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,273 @@
1
+ @import 'variables';
2
+
3
+ .observ-drawer {
4
+ position: fixed;
5
+ top: 0;
6
+ left: 0;
7
+ right: 0;
8
+ bottom: 0;
9
+ z-index: 1000;
10
+ pointer-events: none;
11
+
12
+ &.open {
13
+ pointer-events: auto;
14
+
15
+ .observ-drawer__overlay {
16
+ opacity: 1;
17
+ }
18
+
19
+ .observ-drawer__panel {
20
+ transform: translateX(0);
21
+ }
22
+ }
23
+ }
24
+
25
+ .observ-drawer__overlay {
26
+ position: absolute;
27
+ top: 0;
28
+ left: 0;
29
+ right: 0;
30
+ bottom: 0;
31
+ background-color: rgba($observ-black, 0.5);
32
+ opacity: 0;
33
+ transition: opacity 0.3s ease-in-out;
34
+ cursor: pointer;
35
+ }
36
+
37
+ .observ-drawer__panel {
38
+ position: absolute;
39
+ top: 0;
40
+ right: 0;
41
+ bottom: 0;
42
+ width: 100%;
43
+ max-width: 600px;
44
+ background-color: $observ-white;
45
+ box-shadow: $observ-shadow-lg;
46
+ transform: translateX(100%);
47
+ transition: transform 0.3s ease-in-out;
48
+ display: flex;
49
+ flex-direction: column;
50
+ overflow: hidden;
51
+
52
+ @media (max-width: $observ-breakpoint-md) {
53
+ max-width: 90%;
54
+ }
55
+ }
56
+
57
+ .observ-drawer__header {
58
+ display: flex;
59
+ align-items: center;
60
+ gap: $observ-spacing-md;
61
+ padding: $observ-spacing-lg;
62
+ border-bottom: 1px solid $observ-gray-200;
63
+ background-color: $observ-white;
64
+ flex-shrink: 0;
65
+ }
66
+
67
+ .observ-drawer__close-btn {
68
+ display: flex;
69
+ align-items: center;
70
+ justify-content: center;
71
+ width: 2rem;
72
+ height: 2rem;
73
+ padding: 0;
74
+ border: none;
75
+ background: none;
76
+ color: $observ-gray-600;
77
+ cursor: pointer;
78
+ border-radius: $observ-border-radius-sm;
79
+ transition: $observ-transition;
80
+ flex-shrink: 0;
81
+
82
+ &:hover {
83
+ background-color: $observ-gray-100;
84
+ color: $observ-gray-900;
85
+ }
86
+
87
+ &:active {
88
+ background-color: $observ-gray-200;
89
+ }
90
+
91
+ svg {
92
+ display: block;
93
+ }
94
+ }
95
+
96
+ .observ-drawer__title {
97
+ margin: 0;
98
+ font-size: $observ-font-size-xl;
99
+ font-weight: 600;
100
+ color: $observ-gray-900;
101
+ flex: 1;
102
+ min-width: 0;
103
+
104
+ .observ-spinner {
105
+ width: 1.25rem;
106
+ height: 1.25rem;
107
+ }
108
+ }
109
+
110
+ .observ-drawer__content {
111
+ flex: 1;
112
+ overflow-y: auto;
113
+ padding: $observ-spacing-lg;
114
+ background-color: $observ-white;
115
+ }
116
+
117
+ .observ-drawer__loading {
118
+ display: flex;
119
+ align-items: center;
120
+ justify-content: center;
121
+ padding: $observ-spacing-2xl;
122
+ color: $observ-gray-500;
123
+
124
+ .observ-spinner {
125
+ width: 2rem;
126
+ height: 2rem;
127
+ }
128
+ }
129
+
130
+ .observ-drawer__error {
131
+ padding: $observ-spacing-lg;
132
+ background-color: lighten($observ-danger, 45%);
133
+ color: darken($observ-danger, 10%);
134
+ border-radius: $observ-border-radius;
135
+ text-align: center;
136
+ }
137
+
138
+ // Text output specific styles
139
+ .observ-drawer__text-output {
140
+ position: relative;
141
+ width: 100%;
142
+ }
143
+
144
+ .observ-drawer__textarea {
145
+ width: 100%;
146
+ min-height: 600px;
147
+ padding: $observ-spacing-md;
148
+ border: 1px solid $observ-gray-300;
149
+ border-radius: $observ-border-radius;
150
+ font-family: $observ-font-family-mono;
151
+ font-size: 12px;
152
+ line-height: 1.6;
153
+ resize: vertical;
154
+ background-color: $observ-gray-50;
155
+ color: $observ-gray-900;
156
+
157
+ &:focus {
158
+ outline: none;
159
+ border-color: $observ-primary;
160
+ box-shadow: 0 0 0 3px rgba($observ-primary, 0.1);
161
+ }
162
+ }
163
+
164
+ .observ-drawer__copy-btn {
165
+ position: absolute;
166
+ top: $observ-spacing-sm;
167
+ right: $observ-spacing-sm;
168
+ }
169
+
170
+ .observ-drawer__instructions {
171
+ margin-bottom: $observ-spacing-md;
172
+ padding: $observ-spacing-sm;
173
+ background-color: lighten($observ-info, 45%);
174
+ border-left: 3px solid $observ-info;
175
+ border-radius: $observ-border-radius-sm;
176
+ font-size: $observ-font-size-sm;
177
+ color: $observ-gray-700;
178
+ }
179
+
180
+ .observ-drawer__help {
181
+ margin-top: $observ-spacing-md;
182
+
183
+ details {
184
+ cursor: pointer;
185
+
186
+ summary {
187
+ color: $observ-gray-600;
188
+ font-size: $observ-font-size-sm;
189
+ font-weight: 500;
190
+ padding: $observ-spacing-xs;
191
+
192
+ &:hover {
193
+ color: $observ-gray-900;
194
+ }
195
+ }
196
+ }
197
+
198
+ &-list {
199
+ margin-top: $observ-spacing-sm;
200
+ padding-left: $observ-spacing-lg;
201
+ font-size: $observ-font-size-sm;
202
+ color: $observ-gray-600;
203
+
204
+ li {
205
+ margin-bottom: $observ-spacing-xs;
206
+ }
207
+ }
208
+ }
209
+
210
+ // Field styles for structured content display
211
+ .observ-drawer__field {
212
+ margin-bottom: $observ-spacing-lg;
213
+
214
+ &:last-child {
215
+ margin-bottom: 0;
216
+ }
217
+ }
218
+
219
+ .observ-drawer__field-label {
220
+ display: block;
221
+ margin-bottom: $observ-spacing-xs;
222
+ font-size: $observ-font-size-sm;
223
+ font-weight: 600;
224
+ color: $observ-gray-700;
225
+ text-transform: uppercase;
226
+ letter-spacing: 0.025em;
227
+ }
228
+
229
+ .observ-drawer__field-value {
230
+ display: flex;
231
+ align-items: center;
232
+ gap: $observ-spacing-sm;
233
+ }
234
+
235
+ // Metrics grid for displaying stats
236
+ .observ-drawer__metrics {
237
+ display: grid;
238
+ grid-template-columns: repeat(3, 1fr);
239
+ gap: $observ-spacing-md;
240
+ margin-bottom: $observ-spacing-lg;
241
+ padding: $observ-spacing-md;
242
+ background-color: $observ-gray-50;
243
+ border-radius: $observ-border-radius;
244
+ }
245
+
246
+ .observ-drawer__metric {
247
+ text-align: center;
248
+ }
249
+
250
+ .observ-drawer__metric-label {
251
+ display: block;
252
+ font-size: $observ-font-size-xs;
253
+ color: $observ-gray-500;
254
+ text-transform: uppercase;
255
+ letter-spacing: 0.05em;
256
+ margin-bottom: $observ-spacing-xs;
257
+ }
258
+
259
+ .observ-drawer__metric-value {
260
+ display: block;
261
+ font-size: $observ-font-size-lg;
262
+ font-weight: 600;
263
+ color: $observ-gray-900;
264
+ }
265
+
266
+ // Actions footer
267
+ .observ-drawer__actions {
268
+ display: flex;
269
+ gap: $observ-spacing-sm;
270
+ margin-top: $observ-spacing-lg;
271
+ padding-top: $observ-spacing-lg;
272
+ border-top: 1px solid $observ-gray-200;
273
+ }
@@ -0,0 +1,120 @@
1
+ // Custom JSON Viewer Styles
2
+ // Fully custom implementation with proper newline handling
3
+
4
+ .observ-json-viewer {
5
+ font-family: $observ-font-family-mono;
6
+ background-color: $observ-gray-900;
7
+ color: $observ-gray-100;
8
+ padding: $observ-spacing-md;
9
+ border-radius: $observ-border-radius;
10
+ overflow-x: auto;
11
+ font-size: $observ-font-size-sm;
12
+ line-height: 1.6;
13
+ margin: 0;
14
+
15
+ .json-container {
16
+ white-space: pre-wrap;
17
+ word-wrap: break-word;
18
+ }
19
+
20
+ .json-line {
21
+ display: block;
22
+
23
+ &.json-footer {
24
+ // Closing braces/brackets
25
+ }
26
+ }
27
+
28
+ .json-object,
29
+ .json-array {
30
+ display: inline-block;
31
+ vertical-align: top;
32
+ }
33
+
34
+ .json-content {
35
+ display: block;
36
+ }
37
+
38
+ // Toggle buttons (▸/▾)
39
+ .json-toggle {
40
+ display: inline-block;
41
+ width: 1em;
42
+ margin-right: 0.25em;
43
+ color: $observ-gray-400;
44
+ cursor: pointer;
45
+ user-select: none;
46
+
47
+ &:hover {
48
+ color: $observ-primary;
49
+ }
50
+ }
51
+
52
+ // Indentation
53
+ .json-indent {
54
+ display: inline-block;
55
+ white-space: pre;
56
+ }
57
+
58
+ // Item count when collapsed
59
+ .json-item-count {
60
+ color: $observ-gray-500;
61
+ font-style: italic;
62
+ font-size: 0.9em;
63
+ margin-left: 0.25em;
64
+ }
65
+
66
+ // Keys
67
+ .json-key {
68
+ color: $observ-info;
69
+ font-weight: 500;
70
+ }
71
+
72
+ // String values
73
+ .json-string {
74
+ color: $observ-success;
75
+ display: inline;
76
+
77
+ .json-string-content {
78
+ white-space: pre-wrap;
79
+ word-break: break-word;
80
+ }
81
+
82
+ .json-quote {
83
+ opacity: 0.7;
84
+ }
85
+ }
86
+
87
+ // Number values
88
+ .json-number {
89
+ color: #ff9800;
90
+ }
91
+
92
+ // Boolean values
93
+ .json-boolean {
94
+ color: #9c27b0;
95
+ font-weight: 500;
96
+ }
97
+
98
+ // Null values
99
+ .json-null {
100
+ color: $observ-gray-500;
101
+ font-style: italic;
102
+ }
103
+
104
+ // Punctuation (brackets, braces, colons, commas)
105
+ .json-punctuation {
106
+ color: $observ-gray-400;
107
+ }
108
+
109
+ // Unknown/fallback
110
+ .json-unknown {
111
+ color: $observ-warning;
112
+ }
113
+ }
114
+
115
+ // Compact variant for use in cards
116
+ .observ-json-viewer--compact {
117
+ padding: $observ-spacing-sm;
118
+ font-size: $observ-font-size-xs;
119
+ line-height: 1.5;
120
+ }
@@ -0,0 +1,256 @@
1
+ @import 'variables';
2
+
3
+ .observ-layout {
4
+ min-height: 100vh;
5
+ display: flex;
6
+ flex-direction: column;
7
+ background-color: $observ-gray-50;
8
+ font-family: $observ-font-family;
9
+ color: $observ-gray-900;
10
+ font-size: $observ-font-size-base;
11
+ line-height: 1.5;
12
+ }
13
+
14
+ .observ-nav {
15
+ background-color: $observ-white;
16
+ border-bottom: 1px solid $observ-gray-200;
17
+ box-shadow: $observ-shadow-sm;
18
+
19
+ &__container {
20
+ max-width: 1440px;
21
+ margin: 0 auto;
22
+ padding: 0 $observ-spacing-lg;
23
+ display: flex;
24
+ align-items: center;
25
+ justify-content: space-between;
26
+ height: 64px;
27
+ }
28
+
29
+ &__brand {
30
+ font-size: $observ-font-size-xl;
31
+ font-weight: 600;
32
+ }
33
+
34
+ &__brand-link {
35
+ color: $observ-gray-900;
36
+ text-decoration: none;
37
+ transition: $observ-transition;
38
+
39
+ &:hover {
40
+ color: $observ-primary;
41
+ }
42
+ }
43
+
44
+ &__menu {
45
+ display: flex;
46
+ list-style: none;
47
+ margin: 0;
48
+ padding: 0;
49
+ gap: $observ-spacing-md;
50
+ }
51
+
52
+ &__item {
53
+ margin: 0;
54
+
55
+ &--dropdown {
56
+ position: relative;
57
+ }
58
+ }
59
+
60
+ &__dropdown-toggle {
61
+ background: none;
62
+ border: none;
63
+ cursor: pointer;
64
+ font-family: inherit;
65
+ font-size: inherit;
66
+ }
67
+
68
+ &__dropdown-arrow {
69
+ margin-left: 4px;
70
+ font-size: 0.75em;
71
+ transition: transform 0.2s;
72
+ }
73
+
74
+ &__dropdown-menu {
75
+ position: absolute;
76
+ top: 100%;
77
+ left: 0;
78
+ margin-top: 8px;
79
+ background-color: $observ-white;
80
+ border: 1px solid $observ-gray-200;
81
+ border-radius: $observ-border-radius;
82
+ box-shadow: $observ-shadow-md;
83
+ list-style: none;
84
+ padding: $observ-spacing-xs 0;
85
+ min-width: 160px;
86
+ opacity: 0;
87
+ visibility: hidden;
88
+ transform: translateY(-8px);
89
+ transition: all 0.2s ease;
90
+ z-index: 1000;
91
+
92
+ &--open {
93
+ opacity: 1;
94
+ visibility: visible;
95
+ transform: translateY(0);
96
+ }
97
+
98
+ li {
99
+ margin: 0;
100
+ }
101
+ }
102
+
103
+ &__dropdown-link {
104
+ display: block;
105
+ padding: $observ-spacing-sm $observ-spacing-md;
106
+ color: $observ-gray-700;
107
+ text-decoration: none;
108
+ transition: $observ-transition;
109
+ font-weight: 500;
110
+ white-space: nowrap;
111
+
112
+ &:hover {
113
+ background-color: $observ-gray-100;
114
+ color: $observ-gray-900;
115
+ }
116
+ }
117
+
118
+ &__link {
119
+ display: block;
120
+ padding: $observ-spacing-sm $observ-spacing-md;
121
+ color: $observ-gray-600;
122
+ text-decoration: none;
123
+ border-radius: $observ-border-radius-sm;
124
+ transition: $observ-transition;
125
+ font-weight: 500;
126
+
127
+ &:hover {
128
+ background-color: $observ-gray-100;
129
+ color: $observ-gray-900;
130
+ }
131
+
132
+ &--active {
133
+ background-color: $observ-primary;
134
+ color: $observ-white;
135
+
136
+ &:hover {
137
+ background-color: darken($observ-primary, 5%);
138
+ color: $observ-white;
139
+ }
140
+ }
141
+
142
+ &--secondary {
143
+ color: $observ-gray-500;
144
+ font-size: $observ-font-size-sm;
145
+ }
146
+ }
147
+
148
+ &__actions {
149
+ display: flex;
150
+ gap: $observ-spacing-sm;
151
+ }
152
+ }
153
+
154
+ .observ-main {
155
+ flex: 1;
156
+ max-width: 1440px;
157
+ width: 100%;
158
+ margin: 0 auto;
159
+ padding: $observ-spacing-xl $observ-spacing-lg;
160
+ }
161
+
162
+ .observ-container {
163
+ display: flex;
164
+ flex-direction: column;
165
+ gap: $observ-spacing-xl;
166
+ }
167
+
168
+ .observ-footer {
169
+ background-color: $observ-white;
170
+ border-top: 1px solid $observ-gray-200;
171
+ margin-top: auto;
172
+
173
+ &__container {
174
+ max-width: 1440px;
175
+ margin: 0 auto;
176
+ padding: $observ-spacing-lg;
177
+ text-align: center;
178
+ }
179
+
180
+ &__text {
181
+ color: $observ-gray-500;
182
+ font-size: $observ-font-size-sm;
183
+ margin: 0;
184
+ }
185
+ }
186
+
187
+ .observ-page-header {
188
+ margin-bottom: $observ-spacing-xl;
189
+ display: flex;
190
+ align-items: center;
191
+ gap: $observ-spacing-md;
192
+
193
+ &__content {
194
+ display: flex;
195
+ align-items: center;
196
+ flex-wrap: wrap;
197
+ gap: $observ-spacing-md;
198
+ flex: 1;
199
+ }
200
+
201
+ &__breadcrumb {
202
+ color: $observ-gray-500;
203
+ font-size: $observ-font-size-sm;
204
+ margin-bottom: $observ-spacing-xs;
205
+ }
206
+
207
+ &__title {
208
+ font-size: $observ-font-size-3xl;
209
+ font-weight: 700;
210
+ margin: 0;
211
+ color: $observ-gray-900;
212
+ }
213
+
214
+ &__meta {
215
+ display: flex;
216
+ align-items: center;
217
+ gap: $observ-spacing-sm;
218
+ }
219
+
220
+ &__actions {
221
+ display: flex;
222
+ gap: $observ-spacing-sm;
223
+ align-items: center;
224
+ }
225
+ }
226
+
227
+ .observ-alert {
228
+ padding: $observ-spacing-md;
229
+ border-radius: $observ-border-radius;
230
+ margin-bottom: $observ-spacing-lg;
231
+ font-size: $observ-font-size-sm;
232
+
233
+ &--success {
234
+ background-color: lighten($observ-success, 45%);
235
+ color: darken($observ-success, 20%);
236
+ border: 1px solid lighten($observ-success, 30%);
237
+ }
238
+
239
+ &--danger {
240
+ background-color: lighten($observ-danger, 45%);
241
+ color: darken($observ-danger, 20%);
242
+ border: 1px solid lighten($observ-danger, 30%);
243
+ }
244
+
245
+ &--warning {
246
+ background-color: lighten($observ-warning, 45%);
247
+ color: darken($observ-warning, 30%);
248
+ border: 1px solid lighten($observ-warning, 30%);
249
+ }
250
+
251
+ &--info {
252
+ background-color: lighten($observ-info, 45%);
253
+ color: darken($observ-info, 20%);
254
+ border: 1px solid lighten($observ-info, 30%);
255
+ }
256
+ }