@memberjunction/ng-dashboards 5.30.0 → 5.31.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 (424) hide show
  1. package/dist/AI/components/agents/agent-configuration.component.d.ts.map +1 -1
  2. package/dist/AI/components/agents/agent-configuration.component.js +4 -4
  3. package/dist/AI/components/agents/agent-configuration.component.js.map +1 -1
  4. package/dist/AI/components/agents/agent-editor.component.d.ts +2 -1
  5. package/dist/AI/components/agents/agent-editor.component.d.ts.map +1 -1
  6. package/dist/AI/components/agents/agent-editor.component.js +8 -6
  7. package/dist/AI/components/agents/agent-editor.component.js.map +1 -1
  8. package/dist/AI/components/analytics/agent-runs/agent-run-analysis.component.d.ts +2 -1
  9. package/dist/AI/components/analytics/agent-runs/agent-run-analysis.component.d.ts.map +1 -1
  10. package/dist/AI/components/analytics/agent-runs/agent-run-analysis.component.js +6 -5
  11. package/dist/AI/components/analytics/agent-runs/agent-run-analysis.component.js.map +1 -1
  12. package/dist/AI/components/analytics/cost-budget/cost-budget.component.d.ts +2 -1
  13. package/dist/AI/components/analytics/cost-budget/cost-budget.component.d.ts.map +1 -1
  14. package/dist/AI/components/analytics/cost-budget/cost-budget.component.js +6 -5
  15. package/dist/AI/components/analytics/cost-budget/cost-budget.component.js.map +1 -1
  16. package/dist/AI/components/analytics/error-analysis/error-analysis.component.d.ts +2 -1
  17. package/dist/AI/components/analytics/error-analysis/error-analysis.component.d.ts.map +1 -1
  18. package/dist/AI/components/analytics/error-analysis/error-analysis.component.js +6 -5
  19. package/dist/AI/components/analytics/error-analysis/error-analysis.component.js.map +1 -1
  20. package/dist/AI/components/analytics/executive-summary/executive-summary.component.d.ts +2 -1
  21. package/dist/AI/components/analytics/executive-summary/executive-summary.component.d.ts.map +1 -1
  22. package/dist/AI/components/analytics/executive-summary/executive-summary.component.js +7 -4
  23. package/dist/AI/components/analytics/executive-summary/executive-summary.component.js.map +1 -1
  24. package/dist/AI/components/analytics/model-performance/model-performance.component.d.ts +3 -2
  25. package/dist/AI/components/analytics/model-performance/model-performance.component.d.ts.map +1 -1
  26. package/dist/AI/components/analytics/model-performance/model-performance.component.js +10 -6
  27. package/dist/AI/components/analytics/model-performance/model-performance.component.js.map +1 -1
  28. package/dist/AI/components/analytics/prompt-runs/prompt-run-analysis.component.d.ts +2 -1
  29. package/dist/AI/components/analytics/prompt-runs/prompt-run-analysis.component.d.ts.map +1 -1
  30. package/dist/AI/components/analytics/prompt-runs/prompt-run-analysis.component.js +6 -5
  31. package/dist/AI/components/analytics/prompt-runs/prompt-run-analysis.component.js.map +1 -1
  32. package/dist/AI/components/analytics/usage-patterns/usage-patterns.component.d.ts +2 -1
  33. package/dist/AI/components/analytics/usage-patterns/usage-patterns.component.d.ts.map +1 -1
  34. package/dist/AI/components/analytics/usage-patterns/usage-patterns.component.js +6 -5
  35. package/dist/AI/components/analytics/usage-patterns/usage-patterns.component.js.map +1 -1
  36. package/dist/AI/components/autotagging/autotagging-pipeline-resource.component.d.ts.map +1 -1
  37. package/dist/AI/components/autotagging/autotagging-pipeline-resource.component.js +55 -52
  38. package/dist/AI/components/autotagging/autotagging-pipeline-resource.component.js.map +1 -1
  39. package/dist/AI/components/duplicates/duplicate-detection-resource.component.js +15 -15
  40. package/dist/AI/components/duplicates/duplicate-detection-resource.component.js.map +1 -1
  41. package/dist/AI/components/execution-monitoring.component.d.ts.map +1 -1
  42. package/dist/AI/components/execution-monitoring.component.js +4 -3
  43. package/dist/AI/components/execution-monitoring.component.js.map +1 -1
  44. package/dist/AI/components/models/model-management.component.js +2 -2
  45. package/dist/AI/components/models/model-management.component.js.map +1 -1
  46. package/dist/AI/components/prompts/model-prompt-priority-matrix.component.d.ts +2 -1
  47. package/dist/AI/components/prompts/model-prompt-priority-matrix.component.d.ts.map +1 -1
  48. package/dist/AI/components/prompts/model-prompt-priority-matrix.component.js +10 -8
  49. package/dist/AI/components/prompts/model-prompt-priority-matrix.component.js.map +1 -1
  50. package/dist/AI/components/prompts/prompt-management.component.d.ts.map +1 -1
  51. package/dist/AI/components/prompts/prompt-management.component.js +2 -2
  52. package/dist/AI/components/prompts/prompt-management.component.js.map +1 -1
  53. package/dist/AI/components/prompts/prompt-version-control.component.d.ts +2 -1
  54. package/dist/AI/components/prompts/prompt-version-control.component.d.ts.map +1 -1
  55. package/dist/AI/components/prompts/prompt-version-control.component.js +13 -10
  56. package/dist/AI/components/prompts/prompt-version-control.component.js.map +1 -1
  57. package/dist/AI/components/requests/agent-requests-resource.component.js +1 -1
  58. package/dist/AI/components/requests/agent-requests-resource.component.js.map +1 -1
  59. package/dist/AI/components/tags/tags-resource.component.d.ts +1181 -0
  60. package/dist/AI/components/tags/tags-resource.component.d.ts.map +1 -0
  61. package/dist/AI/components/tags/tags-resource.component.js +7487 -0
  62. package/dist/AI/components/tags/tags-resource.component.js.map +1 -0
  63. package/dist/AI/components/vectors/vector-management-resource.component.d.ts.map +1 -1
  64. package/dist/AI/components/vectors/vector-management-resource.component.js +25 -21
  65. package/dist/AI/components/vectors/vector-management-resource.component.js.map +1 -1
  66. package/dist/AI/index.d.ts +1 -0
  67. package/dist/AI/index.d.ts.map +1 -1
  68. package/dist/AI/index.js +5 -0
  69. package/dist/AI/index.js.map +1 -1
  70. package/dist/AI/services/ai-instrumentation.service.d.ts +5 -0
  71. package/dist/AI/services/ai-instrumentation.service.d.ts.map +1 -1
  72. package/dist/AI/services/ai-instrumentation.service.js +12 -4
  73. package/dist/AI/services/ai-instrumentation.service.js.map +1 -1
  74. package/dist/APIKeys/api-applications-panel.component.d.ts +3 -2
  75. package/dist/APIKeys/api-applications-panel.component.d.ts.map +1 -1
  76. package/dist/APIKeys/api-applications-panel.component.js +5 -4
  77. package/dist/APIKeys/api-applications-panel.component.js.map +1 -1
  78. package/dist/APIKeys/api-key-create-dialog.component.d.ts +2 -1
  79. package/dist/APIKeys/api-key-create-dialog.component.d.ts.map +1 -1
  80. package/dist/APIKeys/api-key-create-dialog.component.js +5 -5
  81. package/dist/APIKeys/api-key-create-dialog.component.js.map +1 -1
  82. package/dist/APIKeys/api-key-edit-panel.component.d.ts +3 -2
  83. package/dist/APIKeys/api-key-edit-panel.component.d.ts.map +1 -1
  84. package/dist/APIKeys/api-key-edit-panel.component.js +11 -10
  85. package/dist/APIKeys/api-key-edit-panel.component.js.map +1 -1
  86. package/dist/APIKeys/api-key-list.component.d.ts +2 -1
  87. package/dist/APIKeys/api-key-list.component.d.ts.map +1 -1
  88. package/dist/APIKeys/api-key-list.component.js +6 -5
  89. package/dist/APIKeys/api-key-list.component.js.map +1 -1
  90. package/dist/APIKeys/api-keys-resource.component.d.ts.map +1 -1
  91. package/dist/APIKeys/api-keys-resource.component.js +4 -4
  92. package/dist/APIKeys/api-keys-resource.component.js.map +1 -1
  93. package/dist/APIKeys/api-scopes-panel.component.d.ts +3 -2
  94. package/dist/APIKeys/api-scopes-panel.component.d.ts.map +1 -1
  95. package/dist/APIKeys/api-scopes-panel.component.js +8 -6
  96. package/dist/APIKeys/api-scopes-panel.component.js.map +1 -1
  97. package/dist/APIKeys/api-usage-panel.component.d.ts +3 -2
  98. package/dist/APIKeys/api-usage-panel.component.d.ts.map +1 -1
  99. package/dist/APIKeys/api-usage-panel.component.js +10 -8
  100. package/dist/APIKeys/api-usage-panel.component.js.map +1 -1
  101. package/dist/Actions/components/actions-list-view.component.d.ts +2 -1
  102. package/dist/Actions/components/actions-list-view.component.d.ts.map +1 -1
  103. package/dist/Actions/components/actions-list-view.component.js +6 -5
  104. package/dist/Actions/components/actions-list-view.component.js.map +1 -1
  105. package/dist/Actions/components/actions-overview.component.js +2 -2
  106. package/dist/Actions/components/actions-overview.component.js.map +1 -1
  107. package/dist/Actions/components/categories-list-view.component.d.ts +2 -1
  108. package/dist/Actions/components/categories-list-view.component.d.ts.map +1 -1
  109. package/dist/Actions/components/categories-list-view.component.js +6 -5
  110. package/dist/Actions/components/categories-list-view.component.js.map +1 -1
  111. package/dist/Actions/components/execution-monitoring.component.js +1 -1
  112. package/dist/Actions/components/execution-monitoring.component.js.map +1 -1
  113. package/dist/Actions/components/explorer/action-card.component.d.ts +2 -1
  114. package/dist/Actions/components/explorer/action-card.component.d.ts.map +1 -1
  115. package/dist/Actions/components/explorer/action-card.component.js +6 -4
  116. package/dist/Actions/components/explorer/action-card.component.js.map +1 -1
  117. package/dist/Actions/components/explorer/action-explorer.component.js +1 -1
  118. package/dist/Actions/components/explorer/action-explorer.component.js.map +1 -1
  119. package/dist/Actions/components/explorer/new-action-panel.component.d.ts +2 -1
  120. package/dist/Actions/components/explorer/new-action-panel.component.d.ts.map +1 -1
  121. package/dist/Actions/components/explorer/new-action-panel.component.js +7 -5
  122. package/dist/Actions/components/explorer/new-action-panel.component.js.map +1 -1
  123. package/dist/Actions/components/explorer/new-category-panel.component.d.ts +2 -1
  124. package/dist/Actions/components/explorer/new-category-panel.component.d.ts.map +1 -1
  125. package/dist/Actions/components/explorer/new-category-panel.component.js +7 -5
  126. package/dist/Actions/components/explorer/new-category-panel.component.js.map +1 -1
  127. package/dist/ApplicationRoles/application-roles-resource.component.js +4 -4
  128. package/dist/ApplicationRoles/application-roles-resource.component.js.map +1 -1
  129. package/dist/Communication/communication-logs-resource.component.js +1 -1
  130. package/dist/Communication/communication-logs-resource.component.js.map +1 -1
  131. package/dist/Communication/communication-monitor-resource.component.js +1 -1
  132. package/dist/Communication/communication-monitor-resource.component.js.map +1 -1
  133. package/dist/Communication/communication-providers-resource.component.js +3 -3
  134. package/dist/Communication/communication-providers-resource.component.js.map +1 -1
  135. package/dist/Communication/communication-runs-resource.component.js +1 -1
  136. package/dist/Communication/communication-runs-resource.component.js.map +1 -1
  137. package/dist/Communication/communication-templates-resource.component.js +3 -3
  138. package/dist/Communication/communication-templates-resource.component.js.map +1 -1
  139. package/dist/ComponentStudio/component-studio-dashboard.component.d.ts +1 -1
  140. package/dist/ComponentStudio/component-studio-dashboard.component.d.ts.map +1 -1
  141. package/dist/ComponentStudio/component-studio-dashboard.component.js +5 -4
  142. package/dist/ComponentStudio/component-studio-dashboard.component.js.map +1 -1
  143. package/dist/ComponentStudio/components/ai-assistant/ai-assistant-panel.component.d.ts +2 -1
  144. package/dist/ComponentStudio/components/ai-assistant/ai-assistant-panel.component.d.ts.map +1 -1
  145. package/dist/ComponentStudio/components/ai-assistant/ai-assistant-panel.component.js +6 -4
  146. package/dist/ComponentStudio/components/ai-assistant/ai-assistant-panel.component.js.map +1 -1
  147. package/dist/ComponentStudio/components/artifact-load-dialog.component.d.ts +11 -2
  148. package/dist/ComponentStudio/components/artifact-load-dialog.component.d.ts.map +1 -1
  149. package/dist/ComponentStudio/components/artifact-load-dialog.component.js +61 -27
  150. package/dist/ComponentStudio/components/artifact-load-dialog.component.js.map +1 -1
  151. package/dist/ComponentStudio/components/artifact-selection-dialog.component.d.ts +4 -2
  152. package/dist/ComponentStudio/components/artifact-selection-dialog.component.d.ts.map +1 -1
  153. package/dist/ComponentStudio/components/artifact-selection-dialog.component.js +19 -11
  154. package/dist/ComponentStudio/components/artifact-selection-dialog.component.js.map +1 -1
  155. package/dist/ComponentStudio/components/editors/data-requirements-editor.component.d.ts +4 -2
  156. package/dist/ComponentStudio/components/editors/data-requirements-editor.component.d.ts.map +1 -1
  157. package/dist/ComponentStudio/components/editors/data-requirements-editor.component.js +11 -3
  158. package/dist/ComponentStudio/components/editors/data-requirements-editor.component.js.map +1 -1
  159. package/dist/ComponentStudio/components/workspace/component-preview.component.d.ts +10 -1
  160. package/dist/ComponentStudio/components/workspace/component-preview.component.d.ts.map +1 -1
  161. package/dist/ComponentStudio/components/workspace/component-preview.component.js +18 -9
  162. package/dist/ComponentStudio/components/workspace/component-preview.component.js.map +1 -1
  163. package/dist/ComponentStudio/services/component-studio-state.service.d.ts +11 -1
  164. package/dist/ComponentStudio/services/component-studio-state.service.d.ts.map +1 -1
  165. package/dist/ComponentStudio/services/component-studio-state.service.js +23 -3
  166. package/dist/ComponentStudio/services/component-studio-state.service.js.map +1 -1
  167. package/dist/ComponentStudio/services/component-version.service.d.ts +6 -1
  168. package/dist/ComponentStudio/services/component-version.service.d.ts.map +1 -1
  169. package/dist/ComponentStudio/services/component-version.service.js +16 -6
  170. package/dist/ComponentStudio/services/component-version.service.js.map +1 -1
  171. package/dist/Credentials/components/credentials-audit-resource.component.js +1 -1
  172. package/dist/Credentials/components/credentials-audit-resource.component.js.map +1 -1
  173. package/dist/Credentials/components/credentials-categories-resource.component.d.ts.map +1 -1
  174. package/dist/Credentials/components/credentials-categories-resource.component.js +3 -3
  175. package/dist/Credentials/components/credentials-categories-resource.component.js.map +1 -1
  176. package/dist/Credentials/components/credentials-list-resource.component.d.ts.map +1 -1
  177. package/dist/Credentials/components/credentials-list-resource.component.js +3 -3
  178. package/dist/Credentials/components/credentials-list-resource.component.js.map +1 -1
  179. package/dist/Credentials/components/credentials-overview-resource.component.d.ts.map +1 -1
  180. package/dist/Credentials/components/credentials-overview-resource.component.js +3 -3
  181. package/dist/Credentials/components/credentials-overview-resource.component.js.map +1 -1
  182. package/dist/Credentials/components/credentials-types-resource.component.d.ts.map +1 -1
  183. package/dist/Credentials/components/credentials-types-resource.component.js +3 -3
  184. package/dist/Credentials/components/credentials-types-resource.component.js.map +1 -1
  185. package/dist/Credentials/credentials-dashboard.component.js +1 -1
  186. package/dist/Credentials/credentials-dashboard.component.js.map +1 -1
  187. package/dist/DashboardBrowser/dashboard-browser-resource.component.js +13 -13
  188. package/dist/DashboardBrowser/dashboard-browser-resource.component.js.map +1 -1
  189. package/dist/DashboardBrowser/dashboard-share-adapter.d.ts +4 -0
  190. package/dist/DashboardBrowser/dashboard-share-adapter.d.ts.map +1 -1
  191. package/dist/DashboardBrowser/dashboard-share-adapter.js +5 -2
  192. package/dist/DashboardBrowser/dashboard-share-adapter.js.map +1 -1
  193. package/dist/DashboardBrowser/dashboard-share-dialog.component.d.ts +3 -1
  194. package/dist/DashboardBrowser/dashboard-share-dialog.component.d.ts.map +1 -1
  195. package/dist/DashboardBrowser/dashboard-share-dialog.component.js +9 -4
  196. package/dist/DashboardBrowser/dashboard-share-dialog.component.js.map +1 -1
  197. package/dist/DataExplorer/components/navigation-panel/navigation-panel.component.d.ts +3 -2
  198. package/dist/DataExplorer/components/navigation-panel/navigation-panel.component.d.ts.map +1 -1
  199. package/dist/DataExplorer/components/navigation-panel/navigation-panel.component.js +7 -6
  200. package/dist/DataExplorer/components/navigation-panel/navigation-panel.component.js.map +1 -1
  201. package/dist/DataExplorer/components/view-selector/view-selector.component.d.ts +3 -2
  202. package/dist/DataExplorer/components/view-selector/view-selector.component.d.ts.map +1 -1
  203. package/dist/DataExplorer/components/view-selector/view-selector.component.js +6 -5
  204. package/dist/DataExplorer/components/view-selector/view-selector.component.js.map +1 -1
  205. package/dist/DataExplorer/data-explorer-dashboard.component.d.ts.map +1 -1
  206. package/dist/DataExplorer/data-explorer-dashboard.component.js +10 -9
  207. package/dist/DataExplorer/data-explorer-dashboard.component.js.map +1 -1
  208. package/dist/DataExplorer/services/explorer-state.service.d.ts +6 -1
  209. package/dist/DataExplorer/services/explorer-state.service.d.ts.map +1 -1
  210. package/dist/DataExplorer/services/explorer-state.service.js +15 -5
  211. package/dist/DataExplorer/services/explorer-state.service.js.map +1 -1
  212. package/dist/DatabaseDesigner/components/create-wizard/database-create-wizard.component.d.ts +64 -0
  213. package/dist/DatabaseDesigner/components/create-wizard/database-create-wizard.component.d.ts.map +1 -0
  214. package/dist/DatabaseDesigner/components/create-wizard/database-create-wizard.component.js +400 -0
  215. package/dist/DatabaseDesigner/components/create-wizard/database-create-wizard.component.js.map +1 -0
  216. package/dist/DatabaseDesigner/components/create-wizard/steps/step-basics.component.d.ts +42 -0
  217. package/dist/DatabaseDesigner/components/create-wizard/steps/step-basics.component.d.ts.map +1 -0
  218. package/dist/DatabaseDesigner/components/create-wizard/steps/step-basics.component.js +232 -0
  219. package/dist/DatabaseDesigner/components/create-wizard/steps/step-basics.component.js.map +1 -0
  220. package/dist/DatabaseDesigner/components/create-wizard/steps/step-fields.component.d.ts +14 -0
  221. package/dist/DatabaseDesigner/components/create-wizard/steps/step-fields.component.d.ts.map +1 -0
  222. package/dist/DatabaseDesigner/components/create-wizard/steps/step-fields.component.js +38 -0
  223. package/dist/DatabaseDesigner/components/create-wizard/steps/step-fields.component.js.map +1 -0
  224. package/dist/DatabaseDesigner/components/create-wizard/steps/step-pipeline.component.d.ts +24 -0
  225. package/dist/DatabaseDesigner/components/create-wizard/steps/step-pipeline.component.d.ts.map +1 -0
  226. package/dist/DatabaseDesigner/components/create-wizard/steps/step-pipeline.component.js +77 -0
  227. package/dist/DatabaseDesigner/components/create-wizard/steps/step-pipeline.component.js.map +1 -0
  228. package/dist/DatabaseDesigner/components/create-wizard/steps/step-relationships.component.d.ts +88 -0
  229. package/dist/DatabaseDesigner/components/create-wizard/steps/step-relationships.component.d.ts.map +1 -0
  230. package/dist/DatabaseDesigner/components/create-wizard/steps/step-relationships.component.js +404 -0
  231. package/dist/DatabaseDesigner/components/create-wizard/steps/step-relationships.component.js.map +1 -0
  232. package/dist/DatabaseDesigner/components/create-wizard/steps/step-review.component.d.ts +12 -0
  233. package/dist/DatabaseDesigner/components/create-wizard/steps/step-review.component.d.ts.map +1 -0
  234. package/dist/DatabaseDesigner/components/create-wizard/steps/step-review.component.js +116 -0
  235. package/dist/DatabaseDesigner/components/create-wizard/steps/step-review.component.js.map +1 -0
  236. package/dist/DatabaseDesigner/components/database-designer-dashboard.component.d.ts +55 -0
  237. package/dist/DatabaseDesigner/components/database-designer-dashboard.component.d.ts.map +1 -0
  238. package/dist/DatabaseDesigner/components/database-designer-dashboard.component.js +204 -0
  239. package/dist/DatabaseDesigner/components/database-designer-dashboard.component.js.map +1 -0
  240. package/dist/DatabaseDesigner/components/entity-list.component.d.ts +61 -0
  241. package/dist/DatabaseDesigner/components/entity-list.component.d.ts.map +1 -0
  242. package/dist/DatabaseDesigner/components/entity-list.component.js +485 -0
  243. package/dist/DatabaseDesigner/components/entity-list.component.js.map +1 -0
  244. package/dist/DatabaseDesigner/components/modify/database-modify.component.d.ts +52 -0
  245. package/dist/DatabaseDesigner/components/modify/database-modify.component.d.ts.map +1 -0
  246. package/dist/DatabaseDesigner/components/modify/database-modify.component.js +320 -0
  247. package/dist/DatabaseDesigner/components/modify/database-modify.component.js.map +1 -0
  248. package/dist/DatabaseDesigner/components/shared/database-preview-pane.component.d.ts +79 -0
  249. package/dist/DatabaseDesigner/components/shared/database-preview-pane.component.d.ts.map +1 -0
  250. package/dist/DatabaseDesigner/components/shared/database-preview-pane.component.js +378 -0
  251. package/dist/DatabaseDesigner/components/shared/database-preview-pane.component.js.map +1 -0
  252. package/dist/DatabaseDesigner/components/shared/entity-fields-grid.component.d.ts +85 -0
  253. package/dist/DatabaseDesigner/components/shared/entity-fields-grid.component.d.ts.map +1 -0
  254. package/dist/DatabaseDesigner/components/shared/entity-fields-grid.component.js +429 -0
  255. package/dist/DatabaseDesigner/components/shared/entity-fields-grid.component.js.map +1 -0
  256. package/dist/DatabaseDesigner/components/shared/entity-pipeline-panel.component.d.ts +58 -0
  257. package/dist/DatabaseDesigner/components/shared/entity-pipeline-panel.component.d.ts.map +1 -0
  258. package/dist/DatabaseDesigner/components/shared/entity-pipeline-panel.component.js +333 -0
  259. package/dist/DatabaseDesigner/components/shared/entity-pipeline-panel.component.js.map +1 -0
  260. package/dist/DatabaseDesigner/components/shared/entity-review-panel.component.d.ts +36 -0
  261. package/dist/DatabaseDesigner/components/shared/entity-review-panel.component.d.ts.map +1 -0
  262. package/dist/DatabaseDesigner/components/shared/entity-review-panel.component.js +345 -0
  263. package/dist/DatabaseDesigner/components/shared/entity-review-panel.component.js.map +1 -0
  264. package/dist/DatabaseDesigner/components/shared/wizard-step-indicator.component.d.ts +13 -0
  265. package/dist/DatabaseDesigner/components/shared/wizard-step-indicator.component.d.ts.map +1 -0
  266. package/dist/DatabaseDesigner/components/shared/wizard-step-indicator.component.js +81 -0
  267. package/dist/DatabaseDesigner/components/shared/wizard-step-indicator.component.js.map +1 -0
  268. package/dist/DatabaseDesigner/database-designer-dashboards.module.d.ts +30 -0
  269. package/dist/DatabaseDesigner/database-designer-dashboards.module.d.ts.map +1 -0
  270. package/dist/DatabaseDesigner/database-designer-dashboards.module.js +131 -0
  271. package/dist/DatabaseDesigner/database-designer-dashboards.module.js.map +1 -0
  272. package/dist/DatabaseDesigner/database-designer-erd.d.ts +19 -0
  273. package/dist/DatabaseDesigner/database-designer-erd.d.ts.map +1 -0
  274. package/dist/DatabaseDesigner/database-designer-erd.js +84 -0
  275. package/dist/DatabaseDesigner/database-designer-erd.js.map +1 -0
  276. package/dist/DatabaseDesigner/database-designer.types.d.ts +167 -0
  277. package/dist/DatabaseDesigner/database-designer.types.d.ts.map +1 -0
  278. package/dist/DatabaseDesigner/database-designer.types.js +26 -0
  279. package/dist/DatabaseDesigner/database-designer.types.js.map +1 -0
  280. package/dist/DatabaseDesigner/services/database-designer.engine.d.ts +94 -0
  281. package/dist/DatabaseDesigner/services/database-designer.engine.d.ts.map +1 -0
  282. package/dist/DatabaseDesigner/services/database-designer.engine.js +392 -0
  283. package/dist/DatabaseDesigner/services/database-designer.engine.js.map +1 -0
  284. package/dist/DatabaseDesigner/services/database-designer.service.d.ts +78 -0
  285. package/dist/DatabaseDesigner/services/database-designer.service.d.ts.map +1 -0
  286. package/dist/DatabaseDesigner/services/database-designer.service.js +273 -0
  287. package/dist/DatabaseDesigner/services/database-designer.service.js.map +1 -0
  288. package/dist/DatabaseDesigner/services/entity-wizard-state.service.d.ts +79 -0
  289. package/dist/DatabaseDesigner/services/entity-wizard-state.service.d.ts.map +1 -0
  290. package/dist/DatabaseDesigner/services/entity-wizard-state.service.js +204 -0
  291. package/dist/DatabaseDesigner/services/entity-wizard-state.service.js.map +1 -0
  292. package/dist/EntityAdmin/entity-admin-dashboard.component.d.ts.map +1 -1
  293. package/dist/EntityAdmin/entity-admin-dashboard.component.js +2 -2
  294. package/dist/EntityAdmin/entity-admin-dashboard.component.js.map +1 -1
  295. package/dist/Home/action-pin-config-dialog.component.d.ts +2 -1
  296. package/dist/Home/action-pin-config-dialog.component.d.ts.map +1 -1
  297. package/dist/Home/action-pin-config-dialog.component.js +6 -4
  298. package/dist/Home/action-pin-config-dialog.component.js.map +1 -1
  299. package/dist/Home/action-pin-runner-dialog.component.d.ts +2 -1
  300. package/dist/Home/action-pin-runner-dialog.component.d.ts.map +1 -1
  301. package/dist/Home/action-pin-runner-dialog.component.js +5 -4
  302. package/dist/Home/action-pin-runner-dialog.component.js.map +1 -1
  303. package/dist/Home/home-application.d.ts.map +1 -1
  304. package/dist/Home/home-application.js +10 -12
  305. package/dist/Home/home-application.js.map +1 -1
  306. package/dist/Home/home-dashboard.component.d.ts.map +1 -1
  307. package/dist/Home/home-dashboard.component.js +3 -3
  308. package/dist/Home/home-dashboard.component.js.map +1 -1
  309. package/dist/Integration/components/activity/activity.component.d.ts.map +1 -1
  310. package/dist/Integration/components/activity/activity.component.js +1 -0
  311. package/dist/Integration/components/activity/activity.component.js.map +1 -1
  312. package/dist/Integration/components/connections/connections.component.d.ts.map +1 -1
  313. package/dist/Integration/components/connections/connections.component.js +11 -10
  314. package/dist/Integration/components/connections/connections.component.js.map +1 -1
  315. package/dist/Integration/components/mapping-workspace/mapping-workspace.component.d.ts.map +1 -1
  316. package/dist/Integration/components/mapping-workspace/mapping-workspace.component.js +2 -2
  317. package/dist/Integration/components/mapping-workspace/mapping-workspace.component.js.map +1 -1
  318. package/dist/Integration/components/overview/overview.component.d.ts.map +1 -1
  319. package/dist/Integration/components/overview/overview.component.js +1 -0
  320. package/dist/Integration/components/overview/overview.component.js.map +1 -1
  321. package/dist/Integration/components/pipelines/pipelines.component.d.ts.map +1 -1
  322. package/dist/Integration/components/pipelines/pipelines.component.js +1 -0
  323. package/dist/Integration/components/pipelines/pipelines.component.js.map +1 -1
  324. package/dist/Integration/components/schedules/schedules.component.d.ts.map +1 -1
  325. package/dist/Integration/components/schedules/schedules.component.js +5 -4
  326. package/dist/Integration/components/schedules/schedules.component.js.map +1 -1
  327. package/dist/Integration/services/integration-data.service.d.ts +5 -1
  328. package/dist/Integration/services/integration-data.service.d.ts.map +1 -1
  329. package/dist/Integration/services/integration-data.service.js +27 -19
  330. package/dist/Integration/services/integration-data.service.js.map +1 -1
  331. package/dist/KnowledgeHub/components/analytics/analytics-resource.component.js +6 -6
  332. package/dist/KnowledgeHub/components/analytics/analytics-resource.component.js.map +1 -1
  333. package/dist/KnowledgeHub/components/clusters/cluster-visualization-resource.component.d.ts.map +1 -1
  334. package/dist/KnowledgeHub/components/clusters/cluster-visualization-resource.component.js +8 -6
  335. package/dist/KnowledgeHub/components/clusters/cluster-visualization-resource.component.js.map +1 -1
  336. package/dist/KnowledgeHub/components/config/knowledge-config-resource.component.js +6 -6
  337. package/dist/KnowledgeHub/components/config/knowledge-config-resource.component.js.map +1 -1
  338. package/dist/KnowledgeHub/components/scheduling/scheduling-resource.component.js +3 -3
  339. package/dist/KnowledgeHub/components/scheduling/scheduling-resource.component.js.map +1 -1
  340. package/dist/Lists/components/lists-browse-resource.component.js +6 -6
  341. package/dist/Lists/components/lists-browse-resource.component.js.map +1 -1
  342. package/dist/Lists/components/lists-categories-resource.component.js +4 -4
  343. package/dist/Lists/components/lists-categories-resource.component.js.map +1 -1
  344. package/dist/Lists/components/lists-my-lists-resource.component.js +6 -6
  345. package/dist/Lists/components/lists-my-lists-resource.component.js.map +1 -1
  346. package/dist/Lists/components/lists-operations-resource.component.d.ts.map +1 -1
  347. package/dist/Lists/components/lists-operations-resource.component.js +11 -10
  348. package/dist/Lists/components/lists-operations-resource.component.js.map +1 -1
  349. package/dist/Lists/services/list-set-operations.service.d.ts +5 -0
  350. package/dist/Lists/services/list-set-operations.service.d.ts.map +1 -1
  351. package/dist/Lists/services/list-set-operations.service.js +11 -3
  352. package/dist/Lists/services/list-set-operations.service.js.map +1 -1
  353. package/dist/MCP/components/mcp-connection-dialog.component.d.ts +2 -1
  354. package/dist/MCP/components/mcp-connection-dialog.component.d.ts.map +1 -1
  355. package/dist/MCP/components/mcp-connection-dialog.component.js +8 -6
  356. package/dist/MCP/components/mcp-connection-dialog.component.js.map +1 -1
  357. package/dist/MCP/components/mcp-server-dialog.component.d.ts +2 -1
  358. package/dist/MCP/components/mcp-server-dialog.component.d.ts.map +1 -1
  359. package/dist/MCP/components/mcp-server-dialog.component.js +7 -5
  360. package/dist/MCP/components/mcp-server-dialog.component.js.map +1 -1
  361. package/dist/MCP/mcp-dashboard.component.d.ts.map +1 -1
  362. package/dist/MCP/mcp-dashboard.component.js +12 -12
  363. package/dist/MCP/mcp-dashboard.component.js.map +1 -1
  364. package/dist/Permissions/permissions-shared.d.ts +3 -3
  365. package/dist/Permissions/permissions-shared.d.ts.map +1 -1
  366. package/dist/Permissions/permissions-shared.js +5 -5
  367. package/dist/Permissions/permissions-shared.js.map +1 -1
  368. package/dist/Permissions/resource-access-resource.component.js +1 -1
  369. package/dist/Permissions/resource-access-resource.component.js.map +1 -1
  370. package/dist/Permissions/user-access-resource.component.js +1 -2
  371. package/dist/Permissions/user-access-resource.component.js.map +1 -1
  372. package/dist/QueryBrowser/query-browser-resource.component.d.ts.map +1 -1
  373. package/dist/QueryBrowser/query-browser-resource.component.js +2 -2
  374. package/dist/QueryBrowser/query-browser-resource.component.js.map +1 -1
  375. package/dist/Scheduling/scheduling-dashboard.component.d.ts.map +1 -1
  376. package/dist/Scheduling/scheduling-dashboard.component.js +1 -0
  377. package/dist/Scheduling/scheduling-dashboard.component.js.map +1 -1
  378. package/dist/Scheduling/services/scheduling-instrumentation.service.d.ts +5 -0
  379. package/dist/Scheduling/services/scheduling-instrumentation.service.d.ts.map +1 -1
  380. package/dist/Scheduling/services/scheduling-instrumentation.service.js +21 -13
  381. package/dist/Scheduling/services/scheduling-instrumentation.service.js.map +1 -1
  382. package/dist/SystemDiagnostics/system-diagnostics.component.d.ts.map +1 -1
  383. package/dist/SystemDiagnostics/system-diagnostics.component.js +2 -2
  384. package/dist/SystemDiagnostics/system-diagnostics.component.js.map +1 -1
  385. package/dist/Testing/components/testing-explorer.component.d.ts +2 -1
  386. package/dist/Testing/components/testing-explorer.component.d.ts.map +1 -1
  387. package/dist/Testing/components/testing-explorer.component.js +9 -7
  388. package/dist/Testing/components/testing-explorer.component.js.map +1 -1
  389. package/dist/Testing/services/testing-instrumentation.service.d.ts +6 -1
  390. package/dist/Testing/services/testing-instrumentation.service.d.ts.map +1 -1
  391. package/dist/Testing/services/testing-instrumentation.service.js +21 -11
  392. package/dist/Testing/services/testing-instrumentation.service.js.map +1 -1
  393. package/dist/Testing/testing-dashboard.component.d.ts +3 -1
  394. package/dist/Testing/testing-dashboard.component.d.ts.map +1 -1
  395. package/dist/Testing/testing-dashboard.component.js +15 -11
  396. package/dist/Testing/testing-dashboard.component.js.map +1 -1
  397. package/dist/VersionHistory/components/diff-resource.component.d.ts.map +1 -1
  398. package/dist/VersionHistory/components/diff-resource.component.js +5 -5
  399. package/dist/VersionHistory/components/diff-resource.component.js.map +1 -1
  400. package/dist/VersionHistory/components/graph-resource.component.d.ts.map +1 -1
  401. package/dist/VersionHistory/components/graph-resource.component.js +1 -2
  402. package/dist/VersionHistory/components/graph-resource.component.js.map +1 -1
  403. package/dist/VersionHistory/components/labels-resource.component.d.ts.map +1 -1
  404. package/dist/VersionHistory/components/labels-resource.component.js +3 -3
  405. package/dist/VersionHistory/components/labels-resource.component.js.map +1 -1
  406. package/dist/VersionHistory/components/restore-resource.component.js +1 -1
  407. package/dist/VersionHistory/components/restore-resource.component.js.map +1 -1
  408. package/dist/ai-dashboards.module.d.ts +35 -34
  409. package/dist/ai-dashboards.module.d.ts.map +1 -1
  410. package/dist/ai-dashboards.module.js +6 -0
  411. package/dist/ai-dashboards.module.js.map +1 -1
  412. package/dist/module.d.ts +13 -12
  413. package/dist/module.d.ts.map +1 -1
  414. package/dist/module.js +7 -0
  415. package/dist/module.js.map +1 -1
  416. package/dist/public-api.d.ts +3 -1
  417. package/dist/public-api.d.ts.map +1 -1
  418. package/dist/public-api.js +3 -1
  419. package/dist/public-api.js.map +1 -1
  420. package/dist/shared/entity-field-display.d.ts +4 -3
  421. package/dist/shared/entity-field-display.d.ts.map +1 -1
  422. package/dist/shared/entity-field-display.js +9 -9
  423. package/dist/shared/entity-field-display.js.map +1 -1
  424. package/package.json +52 -51
@@ -0,0 +1,392 @@
1
+ /**
2
+ * @module database-designer.engine
3
+ * @description Read-only data access engine for the Database Designer Angular UI.
4
+ *
5
+ * `DatabaseDesignerEngine` is a `BaseSingleton` (NOT an Angular service) that:
6
+ * - Loads the list of entities accessible to the current user
7
+ * - Loads full entity detail for the modify wizard
8
+ * - Loads available schemas from `MJ: Schema Info` (async, cached per-user)
9
+ * - Checks whether a proposed entity name / table name is available
10
+ *
11
+ * ### Why BaseSingleton, not @Injectable?
12
+ * Engine classes hold a cross-request cache that must survive Angular's DI
13
+ * lifecycle boundaries (the same instance is reused across multiple wizard
14
+ * openings in the same session). `BaseSingleton` uses the MJ Global Object
15
+ * Store which guarantees exactly one instance per process, even across ESBuild
16
+ * code-split chunks.
17
+ *
18
+ * ### Usage
19
+ * ```typescript
20
+ * const engine = DatabaseDesignerEngine.Instance;
21
+ * const entities = await engine.loadAccessibleEntities();
22
+ * ```
23
+ *
24
+ * @see DatabaseDesignerService — for action invocation (create, modify, validate)
25
+ */
26
+ import { Metadata, RunView, AuthorizationEvaluator } from '@memberjunction/core';
27
+ import { BaseSingleton, UUIDsEqual } from '@memberjunction/global';
28
+ // ─── Constants (mirror database-designer-core/src/interfaces.ts) ───────────────
29
+ // These string values MUST stay in sync with the server-side constants.
30
+ // If the server-side values change, update here in parallel.
31
+ const UDT_SCHEMA_NAME = '__mj_UDT';
32
+ const UDT_SETTINGS = {
33
+ OWNER_KEY: 'MJ:UDT:Owner',
34
+ SOURCE_KEY: 'MJ:UDT:Source',
35
+ SOURCE_DATABASE_DESIGNER: 'DatabaseDesigner',
36
+ SOURCE_AGENT_MANAGER: 'AgentManager',
37
+ };
38
+ const ENTITY_DESIGNER_AUTH = {
39
+ CREATE_IN_UDT_SCHEMA: 'Create in UDT Schema',
40
+ CREATE_IN_CUSTOM_SCHEMA: 'Create in Custom Schema',
41
+ MODIFY_OWN_ENTITIES: 'Modify Own Entities',
42
+ MODIFY_ANY_UDT_ENTITIES: 'Modify Any UDT Entities',
43
+ };
44
+ /** Cache TTL in milliseconds (5 minutes). */
45
+ const CACHE_TTL_MS = 5 * 60 * 1_000;
46
+ // ─── Columns managed automatically by CodeGen (never shown as user-editable) ─
47
+ /**
48
+ * Normalised lower-case set of column names that CodeGen injects on every entity.
49
+ * These are filtered out of the modify wizard so users cannot inadvertently remove or
50
+ * rename them. The set is compared after calling `.toLowerCase()` on each column name
51
+ * so it is case-insensitive regardless of DB platform.
52
+ */
53
+ const AUTO_MANAGED_COLUMNS = new Set(['id', '__mj_createdat', '__mj_updatedat']);
54
+ // ─── Frontend-only schema blocklist ──────────────────────────────────────────
55
+ // `MJ: Schema Info` only surfaces MJ-registered schemas, so system schemas
56
+ // (sys, dbo, information_schema) never appear. Only __mj must be blocked here.
57
+ // Server-side RSM.GetAllProtectedSchemas() remains the authoritative gate.
58
+ const FRONTEND_BLOCKED_SCHEMAS = new Set(['__mj']);
59
+ // ─── Engine ──────────────────────────────────────────────────────────────────
60
+ /**
61
+ * Singleton data access engine for Database Designer.
62
+ * All methods read data; none modifies state (mutations go through `DatabaseDesignerService`).
63
+ */
64
+ export class DatabaseDesignerEngine extends BaseSingleton {
65
+ static get Instance() {
66
+ return super.getInstance();
67
+ }
68
+ constructor() { super(); }
69
+ // ─── Cache ────────────────────────────────────────────────────────────
70
+ /** Per-user entity list cache. Key = userID. */
71
+ _cache = new Map();
72
+ /** Per-user schema list cache. Key = userID. Invalidated alongside entity cache. */
73
+ _schemaCache = new Map();
74
+ /** Invalidate all per-user caches for the current user (call after create/modify). */
75
+ invalidateCache() {
76
+ const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider
77
+ if (userId) {
78
+ this._cache.delete(userId);
79
+ this._schemaCache.delete(userId);
80
+ }
81
+ }
82
+ // ─── Entity list ──────────────────────────────────────────────────────
83
+ /**
84
+ * Load all entities accessible to the current user.
85
+ *
86
+ * - **Admin** (holds `Modify Any UDT Entities`): sees all entities created by
87
+ * Database Designer or Agent Manager, with ownership annotations.
88
+ * - **Regular user**: sees only entities they own (`MJ:UDT:Owner = userId`).
89
+ *
90
+ * Results are cached per-user with a 5-minute TTL.
91
+ */
92
+ async loadAccessibleEntities() {
93
+ const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider
94
+ const userId = md.CurrentUser?.ID;
95
+ if (!userId)
96
+ return [];
97
+ const cached = this._cache.get(userId);
98
+ if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {
99
+ return cached.entities;
100
+ }
101
+ const isAdmin = this.currentUserCanModifyAny();
102
+ const entities = isAdmin
103
+ ? await this.loadAllEntities(userId)
104
+ : await this.loadOwnedEntities(userId);
105
+ this._cache.set(userId, { entities, timestamp: Date.now() });
106
+ return entities;
107
+ }
108
+ /**
109
+ * Load full entity detail (columns, foreign keys, description) for the modify wizard.
110
+ * Not cached — always fetches fresh data so the wizard reflects current DB state.
111
+ */
112
+ async loadEntityDetail(entityId) {
113
+ const rv = new RunView();
114
+ // BypassCache: true on both queries because CodeGen writes Entity Fields via direct
115
+ // SQL (outside BaseEntity.Save()), so the server-side RunView cache never receives
116
+ // an invalidation event after a pipeline run. Without this, the detail view would
117
+ // show stale field data (e.g. old AllowsNull value) until MJAPI restarts.
118
+ const [entityResult, fieldsResult] = await rv.RunViews([
119
+ {
120
+ EntityName: 'MJ: Entities',
121
+ ExtraFilter: `ID = '${escapeSqlId(entityId)}'`,
122
+ Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', 'Description', '__mj_CreatedAt'],
123
+ ResultType: 'simple',
124
+ BypassCache: true,
125
+ },
126
+ {
127
+ EntityName: 'MJ: Entity Fields',
128
+ ExtraFilter: `EntityID = '${escapeSqlId(entityId)}' AND IsVirtual = 0`,
129
+ Fields: ['ID', 'EntityID', 'Name', 'Type', 'AllowsNull', 'Description', 'DefaultValue', 'MaxLength', 'Precision', 'Scale'],
130
+ OrderBy: 'Sequence ASC, Name ASC',
131
+ ResultType: 'simple',
132
+ BypassCache: true,
133
+ },
134
+ ]);
135
+ if (!entityResult.Success || entityResult.Results.length === 0)
136
+ return null;
137
+ const entity = entityResult.Results[0];
138
+ const fieldRows = (fieldsResult.Results ?? []);
139
+ const isOwner = await this.isCurrentUserOwner(entityId);
140
+ // Exclude auto-managed columns (ID, __mj_CreatedAt, __mj_UpdatedAt) from the editable
141
+ // column list. CodeGen injects these automatically — they must not be user-modified.
142
+ const editableFields = fieldRows.filter(f => !AUTO_MANAGED_COLUMNS.has(f.Name.toLowerCase()));
143
+ return {
144
+ entityId: entity.ID,
145
+ entityName: entity.Name,
146
+ tableName: entity.BaseTable,
147
+ schemaName: entity.SchemaName,
148
+ description: entity.Description ?? '',
149
+ fieldCount: editableFields.length,
150
+ createdAt: new Date(entity.__mj_CreatedAt),
151
+ isOwner,
152
+ columns: editableFields.map(f => this.mapFieldToColumnSpec(f)),
153
+ foreignKeys: [], // FK detail loaded on demand — Phase 5e
154
+ };
155
+ }
156
+ // ─── Schema options ───────────────────────────────────────────────────
157
+ /**
158
+ * Return the list of schemas the current user is authorized to create tables in.
159
+ *
160
+ * Async — queries `MJ: Schema Info` to surface real DB schemas for users
161
+ * with `Create in Custom Schema` authorization. Results are cached per-user
162
+ * with the same 5-minute TTL as the entity list.
163
+ *
164
+ * Order:
165
+ * 1. `__mj_UDT` (if user holds `Create in UDT Schema`)
166
+ * 2. Real DB schemas from `MJ: Schema Info`, excluding blocked schemas
167
+ * and `__mj_UDT` (already listed first)
168
+ * 3. "Other (enter schema name)" free-text option, at the end
169
+ *
170
+ * All items beyond #1 require `Create in Custom Schema` authorization.
171
+ */
172
+ async loadAvailableSchemas() {
173
+ const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider
174
+ if (!userId)
175
+ return [];
176
+ const cached = this._schemaCache.get(userId);
177
+ if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {
178
+ return cached.schemas;
179
+ }
180
+ const schemas = await this.fetchAvailableSchemas();
181
+ this._schemaCache.set(userId, { schemas, timestamp: Date.now() });
182
+ return schemas;
183
+ }
184
+ /** Build the schema option list: UDT first, then real DB schemas, then Other. */
185
+ async fetchAvailableSchemas() {
186
+ const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider
187
+ const evaluator = new AuthorizationEvaluator();
188
+ const udtAuth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.CREATE_IN_UDT_SCHEMA);
189
+ const customAuth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.CREATE_IN_CUSTOM_SCHEMA);
190
+ const canUseUdt = !!(udtAuth && evaluator.CurrentUserCanExecuteWithAncestors(udtAuth));
191
+ const canUseCustom = !!(customAuth && evaluator.CurrentUserCanExecuteWithAncestors(customAuth));
192
+ const schemas = [];
193
+ if (canUseUdt) {
194
+ schemas.push({
195
+ value: UDT_SCHEMA_NAME,
196
+ label: `${UDT_SCHEMA_NAME} — User-Defined Tables (default)`,
197
+ isDefault: true,
198
+ requiresElevatedAuth: false,
199
+ });
200
+ }
201
+ if (canUseCustom) {
202
+ const dbSchemas = await this.loadDbSchemaNames();
203
+ for (const schemaName of dbSchemas) {
204
+ // Skip the UDT schema (already first) and any server-blocked schemas
205
+ if (schemaName === UDT_SCHEMA_NAME)
206
+ continue;
207
+ if (FRONTEND_BLOCKED_SCHEMAS.has(schemaName))
208
+ continue;
209
+ schemas.push({
210
+ value: schemaName,
211
+ label: schemaName,
212
+ isDefault: false,
213
+ requiresElevatedAuth: true,
214
+ });
215
+ }
216
+ // Free-text escape hatch for schemas not yet registered with MJ
217
+ schemas.push({
218
+ value: '',
219
+ label: 'Other (enter schema name)',
220
+ isDefault: false,
221
+ requiresElevatedAuth: true,
222
+ });
223
+ }
224
+ return schemas;
225
+ }
226
+ /** Query `MJ: Schema Info` for distinct schema names registered with this MJ instance. */
227
+ async loadDbSchemaNames() {
228
+ const rv = new RunView();
229
+ const result = await rv.RunView({
230
+ EntityName: 'MJ: Schema Info',
231
+ Fields: ['SchemaName'],
232
+ OrderBy: 'SchemaName ASC',
233
+ ResultType: 'simple',
234
+ });
235
+ if (!result.Success)
236
+ return [];
237
+ return result.Results.map(r => r.SchemaName).filter(Boolean);
238
+ }
239
+ // ─── Name availability ────────────────────────────────────────────────
240
+ /**
241
+ * Check whether the proposed entity name and table name are available
242
+ * (no existing MJ entity in the same schema shares either).
243
+ *
244
+ * Returns true when the proposed names are safe to use.
245
+ */
246
+ async isNameAvailable(entityName, tableName, schemaName) {
247
+ const rv = new RunView();
248
+ const result = await rv.RunView({
249
+ EntityName: 'MJ: Entities',
250
+ ExtraFilter: `(Name = '${escapeSqlLiteral(entityName)}' OR BaseTable = '${escapeSqlLiteral(tableName)}') ` +
251
+ `AND SchemaName = '${escapeSqlLiteral(schemaName)}'`,
252
+ Fields: ['ID'],
253
+ ResultType: 'simple',
254
+ });
255
+ return result.Success && result.Results.length === 0;
256
+ }
257
+ // ─── Private helpers ──────────────────────────────────────────────────
258
+ /** Admin path: load all database-Designer-managed entities + ownership annotations. */
259
+ async loadAllEntities(currentUserId) {
260
+ const rv = new RunView();
261
+ // Round 1: get entity IDs from provenance settings and owner map
262
+ const [sourceResult, ownerResult] = await rv.RunViews([
263
+ {
264
+ EntityName: 'MJ: Entity Settings',
265
+ ExtraFilter: (`Name = '${UDT_SETTINGS.SOURCE_KEY}' ` +
266
+ `AND Value IN ('${UDT_SETTINGS.SOURCE_DATABASE_DESIGNER}', '${UDT_SETTINGS.SOURCE_AGENT_MANAGER}')`),
267
+ Fields: ['EntityID', 'Value'],
268
+ ResultType: 'simple',
269
+ },
270
+ {
271
+ EntityName: 'MJ: Entity Settings',
272
+ ExtraFilter: `Name = '${UDT_SETTINGS.OWNER_KEY}'`,
273
+ Fields: ['EntityID', 'Value'],
274
+ ResultType: 'simple',
275
+ },
276
+ ]);
277
+ const sourceRows = (sourceResult.Results ?? []);
278
+ const ownerRows = (ownerResult.Results ?? []);
279
+ const entityIds = Array.from(new Set(sourceRows.map(r => r.EntityID)));
280
+ if (entityIds.length === 0)
281
+ return [];
282
+ const ownerMap = new Map(ownerRows.map(r => [r.EntityID, r.Value]));
283
+ // Round 2: entity details
284
+ const entityResult = await rv.RunView({
285
+ EntityName: 'MJ: Entities',
286
+ ExtraFilter: buildInFilter('ID', entityIds),
287
+ Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', '__mj_CreatedAt'],
288
+ OrderBy: 'Name ASC',
289
+ ResultType: 'simple',
290
+ });
291
+ return (entityResult.Results ?? []).map(row => {
292
+ const ownerUserId = ownerMap.get(row.ID);
293
+ return {
294
+ entityId: row.ID,
295
+ entityName: row.Name,
296
+ tableName: row.BaseTable,
297
+ schemaName: row.SchemaName,
298
+ fieldCount: 0, // loaded on demand in detail view
299
+ createdAt: new Date(row.__mj_CreatedAt),
300
+ // UUIDsEqual handles SQL Server uppercase vs PostgreSQL lowercase UUIDs
301
+ isOwner: ownerUserId != null && UUIDsEqual(ownerUserId, currentUserId),
302
+ };
303
+ });
304
+ }
305
+ /** Non-admin path: load only entities owned by the current user. */
306
+ async loadOwnedEntities(userId) {
307
+ const rv = new RunView();
308
+ // Find entities owned by this user
309
+ const ownerResult = await rv.RunView({
310
+ EntityName: 'MJ: Entity Settings',
311
+ ExtraFilter: `Name = '${UDT_SETTINGS.OWNER_KEY}' AND Value = '${escapeSqlLiteral(userId)}'`,
312
+ Fields: ['EntityID', 'Value'],
313
+ ResultType: 'simple',
314
+ });
315
+ const entityIds = (ownerResult.Results ?? []).map(r => r.EntityID);
316
+ if (entityIds.length === 0)
317
+ return [];
318
+ // Fetch entity details
319
+ const entityResult = await rv.RunView({
320
+ EntityName: 'MJ: Entities',
321
+ ExtraFilter: buildInFilter('ID', entityIds),
322
+ Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', '__mj_CreatedAt'],
323
+ OrderBy: 'Name ASC',
324
+ ResultType: 'simple',
325
+ });
326
+ return (entityResult.Results ?? []).map(row => ({
327
+ entityId: row.ID,
328
+ entityName: row.Name,
329
+ tableName: row.BaseTable,
330
+ schemaName: row.SchemaName,
331
+ fieldCount: 0,
332
+ createdAt: new Date(row.__mj_CreatedAt),
333
+ isOwner: true, // user can only see their own entities in this path
334
+ }));
335
+ }
336
+ /** Check if the current user owns the given entity (via MJ:UDT:Owner setting). */
337
+ async isCurrentUserOwner(entityId) {
338
+ const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider
339
+ if (!userId)
340
+ return false;
341
+ const rv = new RunView();
342
+ const result = await rv.RunView({
343
+ EntityName: 'MJ: Entity Settings',
344
+ ExtraFilter: (`EntityID = '${escapeSqlId(entityId)}' ` +
345
+ `AND Name = '${UDT_SETTINGS.OWNER_KEY}' ` +
346
+ `AND Value = '${escapeSqlLiteral(userId)}'`),
347
+ Fields: ['EntityID', 'Value'],
348
+ ResultType: 'simple',
349
+ });
350
+ return result.Success && result.Results.length > 0;
351
+ }
352
+ /** True when the current user holds the `Modify Any UDT Entities` authorization. */
353
+ currentUserCanModifyAny() {
354
+ const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider
355
+ const auth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.MODIFY_ANY_UDT_ENTITIES);
356
+ if (!auth)
357
+ return false;
358
+ const evaluator = new AuthorizationEvaluator();
359
+ return evaluator.CurrentUserCanExecuteWithAncestors(auth);
360
+ }
361
+ /** Map a `MJ: Entity Fields` row to a `ColumnSpec` for the detail view. */
362
+ mapFieldToColumnSpec(field) {
363
+ return {
364
+ Name: field.Name,
365
+ Type: 'string', // detail view uses RawSqlType for display; Type is semantic
366
+ RawSqlType: field.Type,
367
+ IsNullable: field.AllowsNull,
368
+ // Preserve precision/scale metadata so the modify wizard renders the correct
369
+ // field type controls (e.g. DECIMAL(18,2) vs DECIMAL with no constraints)
370
+ MaxLength: field.MaxLength ?? undefined,
371
+ Precision: field.Precision ?? undefined,
372
+ Scale: field.Scale ?? undefined,
373
+ DefaultValue: field.DefaultValue ?? undefined,
374
+ Description: field.Description ?? undefined,
375
+ };
376
+ }
377
+ }
378
+ // ─── Private module-level utilities ──────────────────────────────────────────
379
+ /** Build a SQL `column IN ('id1', 'id2', ...)` filter for a UUID list. */
380
+ function buildInFilter(column, ids) {
381
+ const quoted = ids.map(id => `'${escapeSqlId(id)}'`).join(', ');
382
+ return `${column} IN (${quoted})`;
383
+ }
384
+ /** Escape a UUID for safe embedding in a SQL filter (UUIDs are alphanumeric + dashes, but be explicit). */
385
+ function escapeSqlId(id) {
386
+ return id.replace(/[^a-zA-Z0-9-]/g, '');
387
+ }
388
+ /** Escape a string value for safe embedding in a SQL single-quoted literal. */
389
+ function escapeSqlLiteral(value) {
390
+ return value.replace(/'/g, "''");
391
+ }
392
+ //# sourceMappingURL=database-designer.engine.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"database-designer.engine.js","sourceRoot":"","sources":["../../../src/DatabaseDesigner/services/database-designer.engine.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AAEH,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,sBAAsB,EAAE,MAAM,sBAAsB,CAAC;AACjF,OAAO,EAAE,aAAa,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AAUnE,kFAAkF;AAClF,wEAAwE;AACxE,6DAA6D;AAE7D,MAAM,eAAe,GAAG,UAAU,CAAC;AAEnC,MAAM,YAAY,GAAG;IACjB,SAAS,EAAE,cAAc;IACzB,UAAU,EAAE,eAAe;IAC3B,wBAAwB,EAAE,kBAAkB;IAC5C,oBAAoB,EAAE,cAAc;CAC9B,CAAC;AAEX,MAAM,oBAAoB,GAAG;IACzB,oBAAoB,EAAE,sBAAsB;IAC5C,uBAAuB,EAAE,yBAAyB;IAClD,mBAAmB,EAAE,qBAAqB;IAC1C,uBAAuB,EAAE,yBAAyB;CAC5C,CAAC;AAEX,6CAA6C;AAC7C,MAAM,YAAY,GAAG,CAAC,GAAG,EAAE,GAAG,KAAK,CAAC;AAEpC,gFAAgF;AAEhF;;;;;GAKG;AACH,MAAM,oBAAoB,GAAG,IAAI,GAAG,CAAC,CAAC,IAAI,EAAE,gBAAgB,EAAE,gBAAgB,CAAC,CAAC,CAAC;AA+BjF,gFAAgF;AAChF,2EAA2E;AAC3E,gFAAgF;AAChF,2EAA2E;AAC3E,MAAM,wBAAwB,GAAG,IAAI,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC;AAEnD,gFAAgF;AAEhF;;;GAGG;AACH,MAAM,OAAO,sBAAuB,SAAQ,aAAqC;IAEtE,MAAM,KAAK,QAAQ;QACtB,OAAO,KAAK,CAAC,WAAW,EAA0B,CAAC;IACvD,CAAC;IAED,gBAA0B,KAAK,EAAE,CAAC,CAAC,CAAC;IAEpC,yEAAyE;IAEzE,gDAAgD;IAC/B,MAAM,GAAG,IAAI,GAAG,EAAgC,CAAC;IAElE,oFAAoF;IACnE,YAAY,GAAG,IAAI,GAAG,EAA4B,CAAC;IAEpE,sFAAsF;IAC/E,eAAe;QAClB,MAAM,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,kEAAkE;QACjH,IAAI,MAAM,EAAE,CAAC;YACT,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YAC3B,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACrC,CAAC;IACL,CAAC;IAED,yEAAyE;IAEzE;;;;;;;;OAQG;IACI,KAAK,CAAC,sBAAsB;QAC/B,MAAM,EAAE,GAAG,IAAI,QAAQ,EAAE,CAAC,CAAC,kEAAkE;QAC7F,MAAM,MAAM,GAAG,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC;QAClC,IAAI,CAAC,MAAM;YAAE,OAAO,EAAE,CAAC;QAEvB,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACvC,IAAI,MAAM,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,SAAS,GAAG,YAAY,EAAE,CAAC;YACzD,OAAO,MAAM,CAAC,QAAQ,CAAC;QAC3B,CAAC;QAED,MAAM,OAAO,GAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/C,MAAM,QAAQ,GAAG,OAAO;YACpB,CAAC,CAAC,MAAM,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC;YACpC,CAAC,CAAC,MAAM,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC;QAE3C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,QAAQ,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QAC7D,OAAO,QAAQ,CAAC;IACpB,CAAC;IAED;;;OAGG;IACI,KAAK,CAAC,gBAAgB,CAAC,QAAgB;QAC1C,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QACzB,oFAAoF;QACpF,mFAAmF;QACnF,mFAAmF;QACnF,0EAA0E;QAC1E,MAAM,CAAC,YAAY,EAAE,YAAY,CAAC,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC;YACnD;gBACI,UAAU,EAAE,cAAc;gBAC1B,WAAW,EAAE,SAAS,WAAW,CAAC,QAAQ,CAAC,GAAG;gBAC9C,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,YAAY,EAAE,aAAa,EAAE,gBAAgB,CAAC;gBAClF,UAAU,EAAE,QAAQ;gBACpB,WAAW,EAAE,IAAI;aACpB;YACD;gBACI,UAAU,EAAE,mBAAmB;gBAC/B,WAAW,EAAE,eAAe,WAAW,CAAC,QAAQ,CAAC,qBAAqB;gBACtE,MAAM,EAAE,CAAC,IAAI,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,EAAE,YAAY,EAAE,aAAa,EAAE,cAAc,EAAE,WAAW,EAAE,WAAW,EAAE,OAAO,CAAC;gBAC1H,OAAO,EAAE,wBAAwB;gBACjC,UAAU,EAAE,QAAQ;gBACpB,WAAW,EAAE,IAAI;aACpB;SACJ,CAAC,CAAC;QAEH,IAAI,CAAC,YAAY,CAAC,OAAO,IAAI,YAAY,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,IAAI,CAAC;QAE5E,MAAM,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC,CAAC,CAAwC,CAAC;QAC9E,MAAM,SAAS,GAAG,CAAC,YAAY,CAAC,OAAO,IAAI,EAAE,CAAe,CAAC;QAE7D,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,QAAQ,CAAC,CAAC;QAExD,sFAAsF;QACtF,sFAAsF;QACtF,MAAM,cAAc,GAAG,SAAS,CAAC,MAAM,CACnC,CAAC,CAAC,EAAE,CAAC,CAAC,oBAAoB,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CACvD,CAAC;QAEF,OAAO;YACH,QAAQ,EAAE,MAAM,CAAC,EAAE;YACnB,UAAU,EAAE,MAAM,CAAC,IAAI;YACvB,SAAS,EAAE,MAAM,CAAC,SAAS;YAC3B,UAAU,EAAE,MAAM,CAAC,UAAU;YAC7B,WAAW,EAAE,MAAM,CAAC,WAAW,IAAI,EAAE;YACrC,UAAU,EAAE,cAAc,CAAC,MAAM;YACjC,SAAS,EAAE,IAAI,IAAI,CAAC,MAAM,CAAC,cAAc,CAAC;YAC1C,OAAO;YACP,OAAO,EAAE,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,CAAC;YAC9D,WAAW,EAAE,EAAE,EAAG,wCAAwC;SAC7D,CAAC;IACN,CAAC;IAED,yEAAyE;IAEzE;;;;;;;;;;;;;;OAcG;IACI,KAAK,CAAC,oBAAoB;QAC7B,MAAM,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,kEAAkE;QACjH,IAAI,CAAC,MAAM;YAAE,OAAO,EAAE,CAAC;QAEvB,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QAC7C,IAAI,MAAM,IAAI,IAAI,CAAC,GAAG,EAAE,GAAG,MAAM,CAAC,SAAS,GAAG,YAAY,EAAE,CAAC;YACzD,OAAO,MAAM,CAAC,OAAO,CAAC;QAC1B,CAAC;QAED,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,qBAAqB,EAAE,CAAC;QACnD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,OAAO,EAAE,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC;QAClE,OAAO,OAAO,CAAC;IACnB,CAAC;IAED,iFAAiF;IACzE,KAAK,CAAC,qBAAqB;QAC/B,MAAM,EAAE,GAAG,IAAI,QAAQ,EAAE,CAAC,CAAC,kEAAkE;QAC7F,MAAM,SAAS,GAAG,IAAI,sBAAsB,EAAE,CAAC;QAE/C,MAAM,OAAO,GAAM,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,oBAAoB,CAAC,oBAAoB,CAAC,CAAC;QACrG,MAAM,UAAU,GAAG,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,oBAAoB,CAAC,uBAAuB,CAAC,CAAC;QAExG,MAAM,SAAS,GAAM,CAAC,CAAC,CAAC,OAAO,IAAO,SAAS,CAAC,kCAAkC,CAAC,OAAO,CAAC,CAAC,CAAC;QAC7F,MAAM,YAAY,GAAG,CAAC,CAAC,CAAC,UAAU,IAAI,SAAS,CAAC,kCAAkC,CAAC,UAAU,CAAC,CAAC,CAAC;QAEhG,MAAM,OAAO,GAAmB,EAAE,CAAC;QAEnC,IAAI,SAAS,EAAE,CAAC;YACZ,OAAO,CAAC,IAAI,CAAC;gBACT,KAAK,EAAE,eAAe;gBACtB,KAAK,EAAE,GAAG,eAAe,kCAAkC;gBAC3D,SAAS,EAAE,IAAI;gBACf,oBAAoB,EAAE,KAAK;aAC9B,CAAC,CAAC;QACP,CAAC;QAED,IAAI,YAAY,EAAE,CAAC;YACf,MAAM,SAAS,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACjD,KAAK,MAAM,UAAU,IAAI,SAAS,EAAE,CAAC;gBACjC,qEAAqE;gBACrE,IAAI,UAAU,KAAK,eAAe;oBAAE,SAAS;gBAC7C,IAAI,wBAAwB,CAAC,GAAG,CAAC,UAAU,CAAC;oBAAE,SAAS;gBACvD,OAAO,CAAC,IAAI,CAAC;oBACT,KAAK,EAAE,UAAU;oBACjB,KAAK,EAAE,UAAU;oBACjB,SAAS,EAAE,KAAK;oBAChB,oBAAoB,EAAE,IAAI;iBAC7B,CAAC,CAAC;YACP,CAAC;YAED,gEAAgE;YAChE,OAAO,CAAC,IAAI,CAAC;gBACT,KAAK,EAAE,EAAE;gBACT,KAAK,EAAE,2BAA2B;gBAClC,SAAS,EAAE,KAAK;gBAChB,oBAAoB,EAAE,IAAI;aAC7B,CAAC,CAAC;QACP,CAAC;QAED,OAAO,OAAO,CAAC;IACnB,CAAC;IAED,0FAA0F;IAClF,KAAK,CAAC,iBAAiB;QAC3B,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QACzB,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAyB;YACpD,UAAU,EAAE,iBAAiB;YAC7B,MAAM,EAAE,CAAC,YAAY,CAAC;YACtB,OAAO,EAAE,gBAAgB;YACzB,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,IAAI,CAAC,MAAM,CAAC,OAAO;YAAE,OAAO,EAAE,CAAC;QAC/B,OAAO,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;IACjE,CAAC;IAED,yEAAyE;IAEzE;;;;;OAKG;IACI,KAAK,CAAC,eAAe,CAAC,UAAkB,EAAE,SAAiB,EAAE,UAAkB;QAClF,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QACzB,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAiB;YAC5C,UAAU,EAAE,cAAc;YAC1B,WAAW,EACP,YAAY,gBAAgB,CAAC,UAAU,CAAC,qBAAqB,gBAAgB,CAAC,SAAS,CAAC,KAAK;gBAC7F,qBAAqB,gBAAgB,CAAC,UAAU,CAAC,GAAG;YACxD,MAAM,EAAE,CAAC,IAAI,CAAC;YACd,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,OAAO,MAAM,CAAC,OAAO,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC;IACzD,CAAC;IAED,yEAAyE;IAEzE,uFAAuF;IAC/E,KAAK,CAAC,eAAe,CAAC,aAAqB;QAC/C,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QAEzB,iEAAiE;QACjE,MAAM,CAAC,YAAY,EAAE,WAAW,CAAC,GAAG,MAAM,EAAE,CAAC,QAAQ,CAAC;YAClD;gBACI,UAAU,EAAE,qBAAqB;gBACjC,WAAW,EAAE,CACT,WAAW,YAAY,CAAC,UAAU,IAAI;oBACtC,kBAAkB,YAAY,CAAC,wBAAwB,OAAO,YAAY,CAAC,oBAAoB,IAAI,CACtG;gBACD,MAAM,EAAE,CAAC,UAAU,EAAE,OAAO,CAAC;gBAC7B,UAAU,EAAE,QAAQ;aACvB;YACD;gBACI,UAAU,EAAE,qBAAqB;gBACjC,WAAW,EAAE,WAAW,YAAY,CAAC,SAAS,GAAG;gBACjD,MAAM,EAAE,CAAC,UAAU,EAAE,OAAO,CAAC;gBAC7B,UAAU,EAAE,QAAQ;aACvB;SACJ,CAAC,CAAC;QAEH,MAAM,UAAU,GAAG,CAAC,YAAY,CAAC,OAAO,IAAI,EAAE,CAAkB,CAAC;QACjE,MAAM,SAAS,GAAG,CAAC,WAAW,CAAC,OAAO,IAAI,EAAE,CAAkB,CAAC;QAE/D,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;QACvE,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,CAAC;QAEtC,MAAM,QAAQ,GAAG,IAAI,GAAG,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QAEpE,0BAA0B;QAC1B,MAAM,YAAY,GAAG,MAAM,EAAE,CAAC,OAAO,CAAY;YAC7C,UAAU,EAAE,cAAc;YAC1B,WAAW,EAAE,aAAa,CAAC,IAAI,EAAE,SAAS,CAAC;YAC3C,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,YAAY,EAAE,gBAAgB,CAAC;YACnE,OAAO,EAAE,UAAU;YACnB,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,OAAO,CAAC,YAAY,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;YAC1C,MAAM,WAAW,GAAG,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YACzC,OAAO;gBACH,QAAQ,EAAE,GAAG,CAAC,EAAE;gBAChB,UAAU,EAAE,GAAG,CAAC,IAAI;gBACpB,SAAS,EAAE,GAAG,CAAC,SAAS;gBACxB,UAAU,EAAE,GAAG,CAAC,UAAU;gBAC1B,UAAU,EAAE,CAAC,EAAG,kCAAkC;gBAClD,SAAS,EAAE,IAAI,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC;gBACvC,wEAAwE;gBACxE,OAAO,EAAE,WAAW,IAAI,IAAI,IAAI,UAAU,CAAC,WAAW,EAAE,aAAa,CAAC;aAC9C,CAAC;QACjC,CAAC,CAAC,CAAC;IACP,CAAC;IAED,oEAAoE;IAC5D,KAAK,CAAC,iBAAiB,CAAC,MAAc;QAC1C,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QAEzB,mCAAmC;QACnC,MAAM,WAAW,GAAG,MAAM,EAAE,CAAC,OAAO,CAAc;YAC9C,UAAU,EAAE,qBAAqB;YACjC,WAAW,EAAE,WAAW,YAAY,CAAC,SAAS,kBAAkB,gBAAgB,CAAC,MAAM,CAAC,GAAG;YAC3F,MAAM,EAAE,CAAC,UAAU,EAAE,OAAO,CAAC;YAC7B,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,CAAC,WAAW,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;QACnE,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,EAAE,CAAC;QAEtC,uBAAuB;QACvB,MAAM,YAAY,GAAG,MAAM,EAAE,CAAC,OAAO,CAAY;YAC7C,UAAU,EAAE,cAAc;YAC1B,WAAW,EAAE,aAAa,CAAC,IAAI,EAAE,SAAS,CAAC;YAC3C,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,EAAE,WAAW,EAAE,YAAY,EAAE,gBAAgB,CAAC;YACnE,OAAO,EAAE,UAAU;YACnB,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,OAAO,CAAC,YAAY,CAAC,OAAO,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC5C,QAAQ,EAAE,GAAG,CAAC,EAAE;YAChB,UAAU,EAAE,GAAG,CAAC,IAAI;YACpB,SAAS,EAAE,GAAG,CAAC,SAAS;YACxB,UAAU,EAAE,GAAG,CAAC,UAAU;YAC1B,UAAU,EAAE,CAAC;YACb,SAAS,EAAE,IAAI,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC;YACvC,OAAO,EAAE,IAAI,EAAG,oDAAoD;SAC3C,CAAA,CAAC,CAAC;IACnC,CAAC;IAED,kFAAkF;IAC1E,KAAK,CAAC,kBAAkB,CAAC,QAAgB;QAC7C,MAAM,MAAM,GAAG,IAAI,QAAQ,EAAE,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC,kEAAkE;QACjH,IAAI,CAAC,MAAM;YAAE,OAAO,KAAK,CAAC;QAE1B,MAAM,EAAE,GAAG,IAAI,OAAO,EAAE,CAAC;QACzB,MAAM,MAAM,GAAG,MAAM,EAAE,CAAC,OAAO,CAAc;YACzC,UAAU,EAAE,qBAAqB;YACjC,WAAW,EAAE,CACT,eAAe,WAAW,CAAC,QAAQ,CAAC,IAAI;gBACxC,eAAe,YAAY,CAAC,SAAS,IAAI;gBACzC,gBAAgB,gBAAgB,CAAC,MAAM,CAAC,GAAG,CAC9C;YACD,MAAM,EAAE,CAAC,UAAU,EAAE,OAAO,CAAC;YAC7B,UAAU,EAAE,QAAQ;SACvB,CAAC,CAAC;QAEH,OAAO,MAAM,CAAC,OAAO,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;IACvD,CAAC;IAED,oFAAoF;IAC5E,uBAAuB;QAC3B,MAAM,EAAE,GAAG,IAAI,QAAQ,EAAE,CAAC,CAAC,kEAAkE;QAC7F,MAAM,IAAI,GAAG,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,oBAAoB,CAAC,uBAAuB,CAAC,CAAC;QAClG,IAAI,CAAC,IAAI;YAAE,OAAO,KAAK,CAAC;QACxB,MAAM,SAAS,GAAG,IAAI,sBAAsB,EAAE,CAAC;QAC/C,OAAO,SAAS,CAAC,kCAAkC,CAAC,IAAI,CAAC,CAAC;IAC9D,CAAC;IAED,2EAA2E;IACnE,oBAAoB,CAAC,KAAe;QACxC,OAAO;YACH,IAAI,EAAE,KAAK,CAAC,IAAI;YAChB,IAAI,EAAE,QAAQ,EAAG,4DAA4D;YAC7E,UAAU,EAAE,KAAK,CAAC,IAAI;YACtB,UAAU,EAAE,KAAK,CAAC,UAAU;YAC5B,6EAA6E;YAC7E,0EAA0E;YAC1E,SAAS,EAAE,KAAK,CAAC,SAAS,IAAI,SAAS;YACvC,SAAS,EAAE,KAAK,CAAC,SAAS,IAAI,SAAS;YACvC,KAAK,EAAE,KAAK,CAAC,KAAK,IAAI,SAAS;YAC/B,YAAY,EAAE,KAAK,CAAC,YAAY,IAAI,SAAS;YAC7C,WAAW,EAAE,KAAK,CAAC,WAAW,IAAI,SAAS;SAC9C,CAAC;IACN,CAAC;CACJ;AAED,gFAAgF;AAEhF,0EAA0E;AAC1E,SAAS,aAAa,CAAC,MAAc,EAAE,GAAa;IAChD,MAAM,MAAM,GAAG,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,WAAW,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IAChE,OAAO,GAAG,MAAM,QAAQ,MAAM,GAAG,CAAC;AACtC,CAAC;AAED,2GAA2G;AAC3G,SAAS,WAAW,CAAC,EAAU;IAC3B,OAAO,EAAE,CAAC,OAAO,CAAC,gBAAgB,EAAE,EAAE,CAAC,CAAC;AAC5C,CAAC;AAED,+EAA+E;AAC/E,SAAS,gBAAgB,CAAC,KAAa;IACnC,OAAO,KAAK,CAAC,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;AACrC,CAAC","sourcesContent":["/**\n * @module database-designer.engine\n * @description Read-only data access engine for the Database Designer Angular UI.\n *\n * `DatabaseDesignerEngine` is a `BaseSingleton` (NOT an Angular service) that:\n * - Loads the list of entities accessible to the current user\n * - Loads full entity detail for the modify wizard\n * - Loads available schemas from `MJ: Schema Info` (async, cached per-user)\n * - Checks whether a proposed entity name / table name is available\n *\n * ### Why BaseSingleton, not @Injectable?\n * Engine classes hold a cross-request cache that must survive Angular's DI\n * lifecycle boundaries (the same instance is reused across multiple wizard\n * openings in the same session). `BaseSingleton` uses the MJ Global Object\n * Store which guarantees exactly one instance per process, even across ESBuild\n * code-split chunks.\n *\n * ### Usage\n * ```typescript\n * const engine = DatabaseDesignerEngine.Instance;\n * const entities = await engine.loadAccessibleEntities();\n * ```\n *\n * @see DatabaseDesignerService — for action invocation (create, modify, validate)\n */\n\nimport { Metadata, RunView, AuthorizationEvaluator } from '@memberjunction/core';\nimport { BaseSingleton, UUIDsEqual } from '@memberjunction/global';\n\nimport type {\n AccessibleEntity,\n AccessibleEntityDetail,\n ColumnSpec,\n ForeignKeySpec,\n SchemaOption,\n} from '../database-designer.types.js';\n\n// ─── Constants (mirror database-designer-core/src/interfaces.ts) ───────────────\n// These string values MUST stay in sync with the server-side constants.\n// If the server-side values change, update here in parallel.\n\nconst UDT_SCHEMA_NAME = '__mj_UDT';\n\nconst UDT_SETTINGS = {\n OWNER_KEY: 'MJ:UDT:Owner',\n SOURCE_KEY: 'MJ:UDT:Source',\n SOURCE_DATABASE_DESIGNER: 'DatabaseDesigner',\n SOURCE_AGENT_MANAGER: 'AgentManager',\n} as const;\n\nconst ENTITY_DESIGNER_AUTH = {\n CREATE_IN_UDT_SCHEMA: 'Create in UDT Schema',\n CREATE_IN_CUSTOM_SCHEMA: 'Create in Custom Schema',\n MODIFY_OWN_ENTITIES: 'Modify Own Entities',\n MODIFY_ANY_UDT_ENTITIES: 'Modify Any UDT Entities',\n} as const;\n\n/** Cache TTL in milliseconds (5 minutes). */\nconst CACHE_TTL_MS = 5 * 60 * 1_000;\n\n// ─── Columns managed automatically by CodeGen (never shown as user-editable) ─\n\n/**\n * Normalised lower-case set of column names that CodeGen injects on every entity.\n * These are filtered out of the modify wizard so users cannot inadvertently remove or\n * rename them. The set is compared after calling `.toLowerCase()` on each column name\n * so it is case-insensitive regardless of DB platform.\n */\nconst AUTO_MANAGED_COLUMNS = new Set(['id', '__mj_createdat', '__mj_updatedat']);\n\n// ─── Internal row types for RunView results ───────────────────────────────────\n\ninterface SettingsRow { EntityID: string; Value: string }\ninterface EntityRow { ID: string; Name: string; BaseTable: string; SchemaName: string; __mj_CreatedAt: string }\ninterface FieldRow {\n ID: string;\n EntityID: string;\n Name: string;\n Type: string;\n AllowsNull: boolean;\n Description: string;\n DefaultValue: string | null;\n MaxLength: number | null;\n Precision: number | null;\n Scale: number | null;\n}\n\n// ─── Cache entry ─────────────────────────────────────────────────────────────\n\ninterface EntityListCacheEntry {\n entities: AccessibleEntity[];\n timestamp: number;\n}\n\ninterface SchemaCacheEntry {\n schemas: SchemaOption[];\n timestamp: number;\n}\n\n// ─── Frontend-only schema blocklist ──────────────────────────────────────────\n// `MJ: Schema Info` only surfaces MJ-registered schemas, so system schemas\n// (sys, dbo, information_schema) never appear. Only __mj must be blocked here.\n// Server-side RSM.GetAllProtectedSchemas() remains the authoritative gate.\nconst FRONTEND_BLOCKED_SCHEMAS = new Set(['__mj']);\n\n// ─── Engine ──────────────────────────────────────────────────────────────────\n\n/**\n * Singleton data access engine for Database Designer.\n * All methods read data; none modifies state (mutations go through `DatabaseDesignerService`).\n */\nexport class DatabaseDesignerEngine extends BaseSingleton<DatabaseDesignerEngine> {\n\n public static get Instance(): DatabaseDesignerEngine {\n return super.getInstance<DatabaseDesignerEngine>();\n }\n\n protected constructor() { super(); }\n\n // ─── Cache ────────────────────────────────────────────────────────────\n\n /** Per-user entity list cache. Key = userID. */\n private readonly _cache = new Map<string, EntityListCacheEntry>();\n\n /** Per-user schema list cache. Key = userID. Invalidated alongside entity cache. */\n private readonly _schemaCache = new Map<string, SchemaCacheEntry>();\n\n /** Invalidate all per-user caches for the current user (call after create/modify). */\n public invalidateCache(): void {\n const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider\n if (userId) {\n this._cache.delete(userId);\n this._schemaCache.delete(userId);\n }\n }\n\n // ─── Entity list ──────────────────────────────────────────────────────\n\n /**\n * Load all entities accessible to the current user.\n *\n * - **Admin** (holds `Modify Any UDT Entities`): sees all entities created by\n * Database Designer or Agent Manager, with ownership annotations.\n * - **Regular user**: sees only entities they own (`MJ:UDT:Owner = userId`).\n *\n * Results are cached per-user with a 5-minute TTL.\n */\n public async loadAccessibleEntities(): Promise<AccessibleEntity[]> {\n const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider\n const userId = md.CurrentUser?.ID;\n if (!userId) return [];\n\n const cached = this._cache.get(userId);\n if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {\n return cached.entities;\n }\n\n const isAdmin = this.currentUserCanModifyAny();\n const entities = isAdmin\n ? await this.loadAllEntities(userId)\n : await this.loadOwnedEntities(userId);\n\n this._cache.set(userId, { entities, timestamp: Date.now() });\n return entities;\n }\n\n /**\n * Load full entity detail (columns, foreign keys, description) for the modify wizard.\n * Not cached — always fetches fresh data so the wizard reflects current DB state.\n */\n public async loadEntityDetail(entityId: string): Promise<AccessibleEntityDetail | null> {\n const rv = new RunView();\n // BypassCache: true on both queries because CodeGen writes Entity Fields via direct\n // SQL (outside BaseEntity.Save()), so the server-side RunView cache never receives\n // an invalidation event after a pipeline run. Without this, the detail view would\n // show stale field data (e.g. old AllowsNull value) until MJAPI restarts.\n const [entityResult, fieldsResult] = await rv.RunViews([\n {\n EntityName: 'MJ: Entities',\n ExtraFilter: `ID = '${escapeSqlId(entityId)}'`,\n Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', 'Description', '__mj_CreatedAt'],\n ResultType: 'simple',\n BypassCache: true,\n },\n {\n EntityName: 'MJ: Entity Fields',\n ExtraFilter: `EntityID = '${escapeSqlId(entityId)}' AND IsVirtual = 0`,\n Fields: ['ID', 'EntityID', 'Name', 'Type', 'AllowsNull', 'Description', 'DefaultValue', 'MaxLength', 'Precision', 'Scale'],\n OrderBy: 'Sequence ASC, Name ASC',\n ResultType: 'simple',\n BypassCache: true,\n },\n ]);\n\n if (!entityResult.Success || entityResult.Results.length === 0) return null;\n\n const entity = entityResult.Results[0] as EntityRow & { Description: string };\n const fieldRows = (fieldsResult.Results ?? []) as FieldRow[];\n\n const isOwner = await this.isCurrentUserOwner(entityId);\n\n // Exclude auto-managed columns (ID, __mj_CreatedAt, __mj_UpdatedAt) from the editable\n // column list. CodeGen injects these automatically — they must not be user-modified.\n const editableFields = fieldRows.filter(\n f => !AUTO_MANAGED_COLUMNS.has(f.Name.toLowerCase())\n );\n\n return {\n entityId: entity.ID,\n entityName: entity.Name,\n tableName: entity.BaseTable,\n schemaName: entity.SchemaName,\n description: entity.Description ?? '',\n fieldCount: editableFields.length,\n createdAt: new Date(entity.__mj_CreatedAt),\n isOwner,\n columns: editableFields.map(f => this.mapFieldToColumnSpec(f)),\n foreignKeys: [], // FK detail loaded on demand — Phase 5e\n };\n }\n\n // ─── Schema options ───────────────────────────────────────────────────\n\n /**\n * Return the list of schemas the current user is authorized to create tables in.\n *\n * Async — queries `MJ: Schema Info` to surface real DB schemas for users\n * with `Create in Custom Schema` authorization. Results are cached per-user\n * with the same 5-minute TTL as the entity list.\n *\n * Order:\n * 1. `__mj_UDT` (if user holds `Create in UDT Schema`)\n * 2. Real DB schemas from `MJ: Schema Info`, excluding blocked schemas\n * and `__mj_UDT` (already listed first)\n * 3. \"Other (enter schema name)\" free-text option, at the end\n *\n * All items beyond #1 require `Create in Custom Schema` authorization.\n */\n public async loadAvailableSchemas(): Promise<SchemaOption[]> {\n const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider\n if (!userId) return [];\n\n const cached = this._schemaCache.get(userId);\n if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {\n return cached.schemas;\n }\n\n const schemas = await this.fetchAvailableSchemas();\n this._schemaCache.set(userId, { schemas, timestamp: Date.now() });\n return schemas;\n }\n\n /** Build the schema option list: UDT first, then real DB schemas, then Other. */\n private async fetchAvailableSchemas(): Promise<SchemaOption[]> {\n const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider\n const evaluator = new AuthorizationEvaluator();\n\n const udtAuth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.CREATE_IN_UDT_SCHEMA);\n const customAuth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.CREATE_IN_CUSTOM_SCHEMA);\n\n const canUseUdt = !!(udtAuth && evaluator.CurrentUserCanExecuteWithAncestors(udtAuth));\n const canUseCustom = !!(customAuth && evaluator.CurrentUserCanExecuteWithAncestors(customAuth));\n\n const schemas: SchemaOption[] = [];\n\n if (canUseUdt) {\n schemas.push({\n value: UDT_SCHEMA_NAME,\n label: `${UDT_SCHEMA_NAME} — User-Defined Tables (default)`,\n isDefault: true,\n requiresElevatedAuth: false,\n });\n }\n\n if (canUseCustom) {\n const dbSchemas = await this.loadDbSchemaNames();\n for (const schemaName of dbSchemas) {\n // Skip the UDT schema (already first) and any server-blocked schemas\n if (schemaName === UDT_SCHEMA_NAME) continue;\n if (FRONTEND_BLOCKED_SCHEMAS.has(schemaName)) continue;\n schemas.push({\n value: schemaName,\n label: schemaName,\n isDefault: false,\n requiresElevatedAuth: true,\n });\n }\n\n // Free-text escape hatch for schemas not yet registered with MJ\n schemas.push({\n value: '',\n label: 'Other (enter schema name)',\n isDefault: false,\n requiresElevatedAuth: true,\n });\n }\n\n return schemas;\n }\n\n /** Query `MJ: Schema Info` for distinct schema names registered with this MJ instance. */\n private async loadDbSchemaNames(): Promise<string[]> {\n const rv = new RunView();\n const result = await rv.RunView<{ SchemaName: string }>({\n EntityName: 'MJ: Schema Info',\n Fields: ['SchemaName'],\n OrderBy: 'SchemaName ASC',\n ResultType: 'simple',\n });\n\n if (!result.Success) return [];\n return result.Results.map(r => r.SchemaName).filter(Boolean);\n }\n\n // ─── Name availability ────────────────────────────────────────────────\n\n /**\n * Check whether the proposed entity name and table name are available\n * (no existing MJ entity in the same schema shares either).\n *\n * Returns true when the proposed names are safe to use.\n */\n public async isNameAvailable(entityName: string, tableName: string, schemaName: string): Promise<boolean> {\n const rv = new RunView();\n const result = await rv.RunView<{ ID: string }>({\n EntityName: 'MJ: Entities',\n ExtraFilter:\n `(Name = '${escapeSqlLiteral(entityName)}' OR BaseTable = '${escapeSqlLiteral(tableName)}') ` +\n `AND SchemaName = '${escapeSqlLiteral(schemaName)}'`,\n Fields: ['ID'],\n ResultType: 'simple',\n });\n\n return result.Success && result.Results.length === 0;\n }\n\n // ─── Private helpers ──────────────────────────────────────────────────\n\n /** Admin path: load all database-Designer-managed entities + ownership annotations. */\n private async loadAllEntities(currentUserId: string): Promise<AccessibleEntity[]> {\n const rv = new RunView();\n\n // Round 1: get entity IDs from provenance settings and owner map\n const [sourceResult, ownerResult] = await rv.RunViews([\n {\n EntityName: 'MJ: Entity Settings',\n ExtraFilter: (\n `Name = '${UDT_SETTINGS.SOURCE_KEY}' ` +\n `AND Value IN ('${UDT_SETTINGS.SOURCE_DATABASE_DESIGNER}', '${UDT_SETTINGS.SOURCE_AGENT_MANAGER}')`\n ),\n Fields: ['EntityID', 'Value'],\n ResultType: 'simple',\n },\n {\n EntityName: 'MJ: Entity Settings',\n ExtraFilter: `Name = '${UDT_SETTINGS.OWNER_KEY}'`,\n Fields: ['EntityID', 'Value'],\n ResultType: 'simple',\n },\n ]);\n\n const sourceRows = (sourceResult.Results ?? []) as SettingsRow[];\n const ownerRows = (ownerResult.Results ?? []) as SettingsRow[];\n\n const entityIds = Array.from(new Set(sourceRows.map(r => r.EntityID)));\n if (entityIds.length === 0) return [];\n\n const ownerMap = new Map(ownerRows.map(r => [r.EntityID, r.Value]));\n\n // Round 2: entity details\n const entityResult = await rv.RunView<EntityRow>({\n EntityName: 'MJ: Entities',\n ExtraFilter: buildInFilter('ID', entityIds),\n Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', '__mj_CreatedAt'],\n OrderBy: 'Name ASC',\n ResultType: 'simple',\n });\n\n return (entityResult.Results ?? []).map(row => {\n const ownerUserId = ownerMap.get(row.ID);\n return {\n entityId: row.ID,\n entityName: row.Name,\n tableName: row.BaseTable,\n schemaName: row.SchemaName,\n fieldCount: 0, // loaded on demand in detail view\n createdAt: new Date(row.__mj_CreatedAt),\n // UUIDsEqual handles SQL Server uppercase vs PostgreSQL lowercase UUIDs\n isOwner: ownerUserId != null && UUIDsEqual(ownerUserId, currentUserId),\n } satisfies AccessibleEntity;\n });\n }\n\n /** Non-admin path: load only entities owned by the current user. */\n private async loadOwnedEntities(userId: string): Promise<AccessibleEntity[]> {\n const rv = new RunView();\n\n // Find entities owned by this user\n const ownerResult = await rv.RunView<SettingsRow>({\n EntityName: 'MJ: Entity Settings',\n ExtraFilter: `Name = '${UDT_SETTINGS.OWNER_KEY}' AND Value = '${escapeSqlLiteral(userId)}'`,\n Fields: ['EntityID', 'Value'],\n ResultType: 'simple',\n });\n\n const entityIds = (ownerResult.Results ?? []).map(r => r.EntityID);\n if (entityIds.length === 0) return [];\n\n // Fetch entity details\n const entityResult = await rv.RunView<EntityRow>({\n EntityName: 'MJ: Entities',\n ExtraFilter: buildInFilter('ID', entityIds),\n Fields: ['ID', 'Name', 'BaseTable', 'SchemaName', '__mj_CreatedAt'],\n OrderBy: 'Name ASC',\n ResultType: 'simple',\n });\n\n return (entityResult.Results ?? []).map(row => ({\n entityId: row.ID,\n entityName: row.Name,\n tableName: row.BaseTable,\n schemaName: row.SchemaName,\n fieldCount: 0,\n createdAt: new Date(row.__mj_CreatedAt),\n isOwner: true, // user can only see their own entities in this path\n } satisfies AccessibleEntity));\n }\n\n /** Check if the current user owns the given entity (via MJ:UDT:Owner setting). */\n private async isCurrentUserOwner(entityId: string): Promise<boolean> {\n const userId = new Metadata().CurrentUser?.ID; // global-provider-ok: client-side Angular engine, single provider\n if (!userId) return false;\n\n const rv = new RunView();\n const result = await rv.RunView<SettingsRow>({\n EntityName: 'MJ: Entity Settings',\n ExtraFilter: (\n `EntityID = '${escapeSqlId(entityId)}' ` +\n `AND Name = '${UDT_SETTINGS.OWNER_KEY}' ` +\n `AND Value = '${escapeSqlLiteral(userId)}'`\n ),\n Fields: ['EntityID', 'Value'],\n ResultType: 'simple',\n });\n\n return result.Success && result.Results.length > 0;\n }\n\n /** True when the current user holds the `Modify Any UDT Entities` authorization. */\n private currentUserCanModifyAny(): boolean {\n const md = new Metadata(); // global-provider-ok: client-side Angular engine, single provider\n const auth = md.Authorizations.find(a => a.Name === ENTITY_DESIGNER_AUTH.MODIFY_ANY_UDT_ENTITIES);\n if (!auth) return false;\n const evaluator = new AuthorizationEvaluator();\n return evaluator.CurrentUserCanExecuteWithAncestors(auth);\n }\n\n /** Map a `MJ: Entity Fields` row to a `ColumnSpec` for the detail view. */\n private mapFieldToColumnSpec(field: FieldRow): ColumnSpec {\n return {\n Name: field.Name,\n Type: 'string', // detail view uses RawSqlType for display; Type is semantic\n RawSqlType: field.Type,\n IsNullable: field.AllowsNull,\n // Preserve precision/scale metadata so the modify wizard renders the correct\n // field type controls (e.g. DECIMAL(18,2) vs DECIMAL with no constraints)\n MaxLength: field.MaxLength ?? undefined,\n Precision: field.Precision ?? undefined,\n Scale: field.Scale ?? undefined,\n DefaultValue: field.DefaultValue ?? undefined,\n Description: field.Description ?? undefined,\n };\n }\n}\n\n// ─── Private module-level utilities ──────────────────────────────────────────\n\n/** Build a SQL `column IN ('id1', 'id2', ...)` filter for a UUID list. */\nfunction buildInFilter(column: string, ids: string[]): string {\n const quoted = ids.map(id => `'${escapeSqlId(id)}'`).join(', ');\n return `${column} IN (${quoted})`;\n}\n\n/** Escape a UUID for safe embedding in a SQL filter (UUIDs are alphanumeric + dashes, but be explicit). */\nfunction escapeSqlId(id: string): string {\n return id.replace(/[^a-zA-Z0-9-]/g, '');\n}\n\n/** Escape a string value for safe embedding in a SQL single-quoted literal. */\nfunction escapeSqlLiteral(value: string): string {\n return value.replace(/'/g, \"''\");\n}\n"]}
@@ -0,0 +1,78 @@
1
+ import type { EntityTableSpec, ClientValidationResult, EntityPipelineResult, EntityDescribeResult } from '../database-designer.types.js';
2
+ import * as i0 from "@angular/core";
3
+ export interface CreateEntityOptions {
4
+ skipGitCommit?: boolean;
5
+ skipRestart?: boolean;
6
+ }
7
+ export interface ModifyEntityOptions extends CreateEntityOptions {
8
+ /** ID of the existing entity being modified. */
9
+ existingEntityId: string;
10
+ }
11
+ export declare class DatabaseDesignerService {
12
+ /**
13
+ * Lazily-created action client. Re-uses `GraphQLDataProvider.Instance` so the
14
+ * singleton provider (and its auth headers) are shared across all invocations.
15
+ */
16
+ private _actionClient;
17
+ private getActionClient;
18
+ /**
19
+ * Cache maps action name → Promise<ID>. Storing a Promise (not a resolved ID)
20
+ * prevents duplicate in-flight lookups when two callers await the same name
21
+ * before the first resolves.
22
+ */
23
+ private readonly _actionIdCache;
24
+ private resolveActionId;
25
+ private fetchActionId;
26
+ /**
27
+ * Create a new MemberJunction entity from a `EntityTableSpec`.
28
+ * Calls the `Create Entity` action on the server which runs the full RSU pipeline.
29
+ */
30
+ createEntity(tableSpec: EntityTableSpec, options?: CreateEntityOptions): Promise<EntityPipelineResult>;
31
+ /**
32
+ * Modify an existing MemberJunction entity.
33
+ * Calls the `Modify Entity` action on the server.
34
+ */
35
+ modifyEntity(tableSpec: EntityTableSpec, options: ModifyEntityOptions): Promise<EntityPipelineResult>;
36
+ /**
37
+ * Validate a `EntityTableSpec` without executing the pipeline.
38
+ * Used by Step 4 (Review) of the wizard to surface errors before the user submits.
39
+ *
40
+ * Note: `ActionResult.Success` is `true` even when `ValidationResult.Valid` is `false`
41
+ * because the action itself succeeded — validation failure is a business outcome, not
42
+ * an infrastructure error.
43
+ */
44
+ validateEntitySchema(tableSpec: EntityTableSpec, modificationType?: 'create' | 'alter'): Promise<ClientValidationResult>;
45
+ /**
46
+ * Load detailed metadata for an existing entity.
47
+ * Used by the entity list slide panel and the modify wizard entry point.
48
+ */
49
+ describeEntity(entityId: string): Promise<EntityDescribeResult>;
50
+ private buildPipelineParams;
51
+ /**
52
+ * Extract server output params from a `RunAction` result.
53
+ *
54
+ * ### Why `result.Result`, not `result.Params`?
55
+ * `ActionResolver.createActionResult` on the server serializes all Output/Both typed
56
+ * params into a JSON string stored in `ResultData`. `GraphQLActionClient` parses that
57
+ * JSON and stores the resulting array in `result.Result`. The `result.Params` field
58
+ * contains only the *input* params passed by the caller — the server's output params
59
+ * are never included there.
60
+ *
61
+ * This helper centralizes the cast so no caller ever needs to understand this transport
62
+ * contract directly.
63
+ */
64
+ private extractOutputParams;
65
+ /**
66
+ * Map a completed `ActionResult` to an `EntityPipelineResult`.
67
+ *
68
+ * Two failure modes are distinguished:
69
+ * - `result.Success === false`: infrastructure failure (auth, pipeline exception) → hard error
70
+ * - `result.Success === true` but `EntityID` missing: pipeline ran but ID unconfirmed →
71
+ * soft warning so the UI can surface guidance without blocking the user
72
+ */
73
+ private extractPipelineResult;
74
+ private errorMessage;
75
+ static ɵfac: i0.ɵɵFactoryDeclaration<DatabaseDesignerService, never>;
76
+ static ɵprov: i0.ɵɵInjectableDeclaration<DatabaseDesignerService>;
77
+ }
78
+ //# sourceMappingURL=database-designer.service.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"database-designer.service.d.ts","sourceRoot":"","sources":["../../../src/DatabaseDesigner/services/database-designer.service.ts"],"names":[],"mappings":"AA6BA,OAAO,KAAK,EACR,eAAe,EACf,sBAAsB,EACtB,oBAAoB,EACpB,oBAAoB,EAEvB,MAAM,+BAA+B,CAAC;;AAevC,MAAM,WAAW,mBAAmB;IAChC,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,WAAW,CAAC,EAAE,OAAO,CAAC;CACzB;AAED,MAAM,WAAW,mBAAoB,SAAQ,mBAAmB;IAC5D,gDAAgD;IAChD,gBAAgB,EAAE,MAAM,CAAC;CAC5B;AAID,qBACa,uBAAuB;IAIhC;;;OAGG;IACH,OAAO,CAAC,aAAa,CAAoC;IAEzD,OAAO,CAAC,eAAe;IASvB;;;;OAIG;IACH,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAsC;IAErE,OAAO,CAAC,eAAe;YAST,aAAa;IAoB3B;;;OAGG;IACU,YAAY,CACrB,SAAS,EAAE,eAAe,EAC1B,OAAO,GAAE,mBAAwB,GAClC,OAAO,CAAC,oBAAoB,CAAC;IAWhC;;;OAGG;IACU,YAAY,CACrB,SAAS,EAAE,eAAe,EAC1B,OAAO,EAAE,mBAAmB,GAC7B,OAAO,CAAC,oBAAoB,CAAC;IAchC;;;;;;;OAOG;IACU,oBAAoB,CAC7B,SAAS,EAAE,eAAe,EAC1B,gBAAgB,GAAE,QAAQ,GAAG,OAAkB,GAChD,OAAO,CAAC,sBAAsB,CAAC;IAuBlC;;;OAGG;IACU,cAAc,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,oBAAoB,CAAC;IAgC5E,OAAO,CAAC,mBAAmB;IAa3B;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,mBAAmB;IAoB3B;;;;;;;OAOG;IACH,OAAO,CAAC,qBAAqB;IAyC7B,OAAO,CAAC,YAAY;yCAnQX,uBAAuB;6CAAvB,uBAAuB;CAsQnC"}