@contractspec/example.analytics-dashboard 1.44.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (62) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +281 -0
  3. package/dist/dashboard/dashboard.enum.d.ts +18 -0
  4. package/dist/dashboard/dashboard.enum.d.ts.map +1 -0
  5. package/dist/dashboard/dashboard.enum.js +43 -0
  6. package/dist/dashboard/dashboard.enum.js.map +1 -0
  7. package/dist/dashboard/dashboard.operation.d.ts +537 -0
  8. package/dist/dashboard/dashboard.operation.d.ts.map +1 -0
  9. package/dist/dashboard/dashboard.operation.js +213 -0
  10. package/dist/dashboard/dashboard.operation.js.map +1 -0
  11. package/dist/dashboard/dashboard.presentation.d.ts +9 -0
  12. package/dist/dashboard/dashboard.presentation.d.ts.map +1 -0
  13. package/dist/dashboard/dashboard.presentation.js +90 -0
  14. package/dist/dashboard/dashboard.presentation.js.map +1 -0
  15. package/dist/dashboard/dashboard.schema.d.ts +333 -0
  16. package/dist/dashboard/dashboard.schema.d.ts.map +1 -0
  17. package/dist/dashboard/dashboard.schema.js +236 -0
  18. package/dist/dashboard/dashboard.schema.js.map +1 -0
  19. package/dist/dashboard/index.d.ts +4 -0
  20. package/dist/dashboard/index.js +5 -0
  21. package/dist/dashboard.feature.d.ts +8 -0
  22. package/dist/dashboard.feature.d.ts.map +1 -0
  23. package/dist/dashboard.feature.js +166 -0
  24. package/dist/dashboard.feature.js.map +1 -0
  25. package/dist/docs/analytics-dashboard.docblock.d.ts +1 -0
  26. package/dist/docs/analytics-dashboard.docblock.js +114 -0
  27. package/dist/docs/analytics-dashboard.docblock.js.map +1 -0
  28. package/dist/docs/index.d.ts +1 -0
  29. package/dist/docs/index.js +1 -0
  30. package/dist/events.d.ts +149 -0
  31. package/dist/events.d.ts.map +1 -0
  32. package/dist/events.js +128 -0
  33. package/dist/events.js.map +1 -0
  34. package/dist/example.d.ts +40 -0
  35. package/dist/example.d.ts.map +1 -0
  36. package/dist/example.js +51 -0
  37. package/dist/example.js.map +1 -0
  38. package/dist/index.d.ts +7 -0
  39. package/dist/index.js +8 -0
  40. package/dist/query/index.d.ts +4 -0
  41. package/dist/query/index.js +5 -0
  42. package/dist/query/query.enum.d.ts +10 -0
  43. package/dist/query/query.enum.d.ts.map +1 -0
  44. package/dist/query/query.enum.js +16 -0
  45. package/dist/query/query.enum.js.map +1 -0
  46. package/dist/query/query.operation.d.ts +181 -0
  47. package/dist/query/query.operation.d.ts.map +1 -0
  48. package/dist/query/query.operation.js +113 -0
  49. package/dist/query/query.operation.js.map +1 -0
  50. package/dist/query/query.presentation.d.ts +8 -0
  51. package/dist/query/query.presentation.d.ts.map +1 -0
  52. package/dist/query/query.presentation.js +60 -0
  53. package/dist/query/query.presentation.js.map +1 -0
  54. package/dist/query/query.schema.d.ts +143 -0
  55. package/dist/query/query.schema.d.ts.map +1 -0
  56. package/dist/query/query.schema.js +157 -0
  57. package/dist/query/query.schema.js.map +1 -0
  58. package/dist/query-engine/index.d.ts +106 -0
  59. package/dist/query-engine/index.d.ts.map +1 -0
  60. package/dist/query-engine/index.js +189 -0
  61. package/dist/query-engine/index.js.map +1 -0
  62. package/package.json +91 -0
@@ -0,0 +1,189 @@
1
+ //#region src/query-engine/index.ts
2
+ var InMemoryQueryCache = class {
3
+ cache = /* @__PURE__ */ new Map();
4
+ async get(key) {
5
+ const entry = this.cache.get(key);
6
+ if (!entry) return null;
7
+ if (entry.expiresAt < /* @__PURE__ */ new Date()) {
8
+ this.cache.delete(key);
9
+ return null;
10
+ }
11
+ return {
12
+ ...entry.result,
13
+ cached: true,
14
+ cachedAt: entry.expiresAt
15
+ };
16
+ }
17
+ async set(key, result, ttlSeconds) {
18
+ const expiresAt = new Date(Date.now() + ttlSeconds * 1e3);
19
+ this.cache.set(key, {
20
+ result,
21
+ expiresAt
22
+ });
23
+ }
24
+ async invalidate(pattern) {
25
+ const regex = new RegExp(pattern);
26
+ for (const key of this.cache.keys()) if (regex.test(key)) this.cache.delete(key);
27
+ }
28
+ };
29
+ var BasicQueryEngine = class {
30
+ cache;
31
+ constructor(cache) {
32
+ this.cache = cache ?? new InMemoryQueryCache();
33
+ }
34
+ async execute(definition, params) {
35
+ const startTime = Date.now();
36
+ const validation = this.validateQuery(definition);
37
+ if (!validation.valid) return {
38
+ data: [],
39
+ columns: [],
40
+ rowCount: 0,
41
+ executionTimeMs: Date.now() - startTime,
42
+ cached: false,
43
+ error: validation.errors.join(", ")
44
+ };
45
+ const cacheKey = this.buildCacheKey(definition, params);
46
+ const cachedResult = await this.cache.get(cacheKey);
47
+ if (cachedResult) return cachedResult;
48
+ let result;
49
+ switch (definition.type) {
50
+ case "AGGREGATION":
51
+ if (!definition.aggregation) throw new Error("Aggregation definition is missing");
52
+ result = await this.executeAggregation(definition.aggregation, params);
53
+ break;
54
+ case "METRIC":
55
+ if (!definition.metricIds) throw new Error("Metric IDs are missing");
56
+ result = await this.executeMetric(definition.metricIds, params);
57
+ break;
58
+ case "SQL":
59
+ if (!definition.sql) throw new Error("SQL query is missing");
60
+ result = await this.executeSql(definition.sql, params);
61
+ break;
62
+ default: result = {
63
+ data: [],
64
+ columns: [],
65
+ rowCount: 0,
66
+ executionTimeMs: Date.now() - startTime,
67
+ cached: false,
68
+ error: `Unknown query type: ${definition.type}`
69
+ };
70
+ }
71
+ result.executionTimeMs = Date.now() - startTime;
72
+ result.cached = false;
73
+ await this.cache.set(cacheKey, result, 300);
74
+ return result;
75
+ }
76
+ validateQuery(definition) {
77
+ const errors = [];
78
+ if (!definition.type) errors.push("Query type is required");
79
+ switch (definition.type) {
80
+ case "SQL":
81
+ if (!definition.sql) errors.push("SQL query is required for SQL type");
82
+ break;
83
+ case "METRIC":
84
+ if (!definition.metricIds || definition.metricIds.length === 0) errors.push("Metric IDs are required for METRIC type");
85
+ break;
86
+ case "AGGREGATION":
87
+ if (!definition.aggregation) errors.push("Aggregation definition is required for AGGREGATION type");
88
+ else {
89
+ if (!definition.aggregation.source) errors.push("Aggregation source is required");
90
+ if (!definition.aggregation.measures || definition.aggregation.measures.length === 0) errors.push("At least one measure is required");
91
+ }
92
+ break;
93
+ }
94
+ return {
95
+ valid: errors.length === 0,
96
+ errors
97
+ };
98
+ }
99
+ buildCacheKey(definition, params) {
100
+ return JSON.stringify({
101
+ definition,
102
+ params
103
+ });
104
+ }
105
+ async executeAggregation(aggregation, params) {
106
+ const columns = [...aggregation.dimensions.map((d) => ({
107
+ name: d.name,
108
+ type: d.type === "NUMBER" ? "NUMBER" : d.type === "TIME" ? "DATE" : "STRING",
109
+ label: d.name
110
+ })), ...aggregation.measures.map((m) => ({
111
+ name: m.name,
112
+ type: "NUMBER",
113
+ label: m.name,
114
+ format: m.format
115
+ }))];
116
+ const data = this.generateMockData(aggregation, params);
117
+ return {
118
+ data,
119
+ columns,
120
+ rowCount: data.length,
121
+ executionTimeMs: 0,
122
+ cached: false
123
+ };
124
+ }
125
+ async executeMetric(metricIds, _params) {
126
+ const data = metricIds.map((id) => ({
127
+ metricId: id,
128
+ value: Math.random() * 1e3,
129
+ change: (Math.random() - .5) * 20
130
+ }));
131
+ return {
132
+ data,
133
+ columns: [
134
+ {
135
+ name: "metricId",
136
+ type: "STRING"
137
+ },
138
+ {
139
+ name: "value",
140
+ type: "NUMBER"
141
+ },
142
+ {
143
+ name: "change",
144
+ type: "NUMBER"
145
+ }
146
+ ],
147
+ rowCount: data.length,
148
+ executionTimeMs: 0,
149
+ cached: false
150
+ };
151
+ }
152
+ async executeSql(_sql, _params) {
153
+ return {
154
+ data: [],
155
+ columns: [],
156
+ rowCount: 0,
157
+ executionTimeMs: 0,
158
+ cached: false,
159
+ error: "SQL execution not implemented in demo"
160
+ };
161
+ }
162
+ generateMockData(aggregation, params) {
163
+ const data = [];
164
+ const rowCount = 10;
165
+ const timeDimension = aggregation.dimensions.find((d) => d.type === "TIME");
166
+ for (let i = 0; i < rowCount; i++) {
167
+ const row = {};
168
+ for (const dim of aggregation.dimensions) if (dim.type === "TIME") {
169
+ const date = new Date(params.dateRange?.start ?? /* @__PURE__ */ new Date());
170
+ date.setDate(date.getDate() + i);
171
+ row[dim.name] = date.toISOString().split("T")[0];
172
+ } else row[dim.name] = `${dim.name}_${i % 5}`;
173
+ for (const measure of aggregation.measures) {
174
+ const baseValue = timeDimension ? 100 + i * 10 : Math.random() * 1e3;
175
+ const noise = (Math.random() - .5) * 20;
176
+ row[measure.name] = Math.round((baseValue + noise) * 100) / 100;
177
+ }
178
+ data.push(row);
179
+ }
180
+ return data;
181
+ }
182
+ };
183
+ function createQueryEngine(cache) {
184
+ return new BasicQueryEngine(cache);
185
+ }
186
+
187
+ //#endregion
188
+ export { BasicQueryEngine, InMemoryQueryCache, createQueryEngine };
189
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":["result: QueryResult","errors: string[]","columns: ColumnDefinition[]","data: Record<string, unknown>[]","row: Record<string, unknown>"],"sources":["../../src/query-engine/index.ts"],"sourcesContent":["/**\n * Analytics Query Engine\n *\n * Provides query execution and caching for analytics dashboards.\n */\n\n// ============ Types ============\n\nexport interface QueryDefinition {\n type: 'SQL' | 'METRIC' | 'AGGREGATION' | 'CUSTOM';\n sql?: string;\n metricIds?: string[];\n aggregation?: AggregationDefinition;\n custom?: CustomQueryDefinition;\n}\n\nexport interface AggregationDefinition {\n source: string;\n measures: MeasureDefinition[];\n dimensions: DimensionDefinition[];\n filters?: FilterDefinition[];\n orderBy?: OrderByDefinition[];\n limit?: number;\n}\n\nexport interface MeasureDefinition {\n name: string;\n field: string;\n aggregation: 'COUNT' | 'SUM' | 'AVG' | 'MIN' | 'MAX' | 'COUNT_DISTINCT';\n format?: string;\n}\n\nexport interface DimensionDefinition {\n name: string;\n field: string;\n type?: 'TIME' | 'STRING' | 'NUMBER';\n granularity?: 'HOUR' | 'DAY' | 'WEEK' | 'MONTH' | 'YEAR';\n}\n\nexport interface FilterDefinition {\n field: string;\n operator:\n | 'eq'\n | 'neq'\n | 'gt'\n | 'gte'\n | 'lt'\n | 'lte'\n | 'in'\n | 'nin'\n | 'contains'\n | 'between';\n value: unknown;\n}\n\nexport interface OrderByDefinition {\n field: string;\n direction: 'ASC' | 'DESC';\n}\n\nexport interface CustomQueryDefinition {\n handler: string;\n parameters: Record<string, unknown>;\n}\n\nexport interface QueryParameters {\n dateRange?: {\n start: Date;\n end: Date;\n granularity?: string;\n };\n filters?: FilterDefinition[];\n parameters?: Record<string, unknown>;\n}\n\nexport interface QueryResult {\n data: Record<string, unknown>[];\n columns: ColumnDefinition[];\n rowCount: number;\n executionTimeMs: number;\n cached: boolean;\n cachedAt?: Date;\n error?: string;\n}\n\nexport interface ColumnDefinition {\n name: string;\n type: 'STRING' | 'NUMBER' | 'DATE' | 'BOOLEAN';\n label?: string;\n format?: string;\n}\n\n// ============ Query Engine Interface ============\n\nexport interface IQueryEngine {\n execute(\n definition: QueryDefinition,\n params: QueryParameters\n ): Promise<QueryResult>;\n validateQuery(definition: QueryDefinition): {\n valid: boolean;\n errors: string[];\n };\n}\n\n// ============ Query Cache ============\n\nexport interface IQueryCache {\n get(key: string): Promise<QueryResult | null>;\n set(key: string, result: QueryResult, ttlSeconds: number): Promise<void>;\n invalidate(pattern: string): Promise<void>;\n}\n\nexport class InMemoryQueryCache implements IQueryCache {\n private cache = new Map<string, { result: QueryResult; expiresAt: Date }>();\n\n async get(key: string): Promise<QueryResult | null> {\n const entry = this.cache.get(key);\n if (!entry) return null;\n if (entry.expiresAt < new Date()) {\n this.cache.delete(key);\n return null;\n }\n return { ...entry.result, cached: true, cachedAt: entry.expiresAt };\n }\n\n async set(\n key: string,\n result: QueryResult,\n ttlSeconds: number\n ): Promise<void> {\n const expiresAt = new Date(Date.now() + ttlSeconds * 1000);\n this.cache.set(key, { result, expiresAt });\n }\n\n async invalidate(pattern: string): Promise<void> {\n const regex = new RegExp(pattern);\n for (const key of this.cache.keys()) {\n if (regex.test(key)) {\n this.cache.delete(key);\n }\n }\n }\n}\n\n// ============ Basic Query Engine ============\n\nexport class BasicQueryEngine implements IQueryEngine {\n private cache: IQueryCache;\n\n constructor(cache?: IQueryCache) {\n this.cache = cache ?? new InMemoryQueryCache();\n }\n\n async execute(\n definition: QueryDefinition,\n params: QueryParameters\n ): Promise<QueryResult> {\n const startTime = Date.now();\n\n // Validate query\n const validation = this.validateQuery(definition);\n if (!validation.valid) {\n return {\n data: [],\n columns: [],\n rowCount: 0,\n executionTimeMs: Date.now() - startTime,\n cached: false,\n error: validation.errors.join(', '),\n };\n }\n\n // Check cache\n const cacheKey = this.buildCacheKey(definition, params);\n const cachedResult = await this.cache.get(cacheKey);\n if (cachedResult) {\n return cachedResult;\n }\n\n // Execute query based on type\n let result: QueryResult;\n switch (definition.type) {\n case 'AGGREGATION':\n if (!definition.aggregation) {\n throw new Error('Aggregation definition is missing');\n }\n result = await this.executeAggregation(definition.aggregation, params);\n break;\n case 'METRIC':\n if (!definition.metricIds) {\n throw new Error('Metric IDs are missing');\n }\n result = await this.executeMetric(definition.metricIds, params);\n break;\n case 'SQL':\n if (!definition.sql) {\n throw new Error('SQL query is missing');\n }\n result = await this.executeSql(definition.sql, params);\n break;\n default:\n result = {\n data: [],\n columns: [],\n rowCount: 0,\n executionTimeMs: Date.now() - startTime,\n cached: false,\n error: `Unknown query type: ${definition.type}`,\n };\n }\n\n result.executionTimeMs = Date.now() - startTime;\n result.cached = false;\n\n // Cache result\n await this.cache.set(cacheKey, result, 300);\n\n return result;\n }\n\n validateQuery(definition: QueryDefinition): {\n valid: boolean;\n errors: string[];\n } {\n const errors: string[] = [];\n\n if (!definition.type) {\n errors.push('Query type is required');\n }\n\n switch (definition.type) {\n case 'SQL':\n if (!definition.sql) {\n errors.push('SQL query is required for SQL type');\n }\n break;\n case 'METRIC':\n if (!definition.metricIds || definition.metricIds.length === 0) {\n errors.push('Metric IDs are required for METRIC type');\n }\n break;\n case 'AGGREGATION':\n if (!definition.aggregation) {\n errors.push(\n 'Aggregation definition is required for AGGREGATION type'\n );\n } else {\n if (!definition.aggregation.source) {\n errors.push('Aggregation source is required');\n }\n if (\n !definition.aggregation.measures ||\n definition.aggregation.measures.length === 0\n ) {\n errors.push('At least one measure is required');\n }\n }\n break;\n }\n\n return { valid: errors.length === 0, errors };\n }\n\n private buildCacheKey(\n definition: QueryDefinition,\n params: QueryParameters\n ): string {\n return JSON.stringify({ definition, params });\n }\n\n private async executeAggregation(\n aggregation: AggregationDefinition,\n params: QueryParameters\n ): Promise<QueryResult> {\n // In production, this would execute against a data warehouse\n // For demo, return mock data\n const columns: ColumnDefinition[] = [\n ...aggregation.dimensions.map((d) => ({\n name: d.name,\n type: (d.type === 'NUMBER'\n ? 'NUMBER'\n : d.type === 'TIME'\n ? 'DATE'\n : 'STRING') as ColumnDefinition['type'],\n label: d.name,\n })),\n ...aggregation.measures.map((m) => ({\n name: m.name,\n type: 'NUMBER' as const,\n label: m.name,\n format: m.format,\n })),\n ];\n\n // Mock data generation\n const data = this.generateMockData(aggregation, params);\n\n return {\n data,\n columns,\n rowCount: data.length,\n executionTimeMs: 0,\n cached: false,\n };\n }\n\n private async executeMetric(\n metricIds: string[],\n _params: QueryParameters\n ): Promise<QueryResult> {\n // In production, this would fetch from metering service\n const data = metricIds.map((id) => ({\n metricId: id,\n value: Math.random() * 1000,\n change: (Math.random() - 0.5) * 20,\n }));\n\n return {\n data,\n columns: [\n { name: 'metricId', type: 'STRING' },\n { name: 'value', type: 'NUMBER' },\n { name: 'change', type: 'NUMBER' },\n ],\n rowCount: data.length,\n executionTimeMs: 0,\n cached: false,\n };\n }\n\n private async executeSql(\n _sql: string,\n _params: QueryParameters\n ): Promise<QueryResult> {\n // In production, this would execute SQL against a database\n return {\n data: [],\n columns: [],\n rowCount: 0,\n executionTimeMs: 0,\n cached: false,\n error: 'SQL execution not implemented in demo',\n };\n }\n\n private generateMockData(\n aggregation: AggregationDefinition,\n params: QueryParameters\n ): Record<string, unknown>[] {\n const data: Record<string, unknown>[] = [];\n const rowCount = 10;\n\n // Generate time series data if there's a time dimension\n const timeDimension = aggregation.dimensions.find((d) => d.type === 'TIME');\n\n for (let i = 0; i < rowCount; i++) {\n const row: Record<string, unknown> = {};\n\n for (const dim of aggregation.dimensions) {\n if (dim.type === 'TIME') {\n const date = new Date(params.dateRange?.start ?? new Date());\n date.setDate(date.getDate() + i);\n row[dim.name] = date.toISOString().split('T')[0];\n } else {\n row[dim.name] = `${dim.name}_${i % 5}`;\n }\n }\n\n for (const measure of aggregation.measures) {\n const baseValue = timeDimension ? 100 + i * 10 : Math.random() * 1000;\n const noise = (Math.random() - 0.5) * 20;\n row[measure.name] = Math.round((baseValue + noise) * 100) / 100;\n }\n\n data.push(row);\n }\n\n return data;\n }\n}\n\n// ============ Factory ============\n\nexport function createQueryEngine(cache?: IQueryCache): IQueryEngine {\n return new BasicQueryEngine(cache);\n}\n"],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,91 @@
1
+ {
2
+ "name": "@contractspec/example.analytics-dashboard",
3
+ "version": "1.44.0",
4
+ "description": "Analytics Dashboard example with widgets and query engine for ContractSpec",
5
+ "main": "./dist/index.js",
6
+ "types": "./dist/index.d.ts",
7
+ "type": "module",
8
+ "scripts": {
9
+ "publish:pkg": "bun publish --tolerate-republish --ignore-scripts --verbose",
10
+ "publish:pkg:canary": "bun publish:pkg --tag canary",
11
+ "build": "bun build:types && bun build:bundle",
12
+ "build:bundle": "tsdown",
13
+ "build:types": "tsc --noEmit",
14
+ "dev": "bun build:bundle --watch",
15
+ "clean": "rimraf dist .turbo",
16
+ "lint": "bun lint:fix",
17
+ "lint:fix": "eslint src --fix",
18
+ "lint:check": "eslint src"
19
+ },
20
+ "dependencies": {
21
+ "@contractspec/lib.schema": "1.44.0",
22
+ "@contractspec/lib.contracts": "1.44.0",
23
+ "@contractspec/lib.bus": "1.44.0",
24
+ "@contractspec/lib.identity-rbac": "1.44.0",
25
+ "@contractspec/lib.metering": "1.44.0",
26
+ "@contractspec/lib.jobs": "1.44.0",
27
+ "@contractspec/module.audit-trail": "1.44.0",
28
+ "zod": "^4.1.13"
29
+ },
30
+ "devDependencies": {
31
+ "@contractspec/tool.typescript": "1.44.0",
32
+ "@contractspec/tool.tsdown": "1.44.0",
33
+ "typescript": "^5.9.3"
34
+ },
35
+ "exports": {
36
+ ".": "./dist/index.js",
37
+ "./dashboard": "./dist/dashboard/index.js",
38
+ "./dashboard.feature": "./dist/dashboard.feature.js",
39
+ "./dashboard/dashboard.enum": "./dist/dashboard/dashboard.enum.js",
40
+ "./dashboard/dashboard.operation": "./dist/dashboard/dashboard.operation.js",
41
+ "./dashboard/dashboard.presentation": "./dist/dashboard/dashboard.presentation.js",
42
+ "./dashboard/dashboard.schema": "./dist/dashboard/dashboard.schema.js",
43
+ "./docs": "./dist/docs/index.js",
44
+ "./docs/analytics-dashboard.docblock": "./dist/docs/analytics-dashboard.docblock.js",
45
+ "./events": "./dist/events.js",
46
+ "./example": "./dist/example.js",
47
+ "./query": "./dist/query/index.js",
48
+ "./query-engine": "./dist/query-engine/index.js",
49
+ "./query/query.enum": "./dist/query/query.enum.js",
50
+ "./query/query.operation": "./dist/query/query.operation.js",
51
+ "./query/query.presentation": "./dist/query/query.presentation.js",
52
+ "./query/query.schema": "./dist/query/query.schema.js",
53
+ "./*": "./*"
54
+ },
55
+ "module": "./dist/index.js",
56
+ "files": [
57
+ "dist",
58
+ "README.md"
59
+ ],
60
+ "publishConfig": {
61
+ "access": "public",
62
+ "exports": {
63
+ ".": "./dist/index.js",
64
+ "./dashboard": "./dist/dashboard/index.js",
65
+ "./dashboard.feature": "./dist/dashboard.feature.js",
66
+ "./dashboard/dashboard.contracts": "./dist/dashboard/dashboard.operations.js",
67
+ "./dashboard/dashboard.enum": "./dist/dashboard/dashboard.enum.js",
68
+ "./dashboard/dashboard.presentation": "./dist/dashboard/dashboard.presentation.js",
69
+ "./dashboard/dashboard.schema": "./dist/dashboard/dashboard.schema.js",
70
+ "./docs": "./dist/docs/index.js",
71
+ "./docs/analytics-dashboard.docblock": "./dist/docs/analytics-dashboard.docblock.js",
72
+ "./events": "./dist/events.js",
73
+ "./example": "./dist/example.js",
74
+ "./query": "./dist/query/index.js",
75
+ "./query-engine": "./dist/query-engine/index.js",
76
+ "./query/query.contracts": "./dist/query/query.operations.js",
77
+ "./query/query.enum": "./dist/query/query.enum.js",
78
+ "./query/query.presentation": "./dist/query/query.presentation.js",
79
+ "./query/query.schema": "./dist/query/query.schema.js",
80
+ "./*": "./*"
81
+ },
82
+ "registry": "https://registry.npmjs.org/"
83
+ },
84
+ "license": "MIT",
85
+ "repository": {
86
+ "type": "git",
87
+ "url": "https://github.com/lssm-tech/contractspec.git",
88
+ "directory": "packages/examples/analytics-dashboard"
89
+ },
90
+ "homepage": "https://contractspec.io"
91
+ }