@unispechq/unispec-core 0.2.13 → 0.3.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 (204) hide show
  1. package/README.md +85 -160
  2. package/dist/cache/cache-factory.d.ts +31 -0
  3. package/dist/cache/cache-factory.js +65 -0
  4. package/dist/cache/cache-manager.d.ts +62 -0
  5. package/dist/cache/cache-manager.js +122 -0
  6. package/dist/cache/constants.d.ts +21 -0
  7. package/dist/cache/constants.js +22 -0
  8. package/dist/cache/hash-utils.d.ts +21 -0
  9. package/dist/cache/hash-utils.js +35 -0
  10. package/dist/cache/hashing.d.ts +19 -0
  11. package/dist/cache/hashing.js +197 -0
  12. package/dist/cache/index.d.ts +6 -0
  13. package/dist/cache/index.js +10 -0
  14. package/dist/cache/lru-cache.d.ts +56 -0
  15. package/dist/cache/lru-cache.js +161 -0
  16. package/dist/cache/types.d.ts +24 -0
  17. package/dist/cache/types.js +4 -0
  18. package/dist/cjs/cache/cache-factory.js +72 -0
  19. package/dist/cjs/cache/cache-manager.js +126 -0
  20. package/dist/cjs/cache/constants.js +25 -0
  21. package/dist/cjs/cache/hash-utils.js +41 -0
  22. package/dist/cjs/cache/hashing.js +236 -0
  23. package/dist/cjs/cache/index.js +26 -0
  24. package/dist/cjs/cache/lru-cache.js +165 -0
  25. package/dist/cjs/cache/types.js +5 -0
  26. package/dist/cjs/diff/annotators.js +132 -0
  27. package/dist/cjs/diff/change-reports.js +369 -0
  28. package/dist/cjs/diff/core.js +158 -0
  29. package/dist/cjs/diff/enhanced-diff.js +79 -0
  30. package/dist/cjs/diff/impact-strategies-refactored.js +230 -0
  31. package/dist/cjs/diff/impact-strategies.js +219 -0
  32. package/dist/cjs/diff/index.js +25 -247
  33. package/dist/cjs/diff/metrics-calculator.js +69 -0
  34. package/dist/cjs/diff/risk-calculator.js +58 -0
  35. package/dist/cjs/diff/suggestion-generator.js +78 -0
  36. package/dist/cjs/diff/types.js +11 -0
  37. package/dist/cjs/errors/base-error.js +33 -0
  38. package/dist/cjs/errors/config-error.js +11 -0
  39. package/dist/cjs/errors/error-factory.js +48 -0
  40. package/dist/cjs/errors/index.js +19 -0
  41. package/dist/cjs/errors/loader-error.js +11 -0
  42. package/dist/cjs/errors/reference-error.js +11 -0
  43. package/dist/cjs/errors/schema-error.js +11 -0
  44. package/dist/cjs/errors/security-error.js +11 -0
  45. package/dist/cjs/errors/semantic-error.js +11 -0
  46. package/dist/cjs/generated-schemas.js +2100 -0
  47. package/dist/cjs/index.js +41 -5
  48. package/dist/cjs/loader/index.js +11 -75
  49. package/dist/cjs/loader/security-validator.js +53 -0
  50. package/dist/cjs/loader/types.js +11 -0
  51. package/dist/cjs/loader/unispec-loader.js +84 -0
  52. package/dist/cjs/loader/yaml-loader.js +76 -0
  53. package/dist/cjs/normalizer/core.js +32 -0
  54. package/dist/cjs/normalizer/graphql-normalizer.js +67 -0
  55. package/dist/cjs/normalizer/index.js +5 -119
  56. package/dist/cjs/normalizer/rest-normalizer.js +51 -0
  57. package/dist/cjs/normalizer/types.js +2 -0
  58. package/dist/cjs/normalizer/utils.js +33 -0
  59. package/dist/cjs/normalizer/websocket-normalizer.js +81 -0
  60. package/dist/cjs/optimizer/core.js +115 -0
  61. package/dist/cjs/optimizer/index.js +17 -0
  62. package/dist/cjs/optimizer/optimization-functions.js +185 -0
  63. package/dist/cjs/optimizer/types.js +2 -0
  64. package/dist/cjs/optimizer/utils.js +32 -0
  65. package/dist/cjs/schemas/dedupe.js +100 -0
  66. package/dist/cjs/schemas/index.js +12 -132
  67. package/dist/cjs/schemas/resolver.js +41 -0
  68. package/dist/cjs/schemas/utils.js +53 -0
  69. package/dist/cjs/types/index.js +0 -1
  70. package/dist/cjs/validator/ajv-validator.js +82 -0
  71. package/dist/cjs/validator/config-validator-main.js +34 -0
  72. package/dist/cjs/validator/config-validator.js +17 -0
  73. package/dist/cjs/validator/index.js +21 -108
  74. package/dist/cjs/validator/object-traversal.js +112 -0
  75. package/dist/cjs/validator/reference-validator.js +233 -0
  76. package/dist/cjs/validator/schema-references.js +116 -0
  77. package/dist/cjs/validator/semantic-validator.js +328 -0
  78. package/dist/cjs/validator/tests-validator.js +16 -0
  79. package/dist/cjs/validator/types.js +2 -0
  80. package/dist/cjs/validator/unispec-validator.js +84 -0
  81. package/dist/cjs/validator/validator-factory.js +77 -0
  82. package/dist/cjs/versions.js +147 -0
  83. package/dist/diff/annotators.d.ts +4 -0
  84. package/dist/diff/annotators.js +127 -0
  85. package/dist/diff/change-reports.d.ts +37 -0
  86. package/dist/diff/change-reports.js +366 -0
  87. package/dist/diff/core.d.ts +26 -0
  88. package/dist/diff/core.js +155 -0
  89. package/dist/diff/enhanced-diff.d.ts +51 -0
  90. package/dist/diff/enhanced-diff.js +76 -0
  91. package/dist/diff/impact-strategies-refactored.d.ts +69 -0
  92. package/dist/diff/impact-strategies-refactored.js +223 -0
  93. package/dist/diff/impact-strategies.d.ts +41 -0
  94. package/dist/diff/impact-strategies.js +212 -0
  95. package/dist/diff/index.d.ts +8 -34
  96. package/dist/diff/index.js +11 -246
  97. package/dist/diff/metrics-calculator.d.ts +23 -0
  98. package/dist/diff/metrics-calculator.js +65 -0
  99. package/dist/diff/risk-calculator.d.ts +23 -0
  100. package/dist/diff/risk-calculator.js +55 -0
  101. package/dist/diff/suggestion-generator.d.ts +18 -0
  102. package/dist/diff/suggestion-generator.js +74 -0
  103. package/dist/diff/types.d.ts +24 -0
  104. package/dist/diff/types.js +8 -0
  105. package/dist/errors/base-error.d.ts +20 -0
  106. package/dist/errors/base-error.js +29 -0
  107. package/dist/errors/config-error.d.ts +4 -0
  108. package/dist/errors/config-error.js +7 -0
  109. package/dist/errors/error-factory.d.ts +22 -0
  110. package/dist/errors/error-factory.js +45 -0
  111. package/dist/errors/index.d.ts +8 -0
  112. package/dist/errors/index.js +8 -0
  113. package/dist/errors/loader-error.d.ts +4 -0
  114. package/dist/errors/loader-error.js +7 -0
  115. package/dist/errors/reference-error.d.ts +4 -0
  116. package/dist/errors/reference-error.js +7 -0
  117. package/dist/errors/schema-error.d.ts +4 -0
  118. package/dist/errors/schema-error.js +7 -0
  119. package/dist/errors/security-error.d.ts +4 -0
  120. package/dist/errors/security-error.js +7 -0
  121. package/dist/errors/semantic-error.d.ts +4 -0
  122. package/dist/errors/semantic-error.js +7 -0
  123. package/dist/generated-schemas.d.ts +2073 -0
  124. package/dist/generated-schemas.js +2097 -0
  125. package/dist/index.cjs +41 -5
  126. package/dist/index.d.ts +11 -5
  127. package/dist/index.js +41 -5
  128. package/dist/loader/index.d.ts +5 -12
  129. package/dist/loader/index.js +5 -41
  130. package/dist/loader/security-validator.d.ts +5 -0
  131. package/dist/loader/security-validator.js +50 -0
  132. package/dist/loader/types.d.ts +30 -0
  133. package/dist/loader/types.js +8 -0
  134. package/dist/loader/unispec-loader.d.ts +10 -0
  135. package/dist/loader/unispec-loader.js +81 -0
  136. package/dist/loader/yaml-loader.d.ts +10 -0
  137. package/dist/loader/yaml-loader.js +39 -0
  138. package/dist/normalizer/core.d.ts +24 -0
  139. package/dist/normalizer/core.js +29 -0
  140. package/dist/normalizer/graphql-normalizer.d.ts +8 -0
  141. package/dist/normalizer/graphql-normalizer.js +64 -0
  142. package/dist/normalizer/index.d.ts +2 -25
  143. package/dist/normalizer/index.js +3 -118
  144. package/dist/normalizer/rest-normalizer.d.ts +8 -0
  145. package/dist/normalizer/rest-normalizer.js +48 -0
  146. package/dist/normalizer/types.d.ts +7 -0
  147. package/dist/normalizer/types.js +1 -0
  148. package/dist/normalizer/utils.d.ts +14 -0
  149. package/dist/normalizer/utils.js +29 -0
  150. package/dist/normalizer/websocket-normalizer.d.ts +8 -0
  151. package/dist/normalizer/websocket-normalizer.js +78 -0
  152. package/dist/optimizer/core.d.ts +17 -0
  153. package/dist/optimizer/core.js +111 -0
  154. package/dist/optimizer/index.d.ts +4 -0
  155. package/dist/optimizer/index.js +7 -0
  156. package/dist/optimizer/optimization-functions.d.ts +32 -0
  157. package/dist/optimizer/optimization-functions.js +179 -0
  158. package/dist/optimizer/types.d.ts +28 -0
  159. package/dist/optimizer/types.js +1 -0
  160. package/dist/optimizer/utils.d.ts +7 -0
  161. package/dist/optimizer/utils.js +29 -0
  162. package/dist/schemas/dedupe.d.ts +9 -0
  163. package/dist/schemas/dedupe.js +97 -0
  164. package/dist/schemas/index.d.ts +3 -4
  165. package/dist/schemas/index.js +6 -129
  166. package/dist/schemas/resolver.d.ts +19 -0
  167. package/dist/schemas/resolver.js +37 -0
  168. package/dist/schemas/utils.d.ts +20 -0
  169. package/dist/schemas/utils.js +49 -0
  170. package/dist/types/index.d.ts +279 -41
  171. package/dist/types/index.js +0 -1
  172. package/dist/validator/ajv-validator.d.ts +15 -0
  173. package/dist/validator/ajv-validator.js +75 -0
  174. package/dist/validator/config-validator-main.d.ts +10 -0
  175. package/dist/validator/config-validator-main.js +31 -0
  176. package/dist/validator/config-validator.d.ts +5 -0
  177. package/dist/validator/config-validator.js +14 -0
  178. package/dist/validator/index.d.ts +10 -23
  179. package/dist/validator/index.js +11 -103
  180. package/dist/validator/object-traversal.d.ts +52 -0
  181. package/dist/validator/object-traversal.js +104 -0
  182. package/dist/validator/reference-validator.d.ts +31 -0
  183. package/dist/validator/reference-validator.js +230 -0
  184. package/dist/validator/schema-references.d.ts +23 -0
  185. package/dist/validator/schema-references.js +111 -0
  186. package/dist/validator/semantic-validator.d.ts +26 -0
  187. package/dist/validator/semantic-validator.js +325 -0
  188. package/dist/validator/tests-validator.d.ts +9 -0
  189. package/dist/validator/tests-validator.js +13 -0
  190. package/dist/validator/types.d.ts +29 -0
  191. package/dist/validator/types.js +1 -0
  192. package/dist/validator/unispec-validator.d.ts +15 -0
  193. package/dist/validator/unispec-validator.js +81 -0
  194. package/dist/validator/validator-factory.d.ts +10 -0
  195. package/dist/validator/validator-factory.js +73 -0
  196. package/dist/versions.d.ts +10 -0
  197. package/dist/versions.js +143 -0
  198. package/package.json +11 -9
  199. package/dist/cjs/converters/index.js +0 -204
  200. package/dist/cjs/validator/generated-schemas.js +0 -827
  201. package/dist/converters/index.d.ts +0 -14
  202. package/dist/converters/index.js +0 -199
  203. package/dist/validator/generated-schemas.d.ts +0 -842
  204. package/dist/validator/generated-schemas.js +0 -824
@@ -0,0 +1,79 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.diffUniSpecEnhanced = diffUniSpecEnhanced;
4
+ const core_1 = require("./core.js");
5
+ const impact_strategies_refactored_1 = require("./impact-strategies-refactored.js");
6
+ const metrics_calculator_1 = require("./metrics-calculator.js");
7
+ const risk_calculator_1 = require("./risk-calculator.js");
8
+ const suggestion_generator_1 = require("./suggestion-generator.js");
9
+ /**
10
+ * Service instances for better separation of concerns.
11
+ */
12
+ const suggestionGenerator = new suggestion_generator_1.SuggestionGeneratorService();
13
+ const metricsCalculator = new metrics_calculator_1.MetricsCalculatorService();
14
+ /**
15
+ * Generate migration suggestions based on change impact.
16
+ * @deprecated Use SuggestionGeneratorService.generateSuggestions instead
17
+ */
18
+ function generateSuggestions(change, impact) {
19
+ return suggestionGenerator.generateSuggestions(change, impact);
20
+ }
21
+ /**
22
+ * Calculate compatibility metrics from enhanced changes.
23
+ * @deprecated Use MetricsCalculatorService.calculateMetrics instead
24
+ */
25
+ function calculateMetrics(changes) {
26
+ return metricsCalculator.calculateMetrics(changes);
27
+ }
28
+ /**
29
+ * Perform enhanced diff analysis with impact assessment and compatibility metrics.
30
+ *
31
+ * This function extends the basic diff functionality with:
32
+ * - Detailed impact analysis for each change
33
+ * - Compatibility metrics
34
+ * - Migration suggestions
35
+ * - Risk assessment
36
+ *
37
+ * @param oldDoc - The previous version of the UniSpec document
38
+ * @param newDoc - The current version of the UniSpec document
39
+ * @returns Enhanced diff result with comprehensive analysis
40
+ */
41
+ function diffUniSpecEnhanced(oldDoc, newDoc) {
42
+ // Get basic diff
43
+ const basicDiff = (0, core_1.diffUniSpec)(oldDoc, newDoc);
44
+ // Enhance each change with impact analysis using strategy pattern
45
+ const enhancedChanges = basicDiff.changes.map((change) => {
46
+ // Get appropriate strategy for the protocol
47
+ const strategy = impact_strategies_refactored_1.ImpactAnalysisStrategyFactory.getStrategy(change.protocol);
48
+ // Analyze impact using strategy
49
+ const impact = strategy.analyze(change);
50
+ // Generate suggestions
51
+ const suggestions = generateSuggestions(change, impact);
52
+ // Calculate risk level using calculator
53
+ const riskLevel = risk_calculator_1.RiskLevelCalculator.calculate(impact);
54
+ return {
55
+ ...change,
56
+ impact,
57
+ suggestions,
58
+ riskLevel,
59
+ };
60
+ });
61
+ // Calculate metrics
62
+ const metrics = calculateMetrics(enhancedChanges);
63
+ // Generate summary
64
+ const summary = {
65
+ totalChanges: enhancedChanges.length,
66
+ criticalChanges: enhancedChanges.filter((c) => c.riskLevel === "critical")
67
+ .length,
68
+ highRiskChanges: enhancedChanges.filter((c) => c.riskLevel === "high")
69
+ .length,
70
+ mediumRiskChanges: enhancedChanges.filter((c) => c.riskLevel === "medium")
71
+ .length,
72
+ lowRiskChanges: enhancedChanges.filter((c) => c.riskLevel === "low").length,
73
+ };
74
+ return {
75
+ changes: enhancedChanges,
76
+ metrics,
77
+ summary,
78
+ };
79
+ }
@@ -0,0 +1,230 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ImpactAnalysisStrategyFactory = exports.DefaultImpactStrategy = exports.WebSocketImpactStrategy = exports.GraphQLOperationImpactStrategy = exports.RestRouteImpactStrategy = void 0;
4
+ /**
5
+ * Base class for impact analysis strategies with common functionality.
6
+ */
7
+ class BaseImpactStrategy {
8
+ createDefaultImpact() {
9
+ return {
10
+ backwardCompatibility: "unknown",
11
+ clientImpact: "none",
12
+ serverImpact: "none",
13
+ migrationComplexity: "none",
14
+ };
15
+ }
16
+ setBreakingChange(impact, clientImpact = "high", serverImpact = "medium", migrationComplexity = "moderate") {
17
+ impact.backwardCompatibility = "incompatible";
18
+ impact.clientImpact = clientImpact;
19
+ impact.serverImpact = serverImpact;
20
+ impact.migrationComplexity = migrationComplexity;
21
+ return impact;
22
+ }
23
+ setNonBreakingChange(impact, clientImpact = "none", serverImpact = "low", migrationComplexity = "none") {
24
+ impact.backwardCompatibility = "compatible";
25
+ impact.clientImpact = clientImpact;
26
+ impact.serverImpact = serverImpact;
27
+ impact.migrationComplexity = migrationComplexity;
28
+ return impact;
29
+ }
30
+ }
31
+ /**
32
+ * REST route impact analysis strategy.
33
+ */
34
+ class RestRouteImpactStrategy extends BaseImpactStrategy {
35
+ analyze(change) {
36
+ const impact = this.createDefaultImpact();
37
+ // Route removal is always breaking
38
+ if (this.isRouteRemoval(change)) {
39
+ return this.setBreakingChange(impact, "high", "medium", "complex");
40
+ }
41
+ // Route addition is non-breaking
42
+ if (this.isRouteAddition(change)) {
43
+ return this.setNonBreakingChange(impact, "none", "low", "none");
44
+ }
45
+ // Method change is breaking
46
+ if (this.isMethodChange(change)) {
47
+ return this.setBreakingChange(impact, "high", "medium", "moderate");
48
+ }
49
+ // Path change is breaking
50
+ if (this.isPathChange(change)) {
51
+ return this.setBreakingChange(impact, "high", "medium", "moderate");
52
+ }
53
+ // Required parameter change is breaking
54
+ if (this.isRequiredParameterChange(change)) {
55
+ return this.setBreakingChange(impact, "medium", "low", "simple");
56
+ }
57
+ // Response status code removal is breaking
58
+ if (this.isResponseRemoval(change)) {
59
+ return this.setBreakingChange(impact, "medium", "low", "simple");
60
+ }
61
+ // Adding optional parameter is non-breaking
62
+ if (this.isOptionalParameterAddition(change)) {
63
+ return this.setNonBreakingChange(impact, "none", "low", "none");
64
+ }
65
+ return impact;
66
+ }
67
+ isRouteRemoval(change) {
68
+ return (change.description === "Item removed" &&
69
+ !!change.kind?.includes("route.removed"));
70
+ }
71
+ isRouteAddition(change) {
72
+ return (change.description === "Item added" &&
73
+ !!change.kind?.includes("route.added"));
74
+ }
75
+ isMethodChange(change) {
76
+ return (change.path.includes("/method") && change.description === "Value changed");
77
+ }
78
+ isPathChange(change) {
79
+ return (change.path.includes("/path") && change.description === "Value changed");
80
+ }
81
+ isRequiredParameterChange(change) {
82
+ return (change.path.includes("/required") &&
83
+ change.description === "Value changed");
84
+ }
85
+ isResponseRemoval(change) {
86
+ return (change.path.includes("/responses/") &&
87
+ change.description === "Item removed");
88
+ }
89
+ isOptionalParameterAddition(change) {
90
+ return (change.path.includes("/queryParams/") &&
91
+ change.description === "Item added");
92
+ }
93
+ }
94
+ exports.RestRouteImpactStrategy = RestRouteImpactStrategy;
95
+ /**
96
+ * GraphQL operation impact analysis strategy.
97
+ */
98
+ class GraphQLOperationImpactStrategy extends BaseImpactStrategy {
99
+ analyze(change) {
100
+ const impact = this.createDefaultImpact();
101
+ // Operation removal is breaking
102
+ if (this.isOperationRemoval(change)) {
103
+ return this.setBreakingChange(impact, "high", "medium", "complex");
104
+ }
105
+ // Operation addition is non-breaking
106
+ if (this.isOperationAddition(change)) {
107
+ return this.setNonBreakingChange(impact, "none", "low", "none");
108
+ }
109
+ // Return type change is breaking
110
+ if (this.isReturnTypeChange(change)) {
111
+ return this.setBreakingChange(impact, "high", "medium", "moderate");
112
+ }
113
+ // Adding optional argument is non-breaking
114
+ if (this.isOptionalArgumentAddition(change)) {
115
+ return this.setNonBreakingChange(impact, "none", "low", "none");
116
+ }
117
+ // Making argument required is breaking
118
+ if (this.isRequiredArgumentChange(change)) {
119
+ return this.setBreakingChange(impact, "medium", "low", "simple");
120
+ }
121
+ return impact;
122
+ }
123
+ isOperationRemoval(change) {
124
+ return (change.description === "Item removed" &&
125
+ !!change.kind?.includes("operation.removed"));
126
+ }
127
+ isOperationAddition(change) {
128
+ return (change.description === "Item added" &&
129
+ !!change.kind?.includes("operation.added"));
130
+ }
131
+ isReturnTypeChange(change) {
132
+ return (change.path.includes("/returnType") &&
133
+ change.description === "Value changed");
134
+ }
135
+ isOptionalArgumentAddition(change) {
136
+ return (change.path.includes("/args/") && change.description === "Item added");
137
+ }
138
+ isRequiredArgumentChange(change) {
139
+ return (change.path.includes("/args/") &&
140
+ change.path.includes("/required") &&
141
+ change.description === "Value changed");
142
+ }
143
+ }
144
+ exports.GraphQLOperationImpactStrategy = GraphQLOperationImpactStrategy;
145
+ /**
146
+ * WebSocket impact analysis strategy.
147
+ */
148
+ class WebSocketImpactStrategy extends BaseImpactStrategy {
149
+ analyze(change) {
150
+ const impact = this.createDefaultImpact();
151
+ // Channel removal is breaking
152
+ if (this.isChannelRemoval(change)) {
153
+ return this.setBreakingChange(impact, "high", "medium", "complex");
154
+ }
155
+ // Channel addition is non-breaking
156
+ if (this.isChannelAddition(change)) {
157
+ return this.setNonBreakingChange(impact, "none", "low", "none");
158
+ }
159
+ // Message removal is breaking
160
+ if (this.isMessageRemoval(change)) {
161
+ return this.setBreakingChange(impact, "medium", "low", "moderate");
162
+ }
163
+ // Message addition is non-breaking
164
+ if (this.isMessageAddition(change)) {
165
+ return this.setNonBreakingChange(impact, "none", "low", "none");
166
+ }
167
+ // Message schema change is breaking
168
+ if (this.isMessageSchemaChange(change)) {
169
+ return this.setBreakingChange(impact, "medium", "low", "moderate");
170
+ }
171
+ return impact;
172
+ }
173
+ isChannelRemoval(change) {
174
+ return (change.description === "Item removed" &&
175
+ !!change.kind?.includes("channel.removed"));
176
+ }
177
+ isChannelAddition(change) {
178
+ return (change.description === "Item added" &&
179
+ !!change.kind?.includes("channel.added"));
180
+ }
181
+ isMessageRemoval(change) {
182
+ return (change.description === "Item removed" &&
183
+ !!change.kind?.includes("message.removed"));
184
+ }
185
+ isMessageAddition(change) {
186
+ return (change.description === "Item added" &&
187
+ !!change.kind?.includes("message.added"));
188
+ }
189
+ isMessageSchemaChange(change) {
190
+ return (change.path.includes("/schemaRef") &&
191
+ change.description === "Value changed");
192
+ }
193
+ }
194
+ exports.WebSocketImpactStrategy = WebSocketImpactStrategy;
195
+ /**
196
+ * Default impact analysis strategy for unknown changes.
197
+ */
198
+ class DefaultImpactStrategy extends BaseImpactStrategy {
199
+ analyze(_change) {
200
+ return this.createDefaultImpact();
201
+ }
202
+ }
203
+ exports.DefaultImpactStrategy = DefaultImpactStrategy;
204
+ /**
205
+ * Factory for creating impact analysis strategies.
206
+ */
207
+ const ImpactAnalysisStrategyFactory = {
208
+ strategies: new Map([
209
+ ["rest", new RestRouteImpactStrategy()],
210
+ ["graphql", new GraphQLOperationImpactStrategy()],
211
+ ["websocket", new WebSocketImpactStrategy()],
212
+ ]),
213
+ getStrategy(protocol) {
214
+ if (!protocol) {
215
+ return new DefaultImpactStrategy();
216
+ }
217
+ const strategy = this.strategies.get(protocol);
218
+ return strategy || new DefaultImpactStrategy();
219
+ },
220
+ registerStrategy(protocol, strategy) {
221
+ this.strategies.set(protocol, strategy);
222
+ },
223
+ getAvailableStrategies() {
224
+ return Array.from(this.strategies.keys());
225
+ },
226
+ clearStrategies() {
227
+ this.strategies.clear();
228
+ },
229
+ };
230
+ exports.ImpactAnalysisStrategyFactory = ImpactAnalysisStrategyFactory;
@@ -0,0 +1,219 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ImpactAnalysisStrategyFactory = exports.DefaultImpactStrategy = exports.WebSocketImpactStrategy = exports.GraphQLOperationImpactStrategy = exports.RestRouteImpactStrategy = void 0;
4
+ /**
5
+ * REST route impact analysis strategy.
6
+ */
7
+ class RestRouteImpactStrategy {
8
+ analyze(change) {
9
+ const impact = {
10
+ backwardCompatibility: "unknown",
11
+ clientImpact: "none",
12
+ serverImpact: "none",
13
+ migrationComplexity: "none",
14
+ };
15
+ // Route removal is always breaking
16
+ if (change.description === "Item removed" &&
17
+ change.kind?.includes("route.removed")) {
18
+ impact.backwardCompatibility = "incompatible";
19
+ impact.clientImpact = "high";
20
+ impact.serverImpact = "medium";
21
+ impact.migrationComplexity = "complex";
22
+ }
23
+ // Route addition is non-breaking
24
+ if (change.description === "Item added" &&
25
+ change.kind?.includes("route.added")) {
26
+ impact.backwardCompatibility = "compatible";
27
+ impact.clientImpact = "none";
28
+ impact.serverImpact = "low";
29
+ impact.migrationComplexity = "none";
30
+ }
31
+ // Method change is breaking
32
+ if (change.path.includes("/method") &&
33
+ change.description === "Value changed") {
34
+ impact.backwardCompatibility = "incompatible";
35
+ impact.clientImpact = "high";
36
+ impact.serverImpact = "medium";
37
+ impact.migrationComplexity = "moderate";
38
+ }
39
+ // Path change is breaking
40
+ if (change.path.includes("/path") &&
41
+ change.description === "Value changed") {
42
+ impact.backwardCompatibility = "incompatible";
43
+ impact.clientImpact = "high";
44
+ impact.serverImpact = "medium";
45
+ impact.migrationComplexity = "moderate";
46
+ }
47
+ // Required parameter change is breaking
48
+ if (change.path.includes("/required") &&
49
+ change.description === "Value changed") {
50
+ impact.backwardCompatibility = "incompatible";
51
+ impact.clientImpact = "medium";
52
+ impact.serverImpact = "low";
53
+ impact.migrationComplexity = "simple";
54
+ }
55
+ // Response status code removal is breaking
56
+ if (change.path.includes("/responses/") &&
57
+ change.description === "Item removed") {
58
+ impact.backwardCompatibility = "incompatible";
59
+ impact.clientImpact = "medium";
60
+ impact.serverImpact = "low";
61
+ impact.migrationComplexity = "simple";
62
+ }
63
+ // Adding optional parameter is non-breaking
64
+ if (change.path.includes("/queryParams/") &&
65
+ change.description === "Item added") {
66
+ impact.backwardCompatibility = "compatible";
67
+ impact.clientImpact = "none";
68
+ impact.serverImpact = "low";
69
+ impact.migrationComplexity = "none";
70
+ }
71
+ return impact;
72
+ }
73
+ }
74
+ exports.RestRouteImpactStrategy = RestRouteImpactStrategy;
75
+ /**
76
+ * GraphQL operation impact analysis strategy.
77
+ */
78
+ class GraphQLOperationImpactStrategy {
79
+ analyze(change) {
80
+ const impact = {
81
+ backwardCompatibility: "unknown",
82
+ clientImpact: "none",
83
+ serverImpact: "none",
84
+ migrationComplexity: "none",
85
+ };
86
+ // Operation removal is breaking
87
+ if (change.description === "Item removed" &&
88
+ change.kind?.includes("operation.removed")) {
89
+ impact.backwardCompatibility = "incompatible";
90
+ impact.clientImpact = "high";
91
+ impact.serverImpact = "medium";
92
+ impact.migrationComplexity = "complex";
93
+ }
94
+ // Operation addition is non-breaking
95
+ if (change.description === "Item added" &&
96
+ change.kind?.includes("operation.added")) {
97
+ impact.backwardCompatibility = "compatible";
98
+ impact.clientImpact = "none";
99
+ impact.serverImpact = "low";
100
+ impact.migrationComplexity = "none";
101
+ }
102
+ // Return type change is breaking
103
+ if (change.path.includes("/returnType") &&
104
+ change.description === "Value changed") {
105
+ impact.backwardCompatibility = "incompatible";
106
+ impact.clientImpact = "high";
107
+ impact.serverImpact = "medium";
108
+ impact.migrationComplexity = "moderate";
109
+ }
110
+ // Adding optional argument is non-breaking
111
+ if (change.path.includes("/args/") && change.description === "Item added") {
112
+ impact.backwardCompatibility = "compatible";
113
+ impact.clientImpact = "none";
114
+ impact.serverImpact = "low";
115
+ impact.migrationComplexity = "none";
116
+ }
117
+ // Making argument required is breaking
118
+ if (change.path.includes("/args/") &&
119
+ change.path.includes("/required") &&
120
+ change.description === "Value changed") {
121
+ impact.backwardCompatibility = "incompatible";
122
+ impact.clientImpact = "medium";
123
+ impact.serverImpact = "low";
124
+ impact.migrationComplexity = "simple";
125
+ }
126
+ return impact;
127
+ }
128
+ }
129
+ exports.GraphQLOperationImpactStrategy = GraphQLOperationImpactStrategy;
130
+ /**
131
+ * WebSocket impact analysis strategy.
132
+ */
133
+ class WebSocketImpactStrategy {
134
+ analyze(change) {
135
+ const impact = {
136
+ backwardCompatibility: "unknown",
137
+ clientImpact: "none",
138
+ serverImpact: "none",
139
+ migrationComplexity: "none",
140
+ };
141
+ // Channel removal is breaking
142
+ if (change.description === "Item removed" &&
143
+ change.kind?.includes("channel.removed")) {
144
+ impact.backwardCompatibility = "incompatible";
145
+ impact.clientImpact = "high";
146
+ impact.serverImpact = "medium";
147
+ impact.migrationComplexity = "complex";
148
+ }
149
+ // Channel addition is non-breaking
150
+ if (change.description === "Item added" &&
151
+ change.kind?.includes("channel.added")) {
152
+ impact.backwardCompatibility = "compatible";
153
+ impact.clientImpact = "none";
154
+ impact.serverImpact = "low";
155
+ impact.migrationComplexity = "none";
156
+ }
157
+ // Message removal is breaking
158
+ if (change.description === "Item removed" &&
159
+ change.kind?.includes("message.removed")) {
160
+ impact.backwardCompatibility = "incompatible";
161
+ impact.clientImpact = "medium";
162
+ impact.serverImpact = "low";
163
+ impact.migrationComplexity = "moderate";
164
+ }
165
+ // Message addition is non-breaking
166
+ if (change.description === "Item added" &&
167
+ change.kind?.includes("message.added")) {
168
+ impact.backwardCompatibility = "compatible";
169
+ impact.clientImpact = "none";
170
+ impact.serverImpact = "low";
171
+ impact.migrationComplexity = "none";
172
+ }
173
+ // Message schema change is breaking
174
+ if (change.path.includes("/schemaRef") &&
175
+ change.description === "Value changed") {
176
+ impact.backwardCompatibility = "incompatible";
177
+ impact.clientImpact = "medium";
178
+ impact.serverImpact = "low";
179
+ impact.migrationComplexity = "moderate";
180
+ }
181
+ return impact;
182
+ }
183
+ }
184
+ exports.WebSocketImpactStrategy = WebSocketImpactStrategy;
185
+ /**
186
+ * Default impact analysis strategy for unknown changes.
187
+ */
188
+ class DefaultImpactStrategy {
189
+ analyze(_change) {
190
+ return {
191
+ backwardCompatibility: "unknown",
192
+ clientImpact: "none",
193
+ serverImpact: "none",
194
+ migrationComplexity: "none",
195
+ };
196
+ }
197
+ }
198
+ exports.DefaultImpactStrategy = DefaultImpactStrategy;
199
+ /**
200
+ * Factory for creating impact analysis strategies.
201
+ */
202
+ const ImpactAnalysisStrategyFactory = {
203
+ strategies: new Map([
204
+ ["rest", new RestRouteImpactStrategy()],
205
+ ["graphql", new GraphQLOperationImpactStrategy()],
206
+ ["websocket", new WebSocketImpactStrategy()],
207
+ ]),
208
+ getStrategy(protocol) {
209
+ if (!protocol) {
210
+ return new DefaultImpactStrategy();
211
+ }
212
+ const strategy = this.strategies.get(protocol);
213
+ return strategy || new DefaultImpactStrategy();
214
+ },
215
+ registerStrategy(protocol, strategy) {
216
+ this.strategies.set(protocol, strategy);
217
+ },
218
+ };
219
+ exports.ImpactAnalysisStrategyFactory = ImpactAnalysisStrategyFactory;