rubyllm-observ 0.5.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.
Files changed (209) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +778 -0
  3. data/Rakefile +49 -0
  4. data/app/assets/javascripts/observ/application.js +12 -0
  5. data/app/assets/javascripts/observ/controllers/autoscroll_controller.js +33 -0
  6. data/app/assets/javascripts/observ/controllers/chat_form_controller.js +93 -0
  7. data/app/assets/javascripts/observ/controllers/copy_controller.js +43 -0
  8. data/app/assets/javascripts/observ/controllers/dashboard_controller.js +58 -0
  9. data/app/assets/javascripts/observ/controllers/drawer_controller.js +58 -0
  10. data/app/assets/javascripts/observ/controllers/expandable_controller.js +33 -0
  11. data/app/assets/javascripts/observ/controllers/filter_controller.js +36 -0
  12. data/app/assets/javascripts/observ/controllers/index.js +52 -0
  13. data/app/assets/javascripts/observ/controllers/json_viewer_controller.js +260 -0
  14. data/app/assets/javascripts/observ/controllers/message_form_controller.js +58 -0
  15. data/app/assets/javascripts/observ/controllers/prompt_variables_controller.js +64 -0
  16. data/app/assets/javascripts/observ/controllers/text_select_controller.js +14 -0
  17. data/app/assets/stylesheets/observ/_annotations.scss +127 -0
  18. data/app/assets/stylesheets/observ/_card.scss +52 -0
  19. data/app/assets/stylesheets/observ/_chat.scss +156 -0
  20. data/app/assets/stylesheets/observ/_components.scss +460 -0
  21. data/app/assets/stylesheets/observ/_dashboard.scss +40 -0
  22. data/app/assets/stylesheets/observ/_datasets.scss +697 -0
  23. data/app/assets/stylesheets/observ/_drawer.scss +273 -0
  24. data/app/assets/stylesheets/observ/_json_viewer.scss +120 -0
  25. data/app/assets/stylesheets/observ/_layout.scss +256 -0
  26. data/app/assets/stylesheets/observ/_metrics.scss +99 -0
  27. data/app/assets/stylesheets/observ/_observations.scss +160 -0
  28. data/app/assets/stylesheets/observ/_pagination.scss +143 -0
  29. data/app/assets/stylesheets/observ/_prompts.scss +365 -0
  30. data/app/assets/stylesheets/observ/_table.scss +53 -0
  31. data/app/assets/stylesheets/observ/_variables.scss +53 -0
  32. data/app/assets/stylesheets/observ/application.scss +15 -0
  33. data/app/controllers/observ/annotations_controller.rb +144 -0
  34. data/app/controllers/observ/application_controller.rb +8 -0
  35. data/app/controllers/observ/chats_controller.rb +58 -0
  36. data/app/controllers/observ/dashboard_controller.rb +159 -0
  37. data/app/controllers/observ/dataset_items_controller.rb +85 -0
  38. data/app/controllers/observ/dataset_run_items_controller.rb +84 -0
  39. data/app/controllers/observ/dataset_runs_controller.rb +110 -0
  40. data/app/controllers/observ/datasets_controller.rb +74 -0
  41. data/app/controllers/observ/messages_controller.rb +26 -0
  42. data/app/controllers/observ/observations_controller.rb +59 -0
  43. data/app/controllers/observ/prompt_versions_controller.rb +148 -0
  44. data/app/controllers/observ/prompts_controller.rb +205 -0
  45. data/app/controllers/observ/sessions_controller.rb +45 -0
  46. data/app/controllers/observ/traces_controller.rb +86 -0
  47. data/app/forms/observ/prompt_form.rb +96 -0
  48. data/app/helpers/observ/application_helper.rb +9 -0
  49. data/app/helpers/observ/chats_helper.rb +47 -0
  50. data/app/helpers/observ/dashboard_helper.rb +154 -0
  51. data/app/helpers/observ/datasets_helper.rb +62 -0
  52. data/app/helpers/observ/pagination_helper.rb +38 -0
  53. data/app/jobs/observ/application_job.rb +4 -0
  54. data/app/jobs/observ/dataset_runner_job.rb +49 -0
  55. data/app/mailers/observ/application_mailer.rb +6 -0
  56. data/app/models/concerns/observ/agent_phaseable.rb +124 -0
  57. data/app/models/concerns/observ/agent_selectable.rb +50 -0
  58. data/app/models/concerns/observ/chat_enhancements.rb +109 -0
  59. data/app/models/concerns/observ/message_enhancements.rb +31 -0
  60. data/app/models/concerns/observ/observability_instrumentation.rb +124 -0
  61. data/app/models/concerns/observ/prompt_management.rb +320 -0
  62. data/app/models/concerns/observ/trace_association.rb +9 -0
  63. data/app/models/observ/annotation.rb +23 -0
  64. data/app/models/observ/application_record.rb +5 -0
  65. data/app/models/observ/dataset.rb +51 -0
  66. data/app/models/observ/dataset_item.rb +41 -0
  67. data/app/models/observ/dataset_run.rb +104 -0
  68. data/app/models/observ/dataset_run_item.rb +111 -0
  69. data/app/models/observ/generation.rb +56 -0
  70. data/app/models/observ/null_prompt.rb +59 -0
  71. data/app/models/observ/observation.rb +38 -0
  72. data/app/models/observ/prompt.rb +315 -0
  73. data/app/models/observ/score.rb +51 -0
  74. data/app/models/observ/session.rb +131 -0
  75. data/app/models/observ/span.rb +13 -0
  76. data/app/models/observ/trace.rb +135 -0
  77. data/app/presenters/observ/agent_select_presenter.rb +59 -0
  78. data/app/services/observ/agent_executor_service.rb +174 -0
  79. data/app/services/observ/agent_provider.rb +60 -0
  80. data/app/services/observ/agent_selection_service.rb +53 -0
  81. data/app/services/observ/chat_instrumenter.rb +523 -0
  82. data/app/services/observ/dataset_runner_service.rb +153 -0
  83. data/app/services/observ/evaluator_runner_service.rb +58 -0
  84. data/app/services/observ/evaluators/base_evaluator.rb +51 -0
  85. data/app/services/observ/evaluators/contains_evaluator.rb +53 -0
  86. data/app/services/observ/evaluators/exact_match_evaluator.rb +23 -0
  87. data/app/services/observ/evaluators/json_structure_evaluator.rb +44 -0
  88. data/app/services/observ/prompt_manager/cache_statistics.rb +82 -0
  89. data/app/services/observ/prompt_manager/caching.rb +167 -0
  90. data/app/services/observ/prompt_manager/comparison.rb +49 -0
  91. data/app/services/observ/prompt_manager/version_management.rb +96 -0
  92. data/app/services/observ/prompt_manager.rb +40 -0
  93. data/app/services/observ/trace_text_formatter.rb +349 -0
  94. data/app/validators/observ/prompt_config_validator.rb +187 -0
  95. data/app/views/kaminari/_first_page.html.erb +11 -0
  96. data/app/views/kaminari/_gap.html.erb +8 -0
  97. data/app/views/kaminari/_last_page.html.erb +11 -0
  98. data/app/views/kaminari/_next_page.html.erb +11 -0
  99. data/app/views/kaminari/_page.html.erb +12 -0
  100. data/app/views/kaminari/_paginator.html.erb +25 -0
  101. data/app/views/kaminari/_prev_page.html.erb +11 -0
  102. data/app/views/kaminari/observ/_first_page.html.erb +11 -0
  103. data/app/views/kaminari/observ/_gap.html.erb +8 -0
  104. data/app/views/kaminari/observ/_last_page.html.erb +11 -0
  105. data/app/views/kaminari/observ/_next_page.html.erb +11 -0
  106. data/app/views/kaminari/observ/_page.html.erb +12 -0
  107. data/app/views/kaminari/observ/_paginator.html.erb +25 -0
  108. data/app/views/kaminari/observ/_prev_page.html.erb +11 -0
  109. data/app/views/layouts/observ/application.html.erb +88 -0
  110. data/app/views/observ/annotations/_annotation.html.erb +13 -0
  111. data/app/views/observ/annotations/_form.html.erb +28 -0
  112. data/app/views/observ/annotations/index.html.erb +28 -0
  113. data/app/views/observ/annotations/sessions_index.html.erb +48 -0
  114. data/app/views/observ/annotations/traces_index.html.erb +48 -0
  115. data/app/views/observ/chats/_form.html.erb +45 -0
  116. data/app/views/observ/chats/index.html.erb +67 -0
  117. data/app/views/observ/chats/new.html.erb +17 -0
  118. data/app/views/observ/chats/show.html.erb +34 -0
  119. data/app/views/observ/dashboard/index.html.erb +236 -0
  120. data/app/views/observ/dataset_items/_form.html.erb +49 -0
  121. data/app/views/observ/dataset_items/edit.html.erb +18 -0
  122. data/app/views/observ/dataset_items/index.html.erb +95 -0
  123. data/app/views/observ/dataset_items/new.html.erb +18 -0
  124. data/app/views/observ/dataset_run_items/_score_close_drawer.html.erb +4 -0
  125. data/app/views/observ/dataset_run_items/_score_drawer.html.erb +75 -0
  126. data/app/views/observ/dataset_run_items/_score_success.html.erb +29 -0
  127. data/app/views/observ/dataset_run_items/_scores_cell.html.erb +19 -0
  128. data/app/views/observ/dataset_run_items/details_drawer.turbo_stream.erb +80 -0
  129. data/app/views/observ/dataset_run_items/score_drawer.turbo_stream.erb +7 -0
  130. data/app/views/observ/dataset_runs/index.html.erb +108 -0
  131. data/app/views/observ/dataset_runs/new.html.erb +57 -0
  132. data/app/views/observ/dataset_runs/review.html.erb +155 -0
  133. data/app/views/observ/dataset_runs/show.html.erb +166 -0
  134. data/app/views/observ/datasets/_form.html.erb +62 -0
  135. data/app/views/observ/datasets/_items_tab.html.erb +66 -0
  136. data/app/views/observ/datasets/_runs_tab.html.erb +82 -0
  137. data/app/views/observ/datasets/edit.html.erb +32 -0
  138. data/app/views/observ/datasets/index.html.erb +105 -0
  139. data/app/views/observ/datasets/new.html.erb +18 -0
  140. data/app/views/observ/datasets/show.html.erb +67 -0
  141. data/app/views/observ/messages/_content.html.erb +1 -0
  142. data/app/views/observ/messages/_form.html.erb +33 -0
  143. data/app/views/observ/messages/_message.html.erb +14 -0
  144. data/app/views/observ/messages/_tool_calls.html.erb +10 -0
  145. data/app/views/observ/messages/create.turbo_stream.erb +9 -0
  146. data/app/views/observ/observations/index.html.erb +97 -0
  147. data/app/views/observ/observations/show_generation.html.erb +195 -0
  148. data/app/views/observ/observations/show_span.html.erb +93 -0
  149. data/app/views/observ/prompts/_diff_content.html.erb +16 -0
  150. data/app/views/observ/prompts/_form.html.erb +111 -0
  151. data/app/views/observ/prompts/_new_form.html.erb +102 -0
  152. data/app/views/observ/prompts/_prompt_actions.html.erb +4 -0
  153. data/app/views/observ/prompts/_prompt_content_highlighted.html.erb +4 -0
  154. data/app/views/observ/prompts/_version_actions.html.erb +40 -0
  155. data/app/views/observ/prompts/compare.html.erb +155 -0
  156. data/app/views/observ/prompts/edit.html.erb +17 -0
  157. data/app/views/observ/prompts/index.html.erb +108 -0
  158. data/app/views/observ/prompts/new.html.erb +17 -0
  159. data/app/views/observ/prompts/show.html.erb +138 -0
  160. data/app/views/observ/prompts/versions.html.erb +87 -0
  161. data/app/views/observ/sessions/annotations_drawer.turbo_stream.erb +25 -0
  162. data/app/views/observ/sessions/drawer_test.turbo_stream.erb +49 -0
  163. data/app/views/observ/sessions/index.html.erb +91 -0
  164. data/app/views/observ/sessions/show.html.erb +251 -0
  165. data/app/views/observ/traces/add_to_dataset_drawer.turbo_stream.erb +48 -0
  166. data/app/views/observ/traces/annotations_drawer.turbo_stream.erb +25 -0
  167. data/app/views/observ/traces/index.html.erb +87 -0
  168. data/app/views/observ/traces/show.html.erb +285 -0
  169. data/app/views/observ/traces/text_output_drawer.turbo_stream.erb +48 -0
  170. data/app/views/shared/_drawer.html.erb +26 -0
  171. data/config/routes.rb +80 -0
  172. data/db/migrate/001_create_observ_sessions.rb +21 -0
  173. data/db/migrate/002_create_observ_traces.rb +25 -0
  174. data/db/migrate/003_create_observ_observations.rb +42 -0
  175. data/db/migrate/004_add_message_id_to_observ_traces.rb +7 -0
  176. data/db/migrate/005_create_observ_prompts.rb +21 -0
  177. data/db/migrate/006_fix_prompt_config_strings.rb +23 -0
  178. data/db/migrate/007_create_observ_annotations.rb +12 -0
  179. data/db/migrate/009_add_prompt_fields_to_observ_chats.rb +11 -0
  180. data/db/migrate/010_create_observ_datasets.rb +15 -0
  181. data/db/migrate/011_create_observ_dataset_items.rb +17 -0
  182. data/db/migrate/012_create_observ_dataset_runs.rb +22 -0
  183. data/db/migrate/013_create_observ_dataset_run_items.rb +16 -0
  184. data/db/migrate/014_create_observ_scores.rb +26 -0
  185. data/lib/generators/observ/add_phase_tracking/add_phase_tracking_generator.rb +150 -0
  186. data/lib/generators/observ/add_phase_tracking/templates/migration.rb.tt +6 -0
  187. data/lib/generators/observ/install/USAGE +27 -0
  188. data/lib/generators/observ/install/install_generator.rb +270 -0
  189. data/lib/generators/observ/install_chat/install_chat_generator.rb +313 -0
  190. data/lib/generators/observ/install_chat/templates/agents/base_agent.rb.tt +147 -0
  191. data/lib/generators/observ/install_chat/templates/agents/simple_agent.rb.tt +55 -0
  192. data/lib/generators/observ/install_chat/templates/concerns/observ_chat_enhancements.rb.tt +34 -0
  193. data/lib/generators/observ/install_chat/templates/concerns/observ_message_enhancements.rb.tt +18 -0
  194. data/lib/generators/observ/install_chat/templates/initializers/observability.rb.tt +20 -0
  195. data/lib/generators/observ/install_chat/templates/jobs/chat_response_job.rb.tt +56 -0
  196. data/lib/generators/observ/install_chat/templates/migrations/add_agent_class_name.rb.tt +6 -0
  197. data/lib/generators/observ/install_chat/templates/migrations/add_observability_session_id.rb.tt +6 -0
  198. data/lib/generators/observ/install_chat/templates/tools/think_tool.rb.tt +29 -0
  199. data/lib/generators/observ/install_chat/templates/views/messages/_content.html.erb.tt +1 -0
  200. data/lib/observ/asset_installer.rb +130 -0
  201. data/lib/observ/asset_syncer.rb +104 -0
  202. data/lib/observ/configuration.rb +108 -0
  203. data/lib/observ/engine.rb +50 -0
  204. data/lib/observ/index_file_generator.rb +142 -0
  205. data/lib/observ/instrumenter/ruby_llm.rb +6 -0
  206. data/lib/observ/version.rb +3 -0
  207. data/lib/observ.rb +29 -0
  208. data/lib/tasks/observ_tasks.rake +75 -0
  209. metadata +453 -0
@@ -0,0 +1,156 @@
1
+ .observ-chat-messages {
2
+ display: flex;
3
+ flex-direction: column;
4
+ gap: 1rem;
5
+ margin-bottom: 1.875rem;
6
+ }
7
+
8
+ .observ-chat-message {
9
+ border-left: 3px solid #e5e7eb;
10
+ padding: 1rem;
11
+ background-color: #f9fafb;
12
+ border-radius: 0.25rem;
13
+ transition: all 0.15s ease-in-out;
14
+ }
15
+
16
+ .observ-chat-message:hover {
17
+ background-color: #f3f4f6;
18
+ }
19
+
20
+ .observ-chat-message--user {
21
+ border-left-color: #3b82f6;
22
+ }
23
+
24
+ .observ-chat-message--assistant {
25
+ border-left-color: #10b981;
26
+ }
27
+
28
+ .observ-chat-message--system {
29
+ border-left-color: #6b7280;
30
+ }
31
+
32
+ .observ-chat-message--tool {
33
+ border-left-color: #f59e0b;
34
+ }
35
+
36
+ .observ-chat-message__header {
37
+ display: flex;
38
+ align-items: center;
39
+ gap: 1rem;
40
+ margin-bottom: 0.75rem;
41
+ flex-wrap: wrap;
42
+ }
43
+
44
+ .observ-chat-message__trace-link {
45
+ font-size: 0.75rem;
46
+ padding: 0.25rem 0.5rem;
47
+ background-color: #3b82f6;
48
+ color: #fff;
49
+ text-decoration: none;
50
+ border-radius: 0.25rem;
51
+ font-weight: 600;
52
+ transition: all 0.15s ease-in-out;
53
+ }
54
+
55
+ .observ-chat-message__trace-link:hover {
56
+ background-color: #2563eb;
57
+ }
58
+
59
+ .observ-chat-message__role {
60
+ font-size: 0.75rem;
61
+ font-weight: 600;
62
+ color: #4b5563;
63
+ text-transform: uppercase;
64
+ letter-spacing: 0.05em;
65
+ }
66
+
67
+ .observ-chat-message__time {
68
+ font-size: 0.75rem;
69
+ color: #9ca3af;
70
+ }
71
+
72
+ .observ-chat-message__tokens {
73
+ display: flex;
74
+ align-items: center;
75
+ gap: 0.75rem;
76
+ font-size: 0.75rem;
77
+ color: #6b7280;
78
+ margin-left: auto;
79
+ }
80
+
81
+ .observ-chat-message__tokens span {
82
+ display: flex;
83
+ align-items: center;
84
+ gap: 0.25rem;
85
+ padding: 0.125rem 0.5rem;
86
+ background-color: #e5e7eb;
87
+ border-radius: 0.25rem;
88
+ font-variant-numeric: tabular-nums;
89
+ }
90
+
91
+ .observ-chat-message__content {
92
+ color: #111827;
93
+ font-size: 0.875rem;
94
+ line-height: 1.6;
95
+ word-wrap: break-word;
96
+ overflow-wrap: break-word;
97
+ }
98
+
99
+ .observ-chat-message__content p {
100
+ margin: 0 0 0.5rem 0;
101
+ }
102
+
103
+ .observ-chat-message__content p:last-child {
104
+ margin-bottom: 0;
105
+ }
106
+
107
+ .observ-chat-message__content code {
108
+ font-family: ui-monospace, SFMono-Regular, 'SF Mono', Menlo, Consolas, 'Liberation Mono', monospace;
109
+ background-color: #e5e7eb;
110
+ padding: 0.125rem 0.375rem;
111
+ border-radius: 0.25rem;
112
+ font-size: 0.875em;
113
+ }
114
+
115
+ .observ-chat-message__content pre {
116
+ background-color: #1f2937;
117
+ color: #f9fafb;
118
+ padding: 1rem;
119
+ border-radius: 0.375rem;
120
+ overflow-x: auto;
121
+ margin: 0.5rem 0;
122
+ }
123
+
124
+ .observ-chat-message__content pre code {
125
+ background-color: transparent;
126
+ padding: 0;
127
+ color: inherit;
128
+ }
129
+
130
+ .observ-chat-message__tool-calls {
131
+ margin-top: 0.75rem;
132
+ display: flex;
133
+ flex-direction: column;
134
+ gap: 0.5rem;
135
+ }
136
+
137
+ .observ-chat-message__tool-calls-label {
138
+ font-size: 0.75rem;
139
+ font-weight: 600;
140
+ color: #6b7280;
141
+ text-transform: uppercase;
142
+ letter-spacing: 0.05em;
143
+ }
144
+
145
+ .observ-chat-message__tool-call {
146
+ background: #f3f4f6;
147
+ padding: 0.5rem;
148
+ border-radius: 0.375rem;
149
+ font-family: ui-monospace, SFMono-Regular, 'SF Mono', Menlo, Consolas, 'Liberation Mono', monospace;
150
+ font-size: 0.875rem;
151
+ border-left: 3px solid #f59e0b;
152
+ }
153
+
154
+ .observ-chat-message__tool-call-name {
155
+ font-weight: 600;
156
+ }
@@ -0,0 +1,460 @@
1
+ @import 'variables';
2
+
3
+ .observ-badge {
4
+ display: inline-flex;
5
+ align-items: center;
6
+ padding: 0.25rem 0.75rem;
7
+ border-radius: 9999px;
8
+ font-size: $observ-font-size-xs;
9
+ font-weight: 600;
10
+ text-transform: uppercase;
11
+ letter-spacing: 0.05em;
12
+
13
+ &--success {
14
+ background-color: lighten($observ-success, 45%);
15
+ color: darken($observ-success, 10%);
16
+ }
17
+
18
+ &--danger {
19
+ background-color: lighten($observ-danger, 45%);
20
+ color: darken($observ-danger, 10%);
21
+ }
22
+
23
+ &--warning {
24
+ background-color: lighten($observ-warning, 45%);
25
+ color: darken($observ-warning, 30%);
26
+ }
27
+
28
+ &--info {
29
+ background-color: lighten($observ-info, 45%);
30
+ color: darken($observ-info, 10%);
31
+ }
32
+
33
+ &--generation {
34
+ background-color: #e0e7ff;
35
+ color: #4338ca;
36
+ }
37
+
38
+ &--span {
39
+ background-color: lighten($observ-info, 45%);
40
+ color: darken($observ-info, 10%);
41
+ }
42
+
43
+ &--light {
44
+ background-color: rgba(255, 255, 255, 0.3);
45
+ color: $observ-white;
46
+ font-size: 0.75rem;
47
+ padding: 0.125rem 0.5rem;
48
+ margin-left: 0.375rem;
49
+ }
50
+ }
51
+
52
+ .observ-button {
53
+ display: inline-flex;
54
+ align-items: center;
55
+ justify-content: center;
56
+ gap: $observ-spacing-xs;
57
+ padding: $observ-spacing-sm $observ-spacing-md;
58
+ border: 1px solid $observ-gray-300;
59
+ border-radius: $observ-border-radius-sm;
60
+ background-color: $observ-white;
61
+ color: $observ-gray-700;
62
+ font-size: $observ-font-size-sm;
63
+ font-weight: 500;
64
+ font-family: inherit;
65
+ line-height: 1.25;
66
+ text-decoration: none;
67
+ cursor: pointer;
68
+ transition: $observ-transition;
69
+
70
+ &:hover {
71
+ background-color: $observ-gray-50;
72
+ border-color: $observ-gray-400;
73
+ }
74
+
75
+ &:active {
76
+ background-color: $observ-gray-100;
77
+ }
78
+
79
+ &--sm {
80
+ padding: 0.375rem 0.75rem;
81
+ font-size: $observ-font-size-xs;
82
+ }
83
+
84
+ &--primary {
85
+ background-color: $observ-primary;
86
+ border-color: $observ-primary;
87
+ color: $observ-white;
88
+
89
+ &:hover {
90
+ background-color: darken($observ-primary, 5%);
91
+ border-color: darken($observ-primary, 5%);
92
+ }
93
+
94
+ &:disabled {
95
+ background-color: $observ-gray-300;
96
+ border-color: $observ-gray-300;
97
+ color: $observ-gray-500;
98
+ cursor: not-allowed;
99
+
100
+ &:hover {
101
+ background-color: $observ-gray-300;
102
+ border-color: $observ-gray-300;
103
+ }
104
+ }
105
+ }
106
+
107
+ &--secondary {
108
+ background-color: $observ-gray-100;
109
+ border-color: $observ-gray-300;
110
+ color: $observ-gray-700;
111
+
112
+ &:hover {
113
+ background-color: $observ-gray-200;
114
+ border-color: $observ-gray-400;
115
+ }
116
+ }
117
+
118
+ &--danger {
119
+ background-color: $observ-white;
120
+ border-color: $observ-danger;
121
+ color: $observ-danger;
122
+
123
+ &:hover {
124
+ background-color: $observ-danger;
125
+ border-color: $observ-danger;
126
+ color: $observ-white;
127
+ }
128
+ }
129
+ }
130
+
131
+ // Ensure button elements inside forms (from button_to) match link styles
132
+ button.observ-button {
133
+ appearance: none;
134
+ -webkit-appearance: none;
135
+ }
136
+
137
+ .observ-link {
138
+ color: $observ-primary;
139
+ text-decoration: none;
140
+ font-weight: 500;
141
+ transition: $observ-transition;
142
+
143
+ &:hover {
144
+ color: darken($observ-primary, 10%);
145
+ text-decoration: underline;
146
+ }
147
+ }
148
+
149
+ .observ-code {
150
+ font-family: $observ-font-family-mono;
151
+ background-color: $observ-gray-100;
152
+ padding: 0.125rem 0.375rem;
153
+ border-radius: $observ-border-radius-sm;
154
+ font-size: 0.875em;
155
+
156
+ &--inline {
157
+ display: inline;
158
+ }
159
+ }
160
+
161
+ .observ-code-block {
162
+ font-family: $observ-font-family-mono;
163
+ background-color: $observ-gray-900;
164
+ color: $observ-gray-100;
165
+ padding: $observ-spacing-md;
166
+ border-radius: $observ-border-radius;
167
+ overflow-x: auto;
168
+ font-size: $observ-font-size-sm;
169
+ line-height: 1.6;
170
+ margin: 0;
171
+ white-space: pre-wrap;
172
+ word-wrap: break-word;
173
+ }
174
+
175
+ .observ-model-badge {
176
+ display: inline-flex;
177
+ align-items: center;
178
+ padding: 0.25rem 0.5rem;
179
+ background-color: lighten($observ-primary, 45%);
180
+ color: darken($observ-primary, 10%);
181
+ border-radius: $observ-border-radius-sm;
182
+ font-size: $observ-font-size-xs;
183
+ font-weight: 600;
184
+ font-family: $observ-font-family-mono;
185
+ }
186
+
187
+ .observ-text {
188
+ &--muted {
189
+ color: $observ-gray-500;
190
+ }
191
+
192
+ &--small {
193
+ font-size: $observ-font-size-sm;
194
+ }
195
+ }
196
+
197
+ .observ-definition-list {
198
+ display: flex;
199
+ flex-direction: column;
200
+ gap: $observ-spacing-md;
201
+ margin: 0;
202
+
203
+ &__item {
204
+ display: flex;
205
+ flex-direction: column;
206
+ gap: $observ-spacing-xs;
207
+ }
208
+
209
+ &__term {
210
+ font-size: $observ-font-size-xs;
211
+ font-weight: 600;
212
+ color: $observ-gray-500;
213
+ text-transform: uppercase;
214
+ letter-spacing: 0.05em;
215
+ margin: 0;
216
+ }
217
+
218
+ &__definition {
219
+ font-size: $observ-font-size-base;
220
+ color: $observ-gray-900;
221
+ margin: 0;
222
+ }
223
+
224
+ &--horizontal {
225
+ flex-direction: row;
226
+ flex-wrap: wrap;
227
+
228
+ .observ-definition-list__item {
229
+ flex-direction: row;
230
+ align-items: baseline;
231
+ gap: $observ-spacing-sm;
232
+ flex: 0 0 auto;
233
+ min-width: 150px;
234
+ }
235
+ }
236
+ }
237
+
238
+ .observ-input,
239
+ .observ-select {
240
+ width: 100%;
241
+ max-width: 600px;
242
+ padding: $observ-spacing-sm $observ-spacing-md;
243
+ border: 1px solid $observ-gray-300;
244
+ border-radius: $observ-border-radius-sm;
245
+ background-color: $observ-white;
246
+ color: $observ-gray-900;
247
+ font-size: $observ-font-size-sm;
248
+ line-height: 1.5;
249
+ transition: $observ-transition;
250
+ font-family: inherit;
251
+
252
+ &:hover {
253
+ border-color: $observ-gray-400;
254
+ }
255
+
256
+ &:focus {
257
+ outline: none;
258
+ border-color: $observ-primary;
259
+ box-shadow: 0 0 0 3px rgba($observ-primary, 0.1);
260
+ }
261
+
262
+ &::placeholder {
263
+ color: $observ-gray-400;
264
+ }
265
+
266
+ &:disabled {
267
+ background-color: $observ-gray-50;
268
+ color: $observ-gray-500;
269
+ cursor: not-allowed;
270
+ }
271
+ }
272
+
273
+ .observ-input--wide {
274
+ max-width: 800px;
275
+ }
276
+
277
+ select.observ-input,
278
+ .observ-select {
279
+ appearance: none;
280
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 12 12'%3E%3Cpath fill='%236b7280' d='M6 9L1 4h10z'/%3E%3C/svg%3E");
281
+ background-repeat: no-repeat;
282
+ background-position: right $observ-spacing-sm center;
283
+ padding-right: calc(#{$observ-spacing-md} + 1.5rem);
284
+ cursor: pointer;
285
+ }
286
+
287
+ .observ-label {
288
+ display: block;
289
+ margin-bottom: $observ-spacing-xs;
290
+ font-size: $observ-font-size-sm;
291
+ font-weight: 500;
292
+ color: $observ-gray-700;
293
+ }
294
+
295
+ .observ-form-group {
296
+ margin-bottom: $observ-spacing-md;
297
+ }
298
+
299
+ // BEM-style form components
300
+ .observ-form {
301
+ &__group {
302
+ margin-bottom: $observ-spacing-lg;
303
+ }
304
+
305
+ &__label {
306
+ display: block;
307
+ margin-bottom: $observ-spacing-xs;
308
+ font-size: $observ-font-size-sm;
309
+ font-weight: 500;
310
+ color: $observ-gray-700;
311
+ }
312
+
313
+ &__input,
314
+ &__select,
315
+ &__textarea {
316
+ width: 100%;
317
+ padding: $observ-spacing-sm $observ-spacing-md;
318
+ font-size: $observ-font-size-base;
319
+ color: $observ-gray-900;
320
+ background-color: #fff;
321
+ border: 1px solid $observ-gray-300;
322
+ border-radius: $observ-border-radius-sm;
323
+ transition: border-color 0.2s ease, box-shadow 0.2s ease;
324
+
325
+ &:focus {
326
+ outline: none;
327
+ border-color: $observ-primary;
328
+ box-shadow: 0 0 0 3px rgba($observ-primary, 0.1);
329
+ }
330
+
331
+ &::placeholder {
332
+ color: $observ-gray-400;
333
+ }
334
+
335
+ &:disabled {
336
+ background-color: $observ-gray-100;
337
+ cursor: not-allowed;
338
+ }
339
+ }
340
+
341
+ &__textarea {
342
+ resize: vertical;
343
+ min-height: 80px;
344
+
345
+ &--code {
346
+ font-family: $observ-font-family-mono;
347
+ font-size: $observ-font-size-sm;
348
+ }
349
+ }
350
+
351
+ &__select {
352
+ cursor: pointer;
353
+ appearance: none;
354
+ background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='12' height='12' viewBox='0 0 12 12'%3E%3Cpath fill='%236b7280' d='M6 8L1 3h10z'/%3E%3C/svg%3E");
355
+ background-repeat: no-repeat;
356
+ background-position: right $observ-spacing-md center;
357
+ padding-right: calc($observ-spacing-md + 1.5rem);
358
+ }
359
+
360
+ &__hint {
361
+ margin-top: $observ-spacing-xs;
362
+ font-size: $observ-font-size-sm;
363
+ color: $observ-gray-500;
364
+
365
+ &--warning {
366
+ color: $observ-warning;
367
+ }
368
+ }
369
+
370
+ &__actions {
371
+ display: flex;
372
+ align-items: center;
373
+ gap: $observ-spacing-md;
374
+ padding-top: $observ-spacing-lg;
375
+ border-top: 1px solid $observ-gray-200;
376
+ margin-top: $observ-spacing-lg;
377
+ }
378
+ }
379
+
380
+ .observ-alert__title {
381
+ margin: 0 0 $observ-spacing-sm 0;
382
+ font-size: $observ-font-size-base;
383
+ font-weight: 600;
384
+ }
385
+
386
+ .observ-alert__list {
387
+ margin: 0;
388
+ padding-left: $observ-spacing-lg;
389
+ }
390
+
391
+ .observ-form-separator {
392
+ border-top: 1px solid $observ-gray-200;
393
+ padding-top: $observ-spacing-lg;
394
+ margin-top: $observ-spacing-lg;
395
+ }
396
+
397
+ .observ-form-actions {
398
+ display: flex;
399
+ align-items: center;
400
+ gap: $observ-spacing-md;
401
+ }
402
+
403
+ .observ-loading-indicator {
404
+ display: inline-flex;
405
+ align-items: center;
406
+ gap: $observ-spacing-sm;
407
+ color: $observ-gray-600;
408
+ font-size: $observ-font-size-sm;
409
+ }
410
+
411
+ .observ-spinner {
412
+ display: inline-block;
413
+ width: 1rem;
414
+ height: 1rem;
415
+ border: 2px solid $observ-gray-300;
416
+ border-top-color: $observ-primary;
417
+ border-radius: 50%;
418
+ animation: observ-spin 0.8s linear infinite;
419
+ }
420
+
421
+ @keyframes observ-spin {
422
+ to {
423
+ transform: rotate(360deg);
424
+ }
425
+ }
426
+
427
+ .observ-tabs {
428
+ margin-bottom: $observ-spacing-lg;
429
+
430
+ &__list {
431
+ display: flex;
432
+ gap: $observ-spacing-sm;
433
+ border-bottom: 2px solid $observ-gray-200;
434
+ }
435
+
436
+ &__tab {
437
+ padding: $observ-spacing-sm $observ-spacing-lg;
438
+ color: $observ-gray-600;
439
+ text-decoration: none;
440
+ font-weight: 500;
441
+ border-bottom: 2px solid transparent;
442
+ margin-bottom: -2px;
443
+ transition: $observ-transition;
444
+
445
+ &:hover {
446
+ color: $observ-gray-900;
447
+ border-bottom-color: $observ-gray-300;
448
+ }
449
+
450
+ &--active {
451
+ color: $observ-primary;
452
+ border-bottom-color: $observ-primary;
453
+
454
+ &:hover {
455
+ color: $observ-primary;
456
+ border-bottom-color: $observ-primary;
457
+ }
458
+ }
459
+ }
460
+ }
@@ -0,0 +1,40 @@
1
+ @import 'variables';
2
+
3
+ .observ-dashboard {
4
+ display: flex;
5
+ flex-direction: column;
6
+ gap: $observ-spacing-xl;
7
+
8
+ &__grid {
9
+ display: grid;
10
+ grid-template-columns: repeat(2, 1fr);
11
+ gap: $observ-spacing-xl;
12
+
13
+ @media (max-width: $observ-breakpoint-md) {
14
+ grid-template-columns: 1fr;
15
+ }
16
+ }
17
+ }
18
+
19
+ .observ-period-selector {
20
+ &__select {
21
+ padding: $observ-spacing-sm $observ-spacing-md;
22
+ border: 1px solid $observ-gray-300;
23
+ border-radius: $observ-border-radius-sm;
24
+ background-color: $observ-white;
25
+ font-size: $observ-font-size-sm;
26
+ color: $observ-gray-700;
27
+ cursor: pointer;
28
+ transition: $observ-transition;
29
+
30
+ &:hover {
31
+ border-color: $observ-gray-400;
32
+ }
33
+
34
+ &:focus {
35
+ outline: none;
36
+ border-color: $observ-primary;
37
+ box-shadow: 0 0 0 3px rgba($observ-primary, 0.1);
38
+ }
39
+ }
40
+ }