@mittwald/api-models 0.1.0-alpha.1 → 0.1.0-alpha.2

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 (136) hide show
  1. package/README.md +107 -10
  2. package/dist/esm/app/AppInstallation/AppInstallation.js +43 -0
  3. package/dist/esm/app/AppInstallation/behaviors/api.js +21 -0
  4. package/dist/esm/app/AppInstallation/behaviors/index.js +2 -0
  5. package/dist/esm/app/AppInstallation/behaviors/types.js +1 -0
  6. package/dist/esm/app/AppInstallation/index.js +2 -0
  7. package/dist/esm/app/AppInstallation/types.js +1 -0
  8. package/dist/esm/app/index.js +1 -0
  9. package/dist/esm/base/DataModel.js +7 -0
  10. package/dist/esm/base/ReferenceModel.js +9 -0
  11. package/dist/esm/base/assertObjectFound.js +9 -0
  12. package/dist/esm/base/index.js +1 -0
  13. package/dist/esm/base/types.js +1 -0
  14. package/dist/esm/config/behaviors/api.js +34 -0
  15. package/dist/esm/config/behaviors/index.js +1 -0
  16. package/dist/esm/config/config.js +9 -0
  17. package/dist/esm/config/index.js +1 -0
  18. package/dist/esm/customer/Customer/Customer.js +43 -0
  19. package/dist/esm/customer/Customer/behaviors/api.js +20 -0
  20. package/dist/esm/customer/Customer/behaviors/index.js +2 -0
  21. package/dist/esm/customer/Customer/behaviors/types.js +1 -0
  22. package/dist/esm/customer/Customer/index.js +2 -0
  23. package/dist/esm/customer/Customer/types.js +1 -0
  24. package/dist/esm/customer/index.js +1 -0
  25. package/dist/esm/domain/Ingress/Ingress.js +56 -0
  26. package/dist/esm/domain/Ingress/behaviors/api.js +23 -0
  27. package/dist/esm/domain/Ingress/behaviors/index.js +2 -0
  28. package/dist/esm/domain/Ingress/behaviors/types.js +1 -0
  29. package/dist/esm/domain/Ingress/index.js +2 -0
  30. package/dist/esm/domain/Ingress/types.js +1 -0
  31. package/dist/esm/domain/IngressPath/IngressPath.js +15 -0
  32. package/dist/esm/domain/IngressPath/IngressPath.test.js +16 -0
  33. package/dist/esm/domain/IngressPath/index.js +2 -0
  34. package/dist/esm/domain/IngressPath/types.js +1 -0
  35. package/dist/esm/domain/IngressTarget/IngressTarget.js +51 -0
  36. package/dist/esm/domain/IngressTarget/IngressTarget.test-types.js +13 -0
  37. package/dist/esm/domain/IngressTarget/index.js +2 -0
  38. package/dist/esm/domain/IngressTarget/types.js +1 -0
  39. package/dist/esm/domain/index.js +1 -0
  40. package/dist/esm/errors/ObjectNotFoundError.js +7 -0
  41. package/dist/esm/index.js +7 -0
  42. package/dist/esm/lib/deepFreeze.js +3 -0
  43. package/dist/esm/lib/types.js +1 -0
  44. package/dist/esm/lib/withAsyncResourceVariant.js +5 -0
  45. package/dist/esm/project/Project/Project.js +69 -0
  46. package/dist/esm/project/Project/behaviors/api.js +51 -0
  47. package/dist/esm/project/Project/behaviors/inMem.js +23 -0
  48. package/dist/esm/project/Project/behaviors/index.js +3 -0
  49. package/dist/esm/project/Project/behaviors/types.js +1 -0
  50. package/dist/esm/project/Project/index.js +2 -0
  51. package/dist/esm/project/Project/types.js +1 -0
  52. package/dist/esm/project/index.js +1 -0
  53. package/dist/esm/react/MittwaldApiModelProvider.js +12 -0
  54. package/dist/esm/react/index.js +3 -0
  55. package/dist/esm/react/reactUsePromise.js +17 -0
  56. package/dist/esm/react.js +1 -0
  57. package/dist/esm/server/Server/Server.js +47 -0
  58. package/dist/esm/server/Server/behaviors/api.js +19 -0
  59. package/dist/esm/server/Server/behaviors/index.js +2 -0
  60. package/dist/esm/server/Server/behaviors/types.js +1 -0
  61. package/dist/esm/server/Server/index.js +1 -0
  62. package/dist/esm/server/Server/types.js +1 -0
  63. package/dist/esm/server/index.js +1 -0
  64. package/dist/types/app/AppInstallation/AppInstallation.d.ts +109 -0
  65. package/dist/types/app/AppInstallation/behaviors/api.d.ts +3 -0
  66. package/dist/types/app/AppInstallation/behaviors/index.d.ts +2 -0
  67. package/dist/types/app/AppInstallation/behaviors/types.d.ts +5 -0
  68. package/dist/types/app/AppInstallation/index.d.ts +2 -0
  69. package/dist/types/app/AppInstallation/types.d.ts +4 -0
  70. package/dist/types/app/index.d.ts +1 -0
  71. package/dist/types/base/DataModel.d.ts +5 -0
  72. package/dist/types/base/ReferenceModel.d.ts +5 -0
  73. package/dist/types/base/assertObjectFound.d.ts +3 -0
  74. package/dist/types/base/index.d.ts +1 -0
  75. package/dist/types/base/types.d.ts +2 -0
  76. package/dist/types/config/behaviors/api.d.ts +11 -0
  77. package/dist/types/config/behaviors/index.d.ts +1 -0
  78. package/dist/types/config/config.d.ts +16 -0
  79. package/dist/types/config/index.d.ts +1 -0
  80. package/dist/types/customer/Customer/Customer.d.ts +113 -0
  81. package/dist/types/customer/Customer/behaviors/api.d.ts +3 -0
  82. package/dist/types/customer/Customer/behaviors/index.d.ts +2 -0
  83. package/dist/types/customer/Customer/behaviors/types.d.ts +5 -0
  84. package/dist/types/customer/Customer/index.d.ts +2 -0
  85. package/dist/types/customer/Customer/types.d.ts +4 -0
  86. package/dist/types/customer/index.d.ts +1 -0
  87. package/dist/types/domain/Ingress/Ingress.d.ts +102 -0
  88. package/dist/types/domain/Ingress/behaviors/api.d.ts +3 -0
  89. package/dist/types/domain/Ingress/behaviors/index.d.ts +2 -0
  90. package/dist/types/domain/Ingress/behaviors/types.d.ts +5 -0
  91. package/dist/types/domain/Ingress/index.d.ts +2 -0
  92. package/dist/types/domain/Ingress/types.d.ts +6 -0
  93. package/dist/types/domain/IngressPath/IngressPath.d.ts +11 -0
  94. package/dist/types/domain/IngressPath/IngressPath.test.d.ts +1 -0
  95. package/dist/types/domain/IngressPath/index.d.ts +2 -0
  96. package/dist/types/domain/IngressPath/types.d.ts +2 -0
  97. package/dist/types/domain/IngressTarget/IngressTarget.d.ts +29 -0
  98. package/dist/types/domain/IngressTarget/IngressTarget.test-types.d.ts +1 -0
  99. package/dist/types/domain/IngressTarget/index.d.ts +2 -0
  100. package/dist/types/domain/IngressTarget/types.d.ts +6 -0
  101. package/dist/types/domain/index.d.ts +1 -0
  102. package/dist/types/errors/ObjectNotFoundError.d.ts +3 -0
  103. package/dist/types/index.d.ts +7 -0
  104. package/dist/types/lib/deepFreeze.d.ts +5 -0
  105. package/dist/types/lib/types.d.ts +2 -0
  106. package/dist/types/lib/withAsyncResourceVariant.d.ts +11 -0
  107. package/dist/types/project/Project/Project.d.ts +230 -0
  108. package/dist/types/project/Project/behaviors/api.d.ts +3 -0
  109. package/dist/types/project/Project/behaviors/inMem.d.ts +3 -0
  110. package/dist/types/project/Project/behaviors/index.d.ts +3 -0
  111. package/dist/types/project/Project/behaviors/types.d.ts +11 -0
  112. package/dist/types/project/Project/index.d.ts +2 -0
  113. package/dist/types/project/Project/types.d.ts +4 -0
  114. package/dist/types/project/index.d.ts +1 -0
  115. package/dist/types/react/MittwaldApiModelProvider.d.ts +2 -0
  116. package/dist/types/react/index.d.ts +4 -0
  117. package/dist/types/react/reactUsePromise.d.ts +1 -0
  118. package/dist/types/react.d.ts +1 -0
  119. package/dist/types/server/Server/Server.d.ts +72 -0
  120. package/dist/types/server/Server/behaviors/api.d.ts +3 -0
  121. package/dist/types/server/Server/behaviors/index.d.ts +2 -0
  122. package/dist/types/server/Server/behaviors/types.d.ts +5 -0
  123. package/dist/types/server/Server/index.d.ts +1 -0
  124. package/dist/types/server/Server/types.d.ts +4 -0
  125. package/dist/types/server/index.d.ts +1 -0
  126. package/package.json +23 -30
  127. package/dist/index.d.mts +0 -16601
  128. package/dist/index.d.ts +0 -16601
  129. package/dist/index.mjs +0 -704
  130. package/dist/index.mjs.map +0 -1
  131. package/dist/react.d.mts +0 -17
  132. package/dist/react.d.ts +0 -17
  133. package/dist/react.mjs +0 -17
  134. package/dist/react.mjs.map +0 -1
  135. package/dist/shared/api-models.54184fa2.mjs +0 -25
  136. package/dist/shared/api-models.54184fa2.mjs.map +0 -1
package/README.md CHANGED
@@ -20,18 +20,14 @@ yarn add @mittwald/api-models
20
20
 
21
21
  ## Setup
22
22
 
23
- You will need an initialized API client in order to operate with the models
24
- provided by this package. Provide it to the `setupApiBehaviors` method before
25
- any model operation is executed.
23
+ You will need to initialize an API client in order to operate with the models
24
+ provided by this package. Use the `api` global instance for initialization with
25
+ some methods.
26
26
 
27
27
  ```typescript
28
- import { setupApiBehaviors } from "@mittwald/api-models";
29
- import { MittwaldAPIV2Client } from "@mittwald/api-client";
28
+ import { api } from "@mittwald/api-models";
30
29
 
31
- // See the documentation of @mittwald/api-client for alternative methods of
32
- // client initialization: https://www.npmjs.com/package/@mittwald/api-client
33
- const client = MittwaldAPIClient.newUnauthenticated();
34
- setupApiBehaviors(client);
30
+ api.setupWithApiToken(process.env.MW_API_TOKEN);
35
31
  ```
36
32
 
37
33
  ## Examples
@@ -291,10 +287,111 @@ class ProjectDetailed {
291
287
  ```typescript
292
288
  class ProjectDetailed {
293
289
  public static async find(id: string): Promise<ProjectDetailed | undefined> {
294
- const data = await Project.behaviors.find(id);
290
+ const data = await config.project.behaviors.find(id);
295
291
  if (data !== undefined) {
296
292
  return new Project(data.id, data);
297
293
  }
298
294
  }
299
295
  }
300
296
  ```
297
+
298
+ #### How-to implement behaviors
299
+
300
+ Place a `behaviors` folder inside the model that should look like this:
301
+
302
+ ```
303
+ Project/
304
+ ├─ behaviors/
305
+ │ ├─ index.ts
306
+ │ ├─ types.ts (behavior interface)
307
+ │ ├─ api.ts (behavior implementation)
308
+ │ ├─ inmem.ts (behavior implementation)
309
+
310
+ ```
311
+
312
+ ##### Define `types.ts` first
313
+
314
+ It is a good starting point to first implement the interface for the behavior.
315
+ The interface usually just defines methods used in the behavior. Like
316
+
317
+ ```ts
318
+ export interface ProjectBehaviors {
319
+ find: (id: string) => Promise<ProjectData | undefined>;
320
+ updateDescription: (projectId: string, description: string) => Promise<void>;
321
+ }
322
+ ```
323
+
324
+ Then register the behavior in the global behavior configuration
325
+ `packages/models/src/config/config.ts`.
326
+
327
+ ##### Use the behaviors in the model
328
+
329
+ If the behavior interface is defined, you can start implementing the model. You
330
+ can also first implement the concrete API behavior, to "proof" the behavior is
331
+ "working" with the real API.
332
+
333
+ ```ts
334
+ import { config } from "../../config/config.js";
335
+
336
+ class ProjectDetailed {
337
+ public static async find(id: string): Promise<ProjectDetailed | undefined> {
338
+ const data = await config.project.behaviors.find(id);
339
+ if (data !== undefined) {
340
+ return new Project(data.id, data);
341
+ }
342
+ }
343
+ }
344
+ ```
345
+
346
+ ##### Implement the API behavior
347
+
348
+ The API behavior depends on an API client. You can implement the behavior as an
349
+ object factory, or a simple class implementing the interface. When using the
350
+ object factory, you do not have to redeclare the method parameter types.
351
+
352
+ Do the implementation specific stuff, thus preparing and executing the request,
353
+ and finally processing the response.
354
+
355
+ ```ts
356
+ import { ProjectBehaviors } from "./types.js";
357
+ import { assertStatus, MittwaldAPIV2Client } from "@mittwald/api-client";
358
+
359
+ export const apiProjectBehaviors = (
360
+ client: MittwaldAPIV2Client,
361
+ ): ProjectBehaviors => ({
362
+ find: async (id) => {
363
+ const response = await client.project.getProject({
364
+ projectId: id,
365
+ });
366
+
367
+ if (response.status === 200) {
368
+ return response.data;
369
+ }
370
+ assertStatus(response, 403);
371
+ },
372
+ });
373
+ ```
374
+
375
+ ### Prepare for React
376
+
377
+ All asynchronous methods should provide a `use`-method property. This method
378
+ uses
379
+ [@mittwald/react-use-promise](https://www.npmjs.com/package/@mittwald/react-use-promise)
380
+ under the hood to "resolve" the promise in the "React way".
381
+
382
+ To provide this feature to your _async_ model methods, wrap the actual method
383
+ with the `provideReact` enhancer.
384
+
385
+ ```ts
386
+ class ProjectDetailed {
387
+ public static find = provideReact(
388
+ async (id: string): Promise<ProjectDetailed | undefined> => {
389
+ const data = await config.behaviors.project.find(id);
390
+
391
+ if (data !== undefined) {
392
+ return new ProjectDetailed(data);
393
+ }
394
+ },
395
+ );
396
+ }
397
+ ```
@@ -0,0 +1,43 @@
1
+ import { config } from "../../config/config.js";
2
+ import { classes } from "polytype";
3
+ import { DataModel } from "../../base/DataModel.js";
4
+ import assertObjectFound from "../../base/assertObjectFound.js";
5
+ import { ReferenceModel } from "../../base/ReferenceModel.js";
6
+ import { withAsyncResourceVariant } from "../../lib/withAsyncResourceVariant.js";
7
+ export class AppInstallation extends ReferenceModel {
8
+ static find = withAsyncResourceVariant(async (id) => {
9
+ const data = await config.behaviors.appInstallation.find(id);
10
+ if (data !== undefined) {
11
+ return new AppInstallationDetailed(data);
12
+ }
13
+ });
14
+ static get = withAsyncResourceVariant(async (id) => {
15
+ const appInstallation = await this.find(id);
16
+ assertObjectFound(appInstallation, this, id);
17
+ return appInstallation;
18
+ });
19
+ static ofId(id) {
20
+ return new AppInstallation(id);
21
+ }
22
+ static list = withAsyncResourceVariant(async (projectId, query = {}) => {
23
+ const data = await config.behaviors.appInstallation.list(projectId, query);
24
+ return data.map((d) => new AppInstallationListItem(d));
25
+ });
26
+ getDetailed = withAsyncResourceVariant(() => AppInstallation.get(this.id));
27
+ }
28
+ // Common class for future extension
29
+ class AppInstallationCommon extends classes((DataModel), AppInstallation) {
30
+ constructor(data) {
31
+ super([data], [data.id]);
32
+ }
33
+ }
34
+ export class AppInstallationDetailed extends classes(AppInstallationCommon, (DataModel)) {
35
+ constructor(data) {
36
+ super([data], [data]);
37
+ }
38
+ }
39
+ export class AppInstallationListItem extends classes(AppInstallationCommon, (DataModel)) {
40
+ constructor(data) {
41
+ super([data], [data]);
42
+ }
43
+ }
@@ -0,0 +1,21 @@
1
+ import { assertStatus } from "@mittwald/api-client";
2
+ import { assertOneOfStatus } from "@mittwald/api-client";
3
+ export const apiAppInstallationBehaviors = (client) => ({
4
+ find: async (id) => {
5
+ const response = await client.app.getAppinstallation({
6
+ appInstallationId: id,
7
+ });
8
+ if (response.status === 200) {
9
+ return response.data;
10
+ }
11
+ assertOneOfStatus(response, [404]);
12
+ },
13
+ list: async (projectId, query) => {
14
+ const response = await client.app.listAppinstallations({
15
+ queryParameters: query,
16
+ projectId,
17
+ });
18
+ assertStatus(response, 200);
19
+ return response.data;
20
+ },
21
+ });
@@ -0,0 +1,2 @@
1
+ export * from "./api.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,2 @@
1
+ export * from "./AppInstallation.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1 @@
1
+ export * from "./AppInstallation/index.js";
@@ -0,0 +1,7 @@
1
+ import deepFreeze from "../lib/deepFreeze.js";
2
+ export class DataModel {
3
+ data;
4
+ constructor(data) {
5
+ this.data = deepFreeze(data);
6
+ }
7
+ }
@@ -0,0 +1,9 @@
1
+ export class ReferenceModel {
2
+ id;
3
+ constructor(id) {
4
+ this.id = id;
5
+ }
6
+ describe() {
7
+ return `${this.constructor.name}@${this.id}`;
8
+ }
9
+ }
@@ -0,0 +1,9 @@
1
+ import ObjectNotFoundError from "../errors/ObjectNotFoundError.js";
2
+ export default function assertObjectFound(obj, theClass, refIdOrObject) {
3
+ if (obj === undefined) {
4
+ const refName = typeof refIdOrObject === "string"
5
+ ? refIdOrObject
6
+ : refIdOrObject.toString();
7
+ throw new ObjectNotFoundError(theClass.name, refName);
8
+ }
9
+ }
@@ -0,0 +1 @@
1
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,34 @@
1
+ import { MittwaldAPIV2Client } from "@mittwald/api-client";
2
+ import { config } from "../config.js";
3
+ import { apiProjectBehaviors } from "../../project/Project/behaviors/index.js";
4
+ import { apiServerBehaviors } from "../../server/Server/behaviors/index.js";
5
+ import { apiCustomerBehaviors } from "../../customer/Customer/behaviors/index.js";
6
+ import { apiIngressBehaviors } from "../../domain/Ingress/behaviors/index.js";
7
+ import { apiAppInstallationBehaviors } from "../../app/AppInstallation/behaviors/index.js";
8
+ class ApiSetupState {
9
+ _client;
10
+ setupWithClient(client) {
11
+ this._client = client;
12
+ config.behaviors.project = apiProjectBehaviors(client);
13
+ config.behaviors.server = apiServerBehaviors(client);
14
+ config.behaviors.customer = apiCustomerBehaviors(client);
15
+ config.behaviors.ingress = apiIngressBehaviors(client);
16
+ config.behaviors.appInstallation = apiAppInstallationBehaviors(client);
17
+ }
18
+ setupWithApiToken(apiToken) {
19
+ return this.setupWithClient(MittwaldAPIV2Client.newWithToken(apiToken));
20
+ }
21
+ get client() {
22
+ if (!this._client) {
23
+ throw new Error("Could not get client. Behavior not initialized.");
24
+ }
25
+ return this._client;
26
+ }
27
+ get defaults() {
28
+ return this.client.axios.defaults;
29
+ }
30
+ get interceptors() {
31
+ return this.client.axios.interceptors;
32
+ }
33
+ }
34
+ export const api = new ApiSetupState();
@@ -0,0 +1 @@
1
+ export * from "./api.js";
@@ -0,0 +1,9 @@
1
+ export const config = {
2
+ behaviors: {
3
+ project: undefined,
4
+ server: undefined,
5
+ customer: undefined,
6
+ ingress: undefined,
7
+ appInstallation: undefined,
8
+ },
9
+ };
@@ -0,0 +1 @@
1
+ export * from "./behaviors/index.js";
@@ -0,0 +1,43 @@
1
+ import { config } from "../../config/config.js";
2
+ import { classes } from "polytype";
3
+ import { DataModel } from "../../base/DataModel.js";
4
+ import assertObjectFound from "../../base/assertObjectFound.js";
5
+ import { ReferenceModel } from "../../base/ReferenceModel.js";
6
+ import { withAsyncResourceVariant } from "../../lib/withAsyncResourceVariant.js";
7
+ export class Customer extends ReferenceModel {
8
+ static ofId(id) {
9
+ return new Customer(id);
10
+ }
11
+ static find = withAsyncResourceVariant(async (id) => {
12
+ const data = await config.behaviors.customer.find(id);
13
+ if (data !== undefined) {
14
+ return new CustomerDetailed(data);
15
+ }
16
+ });
17
+ static list = withAsyncResourceVariant(async (query = {}) => {
18
+ const data = await config.behaviors.customer.list(query);
19
+ return Object.freeze(data.map((d) => new CustomerListItem(d)));
20
+ });
21
+ static get = withAsyncResourceVariant(async (id) => {
22
+ const customer = await this.find(id);
23
+ assertObjectFound(customer, this, id);
24
+ return customer;
25
+ });
26
+ getDetailed = withAsyncResourceVariant(() => Customer.get(this.id));
27
+ }
28
+ // Common class for future extension
29
+ class CustomerCommon extends classes((DataModel), Customer) {
30
+ constructor(data) {
31
+ super([data], [data.customerId]);
32
+ }
33
+ }
34
+ export class CustomerDetailed extends classes(CustomerCommon, (DataModel)) {
35
+ constructor(data) {
36
+ super([data], [data]);
37
+ }
38
+ }
39
+ export class CustomerListItem extends classes(CustomerCommon, (DataModel)) {
40
+ constructor(data) {
41
+ super([data], [data]);
42
+ }
43
+ }
@@ -0,0 +1,20 @@
1
+ import { assertStatus } from "@mittwald/api-client";
2
+ import { assertOneOfStatus } from "@mittwald/api-client";
3
+ export const apiCustomerBehaviors = (client) => ({
4
+ find: async (id) => {
5
+ const response = await client.customer.getCustomer({
6
+ customerId: id,
7
+ });
8
+ if (response.status === 200) {
9
+ return response.data;
10
+ }
11
+ assertOneOfStatus(response, [404]);
12
+ },
13
+ list: async (query) => {
14
+ const response = await client.customer.listCustomers({
15
+ queryParameters: query,
16
+ });
17
+ assertStatus(response, 200);
18
+ return response.data;
19
+ },
20
+ });
@@ -0,0 +1,2 @@
1
+ export * from "./api.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,2 @@
1
+ export * from "./Customer.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1 @@
1
+ export * from "./Customer/index.js";
@@ -0,0 +1,56 @@
1
+ import { config } from "../../config/config.js";
2
+ import { classes } from "polytype";
3
+ import { DataModel } from "../../base/DataModel.js";
4
+ import assertObjectFound from "../../base/assertObjectFound.js";
5
+ import { ReferenceModel } from "../../base/ReferenceModel.js";
6
+ import { withAsyncResourceVariant } from "../../lib/withAsyncResourceVariant.js";
7
+ import { IngressPath } from "../IngressPath/IngressPath.js";
8
+ export class Ingress extends ReferenceModel {
9
+ static ofId(id) {
10
+ return new Ingress(id);
11
+ }
12
+ static ofHostname(hostname) {
13
+ return Ingress.ofId(hostname);
14
+ }
15
+ static list = withAsyncResourceVariant(async (query = {}) => {
16
+ const data = await config.behaviors.ingress.list(query);
17
+ return data.map((d) => new IngressListItem(d));
18
+ });
19
+ static find = withAsyncResourceVariant(async (id) => {
20
+ const data = await config.behaviors.ingress.find(id);
21
+ if (data !== undefined) {
22
+ return new IngressDetailed(data);
23
+ }
24
+ });
25
+ static get = withAsyncResourceVariant(async (id) => {
26
+ const ingress = await this.find(id);
27
+ assertObjectFound(ingress, this, id);
28
+ return ingress;
29
+ });
30
+ getDetailed = withAsyncResourceVariant(() => Ingress.get(this.id));
31
+ }
32
+ export class IngressCommon extends classes((DataModel), Ingress) {
33
+ baseUrl;
34
+ paths;
35
+ defaultPath;
36
+ constructor(data) {
37
+ super([data], [data.id]);
38
+ this.baseUrl = `https://${data.hostname}`;
39
+ this.paths = Object.freeze(data.paths.map((p) => new IngressPath(this, p)));
40
+ const defaultPath = this.paths.find((p) => p.path === "/");
41
+ if (defaultPath === undefined) {
42
+ throw new Error(`Ingress ${this.describe()} has no default path.`);
43
+ }
44
+ this.defaultPath = defaultPath;
45
+ }
46
+ }
47
+ export class IngressDetailed extends classes(IngressCommon, (DataModel)) {
48
+ constructor(data) {
49
+ super([data], [data]);
50
+ }
51
+ }
52
+ export class IngressListItem extends classes(IngressCommon, (DataModel)) {
53
+ constructor(data) {
54
+ super([data], [data]);
55
+ }
56
+ }
@@ -0,0 +1,23 @@
1
+ import { assertStatus } from "@mittwald/api-client";
2
+ import { assertOneOfStatus } from "@mittwald/api-client";
3
+ export const apiIngressBehaviors = (client) => ({
4
+ find: async (id) => {
5
+ const response = await client.domain.ingressGetIngress({
6
+ ingressId: id,
7
+ });
8
+ if (response.status === 200) {
9
+ return response.data;
10
+ }
11
+ assertOneOfStatus(response, [404]);
12
+ },
13
+ list: async (query = {}) => {
14
+ const { projectId } = query;
15
+ const response = await client.domain.ingressListIngresses({
16
+ queryParameters: {
17
+ projectId,
18
+ },
19
+ });
20
+ assertStatus(response, 200);
21
+ return response.data;
22
+ },
23
+ });
@@ -0,0 +1,2 @@
1
+ export * from "./api.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,2 @@
1
+ export * from "./Ingress.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,15 @@
1
+ import { DataModel } from "../../base/DataModel.js";
2
+ import { ingressTargetFactory, } from "../IngressTarget/IngressTarget.js";
3
+ export class IngressPath extends DataModel {
4
+ ingress;
5
+ path;
6
+ url;
7
+ target;
8
+ constructor(ingress, data) {
9
+ super(data);
10
+ this.ingress = ingress;
11
+ this.path = data.path;
12
+ this.url = new URL(data.path, ingress.baseUrl);
13
+ this.target = ingressTargetFactory(this, data.target);
14
+ }
15
+ }
@@ -0,0 +1,16 @@
1
+ import { IngressPath } from "./IngressPath.js";
2
+ import { IngressDetailed } from "../Ingress/index.js";
3
+ const ingressData = {
4
+ id: "abc",
5
+ paths: [],
6
+ };
7
+ const corruptIngressTargetData = {};
8
+ const ingressPathData = {
9
+ path: "/",
10
+ target: corruptIngressTargetData,
11
+ };
12
+ test("Creating IngressPath with corrupt IngressTarget throws error", () => {
13
+ expect(() => {
14
+ new IngressPath(new IngressDetailed(ingressData), ingressPathData);
15
+ }).toThrowError("Ingress IngressDetailed@abc has no default path.");
16
+ });
@@ -0,0 +1,2 @@
1
+ export * from "./IngressPath.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,51 @@
1
+ import { DataModel } from "../../base/DataModel.js";
2
+ import { AppInstallation } from "../../app/AppInstallation/index.js";
3
+ class IngressTargetBase extends DataModel {
4
+ path;
5
+ constructor(path, data) {
6
+ super(data);
7
+ this.path = path;
8
+ }
9
+ }
10
+ export class IngressRedirectTarget extends IngressTargetBase {
11
+ type = "redirect";
12
+ url;
13
+ constructor(path, data) {
14
+ super(path, data);
15
+ this.url = new URL(data.url);
16
+ }
17
+ }
18
+ export class IngressDirectoryTarget extends IngressTargetBase {
19
+ type = "directory";
20
+ directory;
21
+ constructor(path, data) {
22
+ super(path, data);
23
+ this.directory = data.directory;
24
+ }
25
+ }
26
+ export class IngressAppInstallationTarget extends IngressTargetBase {
27
+ type = "appInstallation";
28
+ appInstallation;
29
+ constructor(path, data) {
30
+ super(path, data);
31
+ this.appInstallation = AppInstallation.ofId(data.installationId);
32
+ }
33
+ }
34
+ export class IngressUndefinedTarget extends IngressTargetBase {
35
+ type = "undefined";
36
+ }
37
+ export const ingressTargetFactory = (path, data) => {
38
+ if ("directory" in data) {
39
+ return new IngressDirectoryTarget(path, data);
40
+ }
41
+ if ("url" in data) {
42
+ return new IngressRedirectTarget(path, data);
43
+ }
44
+ if ("installationId" in data) {
45
+ return new IngressAppInstallationTarget(path, data);
46
+ }
47
+ if ("useDefaultPage" in data) {
48
+ return new IngressUndefinedTarget(path, data);
49
+ }
50
+ throw new Error("Ingress target type is not supported.");
51
+ };
@@ -0,0 +1,13 @@
1
+ const target = {};
2
+ // @ts-expect-error use type-guard first
3
+ void target.url;
4
+ if (target.type === "redirect") {
5
+ void target.url;
6
+ }
7
+ if (target.type === "directory") {
8
+ void target.directory;
9
+ }
10
+ if (target.type === "appInstallation") {
11
+ void target.appInstallation.id;
12
+ }
13
+ export {};
@@ -0,0 +1,2 @@
1
+ export * from "./IngressTarget.js";
2
+ export * from "./types.js";
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1 @@
1
+ export * from "./Ingress/index.js";
@@ -0,0 +1,7 @@
1
+ export default class ObjectNotFoundError extends Error {
2
+ constructor(type, refName) {
3
+ super(`${type}@${refName} not found`);
4
+ this.name = "ObjectNotFoundError";
5
+ Object.setPrototypeOf(this, ObjectNotFoundError.prototype);
6
+ }
7
+ }
@@ -0,0 +1,7 @@
1
+ export { MittwaldAPIV2Client } from "@mittwald/api-client";
2
+ export * from "./config/index.js";
3
+ export * from "./project/index.js";
4
+ export * from "./server/index.js";
5
+ export * from "./domain/index.js";
6
+ export * from "./customer/index.js";
7
+ export * from "./base/index.js";
@@ -0,0 +1,3 @@
1
+ import deepFreezeLib from "another-deep-freeze";
2
+ const deepFreeze = deepFreezeLib.default;
3
+ export default deepFreeze;
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,5 @@
1
+ import { reactUsePromise } from "../react/reactUsePromise.js";
2
+ export const withAsyncResourceVariant = (loader) => Object.assign(loader, {
3
+ asResource: (...params) => reactUsePromise.getAsyncResource(loader, params),
4
+ use: (...params) => reactUsePromise.getAsyncResource(loader, params).use(),
5
+ });