@api-client/core 0.19.9 → 0.19.11

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 (523) hide show
  1. package/LICENSE.md +9 -63
  2. package/Testing.md +1 -1
  3. package/bin/test.ts +1 -12
  4. package/build/src/browser.d.ts +0 -15
  5. package/build/src/browser.d.ts.map +1 -1
  6. package/build/src/browser.js +0 -17
  7. package/build/src/browser.js.map +1 -1
  8. package/build/src/decorators/observed.d.ts.map +1 -1
  9. package/build/src/decorators/observed.js +91 -0
  10. package/build/src/decorators/observed.js.map +1 -1
  11. package/build/src/events/EventTypes.d.ts +0 -6
  12. package/build/src/events/EventTypes.d.ts.map +1 -1
  13. package/build/src/events/EventTypes.js +0 -2
  14. package/build/src/events/EventTypes.js.map +1 -1
  15. package/build/src/events/Events.d.ts +0 -2
  16. package/build/src/events/Events.d.ts.map +1 -1
  17. package/build/src/events/Events.js +0 -2
  18. package/build/src/events/Events.js.map +1 -1
  19. package/build/src/index.d.ts +0 -15
  20. package/build/src/index.d.ts.map +1 -1
  21. package/build/src/index.js +0 -17
  22. package/build/src/index.js.map +1 -1
  23. package/build/src/modeling/ApiModel.d.ts +21 -7
  24. package/build/src/modeling/ApiModel.d.ts.map +1 -1
  25. package/build/src/modeling/ApiModel.js +70 -29
  26. package/build/src/modeling/ApiModel.js.map +1 -1
  27. package/build/src/modeling/Bindings.d.ts +3 -3
  28. package/build/src/modeling/Bindings.d.ts.map +1 -1
  29. package/build/src/modeling/Bindings.js.map +1 -1
  30. package/build/src/modeling/DomainAssociation.d.ts +5 -5
  31. package/build/src/modeling/DomainAssociation.d.ts.map +1 -1
  32. package/build/src/modeling/DomainAssociation.js.map +1 -1
  33. package/build/src/modeling/DomainEntity.d.ts +2 -3
  34. package/build/src/modeling/DomainEntity.d.ts.map +1 -1
  35. package/build/src/modeling/DomainEntity.js +1 -1
  36. package/build/src/modeling/DomainEntity.js.map +1 -1
  37. package/build/src/modeling/DomainProperty.d.ts +2 -3
  38. package/build/src/modeling/DomainProperty.d.ts.map +1 -1
  39. package/build/src/modeling/DomainProperty.js +1 -1
  40. package/build/src/modeling/DomainProperty.js.map +1 -1
  41. package/build/src/modeling/DomainValidation.d.ts +1 -1
  42. package/build/src/modeling/DomainValidation.d.ts.map +1 -1
  43. package/build/src/modeling/DomainValidation.js.map +1 -1
  44. package/build/src/modeling/ExposedEntity.d.ts +14 -0
  45. package/build/src/modeling/ExposedEntity.d.ts.map +1 -1
  46. package/build/src/modeling/ExposedEntity.js +59 -6
  47. package/build/src/modeling/ExposedEntity.js.map +1 -1
  48. package/build/src/modeling/actions/Action.d.ts +11 -1
  49. package/build/src/modeling/actions/Action.d.ts.map +1 -1
  50. package/build/src/modeling/actions/Action.js +21 -3
  51. package/build/src/modeling/actions/Action.js.map +1 -1
  52. package/build/src/modeling/actions/CreateAction.d.ts +2 -1
  53. package/build/src/modeling/actions/CreateAction.d.ts.map +1 -1
  54. package/build/src/modeling/actions/CreateAction.js +2 -2
  55. package/build/src/modeling/actions/CreateAction.js.map +1 -1
  56. package/build/src/modeling/actions/DeleteAction.d.ts +2 -1
  57. package/build/src/modeling/actions/DeleteAction.d.ts.map +1 -1
  58. package/build/src/modeling/actions/DeleteAction.js +2 -2
  59. package/build/src/modeling/actions/DeleteAction.js.map +1 -1
  60. package/build/src/modeling/actions/ListAction.d.ts +2 -1
  61. package/build/src/modeling/actions/ListAction.d.ts.map +1 -1
  62. package/build/src/modeling/actions/ListAction.js +2 -2
  63. package/build/src/modeling/actions/ListAction.js.map +1 -1
  64. package/build/src/modeling/actions/ReadAction.d.ts +2 -1
  65. package/build/src/modeling/actions/ReadAction.d.ts.map +1 -1
  66. package/build/src/modeling/actions/ReadAction.js +2 -2
  67. package/build/src/modeling/actions/ReadAction.js.map +1 -1
  68. package/build/src/modeling/actions/SearchAction.d.ts +2 -1
  69. package/build/src/modeling/actions/SearchAction.d.ts.map +1 -1
  70. package/build/src/modeling/actions/SearchAction.js +2 -2
  71. package/build/src/modeling/actions/SearchAction.js.map +1 -1
  72. package/build/src/modeling/actions/UpdateAction.d.ts +2 -1
  73. package/build/src/modeling/actions/UpdateAction.d.ts.map +1 -1
  74. package/build/src/modeling/actions/UpdateAction.js +2 -2
  75. package/build/src/modeling/actions/UpdateAction.js.map +1 -1
  76. package/build/src/modeling/actions/index.d.ts +2 -1
  77. package/build/src/modeling/actions/index.d.ts.map +1 -1
  78. package/build/src/modeling/actions/index.js +7 -7
  79. package/build/src/modeling/actions/index.js.map +1 -1
  80. package/build/src/modeling/ai/types.d.ts +2 -2
  81. package/build/src/modeling/ai/types.d.ts.map +1 -1
  82. package/build/src/modeling/ai/types.js.map +1 -1
  83. package/build/src/{amf → modeling/amf}/DataValueGenerator.d.ts +1 -1
  84. package/build/src/modeling/amf/DataValueGenerator.d.ts.map +1 -0
  85. package/build/src/{amf → modeling/amf}/DataValueGenerator.js +2 -3
  86. package/build/src/modeling/amf/DataValueGenerator.js.map +1 -0
  87. package/build/src/modeling/amf/ShapeGenerator.d.ts +20 -20
  88. package/build/src/modeling/amf/ShapeGenerator.d.ts.map +1 -1
  89. package/build/src/modeling/amf/ShapeGenerator.js +16 -18
  90. package/build/src/modeling/amf/ShapeGenerator.js.map +1 -1
  91. package/build/src/modeling/amf/core_types.d.ts +9 -0
  92. package/build/src/modeling/amf/core_types.d.ts.map +1 -0
  93. package/build/src/modeling/amf/core_types.js +45 -0
  94. package/build/src/modeling/amf/core_types.js.map +1 -0
  95. package/build/src/modeling/index.d.ts +1 -0
  96. package/build/src/modeling/index.d.ts.map +1 -1
  97. package/build/src/modeling/index.js +1 -0
  98. package/build/src/modeling/index.js.map +1 -1
  99. package/build/src/modeling/types.d.ts +67 -0
  100. package/build/src/modeling/types.d.ts.map +1 -1
  101. package/build/src/modeling/types.js.map +1 -1
  102. package/build/src/modeling/validation/api_model_rules.d.ts +15 -0
  103. package/build/src/modeling/validation/api_model_rules.d.ts.map +1 -0
  104. package/build/src/modeling/validation/api_model_rules.js +599 -0
  105. package/build/src/modeling/validation/api_model_rules.js.map +1 -0
  106. package/build/src/modeling/validation/association_validation.d.ts.map +1 -1
  107. package/build/src/modeling/validation/association_validation.js +1 -3
  108. package/build/src/modeling/validation/association_validation.js.map +1 -1
  109. package/build/src/models/http-flows/LegacyTranslator.js +1 -1
  110. package/build/src/models/http-flows/LegacyTranslator.js.map +1 -1
  111. package/build/src/runtime/http-runner/HttpFlows.js +1 -1
  112. package/build/src/runtime/http-runner/HttpFlows.js.map +1 -1
  113. package/build/tsconfig.tsbuildinfo +1 -1
  114. package/eslint.config.js +0 -2
  115. package/package.json +27 -135
  116. package/src/decorators/observed.ts +91 -0
  117. package/src/events/EventTypes.ts +0 -2
  118. package/src/events/Events.ts +0 -2
  119. package/src/modeling/ApiModel.ts +73 -33
  120. package/src/modeling/Bindings.ts +3 -3
  121. package/src/modeling/DomainAssociation.ts +5 -5
  122. package/src/modeling/DomainEntity.ts +3 -4
  123. package/src/modeling/DomainProperty.ts +3 -4
  124. package/src/modeling/DomainValidation.ts +1 -1
  125. package/src/modeling/ExposedEntity.ts +63 -9
  126. package/src/modeling/actions/Action.ts +25 -2
  127. package/src/modeling/actions/CreateAction.ts +3 -2
  128. package/src/modeling/actions/DeleteAction.ts +3 -2
  129. package/src/modeling/actions/ListAction.ts +3 -2
  130. package/src/modeling/actions/ReadAction.ts +3 -2
  131. package/src/modeling/actions/SearchAction.ts +3 -2
  132. package/src/modeling/actions/UpdateAction.ts +3 -2
  133. package/src/modeling/ai/types.ts +2 -2
  134. package/src/{amf → modeling/amf}/DataValueGenerator.ts +3 -4
  135. package/src/modeling/amf/ShapeGenerator.ts +45 -61
  136. package/src/modeling/amf/core_types.ts +47 -0
  137. package/src/modeling/types.ts +70 -0
  138. package/src/modeling/validation/api_model_rules.ts +640 -0
  139. package/src/modeling/validation/api_model_validation_rules.md +58 -0
  140. package/src/modeling/validation/association_validation.ts +1 -3
  141. package/src/models/http-flows/LegacyTranslator.ts +1 -1
  142. package/src/runtime/http-runner/HttpFlows.ts +1 -1
  143. package/tests/unit/modeling/actions/Action.spec.ts +40 -8
  144. package/tests/unit/modeling/actions/CreateAction.spec.ts +5 -5
  145. package/tests/unit/modeling/actions/DeleteAction.spec.ts +6 -6
  146. package/tests/unit/modeling/actions/ListAction.spec.ts +7 -7
  147. package/tests/unit/modeling/actions/ReadAction.spec.ts +6 -6
  148. package/tests/unit/modeling/actions/SearchAction.spec.ts +6 -6
  149. package/tests/unit/modeling/actions/UpdateAction.spec.ts +6 -6
  150. package/tests/unit/{amf → modeling/amf}/data_value_generator.spec.ts +2 -2
  151. package/tests/unit/modeling/amf/shape_generator.spec.ts +146 -167
  152. package/tests/unit/modeling/api_model.spec.ts +190 -13
  153. package/tests/unit/modeling/api_model_expose_entity.spec.ts +43 -19
  154. package/tests/unit/modeling/api_model_remove_entity.spec.ts +6 -6
  155. package/tests/unit/modeling/domain_asociation.spec.ts +13 -13
  156. package/tests/unit/modeling/exposed_entity.spec.ts +123 -3
  157. package/tests/unit/modeling/exposed_entity_actions.spec.ts +41 -18
  158. package/tests/unit/modeling/exposed_entity_setter_validation.spec.ts +1 -1
  159. package/tests/unit/modeling/rules/restoring_rules.spec.ts +9 -5
  160. package/tests/unit/modeling/validation/api_model_rules.spec.ts +324 -0
  161. package/tests/unit/runtime/runners/http_assertion_runner.spec.ts +1 -1
  162. package/tests/unit/runtime/runners/http_flow_runner.spec.ts +1 -1
  163. package/tests/unit/runtime/runners/project_runner.spec.ts +1 -1
  164. package/tests/unit/runtime/runners/request_runner.spec.ts +1 -1
  165. package/tests/unit/runtime/runners/serial_runner.spec.ts +1 -1
  166. package/tsconfig.browser.json +1 -1
  167. package/tsconfig.node.json +1 -1
  168. package/bin/test-web.ts +0 -6
  169. package/build/src/amf/AmfMixin.d.ts +0 -396
  170. package/build/src/amf/AmfMixin.d.ts.map +0 -1
  171. package/build/src/amf/AmfMixin.js +0 -1138
  172. package/build/src/amf/AmfMixin.js.map +0 -1
  173. package/build/src/amf/AmfSerializer.d.ts +0 -137
  174. package/build/src/amf/AmfSerializer.d.ts.map +0 -1
  175. package/build/src/amf/AmfSerializer.js +0 -1947
  176. package/build/src/amf/AmfSerializer.js.map +0 -1
  177. package/build/src/amf/AmfTypes.d.ts +0 -25
  178. package/build/src/amf/AmfTypes.d.ts.map +0 -1
  179. package/build/src/amf/AmfTypes.js +0 -143
  180. package/build/src/amf/AmfTypes.js.map +0 -1
  181. package/build/src/amf/ApiExampleGenerator.d.ts +0 -40
  182. package/build/src/amf/ApiExampleGenerator.d.ts.map +0 -1
  183. package/build/src/amf/ApiExampleGenerator.js +0 -112
  184. package/build/src/amf/ApiExampleGenerator.js.map +0 -1
  185. package/build/src/amf/ApiMonacoSchemaGenerator.d.ts +0 -68
  186. package/build/src/amf/ApiMonacoSchemaGenerator.d.ts.map +0 -1
  187. package/build/src/amf/ApiMonacoSchemaGenerator.js +0 -248
  188. package/build/src/amf/ApiMonacoSchemaGenerator.js.map +0 -1
  189. package/build/src/amf/ApiSchemaGenerator.d.ts +0 -56
  190. package/build/src/amf/ApiSchemaGenerator.d.ts.map +0 -1
  191. package/build/src/amf/ApiSchemaGenerator.js +0 -94
  192. package/build/src/amf/ApiSchemaGenerator.js.map +0 -1
  193. package/build/src/amf/ApiSchemaValues.d.ts +0 -111
  194. package/build/src/amf/ApiSchemaValues.d.ts.map +0 -1
  195. package/build/src/amf/ApiSchemaValues.js +0 -463
  196. package/build/src/amf/ApiSchemaValues.js.map +0 -1
  197. package/build/src/amf/DataValueGenerator.d.ts.map +0 -1
  198. package/build/src/amf/DataValueGenerator.js.map +0 -1
  199. package/build/src/amf/Parsing.d.ts +0 -48
  200. package/build/src/amf/Parsing.d.ts.map +0 -1
  201. package/build/src/amf/Parsing.js +0 -2
  202. package/build/src/amf/Parsing.js.map +0 -1
  203. package/build/src/amf/Utils.d.ts +0 -37
  204. package/build/src/amf/Utils.d.ts.map +0 -1
  205. package/build/src/amf/Utils.js +0 -169
  206. package/build/src/amf/Utils.js.map +0 -1
  207. package/build/src/amf/data-node/DataNodeBase.d.ts +0 -34
  208. package/build/src/amf/data-node/DataNodeBase.d.ts.map +0 -1
  209. package/build/src/amf/data-node/DataNodeBase.js +0 -77
  210. package/build/src/amf/data-node/DataNodeBase.js.map +0 -1
  211. package/build/src/amf/data-node/JsonDataNodeGenerator.d.ts +0 -15
  212. package/build/src/amf/data-node/JsonDataNodeGenerator.d.ts.map +0 -1
  213. package/build/src/amf/data-node/JsonDataNodeGenerator.js +0 -23
  214. package/build/src/amf/data-node/JsonDataNodeGenerator.js.map +0 -1
  215. package/build/src/amf/data-node/UrlEncodedDataNodeGenerator.d.ts +0 -14
  216. package/build/src/amf/data-node/UrlEncodedDataNodeGenerator.d.ts.map +0 -1
  217. package/build/src/amf/data-node/UrlEncodedDataNodeGenerator.js +0 -48
  218. package/build/src/amf/data-node/UrlEncodedDataNodeGenerator.js.map +0 -1
  219. package/build/src/amf/data-node/XmlDataNodeGenerator.d.ts +0 -22
  220. package/build/src/amf/data-node/XmlDataNodeGenerator.d.ts.map +0 -1
  221. package/build/src/amf/data-node/XmlDataNodeGenerator.js +0 -34
  222. package/build/src/amf/data-node/XmlDataNodeGenerator.js.map +0 -1
  223. package/build/src/amf/definitions/Amf.d.ts +0 -363
  224. package/build/src/amf/definitions/Amf.d.ts.map +0 -1
  225. package/build/src/amf/definitions/Amf.js +0 -3
  226. package/build/src/amf/definitions/Amf.js.map +0 -1
  227. package/build/src/amf/definitions/Api.d.ts +0 -383
  228. package/build/src/amf/definitions/Api.d.ts.map +0 -1
  229. package/build/src/amf/definitions/Api.js +0 -2
  230. package/build/src/amf/definitions/Api.js.map +0 -1
  231. package/build/src/amf/definitions/Base.d.ts +0 -12
  232. package/build/src/amf/definitions/Base.d.ts.map +0 -1
  233. package/build/src/amf/definitions/Base.js +0 -2
  234. package/build/src/amf/definitions/Base.js.map +0 -1
  235. package/build/src/amf/definitions/Namespace.d.ts +0 -327
  236. package/build/src/amf/definitions/Namespace.d.ts.map +0 -1
  237. package/build/src/amf/definitions/Namespace.js +0 -346
  238. package/build/src/amf/definitions/Namespace.js.map +0 -1
  239. package/build/src/amf/definitions/Shapes.d.ts +0 -312
  240. package/build/src/amf/definitions/Shapes.d.ts.map +0 -1
  241. package/build/src/amf/definitions/Shapes.js +0 -87
  242. package/build/src/amf/definitions/Shapes.js.map +0 -1
  243. package/build/src/amf/models/AmfDataNode.d.ts +0 -67
  244. package/build/src/amf/models/AmfDataNode.d.ts.map +0 -1
  245. package/build/src/amf/models/AmfDataNode.js +0 -192
  246. package/build/src/amf/models/AmfDataNode.js.map +0 -1
  247. package/build/src/amf/shape/ShapeBase.d.ts +0 -74
  248. package/build/src/amf/shape/ShapeBase.d.ts.map +0 -1
  249. package/build/src/amf/shape/ShapeBase.js +0 -93
  250. package/build/src/amf/shape/ShapeBase.js.map +0 -1
  251. package/build/src/amf/shape/ShapeJsonSchemaGenerator.d.ts +0 -47
  252. package/build/src/amf/shape/ShapeJsonSchemaGenerator.d.ts.map +0 -1
  253. package/build/src/amf/shape/ShapeJsonSchemaGenerator.js +0 -420
  254. package/build/src/amf/shape/ShapeJsonSchemaGenerator.js.map +0 -1
  255. package/build/src/amf/shape/ShapeXmlSchemaGenerator.d.ts +0 -85
  256. package/build/src/amf/shape/ShapeXmlSchemaGenerator.d.ts.map +0 -1
  257. package/build/src/amf/shape/ShapeXmlSchemaGenerator.js +0 -852
  258. package/build/src/amf/shape/ShapeXmlSchemaGenerator.js.map +0 -1
  259. package/build/src/events/amf/AmfEventTypes.d.ts +0 -7
  260. package/build/src/events/amf/AmfEventTypes.d.ts.map +0 -1
  261. package/build/src/events/amf/AmfEventTypes.js +0 -7
  262. package/build/src/events/amf/AmfEventTypes.js.map +0 -1
  263. package/build/src/events/amf/AmfEvents.d.ts +0 -34
  264. package/build/src/events/amf/AmfEvents.d.ts.map +0 -1
  265. package/build/src/events/amf/AmfEvents.js +0 -63
  266. package/build/src/events/amf/AmfEvents.js.map +0 -1
  267. package/build/src/mocking/RandExp.d.ts +0 -55
  268. package/build/src/mocking/RandExp.d.ts.map +0 -1
  269. package/build/src/mocking/RandExp.js +0 -302
  270. package/build/src/mocking/RandExp.js.map +0 -1
  271. package/build/src/mocking/lib/ret.d.ts +0 -16
  272. package/build/src/mocking/lib/ret.d.ts.map +0 -1
  273. package/build/src/mocking/lib/ret.js +0 -284
  274. package/build/src/mocking/lib/ret.js.map +0 -1
  275. package/data/apis/APIC-187/APIC-187.raml +0 -21
  276. package/data/apis/APIC-187/dataType.raml +0 -132
  277. package/data/apis/APIC-187/example.raml +0 -84
  278. package/data/apis/APIC-188/APIC-188.raml +0 -21
  279. package/data/apis/APIC-188/dataType.raml +0 -132
  280. package/data/apis/APIC-188/example.raml +0 -84
  281. package/data/apis/APIC-233/APIC-233.raml +0 -15
  282. package/data/apis/APIC-391/APIC-391.raml +0 -44
  283. package/data/apis/APIC-391/examples/post-body-request-example.raml +0 -7
  284. package/data/apis/APIC-483/APIC-483.raml +0 -16
  285. package/data/apis/APIC-483/example/list-of-bank-succ.json +0 -32
  286. package/data/apis/APIC-483/schema/response-retrieve-list-bank.json +0 -68
  287. package/data/apis/APIC-487/APIC-487.raml +0 -36
  288. package/data/apis/APIC-655/APIC-655.raml +0 -44
  289. package/data/apis/APIC-655/delivery.json +0 -12
  290. package/data/apis/APIC-689/APIC-689.raml +0 -20
  291. package/data/apis/APIC-690/APIC-690.raml +0 -17
  292. package/data/apis/APIC-690/datatypes/Address.raml +0 -10
  293. package/data/apis/APIC-690/datatypes/Customer.raml +0 -17
  294. package/data/apis/APIC-690/examples/CustomerExample2.raml +0 -17
  295. package/data/apis/APIC-690/examples/CustomersExample.raml +0 -34
  296. package/data/apis/SE-10469/Purchase-Order-oagis-id-dd32f9c6c01048a19e15c423c9c741ae.json +0 -1383
  297. package/data/apis/SE-10469/SE-10469.raml +0 -11
  298. package/data/apis/SE-13092/Examples/PersonExample.raml +0 -7
  299. package/data/apis/SE-13092/SE-13092.raml +0 -37
  300. package/data/apis/SE-13092/datatypes/Address.raml +0 -5
  301. package/data/apis/SE-13092/datatypes/InlineType.raml +0 -5
  302. package/data/apis/SE-13092/datatypes/Person.raml +0 -15
  303. package/data/apis/SE-22063/SE-22063.raml +0 -14
  304. package/data/apis/SE-22063/stock.raml +0 -41
  305. package/data/apis/amf-helper-api/amf-helper-api.raml +0 -1024
  306. package/data/apis/amf-helper-api/examples/comment-example.json +0 -42
  307. package/data/apis/amf-helper-api/examples/commentCreateRequest-example.json +0 -7
  308. package/data/apis/amf-helper-api/examples/commentList-example.json +0 -48
  309. package/data/apis/amf-helper-api/examples/person.json +0 -14
  310. package/data/apis/amf-helper-api/examples/person.raml +0 -13
  311. package/data/apis/amf-helper-api/examples/person.xml +0 -14
  312. package/data/apis/amf-helper-api/libraries/comment-lib.raml +0 -143
  313. package/data/apis/amf-helper-api/library.raml +0 -7
  314. package/data/apis/amf-helper-api/schemas/person.json +0 -104
  315. package/data/apis/amf-helper-api/schemas/person.xsd +0 -26
  316. package/data/apis/amf-helper-api/traits/file.raml +0 -33
  317. package/data/apis/amf-helper-api/traits/pagination.raml +0 -80
  318. package/data/apis/amf-helper-api/traits/results.raml +0 -19
  319. package/data/apis/amf-helper-api/traits/visibility.raml +0 -14
  320. package/data/apis/amf-helper-api/types/app-person.raml +0 -101
  321. package/data/apis/amf-helper-api/types/dimensions.raml +0 -24
  322. package/data/apis/amf-helper-api/types/external-type.raml +0 -7
  323. package/data/apis/amf-helper-api/types/image.raml +0 -23
  324. package/data/apis/amf-helper-api/types/product.raml +0 -52
  325. package/data/apis/amf-helper-api/types/resource.raml +0 -11
  326. package/data/apis/amf-helper-api/types/standard-request.raml +0 -50
  327. package/data/apis/amf-helper-api/types/user.raml +0 -37
  328. package/data/apis/arc-demo-api/arc-demo-api.raml +0 -468
  329. package/data/apis/arc-demo-api/examples/e400.xml +0 -4
  330. package/data/apis/arc-demo-api/examples/e401.xml +0 -4
  331. package/data/apis/arc-demo-api/examples/e404.xml +0 -4
  332. package/data/apis/arc-demo-api/examples/image.xml +0 -5
  333. package/data/apis/arc-demo-api/examples/messages-example.json +0 -22
  334. package/data/apis/arc-demo-api/examples/messages-sent-example.json +0 -15
  335. package/data/apis/arc-demo-api/examples/person.json +0 -14
  336. package/data/apis/arc-demo-api/examples/person.url.encoded +0 -1
  337. package/data/apis/arc-demo-api/examples/person.xml +0 -14
  338. package/data/apis/arc-demo-api/examples/product.xml +0 -9
  339. package/data/apis/arc-demo-api/library.raml +0 -36
  340. package/data/apis/arc-demo-api/resourceTypes/app-person.raml +0 -49
  341. package/data/apis/arc-demo-api/resourceTypes/example-types.raml +0 -45
  342. package/data/apis/arc-demo-api/resourceTypes/image.raml +0 -23
  343. package/data/apis/arc-demo-api/resourceTypes/message-sent-type.raml +0 -23
  344. package/data/apis/arc-demo-api/resourceTypes/message-type.raml +0 -33
  345. package/data/apis/arc-demo-api/resourceTypes/product.raml +0 -52
  346. package/data/apis/arc-demo-api/resourceTypes/resource.raml +0 -11
  347. package/data/apis/arc-demo-api/schemas/error-response.xsd +0 -11
  348. package/data/apis/arc-demo-api/schemas/image.xsd +0 -11
  349. package/data/apis/arc-demo-api/schemas/person.json +0 -104
  350. package/data/apis/arc-demo-api/schemas/person.xsd +0 -26
  351. package/data/apis/arc-demo-api/schemas/product.xsd +0 -17
  352. package/data/apis/arc-demo-api/securitySchemes/basic.raml +0 -5
  353. package/data/apis/arc-demo-api/securitySchemes/oauth_2_0.raml +0 -29
  354. package/data/apis/arc-demo-api/securitySchemes/x-custom.raml +0 -26
  355. package/data/apis/arc-demo-api/traits/adminable.raml +0 -23
  356. package/data/apis/arc-demo-api/traits/pagination.raml +0 -83
  357. package/data/apis/arc-demo-api/traits/rate-limited.raml +0 -9
  358. package/data/apis/async-api/async-api.yaml +0 -96
  359. package/data/apis/example-generator-api/contact-email-example.raml +0 -6
  360. package/data/apis/example-generator-api/contact-example.raml +0 -5
  361. package/data/apis/example-generator-api/contact-fax-example.raml +0 -5
  362. package/data/apis/example-generator-api/contact-website-example.raml +0 -2
  363. package/data/apis/example-generator-api/example-1.raml +0 -6
  364. package/data/apis/example-generator-api/example-2.raml +0 -9
  365. package/data/apis/example-generator-api/example-3.raml +0 -4
  366. package/data/apis/example-generator-api/example-4.raml +0 -2
  367. package/data/apis/example-generator-api/example-5.raml +0 -4
  368. package/data/apis/example-generator-api/example-6.raml +0 -2
  369. package/data/apis/example-generator-api/example-7.raml +0 -5
  370. package/data/apis/example-generator-api/example-generator-api.raml +0 -423
  371. package/data/apis/example-generator-api/example.json +0 -14
  372. package/data/apis/example-generator-api/example.xml +0 -15
  373. package/data/apis/example-generator-api/example.xsd +0 -26
  374. package/data/apis/example-generator-api/examples/employee.raml +0 -7
  375. package/data/apis/example-generator-api/examples/employees.raml +0 -7
  376. package/data/apis/example-generator-api/linked-named-example.raml +0 -21
  377. package/data/apis/example-generator-api/named-example-with-link.raml +0 -5
  378. package/data/apis/example-generator-api/named-example.raml +0 -9
  379. package/data/apis/example-generator-api/user-json-example.raml +0 -17
  380. package/data/apis/example-generator-api/user-raml-example.raml +0 -37
  381. package/data/apis/example-generator-api/user.json +0 -8
  382. package/data/apis/example-generator-api/user.raml +0 -18
  383. package/data/apis/example-generator-api/users-json-example.raml +0 -17
  384. package/data/apis/example-generator-api/users-raml-example.raml +0 -32
  385. package/data/apis/example-generator-api/users.json +0 -8
  386. package/data/apis/expanded-api/expanded-api.raml +0 -8
  387. package/data/apis/flattened-api/flattened-api.raml +0 -8
  388. package/data/apis/multiple-servers/multiple-servers.yaml +0 -112
  389. package/data/apis/oas-3-api/oas-3-api.yaml +0 -356
  390. package/data/apis/oas-date/oas-date.yaml +0 -28
  391. package/data/apis/oas-types/oas-types.yaml +0 -159
  392. package/data/apis/oas-unions/oas-unions.yaml +0 -75
  393. package/data/apis/petstore/petstore.yaml +0 -954
  394. package/data/apis/raml-date/raml-date.raml +0 -28
  395. package/data/apis/recursive/recursive.raml +0 -14
  396. package/data/apis/schema-api/examples/person.json +0 -14
  397. package/data/apis/schema-api/examples/person.raml +0 -14
  398. package/data/apis/schema-api/examples/person.url.encoded +0 -1
  399. package/data/apis/schema-api/examples/person.xml +0 -14
  400. package/data/apis/schema-api/library/demo-types.raml +0 -43
  401. package/data/apis/schema-api/schema-api.raml +0 -656
  402. package/data/apis/schema-api/schemas/person.json +0 -104
  403. package/data/apis/schema-api/schemas/person.xsd +0 -26
  404. package/data/apis/schema-api/types/DemoPerson.raml +0 -67
  405. package/data/apis/secured-api/oauth-2-custom-settings.raml +0 -143
  406. package/data/apis/secured-api/oauth2-header-delivery.raml +0 -13
  407. package/data/apis/secured-api/oauth2-no-delivery.raml +0 -8
  408. package/data/apis/secured-api/oauth2-no-grants.raml +0 -12
  409. package/data/apis/secured-api/oauth2-pkce.raml +0 -13
  410. package/data/apis/secured-api/oauth2-query-delivery.raml +0 -13
  411. package/data/apis/secured-api/oauth_1_0.raml +0 -8
  412. package/data/apis/secured-api/oauth_1_0_no-settings.raml +0 -3
  413. package/data/apis/secured-api/oauth_1_0_no-signature.raml +0 -7
  414. package/data/apis/secured-api/oauth_1_0_signature.raml +0 -8
  415. package/data/apis/secured-api/passthrough-querystring.raml +0 -16
  416. package/data/apis/secured-api/passthrough.raml +0 -24
  417. package/data/apis/secured-api/secured-api.raml +0 -231
  418. package/data/apis/secured-api/x-custom.raml +0 -33
  419. package/data/apis/secured-api/x-other.raml +0 -29
  420. package/data/apis/secured-api/x-query-string.raml +0 -16
  421. package/data/apis/tracked-examples/datatypes/employee.raml +0 -5
  422. package/data/apis/tracked-examples/examples/employee.raml +0 -4
  423. package/data/apis/tracked-examples/examples/employees.raml +0 -8
  424. package/data/apis/tracked-examples/tracked-to-linked.raml +0 -28
  425. package/data/model.js +0 -129
  426. package/data/models/APIC-187.json +0 -11416
  427. package/data/models/APIC-188.json +0 -11416
  428. package/data/models/APIC-233.json +0 -539
  429. package/data/models/APIC-391.json +0 -1903
  430. package/data/models/APIC-483.json +0 -3110
  431. package/data/models/APIC-487.json +0 -1225
  432. package/data/models/APIC-655.json +0 -3145
  433. package/data/models/APIC-689.json +0 -797
  434. package/data/models/APIC-690.json +0 -5992
  435. package/data/models/SE-10469.json +0 -22898
  436. package/data/models/SE-13092.json +0 -2769
  437. package/data/models/SE-22063.json +0 -2314
  438. package/data/models/amf-helper-api.json +0 -84457
  439. package/data/models/arc-demo-api.json +0 -52041
  440. package/data/models/async-api.json +0 -3227
  441. package/data/models/example-generator-api.json +0 -45209
  442. package/data/models/expanded-api.json +0 -251
  443. package/data/models/flattened-api.json +0 -251
  444. package/data/models/multiple-servers.json +0 -2769
  445. package/data/models/oas-3-api.json +0 -10590
  446. package/data/models/oas-date.json +0 -637
  447. package/data/models/oas-types.json +0 -5352
  448. package/data/models/oas-unions.json +0 -1935
  449. package/data/models/petstore.json +0 -27712
  450. package/data/models/raml-date.json +0 -1114
  451. package/data/models/recursive.json +0 -610
  452. package/data/models/schema-api.json +0 -35527
  453. package/data/models/secured-api.json +0 -21071
  454. package/data/models/tracked-to-linked.json +0 -1914
  455. package/src/amf/AmfMixin.ts +0 -1726
  456. package/src/amf/AmfSerializer.ts +0 -2358
  457. package/src/amf/AmfTypes.ts +0 -147
  458. package/src/amf/ApiExampleGenerator.ts +0 -122
  459. package/src/amf/ApiMonacoSchemaGenerator.ts +0 -332
  460. package/src/amf/ApiSchemaGenerator.ts +0 -115
  461. package/src/amf/ApiSchemaValues.ts +0 -509
  462. package/src/amf/Parsing.ts +0 -50
  463. package/src/amf/Utils.ts +0 -172
  464. package/src/amf/data-node/DataNodeBase.ts +0 -83
  465. package/src/amf/data-node/JsonDataNodeGenerator.ts +0 -24
  466. package/src/amf/data-node/README.md +0 -3
  467. package/src/amf/data-node/UrlEncodedDataNodeGenerator.ts +0 -49
  468. package/src/amf/data-node/XmlDataNodeGenerator.ts +0 -46
  469. package/src/amf/definitions/Amf.ts +0 -440
  470. package/src/amf/definitions/Api.ts +0 -445
  471. package/src/amf/definitions/Base.ts +0 -13
  472. package/src/amf/definitions/Namespace.ts +0 -357
  473. package/src/amf/definitions/Shapes.ts +0 -438
  474. package/src/amf/models/AmfDataNode.ts +0 -210
  475. package/src/amf/shape/README.md +0 -4
  476. package/src/amf/shape/ShapeBase.ts +0 -174
  477. package/src/amf/shape/ShapeJsonSchemaGenerator.ts +0 -457
  478. package/src/amf/shape/ShapeXmlSchemaGenerator.ts +0 -933
  479. package/src/events/amf/AmfEventTypes.ts +0 -6
  480. package/src/events/amf/AmfEvents.ts +0 -78
  481. package/src/mocking/RandExp.ts +0 -335
  482. package/src/mocking/lib/ret.ts +0 -279
  483. package/tests/unit/amf/APIC-187.spec.ts +0 -41
  484. package/tests/unit/amf/APIC-188.spec.ts +0 -30
  485. package/tests/unit/amf/APIC-233.spec.ts +0 -32
  486. package/tests/unit/amf/APIC-391.spec.ts +0 -39
  487. package/tests/unit/amf/APIC-487.spec.ts +0 -44
  488. package/tests/unit/amf/APIC-655.spec.ts +0 -41
  489. package/tests/unit/amf/APIC-689.spec.ts +0 -28
  490. package/tests/unit/amf/APIC-690.spec.ts +0 -93
  491. package/tests/unit/amf/AmfLoader.ts +0 -538
  492. package/tests/unit/amf/AmfMixin.spec.ts +0 -1707
  493. package/tests/unit/amf/ApiExampleGenerator.spec.ts +0 -250
  494. package/tests/unit/amf/ApiMonacoSchemaGenerator.spec.ts +0 -183
  495. package/tests/unit/amf/ApiSchemaValues.spec.ts +0 -602
  496. package/tests/unit/amf/Namepsace.spec.ts +0 -494
  497. package/tests/unit/amf/SE-10469.spec.ts +0 -35
  498. package/tests/unit/amf/SE-13092.spec.ts +0 -64
  499. package/tests/unit/amf/SE-22063.spec.ts +0 -45
  500. package/tests/unit/amf/raml-examples.spec.ts +0 -43
  501. package/tests/unit/amf/schema_gen_array.spec.ts +0 -199
  502. package/tests/unit/amf/schema_gen_oas.spec.ts +0 -140
  503. package/tests/unit/amf/schema_gen_object.spec.ts +0 -279
  504. package/tests/unit/amf/schema_gen_scalar.spec.ts +0 -143
  505. package/tests/unit/amf/schema_gen_union.spec.ts +0 -401
  506. package/tests/unit/amf/schema_gen_unknown.spec.ts +0 -42
  507. package/tests/unit/amf/schema_gen_xml.spec.ts +0 -726
  508. package/tests/unit/amf/serializer/APIC-483.spec.ts +0 -36
  509. package/tests/unit/amf/serializer/Anotations.spec.ts +0 -50
  510. package/tests/unit/amf/serializer/Api.spec.ts +0 -210
  511. package/tests/unit/amf/serializer/Documentation.spec.ts +0 -52
  512. package/tests/unit/amf/serializer/Endpoints.spec.ts +0 -172
  513. package/tests/unit/amf/serializer/Operations.spec.ts +0 -330
  514. package/tests/unit/amf/serializer/Parameters.spec.ts +0 -149
  515. package/tests/unit/amf/serializer/PayloadExamples.spec.ts +0 -152
  516. package/tests/unit/amf/serializer/Security.spec.ts +0 -626
  517. package/tests/unit/amf/serializer/Server.spec.ts +0 -78
  518. package/tests/unit/amf/serializer/Shapes.spec.ts +0 -304
  519. package/tests/unit/amf/serializer/SourceMaps.spec.ts +0 -52
  520. package/tests/unit/amf/shapes/ShapeXmlSchemaGenerator.spec.ts +0 -28
  521. package/tests/unit/amf/tracked-to-linked.spec.ts +0 -90
  522. package/tests/unit/amf/xml-data-processing.spec.ts +0 -217
  523. package/tests/unit/events/amf.spec.ts +0 -151
@@ -0,0 +1,640 @@
1
+ import type { ApiModel } from '../ApiModel.js'
2
+ import type { ExposedEntity } from '../ExposedEntity.js'
3
+ import type { Action } from '../actions/Action.js'
4
+ import { ListAction } from '../actions/ListAction.js'
5
+ import { DeleteAction } from '../actions/DeleteAction.js'
6
+ import { UpdateAction } from '../actions/UpdateAction.js'
7
+ import { SearchAction } from '../actions/SearchAction.js'
8
+ import type { RolesBasedAccessControl, UsernamePasswordConfiguration } from '../types.js'
9
+ import type { ApiModelValidationItem, ApiModelValidationResult, ApiModelValidationContext } from '../types.js'
10
+ import { ApiModelKind, ExposedEntityKind } from '../../models/kinds.js'
11
+ import { SemanticType } from '../Semantics.js'
12
+
13
+ /**
14
+ * Creates a unique validation code.
15
+ * @param entity The entity type (e.g., 'API', 'EXPOSURE', 'ACTION')
16
+ * @param issue The issue identifier (e.g., 'MISSING_NAME')
17
+ */
18
+ function createCode(entity: string, issue: string): string {
19
+ return `${entity}_${issue}`
20
+ }
21
+
22
+ /**
23
+ * Validates the core properties and metadata of an ApiModel.
24
+ */
25
+ export function validateApiModelInfo(model: ApiModel): ApiModelValidationItem[] {
26
+ const issues: ApiModelValidationItem[] = []
27
+ const context: ApiModelValidationContext = {
28
+ apiModelKey: model.key,
29
+ kind: ApiModelKind,
30
+ key: model.key,
31
+ }
32
+
33
+ if (model.kind !== ApiModelKind) {
34
+ issues.push({
35
+ code: createCode('API', 'INVALID_KIND'),
36
+ message: 'The API model type is incorrect.',
37
+ suggestion: 'Set the model type to "ApiModel".',
38
+ severity: 'error',
39
+ context: { ...context, property: 'kind' },
40
+ })
41
+ }
42
+
43
+ if (!model.key || typeof model.key !== 'string' || model.key.trim() === '') {
44
+ issues.push({
45
+ code: createCode('API', 'MISSING_KEY'),
46
+ message: 'The API model is missing a unique identifier.',
47
+ suggestion: 'Provide a valid name to use as a unique identifier.',
48
+ severity: 'error',
49
+ context: { ...context, property: 'key' },
50
+ })
51
+ }
52
+
53
+ if (!model.info || !model.info.name || model.info.name.trim() === '') {
54
+ issues.push({
55
+ code: createCode('API', 'MISSING_NAME'),
56
+ message: 'The API model has no defined name.',
57
+ suggestion: 'Provide a descriptive name for your API.',
58
+ severity: 'error',
59
+ context: { ...context, property: 'info.name' },
60
+ })
61
+ }
62
+
63
+ if (!model.info || !model.info.description || model.info.description.trim() === '') {
64
+ issues.push({
65
+ code: createCode('API', 'MISSING_DESCRIPTION'),
66
+ message: 'Adding a description helps others understand your API.',
67
+ suggestion: 'Add a description to clarify the purpose of this API.',
68
+ severity: 'warning',
69
+ context: { ...context, property: 'info.description' },
70
+ })
71
+ }
72
+
73
+ return issues
74
+ }
75
+
76
+ export function validateApiModelDependency(model: ApiModel): ApiModelValidationItem[] {
77
+ const issues: ApiModelValidationItem[] = []
78
+ const context: ApiModelValidationContext = {
79
+ apiModelKey: model.key,
80
+ kind: ApiModelKind,
81
+ key: model.key,
82
+ }
83
+
84
+ const domain = model.domain
85
+ if (!domain) {
86
+ issues.push({
87
+ code: createCode('API', 'MISSING_DOMAIN'),
88
+ message: 'No Data Domain is attached to the API.',
89
+ suggestion: 'Select a Data Domain from the settings to power your API.',
90
+ severity: 'error',
91
+ context,
92
+ })
93
+ return issues // Can't validate version if it doesn't exist
94
+ }
95
+
96
+ if (!domain.info || !domain.info.version) {
97
+ issues.push({
98
+ code: createCode('API', 'MISSING_DOMAIN_VERSION'),
99
+ message: 'The selected Data Domain is missing a version number.',
100
+ suggestion: 'Specify a version for the attached Data Domain.',
101
+ severity: 'error',
102
+ context,
103
+ })
104
+ }
105
+
106
+ return issues
107
+ }
108
+
109
+ export function validateApiModelSecurity(model: ApiModel): ApiModelValidationItem[] {
110
+ const issues: ApiModelValidationItem[] = []
111
+ const context: ApiModelValidationContext = {
112
+ apiModelKey: model.key,
113
+ kind: ApiModelKind,
114
+ key: model.key,
115
+ }
116
+
117
+ // Resolve User target
118
+ const userEntity = model.domain && model.user ? model.domain.findEntity(model.user.key, model.user.domain) : undefined
119
+
120
+ // Authentication
121
+ if (!model.authentication) {
122
+ issues.push({
123
+ code: createCode('API', 'MISSING_AUTHENTICATION'),
124
+ message: 'The API is missing authentication settings.',
125
+ suggestion: 'Go to the security settings and configure how users authenticate.',
126
+ severity: 'error',
127
+ context: { ...context, property: 'authentication' },
128
+ })
129
+ } else if (model.authentication.strategy === 'UsernamePassword') {
130
+ const auth = model.authentication as UsernamePasswordConfiguration
131
+ if (!auth.passwordKey) {
132
+ issues.push({
133
+ code: createCode('API', 'MISSING_PASSWORD_KEY'),
134
+ message: 'Username & Password authentication requires a specific field for the password.',
135
+ suggestion:
136
+ 'Select which field in your user profile should store the password. ' +
137
+ 'The data domain model should have a password data semantic on that property.',
138
+ severity: 'error',
139
+ context: { ...context, property: 'authentication.passwordKey' },
140
+ })
141
+ } else if (userEntity) {
142
+ const passwordProp = Array.from(userEntity.properties).find((p) => p.key === auth.passwordKey)
143
+ if (passwordProp && !passwordProp.hasSemantic(SemanticType.Password)) {
144
+ issues.push({
145
+ code: createCode('API', 'MISSING_PASSWORD_SEMANTIC'),
146
+ message: 'The selected password field is missing the Password data semantic.',
147
+ suggestion: 'Go to the Data Modeler and add the "Password" semantic to this property.',
148
+ severity: 'error',
149
+ context: { ...context, property: 'authentication.passwordKey' },
150
+ })
151
+ }
152
+ }
153
+
154
+ if (userEntity) {
155
+ const hasUsernameSemantic = Array.from(userEntity.properties).some(
156
+ (p) => typeof p.hasSemantic === 'function' && p.hasSemantic(SemanticType.Username)
157
+ )
158
+ if (!hasUsernameSemantic) {
159
+ issues.push({
160
+ code: createCode('API', 'MISSING_USERNAME_SEMANTIC'),
161
+ message: 'Username & Password authentication requires a field with the Username data semantic.',
162
+ suggestion: 'Go to the Data Modeler and add the "Username" semantic to the property used for login.',
163
+ severity: 'error',
164
+ context: { ...context, property: 'user' },
165
+ })
166
+ }
167
+ }
168
+ }
169
+
170
+ // Authorization
171
+ if (!model.authorization) {
172
+ issues.push({
173
+ code: createCode('API', 'MISSING_AUTHORIZATION'),
174
+ message: 'The API is missing authorization settings.',
175
+ suggestion: 'Go to the security settings and configure how to handle user permissions.',
176
+ severity: 'error',
177
+ context: { ...context, property: 'authorization' },
178
+ })
179
+ } else if (model.authorization.strategy === 'RBAC') {
180
+ const rbac = model.authorization as RolesBasedAccessControl
181
+ if (!rbac.roleKey) {
182
+ issues.push({
183
+ code: createCode('API', 'MISSING_ROLE_KEY'),
184
+ message: 'Role-based access control is selected but no role field has been defined.',
185
+ suggestion: "Select which field in your user profile determines the user's role.",
186
+ severity: 'error',
187
+ context: { ...context, property: 'authorization.roleKey' },
188
+ })
189
+ } else if (userEntity) {
190
+ const roleProp = Array.from(userEntity.properties).find((p) => p.key === rbac.roleKey)
191
+ if (roleProp && !roleProp.hasSemantic(SemanticType.UserRole)) {
192
+ issues.push({
193
+ code: createCode('API', 'MISSING_ROLE_SEMANTIC'),
194
+ message: 'The selected role field is missing the User Role data semantic.',
195
+ suggestion: 'Go to the Data Modeler and add the "User Role" semantic to this property.',
196
+ severity: 'error',
197
+ context: { ...context, property: 'authorization.roleKey' },
198
+ })
199
+ }
200
+ }
201
+ }
202
+
203
+ // Session
204
+ if (!model.session) {
205
+ issues.push({
206
+ code: createCode('API', 'MISSING_SESSION'),
207
+ message: 'The API is missing session configuration.',
208
+ suggestion: 'Configure how user sessions will be handled.',
209
+ severity: 'error',
210
+ context: { ...context, property: 'session' },
211
+ })
212
+ } else {
213
+ if (!model.session.secret) {
214
+ issues.push({
215
+ code: createCode('API', 'MISSING_SESSION_SECRET'),
216
+ message: 'A secure encryption key is required for sessions.',
217
+ suggestion: 'Provide a strong security key in the session settings.',
218
+ severity: 'error',
219
+ context: { ...context, property: 'session.secret' },
220
+ })
221
+ }
222
+
223
+ if (!model.session.properties || model.session.properties.length === 0) {
224
+ issues.push({
225
+ code: createCode('API', 'MISSING_SESSION_PROPERTIES'),
226
+ message: 'The session token needs to include at least one piece of user information, like an ID.',
227
+ suggestion: 'Select fields that should be stored securely inside the session payload.',
228
+ severity: 'error',
229
+ context: { ...context, property: 'session.properties' },
230
+ })
231
+ } else if (model.authorization && model.authorization.strategy === 'RBAC') {
232
+ const rbac = model.authorization as RolesBasedAccessControl
233
+ if (rbac.roleKey && !model.session.properties.includes(rbac.roleKey)) {
234
+ issues.push({
235
+ code: createCode('API', 'MISSING_RBAC_SESSION_PROPERTY'),
236
+ message: 'The user role must be included in the session data for permissions to work.',
237
+ suggestion: 'Make sure your selected role field is checked in the session settings.',
238
+ severity: 'error',
239
+ context: { ...context, property: 'session.properties' },
240
+ })
241
+ }
242
+ }
243
+
244
+ const { cookie, jwt } = model.session
245
+ if ((!cookie || !cookie.enabled) && (!jwt || !jwt.enabled)) {
246
+ issues.push({
247
+ code: createCode('API', 'MISSING_SESSION_TRANSPORT'),
248
+ message: 'No delivery method for sessions (like cookies or tokens) is enabled.',
249
+ suggestion: 'Enable at least one session delivery mechanism in the security settings.',
250
+ severity: 'error',
251
+ context: { ...context, property: 'session' },
252
+ })
253
+ }
254
+ }
255
+
256
+ // User Target
257
+ if (!model.user) {
258
+ issues.push({
259
+ code: createCode('API', 'MISSING_USER_ENTITY'),
260
+ message: 'You need to specify what kind of object represents your users.',
261
+ suggestion: 'Select a model from your domain that stores your user accounts.',
262
+ severity: 'error',
263
+ context: { ...context, property: 'user' },
264
+ })
265
+ } else if (model.domain && !model.domain.findEntity(model.user.key, model.user.domain)) {
266
+ issues.push({
267
+ code: createCode('API', 'INVALID_USER_ENTITY'),
268
+ message: 'The selected user model no longer exists in your data domain.',
269
+ suggestion: 'Please navigate to security settings and select a valid user model.',
270
+ severity: 'error',
271
+ context: { ...context, property: 'user' },
272
+ })
273
+ }
274
+
275
+ return issues
276
+ }
277
+
278
+ export function validateApiModelMetadata(model: ApiModel): ApiModelValidationItem[] {
279
+ const issues: ApiModelValidationItem[] = []
280
+ const context: ApiModelValidationContext = {
281
+ apiModelKey: model.key,
282
+ kind: ApiModelKind,
283
+ key: model.key,
284
+ }
285
+
286
+ const urlRegex = /^https?:\/\//i
287
+ const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/
288
+
289
+ if (model.contact) {
290
+ if (model.contact.email && !emailRegex.test(model.contact.email)) {
291
+ issues.push({
292
+ code: createCode('API', 'INVALID_CONTACT_EMAIL'),
293
+ message: 'The contact email address looks invalid.',
294
+ suggestion: 'Please double check the email address formatting.',
295
+ severity: 'error',
296
+ context: { ...context, property: 'contact.email' },
297
+ })
298
+ }
299
+ if (model.contact.url && !urlRegex.test(model.contact.url)) {
300
+ issues.push({
301
+ code: createCode('API', 'INVALID_CONTACT_URL'),
302
+ message: 'The contact link looks invalid.',
303
+ suggestion: 'Provide a working link, starting with http:// or https://.',
304
+ severity: 'error',
305
+ context: { ...context, property: 'contact.url' },
306
+ })
307
+ }
308
+ } else {
309
+ issues.push({
310
+ code: createCode('API', 'MISSING_CONTACT'),
311
+ message: 'A contact email or link is highly recommended for API consumers.',
312
+ suggestion: 'Add an email or help desk link in the metadata tab.',
313
+ severity: 'info',
314
+ context: { ...context, property: 'contact' },
315
+ })
316
+ }
317
+
318
+ if (model.license) {
319
+ if (model.license.url && !urlRegex.test(model.license.url)) {
320
+ issues.push({
321
+ code: createCode('API', 'INVALID_LICENSE_URL'),
322
+ message: 'The license link looks invalid.',
323
+ suggestion: 'Provide a working link, starting with http:// or https://.',
324
+ severity: 'error',
325
+ context: { ...context, property: 'license.url' },
326
+ })
327
+ }
328
+ } else {
329
+ issues.push({
330
+ code: createCode('API', 'MISSING_LICENSE'),
331
+ message: 'Adding a license to your API helps users understand its usage rights.',
332
+ suggestion: 'Add the name and a link to your API license terms.',
333
+ severity: 'info',
334
+ context: { ...context, property: 'license' },
335
+ })
336
+ }
337
+
338
+ if (!model.termsOfService) {
339
+ issues.push({
340
+ code: createCode('API', 'MISSING_TOS'),
341
+ message: 'Terms of Service help legal compliance.',
342
+ suggestion: 'Add a link to the terms of your API service.',
343
+ severity: 'info',
344
+ context: { ...context, property: 'termsOfService' },
345
+ })
346
+ }
347
+
348
+ return issues
349
+ }
350
+
351
+ export function validateAction(action: Action, parent: ExposedEntity, apiModelKey: string): ApiModelValidationItem[] {
352
+ const issues: ApiModelValidationItem[] = []
353
+ const context: ApiModelValidationContext = {
354
+ apiModelKey,
355
+ kind: 'Action',
356
+ key: action.kind, // Actions lack nanoids, kind represents its type
357
+ parentExposedEntityKey: parent.key,
358
+ }
359
+
360
+ if (ListAction.isListAction(action)) {
361
+ if (!action.pagination || !action.pagination.kind) {
362
+ issues.push({
363
+ code: createCode('ACTION', 'LIST_MISSING_PAGINATION'),
364
+ message: 'A List action must have a defined pagination strategy.',
365
+ suggestion: 'Configure how results are loaded (e.g. by page or by a continuous cursor).',
366
+ severity: 'error',
367
+ context: { ...context, property: 'pagination' },
368
+ })
369
+ }
370
+ if (
371
+ (!action.filterableFields || action.filterableFields.length === 0) &&
372
+ (!action.sortableFields || action.sortableFields.length === 0)
373
+ ) {
374
+ issues.push({
375
+ code: createCode('ACTION', 'LIST_MISSING_FILTERS'),
376
+ message: 'Listing all elements without filters or sorting could be overwhelming for large tables.',
377
+ suggestion: 'Select a few important fields to allow sorting and searching by.',
378
+ severity: 'warning',
379
+ context,
380
+ })
381
+ }
382
+ }
383
+
384
+ if (SearchAction.isSearchAction(action)) {
385
+ if (!action.fields || action.fields.length === 0) {
386
+ issues.push({
387
+ code: createCode('ACTION', 'SEARCH_MISSING_FIELDS'),
388
+ message: 'A Search action needs to know which fields to look in.',
389
+ suggestion: 'Select the properties (like names or emails) that the search will run on.',
390
+ severity: 'error',
391
+ context: { ...context, property: 'fields' },
392
+ })
393
+ }
394
+ }
395
+
396
+ if (DeleteAction.isDeleteAction(action)) {
397
+ if (!action.strategy) {
398
+ issues.push({
399
+ code: createCode('ACTION', 'DELETE_MISSING_STRATEGY'),
400
+ message: 'A Delete action must know if you want to completely erase the record, or just hide it.',
401
+ suggestion: 'Configure the deletion type (permanent or soft).',
402
+ severity: 'error',
403
+ context: { ...context, property: 'strategy' },
404
+ })
405
+ } else if (action.strategy === 'hard') {
406
+ issues.push({
407
+ code: createCode('ACTION', 'DELETE_HARD_WARNING'),
408
+ message: 'Permanent delete is active. There will be no way to restore deleted data.',
409
+ suggestion: 'Consider switching to "soft delete" if users might need to undo mistakes.',
410
+ severity: 'info',
411
+ context: { ...context, property: 'strategy' },
412
+ })
413
+ }
414
+ }
415
+
416
+ if (UpdateAction.isUpdateAction(action)) {
417
+ if (!action.allowedMethods || action.allowedMethods.length === 0) {
418
+ issues.push({
419
+ code: createCode('ACTION', 'UPDATE_MISSING_METHODS'),
420
+ message:
421
+ 'An Update action must define how the data is sent (PUT replaces everything; PATCH applies partial changes).',
422
+ suggestion: 'Select at least one allowed HTTP method.',
423
+ severity: 'error',
424
+ context: { ...context, property: 'allowedMethods' },
425
+ })
426
+ }
427
+ }
428
+
429
+ const allRateLimiters = action.getAllRateLimiters()
430
+ const oneHasRules = allRateLimiters.some((i) => i.rules.length)
431
+ if (!oneHasRules) {
432
+ issues.push({
433
+ code: createCode('ACTION', 'MISSING_RATE_LIMITS'),
434
+ message: 'It is best practice to configure a rate limit so your API is not overwhelmed.',
435
+ suggestion: 'Set a reasonable maximum request speed for this operation.',
436
+ severity: 'warning',
437
+ context,
438
+ })
439
+ }
440
+
441
+ return issues
442
+ }
443
+
444
+ export function validateExposedEntity(entity: ExposedEntity, apiModel: ApiModel): ApiModelValidationItem[] {
445
+ const issues: ApiModelValidationItem[] = []
446
+ const context: ApiModelValidationContext = {
447
+ apiModelKey: apiModel.key,
448
+ kind: ExposedEntityKind,
449
+ key: entity.key,
450
+ }
451
+
452
+ // Valid Entity Reference
453
+ if (!entity.entity || !entity.entity.key) {
454
+ issues.push({
455
+ code: createCode('EXPOSURE', 'MISSING_ENTITY_REF'),
456
+ message: 'This exposed endpoint does not point to a specific data model.',
457
+ suggestion: 'Select which database entity this endpoint should serve.',
458
+ severity: 'error',
459
+ context: { ...context, property: 'entity' },
460
+ })
461
+ } else if (apiModel.domain && !apiModel.domain.findEntity(entity.entity.key, entity.entity.domain)) {
462
+ issues.push({
463
+ code: createCode('EXPOSURE', 'INVALID_ENTITY_REF'),
464
+ message: 'This endpoint points to a data model that no longer exists.',
465
+ suggestion: 'Select a new valid database model or delete this endpoint.',
466
+ severity: 'error',
467
+ context: { ...context, property: 'entity' },
468
+ })
469
+ }
470
+
471
+ // Path Integrity
472
+ if (entity.hasCollection) {
473
+ if (!entity.collectionPath) {
474
+ issues.push({
475
+ code: createCode('EXPOSURE', 'MISSING_COLLECTION_PATH'),
476
+ message: 'When an endpoint exposes a collection, it must define what its base URL path is.',
477
+ suggestion: 'Add a path (e.g., "/items").',
478
+ severity: 'error',
479
+ context: { ...context, property: 'collectionPath' },
480
+ })
481
+ } else {
482
+ const parts = entity.collectionPath.split('/').filter(Boolean)
483
+ if (parts.length !== 1 || !entity.collectionPath.startsWith('/')) {
484
+ issues.push({
485
+ code: createCode('EXPOSURE', 'INVALID_COLLECTION_PATH'),
486
+ message: 'The collection URL should start with "/" and have no extra slashes.',
487
+ suggestion: `Ensure the path looks like exactly "/${parts[0] || 'subresource'}".`,
488
+ severity: 'error',
489
+ context: { ...context, property: 'collectionPath' },
490
+ })
491
+ }
492
+ }
493
+
494
+ if (!entity.resourcePath) {
495
+ issues.push({
496
+ code: createCode('EXPOSURE', 'MISSING_RESOURCE_PATH'),
497
+ message: 'You need an identifier to locate single items.',
498
+ suggestion: 'Specify the identification parameter format, like "{id}".',
499
+ severity: 'error',
500
+ context: { ...context, property: 'resourcePath' },
501
+ })
502
+ } else if (entity.collectionPath) {
503
+ const colRegex = new RegExp(`^${entity.collectionPath}/\\{[a-zA-Z0-9_]+\\}$`)
504
+ if (!colRegex.test(entity.resourcePath)) {
505
+ issues.push({
506
+ code: createCode('EXPOSURE', 'INVALID_RESOURCE_PATH_FORMAT'),
507
+ message: 'The single item route should match exactly your collection path plus one identifier variable.',
508
+ suggestion: `Make sure the item path is formatted exactly like "${entity.collectionPath}/{id}".`,
509
+ severity: 'error',
510
+ context: { ...context, property: 'resourcePath' },
511
+ })
512
+ }
513
+ }
514
+ } else {
515
+ if (!entity.resourcePath) {
516
+ issues.push({
517
+ code: createCode('EXPOSURE', 'MISSING_RESOURCE_PATH'),
518
+ message: 'Endpoints representing a single item must declare their URL route.',
519
+ suggestion: 'Set the URL path (such as "/profile").',
520
+ severity: 'error',
521
+ context: { ...context, property: 'resourcePath' },
522
+ })
523
+ } else {
524
+ const parts = entity.resourcePath.split('/').filter(Boolean)
525
+ if (parts.length !== 2 || !entity.resourcePath.startsWith('/')) {
526
+ issues.push({
527
+ code: createCode('EXPOSURE', 'INVALID_RESOURCE_PATH_FORMAT'),
528
+ message: 'The URL route must only contain two exact parts or levels.',
529
+ suggestion: 'Simplify the endpoint URL to prevent deep-nesting.',
530
+ severity: 'error',
531
+ context: { ...context, property: 'resourcePath' },
532
+ })
533
+ }
534
+ }
535
+ }
536
+
537
+ // Path Collisions
538
+ if (entity.isRoot) {
539
+ if (entity.collectionPath) {
540
+ const collectionCollision = apiModel.findCollectionPathCollision(entity.collectionPath, entity.key)
541
+ if (collectionCollision) {
542
+ issues.push({
543
+ code: createCode('EXPOSURE', 'ROOT_COLLECTION_COLLISION'),
544
+ message: `The route "${entity.collectionPath}" is already used by another view.`,
545
+ suggestion: 'Give this resource a different path to avoid conflicts.',
546
+ severity: 'error',
547
+ context: { ...context, property: 'collectionPath' },
548
+ })
549
+ }
550
+ }
551
+
552
+ if (entity.resourcePath) {
553
+ const resourceCollision = apiModel.findResourcePathCollision(entity.resourcePath, entity.key)
554
+ if (resourceCollision) {
555
+ issues.push({
556
+ code: createCode('EXPOSURE', 'ROOT_RESOURCE_COLLISION'),
557
+ message: `The route "${entity.resourcePath}" is already used by another view.`,
558
+ suggestion: 'Give this single-item resource a different path to avoid conflicts.',
559
+ severity: 'error',
560
+ context: { ...context, property: 'resourcePath' },
561
+ })
562
+ }
563
+ }
564
+ }
565
+
566
+ // Minimum Actions
567
+ if (!entity.actions || entity.actions.length === 0) {
568
+ issues.push({
569
+ code: createCode('EXPOSURE', 'MISSING_ACTIONS'),
570
+ message: 'This exposed view does not let users do anything.',
571
+ suggestion: 'Enable at least one operation like Create, Read, or Update.',
572
+ severity: 'error',
573
+ context: { ...context, property: 'actions' },
574
+ })
575
+ } else {
576
+ for (const action of entity.actions) {
577
+ issues.push(...validateAction(action, entity, apiModel.key))
578
+
579
+ // Check inheritance of access rules
580
+ // For a rule to exist, it might be on the action, the exposure, any parent, or the apiModel.
581
+ let hasAuth = false
582
+ if (action.accessRule && action.accessRule.length > 0) hasAuth = true
583
+ if (!hasAuth && entity.accessRule && entity.accessRule.length > 0) hasAuth = true
584
+
585
+ let curr = entity.parent?.key
586
+ while (curr && !hasAuth) {
587
+ const p = apiModel.exposes.get(curr)
588
+ if (p && p.accessRule && p.accessRule.length > 0) {
589
+ hasAuth = true
590
+ break
591
+ }
592
+ curr = p?.parent?.key
593
+ }
594
+
595
+ if (!hasAuth && apiModel.accessRule && apiModel.accessRule.length > 0) hasAuth = true
596
+
597
+ if (!hasAuth) {
598
+ issues.push({
599
+ code: createCode('ACTION', 'MISSING_ACCESS_RULES'),
600
+ message: `The ${action.kind} action has no security rules attached, making it entirely inaccessible or open.`,
601
+ suggestion: 'Allow specific user roles to access this operation.',
602
+ severity: 'error',
603
+ context: { ...context, kind: 'Action', key: action.kind }, // using action.kind as the key context equivalent
604
+ })
605
+ }
606
+ }
607
+ }
608
+
609
+ return issues
610
+ }
611
+
612
+ export function validateApiModel(model: ApiModel): ApiModelValidationResult {
613
+ const issues: ApiModelValidationItem[] = []
614
+
615
+ issues.push(...validateApiModelInfo(model))
616
+ issues.push(...validateApiModelDependency(model))
617
+ issues.push(...validateApiModelSecurity(model))
618
+ issues.push(...validateApiModelMetadata(model))
619
+
620
+ if (!model.exposes || model.exposes.size === 0) {
621
+ issues.push({
622
+ code: createCode('API', 'NO_EXPOSURES'),
623
+ message: 'Your API currently has no exposed data for external clients to request.',
624
+ suggestion: 'Expose a data model so apps can communicate with it.',
625
+ severity: 'warning',
626
+ context: { apiModelKey: model.key, kind: ApiModelKind, key: model.key, property: 'exposes' },
627
+ })
628
+ } else {
629
+ for (const exposure of model.exposes.values()) {
630
+ issues.push(...validateExposedEntity(exposure, model))
631
+ }
632
+ }
633
+
634
+ const hasErrors = issues.some((i) => i.severity === 'error')
635
+
636
+ return {
637
+ isValid: !hasErrors,
638
+ issues,
639
+ }
640
+ }