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.
- checksums.yaml +4 -4
- data/.rubocop.yml +130 -10
- data/CHANGELOG.md +80 -1
- data/CLAUDE.md +168 -0
- data/CONTRIBUTING.md +640 -0
- data/README.md +165 -701
- data/RELEASE.md +41 -12
- data/Rakefile +249 -57
- data/config/README.md +1 -1
- data/config/loki-local-config.yaml +12 -0
- data/config/otel-collector-config.yaml +44 -0
- data/cucumber.yml +1 -0
- data/docker-compose.yml +18 -2
- data/docs/ADAPTERS.md +76 -0
- data/docs/ADAPTIVE_SAMPLING.md +59 -0
- data/docs/COMPARISON.md +104 -0
- data/docs/CONFIGURATION.md +52 -0
- data/docs/DISTRIBUTED_TRACING.md +44 -0
- data/docs/LIMITATIONS.md +13 -0
- data/docs/METRICS_DSL.md +84 -0
- data/docs/PERFORMANCE.md +60 -0
- data/docs/PII_FILTERING.md +40 -0
- data/docs/PRESETS.md +65 -0
- data/docs/QUICK-START.md +546 -587
- data/docs/RAILS_INTEGRATION.md +79 -0
- data/docs/SCHEMA_VALIDATION.md +63 -0
- data/docs/SLO-PROMQL-ALERTS.md +161 -0
- data/docs/TESTING.md +69 -0
- data/docs/{ADR-001-architecture.md → architecture/ADR-001-architecture.md} +36 -65
- data/docs/{ADR-002-metrics-yabeda.md → architecture/ADR-002-metrics-yabeda.md} +62 -236
- data/docs/architecture/ADR-003-slo-observability.md +1402 -0
- data/docs/{ADR-004-adapter-architecture.md → architecture/ADR-004-adapter-architecture.md} +163 -146
- data/docs/{ADR-005-tracing-context.md → architecture/ADR-005-tracing-context.md} +10 -9
- data/docs/{ADR-006-security-compliance.md → architecture/ADR-006-security-compliance.md} +184 -191
- data/docs/{ADR-007-opentelemetry-integration.md → architecture/ADR-007-opentelemetry-integration.md} +3 -21
- data/docs/{ADR-008-rails-integration.md → architecture/ADR-008-rails-integration.md} +182 -743
- data/docs/{ADR-009-cost-optimization.md → architecture/ADR-009-cost-optimization.md} +45 -54
- data/docs/architecture/ADR-010-developer-experience.md +522 -0
- data/docs/{ADR-011-testing-strategy.md → architecture/ADR-011-testing-strategy.md} +44 -86
- data/docs/{ADR-012-event-evolution.md → architecture/ADR-012-event-evolution.md} +11 -11
- data/docs/{ADR-013-reliability-error-handling.md → architecture/ADR-013-reliability-error-handling.md} +37 -12
- data/docs/{ADR-014-event-driven-slo.md → architecture/ADR-014-event-driven-slo.md} +12 -24
- data/docs/{ADR-015-middleware-order.md → architecture/ADR-015-middleware-order.md} +43 -59
- data/docs/{ADR-016-self-monitoring-slo.md → architecture/ADR-016-self-monitoring-slo.md} +58 -355
- data/docs/{ADR-017-multi-rails-compatibility.md → architecture/ADR-017-multi-rails-compatibility.md} +4 -11
- data/docs/architecture/ADR-018-memory-optimization.md +366 -0
- data/docs/{ADR-INDEX.md → architecture/ADR-INDEX.md} +11 -6
- data/docs/plans/2026-03-20-browser-overlay-svelte.md +281 -0
- data/docs/{00-ICP-AND-TIMELINE.md → prd/00-ICP-AND-TIMELINE.md} +6 -6
- data/docs/{01-SCALE-REQUIREMENTS.md → prd/01-SCALE-REQUIREMENTS.md} +6 -6
- data/docs/prd/01-overview-vision.md +19 -14
- data/docs/use_cases/README.md +22 -23
- data/docs/use_cases/UC-001-request-scoped-debug-buffering.md +50 -44
- data/docs/use_cases/UC-002-business-event-tracking.md +26 -95
- data/docs/use_cases/UC-003-event-metrics.md +66 -0
- data/docs/use_cases/UC-004-zero-config-slo-tracking.md +33 -684
- data/docs/use_cases/UC-005-sentry-integration.md +13 -15
- data/docs/use_cases/UC-006-trace-context-management.md +30 -28
- data/docs/use_cases/UC-007-pii-filtering.md +35 -87
- data/docs/use_cases/UC-008-opentelemetry-integration.md +51 -89
- data/docs/use_cases/UC-009-multi-service-tracing.md +30 -178
- data/docs/use_cases/UC-010-background-job-tracking.md +24 -91
- data/docs/use_cases/UC-011-rate-limiting.md +95 -168
- data/docs/use_cases/UC-012-audit-trail.md +21 -46
- data/docs/use_cases/UC-013-high-cardinality-protection.md +29 -167
- data/docs/use_cases/UC-014-adaptive-sampling.md +2 -2
- data/docs/use_cases/UC-015-cost-optimization.md +46 -99
- data/docs/use_cases/UC-016-rails-logger-migration.md +39 -213
- data/docs/use_cases/UC-017-local-development.md +203 -777
- data/docs/use_cases/UC-018-testing-events.md +3 -3
- data/docs/use_cases/UC-019-retention-based-routing.md +53 -106
- data/docs/use_cases/UC-020-event-versioning.md +8 -9
- data/docs/use_cases/UC-021-error-handling-retry-dlq.md +18 -22
- data/docs/use_cases/UC-022-event-registry.md +15 -21
- data/docs/use_cases/backlog.md +119 -87
- data/e11y.gemspec +2 -2
- data/gems/e11y-devtools/README.md +158 -0
- data/gems/e11y-devtools/config/routes.rb +15 -0
- data/gems/e11y-devtools/e11y-devtools.gemspec +25 -0
- data/gems/e11y-devtools/exe/e11y +34 -0
- data/gems/e11y-devtools/frontend/.gitignore +24 -0
- data/gems/e11y-devtools/frontend/README.md +51 -0
- data/gems/e11y-devtools/frontend/index.html +14 -0
- data/gems/e11y-devtools/frontend/package-lock.json +3707 -0
- data/gems/e11y-devtools/frontend/package.json +28 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/events/recent.json +4205 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/interactions.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/0a2e04027cfa22d014bc22e8b27cd913/events.json +86 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/0e1543af6a630fb3af6b52283154b3e0/events.json +169 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/1838b691faa49564f97db8592ff3978d/events.json +78 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/29f198f6588dacffb687777eb5f8f118/events.json +197 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/34bc3c9c0097de28a7a6f99b90a8e7bc/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/3ba6c20d068ab9cee00e51b180e66444/events.json +184 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/435bfd8f17b9009146a79812d7c3726d/events.json +144 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/4c7676e3fe668e99edb2b94d7d5678a9/events.json +222 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/6daf0d47974bedfc55d5de7004a3ea9f/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8a81ada42834d15f287bb40010043605/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8c0a98900edaae105469df8daedccf02/events.json +198 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/8e4f645180f8a7d1dce426b07380466b/events.json +222 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/93db346fa5d44a032605a13b627f4b80/events.json +128 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/98ff6146faf7bd9be8bd03a8275817ba/events.json +223 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/9997ddd0247bc7e25f2ca7a5c415c93d/events.json +197 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/99e35f8ef3baedd798cc4fd085980ad9/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/b4f3095c1909924cbc98889a86c83d6d/events.json +131 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/b54b7fc32b7575a7110de809d11ccda0/events.json +128 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/c0b48033fa06746bcc5886745e053cff/events.json +169 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/c44649ac76701b4558927cd2305ab535/events.json +169 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/d601ae3320057580a39dbdac2edfdf4a/events.json +248 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/e67e724bab422d2b52eeb49635e512e1/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/e6c72765a28f158a8485b35fa63f73da/events.json +194 -0
- data/gems/e11y-devtools/frontend/public/mocks/v1/traces/f541b87405c9a54819b18ebe529f6419/events.json +194 -0
- data/gems/e11y-devtools/frontend/scripts/generate_mocks.rb +397 -0
- data/gems/e11y-devtools/frontend/src/App.svelte +827 -0
- data/gems/e11y-devtools/frontend/src/components/Fab.svelte +19 -0
- data/gems/e11y-devtools/frontend/src/components/FilterBar.svelte +38 -0
- data/gems/e11y-devtools/frontend/src/components/FullscreenPanel.svelte +82 -0
- data/gems/e11y-devtools/frontend/src/components/InteractionsTimeline.svelte +264 -0
- data/gems/e11y-devtools/frontend/src/components/RecentHistogram.svelte +354 -0
- data/gems/e11y-devtools/frontend/src/lib/api.ts +37 -0
- data/gems/e11y-devtools/frontend/src/lib/eventIdentity.ts +12 -0
- data/gems/e11y-devtools/frontend/src/lib/format.ts +37 -0
- data/gems/e11y-devtools/frontend/src/lib/listFilter.ts +43 -0
- data/gems/e11y-devtools/frontend/src/lib/recentVolume.ts +80 -0
- data/gems/e11y-devtools/frontend/src/lib/router.ts +12 -0
- data/gems/e11y-devtools/frontend/src/lib/transitions.ts +34 -0
- data/gems/e11y-devtools/frontend/src/lib/viewportOrigin.ts +25 -0
- data/gems/e11y-devtools/frontend/src/main.ts +8 -0
- data/gems/e11y-devtools/frontend/src/overlay-entry.ts +24 -0
- data/gems/e11y-devtools/frontend/src/overlay.css +1080 -0
- data/gems/e11y-devtools/frontend/svelte.config.js +2 -0
- data/gems/e11y-devtools/frontend/test_puppeteer.js +41 -0
- data/gems/e11y-devtools/frontend/test_scale.js +3 -0
- data/gems/e11y-devtools/frontend/tsconfig.app.json +21 -0
- data/gems/e11y-devtools/frontend/tsconfig.json +7 -0
- data/gems/e11y-devtools/frontend/tsconfig.node.json +26 -0
- data/gems/e11y-devtools/frontend/vite.config.ts +36 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/server.rb +96 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tool_base.rb +25 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/clear.rb +31 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/errors.rb +35 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/event_detail.rb +33 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/events_by_trace.rb +33 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/interactions.rb +40 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/recent_events.rb +34 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/search.rb +34 -0
- data/gems/e11y-devtools/lib/e11y/devtools/mcp/tools/stats.rb +30 -0
- data/gems/e11y-devtools/lib/e11y/devtools/overlay/assets/overlay.js +20 -0
- data/gems/e11y-devtools/lib/e11y/devtools/overlay/controller.rb +94 -0
- data/gems/e11y-devtools/lib/e11y/devtools/overlay/engine.rb +26 -0
- data/gems/e11y-devtools/lib/e11y/devtools/overlay/middleware.rb +80 -0
- data/gems/e11y-devtools/lib/e11y/devtools/overlay/rails_controller.rb +67 -0
- data/gems/e11y-devtools/lib/e11y/devtools/tui/app.rb +262 -0
- data/gems/e11y-devtools/lib/e11y/devtools/tui/grouping.rb +66 -0
- data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/event_detail.rb +62 -0
- data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/event_list.rb +70 -0
- data/gems/e11y-devtools/lib/e11y/devtools/tui/widgets/interaction_list.rb +47 -0
- data/gems/e11y-devtools/lib/e11y/devtools/version.rb +8 -0
- data/gems/e11y-devtools/lib/e11y/devtools.rb +13 -0
- data/gems/e11y-devtools/spec/e11y/devtools/mcp/tools_spec.rb +107 -0
- data/gems/e11y-devtools/spec/e11y/devtools/overlay/controller_spec.rb +91 -0
- data/gems/e11y-devtools/spec/e11y/devtools/overlay/middleware_spec.rb +46 -0
- data/gems/e11y-devtools/spec/e11y/devtools/tui/app_spec.rb +85 -0
- data/gems/e11y-devtools/spec/e11y/devtools/tui/grouping_spec.rb +64 -0
- data/gems/e11y-devtools/spec/spec_helper.rb +5 -0
- data/gems/e11y-devtools/spec/tui/widgets/event_list_spec.rb +44 -0
- data/gems/e11y-devtools/spec/tui/widgets/interaction_list_spec.rb +62 -0
- data/lib/e11y/adapters/audit_encrypted.rb +53 -11
- data/lib/e11y/adapters/base.rb +33 -34
- data/lib/e11y/adapters/dev_log/file_store.rb +143 -0
- data/lib/e11y/adapters/dev_log/query.rb +219 -0
- data/lib/e11y/adapters/dev_log.rb +118 -0
- data/lib/e11y/adapters/file.rb +3 -6
- data/lib/e11y/adapters/in_memory.rb +52 -5
- data/lib/e11y/adapters/in_memory_test.rb +29 -0
- data/lib/e11y/adapters/loki.rb +58 -23
- data/lib/e11y/adapters/null.rb +82 -0
- data/lib/e11y/adapters/opentelemetry_collector.rb +183 -0
- data/lib/e11y/adapters/otel_logs.rb +136 -23
- data/lib/e11y/adapters/sentry.rb +4 -7
- data/lib/e11y/adapters/stdout.rb +73 -7
- data/lib/e11y/adapters/yabeda.rb +153 -29
- data/lib/e11y/buffers/adaptive_buffer.rb +3 -17
- data/lib/e11y/buffers/{request_scoped_buffer.rb → ephemeral_buffer.rb} +72 -58
- data/lib/e11y/buffers/ring_buffer.rb +3 -16
- data/lib/e11y/configuration.rb +272 -0
- data/lib/e11y/console.rb +10 -17
- data/lib/e11y/current.rb +53 -1
- data/lib/e11y/debug/pipeline_inspector.rb +96 -0
- data/lib/e11y/documentation/generator.rb +48 -0
- data/lib/e11y/event/base.rb +176 -82
- data/lib/e11y/event/value_sampling_config.rb +1 -5
- data/lib/e11y/events/rails/database/query.rb +1 -4
- data/lib/e11y/events/rails/job/failed.rb +2 -0
- data/lib/e11y/instruments/active_job.rb +44 -12
- data/lib/e11y/instruments/rails_instrumentation.rb +49 -24
- data/lib/e11y/instruments/sidekiq.rb +135 -31
- data/lib/e11y/linters/base.rb +11 -0
- data/lib/e11y/linters/pii/pii_declaration_linter.rb +120 -0
- data/lib/e11y/linters/slo/config_consistency_linter.rb +76 -0
- data/lib/e11y/linters/slo/explicit_declaration_linter.rb +36 -0
- data/lib/e11y/linters/slo/slo_status_from_linter.rb +41 -0
- data/lib/e11y/logger/bridge.rb +26 -7
- data/lib/e11y/metrics/cardinality_protection.rb +10 -15
- data/lib/e11y/metrics/cardinality_tracker.rb +16 -6
- data/lib/e11y/metrics/registry.rb +3 -5
- data/lib/e11y/metrics/test_backend.rb +62 -0
- data/lib/e11y/metrics.rb +56 -10
- data/lib/e11y/middleware/adapter_resolver.rb +40 -0
- data/lib/e11y/middleware/audit_signing.rb +43 -6
- data/lib/e11y/middleware/baggage_protection.rb +75 -0
- data/lib/e11y/middleware/dev_log_source.rb +24 -0
- data/lib/e11y/middleware/event_slo.rb +23 -9
- data/lib/e11y/middleware/otel_span.rb +23 -0
- data/lib/e11y/middleware/pii_filter.rb +104 -75
- data/lib/e11y/middleware/rate_limiting.rb +54 -27
- data/lib/e11y/middleware/request.rb +70 -23
- data/lib/e11y/middleware/routing.rb +78 -21
- data/lib/e11y/middleware/sampling.rb +66 -17
- data/lib/e11y/middleware/self_monitoring_emit.rb +39 -0
- data/lib/e11y/middleware/trace_context.rb +45 -10
- data/lib/e11y/middleware/track_latency.rb +34 -0
- data/lib/e11y/middleware/validation.rb +7 -16
- data/lib/e11y/middleware/versioning.rb +26 -22
- data/lib/e11y/opentelemetry/semantic_conventions.rb +109 -0
- data/lib/e11y/opentelemetry/span_creator.rb +142 -0
- data/lib/e11y/pii/patterns.rb +12 -1
- data/lib/e11y/pipeline/builder.rb +4 -4
- data/lib/e11y/presets/audit_event.rb +13 -2
- data/lib/e11y/railtie.rb +52 -14
- data/lib/e11y/registry.rb +306 -0
- data/lib/e11y/reliability/circuit_breaker.rb +19 -21
- data/lib/e11y/reliability/dlq/base.rb +71 -0
- data/lib/e11y/reliability/dlq/file_adapter.rb +301 -0
- data/lib/e11y/reliability/dlq/file_storage.rb +63 -34
- data/lib/e11y/reliability/dlq/filter.rb +37 -54
- data/lib/e11y/reliability/retry_handler.rb +26 -29
- data/lib/e11y/reliability/retry_rate_limiter.rb +3 -11
- data/lib/e11y/sampling/error_spike_detector.rb +0 -2
- data/lib/e11y/sampling/load_monitor.rb +5 -9
- data/lib/e11y/sampling/stratified_tracker.rb +18 -0
- data/lib/e11y/self_monitoring/buffer_monitor.rb +2 -0
- data/lib/e11y/self_monitoring/performance_monitor.rb +19 -61
- data/lib/e11y/self_monitoring/reliability_monitor.rb +4 -74
- data/lib/e11y/slo/config_loader.rb +40 -0
- data/lib/e11y/slo/config_validator.rb +58 -0
- data/lib/e11y/slo/dashboard_generator.rb +122 -0
- data/lib/e11y/slo/event_driven.rb +8 -0
- data/lib/e11y/slo/tracker.rb +31 -4
- data/lib/e11y/testing/have_tracked_event_matcher.rb +190 -0
- data/lib/e11y/testing/rspec_matchers.rb +21 -0
- data/lib/e11y/testing/snapshot_matcher.rb +86 -0
- data/lib/e11y/trace_context/sampler.rb +35 -0
- data/lib/e11y/tracing/faraday_middleware.rb +31 -0
- data/lib/e11y/tracing/net_http_patch.rb +33 -0
- data/lib/e11y/tracing/propagator.rb +144 -0
- data/lib/e11y/tracing.rb +47 -0
- data/lib/e11y/version.rb +1 -1
- data/lib/e11y/versioning/version_extractor.rb +32 -0
- data/lib/e11y.rb +123 -266
- data/lib/generators/e11y/event/event_generator.rb +22 -0
- data/lib/generators/e11y/event/templates/event.rb.tt +16 -0
- data/lib/generators/e11y/grafana_dashboard/grafana_dashboard_generator.rb +30 -0
- data/lib/generators/e11y/grafana_dashboard/templates/e11y_dashboard.json +81 -0
- data/lib/generators/e11y/install/install_generator.rb +34 -0
- data/lib/generators/e11y/install/templates/e11y.rb +239 -0
- data/lib/generators/e11y/prometheus_alerts/prometheus_alerts_generator.rb +29 -0
- data/lib/generators/e11y/prometheus_alerts/templates/e11y_alerts.yml +28 -0
- data/lib/tasks/e11y_docs.rake +30 -0
- data/lib/tasks/e11y_events.rake +71 -0
- data/lib/tasks/e11y_lint.rake +91 -0
- data/lib/tasks/e11y_slo.rake +29 -0
- metadata +186 -39
- data/docs/ADR-003-slo-observability.md +0 -3337
- data/docs/ADR-010-developer-experience.md +0 -2166
- data/docs/API-REFERENCE-L28.md +0 -914
- data/docs/COMPREHENSIVE-CONFIGURATION.md +0 -2366
- data/docs/CONTRIBUTING.md +0 -312
- data/docs/IMPLEMENTATION_NOTES.md +0 -2804
- data/docs/IMPLEMENTATION_PLAN.md +0 -1971
- data/docs/IMPLEMENTATION_PLAN_ARCHITECTURE.md +0 -586
- data/docs/PLAN.md +0 -148
- data/docs/README.md +0 -296
- data/docs/design/00-memory-optimization.md +0 -593
- data/docs/guides/MIGRATION-L27-L28.md +0 -692
- data/docs/guides/PERFORMANCE-BENCHMARKS.md +0 -434
- data/docs/guides/README.md +0 -44
- data/docs/use_cases/UC-003-pattern-based-metrics.md +0 -1627
- data/lib/e11y/adapters/registry.rb +0 -141
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
type Props = {
|
|
3
|
+
label: string
|
|
4
|
+
onclick: (e: MouseEvent) => void
|
|
5
|
+
stateClass: "" | "e11y-fab--state-warn" | "e11y-fab--state-err"
|
|
6
|
+
pulseClass: "" | "e11y-fab--pulse-warn" | "e11y-fab--pulse-error"
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
let { label, onclick, stateClass, pulseClass }: Props = $props()
|
|
10
|
+
</script>
|
|
11
|
+
|
|
12
|
+
<button
|
|
13
|
+
type="button"
|
|
14
|
+
class="e11y-fab {stateClass} {pulseClass}"
|
|
15
|
+
onclick={onclick}
|
|
16
|
+
aria-label="Open e11y devtools"
|
|
17
|
+
>
|
|
18
|
+
{label}
|
|
19
|
+
</button>
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { ListSeverityFilter } from "../lib/listFilter"
|
|
3
|
+
|
|
4
|
+
let {
|
|
5
|
+
search = $bindable(""),
|
|
6
|
+
severity = $bindable<ListSeverityFilter>("all"),
|
|
7
|
+
placeholder = "Search...",
|
|
8
|
+
class: className = "",
|
|
9
|
+
} = $props()
|
|
10
|
+
|
|
11
|
+
const severities: { id: ListSeverityFilter; label: string; dotClass: string }[] = [
|
|
12
|
+
{ id: "all", label: "All", dotClass: "e11y-chip-dot--all" },
|
|
13
|
+
{ id: "error", label: "Error", dotClass: "e11y-chip-dot--err" },
|
|
14
|
+
{ id: "warn", label: "Warn", dotClass: "e11y-chip-dot--warn" },
|
|
15
|
+
{ id: "rest", label: "Other", dotClass: "e11y-chip-dot--rest" },
|
|
16
|
+
]
|
|
17
|
+
</script>
|
|
18
|
+
|
|
19
|
+
<div class="e11y-list-filters {className}">
|
|
20
|
+
<input
|
|
21
|
+
type="search"
|
|
22
|
+
class="e11y-search"
|
|
23
|
+
{placeholder}
|
|
24
|
+
bind:value={search}
|
|
25
|
+
aria-label="Search"
|
|
26
|
+
/>
|
|
27
|
+
{#each severities as s (s.id)}
|
|
28
|
+
<button
|
|
29
|
+
type="button"
|
|
30
|
+
class="e11y-chip e11y-chip--{s.id}"
|
|
31
|
+
class:e11y-chip--active={severity === s.id}
|
|
32
|
+
onclick={() => (severity = s.id)}
|
|
33
|
+
>
|
|
34
|
+
<i class="e11y-chip-dot {s.dotClass}"></i>
|
|
35
|
+
{s.label}
|
|
36
|
+
</button>
|
|
37
|
+
{/each}
|
|
38
|
+
</div>
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
<script lang="ts">
|
|
2
|
+
import type { Snippet } from "svelte"
|
|
3
|
+
import { circleCollapse, circleExpand, type CircleOrigin } from "../lib/transitions"
|
|
4
|
+
import { originFallbackFabCorner } from "../lib/viewportOrigin"
|
|
5
|
+
|
|
6
|
+
type Props = {
|
|
7
|
+
open: boolean
|
|
8
|
+
onclose: () => void
|
|
9
|
+
/** Circle reveal origin (FAB center + radius); falls back if null. */
|
|
10
|
+
origin: CircleOrigin | null
|
|
11
|
+
headerTopLeft?: Snippet
|
|
12
|
+
headerTopRight?: Snippet
|
|
13
|
+
headerBottom?: Snippet
|
|
14
|
+
children: Snippet
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
let { open, onclose, headerTopLeft, headerTopRight, headerBottom, children, origin }: Props = $props()
|
|
18
|
+
|
|
19
|
+
function motionOk(): boolean {
|
|
20
|
+
return typeof matchMedia === "undefined" || !matchMedia("(prefers-reduced-motion: reduce)").matches
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
const o = $derived(origin ?? originFallbackFabCorner())
|
|
24
|
+
const openMs = $derived(motionOk() ? 440 : 0)
|
|
25
|
+
const closeMs = $derived(motionOk() ? 360 : 0)
|
|
26
|
+
|
|
27
|
+
function handleKeydown(e: KeyboardEvent): void {
|
|
28
|
+
if (e.key === "Escape") onclose()
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
$effect(() => {
|
|
32
|
+
if (!open) return
|
|
33
|
+
window.addEventListener("keydown", handleKeydown)
|
|
34
|
+
return () => window.removeEventListener("keydown", handleKeydown)
|
|
35
|
+
})
|
|
36
|
+
</script>
|
|
37
|
+
|
|
38
|
+
{#if open}
|
|
39
|
+
<!-- svelte-ignore a11y_click_events_have_key_events -->
|
|
40
|
+
<div
|
|
41
|
+
class="e11y-backdrop"
|
|
42
|
+
role="presentation"
|
|
43
|
+
onclick={onclose}
|
|
44
|
+
in:circleExpand={{ ...o, duration: openMs }}
|
|
45
|
+
out:circleCollapse={{ ...o, duration: closeMs }}
|
|
46
|
+
>
|
|
47
|
+
<div
|
|
48
|
+
class="e11y-sheet"
|
|
49
|
+
role="dialog"
|
|
50
|
+
aria-modal="true"
|
|
51
|
+
aria-label="e11y overlay"
|
|
52
|
+
tabindex="-1"
|
|
53
|
+
onclick={(e) => e.stopPropagation()}
|
|
54
|
+
>
|
|
55
|
+
<div class="e11y-panel-header">
|
|
56
|
+
<div class="e11y-panel-header-top">
|
|
57
|
+
<div class="e11y-panel-header-left">
|
|
58
|
+
{#if headerTopLeft}
|
|
59
|
+
{@render headerTopLeft()}
|
|
60
|
+
{/if}
|
|
61
|
+
</div>
|
|
62
|
+
<div class="e11y-panel-header-right">
|
|
63
|
+
{#if headerTopRight}
|
|
64
|
+
{@render headerTopRight()}
|
|
65
|
+
{/if}
|
|
66
|
+
<button type="button" class="e11y-icon-btn" onclick={onclose} aria-label="Close"
|
|
67
|
+
>×</button
|
|
68
|
+
>
|
|
69
|
+
</div>
|
|
70
|
+
</div>
|
|
71
|
+
{#if headerBottom}
|
|
72
|
+
<div class="e11y-panel-header-bottom">
|
|
73
|
+
{@render headerBottom()}
|
|
74
|
+
</div>
|
|
75
|
+
{/if}
|
|
76
|
+
</div>
|
|
77
|
+
<div class="e11y-panel-body">
|
|
78
|
+
{@render children()}
|
|
79
|
+
</div>
|
|
80
|
+
</div>
|
|
81
|
+
</div>
|
|
82
|
+
{/if}
|
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
<script lang="ts" module>
|
|
2
|
+
export type TimelineTimeRange = { startMs: number; endMs: number }
|
|
3
|
+
</script>
|
|
4
|
+
|
|
5
|
+
<script lang="ts">
|
|
6
|
+
import { scaleTime, scaleLinear } from "d3-scale"
|
|
7
|
+
import { Axis, Chart, Svg } from "layerchart"
|
|
8
|
+
|
|
9
|
+
let {
|
|
10
|
+
interactions = [] as Record<string, unknown>[],
|
|
11
|
+
timeRange = $bindable(null as TimelineTimeRange | null),
|
|
12
|
+
} = $props()
|
|
13
|
+
|
|
14
|
+
const CHART_PAD = { top: 8, right: 6, bottom: 20, left: 4 } as const
|
|
15
|
+
const HEIGHT = 64
|
|
16
|
+
|
|
17
|
+
const COL_ERR = "var(--e11y-err)"
|
|
18
|
+
const COL_WARN = "var(--e11y-warn)"
|
|
19
|
+
const COL_REST = "var(--e11y-ok)"
|
|
20
|
+
const COL_SEL_SHADE = "var(--e11y-sel-bg)"
|
|
21
|
+
|
|
22
|
+
let parsed = $derived.by(() => {
|
|
23
|
+
return interactions
|
|
24
|
+
.map((i) => {
|
|
25
|
+
const t = new Date(String(i.started_at || "")).getTime()
|
|
26
|
+
const d = Number(i.duration) || 0
|
|
27
|
+
const s = Number(i.status) || 200
|
|
28
|
+
return { t, d, s, id: String(i.id || Math.random()) }
|
|
29
|
+
})
|
|
30
|
+
.filter((i) => !isNaN(i.t))
|
|
31
|
+
// sort by time so drawing order is consistent
|
|
32
|
+
.sort((a, b) => a.t - b.t)
|
|
33
|
+
})
|
|
34
|
+
|
|
35
|
+
let tExtents = $derived.by(() => {
|
|
36
|
+
if (parsed.length === 0) return { min: 0, max: 1 }
|
|
37
|
+
let min = parsed[0].t
|
|
38
|
+
let max = parsed[parsed.length - 1].t
|
|
39
|
+
if (max === min) {
|
|
40
|
+
min -= 1000
|
|
41
|
+
max += 1000
|
|
42
|
+
}
|
|
43
|
+
return { min, max }
|
|
44
|
+
})
|
|
45
|
+
|
|
46
|
+
let dMax = $derived.by(() => {
|
|
47
|
+
if (parsed.length === 0) return 1
|
|
48
|
+
return Math.max(1, ...parsed.map((p) => p.d))
|
|
49
|
+
})
|
|
50
|
+
|
|
51
|
+
let chartWidth = $state(0)
|
|
52
|
+
|
|
53
|
+
let xTime = $derived(
|
|
54
|
+
scaleTime()
|
|
55
|
+
.domain([new Date(tExtents.min), new Date(tExtents.max)])
|
|
56
|
+
.range([0, Math.max(0, chartWidth - CHART_PAD.left - CHART_PAD.right)])
|
|
57
|
+
)
|
|
58
|
+
|
|
59
|
+
let yLinear = $derived(
|
|
60
|
+
scaleLinear()
|
|
61
|
+
.domain([0, dMax])
|
|
62
|
+
.range([Math.max(0, HEIGHT - CHART_PAD.top - CHART_PAD.bottom), 0])
|
|
63
|
+
)
|
|
64
|
+
|
|
65
|
+
let chartHost: HTMLDivElement | null = $state(null)
|
|
66
|
+
let brushOverlay: HTMLDivElement | null = $state(null)
|
|
67
|
+
let dragA = $state<number | null>(null)
|
|
68
|
+
let dragB = $state<number | null>(null)
|
|
69
|
+
let dragging = $state(false)
|
|
70
|
+
|
|
71
|
+
function formatTick(d: Date): string {
|
|
72
|
+
return d.toISOString().slice(11, 23) + "Z"
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
function clientToInnerPlotX(clientX: number, clientY: number): number | null {
|
|
76
|
+
const root = chartHost
|
|
77
|
+
if (!root) return null
|
|
78
|
+
const svg = root.querySelector<SVGSVGElement>("svg.layercake-layout-svg")
|
|
79
|
+
const g = svg?.querySelector<SVGGElement>(".layercake-layout-svg_g")
|
|
80
|
+
if (!svg || !g) return null
|
|
81
|
+
const m = g.getScreenCTM()
|
|
82
|
+
if (!m) return null
|
|
83
|
+
return new DOMPoint(clientX, clientY).matrixTransform(m.inverse()).x
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
function timeFromClientX(clientX: number, clientY: number): number {
|
|
87
|
+
if (parsed.length === 0) return 0
|
|
88
|
+
const root = chartHost
|
|
89
|
+
if (!root) return 0
|
|
90
|
+
let xInner = clientToInnerPlotX(clientX, clientY)
|
|
91
|
+
if (xInner == null || !Number.isFinite(xInner)) {
|
|
92
|
+
const svg = root.querySelector<SVGSVGElement>("svg.layercake-layout-svg")
|
|
93
|
+
const rect = svg?.getBoundingClientRect() ?? root.getBoundingClientRect()
|
|
94
|
+
xInner = clientX - rect.left - CHART_PAD.left
|
|
95
|
+
}
|
|
96
|
+
const range = xTime.range()
|
|
97
|
+
xInner = Math.max(range[0], Math.min(range[1], xInner))
|
|
98
|
+
return xTime.invert(xInner).getTime()
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
function commitRange(t0: number, t1: number): void {
|
|
102
|
+
const startMs = Math.min(t0, t1)
|
|
103
|
+
const endMs = Math.max(t0, t1)
|
|
104
|
+
timeRange = { startMs, endMs }
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
function onPointerDown(e: PointerEvent): void {
|
|
108
|
+
if (e.button !== 0) return
|
|
109
|
+
if (parsed.length === 0) return
|
|
110
|
+
const cap = brushOverlay ?? chartHost
|
|
111
|
+
if (!cap) return
|
|
112
|
+
try {
|
|
113
|
+
cap.setPointerCapture(e.pointerId)
|
|
114
|
+
} catch {
|
|
115
|
+
/* ignore */
|
|
116
|
+
}
|
|
117
|
+
dragging = true
|
|
118
|
+
const t = timeFromClientX(e.clientX, e.clientY)
|
|
119
|
+
dragA = t
|
|
120
|
+
dragB = t
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
function onPointerMove(e: PointerEvent): void {
|
|
124
|
+
if (!dragging || dragA === null) return
|
|
125
|
+
dragB = timeFromClientX(e.clientX, e.clientY)
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
function onPointerUp(e: PointerEvent): void {
|
|
129
|
+
const cap = brushOverlay ?? chartHost
|
|
130
|
+
if (!cap) return
|
|
131
|
+
try {
|
|
132
|
+
cap.releasePointerCapture(e.pointerId)
|
|
133
|
+
} catch {
|
|
134
|
+
/* ignore */
|
|
135
|
+
}
|
|
136
|
+
if (dragging && dragA !== null && dragB !== null) {
|
|
137
|
+
commitRange(dragA, dragB)
|
|
138
|
+
}
|
|
139
|
+
dragging = false
|
|
140
|
+
dragA = null
|
|
141
|
+
dragB = null
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
function onDoubleClick(): void {
|
|
145
|
+
timeRange = null
|
|
146
|
+
}
|
|
147
|
+
|
|
148
|
+
function clearRange(): void {
|
|
149
|
+
timeRange = null
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
let activeRange = $derived.by(() => {
|
|
153
|
+
if (dragging && dragA !== null && dragB !== null) {
|
|
154
|
+
return { startMs: Math.min(dragA, dragB), endMs: Math.max(dragA, dragB) }
|
|
155
|
+
}
|
|
156
|
+
return timeRange
|
|
157
|
+
})
|
|
158
|
+
|
|
159
|
+
let selectionLayout = $derived.by(() => {
|
|
160
|
+
const r = activeRange
|
|
161
|
+
if (!r) return null
|
|
162
|
+
const x0 = xTime(new Date(r.startMs))
|
|
163
|
+
const x1 = xTime(new Date(r.endMs))
|
|
164
|
+
const plotH = HEIGHT - CHART_PAD.top - CHART_PAD.bottom
|
|
165
|
+
return {
|
|
166
|
+
left: CHART_PAD.left + x0,
|
|
167
|
+
top: CHART_PAD.top,
|
|
168
|
+
width: Math.max(0, x1 - x0),
|
|
169
|
+
height: Math.max(0, plotH),
|
|
170
|
+
}
|
|
171
|
+
})
|
|
172
|
+
|
|
173
|
+
function colorForStatus(s: number): string {
|
|
174
|
+
if (s >= 500) return COL_ERR
|
|
175
|
+
if (s >= 400) return COL_WARN
|
|
176
|
+
return COL_REST
|
|
177
|
+
}
|
|
178
|
+
</script>
|
|
179
|
+
|
|
180
|
+
{#if parsed.length > 0}
|
|
181
|
+
<div class="e11y-histo-wrap">
|
|
182
|
+
<div
|
|
183
|
+
bind:this={chartHost}
|
|
184
|
+
bind:clientWidth={chartWidth}
|
|
185
|
+
class="e11y-histo-chart-host"
|
|
186
|
+
style:height="{HEIGHT}px"
|
|
187
|
+
style:min-height="{HEIGHT}px"
|
|
188
|
+
role="application"
|
|
189
|
+
aria-label="Interactions timeline. Drag to filter."
|
|
190
|
+
>
|
|
191
|
+
<Chart
|
|
192
|
+
data={parsed}
|
|
193
|
+
x="t"
|
|
194
|
+
xScale={xTime}
|
|
195
|
+
y="d"
|
|
196
|
+
yScale={yLinear}
|
|
197
|
+
padding={{ top: CHART_PAD.top, right: CHART_PAD.right, bottom: CHART_PAD.bottom, left: CHART_PAD.left }}
|
|
198
|
+
>
|
|
199
|
+
<Svg class="e11y-histo-svg">
|
|
200
|
+
{#each parsed as p (p.id)}
|
|
201
|
+
<line
|
|
202
|
+
x1={xTime(new Date(p.t))}
|
|
203
|
+
x2={xTime(new Date(p.t))}
|
|
204
|
+
y1={yLinear(0)}
|
|
205
|
+
y2={yLinear(p.d)}
|
|
206
|
+
stroke={colorForStatus(p.s)}
|
|
207
|
+
stroke-width="2"
|
|
208
|
+
stroke-linecap="round"
|
|
209
|
+
/>
|
|
210
|
+
{/each}
|
|
211
|
+
|
|
212
|
+
<Axis
|
|
213
|
+
placement="bottom"
|
|
214
|
+
rule={false}
|
|
215
|
+
grid={false}
|
|
216
|
+
ticks={4}
|
|
217
|
+
format={(d) => formatTick(d as Date)}
|
|
218
|
+
tickLength={3}
|
|
219
|
+
tickLabelProps={{ class: "e11y-histo-axis-tick" }}
|
|
220
|
+
/>
|
|
221
|
+
</Svg>
|
|
222
|
+
</Chart>
|
|
223
|
+
{#if selectionLayout}
|
|
224
|
+
<div
|
|
225
|
+
class="e11y-histo-sel-shade"
|
|
226
|
+
style:left="{selectionLayout.left}px"
|
|
227
|
+
style:top="{selectionLayout.top}px"
|
|
228
|
+
style:width="{selectionLayout.width}px"
|
|
229
|
+
style:height="{selectionLayout.height}px"
|
|
230
|
+
aria-hidden="true"
|
|
231
|
+
></div>
|
|
232
|
+
{/if}
|
|
233
|
+
<div
|
|
234
|
+
bind:this={brushOverlay}
|
|
235
|
+
class="e11y-histo-brush-overlay"
|
|
236
|
+
aria-hidden="true"
|
|
237
|
+
onpointerdown={onPointerDown}
|
|
238
|
+
onpointermove={onPointerMove}
|
|
239
|
+
onpointerup={onPointerUp}
|
|
240
|
+
onpointercancel={onPointerUp}
|
|
241
|
+
ondblclick={onDoubleClick}
|
|
242
|
+
></div>
|
|
243
|
+
</div>
|
|
244
|
+
<div class="e11y-histo-footer">
|
|
245
|
+
<div class="e11y-histo-legend">
|
|
246
|
+
<span><i class="e11y-histo-dot e11y-histo-dot--err"></i> 5xx</span>
|
|
247
|
+
<span><i class="e11y-histo-dot e11y-histo-dot--warn"></i> 4xx</span>
|
|
248
|
+
<span><i class="e11y-histo-dot e11y-histo-dot--rest"></i> ok</span>
|
|
249
|
+
</div>
|
|
250
|
+
<div class="e11y-histo-meta">
|
|
251
|
+
{#if activeRange}
|
|
252
|
+
<span class="e11y-histo-filter">
|
|
253
|
+
Filter: {formatTick(new Date(activeRange.startMs))}–{formatTick(new Date(activeRange.endMs))}
|
|
254
|
+
</span>
|
|
255
|
+
{#if timeRange}
|
|
256
|
+
<button type="button" class="e11y-histo-clear" onclick={clearRange}>Clear</button>
|
|
257
|
+
{/if}
|
|
258
|
+
{:else}
|
|
259
|
+
<span class="e11y-histo-hint">Drag to narrow · double-click to reset</span>
|
|
260
|
+
{/if}
|
|
261
|
+
</div>
|
|
262
|
+
</div>
|
|
263
|
+
</div>
|
|
264
|
+
{/if}
|