lex-agentic-attention 0.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/CHANGELOG.md +12 -0
- data/Gemfile +5 -0
- data/LICENSE +21 -0
- data/README.md +13 -0
- data/lex-agentic-attention.gemspec +30 -0
- data/lib/legion/extensions/agentic/attention/arousal/client.rb +27 -0
- data/lib/legion/extensions/agentic/attention/arousal/helpers/arousal_model.rb +84 -0
- data/lib/legion/extensions/agentic/attention/arousal/helpers/constants.rb +44 -0
- data/lib/legion/extensions/agentic/attention/arousal/runners/arousal.rb +119 -0
- data/lib/legion/extensions/agentic/attention/arousal/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/arousal.rb +18 -0
- data/lib/legion/extensions/agentic/attention/blindspot/client.rb +19 -0
- data/lib/legion/extensions/agentic/attention/blindspot/helpers/blindspot.rb +98 -0
- data/lib/legion/extensions/agentic/attention/blindspot/helpers/blindspot_engine.rb +174 -0
- data/lib/legion/extensions/agentic/attention/blindspot/helpers/constants.rb +63 -0
- data/lib/legion/extensions/agentic/attention/blindspot/helpers/knowledge_boundary.rb +62 -0
- data/lib/legion/extensions/agentic/attention/blindspot/runners/cognitive_blindspot.rb +103 -0
- data/lib/legion/extensions/agentic/attention/blindspot/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/blindspot.rb +20 -0
- data/lib/legion/extensions/agentic/attention/blink/client.rb +19 -0
- data/lib/legion/extensions/agentic/attention/blink/helpers/blink_engine.rb +171 -0
- data/lib/legion/extensions/agentic/attention/blink/helpers/constants.rb +46 -0
- data/lib/legion/extensions/agentic/attention/blink/helpers/stimulus.rb +68 -0
- data/lib/legion/extensions/agentic/attention/blink/runners/attentional_blink.rb +75 -0
- data/lib/legion/extensions/agentic/attention/blink/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/blink.rb +19 -0
- data/lib/legion/extensions/agentic/attention/economy/client.rb +28 -0
- data/lib/legion/extensions/agentic/attention/economy/helpers/attention_budget.rb +143 -0
- data/lib/legion/extensions/agentic/attention/economy/helpers/constants.rb +38 -0
- data/lib/legion/extensions/agentic/attention/economy/helpers/demand.rb +72 -0
- data/lib/legion/extensions/agentic/attention/economy/runners/attention_economy.rb +90 -0
- data/lib/legion/extensions/agentic/attention/economy/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/economy.rb +19 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/actors/decay.rb +45 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/client.rb +29 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/helpers/binding_field.rb +124 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/helpers/bound_object.rb +89 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/helpers/constants.rb +52 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/helpers/feature.rb +52 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/runners/feature_binding.rb +102 -0
- data/lib/legion/extensions/agentic/attention/feature_binding/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/feature_binding.rb +20 -0
- data/lib/legion/extensions/agentic/attention/focus/client.rb +30 -0
- data/lib/legion/extensions/agentic/attention/focus/helpers/constants.rb +38 -0
- data/lib/legion/extensions/agentic/attention/focus/helpers/focus.rb +78 -0
- data/lib/legion/extensions/agentic/attention/focus/helpers/focus_manager.rb +67 -0
- data/lib/legion/extensions/agentic/attention/focus/helpers/habituation.rb +61 -0
- data/lib/legion/extensions/agentic/attention/focus/runners/attention.rb +121 -0
- data/lib/legion/extensions/agentic/attention/focus/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/focus.rb +20 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/client.rb +15 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/helpers/constants.rb +48 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/helpers/facet.rb +90 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/helpers/kaleidoscope_engine.rb +142 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/helpers/pattern.rb +110 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/runners/cognitive_kaleidoscope.rb +91 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/kaleidoscope.rb +22 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/client.rb +28 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/helpers/constants.rb +33 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/helpers/inhibition_engine.rb +118 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/helpers/stimulus_record.rb +70 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/runners/latent_inhibition.rb +78 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/latent_inhibition.rb +19 -0
- data/lib/legion/extensions/agentic/attention/lens/client.rb +29 -0
- data/lib/legion/extensions/agentic/attention/lens/helpers/constants.rb +58 -0
- data/lib/legion/extensions/agentic/attention/lens/helpers/lens.rb +99 -0
- data/lib/legion/extensions/agentic/attention/lens/helpers/lens_engine.rb +101 -0
- data/lib/legion/extensions/agentic/attention/lens/helpers/lens_stack.rb +134 -0
- data/lib/legion/extensions/agentic/attention/lens/runners/cognitive_lens.rb +90 -0
- data/lib/legion/extensions/agentic/attention/lens/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/lens.rb +22 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/client.rb +25 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/helpers/beacon.rb +76 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/helpers/constants.rb +49 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/helpers/fog.rb +74 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/helpers/lighthouse_engine.rb +127 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/runners/cognitive_lighthouse.rb +80 -0
- data/lib/legion/extensions/agentic/attention/lighthouse/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/lighthouse.rb +22 -0
- data/lib/legion/extensions/agentic/attention/priming/client.rb +21 -0
- data/lib/legion/extensions/agentic/attention/priming/helpers/activation_network.rb +130 -0
- data/lib/legion/extensions/agentic/attention/priming/helpers/concept_node.rb +92 -0
- data/lib/legion/extensions/agentic/attention/priming/helpers/constants.rb +54 -0
- data/lib/legion/extensions/agentic/attention/priming/runners/priming.rb +94 -0
- data/lib/legion/extensions/agentic/attention/priming/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/priming.rb +19 -0
- data/lib/legion/extensions/agentic/attention/prism/client.rb +26 -0
- data/lib/legion/extensions/agentic/attention/prism/helpers/beam.rb +169 -0
- data/lib/legion/extensions/agentic/attention/prism/helpers/constants.rb +46 -0
- data/lib/legion/extensions/agentic/attention/prism/helpers/prism_engine.rb +136 -0
- data/lib/legion/extensions/agentic/attention/prism/helpers/spectral_component.rb +72 -0
- data/lib/legion/extensions/agentic/attention/prism/runners/cognitive_prism.rb +79 -0
- data/lib/legion/extensions/agentic/attention/prism/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/prism.rb +22 -0
- data/lib/legion/extensions/agentic/attention/regulation/client.rb +19 -0
- data/lib/legion/extensions/agentic/attention/regulation/helpers/attention_controller.rb +157 -0
- data/lib/legion/extensions/agentic/attention/regulation/helpers/attention_target.rb +64 -0
- data/lib/legion/extensions/agentic/attention/regulation/helpers/constants.rb +41 -0
- data/lib/legion/extensions/agentic/attention/regulation/runners/attention_regulation.rb +88 -0
- data/lib/legion/extensions/agentic/attention/regulation/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/regulation.rb +20 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/helpers/client.rb +17 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/helpers/cognitive_input.rb +88 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/helpers/constants.rb +50 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/helpers/relevance_engine.rb +152 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/runners/relevance_theory.rb +98 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/relevance_theory.rb +19 -0
- data/lib/legion/extensions/agentic/attention/salience/client.rb +21 -0
- data/lib/legion/extensions/agentic/attention/salience/helpers/constants.rb +44 -0
- data/lib/legion/extensions/agentic/attention/salience/helpers/salience_map.rb +122 -0
- data/lib/legion/extensions/agentic/attention/salience/helpers/signal_integrator.rb +97 -0
- data/lib/legion/extensions/agentic/attention/salience/runners/salience.rb +70 -0
- data/lib/legion/extensions/agentic/attention/salience/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/salience.rb +19 -0
- data/lib/legion/extensions/agentic/attention/schema/actors/decay.rb +45 -0
- data/lib/legion/extensions/agentic/attention/schema/client.rb +28 -0
- data/lib/legion/extensions/agentic/attention/schema/helpers/attention_schema_model.rb +226 -0
- data/lib/legion/extensions/agentic/attention/schema/helpers/constants.rb +66 -0
- data/lib/legion/extensions/agentic/attention/schema/helpers/schema_item.rb +68 -0
- data/lib/legion/extensions/agentic/attention/schema/runners/attention_schema.rb +117 -0
- data/lib/legion/extensions/agentic/attention/schema/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/schema.rb +19 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/client.rb +15 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/helpers/constants.rb +44 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/helpers/gating_engine.rb +109 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/helpers/sensory_filter.rb +100 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/runners/sensory_gating.rb +73 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/sensory_gating.rb +19 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/client.rb +28 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/helpers/constants.rb +66 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/helpers/detection_engine.rb +129 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/helpers/detector.rb +136 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/runners/signal_detection.rb +113 -0
- data/lib/legion/extensions/agentic/attention/signal_detection/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/signal_detection.rb +19 -0
- data/lib/legion/extensions/agentic/attention/spotlight/client.rb +29 -0
- data/lib/legion/extensions/agentic/attention/spotlight/helpers/attention_target.rb +61 -0
- data/lib/legion/extensions/agentic/attention/spotlight/helpers/constants.rb +44 -0
- data/lib/legion/extensions/agentic/attention/spotlight/helpers/spotlight.rb +78 -0
- data/lib/legion/extensions/agentic/attention/spotlight/helpers/spotlight_engine.rb +163 -0
- data/lib/legion/extensions/agentic/attention/spotlight/runners/attention_spotlight.rb +122 -0
- data/lib/legion/extensions/agentic/attention/spotlight/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/spotlight.rb +20 -0
- data/lib/legion/extensions/agentic/attention/subliminal/client.rb +19 -0
- data/lib/legion/extensions/agentic/attention/subliminal/helpers/constants.rb +74 -0
- data/lib/legion/extensions/agentic/attention/subliminal/helpers/influence_event.rb +44 -0
- data/lib/legion/extensions/agentic/attention/subliminal/helpers/subliminal_engine.rb +149 -0
- data/lib/legion/extensions/agentic/attention/subliminal/helpers/subliminal_trace.rb +101 -0
- data/lib/legion/extensions/agentic/attention/subliminal/runners/subliminal.rb +71 -0
- data/lib/legion/extensions/agentic/attention/subliminal/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/subliminal.rb +20 -0
- data/lib/legion/extensions/agentic/attention/surprise/client.rb +22 -0
- data/lib/legion/extensions/agentic/attention/surprise/helpers/constants.rb +35 -0
- data/lib/legion/extensions/agentic/attention/surprise/helpers/habituation_model.rb +62 -0
- data/lib/legion/extensions/agentic/attention/surprise/helpers/surprise_event.rb +43 -0
- data/lib/legion/extensions/agentic/attention/surprise/helpers/surprise_store.rb +74 -0
- data/lib/legion/extensions/agentic/attention/surprise/runners/surprise.rb +174 -0
- data/lib/legion/extensions/agentic/attention/surprise/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/surprise.rb +20 -0
- data/lib/legion/extensions/agentic/attention/switching/client.rb +19 -0
- data/lib/legion/extensions/agentic/attention/switching/helpers/constants.rb +64 -0
- data/lib/legion/extensions/agentic/attention/switching/helpers/switch_event.rb +60 -0
- data/lib/legion/extensions/agentic/attention/switching/helpers/switching_engine.rb +172 -0
- data/lib/legion/extensions/agentic/attention/switching/helpers/task_set.rb +91 -0
- data/lib/legion/extensions/agentic/attention/switching/runners/attention_switching.rb +92 -0
- data/lib/legion/extensions/agentic/attention/switching/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/switching.rb +20 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/client.rb +29 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/helpers/constants.rb +52 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/helpers/sensory_mapping.rb +70 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/helpers/synesthesia_engine.rb +177 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/helpers/synesthetic_event.rb +49 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/runners/cognitive_synesthesia.rb +83 -0
- data/lib/legion/extensions/agentic/attention/synesthesia/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/synesthesia.rb +20 -0
- data/lib/legion/extensions/agentic/attention/telescope/client.rb +15 -0
- data/lib/legion/extensions/agentic/attention/telescope/helpers/constants.rb +60 -0
- data/lib/legion/extensions/agentic/attention/telescope/helpers/observation.rb +68 -0
- data/lib/legion/extensions/agentic/attention/telescope/helpers/observatory_engine.rb +126 -0
- data/lib/legion/extensions/agentic/attention/telescope/helpers/telescope.rb +129 -0
- data/lib/legion/extensions/agentic/attention/telescope/runners/cognitive_telescope.rb +103 -0
- data/lib/legion/extensions/agentic/attention/telescope/version.rb +13 -0
- data/lib/legion/extensions/agentic/attention/telescope.rb +22 -0
- data/lib/legion/extensions/agentic/attention/version.rb +11 -0
- data/lib/legion/extensions/agentic/attention.rb +41 -0
- data/spec/legion/extensions/agentic/attention/arousal/client_spec.rb +42 -0
- data/spec/legion/extensions/agentic/attention/arousal/helpers/arousal_model_spec.rb +160 -0
- data/spec/legion/extensions/agentic/attention/arousal/helpers/constants_spec.rb +61 -0
- data/spec/legion/extensions/agentic/attention/arousal/runners/arousal_spec.rb +137 -0
- data/spec/legion/extensions/agentic/attention/blindspot/client_spec.rb +41 -0
- data/spec/legion/extensions/agentic/attention/blindspot/helpers/blindspot_engine_spec.rb +246 -0
- data/spec/legion/extensions/agentic/attention/blindspot/helpers/blindspot_spec.rb +204 -0
- data/spec/legion/extensions/agentic/attention/blindspot/helpers/constants_spec.rb +82 -0
- data/spec/legion/extensions/agentic/attention/blindspot/helpers/knowledge_boundary_spec.rb +104 -0
- data/spec/legion/extensions/agentic/attention/blindspot/runners/cognitive_blindspot_spec.rb +179 -0
- data/spec/legion/extensions/agentic/attention/blink/client_spec.rb +18 -0
- data/spec/legion/extensions/agentic/attention/blink/helpers/blink_engine_spec.rb +147 -0
- data/spec/legion/extensions/agentic/attention/blink/helpers/constants_spec.rb +27 -0
- data/spec/legion/extensions/agentic/attention/blink/helpers/stimulus_spec.rb +77 -0
- data/spec/legion/extensions/agentic/attention/blink/runners/attentional_blink_spec.rb +77 -0
- data/spec/legion/extensions/agentic/attention/economy/client_spec.rb +30 -0
- data/spec/legion/extensions/agentic/attention/economy/helpers/attention_budget_spec.rb +222 -0
- data/spec/legion/extensions/agentic/attention/economy/helpers/constants_spec.rb +112 -0
- data/spec/legion/extensions/agentic/attention/economy/helpers/demand_spec.rb +128 -0
- data/spec/legion/extensions/agentic/attention/economy/runners/attention_economy_spec.rb +163 -0
- data/spec/legion/extensions/agentic/attention/feature_binding/client_spec.rb +20 -0
- data/spec/legion/extensions/agentic/attention/feature_binding/helpers/binding_field_spec.rb +207 -0
- data/spec/legion/extensions/agentic/attention/feature_binding/helpers/bound_object_spec.rb +149 -0
- data/spec/legion/extensions/agentic/attention/feature_binding/helpers/feature_spec.rb +86 -0
- data/spec/legion/extensions/agentic/attention/feature_binding/runners/feature_binding_spec.rb +106 -0
- data/spec/legion/extensions/agentic/attention/focus/client_spec.rb +19 -0
- data/spec/legion/extensions/agentic/attention/focus/helpers/focus_manager_spec.rb +105 -0
- data/spec/legion/extensions/agentic/attention/focus/helpers/focus_spec.rb +148 -0
- data/spec/legion/extensions/agentic/attention/focus/helpers/habituation_spec.rb +97 -0
- data/spec/legion/extensions/agentic/attention/focus/runners/attention_spec.rb +134 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/client_spec.rb +38 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/helpers/constants_spec.rb +79 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/helpers/facet_spec.rb +155 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/helpers/kaleidoscope_engine_spec.rb +164 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/helpers/pattern_spec.rb +157 -0
- data/spec/legion/extensions/agentic/attention/kaleidoscope/runners/cognitive_kaleidoscope_spec.rb +112 -0
- data/spec/legion/extensions/agentic/attention/latent_inhibition/client_spec.rb +49 -0
- data/spec/legion/extensions/agentic/attention/latent_inhibition/helpers/constants_spec.rb +57 -0
- data/spec/legion/extensions/agentic/attention/latent_inhibition/helpers/inhibition_engine_spec.rb +207 -0
- data/spec/legion/extensions/agentic/attention/latent_inhibition/helpers/stimulus_record_spec.rb +175 -0
- data/spec/legion/extensions/agentic/attention/latent_inhibition/runners/latent_inhibition_spec.rb +148 -0
- data/spec/legion/extensions/agentic/attention/lens/client_spec.rb +64 -0
- data/spec/legion/extensions/agentic/attention/lens/helpers/constants_spec.rb +91 -0
- data/spec/legion/extensions/agentic/attention/lens/helpers/lens_engine_spec.rb +158 -0
- data/spec/legion/extensions/agentic/attention/lens/helpers/lens_spec.rb +222 -0
- data/spec/legion/extensions/agentic/attention/lens/helpers/lens_stack_spec.rb +179 -0
- data/spec/legion/extensions/agentic/attention/lens/runners/cognitive_lens_spec.rb +182 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/client_spec.rb +103 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/helpers/beacon_spec.rb +170 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/helpers/constants_spec.rb +94 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/helpers/fog_spec.rb +166 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/helpers/lighthouse_engine_spec.rb +238 -0
- data/spec/legion/extensions/agentic/attention/lighthouse/runners/cognitive_lighthouse_spec.rb +182 -0
- data/spec/legion/extensions/agentic/attention/priming/client_spec.rb +80 -0
- data/spec/legion/extensions/agentic/attention/priming/helpers/activation_network_spec.rb +176 -0
- data/spec/legion/extensions/agentic/attention/priming/helpers/concept_node_spec.rb +165 -0
- data/spec/legion/extensions/agentic/attention/priming/helpers/constants_spec.rb +51 -0
- data/spec/legion/extensions/agentic/attention/priming/runners/priming_spec.rb +141 -0
- data/spec/legion/extensions/agentic/attention/prism/client_spec.rb +86 -0
- data/spec/legion/extensions/agentic/attention/prism/helpers/beam_spec.rb +182 -0
- data/spec/legion/extensions/agentic/attention/prism/helpers/constants_spec.rb +93 -0
- data/spec/legion/extensions/agentic/attention/prism/helpers/prism_engine_spec.rb +253 -0
- data/spec/legion/extensions/agentic/attention/prism/helpers/spectral_component_spec.rb +171 -0
- data/spec/legion/extensions/agentic/attention/prism/runners/cognitive_prism_spec.rb +132 -0
- data/spec/legion/extensions/agentic/attention/regulation/client_spec.rb +58 -0
- data/spec/legion/extensions/agentic/attention/regulation/helpers/attention_controller_spec.rb +295 -0
- data/spec/legion/extensions/agentic/attention/regulation/helpers/attention_target_spec.rb +122 -0
- data/spec/legion/extensions/agentic/attention/regulation/runners/attention_regulation_spec.rb +147 -0
- data/spec/legion/extensions/agentic/attention/relevance_theory/helpers/cognitive_input_spec.rb +121 -0
- data/spec/legion/extensions/agentic/attention/relevance_theory/helpers/relevance_engine_spec.rb +149 -0
- data/spec/legion/extensions/agentic/attention/relevance_theory/relevance_theory_spec.rb +11 -0
- data/spec/legion/extensions/agentic/attention/relevance_theory/runners/relevance_theory_spec.rb +86 -0
- data/spec/legion/extensions/agentic/attention/salience/client_spec.rb +60 -0
- data/spec/legion/extensions/agentic/attention/salience/helpers/constants_spec.rb +100 -0
- data/spec/legion/extensions/agentic/attention/salience/helpers/salience_map_spec.rb +189 -0
- data/spec/legion/extensions/agentic/attention/salience/helpers/signal_integrator_spec.rb +149 -0
- data/spec/legion/extensions/agentic/attention/salience/runners/salience_spec.rb +143 -0
- data/spec/legion/extensions/agentic/attention/schema/client_spec.rb +47 -0
- data/spec/legion/extensions/agentic/attention/schema/helpers/attention_schema_model_spec.rb +219 -0
- data/spec/legion/extensions/agentic/attention/schema/helpers/schema_item_spec.rb +114 -0
- data/spec/legion/extensions/agentic/attention/schema/runners/attention_schema_spec.rb +185 -0
- data/spec/legion/extensions/agentic/attention/sensory_gating/helpers/gating_engine_spec.rb +112 -0
- data/spec/legion/extensions/agentic/attention/sensory_gating/helpers/sensory_filter_spec.rb +121 -0
- data/spec/legion/extensions/agentic/attention/sensory_gating/runners/sensory_gating_spec.rb +61 -0
- data/spec/legion/extensions/agentic/attention/signal_detection/client_spec.rb +20 -0
- data/spec/legion/extensions/agentic/attention/signal_detection/helpers/constants_spec.rb +85 -0
- data/spec/legion/extensions/agentic/attention/signal_detection/helpers/detection_engine_spec.rb +143 -0
- data/spec/legion/extensions/agentic/attention/signal_detection/helpers/detector_spec.rb +179 -0
- data/spec/legion/extensions/agentic/attention/signal_detection/runners/signal_detection_spec.rb +151 -0
- data/spec/legion/extensions/agentic/attention/spotlight/client_spec.rb +31 -0
- data/spec/legion/extensions/agentic/attention/spotlight/helpers/attention_target_spec.rb +121 -0
- data/spec/legion/extensions/agentic/attention/spotlight/helpers/constants_spec.rb +82 -0
- data/spec/legion/extensions/agentic/attention/spotlight/helpers/spotlight_engine_spec.rb +275 -0
- data/spec/legion/extensions/agentic/attention/spotlight/helpers/spotlight_spec.rb +196 -0
- data/spec/legion/extensions/agentic/attention/spotlight/runners/attention_spotlight_spec.rb +251 -0
- data/spec/legion/extensions/agentic/attention/subliminal/client_spec.rb +21 -0
- data/spec/legion/extensions/agentic/attention/subliminal/helpers/influence_event_spec.rb +58 -0
- data/spec/legion/extensions/agentic/attention/subliminal/helpers/subliminal_engine_spec.rb +161 -0
- data/spec/legion/extensions/agentic/attention/subliminal/helpers/subliminal_trace_spec.rb +168 -0
- data/spec/legion/extensions/agentic/attention/subliminal/runners_spec.rb +78 -0
- data/spec/legion/extensions/agentic/attention/subliminal/subliminal_spec.rb +7 -0
- data/spec/legion/extensions/agentic/attention/surprise/client_spec.rb +122 -0
- data/spec/legion/extensions/agentic/attention/surprise/helpers/constants_spec.rb +112 -0
- data/spec/legion/extensions/agentic/attention/surprise/helpers/habituation_model_spec.rb +127 -0
- data/spec/legion/extensions/agentic/attention/surprise/helpers/surprise_event_spec.rb +102 -0
- data/spec/legion/extensions/agentic/attention/surprise/helpers/surprise_store_spec.rb +171 -0
- data/spec/legion/extensions/agentic/attention/surprise/runners/surprise_spec.rb +266 -0
- data/spec/legion/extensions/agentic/attention/switching/attention_switching_spec.rb +7 -0
- data/spec/legion/extensions/agentic/attention/switching/helpers/switch_event_spec.rb +74 -0
- data/spec/legion/extensions/agentic/attention/switching/helpers/switching_engine_spec.rb +164 -0
- data/spec/legion/extensions/agentic/attention/switching/helpers/task_set_spec.rb +134 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/client_spec.rb +57 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/helpers/constants_spec.rb +89 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/helpers/sensory_mapping_spec.rb +135 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/helpers/synesthesia_engine_spec.rb +260 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/helpers/synesthetic_event_spec.rb +83 -0
- data/spec/legion/extensions/agentic/attention/synesthesia/runners/cognitive_synesthesia_spec.rb +192 -0
- data/spec/legion/extensions/agentic/attention/telescope/client_spec.rb +30 -0
- data/spec/legion/extensions/agentic/attention/telescope/helpers/constants_spec.rb +108 -0
- data/spec/legion/extensions/agentic/attention/telescope/helpers/observation_spec.rb +138 -0
- data/spec/legion/extensions/agentic/attention/telescope/helpers/observatory_engine_spec.rb +197 -0
- data/spec/legion/extensions/agentic/attention/telescope/helpers/telescope_spec.rb +251 -0
- data/spec/legion/extensions/agentic/attention/telescope/runners/cognitive_telescope_spec.rb +191 -0
- data/spec/spec_helper.rb +46 -0
- metadata +399 -0
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SensoryGating::Helpers::SensoryFilter do
|
|
4
|
+
subject(:filter) { described_class.new(modality: :visual) }
|
|
5
|
+
|
|
6
|
+
describe '#initialize' do
|
|
7
|
+
it 'assigns a UUID id' do
|
|
8
|
+
expect(filter.id).to match(/\A[0-9a-f-]{36}\z/)
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
it 'sets modality' do
|
|
12
|
+
expect(filter.modality).to eq(:visual)
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
it 'defaults gate_threshold' do
|
|
16
|
+
default = Legion::Extensions::Agentic::Attention::SensoryGating::Helpers::Constants::DEFAULT_GATE_THRESHOLD
|
|
17
|
+
expect(filter.gate_threshold).to eq(default)
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
it 'starts with 0 habituation' do
|
|
21
|
+
expect(filter.habituation_level).to eq(0.0)
|
|
22
|
+
end
|
|
23
|
+
|
|
24
|
+
it 'starts with 0 passed and blocked' do
|
|
25
|
+
expect(filter.stimuli_passed).to eq(0)
|
|
26
|
+
expect(filter.stimuli_blocked).to eq(0)
|
|
27
|
+
end
|
|
28
|
+
end
|
|
29
|
+
|
|
30
|
+
describe '#process' do
|
|
31
|
+
it 'passes high-intensity stimuli' do
|
|
32
|
+
expect(filter.process(intensity: 0.9)).to eq(:passed)
|
|
33
|
+
expect(filter.stimuli_passed).to eq(1)
|
|
34
|
+
end
|
|
35
|
+
|
|
36
|
+
it 'blocks low-intensity stimuli' do
|
|
37
|
+
expect(filter.process(intensity: 0.1)).to eq(:blocked)
|
|
38
|
+
expect(filter.stimuli_blocked).to eq(1)
|
|
39
|
+
end
|
|
40
|
+
|
|
41
|
+
it 'habituates after passing stimuli' do
|
|
42
|
+
filter.process(intensity: 0.9)
|
|
43
|
+
expect(filter.habituation_level).to be > 0.0
|
|
44
|
+
end
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
describe '#open_gate!' do
|
|
48
|
+
it 'decreases gate threshold' do
|
|
49
|
+
original = filter.gate_threshold
|
|
50
|
+
filter.open_gate!
|
|
51
|
+
expect(filter.gate_threshold).to be < original
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
it 'clamps at 0.0' do
|
|
55
|
+
f = described_class.new(modality: :visual, gate_threshold: 0.01)
|
|
56
|
+
f.open_gate!(amount: 0.5)
|
|
57
|
+
expect(f.gate_threshold).to eq(0.0)
|
|
58
|
+
end
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
describe '#close_gate!' do
|
|
62
|
+
it 'increases gate threshold' do
|
|
63
|
+
original = filter.gate_threshold
|
|
64
|
+
filter.close_gate!
|
|
65
|
+
expect(filter.gate_threshold).to be > original
|
|
66
|
+
end
|
|
67
|
+
end
|
|
68
|
+
|
|
69
|
+
describe '#sensitize!' do
|
|
70
|
+
it 'decreases habituation level' do
|
|
71
|
+
filter.process(intensity: 0.9)
|
|
72
|
+
hab = filter.habituation_level
|
|
73
|
+
filter.sensitize!
|
|
74
|
+
expect(filter.habituation_level).to be < hab
|
|
75
|
+
end
|
|
76
|
+
end
|
|
77
|
+
|
|
78
|
+
describe '#reset_habituation!' do
|
|
79
|
+
it 'resets habituation to 0' do
|
|
80
|
+
filter.process(intensity: 0.9)
|
|
81
|
+
filter.reset_habituation!
|
|
82
|
+
expect(filter.habituation_level).to eq(0.0)
|
|
83
|
+
end
|
|
84
|
+
end
|
|
85
|
+
|
|
86
|
+
describe '#effective_threshold' do
|
|
87
|
+
it 'combines gate_threshold and habituation' do
|
|
88
|
+
filter.process(intensity: 0.9)
|
|
89
|
+
expect(filter.effective_threshold).to be > filter.gate_threshold
|
|
90
|
+
end
|
|
91
|
+
end
|
|
92
|
+
|
|
93
|
+
describe '#pass_rate' do
|
|
94
|
+
it 'returns 0.0 with no stimuli' do
|
|
95
|
+
expect(filter.pass_rate).to eq(0.0)
|
|
96
|
+
end
|
|
97
|
+
|
|
98
|
+
it 'calculates ratio' do
|
|
99
|
+
filter.process(intensity: 0.9)
|
|
100
|
+
filter.process(intensity: 0.1)
|
|
101
|
+
expect(filter.pass_rate).to eq(0.5)
|
|
102
|
+
end
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
describe '#gate_label' do
|
|
106
|
+
it 'returns a symbol' do
|
|
107
|
+
expect(filter.gate_label).to be_a(Symbol)
|
|
108
|
+
end
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
describe '#to_h' do
|
|
112
|
+
it 'includes all fields' do
|
|
113
|
+
hash = filter.to_h
|
|
114
|
+
expect(hash).to include(
|
|
115
|
+
:id, :modality, :gate_threshold, :effective_threshold,
|
|
116
|
+
:habituation_level, :gate_label, :stimuli_passed,
|
|
117
|
+
:stimuli_blocked, :pass_rate, :created_at
|
|
118
|
+
)
|
|
119
|
+
end
|
|
120
|
+
end
|
|
121
|
+
end
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SensoryGating::Runners::SensoryGating do
|
|
4
|
+
let(:client) { Legion::Extensions::Agentic::Attention::SensoryGating::Client.new }
|
|
5
|
+
|
|
6
|
+
describe '#create_filter' do
|
|
7
|
+
it 'returns success with filter hash' do
|
|
8
|
+
result = client.create_filter(modality: :visual)
|
|
9
|
+
expect(result[:success]).to be true
|
|
10
|
+
expect(result[:filter]).to include(:id, :modality, :gate_threshold)
|
|
11
|
+
end
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
describe '#process_stimulus' do
|
|
15
|
+
it 'returns outcome' do
|
|
16
|
+
f = client.create_filter(modality: :visual)
|
|
17
|
+
result = client.process_stimulus(filter_id: f[:filter][:id], intensity: 0.9)
|
|
18
|
+
expect(result[:success]).to be true
|
|
19
|
+
expect(result[:outcome]).to eq(:passed)
|
|
20
|
+
end
|
|
21
|
+
|
|
22
|
+
it 'returns failure for unknown id' do
|
|
23
|
+
result = client.process_stimulus(filter_id: 'fake', intensity: 0.5)
|
|
24
|
+
expect(result[:success]).to be false
|
|
25
|
+
end
|
|
26
|
+
end
|
|
27
|
+
|
|
28
|
+
describe '#open_gate' do
|
|
29
|
+
it 'decreases threshold' do
|
|
30
|
+
f = client.create_filter(modality: :visual)
|
|
31
|
+
result = client.open_gate(filter_id: f[:filter][:id])
|
|
32
|
+
expect(result[:success]).to be true
|
|
33
|
+
expect(result[:filter][:gate_threshold]).to be < 0.5
|
|
34
|
+
end
|
|
35
|
+
end
|
|
36
|
+
|
|
37
|
+
describe '#close_gate' do
|
|
38
|
+
it 'increases threshold' do
|
|
39
|
+
f = client.create_filter(modality: :visual)
|
|
40
|
+
result = client.close_gate(filter_id: f[:filter][:id])
|
|
41
|
+
expect(result[:success]).to be true
|
|
42
|
+
expect(result[:filter][:gate_threshold]).to be > 0.5
|
|
43
|
+
end
|
|
44
|
+
end
|
|
45
|
+
|
|
46
|
+
describe '#average_pass_rate' do
|
|
47
|
+
it 'returns pass rate' do
|
|
48
|
+
result = client.average_pass_rate
|
|
49
|
+
expect(result[:success]).to be true
|
|
50
|
+
expect(result[:pass_rate]).to be_a(Numeric)
|
|
51
|
+
end
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
describe '#gating_report' do
|
|
55
|
+
it 'returns a full report' do
|
|
56
|
+
result = client.gating_report
|
|
57
|
+
expect(result[:success]).to be true
|
|
58
|
+
expect(result[:report]).to include(:total_filters, :average_pass_rate)
|
|
59
|
+
end
|
|
60
|
+
end
|
|
61
|
+
end
|
|
@@ -0,0 +1,20 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
require 'legion/extensions/agentic/attention/signal_detection/client'
|
|
4
|
+
|
|
5
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SignalDetection::Client do
|
|
6
|
+
let(:client) { described_class.new }
|
|
7
|
+
|
|
8
|
+
it 'responds to all runner methods' do
|
|
9
|
+
expect(client).to respond_to(:create_detector)
|
|
10
|
+
expect(client).to respond_to(:record_detection_trial)
|
|
11
|
+
expect(client).to respond_to(:compute_detector_sensitivity)
|
|
12
|
+
expect(client).to respond_to(:adjust_detector_bias)
|
|
13
|
+
expect(client).to respond_to(:best_detectors)
|
|
14
|
+
expect(client).to respond_to(:domain_detectors)
|
|
15
|
+
expect(client).to respond_to(:optimal_detector_criterion)
|
|
16
|
+
expect(client).to respond_to(:detector_roc_point)
|
|
17
|
+
expect(client).to respond_to(:update_signal_detection)
|
|
18
|
+
expect(client).to respond_to(:signal_detection_stats)
|
|
19
|
+
end
|
|
20
|
+
end
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants do
|
|
4
|
+
describe '.sensitivity_label' do
|
|
5
|
+
it 'labels exceptional for d_prime >= 3.0' do
|
|
6
|
+
expect(described_class.sensitivity_label(3.5)).to eq(:exceptional)
|
|
7
|
+
end
|
|
8
|
+
|
|
9
|
+
it 'labels excellent for 2.0 <= d_prime < 3.0' do
|
|
10
|
+
expect(described_class.sensitivity_label(2.5)).to eq(:excellent)
|
|
11
|
+
end
|
|
12
|
+
|
|
13
|
+
it 'labels good for 1.0 <= d_prime < 2.0' do
|
|
14
|
+
expect(described_class.sensitivity_label(1.5)).to eq(:good)
|
|
15
|
+
end
|
|
16
|
+
|
|
17
|
+
it 'labels moderate for 0.5 <= d_prime < 1.0' do
|
|
18
|
+
expect(described_class.sensitivity_label(0.7)).to eq(:moderate)
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
it 'labels poor for d_prime < 0.5' do
|
|
22
|
+
expect(described_class.sensitivity_label(0.2)).to eq(:poor)
|
|
23
|
+
end
|
|
24
|
+
end
|
|
25
|
+
|
|
26
|
+
describe '.bias_label' do
|
|
27
|
+
it 'labels very_conservative for criterion >= 1.0' do
|
|
28
|
+
expect(described_class.bias_label(1.5)).to eq(:very_conservative)
|
|
29
|
+
end
|
|
30
|
+
|
|
31
|
+
it 'labels conservative for 0.3 <= criterion < 1.0' do
|
|
32
|
+
expect(described_class.bias_label(0.5)).to eq(:conservative)
|
|
33
|
+
end
|
|
34
|
+
|
|
35
|
+
it 'labels neutral for -0.3 <= criterion < 0.3' do
|
|
36
|
+
expect(described_class.bias_label(0.0)).to eq(:neutral)
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
it 'labels liberal for -1.0 <= criterion < -0.3' do
|
|
40
|
+
expect(described_class.bias_label(-0.5)).to eq(:liberal)
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
it 'labels very_liberal for criterion < -1.0' do
|
|
44
|
+
expect(described_class.bias_label(-1.5)).to eq(:very_liberal)
|
|
45
|
+
end
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
describe '.clamp_sensitivity' do
|
|
49
|
+
it 'clamps to floor' do
|
|
50
|
+
expect(described_class.clamp_sensitivity(-1.0)).to eq(0.0)
|
|
51
|
+
end
|
|
52
|
+
|
|
53
|
+
it 'clamps to ceiling' do
|
|
54
|
+
expect(described_class.clamp_sensitivity(10.0)).to eq(5.0)
|
|
55
|
+
end
|
|
56
|
+
|
|
57
|
+
it 'passes through valid values' do
|
|
58
|
+
expect(described_class.clamp_sensitivity(2.5)).to eq(2.5)
|
|
59
|
+
end
|
|
60
|
+
end
|
|
61
|
+
|
|
62
|
+
describe '.clamp_criterion' do
|
|
63
|
+
it 'clamps to floor' do
|
|
64
|
+
expect(described_class.clamp_criterion(-5.0)).to eq(-3.0)
|
|
65
|
+
end
|
|
66
|
+
|
|
67
|
+
it 'clamps to ceiling' do
|
|
68
|
+
expect(described_class.clamp_criterion(5.0)).to eq(3.0)
|
|
69
|
+
end
|
|
70
|
+
|
|
71
|
+
it 'passes through valid values' do
|
|
72
|
+
expect(described_class.clamp_criterion(1.0)).to eq(1.0)
|
|
73
|
+
end
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
describe 'TRIAL_OUTCOMES' do
|
|
77
|
+
it 'includes all four outcomes' do
|
|
78
|
+
expect(described_class::TRIAL_OUTCOMES).to include(:hit, :miss, :false_alarm, :correct_rejection)
|
|
79
|
+
end
|
|
80
|
+
|
|
81
|
+
it 'is frozen' do
|
|
82
|
+
expect(described_class::TRIAL_OUTCOMES).to be_frozen
|
|
83
|
+
end
|
|
84
|
+
end
|
|
85
|
+
end
|
data/spec/legion/extensions/agentic/attention/signal_detection/helpers/detection_engine_spec.rb
ADDED
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::DetectionEngine do
|
|
4
|
+
subject(:engine) { described_class.new }
|
|
5
|
+
|
|
6
|
+
let(:detector_id) { engine.create_detector(domain: :vision).id }
|
|
7
|
+
|
|
8
|
+
describe '#create_detector' do
|
|
9
|
+
it 'creates a detector and returns it' do
|
|
10
|
+
detector = engine.create_detector(domain: :auditory)
|
|
11
|
+
expect(detector).to be_a(Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Detector)
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
it 'increments count' do
|
|
15
|
+
engine.create_detector(domain: :auditory)
|
|
16
|
+
engine.create_detector(domain: :visual)
|
|
17
|
+
expect(engine.count).to eq(2)
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
it 'raises when max detectors reached' do
|
|
21
|
+
stub_const('Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::MAX_DETECTORS', 1)
|
|
22
|
+
engine.create_detector(domain: :test)
|
|
23
|
+
expect { engine.create_detector(domain: :overflow) }.to raise_error(ArgumentError, /max detectors/)
|
|
24
|
+
end
|
|
25
|
+
end
|
|
26
|
+
|
|
27
|
+
describe '#record_trial' do
|
|
28
|
+
it 'records a hit when signal present and responded present' do
|
|
29
|
+
result = engine.record_trial(detector_id: detector_id, signal_present: true, responded_present: true)
|
|
30
|
+
expect(result[:outcome]).to eq(:hit)
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
it 'records a miss when signal present and not responded' do
|
|
34
|
+
result = engine.record_trial(detector_id: detector_id, signal_present: true, responded_present: false)
|
|
35
|
+
expect(result[:outcome]).to eq(:miss)
|
|
36
|
+
end
|
|
37
|
+
|
|
38
|
+
it 'records a false alarm when no signal and responded present' do
|
|
39
|
+
result = engine.record_trial(detector_id: detector_id, signal_present: false, responded_present: true)
|
|
40
|
+
expect(result[:outcome]).to eq(:false_alarm)
|
|
41
|
+
end
|
|
42
|
+
|
|
43
|
+
it 'records a correct rejection when no signal and not responded' do
|
|
44
|
+
result = engine.record_trial(detector_id: detector_id, signal_present: false, responded_present: false)
|
|
45
|
+
expect(result[:outcome]).to eq(:correct_rejection)
|
|
46
|
+
end
|
|
47
|
+
|
|
48
|
+
it 'raises KeyError for unknown detector' do
|
|
49
|
+
expect do
|
|
50
|
+
engine.record_trial(detector_id: 'bogus', signal_present: true, responded_present: true)
|
|
51
|
+
end.to raise_error(KeyError)
|
|
52
|
+
end
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
describe '#compute_sensitivity' do
|
|
56
|
+
it 'returns d_prime, criterion, accuracy' do
|
|
57
|
+
5.times { engine.record_trial(detector_id: detector_id, signal_present: true, responded_present: true) }
|
|
58
|
+
2.times { engine.record_trial(detector_id: detector_id, signal_present: false, responded_present: false) }
|
|
59
|
+
result = engine.compute_sensitivity(detector_id: detector_id)
|
|
60
|
+
expect(result).to include(:d_prime, :criterion, :accuracy, :hit_rate, :false_alarm_rate)
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
it 'raises KeyError for unknown detector' do
|
|
64
|
+
expect { engine.compute_sensitivity(detector_id: 'unknown') }.to raise_error(KeyError)
|
|
65
|
+
end
|
|
66
|
+
end
|
|
67
|
+
|
|
68
|
+
describe '#adjust_bias' do
|
|
69
|
+
it 'shifts the criterion' do
|
|
70
|
+
before = engine.get(detector_id).criterion
|
|
71
|
+
engine.adjust_bias(detector_id: detector_id, amount: 0.5)
|
|
72
|
+
expect(engine.get(detector_id).criterion).to be > before
|
|
73
|
+
end
|
|
74
|
+
end
|
|
75
|
+
|
|
76
|
+
describe '#best_detectors' do
|
|
77
|
+
it 'returns detectors sorted by sensitivity' do
|
|
78
|
+
id1 = engine.create_detector(domain: :A).id
|
|
79
|
+
id2 = engine.create_detector(domain: :B).id
|
|
80
|
+
10.times { engine.record_trial(detector_id: id1, signal_present: true, responded_present: true) }
|
|
81
|
+
engine.record_trial(detector_id: id1, signal_present: false, responded_present: true)
|
|
82
|
+
2.times { engine.record_trial(detector_id: id2, signal_present: true, responded_present: true) }
|
|
83
|
+
10.times { engine.record_trial(detector_id: id2, signal_present: false, responded_present: true) }
|
|
84
|
+
result = engine.best_detectors(limit: 2)
|
|
85
|
+
expect(result.first[:sensitivity]).to be >= result.last[:sensitivity]
|
|
86
|
+
end
|
|
87
|
+
|
|
88
|
+
it 'respects limit' do
|
|
89
|
+
3.times { |i| engine.create_detector(domain: :"d#{i}") }
|
|
90
|
+
expect(engine.best_detectors(limit: 2).size).to eq(2)
|
|
91
|
+
end
|
|
92
|
+
end
|
|
93
|
+
|
|
94
|
+
describe '#by_domain' do
|
|
95
|
+
it 'filters by domain' do
|
|
96
|
+
engine.create_detector(domain: :audio)
|
|
97
|
+
engine.create_detector(domain: :audio)
|
|
98
|
+
engine.create_detector(domain: :visual)
|
|
99
|
+
result = engine.by_domain(domain: :audio)
|
|
100
|
+
expect(result.size).to eq(2)
|
|
101
|
+
expect(result.all? { |d| d[:domain] == :audio }).to be true
|
|
102
|
+
end
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
describe '#optimal_criterion' do
|
|
106
|
+
it 'returns optimal criterion for balanced prior' do
|
|
107
|
+
result = engine.optimal_criterion(detector_id: detector_id, signal_probability: 0.5)
|
|
108
|
+
expect(result[:optimal_criterion]).to be_within(0.001).of(0.0)
|
|
109
|
+
end
|
|
110
|
+
|
|
111
|
+
it 'returns positive criterion for low signal probability' do
|
|
112
|
+
result = engine.optimal_criterion(detector_id: detector_id, signal_probability: 0.2)
|
|
113
|
+
expect(result[:optimal_criterion]).to be > 0
|
|
114
|
+
end
|
|
115
|
+
end
|
|
116
|
+
|
|
117
|
+
describe '#roc_point' do
|
|
118
|
+
it 'returns hit_rate and false_alarm_rate' do
|
|
119
|
+
result = engine.roc_point(detector_id: detector_id)
|
|
120
|
+
expect(result).to include(:hit_rate, :false_alarm_rate, :d_prime)
|
|
121
|
+
end
|
|
122
|
+
end
|
|
123
|
+
|
|
124
|
+
describe '#decay_all' do
|
|
125
|
+
it 'returns count of detectors updated' do
|
|
126
|
+
engine.record_trial(detector_id: detector_id, signal_present: true, responded_present: true)
|
|
127
|
+
count = engine.decay_all
|
|
128
|
+
expect(count).to eq(1)
|
|
129
|
+
end
|
|
130
|
+
|
|
131
|
+
it 'skips detectors with no trials' do
|
|
132
|
+
count = engine.decay_all
|
|
133
|
+
expect(count).to eq(0)
|
|
134
|
+
end
|
|
135
|
+
end
|
|
136
|
+
|
|
137
|
+
describe '#to_h' do
|
|
138
|
+
it 'returns detector count and map' do
|
|
139
|
+
result = engine.to_h
|
|
140
|
+
expect(result).to include(:detector_count, :detectors)
|
|
141
|
+
end
|
|
142
|
+
end
|
|
143
|
+
end
|
|
@@ -0,0 +1,179 @@
|
|
|
1
|
+
# frozen_string_literal: true
|
|
2
|
+
|
|
3
|
+
RSpec.describe Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Detector do
|
|
4
|
+
subject(:detector) { described_class.new(domain: :threat) }
|
|
5
|
+
|
|
6
|
+
describe '#initialize' do
|
|
7
|
+
it 'assigns an id' do
|
|
8
|
+
expect(detector.id).to match(/\A[0-9a-f-]{36}\z/)
|
|
9
|
+
end
|
|
10
|
+
|
|
11
|
+
it 'sets domain' do
|
|
12
|
+
expect(detector.domain).to eq(:threat)
|
|
13
|
+
end
|
|
14
|
+
|
|
15
|
+
it 'starts with default sensitivity' do
|
|
16
|
+
expect(detector.sensitivity).to eq(Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::DEFAULT_SENSITIVITY)
|
|
17
|
+
end
|
|
18
|
+
|
|
19
|
+
it 'starts with default criterion' do
|
|
20
|
+
expect(detector.criterion).to eq(Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::DEFAULT_CRITERION)
|
|
21
|
+
end
|
|
22
|
+
|
|
23
|
+
it 'starts with zero counts' do
|
|
24
|
+
expect(detector.hits).to eq(0)
|
|
25
|
+
expect(detector.misses).to eq(0)
|
|
26
|
+
expect(detector.false_alarms).to eq(0)
|
|
27
|
+
expect(detector.correct_rejections).to eq(0)
|
|
28
|
+
expect(detector.trial_count).to eq(0)
|
|
29
|
+
end
|
|
30
|
+
end
|
|
31
|
+
|
|
32
|
+
describe '#record_trial' do
|
|
33
|
+
it 'increments hits on :hit' do
|
|
34
|
+
detector.record_trial(outcome: :hit)
|
|
35
|
+
expect(detector.hits).to eq(1)
|
|
36
|
+
expect(detector.trial_count).to eq(1)
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
it 'increments misses on :miss' do
|
|
40
|
+
detector.record_trial(outcome: :miss)
|
|
41
|
+
expect(detector.misses).to eq(1)
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
it 'increments false_alarms on :false_alarm' do
|
|
45
|
+
detector.record_trial(outcome: :false_alarm)
|
|
46
|
+
expect(detector.false_alarms).to eq(1)
|
|
47
|
+
end
|
|
48
|
+
|
|
49
|
+
it 'increments correct_rejections on :correct_rejection' do
|
|
50
|
+
detector.record_trial(outcome: :correct_rejection)
|
|
51
|
+
expect(detector.correct_rejections).to eq(1)
|
|
52
|
+
end
|
|
53
|
+
|
|
54
|
+
it 'raises on invalid outcome' do
|
|
55
|
+
expect { detector.record_trial(outcome: :bogus) }.to raise_error(ArgumentError, /invalid outcome/)
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
it 'updates last_trial_at' do
|
|
59
|
+
expect(detector.last_trial_at).to be_nil
|
|
60
|
+
detector.record_trial(outcome: :hit)
|
|
61
|
+
expect(detector.last_trial_at).to be_a(Time)
|
|
62
|
+
end
|
|
63
|
+
end
|
|
64
|
+
|
|
65
|
+
describe '#hit_rate' do
|
|
66
|
+
it 'applies Hautus correction' do
|
|
67
|
+
expect(detector.hit_rate).to be_between(0.0, 1.0)
|
|
68
|
+
end
|
|
69
|
+
|
|
70
|
+
it 'increases as hits accumulate' do
|
|
71
|
+
before = detector.hit_rate
|
|
72
|
+
5.times { detector.record_trial(outcome: :hit) }
|
|
73
|
+
expect(detector.hit_rate).to be > before
|
|
74
|
+
end
|
|
75
|
+
end
|
|
76
|
+
|
|
77
|
+
describe '#false_alarm_rate' do
|
|
78
|
+
it 'applies Hautus correction' do
|
|
79
|
+
expect(detector.false_alarm_rate).to be_between(0.0, 1.0)
|
|
80
|
+
end
|
|
81
|
+
|
|
82
|
+
it 'increases as false alarms accumulate' do
|
|
83
|
+
before = detector.false_alarm_rate
|
|
84
|
+
5.times { detector.record_trial(outcome: :false_alarm) }
|
|
85
|
+
expect(detector.false_alarm_rate).to be > before
|
|
86
|
+
end
|
|
87
|
+
end
|
|
88
|
+
|
|
89
|
+
describe '#compute_dprime' do
|
|
90
|
+
it 'returns a float within sensitivity bounds' do
|
|
91
|
+
5.times { detector.record_trial(outcome: :hit) }
|
|
92
|
+
2.times { detector.record_trial(outcome: :miss) }
|
|
93
|
+
2.times { detector.record_trial(outcome: :false_alarm) }
|
|
94
|
+
3.times { detector.record_trial(outcome: :correct_rejection) }
|
|
95
|
+
expect(detector.compute_dprime).to be_between(
|
|
96
|
+
Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::SENSITIVITY_FLOOR,
|
|
97
|
+
Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::SENSITIVITY_CEILING
|
|
98
|
+
)
|
|
99
|
+
end
|
|
100
|
+
|
|
101
|
+
it 'is positive when hit_rate > false_alarm_rate' do
|
|
102
|
+
10.times { detector.record_trial(outcome: :hit) }
|
|
103
|
+
detector.record_trial(outcome: :false_alarm)
|
|
104
|
+
expect(detector.compute_dprime).to be > 0
|
|
105
|
+
end
|
|
106
|
+
end
|
|
107
|
+
|
|
108
|
+
describe '#compute_criterion' do
|
|
109
|
+
it 'returns a float within criterion bounds' do
|
|
110
|
+
5.times { detector.record_trial(outcome: :hit) }
|
|
111
|
+
5.times { detector.record_trial(outcome: :miss) }
|
|
112
|
+
result = detector.compute_criterion
|
|
113
|
+
expect(result).to be_between(
|
|
114
|
+
Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::CRITERION_FLOOR,
|
|
115
|
+
Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::CRITERION_CEILING
|
|
116
|
+
)
|
|
117
|
+
end
|
|
118
|
+
end
|
|
119
|
+
|
|
120
|
+
describe '#accuracy' do
|
|
121
|
+
it 'returns 0.0 with no trials' do
|
|
122
|
+
expect(detector.accuracy).to eq(0.0)
|
|
123
|
+
end
|
|
124
|
+
|
|
125
|
+
it 'computes correctly' do
|
|
126
|
+
3.times { detector.record_trial(outcome: :hit) }
|
|
127
|
+
2.times { detector.record_trial(outcome: :correct_rejection) }
|
|
128
|
+
expect(detector.accuracy).to eq(1.0)
|
|
129
|
+
end
|
|
130
|
+
|
|
131
|
+
it 'decreases with errors' do
|
|
132
|
+
2.times { detector.record_trial(outcome: :hit) }
|
|
133
|
+
2.times { detector.record_trial(outcome: :miss) }
|
|
134
|
+
2.times { detector.record_trial(outcome: :false_alarm) }
|
|
135
|
+
2.times { detector.record_trial(outcome: :correct_rejection) }
|
|
136
|
+
expect(detector.accuracy).to eq(0.5)
|
|
137
|
+
end
|
|
138
|
+
end
|
|
139
|
+
|
|
140
|
+
describe '#sensitivity_label' do
|
|
141
|
+
it 'returns a symbol' do
|
|
142
|
+
expect(detector.sensitivity_label).to be_a(Symbol)
|
|
143
|
+
end
|
|
144
|
+
end
|
|
145
|
+
|
|
146
|
+
describe '#bias_label' do
|
|
147
|
+
it 'returns a symbol' do
|
|
148
|
+
expect(detector.bias_label).to be_a(Symbol)
|
|
149
|
+
end
|
|
150
|
+
end
|
|
151
|
+
|
|
152
|
+
describe '#adjust_criterion' do
|
|
153
|
+
it 'shifts criterion by amount' do
|
|
154
|
+
initial = detector.criterion
|
|
155
|
+
detector.adjust_criterion(amount: 0.5)
|
|
156
|
+
expect(detector.criterion).to eq(initial + 0.5)
|
|
157
|
+
end
|
|
158
|
+
|
|
159
|
+
it 'clamps at ceiling' do
|
|
160
|
+
detector.adjust_criterion(amount: 10.0)
|
|
161
|
+
expect(detector.criterion).to eq(Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::CRITERION_CEILING)
|
|
162
|
+
end
|
|
163
|
+
|
|
164
|
+
it 'clamps at floor' do
|
|
165
|
+
detector.adjust_criterion(amount: -10.0)
|
|
166
|
+
expect(detector.criterion).to eq(Legion::Extensions::Agentic::Attention::SignalDetection::Helpers::Constants::CRITERION_FLOOR)
|
|
167
|
+
end
|
|
168
|
+
end
|
|
169
|
+
|
|
170
|
+
describe '#to_h' do
|
|
171
|
+
it 'returns a hash with expected keys' do
|
|
172
|
+
result = detector.to_h
|
|
173
|
+
expect(result).to include(:id, :domain, :sensitivity, :criterion, :hits, :misses,
|
|
174
|
+
:false_alarms, :correct_rejections, :trial_count,
|
|
175
|
+
:hit_rate, :false_alarm_rate, :accuracy,
|
|
176
|
+
:sensitivity_label, :bias_label, :created_at)
|
|
177
|
+
end
|
|
178
|
+
end
|
|
179
|
+
end
|