dsl-to-sql 1.0.0-fabric-1p-development.1

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 (225) hide show
  1. package/.turbo/turbo-build.log +9 -0
  2. package/.turbo/turbo-check-types.log +4 -0
  3. package/.turbo/turbo-lint.log +126 -0
  4. package/README.md +45 -0
  5. package/coverage/base.css +224 -0
  6. package/coverage/block-navigation.js +87 -0
  7. package/coverage/favicon.png +0 -0
  8. package/coverage/index.html +236 -0
  9. package/coverage/prettify.css +1 -0
  10. package/coverage/prettify.js +2 -0
  11. package/coverage/sort-arrow-sprite.png +0 -0
  12. package/coverage/sorter.js +210 -0
  13. package/coverage/src/constants.ts.html +826 -0
  14. package/coverage/src/database_types.ts.html +136 -0
  15. package/coverage/src/epm-query-builder/EpmQueryBuilder.ts.html +166 -0
  16. package/coverage/src/epm-query-builder/base/BaseAdvancedAggregations.ts.html +568 -0
  17. package/coverage/src/epm-query-builder/base/BaseAnalyticalFunctions.ts.html +694 -0
  18. package/coverage/src/epm-query-builder/base/BaseCTEGenerator.ts.html +1459 -0
  19. package/coverage/src/epm-query-builder/base/BaseCountQueryBuilder.ts.html +400 -0
  20. package/coverage/src/epm-query-builder/base/BaseMeasureBuilder.ts.html +295 -0
  21. package/coverage/src/epm-query-builder/base/BaseOrderBuilder.ts.html +670 -0
  22. package/coverage/src/epm-query-builder/base/BasePaginationBuilder.ts.html +364 -0
  23. package/coverage/src/epm-query-builder/base/BaseQueryBuilder.ts.html +238 -0
  24. package/coverage/src/epm-query-builder/base/BaseRollupBuilder.ts.html +532 -0
  25. package/coverage/src/epm-query-builder/base/BaseSqlBuilder.ts.html +601 -0
  26. package/coverage/src/epm-query-builder/base/BaseSuperFilterBuilder.ts.html +1966 -0
  27. package/coverage/src/epm-query-builder/base/BaseUtilities.ts.html +1798 -0
  28. package/coverage/src/epm-query-builder/base/ColumnRefUtils.ts.html +211 -0
  29. package/coverage/src/epm-query-builder/base/RelationshipResolver.ts.html +706 -0
  30. package/coverage/src/epm-query-builder/base/SharedFilterBuilder.ts.html +1717 -0
  31. package/coverage/src/epm-query-builder/base/index.html +326 -0
  32. package/coverage/src/epm-query-builder/constants/Aggregations.ts.html +133 -0
  33. package/coverage/src/epm-query-builder/constants/Database.ts.html +103 -0
  34. package/coverage/src/epm-query-builder/constants/Source.ts.html +106 -0
  35. package/coverage/src/epm-query-builder/constants/index.html +146 -0
  36. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbAdvancedAggregations.ts.html +286 -0
  37. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbJoinBuilder.ts.html +280 -0
  38. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbMeasureBuilder.ts.html +1924 -0
  39. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbOrderBuilder.ts.html +769 -0
  40. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbPaginationBuilder.ts.html +643 -0
  41. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbQueryBuilder.ts.html +2644 -0
  42. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbRollupBuilder.ts.html +478 -0
  43. package/coverage/src/epm-query-builder/dialects/duckdb/DuckDbSuperFilterBuilder.ts.html +1195 -0
  44. package/coverage/src/epm-query-builder/dialects/duckdb/index.html +221 -0
  45. package/coverage/src/epm-query-builder/errors/QueryBuilderErrors.ts.html +280 -0
  46. package/coverage/src/epm-query-builder/errors/index.html +116 -0
  47. package/coverage/src/epm-query-builder/index.html +116 -0
  48. package/coverage/src/epm-query-builder/interfaces/IDatabaseQueryBuilder.ts.html +100 -0
  49. package/coverage/src/epm-query-builder/interfaces/index.html +131 -0
  50. package/coverage/src/epm-query-builder/interfaces/index.ts.html +88 -0
  51. package/coverage/src/epm-query-builder/utils/format.ts.html +151 -0
  52. package/coverage/src/epm-query-builder/utils/index.html +146 -0
  53. package/coverage/src/epm-query-builder/utils/sql.ts.html +247 -0
  54. package/coverage/src/epm-query-builder/utils/validation.ts.html +124 -0
  55. package/coverage/src/epm-query-builder/validation/QueryOptionsValidator.ts.html +631 -0
  56. package/coverage/src/epm-query-builder/validation/SqlQueryValidator.ts.html +475 -0
  57. package/coverage/src/epm-query-builder/validation/index.html +131 -0
  58. package/coverage/src/filters/FilterConditionBuilder.ts.html +427 -0
  59. package/coverage/src/filters/filter-types.ts.html +484 -0
  60. package/coverage/src/filters/index.html +131 -0
  61. package/coverage/src/index.html +176 -0
  62. package/coverage/src/index.ts.html +103 -0
  63. package/coverage/src/js-lib/JsToSqlParser.ts.html +736 -0
  64. package/coverage/src/js-lib/ParseContext.ts.html +532 -0
  65. package/coverage/src/js-lib/db/azuresql/AzureSqlCallExpressionVisitor.ts.html +196 -0
  66. package/coverage/src/js-lib/db/azuresql/index.html +116 -0
  67. package/coverage/src/js-lib/db/base/ArrayExpressionVisitor.ts.html +133 -0
  68. package/coverage/src/js-lib/db/base/AssignmentExpressionVisitor.ts.html +187 -0
  69. package/coverage/src/js-lib/db/base/BinaryExpressionVisitor.ts.html +223 -0
  70. package/coverage/src/js-lib/db/base/CallExpressionVisitor.ts.html +5479 -0
  71. package/coverage/src/js-lib/db/base/IdentifierVisitor.ts.html +283 -0
  72. package/coverage/src/js-lib/db/base/LiteralVisitor.ts.html +199 -0
  73. package/coverage/src/js-lib/db/base/MemberExpressionVisitor.ts.html +193 -0
  74. package/coverage/src/js-lib/db/base/ProgramVisitor.ts.html +139 -0
  75. package/coverage/src/js-lib/db/base/UnaryExpressionVisitor.ts.html +181 -0
  76. package/coverage/src/js-lib/db/base/VisitorInterface.ts.html +103 -0
  77. package/coverage/src/js-lib/db/base/index.html +251 -0
  78. package/coverage/src/js-lib/db/bigquery/BigQueryCallExpressionVisitor.ts.html +1747 -0
  79. package/coverage/src/js-lib/db/bigquery/index.html +116 -0
  80. package/coverage/src/js-lib/db/commonTransforms.ts.html +2074 -0
  81. package/coverage/src/js-lib/db/databricks/DatabricksCallExpressionVisitor.ts.html +1303 -0
  82. package/coverage/src/js-lib/db/databricks/index.html +116 -0
  83. package/coverage/src/js-lib/db/fabricsql/FabricSqlCallExpressionVisitor.ts.html +196 -0
  84. package/coverage/src/js-lib/db/fabricsql/index.html +116 -0
  85. package/coverage/src/js-lib/db/fabricwarehouse/FabricWarehouseCallExpressionVisitor.ts.html +292 -0
  86. package/coverage/src/js-lib/db/fabricwarehouse/index.html +116 -0
  87. package/coverage/src/js-lib/db/index.html +116 -0
  88. package/coverage/src/js-lib/db/postgresql/PostgreSqlCallExpressionVisitor.ts.html +985 -0
  89. package/coverage/src/js-lib/db/postgresql/index.html +116 -0
  90. package/coverage/src/js-lib/db/redshift/RedshiftCallExpressionVisitor.ts.html +685 -0
  91. package/coverage/src/js-lib/db/redshift/index.html +116 -0
  92. package/coverage/src/js-lib/db/sample/SampleCallExpressionVisitor.ts.html +196 -0
  93. package/coverage/src/js-lib/db/sample/index.html +116 -0
  94. package/coverage/src/js-lib/db/snowflake/SnowflakeCallExpressionVisitor.ts.html +1447 -0
  95. package/coverage/src/js-lib/db/snowflake/index.html +116 -0
  96. package/coverage/src/js-lib/db/validator/FormulaValidator.ts.html +4162 -0
  97. package/coverage/src/js-lib/db/validator/index.html +116 -0
  98. package/coverage/src/js-lib/index.html +131 -0
  99. package/coverage/src/js-lib/objects/BaseObject.ts.html +169 -0
  100. package/coverage/src/js-lib/objects/DateObject.ts.html +169 -0
  101. package/coverage/src/js-lib/objects/PctObject.ts.html +178 -0
  102. package/coverage/src/js-lib/objects/index.html +146 -0
  103. package/coverage/src/query-builder/PaginationBuilder.ts.html +142 -0
  104. package/coverage/src/query-builder/QueryBuilder.ts.html +3118 -0
  105. package/coverage/src/query-builder/SuperFilterBuilder.ts.html +1969 -0
  106. package/coverage/src/query-builder/index.html +146 -0
  107. package/coverage/src/runtime_var.ts.html +109 -0
  108. package/coverage/src/sql-lib/binary_expr.ts.html +133 -0
  109. package/coverage/src/sql-lib/case.ts.html +133 -0
  110. package/coverage/src/sql-lib/column.ts.html +139 -0
  111. package/coverage/src/sql-lib/else.ts.html +124 -0
  112. package/coverage/src/sql-lib/function.ts.html +112 -0
  113. package/coverage/src/sql-lib/index.html +251 -0
  114. package/coverage/src/sql-lib/join.ts.html +127 -0
  115. package/coverage/src/sql-lib/literal.ts.html +130 -0
  116. package/coverage/src/sql-lib/select.ts.html +547 -0
  117. package/coverage/src/sql-lib/unary_expr.ts.html +112 -0
  118. package/coverage/src/sql-lib/when.ts.html +130 -0
  119. package/coverage/src/sql_query_gen.ts.html +535 -0
  120. package/coverage/src/superFilter/DateFilterFactory.ts.html +625 -0
  121. package/coverage/src/superFilter/dateFunction.ts.html +193 -0
  122. package/coverage/src/superFilter/index.html +131 -0
  123. package/coverage/src/utils.ts.html +571 -0
  124. package/dist/index.cjs +8440 -0
  125. package/dist/index.d.cts +927 -0
  126. package/dist/index.d.cts.map +1 -0
  127. package/dist/index.d.ts +927 -0
  128. package/dist/index.d.ts.map +1 -0
  129. package/dist/index.js +8387 -0
  130. package/dist/index.js.map +1 -0
  131. package/eslint.config.js +4 -0
  132. package/jest.config.ts +44 -0
  133. package/package.json +45 -0
  134. package/src/constants.ts +247 -0
  135. package/src/epm-query-builder/EpmQueryBuilder.ts +27 -0
  136. package/src/epm-query-builder/base/BaseAdvancedAggregations.ts +161 -0
  137. package/src/epm-query-builder/base/BaseAnalyticalFunctions.ts +203 -0
  138. package/src/epm-query-builder/base/BaseCTEGenerator.ts +458 -0
  139. package/src/epm-query-builder/base/BaseCountQueryBuilder.ts +105 -0
  140. package/src/epm-query-builder/base/BaseMeasureBuilder.ts +87 -0
  141. package/src/epm-query-builder/base/BaseOrderBuilder.ts +195 -0
  142. package/src/epm-query-builder/base/BasePaginationBuilder.ts +93 -0
  143. package/src/epm-query-builder/base/BaseQueryBuilder.ts +51 -0
  144. package/src/epm-query-builder/base/BaseRollupBuilder.ts +149 -0
  145. package/src/epm-query-builder/base/BaseSqlBuilder.ts +172 -0
  146. package/src/epm-query-builder/base/BaseSuperFilterBuilder.ts +627 -0
  147. package/src/epm-query-builder/base/BaseUtilities.ts +571 -0
  148. package/src/epm-query-builder/base/ColumnRefUtils.ts +42 -0
  149. package/src/epm-query-builder/base/RelationshipResolver.ts +207 -0
  150. package/src/epm-query-builder/base/SharedFilterBuilder.ts +544 -0
  151. package/src/epm-query-builder/constants/Aggregations.ts +16 -0
  152. package/src/epm-query-builder/constants/Database.ts +6 -0
  153. package/src/epm-query-builder/constants/Source.ts +7 -0
  154. package/src/epm-query-builder/dialects/duckdb/DuckDbAdvancedAggregations.ts +67 -0
  155. package/src/epm-query-builder/dialects/duckdb/DuckDbJoinBuilder.ts +65 -0
  156. package/src/epm-query-builder/dialects/duckdb/DuckDbMeasureBuilder.ts +626 -0
  157. package/src/epm-query-builder/dialects/duckdb/DuckDbOrderBuilder.ts +228 -0
  158. package/src/epm-query-builder/dialects/duckdb/DuckDbPaginationBuilder.ts +186 -0
  159. package/src/epm-query-builder/dialects/duckdb/DuckDbQueryBuilder.ts +853 -0
  160. package/src/epm-query-builder/dialects/duckdb/DuckDbRollupBuilder.ts +131 -0
  161. package/src/epm-query-builder/dialects/duckdb/DuckDbSuperFilterBuilder.ts +370 -0
  162. package/src/epm-query-builder/errors/QueryBuilderErrors.ts +65 -0
  163. package/src/epm-query-builder/interfaces/IDatabaseQueryBuilder.ts +5 -0
  164. package/src/epm-query-builder/interfaces/index.ts +1 -0
  165. package/src/epm-query-builder/types/query-builder-types.d.ts +289 -0
  166. package/src/epm-query-builder/utils/format.ts +22 -0
  167. package/src/epm-query-builder/utils/sql.ts +54 -0
  168. package/src/epm-query-builder/utils/validation.ts +13 -0
  169. package/src/epm-query-builder/validation/QueryOptionsValidator.ts +182 -0
  170. package/src/epm-query-builder/validation/SqlQueryValidator.ts +130 -0
  171. package/src/filters/FilterConditionBuilder.ts +114 -0
  172. package/src/filters/filter-types.ts +133 -0
  173. package/src/index.ts +10 -0
  174. package/src/js-lib/JsToSqlParser.ts +217 -0
  175. package/src/js-lib/ParseContext.ts +149 -0
  176. package/src/js-lib/db/base/ArrayExpressionVisitor.ts +16 -0
  177. package/src/js-lib/db/base/AssignmentExpressionVisitor.ts +34 -0
  178. package/src/js-lib/db/base/BinaryExpressionVisitor.ts +46 -0
  179. package/src/js-lib/db/base/CallExpressionVisitor.ts +1798 -0
  180. package/src/js-lib/db/base/IdentifierVisitor.ts +66 -0
  181. package/src/js-lib/db/base/LiteralVisitor.ts +38 -0
  182. package/src/js-lib/db/base/MemberExpressionVisitor.ts +36 -0
  183. package/src/js-lib/db/base/ProgramVisitor.ts +18 -0
  184. package/src/js-lib/db/base/UnaryExpressionVisitor.ts +32 -0
  185. package/src/js-lib/db/base/VisitorInterface.ts +6 -0
  186. package/src/js-lib/db/validator/FormulaValidator.ts +1235 -0
  187. package/src/js-lib/objects/BaseObject.ts +28 -0
  188. package/src/js-lib/objects/DateObject.ts +28 -0
  189. package/src/js-lib/objects/PctObject.ts +31 -0
  190. package/src/query-builder/PaginationBuilder.ts +19 -0
  191. package/src/query-builder/QueryBuilder.ts +1035 -0
  192. package/src/query-builder/SuperFilterBuilder.ts +628 -0
  193. package/src/runtime_var.ts +8 -0
  194. package/src/sql-lib/binary_expr.ts +16 -0
  195. package/src/sql-lib/case.ts +16 -0
  196. package/src/sql-lib/column.ts +18 -0
  197. package/src/sql-lib/else.ts +13 -0
  198. package/src/sql-lib/function.ts +9 -0
  199. package/src/sql-lib/join.ts +14 -0
  200. package/src/sql-lib/literal.ts +15 -0
  201. package/src/sql-lib/select.ts +154 -0
  202. package/src/sql-lib/unary_expr.ts +9 -0
  203. package/src/sql-lib/when.ts +15 -0
  204. package/src/sql-types.d.ts +565 -0
  205. package/src/sql_query_gen.ts +150 -0
  206. package/src/superFilter/DateFilterFactory.ts +180 -0
  207. package/src/superFilter/dateFunction.ts +36 -0
  208. package/src/utils.ts +354 -0
  209. package/test-output/report/junit.xml +329 -0
  210. package/tests/JsToSqlParser.test.ts +163 -0
  211. package/tests/QueryBuilder.test.ts +1320 -0
  212. package/tests/js-lib/CallExpressionVisitor.test.ts +820 -0
  213. package/tests/mocks/MockQueryResolver.ts +14 -0
  214. package/tests/sanity.test.ts +146 -0
  215. package/tests/sql-lib/binary_expr.test.ts +75 -0
  216. package/tests/sql-lib/case.test.ts +117 -0
  217. package/tests/sql-lib/column.test.ts +87 -0
  218. package/tests/sql-lib/else.test.ts +56 -0
  219. package/tests/sql-lib/function.test.ts +96 -0
  220. package/tests/sql-lib/literal.test.ts +75 -0
  221. package/tests/sql-lib/select.test.ts +245 -0
  222. package/tests/sql-lib/unary_expr.test.ts +32 -0
  223. package/tests/utils.test.ts +13 -0
  224. package/tsconfig.json +24 -0
  225. package/tsdown.config.ts +23 -0
@@ -0,0 +1,207 @@
1
+ import { BaseUtilities } from '@epm-query-builder/base/BaseUtilities';
2
+ import { EpmQueryBuilderOptions, Relationship } from '@epm-query-builder/types/query-builder-types';
3
+ import { SOURCE_TYPES } from '@epm-query-builder/constants/Source';
4
+
5
+ type JoinType = 'LEFT' | 'INNER' | 'FULL OUTER';
6
+
7
+ export interface JoinStep {
8
+ leftTable: string;
9
+ leftColumn: string;
10
+ rightTable: string;
11
+ rightColumn: string;
12
+ joinType: JoinType;
13
+ dedupeRightOn?: string;
14
+ }
15
+
16
+ export interface JoinPlan {
17
+ baseTable: string;
18
+ joins: JoinStep[];
19
+ }
20
+
21
+ interface Edge {
22
+ from: string;
23
+ to: string;
24
+ relation: Relationship;
25
+ }
26
+
27
+ function isActive(rel: Relationship, includeInactive: boolean): boolean {
28
+ if (includeInactive) return true;
29
+ return rel.isActive !== false;
30
+ }
31
+
32
+ function buildEdges(rels: Relationship[], includeInactive: boolean): Edge[] {
33
+ const edges: Edge[] = [];
34
+ for (const r of rels) {
35
+ if (!isActive(r, includeInactive)) continue;
36
+ const fromIsMany = r.fromColumnMultiplicity === '*';
37
+ const toIsMany = r.toColumnMultiplicity === '*';
38
+
39
+ if (fromIsMany && toIsMany) {
40
+ edges.push({ from: r.fromTable, to: r.toTable, relation: r });
41
+ edges.push({ from: r.toTable, to: r.fromTable, relation: r });
42
+ } else if (!fromIsMany && toIsMany) {
43
+ edges.push({ from: r.fromTable, to: r.toTable, relation: r });
44
+ } else if (fromIsMany && !toIsMany) {
45
+ edges.push({ from: r.toTable, to: r.fromTable, relation: r });
46
+ } else {
47
+ edges.push({ from: r.fromTable, to: r.toTable, relation: r });
48
+ edges.push({ from: r.toTable, to: r.fromTable, relation: r });
49
+ }
50
+ }
51
+ return edges;
52
+ }
53
+
54
+ function buildGraph(edges: Edge[]): Map<string, Edge[]> {
55
+ const graph = new Map<string, Edge[]>();
56
+ for (const e of edges) {
57
+ if (!graph.has(e.from)) graph.set(e.from, []);
58
+ graph.get(e.from)!.push(e);
59
+ }
60
+ return graph;
61
+ }
62
+
63
+ function bfsReachable(graph: Map<string, Edge[]>, start: string): Set<string> {
64
+ const visited = new Set<string>();
65
+ if (!start) return visited;
66
+ const q: string[] = [start];
67
+ visited.add(start);
68
+ while (q.length) {
69
+ const u = q.shift() as string;
70
+ const outs = graph.get(u) || [];
71
+ for (const e of outs) {
72
+ if (!visited.has(e.to)) {
73
+ visited.add(e.to);
74
+ q.push(e.to);
75
+ }
76
+ }
77
+ }
78
+ return visited;
79
+ }
80
+
81
+ function findShortestPath(
82
+ graph: Map<string, Edge[]>,
83
+ start: string,
84
+ target: string,
85
+ ): Edge[] | null {
86
+ if (start === target) return [];
87
+ const parent = new Map<string, Edge | null>();
88
+ const q: string[] = [start];
89
+ parent.set(start, null);
90
+
91
+ while (q.length) {
92
+ const u = q.shift() as string;
93
+ const outs = graph.get(u) || [];
94
+ for (const e of outs) {
95
+ if (!parent.has(e.to)) {
96
+ parent.set(e.to, e);
97
+ if (e.to === target) {
98
+ const path: Edge[] = [];
99
+ let cur = target;
100
+ while (cur !== start) {
101
+ const pe = parent.get(cur);
102
+ if (!pe) break;
103
+ path.push(pe);
104
+ cur = pe.from;
105
+ }
106
+ path.reverse();
107
+ return path;
108
+ }
109
+ q.push(e.to);
110
+ }
111
+ }
112
+ }
113
+ return null;
114
+ }
115
+
116
+ function selectBaseTable(requiredTables: string[], edges: Edge[], fallbackBase: string): string {
117
+ const graph = buildGraph(edges);
118
+ if (fallbackBase) {
119
+ const reach = bfsReachable(graph, fallbackBase);
120
+ const allCovered = requiredTables.every((t) => reach.has(t) || t === fallbackBase);
121
+ if (allCovered) return fallbackBase;
122
+ }
123
+
124
+ let best: { table: string; covered: number } = { table: fallbackBase, covered: -1 };
125
+ for (const t of requiredTables) {
126
+ const cover = bfsReachable(graph, t);
127
+ const cnt = requiredTables.filter((x) => cover.has(x) || x === t).length;
128
+ if (cnt > best.covered) {
129
+ best = { table: t, covered: cnt };
130
+ }
131
+ }
132
+ return best.table || fallbackBase;
133
+ }
134
+
135
+ function isManyToMany(r: Relationship): boolean {
136
+ return r.fromColumnMultiplicity === '*' && r.toColumnMultiplicity === '*';
137
+ }
138
+
139
+ function edgeToJoin(leftTable: string, e: Edge, dimensionOnly: boolean): JoinStep {
140
+ const r = e.relation;
141
+ const mm = isManyToMany(r);
142
+ const joinType: JoinType = dimensionOnly ? 'FULL OUTER' : 'LEFT';
143
+ if (e.from === r.fromTable && e.to === r.toTable) {
144
+ return {
145
+ leftTable,
146
+ leftColumn: r.fromColumn,
147
+ rightTable: r.toTable,
148
+ rightColumn: r.toColumn,
149
+ joinType,
150
+ dedupeRightOn: dimensionOnly && mm ? r.toColumn : undefined,
151
+ };
152
+ }
153
+ return {
154
+ leftTable,
155
+ leftColumn: r.toColumn,
156
+ rightTable: r.fromTable,
157
+ rightColumn: r.fromColumn,
158
+ joinType,
159
+ dedupeRightOn: dimensionOnly && mm ? r.fromColumn : undefined,
160
+ };
161
+ }
162
+
163
+ export function resolveJoinPlan(options: EpmQueryBuilderOptions): JoinPlan {
164
+ const relationships = options.relationships || [];
165
+ if (relationships.length === 0) {
166
+ return { baseTable: BaseUtilities.getDefaultTableName(options), joins: [] };
167
+ }
168
+
169
+ const requiredTables = BaseUtilities.extractUniqueTableNames(options);
170
+
171
+ const dimensionOnly = !BaseUtilities.hasMeasuresInOptions(options);
172
+ const source = BaseUtilities.getSource(options);
173
+ const includeInactive = source === SOURCE_TYPES.XMLA && dimensionOnly;
174
+
175
+ const edges = buildEdges(relationships, includeInactive);
176
+ if (requiredTables.length <= 1 || edges.length === 0) {
177
+ return { baseTable: BaseUtilities.getDefaultTableName(options), joins: [] };
178
+ }
179
+
180
+ const base = selectBaseTable(requiredTables, edges, BaseUtilities.getDefaultTableName(options));
181
+ const graph = buildGraph(edges);
182
+
183
+ const usedPairs = new Set<string>();
184
+ const joins: JoinStep[] = [];
185
+
186
+ for (const t of requiredTables) {
187
+ if (t === base) continue;
188
+ const path = findShortestPath(graph, base, t);
189
+ if (!path) {
190
+ throw new Error(`No relationship path found from ${base} to ${t}`);
191
+ }
192
+ let currentLeft = base;
193
+ for (const edge of path) {
194
+ const key = `${edge.from}->${edge.to}`;
195
+ if (usedPairs.has(key)) {
196
+ currentLeft = edge.to;
197
+ continue;
198
+ }
199
+ const step = edgeToJoin(currentLeft, edge, dimensionOnly);
200
+ joins.push(step);
201
+ usedPairs.add(key);
202
+ currentLeft = edge.to;
203
+ }
204
+ }
205
+
206
+ return { baseTable: base, joins };
207
+ }