@contractspec/example.analytics-dashboard 1.46.1 → 1.47.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 (67) hide show
  1. package/dist/dashboard/dashboard.enum.d.ts +4 -4
  2. package/dist/dashboard/dashboard.enum.d.ts.map +1 -1
  3. package/dist/dashboard/dashboard.operation.d.ts +131 -131
  4. package/dist/dashboard/dashboard.operation.d.ts.map +1 -1
  5. package/dist/dashboard/dashboard.presentation.d.ts +4 -4
  6. package/dist/dashboard/dashboard.presentation.d.ts.map +1 -1
  7. package/dist/dashboard/dashboard.presentation.js +7 -7
  8. package/dist/dashboard/dashboard.presentation.js.map +1 -1
  9. package/dist/dashboard/dashboard.schema.d.ts +79 -79
  10. package/dist/dashboard/dashboard.test-spec.d.ts +10 -0
  11. package/dist/dashboard/dashboard.test-spec.d.ts.map +1 -0
  12. package/dist/dashboard/dashboard.test-spec.js +233 -0
  13. package/dist/dashboard/dashboard.test-spec.js.map +1 -0
  14. package/dist/dashboard.feature.d.ts +11 -2
  15. package/dist/dashboard.feature.d.ts.map +1 -1
  16. package/dist/dashboard.feature.js +13 -2
  17. package/dist/dashboard.feature.js.map +1 -1
  18. package/dist/events.d.ts +40 -40
  19. package/dist/events.d.ts.map +1 -1
  20. package/dist/events.js +1 -1
  21. package/dist/example.d.ts +2 -2
  22. package/dist/example.d.ts.map +1 -1
  23. package/dist/example.js +4 -2
  24. package/dist/example.js.map +1 -1
  25. package/dist/handlers/analytics.handlers.d.ts +122 -0
  26. package/dist/handlers/analytics.handlers.d.ts.map +1 -0
  27. package/dist/handlers/analytics.handlers.js +310 -0
  28. package/dist/handlers/analytics.handlers.js.map +1 -0
  29. package/dist/handlers/index.d.ts +2 -0
  30. package/dist/handlers/index.js +3 -0
  31. package/dist/index.d.ts +6 -1
  32. package/dist/index.js +6 -1
  33. package/dist/query/query.operation.d.ts +3 -3
  34. package/dist/query/query.operation.d.ts.map +1 -1
  35. package/dist/query/query.presentation.d.ts +3 -3
  36. package/dist/query/query.presentation.d.ts.map +1 -1
  37. package/dist/query/query.presentation.js +5 -5
  38. package/dist/query/query.presentation.js.map +1 -1
  39. package/dist/query/query.test-spec.d.ts +8 -0
  40. package/dist/query/query.test-spec.d.ts.map +1 -0
  41. package/dist/query/query.test-spec.js +124 -0
  42. package/dist/query/query.test-spec.js.map +1 -0
  43. package/dist/query-engine/index.d.ts.map +1 -1
  44. package/dist/query-engine/index.js.map +1 -1
  45. package/dist/seeders/index.d.ts +10 -0
  46. package/dist/seeders/index.d.ts.map +1 -0
  47. package/dist/seeders/index.js +19 -0
  48. package/dist/seeders/index.js.map +1 -0
  49. package/dist/ui/AnalyticsDashboard.d.ts +7 -0
  50. package/dist/ui/AnalyticsDashboard.d.ts.map +1 -0
  51. package/dist/ui/AnalyticsDashboard.js +265 -0
  52. package/dist/ui/AnalyticsDashboard.js.map +1 -0
  53. package/dist/ui/hooks/index.d.ts +2 -0
  54. package/dist/ui/hooks/index.js +5 -0
  55. package/dist/ui/hooks/useAnalyticsData.d.ts +23 -0
  56. package/dist/ui/hooks/useAnalyticsData.d.ts.map +1 -0
  57. package/dist/ui/hooks/useAnalyticsData.js +73 -0
  58. package/dist/ui/hooks/useAnalyticsData.js.map +1 -0
  59. package/dist/ui/index.d.ts +6 -0
  60. package/dist/ui/index.js +6 -0
  61. package/dist/ui/renderers/analytics.markdown.d.ts +28 -0
  62. package/dist/ui/renderers/analytics.markdown.d.ts.map +1 -0
  63. package/dist/ui/renderers/analytics.markdown.js +264 -0
  64. package/dist/ui/renderers/analytics.markdown.js.map +1 -0
  65. package/dist/ui/renderers/index.d.ts +2 -0
  66. package/dist/ui/renderers/index.js +3 -0
  67. package/package.json +24 -8
@@ -0,0 +1,124 @@
1
+ import { defineTestSpec } from "@contractspec/lib.contracts";
2
+
3
+ //#region src/query/query.test-spec.ts
4
+ const CreateQueryTest = defineTestSpec({
5
+ meta: {
6
+ key: "analytics.query.create.test",
7
+ version: "1.0.0",
8
+ title: "Create Query Test",
9
+ description: "Verifies query creation flow",
10
+ owners: ["@example.analytics-dashboard"],
11
+ tags: [
12
+ "analytics",
13
+ "query",
14
+ "test"
15
+ ],
16
+ stability: "stable"
17
+ },
18
+ target: {
19
+ type: "operation",
20
+ operation: {
21
+ key: "analytics.query.create",
22
+ version: "1.0.0"
23
+ }
24
+ },
25
+ scenarios: [{
26
+ key: "success",
27
+ description: "Successfully create a query",
28
+ when: {
29
+ operation: {
30
+ key: "analytics.query.create",
31
+ version: "1.0.0"
32
+ },
33
+ input: {
34
+ name: "Revenue Query",
35
+ sql: "SELECT * FROM revenue"
36
+ }
37
+ },
38
+ then: [{
39
+ type: "expectOutput",
40
+ match: {
41
+ name: "Revenue Query",
42
+ type: "sql"
43
+ }
44
+ }]
45
+ }, {
46
+ key: "error-invalid-sql",
47
+ description: "Fail with invalid SQL",
48
+ when: {
49
+ operation: {
50
+ key: "analytics.query.create",
51
+ version: "1.0.0"
52
+ },
53
+ input: {
54
+ name: "Bad Query",
55
+ sql: ""
56
+ }
57
+ },
58
+ then: [{
59
+ type: "expectError",
60
+ messageIncludes: "VALIDATION_ERROR"
61
+ }]
62
+ }]
63
+ });
64
+ const ExecuteQueryTest = defineTestSpec({
65
+ meta: {
66
+ key: "analytics.query.execute.test",
67
+ version: "1.0.0",
68
+ title: "Execute Query Test",
69
+ description: "Verifies query execution",
70
+ owners: ["@example.analytics-dashboard"],
71
+ tags: [
72
+ "analytics",
73
+ "query",
74
+ "test"
75
+ ],
76
+ stability: "stable"
77
+ },
78
+ target: {
79
+ type: "operation",
80
+ operation: {
81
+ key: "analytics.query.execute",
82
+ version: "1.0.0"
83
+ }
84
+ },
85
+ scenarios: [{
86
+ key: "success",
87
+ description: "Successfully execute query",
88
+ when: {
89
+ operation: {
90
+ key: "analytics.query.execute",
91
+ version: "1.0.0"
92
+ },
93
+ input: {
94
+ queryId: "q-123",
95
+ params: { limit: 10 }
96
+ }
97
+ },
98
+ then: [{
99
+ type: "expectOutput",
100
+ match: { rowCount: 1 }
101
+ }]
102
+ }, {
103
+ key: "error-query-not-found",
104
+ description: "Fail when query not found",
105
+ when: {
106
+ operation: {
107
+ key: "analytics.query.execute",
108
+ version: "1.0.0"
109
+ },
110
+ input: {
111
+ queryId: "q-999",
112
+ params: {}
113
+ }
114
+ },
115
+ then: [{
116
+ type: "expectError",
117
+ messageIncludes: "NOT_FOUND"
118
+ }]
119
+ }]
120
+ });
121
+
122
+ //#endregion
123
+ export { CreateQueryTest, ExecuteQueryTest };
124
+ //# sourceMappingURL=query.test-spec.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"query.test-spec.js","names":[],"sources":["../../src/query/query.test-spec.ts"],"sourcesContent":["import { defineTestSpec } from '@contractspec/lib.contracts';\n\nexport const CreateQueryTest = defineTestSpec({\n meta: {\n key: 'analytics.query.create.test',\n version: '1.0.0',\n title: 'Create Query Test',\n description: 'Verifies query creation flow',\n owners: ['@example.analytics-dashboard'],\n tags: ['analytics', 'query', 'test'],\n stability: 'stable',\n },\n target: {\n type: 'operation',\n operation: { key: 'analytics.query.create', version: '1.0.0' },\n },\n scenarios: [\n {\n key: 'success',\n description: 'Successfully create a query',\n when: {\n operation: { key: 'analytics.query.create', version: '1.0.0' },\n input: {\n name: 'Revenue Query',\n sql: 'SELECT * FROM revenue',\n },\n },\n then: [\n {\n type: 'expectOutput',\n match: {\n name: 'Revenue Query',\n type: 'sql',\n },\n },\n ],\n },\n {\n key: 'error-invalid-sql',\n description: 'Fail with invalid SQL',\n when: {\n operation: { key: 'analytics.query.create', version: '1.0.0' },\n input: {\n name: 'Bad Query',\n sql: '',\n },\n },\n then: [\n {\n type: 'expectError',\n messageIncludes: 'VALIDATION_ERROR',\n },\n ],\n },\n ],\n});\n\nexport const ExecuteQueryTest = defineTestSpec({\n meta: {\n key: 'analytics.query.execute.test',\n version: '1.0.0',\n title: 'Execute Query Test',\n description: 'Verifies query execution',\n owners: ['@example.analytics-dashboard'],\n tags: ['analytics', 'query', 'test'],\n stability: 'stable',\n },\n target: {\n type: 'operation',\n operation: { key: 'analytics.query.execute', version: '1.0.0' },\n },\n scenarios: [\n {\n key: 'success',\n description: 'Successfully execute query',\n when: {\n operation: { key: 'analytics.query.execute', version: '1.0.0' },\n input: {\n queryId: 'q-123',\n params: { limit: 10 },\n },\n },\n then: [\n {\n type: 'expectOutput',\n match: {\n rowCount: 1,\n },\n },\n ],\n },\n {\n key: 'error-query-not-found',\n description: 'Fail when query not found',\n when: {\n operation: { key: 'analytics.query.execute', version: '1.0.0' },\n input: {\n queryId: 'q-999',\n params: {},\n },\n },\n then: [\n {\n type: 'expectError',\n messageIncludes: 'NOT_FOUND',\n },\n ],\n },\n ],\n});\n"],"mappings":";;;AAEA,MAAa,kBAAkB,eAAe;CAC5C,MAAM;EACJ,KAAK;EACL,SAAS;EACT,OAAO;EACP,aAAa;EACb,QAAQ,CAAC,+BAA+B;EACxC,MAAM;GAAC;GAAa;GAAS;GAAO;EACpC,WAAW;EACZ;CACD,QAAQ;EACN,MAAM;EACN,WAAW;GAAE,KAAK;GAA0B,SAAS;GAAS;EAC/D;CACD,WAAW,CACT;EACE,KAAK;EACL,aAAa;EACb,MAAM;GACJ,WAAW;IAAE,KAAK;IAA0B,SAAS;IAAS;GAC9D,OAAO;IACL,MAAM;IACN,KAAK;IACN;GACF;EACD,MAAM,CACJ;GACE,MAAM;GACN,OAAO;IACL,MAAM;IACN,MAAM;IACP;GACF,CACF;EACF,EACD;EACE,KAAK;EACL,aAAa;EACb,MAAM;GACJ,WAAW;IAAE,KAAK;IAA0B,SAAS;IAAS;GAC9D,OAAO;IACL,MAAM;IACN,KAAK;IACN;GACF;EACD,MAAM,CACJ;GACE,MAAM;GACN,iBAAiB;GAClB,CACF;EACF,CACF;CACF,CAAC;AAEF,MAAa,mBAAmB,eAAe;CAC7C,MAAM;EACJ,KAAK;EACL,SAAS;EACT,OAAO;EACP,aAAa;EACb,QAAQ,CAAC,+BAA+B;EACxC,MAAM;GAAC;GAAa;GAAS;GAAO;EACpC,WAAW;EACZ;CACD,QAAQ;EACN,MAAM;EACN,WAAW;GAAE,KAAK;GAA2B,SAAS;GAAS;EAChE;CACD,WAAW,CACT;EACE,KAAK;EACL,aAAa;EACb,MAAM;GACJ,WAAW;IAAE,KAAK;IAA2B,SAAS;IAAS;GAC/D,OAAO;IACL,SAAS;IACT,QAAQ,EAAE,OAAO,IAAI;IACtB;GACF;EACD,MAAM,CACJ;GACE,MAAM;GACN,OAAO,EACL,UAAU,GACX;GACF,CACF;EACF,EACD;EACE,KAAK;EACL,aAAa;EACb,MAAM;GACJ,WAAW;IAAE,KAAK;IAA2B,SAAS;IAAS;GAC/D,OAAO;IACL,SAAS;IACT,QAAQ,EAAE;IACX;GACF;EACD,MAAM,CACJ;GACE,MAAM;GACN,iBAAiB;GAClB,CACF;EACF,CACF;CACF,CAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","names":[],"sources":["../../src/query-engine/index.ts"],"sourcesContent":[],"mappings":";;AAQA;AAQA;;;AAIY,UAZK,eAAA,CAYL;EACA,IAAA,EAAA,KAAA,GAAA,QAAA,GAAA,aAAA,GAAA,QAAA;EAAiB,GAAA,CAAA,EAAA,MAAA;EAIZ,SAAA,CAAA,EAAA,MAAA,EAAiB;EAOjB,WAAA,CAAA,EApBD,qBAoBoB;EAOnB,MAAA,CAAA,EA1BN,qBA0BsB;AAgBjC;AAKiB,UA5CA,qBAAA,CA8CH;EAGG,MAAA,EAAA,MAAA;EAEN,QAAA,EAjDC,iBAiDD,EAAA;EACF,UAAA,EAjDK,mBAiDL,EAAA;EAGG,OAAA,CAAA,EAnDA,gBAmDA,EAAA;EACG,OAAA,CAAA,EAnDH,iBAmDG,EAAA;EAAM,KAAA,CAAA,EAAA,MAAA;AAGrB;AACQ,UAnDS,iBAAA,CAmDT;EACG,IAAA,EAAA,MAAA;EAIE,KAAA,EAAA,MAAA;EAAI,WAAA,EAAA,OAAA,GAAA,KAAA,GAAA,KAAA,GAAA,KAAA,GAAA,KAAA,GAAA,gBAAA;EAIA,MAAA,CAAA,EAAA,MAAA;AASjB;AAEgB,UAhEC,mBAAA,CAgED;EACJ,IAAA,EAAA,MAAA;EACC,KAAA,EAAA,MAAA;EAAR,IAAA,CAAA,EAAA,MAAA,GAAA,QAAA,GAAA,QAAA;EACuB,WAAA,CAAA,EAAA,MAAA,GAAA,KAAA,GAAA,MAAA,GAAA,OAAA,GAAA,MAAA;;AAQX,UApEA,gBAAA,CAoEW;EACA,KAAA,EAAA,MAAA;EAAR,QAAA,EAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,UAAA,GAAA,SAAA;EACO,KAAA,EAAA,OAAA;;AACI,UAvDd,iBAAA,CAuDc;EAAO,KAAA,EAAA,MAAA;EAGzB,SAAA,EAAA,KAAA,GAAA,MAAmB;;AAGN,UAxDT,qBAAA,CAwDS;EAYd,OAAA,EAAA,MAAA;EAEP,UAAA,EApES,MAoET,CAAA,MAAA,EAAA,OAAA,CAAA;;AAjBsC,UAhD1B,eAAA,CAgD0B;EAAW,SAAA,CAAA,EAAA;IAkCzC,KAAA,EAhFF,IAgFE;IAGS,GAAA,EAlFb,IAkFa;IAKN,WAAA,CAAA,EAAA,MAAA;EACJ,CAAA;EACC,OAAA,CAAA,EAtFD,gBAsFC,EAAA;EAAR,UAAA,CAAA,EArFU,MAqFV,CAAA,MAAA,EAAA,OAAA,CAAA;;AAVoC,UAxExB,WAAA,CAwEwB;EAAY,IAAA,EAvE7C,MAuE6C,CAAA,MAAA,EAAA,OAAA,CAAA,EAAA;EA6OrC,OAAA,EAnTL,gBAmTsB,EAAA;;;;aA/SpB;;;UAII,gBAAA;;;;;;UASA,YAAA;sBAED,yBACJ,kBACP,QAAQ;4BACe;;;;;UAQX,WAAA;oBACG,QAAQ;2BACD,kCAAkC;+BAC9B;;cAGlB,kBAAA,YAA8B;;oBAGjB,QAAQ;2BAYtB,kCAEP;+BAKgC;;cAYxB,gBAAA,YAA4B;;sBAGnB;sBAKN,yBACJ,kBACP,QAAQ;4BAgEe;;;;;;;;;;iBAmKZ,iBAAA,SAA0B,cAAc"}
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../src/query-engine/index.ts"],"sourcesContent":[],"mappings":";;AAQA;AAQA;;;AAIY,UAZK,eAAA,CAYL;EACA,IAAA,EAAA,KAAA,GAAA,QAAA,GAAA,aAAA,GAAA,QAAA;EAAiB,GAAA,CAAA,EAAA,MAAA;EAIZ,SAAA,CAAA,EAAA,MAAA,EAAiB;EAOjB,WAAA,CAAA,EApBD,qBAoBoB;EAOnB,MAAA,CAAA,EA1BN,qBA0BsB;AAgBjC;AAKiB,UA5CA,qBAAA,CA4CqB;EAKrB,MAAA,EAAA,MAAA;EAEN,QAAA,EAjDC,iBAiDD,EAAA;EACF,UAAA,EAjDK,mBAiDL,EAAA;EAGG,OAAA,CAAA,EAnDA,gBAmDA,EAAA;EACG,OAAA,CAAA,EAnDH,iBAmDG,EAAA;EAAM,KAAA,CAAA,EAAA,MAAA;AAGrB;AACQ,UAnDS,iBAAA,CAmDT;EACG,IAAA,EAAA,MAAA;EAIE,KAAA,EAAA,MAAA;EAAI,WAAA,EAAA,OAAA,GAAA,KAAA,GAAA,KAAA,GAAA,KAAA,GAAA,KAAA,GAAA,gBAAA;EAIA,MAAA,CAAA,EAAA,MAAA;AASjB;AAEgB,UAhEC,mBAAA,CAgED;EACJ,IAAA,EAAA,MAAA;EACC,KAAA,EAAA,MAAA;EAAR,IAAA,CAAA,EAAA,MAAA,GAAA,QAAA,GAAA,QAAA;EACuB,WAAA,CAAA,EAAA,MAAA,GAAA,KAAA,GAAA,MAAA,GAAA,OAAA,GAAA,MAAA;;AAQX,UApEA,gBAAA,CAoEW;EACA,KAAA,EAAA,MAAA;EAAR,QAAA,EAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,IAAA,GAAA,KAAA,GAAA,UAAA,GAAA,SAAA;EACO,KAAA,EAAA,OAAA;;AACI,UAvDd,iBAAA,CAuDc;EAAO,KAAA,EAAA,MAAA;EAGzB,SAAA,EAAA,KAAA,GAAA,MAAmB;;AAGN,UAxDT,qBAAA,CAwDS;EAYd,OAAA,EAAA,MAAA;EAEP,UAAA,EApES,MAoET,CAAA,MAAA,EAAA,OAAA,CAAA;;AAjBsC,UAhD1B,eAAA,CAgD0B;EAAW,SAAA,CAAA,EAAA;IAkCzC,KAAA,EAhFF,IAgFE;IAGS,GAAA,EAlFb,IAkFa;IAKN,WAAA,CAAA,EAAA,MAAA;EACJ,CAAA;EACC,OAAA,CAAA,EAtFD,gBAsFC,EAAA;EAAR,UAAA,CAAA,EArFU,MAqFV,CAAA,MAAA,EAAA,OAAA,CAAA;;AAVoC,UAxExB,WAAA,CAwEwB;EAAY,IAAA,EAvE7C,MAuE6C,CAAA,MAAA,EAAA,OAAA,CAAA,EAAA;EA6OrC,OAAA,EAnTL,gBAmTsB,EAAA;;;;aA/SpB;;;UAII,gBAAA;;;;;;UASA,YAAA;sBAED,yBACJ,kBACP,QAAQ;4BACe;;;;;UAQX,WAAA;oBACG,QAAQ;2BACD,kCAAkC;+BAC9B;;cAGlB,kBAAA,YAA8B;;oBAGjB,QAAQ;2BAYtB,kCAEP;+BAKgC;;cAYxB,gBAAA,YAA4B;;sBAGnB;sBAKN,yBACJ,kBACP,QAAQ;4BAgEe;;;;;;;;;;iBAmKZ,iBAAA,SAA0B,cAAc"}
@@ -1 +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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"}
1
+ {"version":3,"file":"index.js","names":[],"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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"}
@@ -0,0 +1,10 @@
1
+ import { DatabasePort } from "@contractspec/lib.runtime-sandbox";
2
+
3
+ //#region src/seeders/index.d.ts
4
+ declare function seedAnalyticsDashboard(params: {
5
+ projectId: string;
6
+ db: DatabasePort;
7
+ }): Promise<void>;
8
+ //#endregion
9
+ export { seedAnalyticsDashboard };
10
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","names":[],"sources":["../../src/seeders/index.ts"],"sourcesContent":[],"mappings":";;;iBAEsB,sBAAA;;EAAA,EAAA,EAEhB,YAFgB;IAGrB"}
@@ -0,0 +1,19 @@
1
+ //#region src/seeders/index.ts
2
+ async function seedAnalyticsDashboard(params) {
3
+ const { projectId, db } = params;
4
+ if ((await db.query(`SELECT COUNT(*) as count FROM analytics_dashboard WHERE "projectId" = $1`, [projectId])).rows[0]?.count > 0) return;
5
+ await db.execute(`INSERT INTO analytics_dashboard (id, "projectId", "organizationId", name, slug, description, status)
6
+ VALUES ($1, $2, $3, $4, $5, $6, $7)`, [
7
+ "dash_1",
8
+ projectId,
9
+ "org_demo",
10
+ "Sales Overview",
11
+ "sales-overview",
12
+ "Sales performance dashboard",
13
+ "PUBLISHED"
14
+ ]);
15
+ }
16
+
17
+ //#endregion
18
+ export { seedAnalyticsDashboard };
19
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","names":[],"sources":["../../src/seeders/index.ts"],"sourcesContent":["import type { DatabasePort } from '@contractspec/lib.runtime-sandbox';\n\nexport async function seedAnalyticsDashboard(params: {\n projectId: string;\n db: DatabasePort;\n}) {\n const { projectId, db } = params;\n\n const existing = await db.query(\n `SELECT COUNT(*) as count FROM analytics_dashboard WHERE \"projectId\" = $1`,\n [projectId]\n );\n if ((existing.rows[0]?.count as number) > 0) return;\n\n await db.execute(\n `INSERT INTO analytics_dashboard (id, \"projectId\", \"organizationId\", name, slug, description, status)\n VALUES ($1, $2, $3, $4, $5, $6, $7)`,\n [\n 'dash_1',\n projectId,\n 'org_demo',\n 'Sales Overview',\n 'sales-overview',\n 'Sales performance dashboard',\n 'PUBLISHED',\n ]\n );\n}\n"],"mappings":";AAEA,eAAsB,uBAAuB,QAG1C;CACD,MAAM,EAAE,WAAW,OAAO;AAM1B,MAJiB,MAAM,GAAG,MACxB,4EACA,CAAC,UAAU,CACZ,EACa,KAAK,IAAI,QAAmB,EAAG;AAE7C,OAAM,GAAG,QACP;2CAEA;EACE;EACA;EACA;EACA;EACA;EACA;EACA;EACD,CACF"}
@@ -0,0 +1,7 @@
1
+ import * as react_jsx_runtime0 from "react/jsx-runtime";
2
+
3
+ //#region src/ui/AnalyticsDashboard.d.ts
4
+ declare function AnalyticsDashboard(): react_jsx_runtime0.JSX.Element;
5
+ //#endregion
6
+ export { AnalyticsDashboard };
7
+ //# sourceMappingURL=AnalyticsDashboard.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnalyticsDashboard.d.ts","names":[],"sources":["../../src/ui/AnalyticsDashboard.tsx"],"sourcesContent":[],"mappings":";;;iBAoDgB,kBAAA,CAAA,GAAkB,kBAAA,CAAA,GAAA,CAAA"}
@@ -0,0 +1,265 @@
1
+ 'use client';
2
+
3
+ import { useAnalyticsData } from "./hooks/useAnalyticsData.js";
4
+ import { useState } from "react";
5
+ import { Button, ErrorState, LoaderBlock, StatCard, StatCardGroup } from "@contractspec/lib.design-system";
6
+ import { jsx, jsxs } from "react/jsx-runtime";
7
+
8
+ //#region src/ui/AnalyticsDashboard.tsx
9
+ /**
10
+ * Analytics Dashboard
11
+ *
12
+ * Interactive dashboard for the analytics-dashboard template.
13
+ * Displays dashboards, widgets, and queries.
14
+ */
15
+ const STATUS_COLORS = {
16
+ PUBLISHED: "bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400",
17
+ DRAFT: "bg-gray-100 text-gray-700 dark:bg-gray-900/30 dark:text-gray-400",
18
+ ARCHIVED: "bg-yellow-100 text-yellow-700 dark:bg-yellow-900/30 dark:text-yellow-400"
19
+ };
20
+ const WIDGET_ICONS = {
21
+ LINE_CHART: "📈",
22
+ BAR_CHART: "📊",
23
+ PIE_CHART: "🥧",
24
+ AREA_CHART: "📉",
25
+ SCATTER_PLOT: "⚬",
26
+ METRIC: "🔢",
27
+ TABLE: "📋",
28
+ HEATMAP: "🗺️",
29
+ FUNNEL: "⏬",
30
+ MAP: "🌍",
31
+ TEXT: "📝",
32
+ EMBED: "🔗"
33
+ };
34
+ const QUERY_TYPE_COLORS = {
35
+ SQL: "bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-400",
36
+ METRIC: "bg-purple-100 text-purple-700 dark:bg-purple-900/30 dark:text-purple-400",
37
+ AGGREGATION: "bg-indigo-100 text-indigo-700 dark:bg-indigo-900/30 dark:text-indigo-400",
38
+ CUSTOM: "bg-gray-100 text-gray-700 dark:bg-gray-900/30 dark:text-gray-400"
39
+ };
40
+ function AnalyticsDashboard() {
41
+ const [activeTab, setActiveTab] = useState("dashboards");
42
+ const { dashboards, queries, selectedDashboard, widgets, loading, error, stats, refetch, selectDashboard } = useAnalyticsData();
43
+ const tabs = [{
44
+ id: "dashboards",
45
+ label: "Dashboards",
46
+ icon: "📊"
47
+ }, {
48
+ id: "queries",
49
+ label: "Queries",
50
+ icon: "🔍"
51
+ }];
52
+ if (loading) return /* @__PURE__ */ jsx(LoaderBlock, { label: "Loading Analytics..." });
53
+ if (error) return /* @__PURE__ */ jsx(ErrorState, {
54
+ title: "Failed to load Analytics",
55
+ description: error.message,
56
+ onRetry: refetch,
57
+ retryLabel: "Retry"
58
+ });
59
+ return /* @__PURE__ */ jsxs("div", {
60
+ className: "space-y-6",
61
+ children: [
62
+ /* @__PURE__ */ jsxs("div", {
63
+ className: "flex items-center justify-between",
64
+ children: [/* @__PURE__ */ jsx("h2", {
65
+ className: "text-2xl font-bold",
66
+ children: "Analytics Dashboard"
67
+ }), /* @__PURE__ */ jsxs(Button, {
68
+ onClick: () => alert("Create dashboard modal"),
69
+ children: [/* @__PURE__ */ jsx("span", {
70
+ className: "mr-2",
71
+ children: "+"
72
+ }), " New Dashboard"]
73
+ })]
74
+ }),
75
+ /* @__PURE__ */ jsxs(StatCardGroup, { children: [
76
+ /* @__PURE__ */ jsx(StatCard, {
77
+ label: "Dashboards",
78
+ value: stats.totalDashboards,
79
+ hint: `${stats.publishedDashboards} published`
80
+ }),
81
+ /* @__PURE__ */ jsx(StatCard, {
82
+ label: "Queries",
83
+ value: stats.totalQueries,
84
+ hint: `${stats.sharedQueries} shared`
85
+ }),
86
+ /* @__PURE__ */ jsx(StatCard, {
87
+ label: "Widgets",
88
+ value: widgets.length,
89
+ hint: "on current dashboard"
90
+ })
91
+ ] }),
92
+ /* @__PURE__ */ jsx("nav", {
93
+ className: "bg-muted flex gap-1 rounded-lg p-1",
94
+ role: "tablist",
95
+ children: tabs.map((tab) => /* @__PURE__ */ jsxs(Button, {
96
+ type: "button",
97
+ role: "tab",
98
+ "aria-selected": activeTab === tab.id,
99
+ onClick: () => setActiveTab(tab.id),
100
+ className: `flex flex-1 items-center justify-center gap-2 rounded-md px-4 py-2 text-sm font-medium transition-colors ${activeTab === tab.id ? "bg-background text-foreground shadow-sm" : "text-muted-foreground hover:text-foreground"}`,
101
+ children: [/* @__PURE__ */ jsx("span", { children: tab.icon }), tab.label]
102
+ }, tab.id))
103
+ }),
104
+ /* @__PURE__ */ jsxs("div", {
105
+ className: "min-h-[400px]",
106
+ role: "tabpanel",
107
+ children: [activeTab === "dashboards" && /* @__PURE__ */ jsxs("div", {
108
+ className: "space-y-6",
109
+ children: [/* @__PURE__ */ jsxs("div", {
110
+ className: "grid gap-4 sm:grid-cols-2 lg:grid-cols-3",
111
+ children: [dashboards.map((dashboard) => /* @__PURE__ */ jsxs("div", {
112
+ onClick: () => selectDashboard(dashboard),
113
+ className: `border-border bg-card cursor-pointer rounded-lg border p-4 transition-all ${selectedDashboard?.id === dashboard.id ? "ring-primary ring-2" : "hover:bg-muted/50"}`,
114
+ role: "button",
115
+ tabIndex: 0,
116
+ onKeyDown: (e) => {
117
+ if (e.key === "Enter" || e.key === " ") selectDashboard(dashboard);
118
+ },
119
+ children: [
120
+ /* @__PURE__ */ jsxs("div", {
121
+ className: "mb-2 flex items-center justify-between",
122
+ children: [/* @__PURE__ */ jsx("h3", {
123
+ className: "font-medium",
124
+ children: dashboard.name
125
+ }), /* @__PURE__ */ jsx("span", {
126
+ className: `inline-flex rounded-full px-2 py-0.5 text-xs font-medium ${STATUS_COLORS[dashboard.status] ?? ""}`,
127
+ children: dashboard.status
128
+ })]
129
+ }),
130
+ /* @__PURE__ */ jsx("p", {
131
+ className: "text-muted-foreground mb-3 text-sm",
132
+ children: dashboard.description
133
+ }),
134
+ /* @__PURE__ */ jsxs("div", {
135
+ className: "text-muted-foreground flex items-center justify-between text-xs",
136
+ children: [/* @__PURE__ */ jsxs("span", { children: ["/", dashboard.slug] }), dashboard.isPublic && /* @__PURE__ */ jsx("span", {
137
+ className: "text-green-600 dark:text-green-400",
138
+ children: "🌐 Public"
139
+ })]
140
+ })
141
+ ]
142
+ }, dashboard.id)), dashboards.length === 0 && /* @__PURE__ */ jsx("div", {
143
+ className: "text-muted-foreground col-span-full flex h-64 items-center justify-center",
144
+ children: "No dashboards created yet"
145
+ })]
146
+ }), selectedDashboard && widgets.length > 0 && /* @__PURE__ */ jsxs("div", { children: [/* @__PURE__ */ jsxs("h3", {
147
+ className: "mb-4 text-lg font-semibold",
148
+ children: [
149
+ "Widgets in \"",
150
+ selectedDashboard.name,
151
+ "\""
152
+ ]
153
+ }), /* @__PURE__ */ jsx("div", {
154
+ className: "grid gap-4 sm:grid-cols-2 lg:grid-cols-3",
155
+ children: widgets.map((widget) => /* @__PURE__ */ jsxs("div", {
156
+ className: "border-border bg-card rounded-lg border p-4",
157
+ children: [
158
+ /* @__PURE__ */ jsxs("div", {
159
+ className: "mb-2 flex items-center gap-2",
160
+ children: [/* @__PURE__ */ jsx("span", {
161
+ className: "text-xl",
162
+ children: WIDGET_ICONS[widget.type] ?? "📊"
163
+ }), /* @__PURE__ */ jsx("span", {
164
+ className: "font-medium",
165
+ children: widget.name
166
+ })]
167
+ }),
168
+ /* @__PURE__ */ jsx("div", {
169
+ className: "text-muted-foreground text-sm",
170
+ children: widget.type.replace(/_/g, " ")
171
+ }),
172
+ /* @__PURE__ */ jsxs("div", {
173
+ className: "text-muted-foreground mt-2 text-xs",
174
+ children: [
175
+ "Position: (",
176
+ widget.gridX,
177
+ ", ",
178
+ widget.gridY,
179
+ ") •",
180
+ " ",
181
+ widget.gridWidth,
182
+ "x",
183
+ widget.gridHeight
184
+ ]
185
+ })
186
+ ]
187
+ }, widget.id))
188
+ })] })]
189
+ }), activeTab === "queries" && /* @__PURE__ */ jsx("div", {
190
+ className: "border-border rounded-lg border",
191
+ children: /* @__PURE__ */ jsxs("table", {
192
+ className: "w-full",
193
+ children: [/* @__PURE__ */ jsx("thead", {
194
+ className: "border-border bg-muted/30 border-b",
195
+ children: /* @__PURE__ */ jsxs("tr", { children: [
196
+ /* @__PURE__ */ jsx("th", {
197
+ className: "px-4 py-3 text-left text-sm font-medium",
198
+ children: "Query"
199
+ }),
200
+ /* @__PURE__ */ jsx("th", {
201
+ className: "px-4 py-3 text-left text-sm font-medium",
202
+ children: "Type"
203
+ }),
204
+ /* @__PURE__ */ jsx("th", {
205
+ className: "px-4 py-3 text-left text-sm font-medium",
206
+ children: "Cache TTL"
207
+ }),
208
+ /* @__PURE__ */ jsx("th", {
209
+ className: "px-4 py-3 text-left text-sm font-medium",
210
+ children: "Shared"
211
+ })
212
+ ] })
213
+ }), /* @__PURE__ */ jsxs("tbody", {
214
+ className: "divide-border divide-y",
215
+ children: [queries.map((query) => /* @__PURE__ */ jsxs("tr", {
216
+ className: "hover:bg-muted/50",
217
+ children: [
218
+ /* @__PURE__ */ jsxs("td", {
219
+ className: "px-4 py-3",
220
+ children: [/* @__PURE__ */ jsx("div", {
221
+ className: "font-medium",
222
+ children: query.name
223
+ }), /* @__PURE__ */ jsx("div", {
224
+ className: "text-muted-foreground text-sm",
225
+ children: query.description
226
+ })]
227
+ }),
228
+ /* @__PURE__ */ jsx("td", {
229
+ className: "px-4 py-3",
230
+ children: /* @__PURE__ */ jsx("span", {
231
+ className: `inline-flex rounded-full px-2 py-0.5 text-xs font-medium ${QUERY_TYPE_COLORS[query.type] ?? ""}`,
232
+ children: query.type
233
+ })
234
+ }),
235
+ /* @__PURE__ */ jsxs("td", {
236
+ className: "text-muted-foreground px-4 py-3 text-sm",
237
+ children: [query.cacheTtlSeconds, "s"]
238
+ }),
239
+ /* @__PURE__ */ jsx("td", {
240
+ className: "px-4 py-3",
241
+ children: query.isShared ? /* @__PURE__ */ jsx("span", {
242
+ className: "text-green-600 dark:text-green-400",
243
+ children: "✓"
244
+ }) : /* @__PURE__ */ jsx("span", {
245
+ className: "text-muted-foreground",
246
+ children: "—"
247
+ })
248
+ })
249
+ ]
250
+ }, query.id)), queries.length === 0 && /* @__PURE__ */ jsx("tr", { children: /* @__PURE__ */ jsx("td", {
251
+ colSpan: 4,
252
+ className: "text-muted-foreground px-4 py-8 text-center",
253
+ children: "No queries saved"
254
+ }) })]
255
+ })]
256
+ })
257
+ })]
258
+ })
259
+ ]
260
+ });
261
+ }
262
+
263
+ //#endregion
264
+ export { AnalyticsDashboard };
265
+ //# sourceMappingURL=AnalyticsDashboard.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AnalyticsDashboard.js","names":[],"sources":["../../src/ui/AnalyticsDashboard.tsx"],"sourcesContent":["'use client';\n\n/**\n * Analytics Dashboard\n *\n * Interactive dashboard for the analytics-dashboard template.\n * Displays dashboards, widgets, and queries.\n */\nimport { useState } from 'react';\nimport {\n Button,\n ErrorState,\n LoaderBlock,\n StatCard,\n StatCardGroup,\n} from '@contractspec/lib.design-system';\nimport { useAnalyticsData } from './hooks/useAnalyticsData';\n\ntype Tab = 'dashboards' | 'queries';\n\nconst STATUS_COLORS: Record<string, string> = {\n PUBLISHED:\n 'bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-400',\n DRAFT: 'bg-gray-100 text-gray-700 dark:bg-gray-900/30 dark:text-gray-400',\n ARCHIVED:\n 'bg-yellow-100 text-yellow-700 dark:bg-yellow-900/30 dark:text-yellow-400',\n};\n\nconst WIDGET_ICONS: Record<string, string> = {\n LINE_CHART: '📈',\n BAR_CHART: '📊',\n PIE_CHART: '🥧',\n AREA_CHART: '📉',\n SCATTER_PLOT: '⚬',\n METRIC: '🔢',\n TABLE: '📋',\n HEATMAP: '🗺️',\n FUNNEL: '⏬',\n MAP: '🌍',\n TEXT: '📝',\n EMBED: '🔗',\n};\n\nconst QUERY_TYPE_COLORS: Record<string, string> = {\n SQL: 'bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-400',\n METRIC:\n 'bg-purple-100 text-purple-700 dark:bg-purple-900/30 dark:text-purple-400',\n AGGREGATION:\n 'bg-indigo-100 text-indigo-700 dark:bg-indigo-900/30 dark:text-indigo-400',\n CUSTOM: 'bg-gray-100 text-gray-700 dark:bg-gray-900/30 dark:text-gray-400',\n};\n\nexport function AnalyticsDashboard() {\n const [activeTab, setActiveTab] = useState<Tab>('dashboards');\n const {\n dashboards,\n queries,\n selectedDashboard,\n widgets,\n loading,\n error,\n stats,\n refetch,\n selectDashboard,\n } = useAnalyticsData();\n\n const tabs: { id: Tab; label: string; icon: string }[] = [\n { id: 'dashboards', label: 'Dashboards', icon: '📊' },\n { id: 'queries', label: 'Queries', icon: '🔍' },\n ];\n\n if (loading) {\n return <LoaderBlock label=\"Loading Analytics...\" />;\n }\n\n if (error) {\n return (\n <ErrorState\n title=\"Failed to load Analytics\"\n description={error.message}\n onRetry={refetch}\n retryLabel=\"Retry\"\n />\n );\n }\n\n return (\n <div className=\"space-y-6\">\n {/* Header */}\n <div className=\"flex items-center justify-between\">\n <h2 className=\"text-2xl font-bold\">Analytics Dashboard</h2>\n <Button onClick={() => alert('Create dashboard modal')}>\n <span className=\"mr-2\">+</span> New Dashboard\n </Button>\n </div>\n\n {/* Stats Row */}\n <StatCardGroup>\n <StatCard\n label=\"Dashboards\"\n value={stats.totalDashboards}\n hint={`${stats.publishedDashboards} published`}\n />\n <StatCard\n label=\"Queries\"\n value={stats.totalQueries}\n hint={`${stats.sharedQueries} shared`}\n />\n <StatCard\n label=\"Widgets\"\n value={widgets.length}\n hint=\"on current dashboard\"\n />\n </StatCardGroup>\n\n {/* Navigation Tabs */}\n <nav className=\"bg-muted flex gap-1 rounded-lg p-1\" role=\"tablist\">\n {tabs.map((tab) => (\n <Button\n key={tab.id}\n type=\"button\"\n role=\"tab\"\n aria-selected={activeTab === tab.id}\n onClick={() => setActiveTab(tab.id)}\n className={`flex flex-1 items-center justify-center gap-2 rounded-md px-4 py-2 text-sm font-medium transition-colors ${\n activeTab === tab.id\n ? 'bg-background text-foreground shadow-sm'\n : 'text-muted-foreground hover:text-foreground'\n }`}\n >\n <span>{tab.icon}</span>\n {tab.label}\n </Button>\n ))}\n </nav>\n\n {/* Tab Content */}\n <div className=\"min-h-[400px]\" role=\"tabpanel\">\n {activeTab === 'dashboards' && (\n <div className=\"space-y-6\">\n {/* Dashboard List */}\n <div className=\"grid gap-4 sm:grid-cols-2 lg:grid-cols-3\">\n {dashboards.map((dashboard) => (\n <div\n key={dashboard.id}\n onClick={() => selectDashboard(dashboard)}\n className={`border-border bg-card cursor-pointer rounded-lg border p-4 transition-all ${\n selectedDashboard?.id === dashboard.id\n ? 'ring-primary ring-2'\n : 'hover:bg-muted/50'\n }`}\n role=\"button\"\n tabIndex={0}\n onKeyDown={(e) => {\n if (e.key === 'Enter' || e.key === ' ')\n selectDashboard(dashboard);\n }}\n >\n <div className=\"mb-2 flex items-center justify-between\">\n <h3 className=\"font-medium\">{dashboard.name}</h3>\n <span\n className={`inline-flex rounded-full px-2 py-0.5 text-xs font-medium ${STATUS_COLORS[dashboard.status] ?? ''}`}\n >\n {dashboard.status}\n </span>\n </div>\n <p className=\"text-muted-foreground mb-3 text-sm\">\n {dashboard.description}\n </p>\n <div className=\"text-muted-foreground flex items-center justify-between text-xs\">\n <span>/{dashboard.slug}</span>\n {dashboard.isPublic && (\n <span className=\"text-green-600 dark:text-green-400\">\n 🌐 Public\n </span>\n )}\n </div>\n </div>\n ))}\n {dashboards.length === 0 && (\n <div className=\"text-muted-foreground col-span-full flex h-64 items-center justify-center\">\n No dashboards created yet\n </div>\n )}\n </div>\n\n {/* Widget Grid for Selected Dashboard */}\n {selectedDashboard && widgets.length > 0 && (\n <div>\n <h3 className=\"mb-4 text-lg font-semibold\">\n Widgets in \"{selectedDashboard.name}\"\n </h3>\n <div className=\"grid gap-4 sm:grid-cols-2 lg:grid-cols-3\">\n {widgets.map((widget) => (\n <div\n key={widget.id}\n className=\"border-border bg-card rounded-lg border p-4\"\n >\n <div className=\"mb-2 flex items-center gap-2\">\n <span className=\"text-xl\">\n {WIDGET_ICONS[widget.type] ?? '📊'}\n </span>\n <span className=\"font-medium\">{widget.name}</span>\n </div>\n <div className=\"text-muted-foreground text-sm\">\n {widget.type.replace(/_/g, ' ')}\n </div>\n <div className=\"text-muted-foreground mt-2 text-xs\">\n Position: ({widget.gridX}, {widget.gridY}) •{' '}\n {widget.gridWidth}x{widget.gridHeight}\n </div>\n </div>\n ))}\n </div>\n </div>\n )}\n </div>\n )}\n\n {activeTab === 'queries' && (\n <div className=\"border-border rounded-lg border\">\n <table className=\"w-full\">\n <thead className=\"border-border bg-muted/30 border-b\">\n <tr>\n <th className=\"px-4 py-3 text-left text-sm font-medium\">\n Query\n </th>\n <th className=\"px-4 py-3 text-left text-sm font-medium\">\n Type\n </th>\n <th className=\"px-4 py-3 text-left text-sm font-medium\">\n Cache TTL\n </th>\n <th className=\"px-4 py-3 text-left text-sm font-medium\">\n Shared\n </th>\n </tr>\n </thead>\n <tbody className=\"divide-border divide-y\">\n {queries.map((query) => (\n <tr key={query.id} className=\"hover:bg-muted/50\">\n <td className=\"px-4 py-3\">\n <div className=\"font-medium\">{query.name}</div>\n <div className=\"text-muted-foreground text-sm\">\n {query.description}\n </div>\n </td>\n <td className=\"px-4 py-3\">\n <span\n className={`inline-flex rounded-full px-2 py-0.5 text-xs font-medium ${QUERY_TYPE_COLORS[query.type] ?? ''}`}\n >\n {query.type}\n </span>\n </td>\n <td className=\"text-muted-foreground px-4 py-3 text-sm\">\n {query.cacheTtlSeconds}s\n </td>\n <td className=\"px-4 py-3\">\n {query.isShared ? (\n <span className=\"text-green-600 dark:text-green-400\">\n ✓\n </span>\n ) : (\n <span className=\"text-muted-foreground\">—</span>\n )}\n </td>\n </tr>\n ))}\n {queries.length === 0 && (\n <tr>\n <td\n colSpan={4}\n className=\"text-muted-foreground px-4 py-8 text-center\"\n >\n No queries saved\n </td>\n </tr>\n )}\n </tbody>\n </table>\n </div>\n )}\n </div>\n </div>\n );\n}\n"],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { AnalyticsStats, useAnalyticsData } from "./useAnalyticsData.js";
2
+ export { type AnalyticsStats, useAnalyticsData };
@@ -0,0 +1,5 @@
1
+ 'use client';
2
+
3
+ import { useAnalyticsData } from "./useAnalyticsData.js";
4
+
5
+ export { useAnalyticsData };
@@ -0,0 +1,23 @@
1
+ import { Dashboard, Query, Widget } from "../../handlers/analytics.handlers.js";
2
+
3
+ //#region src/ui/hooks/useAnalyticsData.d.ts
4
+ interface AnalyticsStats {
5
+ totalDashboards: number;
6
+ publishedDashboards: number;
7
+ totalQueries: number;
8
+ sharedQueries: number;
9
+ }
10
+ declare function useAnalyticsData(projectId?: string): {
11
+ dashboards: Dashboard[];
12
+ queries: Query[];
13
+ selectedDashboard: Dashboard | null;
14
+ widgets: Widget[];
15
+ loading: boolean;
16
+ error: Error | null;
17
+ stats: AnalyticsStats;
18
+ refetch: () => Promise<void>;
19
+ selectDashboard: (dashboard: Dashboard) => Promise<void>;
20
+ };
21
+ //#endregion
22
+ export { AnalyticsStats, useAnalyticsData };
23
+ //# sourceMappingURL=useAnalyticsData.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useAnalyticsData.d.ts","names":[],"sources":["../../../src/ui/hooks/useAnalyticsData.ts"],"sourcesContent":[],"mappings":";;;UAWiB,cAAA;;EAAA,mBAAc,EAAA,MAAA;EAOf,YAAA,EAAA,MAAgB;;;iBAAhB,gBAAA;;;;;EAgDM,OAAA,EAAA,OAAA;EAAS,KAAA,OAAA,GAAA,IAAA;EAAA,KAAA,gBAAA;;+BAAT,cAAS"}