@plumile/router 0.1.32 → 0.1.34

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 (47) hide show
  1. package/CHANGELOG.md +7 -0
  2. package/README.md +42 -21
  3. package/lib/esm/index.d.ts +1 -0
  4. package/lib/esm/index.d.ts.map +1 -1
  5. package/lib/esm/index.js +2 -1
  6. package/lib/esm/instrumentation/Instrumentation.d.ts +66 -0
  7. package/lib/esm/instrumentation/Instrumentation.d.ts.map +1 -0
  8. package/lib/esm/instrumentation/Instrumentation.js +59 -0
  9. package/lib/esm/instrumentation/adapters/devtoolsBridge.d.ts +14 -0
  10. package/lib/esm/instrumentation/adapters/devtoolsBridge.d.ts.map +1 -0
  11. package/lib/esm/instrumentation/adapters/devtoolsBridge.js +118 -0
  12. package/lib/esm/instrumentation/adapters/logger.d.ts +10 -0
  13. package/lib/esm/instrumentation/adapters/logger.d.ts.map +1 -0
  14. package/lib/esm/instrumentation/adapters/logger.js +19 -0
  15. package/lib/esm/instrumentation/index.d.ts +4 -0
  16. package/lib/esm/instrumentation/index.d.ts.map +1 -0
  17. package/lib/esm/instrumentation/index.js +4 -0
  18. package/lib/esm/routing/Link.d.ts.map +1 -1
  19. package/lib/esm/routing/Link.js +8 -17
  20. package/lib/esm/routing/createRouter.d.ts +2 -1
  21. package/lib/esm/routing/createRouter.d.ts.map +1 -1
  22. package/lib/esm/routing/createRouter.js +184 -261
  23. package/lib/esm/tools.d.ts +6 -5
  24. package/lib/esm/tools.d.ts.map +1 -1
  25. package/lib/esm/tools.js +17 -34
  26. package/lib/esm/types.d.ts +7 -49
  27. package/lib/esm/types.d.ts.map +1 -1
  28. package/lib/esm/types.js +2 -28
  29. package/lib/tsconfig.esm.tsbuildinfo +1 -1
  30. package/lib/types/index.d.ts +1 -0
  31. package/lib/types/index.d.ts.map +1 -1
  32. package/lib/types/instrumentation/Instrumentation.d.ts +66 -0
  33. package/lib/types/instrumentation/Instrumentation.d.ts.map +1 -0
  34. package/lib/types/instrumentation/adapters/devtoolsBridge.d.ts +14 -0
  35. package/lib/types/instrumentation/adapters/devtoolsBridge.d.ts.map +1 -0
  36. package/lib/types/instrumentation/adapters/logger.d.ts +10 -0
  37. package/lib/types/instrumentation/adapters/logger.d.ts.map +1 -0
  38. package/lib/types/instrumentation/index.d.ts +4 -0
  39. package/lib/types/instrumentation/index.d.ts.map +1 -0
  40. package/lib/types/routing/Link.d.ts.map +1 -1
  41. package/lib/types/routing/createRouter.d.ts +2 -1
  42. package/lib/types/routing/createRouter.d.ts.map +1 -1
  43. package/lib/types/tools.d.ts +6 -5
  44. package/lib/types/tools.d.ts.map +1 -1
  45. package/lib/types/types.d.ts +7 -49
  46. package/lib/types/types.d.ts.map +1 -1
  47. package/package.json +33 -18
package/CHANGELOG.md ADDED
@@ -0,0 +1,7 @@
1
+ # Changelog
2
+
3
+ All notable changes to this project will be documented in this file.
4
+
5
+ ## 1.0.0 - First stable version
6
+
7
+ - First public release of `@plumile/router`.
package/README.md CHANGED
@@ -21,10 +21,11 @@ npm install @plumile/router
21
21
 
22
22
  ## Peer Dependencies
23
23
 
24
- This package requires the following peer dependencies:
24
+ This package expects `react` and `react-dom` to be available in your project
25
+ (React 18 or 19).
25
26
 
26
27
  ```bash
27
- npm install react react-dom react-relay relay-runtime @types/react @types/react-dom @types/react-relay
28
+ npm install react react-dom
28
29
  ```
29
30
 
30
31
  ## Module Entry Points
@@ -37,11 +38,11 @@ This package is ESM-only. If your tooling expects CommonJS, enable ESM support (
37
38
 
38
39
  ## Documentation & Guides
39
40
 
40
- - Getting started: [docs/index.md](../../docs/index.md)
41
- - Relay integration walkthrough: [docs/router-relay.md](../../docs/router-relay.md)
42
- - Examples: [EXAMPLES.md](./EXAMPLES.md)
43
- - Migration strategies: [MIGRATION.md](./MIGRATION.md)
44
- - DevTools extension usage: [docs/router-devtools-extension.md](../../docs/router-devtools-extension.md)
41
+ - Getting started: [User guide](https://gitlab.com/plumile/js/-/blob/main/docs/index.md)
42
+ - Relay integration walkthrough: [Relay guide](https://gitlab.com/plumile/js/-/blob/main/docs/router-relay.md)
43
+ - Examples: [Examples](https://gitlab.com/plumile/js/-/blob/main/packages/router/EXAMPLES.md)
44
+ - Migration strategies: [Migration guide](https://gitlab.com/plumile/js/-/blob/main/packages/router/MIGRATION.md)
45
+ - DevTools extension usage: [DevTools documentation](https://gitlab.com/plumile/js/-/blob/main/docs/router-devtools-extension.md)
45
46
 
46
47
  ## Quick Start
47
48
 
@@ -130,14 +131,14 @@ function Navigation() {
130
131
 
131
132
  ## Hooks Overview
132
133
 
133
- | Hook | Signature | Purpose | Example |
134
- | --------------------------- | -------------------------------------- | ----------------------------------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------- | --------------------------- |
135
- | `useNavigate()` | `() => Navigate` | Imperative navigation with type-safe filters + params. | `const navigate = useNavigate(); navigate({ pathname: '/products', filters: { page: { eq: 2 } } });` |
136
- | `useFilters(schema)` | `(schema) => [filters, actions]` | Read/mutate filters derived from the given schema; actions include `set`, `remove`, `merge`, `clear`. | `const [filters, actions] = useFilters(productFilters); actions.set('price', 'gt', 10);` |
137
- | `useQuery()` | `() => Record<string, string | string[]>` | Raw query aggregation (legacy/simple use). | `const query = useQuery();` |
138
- | `useQueryState(key, opts?)` | `(key, options?) => [value, setValue]` | Two-way binding for a single query parameter with default/replace options. | `const [page, setPage] = useQueryState('page', { defaultValue: 1 });` |
139
- | `useFilterDiagnostics()` | `() => Diagnostic[]` | Surface parsing issues (unknown field/operator) for UI display or logging. | `const diagnostics = useFilterDiagnostics();` |
140
- | `useAllQuery(opts?)` | `(options?) => QueryLike` | Access filters + raw query merged together, useful for debugging or incremental migrations. | `const all = useAllQuery();` |
134
+ | Hook | Signature | Purpose | Example |
135
+ | ------------------------------ | ------------------------------------------ | -------------------------------------------------------------------------- | ---------------------------------------------------------------------------------------------------- |
136
+ | `useNavigate()` | `() => Navigate` | Imperative navigation with type-safe params and filters. | `const navigate = useNavigate(); navigate({ pathname: '/products', filters: { page: { eq: 2 } } });` |
137
+ | `useFilters(schema)` | `(schema) => [filters, actions]` | Read and mutate filters inferred from a schema. | `const [filters, actions] = useFilters(productFilters); actions.set('price', 'gt', 10);` |
138
+ | `useQuery()` | `() => Record<string, string \| string[]>` | Access the raw query aggregation (legacy/simple use). | `const query = useQuery();` |
139
+ | `useQueryState(key, options?)` | `(key, options?) => [value, setValue]` | Two-way binding for a single query parameter with default/replace options. | `const [page, setPage] = useQueryState('page', { defaultValue: 1 });` |
140
+ | `useFilterDiagnostics()` | `() => Diagnostic[]` | Surface parsing issues (unknown fields/operators) for UI or logging. | `const diagnostics = useFilterDiagnostics();` |
141
+ | `useAllQuery(options?)` | `(options?) => QueryLike` | Merge filters and raw query, helpful during migrations. | `const all = useAllQuery();` |
141
142
 
142
143
  ## API Reference
143
144
 
@@ -366,15 +367,35 @@ const search = buildCombinedSearch({
366
367
 
367
368
  Guideline: If you need to derive lightweight projections (e.g. `const { page } = typed`), you can still destructure; but avoid spreading into a new object if you rely on reference equality downstream.
368
369
 
369
- #### Inspection & Debugging
370
+ #### Inspection & Instrumentation
370
371
 
371
- Activez le bridge de debug en développement avec l’option `debug` :
372
+ Activez l’instrumentation en développement pour exposer un bridge DevTools :
372
373
 
373
374
  ```ts
374
- createRouter(routes, { debug: true });
375
+ import {
376
+ createRouter,
377
+ createDevtoolsBridgeInstrumentation,
378
+ } from '@plumile/router';
379
+
380
+ const devtools = createDevtoolsBridgeInstrumentation();
381
+ const { context } = createRouter(routes, { instrumentations: [devtools] });
375
382
  ```
376
383
 
377
- Cela expose `window.__PLUMILE_ROUTER__` avec `get()` et `subscribe()` (non présent en production).
384
+ Aucun bridge n’est publié en production. Vous pouvez chaîner plusieurs instrumentations (par exemple la console) :
385
+
386
+ ```ts
387
+ import {
388
+ createConsoleLoggerInstrumentation,
389
+ createDevtoolsBridgeInstrumentation,
390
+ } from '@plumile/router';
391
+
392
+ const instrumentations = [
393
+ createDevtoolsBridgeInstrumentation(),
394
+ createConsoleLoggerInstrumentation({ label: 'router' }),
395
+ ];
396
+
397
+ createRouter(routes, { instrumentations });
398
+ ```
378
399
 
379
400
  Pour l’inspection visuelle (timeline, filtres, prepared data), installez la Plumile Router DevTools extension (voir [docs/router-devtools-extension.md](../../docs/router-devtools-extension.md)).
380
401
 
@@ -584,10 +605,10 @@ function MyComponent() {
584
605
 
585
606
  const handleHover = () => {
586
607
  // Preload code only
587
- router.preloadCode('/users/123');
608
+ router.preloadCode({ pathname: '/users/123' });
588
609
 
589
610
  // Preload code and data
590
- router.preload('/users/123');
611
+ router.preload({ pathname: '/users/123' });
591
612
  };
592
613
 
593
614
  return (
@@ -5,4 +5,5 @@ export * from './builder.js';
5
5
  export * from './ResourcePage.js';
6
6
  export * from './tools.js';
7
7
  export type * from './types.js';
8
+ export * from './instrumentation/index.js';
8
9
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AACA,cAAc,mBAAmB,CAAC;AAGlC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,cAAc,CAAC;AAG7B,cAAc,mBAAmB,CAAC;AAGlC,cAAc,YAAY,CAAC;AAG3B,mBAAmB,YAAY,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/index.ts"],"names":[],"mappings":"AACA,cAAc,mBAAmB,CAAC;AAGlC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,oBAAoB,CAAC;AAGnC,cAAc,cAAc,CAAC;AAG7B,cAAc,mBAAmB,CAAC;AAGlC,cAAc,YAAY,CAAC;AAG3B,mBAAmB,YAAY,CAAC;AAGhC,cAAc,4BAA4B,CAAC"}
package/lib/esm/index.js CHANGED
@@ -4,4 +4,5 @@ export * from './routing/index.js';
4
4
  export * from './builder.js';
5
5
  export * from './ResourcePage.js';
6
6
  export * from './tools.js';
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsY0FBYyxtQkFBbUIsQ0FBQztBQUdsQyxjQUFjLG9CQUFvQixDQUFDO0FBR25DLGNBQWMsb0JBQW9CLENBQUM7QUFHbkMsY0FBYyxjQUFjLENBQUM7QUFHN0IsY0FBYyxtQkFBbUIsQ0FBQztBQUdsQyxjQUFjLFlBQVksQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8vIEV4cG9ydCBlcnJvciBoYW5kbGluZyB1dGlsaXRpZXNcbmV4cG9ydCAqIGZyb20gJy4vZXJyb3JzL2luZGV4LmpzJztcblxuLy8gRXhwb3J0IGJyb3dzZXIgaGlzdG9yeSBtYW5hZ2VtZW50XG5leHBvcnQgKiBmcm9tICcuL2hpc3RvcnkvaW5kZXguanMnO1xuXG4vLyBFeHBvcnQgcm91dGluZyBjb21wb25lbnRzIGFuZCB1dGlsaXRpZXNcbmV4cG9ydCAqIGZyb20gJy4vcm91dGluZy9pbmRleC5qcyc7XG5cbi8vIEV4cG9ydCByb3V0ZSBidWlsZGluZyB1dGlsaXRpZXNcbmV4cG9ydCAqIGZyb20gJy4vYnVpbGRlci5qcyc7XG5cbi8vIEV4cG9ydCByZXNvdXJjZSBwYWdlIG1hbmFnZW1lbnQgZm9yIGxhenkgbG9hZGluZ1xuZXhwb3J0ICogZnJvbSAnLi9SZXNvdXJjZVBhZ2UuanMnO1xuXG4vLyBFeHBvcnQgdXRpbGl0eSBmdW5jdGlvbnNcbmV4cG9ydCAqIGZyb20gJy4vdG9vbHMuanMnO1xuXG4vLyBFeHBvcnQgYWxsIFR5cGVTY3JpcHQgdHlwZXNcbmV4cG9ydCB0eXBlICogZnJvbSAnLi90eXBlcy5qcyc7XG4iXX0=
7
+ export * from './instrumentation/index.js';
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsY0FBYyxtQkFBbUIsQ0FBQztBQUdsQyxjQUFjLG9CQUFvQixDQUFDO0FBR25DLGNBQWMsb0JBQW9CLENBQUM7QUFHbkMsY0FBYyxjQUFjLENBQUM7QUFHN0IsY0FBYyxtQkFBbUIsQ0FBQztBQUdsQyxjQUFjLFlBQVksQ0FBQztBQU0zQixjQUFjLDRCQUE0QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gRXhwb3J0IGVycm9yIGhhbmRsaW5nIHV0aWxpdGllc1xuZXhwb3J0ICogZnJvbSAnLi9lcnJvcnMvaW5kZXguanMnO1xuXG4vLyBFeHBvcnQgYnJvd3NlciBoaXN0b3J5IG1hbmFnZW1lbnRcbmV4cG9ydCAqIGZyb20gJy4vaGlzdG9yeS9pbmRleC5qcyc7XG5cbi8vIEV4cG9ydCByb3V0aW5nIGNvbXBvbmVudHMgYW5kIHV0aWxpdGllc1xuZXhwb3J0ICogZnJvbSAnLi9yb3V0aW5nL2luZGV4LmpzJztcblxuLy8gRXhwb3J0IHJvdXRlIGJ1aWxkaW5nIHV0aWxpdGllc1xuZXhwb3J0ICogZnJvbSAnLi9idWlsZGVyLmpzJztcblxuLy8gRXhwb3J0IHJlc291cmNlIHBhZ2UgbWFuYWdlbWVudCBmb3IgbGF6eSBsb2FkaW5nXG5leHBvcnQgKiBmcm9tICcuL1Jlc291cmNlUGFnZS5qcyc7XG5cbi8vIEV4cG9ydCB1dGlsaXR5IGZ1bmN0aW9uc1xuZXhwb3J0ICogZnJvbSAnLi90b29scy5qcyc7XG5cbi8vIEV4cG9ydCBhbGwgVHlwZVNjcmlwdCB0eXBlc1xuZXhwb3J0IHR5cGUgKiBmcm9tICcuL3R5cGVzLmpzJztcblxuLy8gRXhwb3J0IGluc3RydW1lbnRhdGlvbiBoZWxwZXJzXG5leHBvcnQgKiBmcm9tICcuL2luc3RydW1lbnRhdGlvbi9pbmRleC5qcyc7XG4iXX0=
@@ -0,0 +1,66 @@
1
+ import type { RouteEntry } from '../types.js';
2
+ export type RouterNavigationSource = 'link-click' | 'link-hover' | 'preload-hover' | 'programmatic' | 'popstate-back' | 'popstate-forward' | 'popstate-unknown' | 'external' | 'normalize';
3
+ export type RouterLocationSnapshot = {
4
+ pathname: string;
5
+ search: string;
6
+ hash: string;
7
+ };
8
+ export type RouterEntrySnapshot = {
9
+ location: RouterLocationSnapshot;
10
+ routePath?: string;
11
+ preparedMatch?: RouteEntry<any>['preparedMatch'];
12
+ filters?: RouteEntry<any>['filters'];
13
+ filterDiagnostics?: RouteEntry<any>['filterDiagnostics'];
14
+ activeQuerySchema?: RouteEntry<any>['activeQuerySchema'];
15
+ };
16
+ export type RouterEventBase = {
17
+ timestamp: number;
18
+ location: RouterLocationSnapshot;
19
+ };
20
+ export type RouterSnapshotEvent = RouterEventBase & {
21
+ kind: 'snapshot';
22
+ source: RouterNavigationSource;
23
+ routePath?: string;
24
+ };
25
+ export type RouterPreloadEvent = RouterEventBase & {
26
+ kind: 'preload';
27
+ source: RouterNavigationSource;
28
+ targetPathname: string;
29
+ mode: 'code' | 'full';
30
+ };
31
+ export type RouterHistoryEvent = RouterEventBase & {
32
+ kind: 'history';
33
+ source: RouterNavigationSource;
34
+ action: 'push' | 'replace' | 'normalize';
35
+ details?: Record<string, unknown>;
36
+ };
37
+ export type RouterPopstateEvent = RouterEventBase & {
38
+ kind: 'popstate';
39
+ source: RouterNavigationSource;
40
+ direction: 'back' | 'forward' | 'unknown';
41
+ details?: Record<string, unknown>;
42
+ };
43
+ export type RouterPrepareEvent = RouterEventBase & {
44
+ kind: 'prepare-start' | 'prepare-end';
45
+ source: RouterNavigationSource;
46
+ routePath?: string;
47
+ moduleId?: string;
48
+ durationMs?: number;
49
+ hasPrepared?: boolean;
50
+ };
51
+ export type RouterEvent = RouterSnapshotEvent | RouterPreloadEvent | RouterHistoryEvent | RouterPopstateEvent | RouterPrepareEvent;
52
+ export type InstrumentationAPI = {
53
+ onEvent?: (event: RouterEvent) => void;
54
+ onEntryChange?: (entry: RouterEntrySnapshot) => void;
55
+ dispose?: () => void;
56
+ };
57
+ export type InstrumentationRegistry = {
58
+ register: (instrumentation: InstrumentationAPI) => void;
59
+ unregister: (instrumentation: InstrumentationAPI) => void;
60
+ emitEvent: (event: RouterEvent) => void;
61
+ notifyEntryChange: (entry: RouterEntrySnapshot) => void;
62
+ dispose: () => void;
63
+ readonly size: number;
64
+ };
65
+ export declare function createInstrumentationRegistry(initial?: InstrumentationAPI[]): InstrumentationRegistry;
66
+ //# sourceMappingURL=Instrumentation.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Instrumentation.d.ts","sourceRoot":"","sources":["../../../src/instrumentation/Instrumentation.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,59 @@
1
+ export function createInstrumentationRegistry(initial = []) {
2
+ const listeners = new Set();
3
+ initial.forEach((instrumentation) => {
4
+ listeners.add(instrumentation);
5
+ });
6
+ let disposed = false;
7
+ function guard() {
8
+ if (disposed) {
9
+ throw new Error('Instrumentation registry was disposed');
10
+ }
11
+ }
12
+ function safeInvoke(instrumentation, method, payload) {
13
+ const handler = instrumentation[method];
14
+ if (typeof handler !== 'function') {
15
+ return;
16
+ }
17
+ try {
18
+ handler(...payload);
19
+ }
20
+ catch {
21
+ }
22
+ }
23
+ return {
24
+ register(instrumentation) {
25
+ guard();
26
+ listeners.add(instrumentation);
27
+ },
28
+ unregister(instrumentation) {
29
+ guard();
30
+ listeners.delete(instrumentation);
31
+ },
32
+ emitEvent(event) {
33
+ guard();
34
+ listeners.forEach((instrumentation) => {
35
+ safeInvoke(instrumentation, 'onEvent', [event]);
36
+ });
37
+ },
38
+ notifyEntryChange(entry) {
39
+ guard();
40
+ listeners.forEach((instrumentation) => {
41
+ safeInvoke(instrumentation, 'onEntryChange', [entry]);
42
+ });
43
+ },
44
+ dispose() {
45
+ if (disposed) {
46
+ return;
47
+ }
48
+ listeners.forEach((instrumentation) => {
49
+ safeInvoke(instrumentation, 'dispose', []);
50
+ });
51
+ listeners.clear();
52
+ disposed = true;
53
+ },
54
+ get size() {
55
+ return listeners.size;
56
+ },
57
+ };
58
+ }
59
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Instrumentation.js","sourceRoot":"","sources":["../../../src/instrumentation/Instrumentation.ts"],"names":[],"mappings":"AAoHA,MAAM,UAAU,6BAA6B,CAC3C,UAAgC,EAAE;IAElC,MAAM,SAAS,GAAG,IAAI,GAAG,EAAsB,CAAC;IAChD,OAAO,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;QAClC,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;IACjC,CAAC,CAAC,CAAC;IAEH,IAAI,QAAQ,GAAG,KAAK,CAAC;IAKrB,SAAS,KAAK;QACZ,IAAI,QAAQ,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,uCAAuC,CAAC,CAAC;QAC3D,CAAC;IACH,CAAC;IAMD,SAAS,UAAU,CACjB,eAAmC,EACnC,MAAgC,EAChC,OAAkB;QAElB,MAAM,OAAO,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC;QACxC,IAAI,OAAO,OAAO,KAAK,UAAU,EAAE,CAAC;YAClC,OAAO;QACT,CAAC;QACD,IAAI,CAAC;YACF,OAAwC,CAAC,GAAG,OAAO,CAAC,CAAC;QACxD,CAAC;QAAC,MAAM,CAAC;QAET,CAAC;IACH,CAAC;IAED,OAAO;QACL,QAAQ,CAAC,eAAe;YACtB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC;QACjC,CAAC;QACD,UAAU,CAAC,eAAe;YACxB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;QACpC,CAAC;QACD,SAAS,CAAC,KAAK;YACb,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YAClD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,iBAAiB,CAAC,KAAK;YACrB,KAAK,EAAE,CAAC;YACR,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,eAAe,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC;YACxD,CAAC,CAAC,CAAC;QACL,CAAC;QACD,OAAO;YACL,IAAI,QAAQ,EAAE,CAAC;gBACb,OAAO;YACT,CAAC;YACD,SAAS,CAAC,OAAO,CAAC,CAAC,eAAe,EAAE,EAAE;gBACpC,UAAU,CAAC,eAAe,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;YACH,SAAS,CAAC,KAAK,EAAE,CAAC;YAClB,QAAQ,GAAG,IAAI,CAAC;QAClB,CAAC;QACD,IAAI,IAAI;YACN,OAAO,SAAS,CAAC,IAAI,CAAC;QACxB,CAAC;KACF,CAAC;AACJ,CAAC","sourcesContent":["import type { RouteEntry } from '../types.js';\n\n/**\n * Enumerates the origin of navigation-related events exposed to instrumentations.\n *\n * Kept as string literals to avoid leaking debug enums into the production bundle.\n */\nexport type RouterNavigationSource =\n  | 'link-click'\n  | 'link-hover'\n  | 'preload-hover'\n  | 'programmatic'\n  | 'popstate-back'\n  | 'popstate-forward'\n  | 'popstate-unknown'\n  | 'external'\n  | 'normalize';\n\n/**\n * Compact location snapshot shared by all instrumentation events.\n */\nexport type RouterLocationSnapshot = {\n  pathname: string;\n  search: string;\n  hash: string;\n};\n\n/**\n * Summary of the current router entry that can be broadcast to instrumentations.\n * The shape intentionally mirrors public data already available to applications.\n */\nexport type RouterEntrySnapshot = {\n  location: RouterLocationSnapshot;\n  routePath?: string;\n  preparedMatch?: RouteEntry<any>['preparedMatch'];\n  filters?: RouteEntry<any>['filters'];\n  filterDiagnostics?: RouteEntry<any>['filterDiagnostics'];\n  activeQuerySchema?: RouteEntry<any>['activeQuerySchema'];\n};\n\n/**\n * Base payload shared by router events. Each event extends the base with\n * specific details relevant to the instrumentation.\n */\nexport type RouterEventBase = {\n  timestamp: number;\n  location: RouterLocationSnapshot;\n};\n\nexport type RouterSnapshotEvent = RouterEventBase & {\n  kind: 'snapshot';\n  source: RouterNavigationSource;\n  routePath?: string;\n};\n\nexport type RouterPreloadEvent = RouterEventBase & {\n  kind: 'preload';\n  source: RouterNavigationSource;\n  targetPathname: string;\n  mode: 'code' | 'full';\n};\n\nexport type RouterHistoryEvent = RouterEventBase & {\n  kind: 'history';\n  source: RouterNavigationSource;\n  action: 'push' | 'replace' | 'normalize';\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPopstateEvent = RouterEventBase & {\n  kind: 'popstate';\n  source: RouterNavigationSource;\n  direction: 'back' | 'forward' | 'unknown';\n  details?: Record<string, unknown>;\n};\n\nexport type RouterPrepareEvent = RouterEventBase & {\n  kind: 'prepare-start' | 'prepare-end';\n  source: RouterNavigationSource;\n  routePath?: string;\n  moduleId?: string;\n  durationMs?: number;\n  hasPrepared?: boolean;\n};\n\nexport type RouterEvent =\n  | RouterSnapshotEvent\n  | RouterPreloadEvent\n  | RouterHistoryEvent\n  | RouterPopstateEvent\n  | RouterPrepareEvent;\n\n/**\n * Instrumentations can observe router events, entry changes, and optionally\n * dispose resources when the registry is torn down.\n */\nexport type InstrumentationAPI = {\n  onEvent?: (event: RouterEvent) => void;\n  onEntryChange?: (entry: RouterEntrySnapshot) => void;\n  dispose?: () => void;\n};\n\nexport type InstrumentationRegistry = {\n  register: (instrumentation: InstrumentationAPI) => void;\n  unregister: (instrumentation: InstrumentationAPI) => void;\n  emitEvent: (event: RouterEvent) => void;\n  notifyEntryChange: (entry: RouterEntrySnapshot) => void;\n  dispose: () => void;\n  readonly size: number;\n};\n\n/**\n * Creates a registry coordinating a set of instrumentations. The registry keeps\n * consumer code simple: router internals broadcast events without caring about\n * the number or nature of attached instrumentations.\n */\nexport function createInstrumentationRegistry(\n  initial: InstrumentationAPI[] = [],\n): InstrumentationRegistry {\n  const listeners = new Set<InstrumentationAPI>();\n  initial.forEach((instrumentation) => {\n    listeners.add(instrumentation);\n  });\n\n  let disposed = false;\n\n  /**\n   * Throws when the registry has already been disposed. Guards event emission.\n   */\n  function guard() {\n    if (disposed) {\n      throw new Error('Instrumentation registry was disposed');\n    }\n  }\n\n  /**\n   * Safely invokes an optional handler provided by an instrumentation.\n   * Failures are swallowed so router execution continues uninterrupted.\n   */\n  function safeInvoke(\n    instrumentation: InstrumentationAPI,\n    method: keyof InstrumentationAPI,\n    payload: unknown[],\n  ): void {\n    const handler = instrumentation[method];\n    if (typeof handler !== 'function') {\n      return;\n    }\n    try {\n      (handler as (...args: unknown[]) => void)(...payload);\n    } catch {\n      // Instrumentations should never break router execution. Errors are swallowed.\n    }\n  }\n\n  return {\n    register(instrumentation) {\n      guard();\n      listeners.add(instrumentation);\n    },\n    unregister(instrumentation) {\n      guard();\n      listeners.delete(instrumentation);\n    },\n    emitEvent(event) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEvent', [event]);\n      });\n    },\n    notifyEntryChange(entry) {\n      guard();\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'onEntryChange', [entry]);\n      });\n    },\n    dispose() {\n      if (disposed) {\n        return;\n      }\n      listeners.forEach((instrumentation) => {\n        safeInvoke(instrumentation, 'dispose', []);\n      });\n      listeners.clear();\n      disposed = true;\n    },\n    get size() {\n      return listeners.size;\n    },\n  };\n}\n"]}
@@ -0,0 +1,14 @@
1
+ import type { InstrumentationAPI, RouterEntrySnapshot, RouterEvent } from '../Instrumentation.js';
2
+ export type DevtoolsBridgeOptions = {
3
+ historyLimit?: number;
4
+ global?: Record<string, unknown>;
5
+ };
6
+ export type DevtoolsBridge = {
7
+ getEntry: () => RouterEntrySnapshot | null;
8
+ getEvents: () => RouterEvent[];
9
+ subscribeEntries: (callback: (entry: RouterEntrySnapshot) => void) => () => void;
10
+ subscribeEvents: (callback: (event: RouterEvent) => void) => () => void;
11
+ historyLimit: number;
12
+ };
13
+ export declare function createDevtoolsBridgeInstrumentation(options?: DevtoolsBridgeOptions): InstrumentationAPI;
14
+ //# sourceMappingURL=devtoolsBridge.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"devtoolsBridge.d.ts","sourceRoot":"","sources":["../../../../src/instrumentation/adapters/devtoolsBridge.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,kBAAkB,EAClB,mBAAmB,EACnB,WAAW,EACZ,MAAM,uBAAuB,CAAC;AAK/B,MAAM,MAAM,qBAAqB,GAAG;IAClC,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;CAClC,CAAC;AAEF,MAAM,MAAM,cAAc,GAAG;IAC3B,QAAQ,EAAE,MAAM,mBAAmB,GAAG,IAAI,CAAC;IAC3C,SAAS,EAAE,MAAM,WAAW,EAAE,CAAC;IAC/B,gBAAgB,EAAE,CAChB,QAAQ,EAAE,CAAC,KAAK,EAAE,mBAAmB,KAAK,IAAI,KAC3C,MAAM,IAAI,CAAC;IAChB,eAAe,EAAE,CAAC,QAAQ,EAAE,CAAC,KAAK,EAAE,WAAW,KAAK,IAAI,KAAK,MAAM,IAAI,CAAC;IACxE,YAAY,EAAE,MAAM,CAAC;CACtB,CAAC;AAYF,wBAAgB,mCAAmC,CACjD,OAAO,GAAE,qBAA0B,GAClC,kBAAkB,CA8HpB"}
@@ -0,0 +1,118 @@
1
+ const BRIDGE_KEY = '__PLUMILE_ROUTER__';
2
+ const DEFAULT_HISTORY_LIMIT = 200;
3
+ function isObject(value) {
4
+ return typeof value === 'object' && value !== null;
5
+ }
6
+ export function createDevtoolsBridgeInstrumentation(options = {}) {
7
+ const historyLimit = options.historyLimit ?? DEFAULT_HISTORY_LIMIT;
8
+ let globalObject;
9
+ if (options.global != null) {
10
+ globalObject = options.global;
11
+ }
12
+ else if (typeof window !== 'undefined') {
13
+ globalObject = window;
14
+ }
15
+ const eventSubscribers = new Set();
16
+ const entrySubscribers = new Set();
17
+ let disposed = false;
18
+ let bridgeInstalled = false;
19
+ let lastEntry = null;
20
+ let events = [];
21
+ const bridge = {
22
+ getEntry() {
23
+ return lastEntry;
24
+ },
25
+ getEvents() {
26
+ return events.slice();
27
+ },
28
+ subscribeEntries(callback) {
29
+ entrySubscribers.add(callback);
30
+ if (lastEntry != null) {
31
+ callback(lastEntry);
32
+ }
33
+ return () => {
34
+ entrySubscribers.delete(callback);
35
+ };
36
+ },
37
+ subscribeEvents(callback) {
38
+ eventSubscribers.add(callback);
39
+ return () => {
40
+ eventSubscribers.delete(callback);
41
+ };
42
+ },
43
+ historyLimit,
44
+ };
45
+ function installBridge() {
46
+ if (bridgeInstalled || globalObject == null) {
47
+ return;
48
+ }
49
+ globalObject[BRIDGE_KEY] = bridge;
50
+ bridgeInstalled = true;
51
+ }
52
+ function removeBridge() {
53
+ if (!bridgeInstalled || globalObject == null) {
54
+ return;
55
+ }
56
+ if (Object.prototype.hasOwnProperty.call(globalObject, BRIDGE_KEY) &&
57
+ globalObject[BRIDGE_KEY] === bridge) {
58
+ globalObject[BRIDGE_KEY] = undefined;
59
+ }
60
+ bridgeInstalled = false;
61
+ }
62
+ installBridge();
63
+ function publishEvent(event) {
64
+ events = [...events, event];
65
+ if (events.length > historyLimit) {
66
+ events = events.slice(-historyLimit);
67
+ }
68
+ eventSubscribers.forEach((subscriber) => {
69
+ try {
70
+ subscriber(event);
71
+ }
72
+ catch {
73
+ }
74
+ });
75
+ }
76
+ function publishEntry(entry) {
77
+ lastEntry = entry;
78
+ entrySubscribers.forEach((subscriber) => {
79
+ try {
80
+ subscriber(entry);
81
+ }
82
+ catch {
83
+ }
84
+ });
85
+ }
86
+ return {
87
+ onEvent(event) {
88
+ if (disposed) {
89
+ return;
90
+ }
91
+ if (!isObject(event)) {
92
+ return;
93
+ }
94
+ publishEvent(event);
95
+ },
96
+ onEntryChange(entry) {
97
+ if (disposed) {
98
+ return;
99
+ }
100
+ if (!isObject(entry)) {
101
+ return;
102
+ }
103
+ publishEntry(entry);
104
+ },
105
+ dispose() {
106
+ if (disposed) {
107
+ return;
108
+ }
109
+ disposed = true;
110
+ eventSubscribers.clear();
111
+ entrySubscribers.clear();
112
+ events = [];
113
+ lastEntry = null;
114
+ removeBridge();
115
+ },
116
+ };
117
+ }
118
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"devtoolsBridge.js","sourceRoot":"","sources":["../../../../src/instrumentation/adapters/devtoolsBridge.ts"],"names":[],"mappings":"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","sourcesContent":["import type {\n  InstrumentationAPI,\n  RouterEntrySnapshot,\n  RouterEvent,\n} from '../Instrumentation.js';\n\nconst BRIDGE_KEY = '__PLUMILE_ROUTER__';\nconst DEFAULT_HISTORY_LIMIT = 200;\n\nexport type DevtoolsBridgeOptions = {\n  historyLimit?: number;\n  global?: Record<string, unknown>;\n};\n\nexport type DevtoolsBridge = {\n  getEntry: () => RouterEntrySnapshot | null;\n  getEvents: () => RouterEvent[];\n  subscribeEntries: (\n    callback: (entry: RouterEntrySnapshot) => void,\n  ) => () => void;\n  subscribeEvents: (callback: (event: RouterEvent) => void) => () => void;\n  historyLimit: number;\n};\n\n/** Determines whether a value is a non-null object. */\nfunction isObject(value: unknown): value is Record<string, unknown> {\n  return typeof value === 'object' && value !== null;\n}\n\n/**\n * DevTools-oriented instrumentation that publishes a bridge on the global\n * object (by default `window.__PLUMILE_ROUTER__`). Consumers such as the Chrome\n * extension can subscribe to events or inspect the latest router entry.\n */\nexport function createDevtoolsBridgeInstrumentation(\n  options: DevtoolsBridgeOptions = {},\n): InstrumentationAPI {\n  const historyLimit = options.historyLimit ?? DEFAULT_HISTORY_LIMIT;\n\n  let globalObject: Record<string, unknown> | undefined;\n  if (options.global != null) {\n    globalObject = options.global;\n  } else if (typeof window !== 'undefined') {\n    globalObject = window as unknown as Record<string, unknown>;\n  }\n\n  const eventSubscribers = new Set<(event: RouterEvent) => void>();\n  const entrySubscribers = new Set<(entry: RouterEntrySnapshot) => void>();\n\n  let disposed = false;\n  let bridgeInstalled = false;\n  let lastEntry: RouterEntrySnapshot | null = null;\n  let events: RouterEvent[] = [];\n\n  const bridge: DevtoolsBridge = {\n    getEntry() {\n      return lastEntry;\n    },\n    getEvents() {\n      return events.slice();\n    },\n    subscribeEntries(callback) {\n      entrySubscribers.add(callback);\n      if (lastEntry != null) {\n        callback(lastEntry);\n      }\n      return () => {\n        entrySubscribers.delete(callback);\n      };\n    },\n    subscribeEvents(callback) {\n      eventSubscribers.add(callback);\n      return () => {\n        eventSubscribers.delete(callback);\n      };\n    },\n    historyLimit,\n  };\n\n  /** Publishes the bridge on the chosen global object. */\n  function installBridge(): void {\n    if (bridgeInstalled || globalObject == null) {\n      return;\n    }\n    globalObject[BRIDGE_KEY] = bridge;\n    bridgeInstalled = true;\n  }\n\n  /** Removes the bridge handle from the global object when cleaning up. */\n  function removeBridge(): void {\n    if (!bridgeInstalled || globalObject == null) {\n      return;\n    }\n    if (\n      Object.prototype.hasOwnProperty.call(globalObject, BRIDGE_KEY) &&\n      globalObject[BRIDGE_KEY] === bridge\n    ) {\n      globalObject[BRIDGE_KEY] = undefined;\n    }\n    bridgeInstalled = false;\n  }\n\n  installBridge();\n\n  /** Broadcasts an incoming router event to DevTools subscribers. */\n  function publishEvent(event: RouterEvent): void {\n    events = [...events, event];\n    if (events.length > historyLimit) {\n      events = events.slice(-historyLimit);\n    }\n    eventSubscribers.forEach((subscriber) => {\n      try {\n        subscriber(event);\n      } catch {\n        // Ignore subscriber failures.\n      }\n    });\n  }\n\n  /** Broadcasts the latest router entry snapshot. */\n  function publishEntry(entry: RouterEntrySnapshot): void {\n    lastEntry = entry;\n    entrySubscribers.forEach((subscriber) => {\n      try {\n        subscriber(entry);\n      } catch {\n        // Ignore subscriber failures.\n      }\n    });\n  }\n\n  return {\n    onEvent(event) {\n      if (disposed) {\n        return;\n      }\n      if (!isObject(event)) {\n        return;\n      }\n      publishEvent(event);\n    },\n    onEntryChange(entry) {\n      if (disposed) {\n        return;\n      }\n      if (!isObject(entry)) {\n        return;\n      }\n      publishEntry(entry);\n    },\n    dispose() {\n      if (disposed) {\n        return;\n      }\n      disposed = true;\n      eventSubscribers.clear();\n      entrySubscribers.clear();\n      events = [];\n      lastEntry = null;\n      removeBridge();\n    },\n  };\n}\n"]}
@@ -0,0 +1,10 @@
1
+ import type { InstrumentationAPI } from '../Instrumentation.js';
2
+ export type ConsoleLike = Pick<Console, 'log' | 'info' | 'warn' | 'error'>;
3
+ export type ConsoleLoggerOptions = {
4
+ level?: keyof ConsoleLike;
5
+ label?: string;
6
+ console?: ConsoleLike;
7
+ };
8
+ export declare function createConsoleLoggerInstrumentation(options?: ConsoleLoggerOptions): InstrumentationAPI;
9
+ export type { ConsoleLoggerOptions as LoggerOptions };
10
+ //# sourceMappingURL=logger.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logger.d.ts","sourceRoot":"","sources":["../../../../src/instrumentation/adapters/logger.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EACV,kBAAkB,EAGnB,MAAM,uBAAuB,CAAC;AAE/B,MAAM,MAAM,WAAW,GAAG,IAAI,CAAC,OAAO,EAAE,KAAK,GAAG,MAAM,GAAG,MAAM,GAAG,OAAO,CAAC,CAAC;AAE3E,MAAM,MAAM,oBAAoB,GAAG;IACjC,KAAK,CAAC,EAAE,MAAM,WAAW,CAAC;IAC1B,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,OAAO,CAAC,EAAE,WAAW,CAAC;CACvB,CAAC;AAMF,wBAAgB,kCAAkC,CAChD,OAAO,GAAE,oBAAyB,GACjC,kBAAkB,CA2BpB;AAED,YAAY,EAAE,oBAAoB,IAAI,aAAa,EAAE,CAAC"}
@@ -0,0 +1,19 @@
1
+ export function createConsoleLoggerInstrumentation(options = {}) {
2
+ const { level = 'info', label = 'router', console: targetConsole = console, } = options;
3
+ const log = targetConsole[level];
4
+ function formatEvent(event) {
5
+ return [`[${label}] event:${event.kind}`, event];
6
+ }
7
+ function formatEntry(entry) {
8
+ return [`[${label}] entry`, entry];
9
+ }
10
+ return {
11
+ onEvent(event) {
12
+ log.apply(targetConsole, formatEvent(event));
13
+ },
14
+ onEntryChange(entry) {
15
+ log.apply(targetConsole, formatEntry(entry));
16
+ },
17
+ };
18
+ }
19
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL2luc3RydW1lbnRhdGlvbi9hZGFwdGVycy9sb2dnZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBa0JBLE1BQU0sVUFBVSxrQ0FBa0MsQ0FDaEQsVUFBZ0MsRUFBRTtJQUVsQyxNQUFNLEVBQ0osS0FBSyxHQUFHLE1BQU0sRUFDZCxLQUFLLEdBQUcsUUFBUSxFQUNoQixPQUFPLEVBQUUsYUFBYSxHQUFHLE9BQU8sR0FDakMsR0FBRyxPQUFPLENBQUM7SUFFWixNQUFNLEdBQUcsR0FBRyxhQUFhLENBQUMsS0FBSyxDQUFDLENBQUM7SUFHakMsU0FBUyxXQUFXLENBQUMsS0FBa0I7UUFDckMsT0FBTyxDQUFDLElBQUksS0FBSyxXQUFXLEtBQUssQ0FBQyxJQUFJLEVBQUUsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNuRCxDQUFDO0lBR0QsU0FBUyxXQUFXLENBQUMsS0FBMEI7UUFDN0MsT0FBTyxDQUFDLElBQUksS0FBSyxTQUFTLEVBQUUsS0FBSyxDQUFDLENBQUM7SUFDckMsQ0FBQztJQUVELE9BQU87UUFDTCxPQUFPLENBQUMsS0FBSztZQUNYLEdBQUcsQ0FBQyxLQUFLLENBQUMsYUFBYSxFQUFFLFdBQVcsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDO1FBQy9DLENBQUM7UUFDRCxhQUFhLENBQUMsS0FBSztZQUNqQixHQUFHLENBQUMsS0FBSyxDQUFDLGFBQWEsRUFBRSxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQztRQUMvQyxDQUFDO0tBQ0YsQ0FBQztBQUNKLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7XG4gIEluc3RydW1lbnRhdGlvbkFQSSxcbiAgUm91dGVyRW50cnlTbmFwc2hvdCxcbiAgUm91dGVyRXZlbnQsXG59IGZyb20gJy4uL0luc3RydW1lbnRhdGlvbi5qcyc7XG5cbmV4cG9ydCB0eXBlIENvbnNvbGVMaWtlID0gUGljazxDb25zb2xlLCAnbG9nJyB8ICdpbmZvJyB8ICd3YXJuJyB8ICdlcnJvcic+O1xuXG5leHBvcnQgdHlwZSBDb25zb2xlTG9nZ2VyT3B0aW9ucyA9IHtcbiAgbGV2ZWw/OiBrZXlvZiBDb25zb2xlTGlrZTtcbiAgbGFiZWw/OiBzdHJpbmc7XG4gIGNvbnNvbGU/OiBDb25zb2xlTGlrZTtcbn07XG5cbi8qKlxuICogTGlnaHR3ZWlnaHQgaW5zdHJ1bWVudGF0aW9uIHRoYXQgbG9ncyByb3V0ZXIgZXZlbnRzIHRvIHRoZSBwcm92aWRlZCBjb25zb2xlLlxuICogVXNlZnVsIGR1cmluZyBkZXZlbG9wbWVudCBvciB3aGVuIGNvbXBvc2luZyBjdXN0b20gdG9vbGluZy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZUNvbnNvbGVMb2dnZXJJbnN0cnVtZW50YXRpb24oXG4gIG9wdGlvbnM6IENvbnNvbGVMb2dnZXJPcHRpb25zID0ge30sXG4pOiBJbnN0cnVtZW50YXRpb25BUEkge1xuICBjb25zdCB7XG4gICAgbGV2ZWwgPSAnaW5mbycsXG4gICAgbGFiZWwgPSAncm91dGVyJyxcbiAgICBjb25zb2xlOiB0YXJnZXRDb25zb2xlID0gY29uc29sZSxcbiAgfSA9IG9wdGlvbnM7XG5cbiAgY29uc3QgbG9nID0gdGFyZ2V0Q29uc29sZVtsZXZlbF07XG5cbiAgLyoqIEZvcm1hdHMgYW4gZXZlbnQgZW50cnkgZm9yIGxvZ2dpbmcuICovXG4gIGZ1bmN0aW9uIGZvcm1hdEV2ZW50KGV2ZW50OiBSb3V0ZXJFdmVudCk6IHVua25vd25bXSB7XG4gICAgcmV0dXJuIFtgWyR7bGFiZWx9XSBldmVudDoke2V2ZW50LmtpbmR9YCwgZXZlbnRdO1xuICB9XG5cbiAgLyoqIEZvcm1hdHMgdGhlIGxhdGVzdCByb3V0ZXIgZW50cnkgc25hcHNob3QgZm9yIGxvZ2dpbmcuICovXG4gIGZ1bmN0aW9uIGZvcm1hdEVudHJ5KGVudHJ5OiBSb3V0ZXJFbnRyeVNuYXBzaG90KTogdW5rbm93bltdIHtcbiAgICByZXR1cm4gW2BbJHtsYWJlbH1dIGVudHJ5YCwgZW50cnldO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBvbkV2ZW50KGV2ZW50KSB7XG4gICAgICBsb2cuYXBwbHkodGFyZ2V0Q29uc29sZSwgZm9ybWF0RXZlbnQoZXZlbnQpKTtcbiAgICB9LFxuICAgIG9uRW50cnlDaGFuZ2UoZW50cnkpIHtcbiAgICAgIGxvZy5hcHBseSh0YXJnZXRDb25zb2xlLCBmb3JtYXRFbnRyeShlbnRyeSkpO1xuICAgIH0sXG4gIH07XG59XG5cbmV4cG9ydCB0eXBlIHsgQ29uc29sZUxvZ2dlck9wdGlvbnMgYXMgTG9nZ2VyT3B0aW9ucyB9O1xuIl19
@@ -0,0 +1,4 @@
1
+ export * from './Instrumentation.js';
2
+ export * from './adapters/logger.js';
3
+ export * from './adapters/devtoolsBridge.js';
4
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/instrumentation/index.ts"],"names":[],"mappings":"AAAA,cAAc,sBAAsB,CAAC;AACrC,cAAc,sBAAsB,CAAC;AACrC,cAAc,8BAA8B,CAAC"}
@@ -0,0 +1,4 @@
1
+ export * from './Instrumentation.js';
2
+ export * from './adapters/logger.js';
3
+ export * from './adapters/devtoolsBridge.js';
4
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvaW5zdHJ1bWVudGF0aW9uL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsc0JBQXNCLENBQUM7QUFDckMsY0FBYyxzQkFBc0IsQ0FBQztBQUNyQyxjQUFjLDhCQUE4QixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9JbnN0cnVtZW50YXRpb24uanMnO1xuZXhwb3J0ICogZnJvbSAnLi9hZGFwdGVycy9sb2dnZXIuanMnO1xuZXhwb3J0ICogZnJvbSAnLi9hZGFwdGVycy9kZXZ0b29sc0JyaWRnZS5qcyc7XG4iXX0=
@@ -1 +1 @@
1
- {"version":3,"file":"Link.d.ts","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EACZ,KAAK,yBAAyB,EAE9B,KAAK,SAAS,EAIf,MAAM,OAAO,CAAC;AAEf,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAc3E,wBAAgB,iBAAiB,CAC/B,KAAK,EAAE,OAAO,EACd,OAAO,EAAE,cAAc,GAAG,SAAS,EACnC,QAAQ,EAAE,MAAM,GACf,OAAO,CAmBT;AAKD,KAAK,KAAK,CAAC,MAAM,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI;IAE7E,eAAe,CAAC,EAAE,MAAM,CAAC;IAEzB,QAAQ,EAAE,SAAS,CAAC;IAEpB,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB,mBAAmB,CAAC,EAAE,OAAO,CAAC;IAE9B,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAE7B,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,WAAW,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAEzD,MAAM,CAAC,EAAE,yBAAyB,CAAC;IAEnC,EAAE,CAAC,EAAE,eAAe,GAAG,MAAM,CAAC;IAE9B,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC;AAyBF,QAAA,MAAM,IAAI,0GAsLR,CAAC;AAEH,eAAe,IAAI,CAAC"}
1
+ {"version":3,"file":"Link.d.ts","sourceRoot":"","sources":["../../../src/routing/Link.tsx"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,EACZ,KAAK,yBAAyB,EAE9B,KAAK,SAAS,EAIf,MAAM,OAAO,CAAC;AAEf,OAAO,KAAK,EAAE,cAAc,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AAa3E,wBAAgB,iBAAiB,CAC/B,KAAK,EAAE,OAAO,EACd,OAAO,EAAE,cAAc,GAAG,SAAS,EACnC,QAAQ,EAAE,MAAM,GACf,OAAO,CAmBT;AAKD,KAAK,KAAK,CAAC,MAAM,SAAS,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,IAAI;IAE7E,eAAe,CAAC,EAAE,MAAM,CAAC;IAEzB,QAAQ,EAAE,SAAS,CAAC;IAEpB,SAAS,CAAC,EAAE,MAAM,CAAC;IAEnB,KAAK,CAAC,EAAE,OAAO,CAAC;IAEhB,mBAAmB,CAAC,EAAE,OAAO,CAAC;IAE9B,kBAAkB,CAAC,EAAE,OAAO,CAAC;IAE7B,IAAI,CAAC,EAAE,MAAM,CAAC;IAEd,UAAU,CAAC,EAAE,OAAO,CAAC;IAErB,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,OAAO,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAErD,WAAW,CAAC,EAAE,KAAK,CAAC,iBAAiB,CAAC,iBAAiB,CAAC,CAAC;IAEzD,MAAM,CAAC,EAAE,yBAAyB,CAAC;IAEnC,EAAE,CAAC,EAAE,eAAe,GAAG,MAAM,CAAC;IAE9B,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB,CAAC;AAyBF,QAAA,MAAM,IAAI,0GA6JR,CAAC;AAEH,eAAe,IAAI,CAAC"}