@cdklabs/cdk-atmosphere-client 0.0.82 → 0.0.84

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.
package/lib/client.d.ts CHANGED
@@ -37,6 +37,20 @@ export interface Environment {
37
37
  */
38
38
  readonly region: string;
39
39
  }
40
+ /**
41
+ * An allocation constraint
42
+ */
43
+ export interface Constraint {
44
+ /**
45
+ * The constraint type.
46
+ */
47
+ readonly type: string;
48
+ /**
49
+ * Value qualifying the constraint.
50
+ * E.g. a list of regions.
51
+ */
52
+ readonly value: unknown;
53
+ }
40
54
  /**
41
55
  * An allocation of a single environment.
42
56
  */
@@ -53,6 +67,10 @@ export interface Allocation {
53
67
  * Credentials.
54
68
  */
55
69
  readonly credentials: Credentials;
70
+ /**
71
+ * Constraints used to fullfil this allocation, if any.
72
+ */
73
+ readonly constraints?: Constraint[];
56
74
  }
57
75
  export interface AcquireOptions {
58
76
  /**
@@ -69,6 +87,12 @@ export interface AcquireOptions {
69
87
  * @default 600
70
88
  */
71
89
  readonly timeoutSeconds?: number;
90
+ /**
91
+ * Constraints imposed on the environment request.
92
+ *
93
+ * @default - no constraints
94
+ */
95
+ readonly constraints?: Constraint[];
72
96
  }
73
97
  /**
74
98
  * Interface of a writable log stream
package/lib/client.js CHANGED
@@ -47,6 +47,7 @@ class AtmosphereClient {
47
47
  const acquired = await this.request('POST', '/allocations', {
48
48
  pool: options.pool,
49
49
  requester: options.requester,
50
+ constraints: options.constraints,
50
51
  });
51
52
  this.log(`Acquire | Successfully acquired environment from pool ${options.pool} (requester: ${options.requester})`);
52
53
  return acquired;
@@ -112,4 +113,4 @@ class AtmosphereClient {
112
113
  }
113
114
  }
114
115
  exports.AtmosphereClient = AtmosphereClient;
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":";;;AAAA,wEAAsE;AACtE,yCAAsC;AAEtC;;GAEG;AACH,MAAa,YAAa,SAAQ,KAAK;IACrC,YAA4B,UAAkB,EAAE,OAAe,EAAE,UAAmB;QAClF,KAAK,CAAC,GAAG,UAAU,IAAI,UAAU,CAAC,CAAC,CAAC,IAAI,UAAU,GAAG,CAAC,CAAC,CAAC,EAAE,KAAK,OAAO,EAAE,CAAC,CAAC;QADhD,eAAU,GAAV,UAAU,CAAQ;IAE9C,CAAC;CACF;AAJD,oCAIC;AAyGD;;;GAGG;AACH,MAAa,gBAAgB;IAI3B,YAAoC,QAAgB,EAAmB,UAAmC,EAAE;QAAxE,aAAQ,GAAR,QAAQ,CAAQ;QAAmB,YAAO,GAAP,OAAO,CAA8B;QAE1G,yDAAyD;QACzD,sEAAsE;QACtE,gDAAgD;QAChD,IAAK,MAAc,CAAC,MAAM,IAAI,IAAI,EAAE,CAAC;YACnC,iEAAiE;YAChE,MAAc,CAAC,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACI,KAAK,CAAC,OAAO,CAAC,OAAuB;QAE1C,MAAM,cAAc,GAAG,OAAO,CAAC,cAAc,IAAI,GAAG,CAAC;QACrD,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAC7B,MAAM,SAAS,GAAG,cAAc,GAAG,IAAI,CAAC;QAExC,IAAI,UAAU,GAAG,IAAI,CAAC,CAAC,sBAAsB;QAC7C,MAAM,aAAa,GAAG,KAAK,CAAC,CAAC,eAAe;QAE5C,IAAI,CAAC,GAAG,CAAC,oCAAoC,OAAO,CAAC,IAAI,kBAAkB,OAAO,CAAC,SAAS,IAAI,CAAC,CAAC;QAClG,OAAO,IAAI,EAAE,CAAC;YACZ,IAAI,CAAC;gBACH,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,cAAc,EAAE;oBAC1D,IAAI,EAAE,OAAO,CAAC,IAAI;oBAClB,SAAS,EAAE,OAAO,CAAC,SAAS;iBAC7B,CAAC,CAAC;gBACH,IAAI,CAAC,GAAG,CAAC,yDAAyD,OAAO,CAAC,IAAI,gBAAgB,OAAO,CAAC,SAAS,GAAG,CAAC,CAAC;gBACpH,OAAO,QAAQ,CAAC;YAClB,CAAC;YAAC,OAAO,KAAU,EAAE,CAAC;gBAEpB,4CAA4C;gBAC5C,IAAI,KAAK,CAAC,UAAU,KAAK,GAAG,EAAE,CAAC;oBAE7B,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC;oBACvC,IAAI,OAAO,IAAI,SAAS,EAAE,CAAC;wBACzB,MAAM,KAAK,CAAC;oBACd,CAAC;oBAED,IAAI,CAAC,GAAG,CAAC,8BAA8B,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;oBAExD,MAAM,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,UAAU,CAAC,CAAC,CAAC;oBAC9D,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,GAAG,CAAC,EAAE,aAAa,CAAC,CAAC;oBACrD,SAAS;gBACX,CAAC;gBAED,MAAM,KAAK,CAAC;YACd,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;OAGG;IACI,KAAK,CAAC,OAAO,CAAC,YAAoB,EAAE,OAAe;QACxD,IAAI,CAAC,GAAG,CAAC,yBAAyB,YAAY,gBAAgB,OAAO,IAAI,CAAC,CAAC;QAC3E,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,gBAAgB,YAAY,EAAE,EAAE,EAAE,OAAO,EAAE,CAAC,CAAC;QAC3F,IAAI,CAAC,GAAG,CAAC,+CAA+C,YAAY,gBAAgB,OAAO,IAAI,CAAC,CAAC;QACjG,OAAO,QAAQ,CAAC;IAClB,CAAC;IAEO,KAAK,CAAC,GAAG;QACf,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;YACf,MAAM,KAAK,GAAG,IAAI,CAAC,OAAO,CAAC,WAAW,IAAI,MAAM,IAAA,4CAAqB,GAAE,EAAE,CAAC;YAC1E,IAAI,CAAC,IAAI,GAAG,IAAI,qBAAS,CAAC;gBACxB,WAAW,EAAE,KAAK,CAAC,WAAW;gBAC9B,eAAe,EAAE,KAAK,CAAC,eAAe;gBACtC,YAAY,EAAE,KAAK,CAAC,YAAY;gBAChC,OAAO,EAAE,aAAa;aACvB,CAAC,CAAC;QACL,CAAC;QACD,OAAO,IAAI,CAAC,IAAI,CAAC;IACnB,CAAC;IAEO,KAAK,CAAC,OAAO,CAAC,MAAc,EAAE,IAAY,EAAE,IAAS;QAE3D,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,GAAG,EAAE,CAAC;QAE7B,MAAM,QAAQ,GAAG,MAAM,GAAG,CAAC,KAAK,CAAC,GAAG,IAAI,CAAC,QAAQ,GAAG,IAAI,EAAE,EAAE;YAC1D,MAAM;YACN,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC;SAC3B,CAAC,CAAC;QAEH,MAAM,YAAY,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAS,CAAC;QAElD,IAAI,QAAQ,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;YAC5B,OAAO,YAAY,CAAC;QACtB,CAAC;QAED,MAAM,IAAI,YAAY,CAAC,QAAQ,CAAC,MAAM,EAAE,YAAY,CAAC,OAAO,IAAI,eAAe,EAAE,QAAQ,CAAC,UAAU,CAAC,CAAC;IACxG,CAAC;IAEO,GAAG,CAAC,OAAe;QACzB,MAAM,IAAI,GAAG,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,KAAK,OAAO,EAAE,CAAC;QACxD,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,CAAC;YAC3B,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,KAAK,CAAC,GAAG,IAAI,IAAI,CAAC,CAAC;QAC5C,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACpB,CAAC;IACH,CAAC;CAEF;AAhHD,4CAgHC","sourcesContent":["import { fromNodeProviderChain } from '@aws-sdk/credential-providers';\nimport { AwsClient } from 'aws4fetch';\n\n/**\n * Error coming from the service.\n */\nexport class ServiceError extends Error {\n  constructor(public readonly statusCode: number, message: string, statusText?: string) {\n    super(`${statusCode} ${statusText ? `(${statusText})` : ''}: ${message}`);\n  }\n}\n\n/**\n * Credentials for a specific environment.\n *\n */\nexport interface Credentials {\n  /**\n   * AccessKeyId\n   */\n  readonly accessKeyId: string;\n\n  /**\n   * SecretAccessKey\n   */\n  readonly secretAccessKey: string;\n\n  /**\n   * SessionToken\n   *\n   */\n  readonly sessionToken: string;\n}\n\n/**\n * Environment information.\n */\nexport interface Environment {\n\n  /**\n   * Account ID.\n   */\n  readonly account: string;\n\n  /**\n   * Region.\n   */\n  readonly region: string;\n}\n\n/**\n * An allocation of a single environment.\n */\nexport interface Allocation {\n\n  /**\n   * The allocation id.\n   */\n  readonly id: string;\n\n  /**\n   * The allocated environment.\n   */\n  readonly environment: Environment;\n\n  /**\n   * Credentials.\n   */\n  readonly credentials: Credentials;\n\n}\n\nexport interface AcquireOptions {\n  /**\n   * Which pool to acquire an environment from.\n   */\n  readonly pool: string;\n  /**\n   * Identity for the requester.\n   */\n  readonly requester: string;\n  /**\n   * How many seconds to wait in case an environment is not immediately available.\n   *\n   * @default 600\n   */\n  readonly timeoutSeconds?: number;\n}\n\n/**\n * Interface of a writable log stream\n *\n * This interface should be assignment-compatible with `process.stdout` and\n * `process.stderr`, but at the same time not place too many Node implementation\n * requirements on implementors.\n */\nexport interface IWritable {\n  write(chunk: string): void;\n}\n\nexport interface AtmosphereClientOptions {\n  /**\n   * Direct logging messages to this stream if given\n   *\n   * @default - Use `console.log()`.\n   */\n  readonly logStream?: IWritable;\n  /**\n   * AWS credentials to use for requests\n   *\n   * @default - Use standard AWS credential provider chain\n   */\n  readonly credentials?: Credentials;\n}\n\n/**\n * Client for the Atmosphere service. Requires AWS credentials to be available\n * via standard mechanisms.\n */\nexport class AtmosphereClient {\n\n  private _aws: AwsClient | undefined;\n\n  public constructor(private readonly endpoint: string, private readonly options: AtmosphereClientOptions = {}) {\n\n    // aws4fetch relies on `crypto` being available globally.\n    // looks like in node < 20, even though it is included in the runtime,\n    // it isn't defined globally, so we polyfill it.\n    if ((global as any).crypto == null) {\n      // eslint-disable-next-line @typescript-eslint/no-require-imports\n      (global as any).crypto = require('crypto');\n    }\n  }\n\n  /**\n   * Waits until an environment could be allocated by the service.\n   *\n   * @returns allocation information.\n   * @throws if an environment could not be acquired within the specified timeout.\n   */\n  public async acquire(options: AcquireOptions): Promise<Allocation> {\n\n    const timeoutSeconds = options.timeoutSeconds ?? 600;\n    const startTime = Date.now();\n    const timeoutMs = timeoutSeconds * 1000;\n\n    let retryDelay = 1000; // start with 1 second\n    const maxRetryDelay = 60000; // max 1 minute\n\n    this.log(`Acquire | environment from pool '${options.pool}' (requester: '${options.requester}')`);\n    while (true) {\n      try {\n        const acquired = await this.request('POST', '/allocations', {\n          pool: options.pool,\n          requester: options.requester,\n        });\n        this.log(`Acquire | Successfully acquired environment from pool ${options.pool} (requester: ${options.requester})`);\n        return acquired;\n      } catch (error: any) {\n\n        // retry if no environment is available yet.\n        if (error.statusCode === 423) {\n\n          const elapsed = Date.now() - startTime;\n          if (elapsed >= timeoutMs) {\n            throw error;\n          }\n\n          this.log(`Acquire | Retrying due to: ${error.message}`);\n\n          await new Promise(resolve => setTimeout(resolve, retryDelay));\n          retryDelay = Math.min(retryDelay * 2, maxRetryDelay);\n          continue;\n        }\n\n        throw error;\n      }\n    }\n  }\n\n  /**\n   * Release an environment based on the allocation id. After releasing an environment,\n   * its provided credentials are deactivated.\n   */\n  public async release(allocationId: string, outcome: string) {\n    this.log(`Release | Allocation '${allocationId}' (outcome: '${outcome}')`);\n    const released = await this.request('DELETE', `/allocations/${allocationId}`, { outcome });\n    this.log(`Release | Successfully released allocation '${allocationId}' (outcome: '${outcome}')`);\n    return released;\n  }\n\n  private async aws(): Promise<AwsClient> {\n    if (!this._aws) {\n      const creds = this.options.credentials ?? await fromNodeProviderChain()();\n      this._aws = new AwsClient({\n        accessKeyId: creds.accessKeyId,\n        secretAccessKey: creds.secretAccessKey,\n        sessionToken: creds.sessionToken,\n        service: 'execute-api',\n      });\n    }\n    return this._aws;\n  }\n\n  private async request(method: string, path: string, body: any): Promise<any> {\n\n    const aws = await this.aws();\n\n    const response = await aws.fetch(`${this.endpoint}${path}`, {\n      method,\n      body: JSON.stringify(body),\n    });\n\n    const responseBody = await response.json() as any;\n\n    if (response.status === 200) {\n      return responseBody;\n    }\n\n    throw new ServiceError(response.status, responseBody.message ?? 'Unknown error', response.statusText);\n  }\n\n  private log(message: string) {\n    const line = `[${new Date().toISOString()}] ${message}`;\n    if (this.options.logStream) {\n      this.options.logStream.write(`${line}\\n`);\n    } else {\n      console.log(line);\n    }\n  }\n\n}"]}
116
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"client.js","sourceRoot":"","sources":["../src/client.ts"],"names":[],"mappings":"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","sourcesContent":["import { fromNodeProviderChain } from '@aws-sdk/credential-providers';\nimport { AwsClient } from 'aws4fetch';\n\n/**\n * Error coming from the service.\n */\nexport class ServiceError extends Error {\n  constructor(public readonly statusCode: number, message: string, statusText?: string) {\n    super(`${statusCode} ${statusText ? `(${statusText})` : ''}: ${message}`);\n  }\n}\n\n/**\n * Credentials for a specific environment.\n *\n */\nexport interface Credentials {\n  /**\n   * AccessKeyId\n   */\n  readonly accessKeyId: string;\n\n  /**\n   * SecretAccessKey\n   */\n  readonly secretAccessKey: string;\n\n  /**\n   * SessionToken\n   *\n   */\n  readonly sessionToken: string;\n}\n\n/**\n * Environment information.\n */\nexport interface Environment {\n\n  /**\n   * Account ID.\n   */\n  readonly account: string;\n\n  /**\n   * Region.\n   */\n  readonly region: string;\n}\n\n/**\n * An allocation constraint\n */\nexport interface Constraint {\n  /**\n   * The constraint type.\n   */\n  readonly type: string;\n  /**\n   * Value qualifying the constraint.\n   * E.g. a list of regions.\n   */\n  readonly value: unknown;\n}\n\n\n/**\n * An allocation of a single environment.\n */\nexport interface Allocation {\n\n  /**\n   * The allocation id.\n   */\n  readonly id: string;\n\n  /**\n   * The allocated environment.\n   */\n  readonly environment: Environment;\n\n  /**\n   * Credentials.\n   */\n  readonly credentials: Credentials;\n\n  /**\n   * Constraints used to fullfil this allocation, if any.\n   */\n  readonly constraints?: Constraint[];\n}\n\nexport interface AcquireOptions {\n  /**\n   * Which pool to acquire an environment from.\n   */\n  readonly pool: string;\n  /**\n   * Identity for the requester.\n   */\n  readonly requester: string;\n  /**\n   * How many seconds to wait in case an environment is not immediately available.\n   *\n   * @default 600\n   */\n  readonly timeoutSeconds?: number;\n  /**\n   * Constraints imposed on the environment request.\n   *\n   * @default - no constraints\n   */\n  readonly constraints?: Constraint[];\n}\n\n/**\n * Interface of a writable log stream\n *\n * This interface should be assignment-compatible with `process.stdout` and\n * `process.stderr`, but at the same time not place too many Node implementation\n * requirements on implementors.\n */\nexport interface IWritable {\n  write(chunk: string): void;\n}\n\nexport interface AtmosphereClientOptions {\n  /**\n   * Direct logging messages to this stream if given\n   *\n   * @default - Use `console.log()`.\n   */\n  readonly logStream?: IWritable;\n  /**\n   * AWS credentials to use for requests\n   *\n   * @default - Use standard AWS credential provider chain\n   */\n  readonly credentials?: Credentials;\n}\n\n/**\n * Client for the Atmosphere service. Requires AWS credentials to be available\n * via standard mechanisms.\n */\nexport class AtmosphereClient {\n\n  private _aws: AwsClient | undefined;\n\n  public constructor(private readonly endpoint: string, private readonly options: AtmosphereClientOptions = {}) {\n\n    // aws4fetch relies on `crypto` being available globally.\n    // looks like in node < 20, even though it is included in the runtime,\n    // it isn't defined globally, so we polyfill it.\n    if ((global as any).crypto == null) {\n      // eslint-disable-next-line @typescript-eslint/no-require-imports\n      (global as any).crypto = require('crypto');\n    }\n  }\n\n  /**\n   * Waits until an environment could be allocated by the service.\n   *\n   * @returns allocation information.\n   * @throws if an environment could not be acquired within the specified timeout.\n   */\n  public async acquire(options: AcquireOptions): Promise<Allocation> {\n\n    const timeoutSeconds = options.timeoutSeconds ?? 600;\n    const startTime = Date.now();\n    const timeoutMs = timeoutSeconds * 1000;\n\n    let retryDelay = 1000; // start with 1 second\n    const maxRetryDelay = 60000; // max 1 minute\n\n    this.log(`Acquire | environment from pool '${options.pool}' (requester: '${options.requester}')`);\n    while (true) {\n      try {\n        const acquired = await this.request('POST', '/allocations', {\n          pool: options.pool,\n          requester: options.requester,\n          constraints: options.constraints,\n        });\n        this.log(`Acquire | Successfully acquired environment from pool ${options.pool} (requester: ${options.requester})`);\n        return acquired;\n      } catch (error: any) {\n\n        // retry if no environment is available yet.\n        if (error.statusCode === 423) {\n\n          const elapsed = Date.now() - startTime;\n          if (elapsed >= timeoutMs) {\n            throw error;\n          }\n\n          this.log(`Acquire | Retrying due to: ${error.message}`);\n\n          await new Promise(resolve => setTimeout(resolve, retryDelay));\n          retryDelay = Math.min(retryDelay * 2, maxRetryDelay);\n          continue;\n        }\n\n        throw error;\n      }\n    }\n  }\n\n  /**\n   * Release an environment based on the allocation id. After releasing an environment,\n   * its provided credentials are deactivated.\n   */\n  public async release(allocationId: string, outcome: string) {\n    this.log(`Release | Allocation '${allocationId}' (outcome: '${outcome}')`);\n    const released = await this.request('DELETE', `/allocations/${allocationId}`, { outcome });\n    this.log(`Release | Successfully released allocation '${allocationId}' (outcome: '${outcome}')`);\n    return released;\n  }\n\n  private async aws(): Promise<AwsClient> {\n    if (!this._aws) {\n      const creds = this.options.credentials ?? await fromNodeProviderChain()();\n      this._aws = new AwsClient({\n        accessKeyId: creds.accessKeyId,\n        secretAccessKey: creds.secretAccessKey,\n        sessionToken: creds.sessionToken,\n        service: 'execute-api',\n      });\n    }\n    return this._aws;\n  }\n\n  private async request(method: string, path: string, body: any): Promise<any> {\n\n    const aws = await this.aws();\n\n    const response = await aws.fetch(`${this.endpoint}${path}`, {\n      method,\n      body: JSON.stringify(body),\n    });\n\n    const responseBody = await response.json() as any;\n\n    if (response.status === 200) {\n      return responseBody;\n    }\n\n    throw new ServiceError(response.status, responseBody.message ?? 'Unknown error', response.statusText);\n  }\n\n  private log(message: string) {\n    const line = `[${new Date().toISOString()}] ${message}`;\n    if (this.options.logStream) {\n      this.options.logStream.write(`${line}\\n`);\n    } else {\n      console.log(line);\n    }\n  }\n\n}\n"]}
package/package.json CHANGED
@@ -52,7 +52,7 @@
52
52
  "typescript": "^5.9.3"
53
53
  },
54
54
  "dependencies": {
55
- "@aws-sdk/credential-providers": "^3.962.0",
55
+ "@aws-sdk/credential-providers": "^3.966.0",
56
56
  "aws4fetch": "^1.0.20"
57
57
  },
58
58
  "engines": {
@@ -63,7 +63,7 @@
63
63
  "publishConfig": {
64
64
  "access": "public"
65
65
  },
66
- "version": "0.0.82",
66
+ "version": "0.0.84",
67
67
  "jest": {
68
68
  "coverageProvider": "v8",
69
69
  "coverageThreshold": {