@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.
- package/CHANGELOG.md +7 -0
- package/README.md +42 -21
- package/lib/esm/index.d.ts +1 -0
- package/lib/esm/index.d.ts.map +1 -1
- package/lib/esm/index.js +2 -1
- package/lib/esm/instrumentation/Instrumentation.d.ts +66 -0
- package/lib/esm/instrumentation/Instrumentation.d.ts.map +1 -0
- package/lib/esm/instrumentation/Instrumentation.js +59 -0
- package/lib/esm/instrumentation/adapters/devtoolsBridge.d.ts +14 -0
- package/lib/esm/instrumentation/adapters/devtoolsBridge.d.ts.map +1 -0
- package/lib/esm/instrumentation/adapters/devtoolsBridge.js +118 -0
- package/lib/esm/instrumentation/adapters/logger.d.ts +10 -0
- package/lib/esm/instrumentation/adapters/logger.d.ts.map +1 -0
- package/lib/esm/instrumentation/adapters/logger.js +19 -0
- package/lib/esm/instrumentation/index.d.ts +4 -0
- package/lib/esm/instrumentation/index.d.ts.map +1 -0
- package/lib/esm/instrumentation/index.js +4 -0
- package/lib/esm/routing/Link.d.ts.map +1 -1
- package/lib/esm/routing/Link.js +8 -17
- package/lib/esm/routing/createRouter.d.ts +2 -1
- package/lib/esm/routing/createRouter.d.ts.map +1 -1
- package/lib/esm/routing/createRouter.js +184 -261
- package/lib/esm/tools.d.ts +6 -5
- package/lib/esm/tools.d.ts.map +1 -1
- package/lib/esm/tools.js +17 -34
- package/lib/esm/types.d.ts +7 -49
- package/lib/esm/types.d.ts.map +1 -1
- package/lib/esm/types.js +2 -28
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/lib/types/index.d.ts +1 -0
- package/lib/types/index.d.ts.map +1 -1
- package/lib/types/instrumentation/Instrumentation.d.ts +66 -0
- package/lib/types/instrumentation/Instrumentation.d.ts.map +1 -0
- package/lib/types/instrumentation/adapters/devtoolsBridge.d.ts +14 -0
- package/lib/types/instrumentation/adapters/devtoolsBridge.d.ts.map +1 -0
- package/lib/types/instrumentation/adapters/logger.d.ts +10 -0
- package/lib/types/instrumentation/adapters/logger.d.ts.map +1 -0
- package/lib/types/instrumentation/index.d.ts +4 -0
- package/lib/types/instrumentation/index.d.ts.map +1 -0
- package/lib/types/routing/Link.d.ts.map +1 -1
- package/lib/types/routing/createRouter.d.ts +2 -1
- package/lib/types/routing/createRouter.d.ts.map +1 -1
- package/lib/types/tools.d.ts +6 -5
- package/lib/types/tools.d.ts.map +1 -1
- package/lib/types/types.d.ts +7 -49
- package/lib/types/types.d.ts.map +1 -1
- package/package.json +33 -18
package/CHANGELOG.md
ADDED
package/README.md
CHANGED
|
@@ -21,10 +21,11 @@ npm install @plumile/router
|
|
|
21
21
|
|
|
22
22
|
## Peer Dependencies
|
|
23
23
|
|
|
24
|
-
This package
|
|
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
|
|
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: [
|
|
41
|
-
- Relay integration walkthrough: [
|
|
42
|
-
- Examples: [
|
|
43
|
-
- Migration strategies: [
|
|
44
|
-
- DevTools extension usage: [
|
|
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
|
|
134
|
-
|
|
|
135
|
-
| `useNavigate()`
|
|
136
|
-
| `useFilters(schema)`
|
|
137
|
-
| `useQuery()`
|
|
138
|
-
| `useQueryState(key,
|
|
139
|
-
| `useFilterDiagnostics()`
|
|
140
|
-
| `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 &
|
|
370
|
+
#### Inspection & Instrumentation
|
|
370
371
|
|
|
371
|
-
Activez
|
|
372
|
+
Activez l’instrumentation en développement pour exposer un bridge DevTools :
|
|
372
373
|
|
|
373
374
|
```ts
|
|
374
|
-
|
|
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
|
-
|
|
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 (
|
package/lib/esm/index.d.ts
CHANGED
package/lib/esm/index.d.ts.map
CHANGED
|
@@ -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
|
-
|
|
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 @@
|
|
|
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;
|
|
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"}
|