e11y 0.2.0 → 1.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.
Files changed (288) hide show
  1. checksums.yaml +4 -4
  2. data/.rubocop.yml +130 -10
  3. data/CHANGELOG.md +80 -1
  4. data/CLAUDE.md +168 -0
  5. data/CONTRIBUTING.md +640 -0
  6. data/README.md +165 -701
  7. data/RELEASE.md +41 -12
  8. data/Rakefile +249 -57
  9. data/config/README.md +1 -1
  10. data/config/loki-local-config.yaml +12 -0
  11. data/config/otel-collector-config.yaml +44 -0
  12. data/cucumber.yml +1 -0
  13. data/docker-compose.yml +18 -2
  14. data/docs/ADAPTERS.md +76 -0
  15. data/docs/ADAPTIVE_SAMPLING.md +59 -0
  16. data/docs/COMPARISON.md +104 -0
  17. data/docs/CONFIGURATION.md +52 -0
  18. data/docs/DISTRIBUTED_TRACING.md +44 -0
  19. data/docs/LIMITATIONS.md +13 -0
  20. data/docs/METRICS_DSL.md +84 -0
  21. data/docs/PERFORMANCE.md +60 -0
  22. data/docs/PII_FILTERING.md +40 -0
  23. data/docs/PRESETS.md +65 -0
  24. data/docs/QUICK-START.md +546 -587
  25. data/docs/RAILS_INTEGRATION.md +79 -0
  26. data/docs/SCHEMA_VALIDATION.md +63 -0
  27. data/docs/SLO-PROMQL-ALERTS.md +161 -0
  28. data/docs/TESTING.md +69 -0
  29. data/docs/{ADR-001-architecture.md → architecture/ADR-001-architecture.md} +36 -65
  30. data/docs/{ADR-002-metrics-yabeda.md → architecture/ADR-002-metrics-yabeda.md} +62 -236
  31. data/docs/architecture/ADR-003-slo-observability.md +1402 -0
  32. data/docs/{ADR-004-adapter-architecture.md → architecture/ADR-004-adapter-architecture.md} +163 -146
  33. data/docs/{ADR-005-tracing-context.md → architecture/ADR-005-tracing-context.md} +10 -9
  34. data/docs/{ADR-006-security-compliance.md → architecture/ADR-006-security-compliance.md} +184 -191
  35. data/docs/{ADR-007-opentelemetry-integration.md → architecture/ADR-007-opentelemetry-integration.md} +3 -21
  36. data/docs/{ADR-008-rails-integration.md → architecture/ADR-008-rails-integration.md} +182 -743
  37. data/docs/{ADR-009-cost-optimization.md → architecture/ADR-009-cost-optimization.md} +45 -54
  38. data/docs/architecture/ADR-010-developer-experience.md +522 -0
  39. data/docs/{ADR-011-testing-strategy.md → architecture/ADR-011-testing-strategy.md} +44 -86
  40. data/docs/{ADR-012-event-evolution.md → architecture/ADR-012-event-evolution.md} +11 -11
  41. data/docs/{ADR-013-reliability-error-handling.md → architecture/ADR-013-reliability-error-handling.md} +37 -12
  42. data/docs/{ADR-014-event-driven-slo.md → architecture/ADR-014-event-driven-slo.md} +12 -24
  43. data/docs/{ADR-015-middleware-order.md → architecture/ADR-015-middleware-order.md} +43 -59
  44. data/docs/{ADR-016-self-monitoring-slo.md → architecture/ADR-016-self-monitoring-slo.md} +58 -355
  45. data/docs/{ADR-017-multi-rails-compatibility.md → architecture/ADR-017-multi-rails-compatibility.md} +4 -11
  46. data/docs/architecture/ADR-018-memory-optimization.md +366 -0
  47. data/docs/{ADR-INDEX.md → architecture/ADR-INDEX.md} +11 -6
  48. data/docs/plans/2026-03-20-browser-overlay-svelte.md +281 -0
  49. data/docs/{00-ICP-AND-TIMELINE.md → prd/00-ICP-AND-TIMELINE.md} +6 -6
  50. data/docs/{01-SCALE-REQUIREMENTS.md → prd/01-SCALE-REQUIREMENTS.md} +6 -6
  51. data/docs/prd/01-overview-vision.md +19 -14
  52. data/docs/use_cases/README.md +22 -23
  53. data/docs/use_cases/UC-001-request-scoped-debug-buffering.md +50 -44
  54. data/docs/use_cases/UC-002-business-event-tracking.md +26 -95
  55. data/docs/use_cases/UC-003-event-metrics.md +66 -0
  56. data/docs/use_cases/UC-004-zero-config-slo-tracking.md +33 -684
  57. data/docs/use_cases/UC-005-sentry-integration.md +13 -15
  58. data/docs/use_cases/UC-006-trace-context-management.md +30 -28
  59. data/docs/use_cases/UC-007-pii-filtering.md +35 -87
  60. data/docs/use_cases/UC-008-opentelemetry-integration.md +51 -89
  61. data/docs/use_cases/UC-009-multi-service-tracing.md +30 -178
  62. data/docs/use_cases/UC-010-background-job-tracking.md +24 -91
  63. data/docs/use_cases/UC-011-rate-limiting.md +95 -168
  64. data/docs/use_cases/UC-012-audit-trail.md +21 -46
  65. data/docs/use_cases/UC-013-high-cardinality-protection.md +29 -167
  66. data/docs/use_cases/UC-014-adaptive-sampling.md +2 -2
  67. data/docs/use_cases/UC-015-cost-optimization.md +46 -99
  68. data/docs/use_cases/UC-016-rails-logger-migration.md +39 -213
  69. data/docs/use_cases/UC-017-local-development.md +203 -777
  70. data/docs/use_cases/UC-018-testing-events.md +3 -3
  71. data/docs/use_cases/UC-019-retention-based-routing.md +53 -106
  72. data/docs/use_cases/UC-020-event-versioning.md +8 -9
  73. data/docs/use_cases/UC-021-error-handling-retry-dlq.md +18 -22
  74. data/docs/use_cases/UC-022-event-registry.md +15 -21
  75. data/docs/use_cases/backlog.md +119 -87
  76. data/e11y.gemspec +2 -2
  77. data/gems/e11y-devtools/README.md +158 -0
  78. data/gems/e11y-devtools/config/routes.rb +15 -0
  79. data/gems/e11y-devtools/e11y-devtools.gemspec +25 -0
  80. data/gems/e11y-devtools/exe/e11y +34 -0
  81. data/gems/e11y-devtools/frontend/.gitignore +24 -0
  82. data/gems/e11y-devtools/frontend/README.md +51 -0
  83. data/gems/e11y-devtools/frontend/index.html +14 -0
  84. data/gems/e11y-devtools/frontend/package-lock.json +3707 -0
  85. data/gems/e11y-devtools/frontend/package.json +28 -0
  86. data/gems/e11y-devtools/frontend/public/mocks/v1/events/recent.json +4205 -0
  87. data/gems/e11y-devtools/frontend/public/mocks/v1/interactions.json +194 -0
  88. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/0a2e04027cfa22d014bc22e8b27cd913/events.json +86 -0
  89. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/0e1543af6a630fb3af6b52283154b3e0/events.json +169 -0
  90. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/1838b691faa49564f97db8592ff3978d/events.json +78 -0
  91. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/29f198f6588dacffb687777eb5f8f118/events.json +197 -0
  92. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/34bc3c9c0097de28a7a6f99b90a8e7bc/events.json +194 -0
  93. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/3ba6c20d068ab9cee00e51b180e66444/events.json +184 -0
  94. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/435bfd8f17b9009146a79812d7c3726d/events.json +144 -0
  95. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/4c7676e3fe668e99edb2b94d7d5678a9/events.json +222 -0
  96. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/6daf0d47974bedfc55d5de7004a3ea9f/events.json +194 -0
  97. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8a81ada42834d15f287bb40010043605/events.json +194 -0
  98. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8c0a98900edaae105469df8daedccf02/events.json +198 -0
  99. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8e4f645180f8a7d1dce426b07380466b/events.json +222 -0
  100. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/93db346fa5d44a032605a13b627f4b80/events.json +128 -0
  101. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/98ff6146faf7bd9be8bd03a8275817ba/events.json +223 -0
  102. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/9997ddd0247bc7e25f2ca7a5c415c93d/events.json +197 -0
  103. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/99e35f8ef3baedd798cc4fd085980ad9/events.json +194 -0
  104. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/b4f3095c1909924cbc98889a86c83d6d/events.json +131 -0
  105. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/b54b7fc32b7575a7110de809d11ccda0/events.json +128 -0
  106. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/c0b48033fa06746bcc5886745e053cff/events.json +169 -0
  107. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/c44649ac76701b4558927cd2305ab535/events.json +169 -0
  108. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/d601ae3320057580a39dbdac2edfdf4a/events.json +248 -0
  109. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/e67e724bab422d2b52eeb49635e512e1/events.json +194 -0
  110. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/e6c72765a28f158a8485b35fa63f73da/events.json +194 -0
  111. data/gems/e11y-devtools/frontend/public/mocks/v1/traces/f541b87405c9a54819b18ebe529f6419/events.json +194 -0
  112. data/gems/e11y-devtools/frontend/scripts/generate_mocks.rb +397 -0
  113. data/gems/e11y-devtools/frontend/src/App.svelte +827 -0
  114. data/gems/e11y-devtools/frontend/src/components/Fab.svelte +19 -0
  115. data/gems/e11y-devtools/frontend/src/components/FilterBar.svelte +38 -0
  116. data/gems/e11y-devtools/frontend/src/components/FullscreenPanel.svelte +82 -0
  117. data/gems/e11y-devtools/frontend/src/components/InteractionsTimeline.svelte +264 -0
  118. data/gems/e11y-devtools/frontend/src/components/RecentHistogram.svelte +354 -0
  119. data/gems/e11y-devtools/frontend/src/lib/api.ts +37 -0
  120. data/gems/e11y-devtools/frontend/src/lib/eventIdentity.ts +12 -0
  121. data/gems/e11y-devtools/frontend/src/lib/format.ts +37 -0
  122. data/gems/e11y-devtools/frontend/src/lib/listFilter.ts +43 -0
  123. data/gems/e11y-devtools/frontend/src/lib/recentVolume.ts +80 -0
  124. data/gems/e11y-devtools/frontend/src/lib/router.ts +12 -0
  125. data/gems/e11y-devtools/frontend/src/lib/transitions.ts +34 -0
  126. data/gems/e11y-devtools/frontend/src/lib/viewportOrigin.ts +25 -0
  127. data/gems/e11y-devtools/frontend/src/main.ts +8 -0
  128. data/gems/e11y-devtools/frontend/src/overlay-entry.ts +24 -0
  129. data/gems/e11y-devtools/frontend/src/overlay.css +1080 -0
  130. data/gems/e11y-devtools/frontend/svelte.config.js +2 -0
  131. data/gems/e11y-devtools/frontend/test_puppeteer.js +41 -0
  132. data/gems/e11y-devtools/frontend/test_scale.js +3 -0
  133. data/gems/e11y-devtools/frontend/tsconfig.app.json +21 -0
  134. data/gems/e11y-devtools/frontend/tsconfig.json +7 -0
  135. data/gems/e11y-devtools/frontend/tsconfig.node.json +26 -0
  136. data/gems/e11y-devtools/frontend/vite.config.ts +36 -0
  137. data/gems/e11y-devtools/lib/e11y/devtools/mcp/server.rb +96 -0
  138. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tool_base.rb +25 -0
  139. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/clear.rb +31 -0
  140. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/errors.rb +35 -0
  141. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/event_detail.rb +33 -0
  142. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/events_by_trace.rb +33 -0
  143. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/interactions.rb +40 -0
  144. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/recent_events.rb +34 -0
  145. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/search.rb +34 -0
  146. data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/stats.rb +30 -0
  147. data/gems/e11y-devtools/lib/e11y/devtools/overlay/assets/overlay.js +20 -0
  148. data/gems/e11y-devtools/lib/e11y/devtools/overlay/controller.rb +94 -0
  149. data/gems/e11y-devtools/lib/e11y/devtools/overlay/engine.rb +26 -0
  150. data/gems/e11y-devtools/lib/e11y/devtools/overlay/middleware.rb +80 -0
  151. data/gems/e11y-devtools/lib/e11y/devtools/overlay/rails_controller.rb +67 -0
  152. data/gems/e11y-devtools/lib/e11y/devtools/tui/app.rb +262 -0
  153. data/gems/e11y-devtools/lib/e11y/devtools/tui/grouping.rb +66 -0
  154. data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/event_detail.rb +62 -0
  155. data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/event_list.rb +70 -0
  156. data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/interaction_list.rb +47 -0
  157. data/gems/e11y-devtools/lib/e11y/devtools/version.rb +8 -0
  158. data/gems/e11y-devtools/lib/e11y/devtools.rb +13 -0
  159. data/gems/e11y-devtools/spec/e11y/devtools/mcp/tools_spec.rb +107 -0
  160. data/gems/e11y-devtools/spec/e11y/devtools/overlay/controller_spec.rb +91 -0
  161. data/gems/e11y-devtools/spec/e11y/devtools/overlay/middleware_spec.rb +46 -0
  162. data/gems/e11y-devtools/spec/e11y/devtools/tui/app_spec.rb +85 -0
  163. data/gems/e11y-devtools/spec/e11y/devtools/tui/grouping_spec.rb +64 -0
  164. data/gems/e11y-devtools/spec/spec_helper.rb +5 -0
  165. data/gems/e11y-devtools/spec/tui/widgets/event_list_spec.rb +44 -0
  166. data/gems/e11y-devtools/spec/tui/widgets/interaction_list_spec.rb +62 -0
  167. data/lib/e11y/adapters/audit_encrypted.rb +53 -11
  168. data/lib/e11y/adapters/base.rb +33 -34
  169. data/lib/e11y/adapters/dev_log/file_store.rb +143 -0
  170. data/lib/e11y/adapters/dev_log/query.rb +219 -0
  171. data/lib/e11y/adapters/dev_log.rb +118 -0
  172. data/lib/e11y/adapters/file.rb +3 -6
  173. data/lib/e11y/adapters/in_memory.rb +52 -5
  174. data/lib/e11y/adapters/in_memory_test.rb +29 -0
  175. data/lib/e11y/adapters/loki.rb +58 -23
  176. data/lib/e11y/adapters/null.rb +82 -0
  177. data/lib/e11y/adapters/opentelemetry_collector.rb +183 -0
  178. data/lib/e11y/adapters/otel_logs.rb +136 -23
  179. data/lib/e11y/adapters/sentry.rb +4 -7
  180. data/lib/e11y/adapters/stdout.rb +73 -7
  181. data/lib/e11y/adapters/yabeda.rb +153 -29
  182. data/lib/e11y/buffers/adaptive_buffer.rb +3 -17
  183. data/lib/e11y/buffers/{request_scoped_buffer.rb → ephemeral_buffer.rb} +72 -58
  184. data/lib/e11y/buffers/ring_buffer.rb +3 -16
  185. data/lib/e11y/configuration.rb +272 -0
  186. data/lib/e11y/console.rb +10 -17
  187. data/lib/e11y/current.rb +53 -1
  188. data/lib/e11y/debug/pipeline_inspector.rb +96 -0
  189. data/lib/e11y/documentation/generator.rb +48 -0
  190. data/lib/e11y/event/base.rb +176 -82
  191. data/lib/e11y/event/value_sampling_config.rb +1 -5
  192. data/lib/e11y/events/rails/database/query.rb +1 -4
  193. data/lib/e11y/events/rails/job/failed.rb +2 -0
  194. data/lib/e11y/instruments/active_job.rb +44 -12
  195. data/lib/e11y/instruments/rails_instrumentation.rb +49 -24
  196. data/lib/e11y/instruments/sidekiq.rb +135 -31
  197. data/lib/e11y/linters/base.rb +11 -0
  198. data/lib/e11y/linters/pii/pii_declaration_linter.rb +120 -0
  199. data/lib/e11y/linters/slo/config_consistency_linter.rb +76 -0
  200. data/lib/e11y/linters/slo/explicit_declaration_linter.rb +36 -0
  201. data/lib/e11y/linters/slo/slo_status_from_linter.rb +41 -0
  202. data/lib/e11y/logger/bridge.rb +26 -7
  203. data/lib/e11y/metrics/cardinality_protection.rb +10 -15
  204. data/lib/e11y/metrics/cardinality_tracker.rb +16 -6
  205. data/lib/e11y/metrics/registry.rb +3 -5
  206. data/lib/e11y/metrics/test_backend.rb +62 -0
  207. data/lib/e11y/metrics.rb +56 -10
  208. data/lib/e11y/middleware/adapter_resolver.rb +40 -0
  209. data/lib/e11y/middleware/audit_signing.rb +43 -6
  210. data/lib/e11y/middleware/baggage_protection.rb +75 -0
  211. data/lib/e11y/middleware/dev_log_source.rb +24 -0
  212. data/lib/e11y/middleware/event_slo.rb +23 -9
  213. data/lib/e11y/middleware/otel_span.rb +23 -0
  214. data/lib/e11y/middleware/pii_filter.rb +104 -75
  215. data/lib/e11y/middleware/rate_limiting.rb +54 -27
  216. data/lib/e11y/middleware/request.rb +70 -23
  217. data/lib/e11y/middleware/routing.rb +78 -21
  218. data/lib/e11y/middleware/sampling.rb +66 -17
  219. data/lib/e11y/middleware/self_monitoring_emit.rb +39 -0
  220. data/lib/e11y/middleware/trace_context.rb +45 -10
  221. data/lib/e11y/middleware/track_latency.rb +34 -0
  222. data/lib/e11y/middleware/validation.rb +7 -16
  223. data/lib/e11y/middleware/versioning.rb +26 -22
  224. data/lib/e11y/opentelemetry/semantic_conventions.rb +109 -0
  225. data/lib/e11y/opentelemetry/span_creator.rb +142 -0
  226. data/lib/e11y/pii/patterns.rb +12 -1
  227. data/lib/e11y/pipeline/builder.rb +4 -4
  228. data/lib/e11y/presets/audit_event.rb +13 -2
  229. data/lib/e11y/railtie.rb +52 -14
  230. data/lib/e11y/registry.rb +306 -0
  231. data/lib/e11y/reliability/circuit_breaker.rb +19 -21
  232. data/lib/e11y/reliability/dlq/base.rb +71 -0
  233. data/lib/e11y/reliability/dlq/file_adapter.rb +301 -0
  234. data/lib/e11y/reliability/dlq/file_storage.rb +63 -34
  235. data/lib/e11y/reliability/dlq/filter.rb +37 -54
  236. data/lib/e11y/reliability/retry_handler.rb +26 -29
  237. data/lib/e11y/reliability/retry_rate_limiter.rb +3 -11
  238. data/lib/e11y/sampling/error_spike_detector.rb +0 -2
  239. data/lib/e11y/sampling/load_monitor.rb +5 -9
  240. data/lib/e11y/sampling/stratified_tracker.rb +18 -0
  241. data/lib/e11y/self_monitoring/buffer_monitor.rb +2 -0
  242. data/lib/e11y/self_monitoring/performance_monitor.rb +19 -61
  243. data/lib/e11y/self_monitoring/reliability_monitor.rb +4 -74
  244. data/lib/e11y/slo/config_loader.rb +40 -0
  245. data/lib/e11y/slo/config_validator.rb +58 -0
  246. data/lib/e11y/slo/dashboard_generator.rb +122 -0
  247. data/lib/e11y/slo/event_driven.rb +8 -0
  248. data/lib/e11y/slo/tracker.rb +31 -4
  249. data/lib/e11y/testing/have_tracked_event_matcher.rb +190 -0
  250. data/lib/e11y/testing/rspec_matchers.rb +21 -0
  251. data/lib/e11y/testing/snapshot_matcher.rb +86 -0
  252. data/lib/e11y/trace_context/sampler.rb +35 -0
  253. data/lib/e11y/tracing/faraday_middleware.rb +31 -0
  254. data/lib/e11y/tracing/net_http_patch.rb +33 -0
  255. data/lib/e11y/tracing/propagator.rb +144 -0
  256. data/lib/e11y/tracing.rb +47 -0
  257. data/lib/e11y/version.rb +1 -1
  258. data/lib/e11y/versioning/version_extractor.rb +32 -0
  259. data/lib/e11y.rb +123 -266
  260. data/lib/generators/e11y/event/event_generator.rb +22 -0
  261. data/lib/generators/e11y/event/templates/event.rb.tt +16 -0
  262. data/lib/generators/e11y/grafana_dashboard/grafana_dashboard_generator.rb +30 -0
  263. data/lib/generators/e11y/grafana_dashboard/templates/e11y_dashboard.json +81 -0
  264. data/lib/generators/e11y/install/install_generator.rb +34 -0
  265. data/lib/generators/e11y/install/templates/e11y.rb +239 -0
  266. data/lib/generators/e11y/prometheus_alerts/prometheus_alerts_generator.rb +29 -0
  267. data/lib/generators/e11y/prometheus_alerts/templates/e11y_alerts.yml +28 -0
  268. data/lib/tasks/e11y_docs.rake +30 -0
  269. data/lib/tasks/e11y_events.rake +71 -0
  270. data/lib/tasks/e11y_lint.rake +91 -0
  271. data/lib/tasks/e11y_slo.rake +29 -0
  272. metadata +186 -39
  273. data/docs/ADR-003-slo-observability.md +0 -3337
  274. data/docs/ADR-010-developer-experience.md +0 -2166
  275. data/docs/API-REFERENCE-L28.md +0 -914
  276. data/docs/COMPREHENSIVE-CONFIGURATION.md +0 -2366
  277. data/docs/CONTRIBUTING.md +0 -312
  278. data/docs/IMPLEMENTATION_NOTES.md +0 -2804
  279. data/docs/IMPLEMENTATION_PLAN.md +0 -1971
  280. data/docs/IMPLEMENTATION_PLAN_ARCHITECTURE.md +0 -586
  281. data/docs/PLAN.md +0 -148
  282. data/docs/README.md +0 -296
  283. data/docs/design/00-memory-optimization.md +0 -593
  284. data/docs/guides/MIGRATION-L27-L28.md +0 -692
  285. data/docs/guides/PERFORMANCE-BENCHMARKS.md +0 -434
  286. data/docs/guides/README.md +0 -44
  287. data/docs/use_cases/UC-003-pattern-based-metrics.md +0 -1627
  288. data/lib/e11y/adapters/registry.rb +0 -141
@@ -0,0 +1,194 @@
1
+ [
2
+ {
3
+ "id": "1b50a45e-4f30-8644-793dbef82fbb",
4
+ "timestamp": "2026-03-20T09:47:32.279Z",
5
+ "event_name": "Events::LogInfo",
6
+ "severity": "info",
7
+ "version": 1,
8
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
9
+ "span_id": "f70c00eb6894a7b1",
10
+ "service_name": "dummy",
11
+ "environment": "development",
12
+ "retention_until": "2027-03-20T09:47:32Z",
13
+ "payload": {
14
+ "message": "Started POST \"/webhooks/stripe\" for ::1 at 2026-03-20 09:47:32 +0000",
15
+ "level": "info"
16
+ },
17
+ "metadata": {
18
+ "source": "web",
19
+ "started_at": "2026-03-20T09:47:32.265Z",
20
+ "request_id": "req-99e35f8e-15",
21
+ "path": "/webhooks/stripe",
22
+ "method": "POST",
23
+ "controller": "Admin::UsersController",
24
+ "action": "index",
25
+ "duration_ms": 1
26
+ }
27
+ },
28
+ {
29
+ "id": "2e654a4b-4fe2-a622-ca00bdba1527",
30
+ "timestamp": "2026-03-20T09:47:32.294Z",
31
+ "event_name": "Events::ApiRequest",
32
+ "severity": "info",
33
+ "version": 1,
34
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
35
+ "span_id": "b8ba8d3f9310ab95",
36
+ "service_name": "dummy",
37
+ "environment": "development",
38
+ "retention_until": "2027-03-20T09:47:32Z",
39
+ "payload": {
40
+ "endpoint": "POST /webhooks/stripe",
41
+ "status": "ok",
42
+ "http_status": 200
43
+ },
44
+ "metadata": {
45
+ "source": "web",
46
+ "started_at": "2026-03-20T09:47:32.265Z",
47
+ "request_id": "req-99e35f8e-15",
48
+ "path": "/webhooks/stripe",
49
+ "method": "POST",
50
+ "controller": "Admin::UsersController",
51
+ "action": "index",
52
+ "duration_ms": 60
53
+ }
54
+ },
55
+ {
56
+ "id": "339dd90d-473b-9b1d-e5222a0d52d4",
57
+ "timestamp": "2026-03-20T09:47:32.308Z",
58
+ "event_name": "Events::PostDebug",
59
+ "severity": "debug",
60
+ "version": 1,
61
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
62
+ "span_id": "2fb83ca4d087f804",
63
+ "service_name": "dummy",
64
+ "environment": "development",
65
+ "retention_until": "2027-03-20T09:47:32Z",
66
+ "payload": {
67
+ "message": " → Render — /webhooks/stripe id=23931"
68
+ },
69
+ "metadata": {
70
+ "source": "web",
71
+ "started_at": "2026-03-20T09:47:32.265Z",
72
+ "request_id": "req-99e35f8e-15",
73
+ "path": "/webhooks/stripe",
74
+ "method": "POST",
75
+ "controller": "Admin::UsersController",
76
+ "action": "index",
77
+ "duration_ms": 14
78
+ }
79
+ },
80
+ {
81
+ "id": "50f24258-46ec-8429-4858aed7e991",
82
+ "timestamp": "2026-03-20T09:47:32.324Z",
83
+ "event_name": "Events::PostDebug",
84
+ "severity": "debug",
85
+ "version": 1,
86
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
87
+ "span_id": "d10ca92773b2e747",
88
+ "service_name": "dummy",
89
+ "environment": "development",
90
+ "retention_until": "2027-03-20T09:47:32Z",
91
+ "payload": {
92
+ "message": " → Render — /webhooks/stripe id=29288"
93
+ },
94
+ "metadata": {
95
+ "source": "web",
96
+ "started_at": "2026-03-20T09:47:32.265Z",
97
+ "request_id": "req-99e35f8e-15",
98
+ "path": "/webhooks/stripe",
99
+ "method": "POST",
100
+ "controller": "Admin::UsersController",
101
+ "action": "index",
102
+ "duration_ms": 14
103
+ }
104
+ },
105
+ {
106
+ "id": "4d2719a5-4760-9416-923e80b2488e",
107
+ "timestamp": "2026-03-20T09:47:32.337Z",
108
+ "event_name": "Events::PostDebug",
109
+ "severity": "debug",
110
+ "version": 1,
111
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
112
+ "span_id": "93d623e03b724a98",
113
+ "service_name": "dummy",
114
+ "environment": "development",
115
+ "retention_until": "2027-03-20T09:47:32Z",
116
+ "payload": {
117
+ "message": " → CACHE — /webhooks/stripe id=34017"
118
+ },
119
+ "metadata": {
120
+ "source": "web",
121
+ "started_at": "2026-03-20T09:47:32.265Z",
122
+ "request_id": "req-99e35f8e-15",
123
+ "path": "/webhooks/stripe",
124
+ "method": "POST",
125
+ "controller": "Admin::UsersController",
126
+ "action": "index",
127
+ "duration_ms": 15
128
+ }
129
+ },
130
+ {
131
+ "id": "110df334-4e67-a867-0fb28af82c26",
132
+ "timestamp": "2026-03-20T09:47:32.350Z",
133
+ "event_name": "Events::OrderCreated",
134
+ "severity": "success",
135
+ "version": 1,
136
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
137
+ "span_id": "1ded07b26e78a0c5",
138
+ "service_name": "dummy",
139
+ "environment": "development",
140
+ "retention_until": "2027-03-20T09:47:32Z",
141
+ "payload": {
142
+ "order_id": "ord_944119",
143
+ "status": "completed",
144
+ "customer": {
145
+ "id": 1203,
146
+ "email": "[FILTERED]",
147
+ "locale": "en"
148
+ },
149
+ "items": [
150
+ {
151
+ "sku": "SKU-653",
152
+ "qty": 1,
153
+ "unit_price": 73.8
154
+ }
155
+ ]
156
+ },
157
+ "metadata": {
158
+ "source": "web",
159
+ "started_at": "2026-03-20T09:47:32.265Z",
160
+ "request_id": "req-99e35f8e-15",
161
+ "path": "/webhooks/stripe",
162
+ "method": "POST",
163
+ "controller": "Admin::UsersController",
164
+ "action": "index",
165
+ "duration_ms": 15
166
+ }
167
+ },
168
+ {
169
+ "id": "92ba039a-41a3-889c-2a25c09d7db8",
170
+ "timestamp": "2026-03-20T09:47:32.364Z",
171
+ "event_name": "Events::LogInfo",
172
+ "severity": "info",
173
+ "version": 1,
174
+ "trace_id": "99e35f8ef3baedd798cc4fd085980ad9",
175
+ "span_id": "4fcd0460b4fcabfc",
176
+ "service_name": "dummy",
177
+ "environment": "development",
178
+ "retention_until": "2027-03-20T09:47:32Z",
179
+ "payload": {
180
+ "message": "Completed POST /webhooks/stripe in 422ms",
181
+ "level": "info"
182
+ },
183
+ "metadata": {
184
+ "source": "web",
185
+ "started_at": "2026-03-20T09:47:32.265Z",
186
+ "request_id": "req-99e35f8e-15",
187
+ "path": "/webhooks/stripe",
188
+ "method": "POST",
189
+ "controller": "Admin::UsersController",
190
+ "action": "index",
191
+ "duration_ms": 190
192
+ }
193
+ }
194
+ ]
@@ -0,0 +1,131 @@
1
+ [
2
+ {
3
+ "id": "3cfd0b62-4368-971d-b51088c9af70",
4
+ "timestamp": "2026-03-20T09:47:21.875Z",
5
+ "event_name": "Events::LogInfo",
6
+ "severity": "info",
7
+ "version": 1,
8
+ "trace_id": "b4f3095c1909924cbc98889a86c83d6d",
9
+ "span_id": "8158383d7f1ea253",
10
+ "service_name": "dummy",
11
+ "environment": "development",
12
+ "retention_until": "2027-03-20T09:47:21Z",
13
+ "payload": {
14
+ "message": "Started GET \"/up\" for ::1 at 2026-03-20 09:47:21 +0000",
15
+ "level": "info"
16
+ },
17
+ "metadata": {
18
+ "source": "web",
19
+ "started_at": "2026-03-20T09:47:21.861Z",
20
+ "request_id": "req-b4f3095c-8",
21
+ "path": "/up",
22
+ "method": "GET",
23
+ "controller": "Webhooks::StripeController",
24
+ "action": "create",
25
+ "duration_ms": 3
26
+ }
27
+ },
28
+ {
29
+ "id": "63a2b8b0-4663-818b-f9bce7c71ce8",
30
+ "timestamp": "2026-03-20T09:47:21.887Z",
31
+ "event_name": "Events::ApiRequest",
32
+ "severity": "info",
33
+ "version": 1,
34
+ "trace_id": "b4f3095c1909924cbc98889a86c83d6d",
35
+ "span_id": "d8de5f92175e904b",
36
+ "service_name": "dummy",
37
+ "environment": "development",
38
+ "retention_until": "2027-03-20T09:47:21Z",
39
+ "payload": {
40
+ "endpoint": "GET /up",
41
+ "status": "ok",
42
+ "http_status": 200
43
+ },
44
+ "metadata": {
45
+ "source": "web",
46
+ "started_at": "2026-03-20T09:47:21.861Z",
47
+ "request_id": "req-b4f3095c-8",
48
+ "path": "/up",
49
+ "method": "GET",
50
+ "controller": "Webhooks::StripeController",
51
+ "action": "create",
52
+ "duration_ms": 9
53
+ }
54
+ },
55
+ {
56
+ "id": "f35afd7f-4e33-9352-edf9a1c0156e",
57
+ "timestamp": "2026-03-20T09:47:21.903Z",
58
+ "event_name": "Events::PostDebug",
59
+ "severity": "debug",
60
+ "version": 1,
61
+ "trace_id": "b4f3095c1909924cbc98889a86c83d6d",
62
+ "span_id": "1552208f72cd5374",
63
+ "service_name": "dummy",
64
+ "environment": "development",
65
+ "retention_until": "2027-03-20T09:47:21Z",
66
+ "payload": {
67
+ "message": " → CACHE /up — 35.8ms"
68
+ },
69
+ "metadata": {
70
+ "source": "web",
71
+ "started_at": "2026-03-20T09:47:21.861Z",
72
+ "request_id": "req-b4f3095c-8",
73
+ "path": "/up",
74
+ "method": "GET",
75
+ "controller": "Webhooks::StripeController",
76
+ "action": "create",
77
+ "duration_ms": 4
78
+ }
79
+ },
80
+ {
81
+ "id": "ca246a38-4426-886c-5eb47e925429",
82
+ "timestamp": "2026-03-20T09:47:21.917Z",
83
+ "event_name": "Events::PostDebug",
84
+ "severity": "debug",
85
+ "version": 1,
86
+ "trace_id": "b4f3095c1909924cbc98889a86c83d6d",
87
+ "span_id": "f96854b90ebc2ba6",
88
+ "service_name": "dummy",
89
+ "environment": "development",
90
+ "retention_until": "2027-03-20T09:47:21Z",
91
+ "payload": {
92
+ "message": " → SQL — /up id=37498"
93
+ },
94
+ "metadata": {
95
+ "source": "web",
96
+ "started_at": "2026-03-20T09:47:21.861Z",
97
+ "request_id": "req-b4f3095c-8",
98
+ "path": "/up",
99
+ "method": "GET",
100
+ "controller": "Webhooks::StripeController",
101
+ "action": "create",
102
+ "duration_ms": 5
103
+ }
104
+ },
105
+ {
106
+ "id": "249b1a4b-4c08-9c49-43b99e4389eb",
107
+ "timestamp": "2026-03-20T09:47:21.928Z",
108
+ "event_name": "Events::LogInfo",
109
+ "severity": "info",
110
+ "version": 1,
111
+ "trace_id": "b4f3095c1909924cbc98889a86c83d6d",
112
+ "span_id": "ddeb1b9019e32cfc",
113
+ "service_name": "dummy",
114
+ "environment": "development",
115
+ "retention_until": "2027-03-20T09:47:21Z",
116
+ "payload": {
117
+ "message": "Completed GET /up in 241ms",
118
+ "level": "info"
119
+ },
120
+ "metadata": {
121
+ "source": "web",
122
+ "started_at": "2026-03-20T09:47:21.861Z",
123
+ "request_id": "req-b4f3095c-8",
124
+ "path": "/up",
125
+ "method": "GET",
126
+ "controller": "Webhooks::StripeController",
127
+ "action": "create",
128
+ "duration_ms": 14
129
+ }
130
+ }
131
+ ]
@@ -0,0 +1,128 @@
1
+ [
2
+ {
3
+ "id": "a2eb589f-4dda-8353-4697ad5ae78b",
4
+ "timestamp": "2026-03-20T09:47:21.550Z",
5
+ "event_name": "Events::BackgroundJobStarted",
6
+ "severity": "info",
7
+ "version": 1,
8
+ "trace_id": "b54b7fc32b7575a7110de809d11ccda0",
9
+ "span_id": "635f1ab192c82ea2",
10
+ "service_name": "dummy",
11
+ "environment": "development",
12
+ "retention_until": "2027-03-20T09:47:21Z",
13
+ "payload": {
14
+ "job_class": "WebhookRetryJob",
15
+ "job_id": "af183fec-4203-ac0f-cb17f1a2e14f",
16
+ "queue": "default",
17
+ "arguments": [
18
+ {
19
+ "batch_id": 4001,
20
+ "shard": 12
21
+ }
22
+ ],
23
+ "enqueued_at": "2026-03-20T09:47:19.474Z"
24
+ },
25
+ "metadata": {
26
+ "source": "job",
27
+ "started_at": "2026-03-20T09:47:21.474Z",
28
+ "duration_ms": 897
29
+ }
30
+ },
31
+ {
32
+ "id": "62d79f42-4e40-95a0-b4a7e7c50f49",
33
+ "timestamp": "2026-03-20T09:47:22.090Z",
34
+ "event_name": "Events::ReportCreated",
35
+ "severity": "info",
36
+ "version": 1,
37
+ "trace_id": "b54b7fc32b7575a7110de809d11ccda0",
38
+ "span_id": "34e9442a321f5e2b",
39
+ "service_name": "dummy",
40
+ "environment": "development",
41
+ "retention_until": "2027-03-20T09:47:22Z",
42
+ "payload": {
43
+ "title": "Export batch 5380",
44
+ "description": "Queued by WebhookRetryJob",
45
+ "employee_ids": [
46
+ 141,
47
+ 12
48
+ ],
49
+ "author": "system"
50
+ },
51
+ "metadata": {
52
+ "source": "job",
53
+ "started_at": "2026-03-20T09:47:21.474Z",
54
+ "duration_ms": 269
55
+ }
56
+ },
57
+ {
58
+ "id": "9e45e2f1-4822-aad6-6b50a7dd7059",
59
+ "timestamp": "2026-03-20T09:47:22.583Z",
60
+ "event_name": "Events::ReportExportCompleted",
61
+ "severity": "fatal",
62
+ "version": 1,
63
+ "trace_id": "b54b7fc32b7575a7110de809d11ccda0",
64
+ "span_id": "51a8630762a95a5c",
65
+ "service_name": "dummy",
66
+ "environment": "development",
67
+ "retention_until": "2027-03-20T09:47:22Z",
68
+ "payload": {
69
+ "report_id": 1417,
70
+ "error": "No space left on device",
71
+ "exception": {
72
+ "class": "Errno::ENOSPC",
73
+ "message": "No space left on device @ rb_sysopen",
74
+ "backtrace": [
75
+ "app/jobs/webhook_retry_job.rb:26:in `perform'",
76
+ "activejob (7.0.0) lib/active_job/execution.rb:68:in `perform_now'"
77
+ ]
78
+ }
79
+ },
80
+ "metadata": {
81
+ "source": "job",
82
+ "started_at": "2026-03-20T09:47:21.474Z",
83
+ "duration_ms": 899
84
+ }
85
+ },
86
+ {
87
+ "id": "bc2d9da3-42d8-ad14-5bfc48336af8",
88
+ "timestamp": "2026-03-20T09:47:23.152Z",
89
+ "event_name": "Events::LogInfo",
90
+ "severity": "info",
91
+ "version": 1,
92
+ "trace_id": "b54b7fc32b7575a7110de809d11ccda0",
93
+ "span_id": "8c47e2a32d6cad89",
94
+ "service_name": "dummy",
95
+ "environment": "development",
96
+ "retention_until": "2027-03-20T09:47:23Z",
97
+ "payload": {
98
+ "message": "Job WebhookRetryJob done shard=4",
99
+ "level": "info"
100
+ },
101
+ "metadata": {
102
+ "source": "job",
103
+ "started_at": "2026-03-20T09:47:21.474Z",
104
+ "duration_ms": 76
105
+ }
106
+ },
107
+ {
108
+ "id": "bb4d4462-4fef-8623-93d1400113f0",
109
+ "timestamp": "2026-03-20T09:47:23.735Z",
110
+ "event_name": "Events::LogInfo",
111
+ "severity": "info",
112
+ "version": 1,
113
+ "trace_id": "b54b7fc32b7575a7110de809d11ccda0",
114
+ "span_id": "c6c2ffe6949e885f",
115
+ "service_name": "dummy",
116
+ "environment": "development",
117
+ "retention_until": "2027-03-20T09:47:23Z",
118
+ "payload": {
119
+ "message": "Job WebhookRetryJob done shard=7",
120
+ "level": "info"
121
+ },
122
+ "metadata": {
123
+ "source": "job",
124
+ "started_at": "2026-03-20T09:47:21.474Z",
125
+ "duration_ms": 193
126
+ }
127
+ }
128
+ ]
@@ -0,0 +1,169 @@
1
+ [
2
+ {
3
+ "id": "898a186e-42f8-9e96-9cb695b33b4f",
4
+ "timestamp": "2026-03-20T09:47:27.469Z",
5
+ "event_name": "Events::LogInfo",
6
+ "severity": "info",
7
+ "version": 1,
8
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
9
+ "span_id": "26e3185e667a17ca",
10
+ "service_name": "dummy",
11
+ "environment": "development",
12
+ "retention_until": "2027-03-20T09:47:27Z",
13
+ "payload": {
14
+ "message": "Started GET \"/sidekiq/busy\" for ::1 at 2026-03-20 09:47:27 +0000",
15
+ "level": "info"
16
+ },
17
+ "metadata": {
18
+ "source": "web",
19
+ "started_at": "2026-03-20T09:47:27.457Z",
20
+ "request_id": "req-c0b48033-12",
21
+ "path": "/sidekiq/busy",
22
+ "method": "GET",
23
+ "controller": "OrdersController",
24
+ "action": "create",
25
+ "duration_ms": 1
26
+ }
27
+ },
28
+ {
29
+ "id": "587da6b7-4118-acc2-475f79558cc2",
30
+ "timestamp": "2026-03-20T09:47:27.482Z",
31
+ "event_name": "Events::ApiRequest",
32
+ "severity": "info",
33
+ "version": 1,
34
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
35
+ "span_id": "83e6829aaae5105c",
36
+ "service_name": "dummy",
37
+ "environment": "development",
38
+ "retention_until": "2027-03-20T09:47:27Z",
39
+ "payload": {
40
+ "endpoint": "GET /sidekiq/busy",
41
+ "status": "ok",
42
+ "http_status": 200
43
+ },
44
+ "metadata": {
45
+ "source": "web",
46
+ "started_at": "2026-03-20T09:47:27.457Z",
47
+ "request_id": "req-c0b48033-12",
48
+ "path": "/sidekiq/busy",
49
+ "method": "GET",
50
+ "controller": "OrdersController",
51
+ "action": "create",
52
+ "duration_ms": 32
53
+ }
54
+ },
55
+ {
56
+ "id": "0a765d92-4f14-9404-55d14751b25b",
57
+ "timestamp": "2026-03-20T09:47:27.494Z",
58
+ "event_name": "Events::PostDebug",
59
+ "severity": "debug",
60
+ "version": 1,
61
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
62
+ "span_id": "58fd92a92d508f3c",
63
+ "service_name": "dummy",
64
+ "environment": "development",
65
+ "retention_until": "2027-03-20T09:47:27Z",
66
+ "payload": {
67
+ "message": " → SQL /sidekiq/busy — 21.5ms"
68
+ },
69
+ "metadata": {
70
+ "source": "web",
71
+ "started_at": "2026-03-20T09:47:27.457Z",
72
+ "request_id": "req-c0b48033-12",
73
+ "path": "/sidekiq/busy",
74
+ "method": "GET",
75
+ "controller": "OrdersController",
76
+ "action": "create",
77
+ "duration_ms": 1
78
+ }
79
+ },
80
+ {
81
+ "id": "e4eb6c2d-4a21-a7b0-f44d548166d9",
82
+ "timestamp": "2026-03-20T09:47:27.506Z",
83
+ "event_name": "Events::PostDebug",
84
+ "severity": "debug",
85
+ "version": 1,
86
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
87
+ "span_id": "b0bdb72c52533f9a",
88
+ "service_name": "dummy",
89
+ "environment": "development",
90
+ "retention_until": "2027-03-20T09:47:27Z",
91
+ "payload": {
92
+ "message": " → CACHE — /sidekiq/busy id=23322"
93
+ },
94
+ "metadata": {
95
+ "source": "web",
96
+ "started_at": "2026-03-20T09:47:27.457Z",
97
+ "request_id": "req-c0b48033-12",
98
+ "path": "/sidekiq/busy",
99
+ "method": "GET",
100
+ "controller": "OrdersController",
101
+ "action": "create",
102
+ "duration_ms": 25
103
+ }
104
+ },
105
+ {
106
+ "id": "14ee28ee-4437-995d-b06aebbe4b3e",
107
+ "timestamp": "2026-03-20T09:47:27.519Z",
108
+ "event_name": "Events::OrderCreated",
109
+ "severity": "success",
110
+ "version": 1,
111
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
112
+ "span_id": "6095d02f0fa2ce3c",
113
+ "service_name": "dummy",
114
+ "environment": "development",
115
+ "retention_until": "2027-03-20T09:47:27Z",
116
+ "payload": {
117
+ "order_id": "ord_667376",
118
+ "status": "pending_payment",
119
+ "customer": {
120
+ "id": 1277,
121
+ "email": "[FILTERED]",
122
+ "locale": "en"
123
+ },
124
+ "items": [
125
+ {
126
+ "sku": "SKU-791",
127
+ "qty": 1,
128
+ "unit_price": 108.42
129
+ }
130
+ ]
131
+ },
132
+ "metadata": {
133
+ "source": "web",
134
+ "started_at": "2026-03-20T09:47:27.457Z",
135
+ "request_id": "req-c0b48033-12",
136
+ "path": "/sidekiq/busy",
137
+ "method": "GET",
138
+ "controller": "OrdersController",
139
+ "action": "create",
140
+ "duration_ms": 56
141
+ }
142
+ },
143
+ {
144
+ "id": "7668acd7-4d81-816c-9a990964120d",
145
+ "timestamp": "2026-03-20T09:47:27.532Z",
146
+ "event_name": "Events::LogInfo",
147
+ "severity": "info",
148
+ "version": 1,
149
+ "trace_id": "c0b48033fa06746bcc5886745e053cff",
150
+ "span_id": "25d23aba409fb537",
151
+ "service_name": "dummy",
152
+ "environment": "development",
153
+ "retention_until": "2027-03-20T09:47:27Z",
154
+ "payload": {
155
+ "message": "Completed GET /sidekiq/busy in 18ms",
156
+ "level": "info"
157
+ },
158
+ "metadata": {
159
+ "source": "web",
160
+ "started_at": "2026-03-20T09:47:27.457Z",
161
+ "request_id": "req-c0b48033-12",
162
+ "path": "/sidekiq/busy",
163
+ "method": "GET",
164
+ "controller": "OrdersController",
165
+ "action": "create",
166
+ "duration_ms": 10
167
+ }
168
+ }
169
+ ]