UNPKG

trimble-connect-sdk

Version:

Trimble Connect SDK for JavaScript

265 lines 42.7 kB
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; import { ServiceError } from './error'; import { ServiceResponse } from './response'; export function extractEndpoint(url) { const schemaIndex = url.indexOf('//'); const firstSeparatorIndex = url.indexOf('/', schemaIndex + 2); return url.substring(0, firstSeparatorIndex); } function delay(ms) { return new Promise((resolve) => { setTimeout(resolve, ms); }); } export class Service { constructor(config) { this.config = config; this.defaultRetryCount = 3; if (!this.config.serviceUri) { throw new Error('The serviceUri is required'); } } makeRequest(url, method = 'GET', body, customHeaders, auth = true) { return __awaiter(this, void 0, void 0, function* () { const headers = new Headers({ Accept: 'application/json' }); if (customHeaders) { customHeaders.forEach((value, key) => { headers.append(key, value); }); } const response = (yield this.request(url, method, body, headers, auth)); if (response.response.status !== 204) { const contentType = response.response.headers.get('content-type'); if (contentType === null || contentType.indexOf('application/json') !== -1) { response.data = (yield response.response.json()); } else { throw new ServiceError(response.response, 'Cannot deserialize response body as it is not in a JSON format.'); } } return response; }); } getItemsWithPages(url, onPageRetrieved, pageSize, body, customHeaders, auth = true) { return __awaiter(this, void 0, void 0, function* () { const headers = customHeaders !== null && customHeaders !== void 0 ? customHeaders : new Headers(); if (headers.has('Range')) { headers.delete('Range'); } const range = { start: 0, end: pageSize - 1 }; headers.append('Range', `items=${range.start}-${range.end}`); let response = yield this.makeRequest(url, 'GET', body, headers, auth); do { const responseRange = this.getRange(response); const hasNextPage = Array.isArray(response.data) && response.data.length !== 0 && responseRange && responseRange.total && responseRange.end < responseRange.total - 1; let nextPage; if (hasNextPage && responseRange && responseRange.total) { headers.delete('Range'); range.start = responseRange.end + 1; range.end = Math.min(responseRange.total - 1, range.start + pageSize - 1); headers.append('Range', `items=${range.start}-${range.end}`); nextPage = this.makeRequest(url, 'GET', body, headers, auth); } onPageRetrieved(response); if (nextPage) { response = yield nextPage; } else { break; } } while (true); }); } request(url, method = 'GET', body, customHeaders, auth = true) { return __awaiter(this, void 0, void 0, function* () { const requestUrl = url.startsWith('http') ? url : url.startsWith('/') ? extractEndpoint(this.config.serviceUri) + url : this.config.serviceUri + url; const headers = new Headers(); const contentType = customHeaders ? customHeaders.get('Content-Type') : undefined; if (!contentType && body !== undefined) { headers.set('Content-Type', 'application/json'); } if (customHeaders) { customHeaders.forEach((value, key) => { headers.append(key, value); }); } return this.fetchWithRetry(requestUrl, { body, headers, method, redirect: 'follow', }, auth); }); } maxRetries() { if (this.config.maxRetries !== undefined) { return this.config.maxRetries; } else { return this.defaultRetryCount; } } calculateRetryDelay(retryCount) { if (retryCount === 0) { return 0; } else { const base = 100; return Math.random() * (Math.pow(2, retryCount - 1) * base); } } retryableError(error) { if (this.timeoutError(error) || this.networkingError(error) || this.expiredCredentialsError(error) || this.throttledError(error) || error.response.status >= 500) { return true; } return false; } networkingError(error) { return error.errorCode === 'NetworkingError'; } timeoutError(error) { return error.errorCode === 'TimeoutError'; } expiredCredentialsError(error) { if (error.response.status === 401) { const credentials = this.config.credentials; if (credentials && typeof credentials.expired === 'boolean') { credentials.expired = true; } return true; } return false; } throttledError(error) { if (error.response.status === 429) { return true; } else { return false; } } fetchWithRetry(url, params, auth = true) { return __awaiter(this, void 0, void 0, function* () { const logger = this.config.logger; const result = new ServiceResponse(this, new Response(), void 0); do { const credentials = this.config.credentials; if (auth && credentials) { if (typeof credentials.get === 'function') { try { yield credentials.get(); } catch (err) { if (logger !== undefined) { this.log(`[TID] Failed to acquire tokens: ${err.message}`); } throw err; } } if (credentials.token) { params.headers.set('Authorization', 'Bearer ' + credentials.token); } } const startTime = Date.now(); const response = yield this.fetch(url, params); if (logger !== undefined) { const delta = (Date.now() - startTime) / 1000; const isoDate = new Date(startTime).toISOString(); const requestContentLength = params.body ? params.body.length : 0; const responseContentLength = response.headers.get('content-length'); const line = `${isoDate} [TC HTTP] ${params.method} ${url} ${response.status} ${delta} ${result.retryCount} ${requestContentLength} ${responseContentLength}`; this.log(line); } result.response = response; if (response.ok) { return result; } else { let err; const contentType = response.headers.get('content-type'); if (contentType && contentType.indexOf('application/json') !== -1) { const errorInfo = yield response.json(); err = new ServiceError(response, errorInfo.message, errorInfo.code || errorInfo.errorcode); } else { const message = yield response.text(); err = new ServiceError(response, message); } if (result.retryCount + 1 < this.maxRetries() && this.retryableError(err)) { const retryAfterHeader = response.headers.get('retry-after'); const retryAfterMS = retryAfterHeader ? parseInt(retryAfterHeader, 10) * 1000 : 0; const ms = this.calculateRetryDelay(result.retryCount) + retryAfterMS; yield delay(ms); result.retryCount++; } else { throw err; } } } while (true); }); } fetch(url, params) { return fetch(url, params); } getRange(response) { if (!response.response.headers.has('Content-Range')) { return undefined; } try { const contentRange = response.response.headers .get('Content-Range') .split(' '); const rangeTokens = contentRange[1].split('/'); const tokens = rangeTokens[0].split('-'); const start = Number(tokens[0]); const end = Number(tokens[1]); const total = Number(rangeTokens[1]); return { start, end, total }; } catch (_a) { return undefined; } } log(line) { return __awaiter(this, void 0, void 0, function* () { const logger = this.config.logger; if (logger !== undefined) { if (typeof logger.log === 'function') { logger.log(line); } else if (typeof logger.write === 'function') { logger.write(line + '\n'); } } }); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"service.js","sourceRoot":"./","sources":["service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Configuration } from './config';\nimport { Credentials } from './credentials';\nimport { Range } from './common';\nimport { ServiceError } from './error';\nimport { ServiceResponse } from './response';\nimport { ServiceCredentials } from './service_credentials';\n\n/** @internal */\nexport function extractEndpoint(url: string): string {\n  const schemaIndex = url.indexOf('//');\n  const firstSeparatorIndex = url.indexOf('/', schemaIndex + 2);\n\n  return url.substring(0, firstSeparatorIndex);\n}\n\nfunction delay(ms: number): Promise<void> {\n  return new Promise<void>((resolve) => {\n    setTimeout(resolve, ms);\n  });\n}\n\n/**\n * The service client that represents connection to the Organizer Service.\n * Each API operation is exposed as a function on service.\n */\nexport class Service {\n  /** Used if maxRetries is not specified in {@see Configuration#maxRetries }. The defaultRetryCount can be overriden by service classes. */\n  private defaultRetryCount: number = 3;\n\n  /**\n   * @constructor Constructs a service object.\n   * @param {Config} config The configuration options (e.g. service url).\n   */\n  constructor(public readonly config: Configuration) {\n    if (!this.config.serviceUri) {\n      throw new Error('The serviceUri is required');\n    }\n  }\n\n  /**\n   * Fetched the next page fromt he sequence of pages initiated by {@see Organizer.listTrees }.\n   * @param {string} url The url to make request to. Could be absolute or relative to the service base uri.\n   * @param {string} method The HTTP method.\n   * @param {string} body The body to send in the request (will be serialized to JSON).\n   * @param {string} headers The headers to attach to request.\n   * @param {boolean} auth The value indicating wether the request requires authentication. If true, the credentials will be requested from the configuration {@see Config#credentials } and Authorization header will be attached.\n   * @returns {Promise<ServiceResponse<D>>} The service response.\n   * @throws {ServiceError} in case of the error response from the service.\n   */\n  public async makeRequest<D>(\n    url: string,\n    method: 'GET' | 'PUT' | 'POST' | 'PATCH' | 'DELETE' = 'GET',\n    body?: string | FormData,\n    customHeaders?: Headers,\n    auth: boolean = true\n  ): Promise<ServiceResponse<D>> {\n    const headers = new Headers({ Accept: 'application/json' });\n    if (customHeaders) {\n      customHeaders.forEach((value: string, key: string) => {\n        headers.append(key, value);\n      });\n    }\n    const response = (await this.request(\n      url,\n      method,\n      body,\n      headers,\n      auth\n    )) as any as ServiceResponse<D>;\n\n    // deserialize response body. If service responded with an error it was thrown already.\n    if (response.response.status !== 204) {\n      const contentType = response.response.headers.get('content-type');\n      if (\n        contentType === null ||\n        contentType.indexOf('application/json') !== -1\n      ) {\n        response.data = (await response.response.json()) as D;\n      } else {\n        throw new ServiceError(\n          response.response,\n          'Cannot deserialize response body as it is not in a JSON format.'\n        );\n      }\n    }\n\n    return response;\n  }\n\n  /**\n   * Fetches all the items page by page, returning each page results in a callback.\n   * @param {string} url The url to make request to. Could be absolute or relative to the service base uri.\n   * @param {(response: ServiceResponse<D>) => void} onPageRetrieved The callback used to return results, page by page.\n   * @param {number} pageSize The page size used to request items.\n   * @param {string} body The body to send in the request (will be serialized to JSON).\n   * @param {string} customHeaders The headers to attach to request.\n   * @param {boolean} auth The value indicating wether the request requires authentication. If true, the credentials will be requested from the configuration {@see Config#credentials } and Authorization header will be attached.\n   * @returns {Promise<ServiceResponse<D>>} The service response.\n   * @throws {ServiceError} in case of the error response from the service.\n   */\n  public async getItemsWithPages<D>(\n    url: string,\n    onPageRetrieved: (response: ServiceResponse<D>) => void,\n    pageSize: number,\n    body?: string | FormData,\n    customHeaders?: Headers,\n    auth: boolean = true\n  ): Promise<void> {\n    const headers = customHeaders ?? new Headers();\n    if (headers.has('Range')) {\n      headers.delete('Range');\n    }\n    const range: Range = { start: 0, end: pageSize - 1 };\n    headers.append('Range', `items=${range.start}-${range.end}`);\n\n    let response = await this.makeRequest<D>(url, 'GET', body, headers, auth);\n\n    do {\n      const responseRange = this.getRange(response);\n\n      // if the response returns an empty array, or we have no content range header, we have acquired all the items.\n      // if the response range is equal to the total number of items, we have acquired all the items.\n      const hasNextPage =\n        Array.isArray(response.data) &&\n        response.data.length !== 0 &&\n        responseRange &&\n        responseRange.total &&\n        responseRange.end < responseRange.total - 1;\n\n      let nextPage: Promise<ServiceResponse<D>> | undefined;\n      if (hasNextPage && responseRange && responseRange.total) {\n        headers.delete('Range');\n        range.start = responseRange.end + 1;\n        range.end = Math.min(\n          responseRange.total - 1,\n          range.start + pageSize - 1\n        );\n        headers.append('Range', `items=${range.start}-${range.end}`);\n\n        // initiate next page request, but don't wait on it yet\n        nextPage = this.makeRequest<D>(url, 'GET', body, headers, auth);\n      }\n\n      onPageRetrieved(response);\n\n      if (nextPage) {\n        response = await nextPage;\n      } else {\n        break;\n      }\n\n      // if we reached this point, we still have items to acquire.\n    } while (true);\n  }\n\n  /**\n   * Makes a service request with all needed authenticationand retry logic. Converts error response to exception.\n   * @param {string} url The url to make request to. Could be absolute or relative to the service base uri.\n   * @param {string} method The HTTP method.\n   * @param {string} body The body to send in the request (will be serialized to JSON).\n   * @param {string} customHeaders The custom headers to attach to request.\n   * @param {boolean} auth The value indicating wether the request requires authentication. If true, the credentials will be requested from the configuration {@see Config#credentials } and Authorization header will be attached.\n   * @returns {Promise<ServiceResponse<D>>} The service response.\n   * @throws {ServiceError} in case of the error response from the service.\n   */\n  public async request(\n    url: string,\n    method: 'GET' | 'PUT' | 'POST' | 'PATCH' | 'DELETE' = 'GET',\n    body?: string | FormData | Blob,\n    customHeaders?: Headers,\n    auth: boolean = true\n  ): Promise<ServiceResponse<void>> {\n    // this method supports 3 cases: absolute url, absolute path, relative path\n    const requestUrl = url.startsWith('http')\n      ? url\n      : url.startsWith('/')\n      ? extractEndpoint(this.config.serviceUri) + url\n      : this.config.serviceUri + url;\n\n    const headers = new Headers();\n    const contentType = customHeaders\n      ? customHeaders.get('Content-Type')\n      : undefined;\n    if (!contentType && body !== undefined) {\n      headers.set('Content-Type', 'application/json');\n    }\n\n    if (customHeaders) {\n      customHeaders.forEach((value: string, key: string) => {\n        headers.append(key, value);\n      });\n    }\n\n    return this.fetchWithRetry(\n      requestUrl,\n      {\n        body,\n        headers,\n        method,\n        redirect: 'follow',\n      },\n      auth\n    );\n  }\n\n  /**\n   * How many times a failed request should be retried before giving up.\n   * the defaultRetryCount can be overriden by service classes.\n   *\n   * @api private\n   * @private\n   */\n  public maxRetries(): number {\n    if (this.config.maxRetries !== undefined) {\n      return this.config.maxRetries;\n    } else {\n      return this.defaultRetryCount;\n    }\n  }\n\n  /**\n   * First retry goes immediatly (0ms delay) then exponential growth with 100ms as a base.\n   * @api private\n   * @private\n   */\n  public calculateRetryDelay(retryCount: number): number {\n    if (retryCount === 0) {\n      return 0;\n    } else {\n      const base = 100;\n      return Math.random() * (Math.pow(2, retryCount - 1) * base);\n    }\n  }\n\n  /**\n   * @api private\n   * @private\n   */\n  private retryableError(error: ServiceError) {\n    if (\n      this.timeoutError(error) ||\n      this.networkingError(error) ||\n      this.expiredCredentialsError(error) ||\n      this.throttledError(error) ||\n      error.response.status >= 500\n    ) {\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * @api private\n   * @private\n   */\n  private networkingError(error: ServiceError) {\n    return error.errorCode === 'NetworkingError';\n  }\n\n  /**\n   * @api private\n   * @private\n   */\n  private timeoutError(error: ServiceError) {\n    return error.errorCode === 'TimeoutError';\n  }\n\n  /**\n   * @api private\n   * @private\n   */\n  private expiredCredentialsError(error: ServiceError) {\n    if (error.response.status === 401) {\n      const credentials = this.config.credentials;\n      if (\n        credentials &&\n        typeof (credentials as Credentials).expired === 'boolean'\n      ) {\n        (credentials as Credentials).expired = true;\n      }\n      return true;\n    }\n    return false;\n  }\n\n  /**\n   * @api private\n   * @private\n   */\n  private throttledError(error: ServiceError) {\n    // AWS API Gateway returns 429 in case of throttling errors\n    if (error.response.status === 429) {\n      return true;\n    } else {\n      return false;\n    }\n  }\n\n  /**\n   * Makes a service call and takes care about retry logic (re-authentication if needed).\n   * Note the <void> as a generic parameter. This method does not read the response body. The ServiceResponse return value will be cased to needed type on upper levels of the code.\n   * @param {string} url The url to make request to. Could be absolute or relative to the service base uri.\n   * @param {string} params The fetch parameters.\n   * @param {boolean} auth The value indicating wether the request requires authentication. If true, the credentials will be requested from the configuration {@see Config#credentials } and Authorization header will be attached.\n   * @returns {Promise<ServiceResponse<void>>} The service response.\n   * @throws {ServiceError} in case of the error response from the service.\n   * @private\n   */\n  private async fetchWithRetry(\n    url: string,\n    params: any,\n    auth: boolean = true\n  ): Promise<ServiceResponse<void>> {\n    const logger = this.config.logger;\n    const result: ServiceResponse<void> = new ServiceResponse<void>(\n      this,\n      new Response(),\n      void 0\n    );\n    do {\n      const credentials = this.config.credentials;\n      if (auth && credentials) {\n        if (typeof (credentials as ServiceCredentials).get === 'function') {\n          try {\n            await (credentials as ServiceCredentials).get();\n          } catch (err: any) {\n            if (logger !== undefined) {\n              this.log(`[TID] Failed to acquire tokens: ${err.message}`);\n            }\n\n            throw err;\n          }\n        }\n\n        if (credentials.token) {\n          params.headers.set('Authorization', 'Bearer ' + credentials.token);\n        }\n      }\n\n      const startTime = Date.now();\n      const response = await this.fetch(url, params);\n      if (logger !== undefined) {\n        const delta = (Date.now() - startTime) / 1000;\n        const isoDate = new Date(startTime).toISOString();\n        const requestContentLength = params.body ? params.body.length : 0;\n        const responseContentLength = response.headers.get('content-length');\n        const line = `${isoDate} [TC HTTP] ${params.method} ${url} ${response.status} ${delta} ${result.retryCount} ${requestContentLength} ${responseContentLength}`;\n        this.log(line);\n      }\n\n      result.response = response;\n\n      if (response.ok) {\n        return result;\n      } else {\n        let err: ServiceError;\n        const contentType = response.headers.get('content-type');\n        if (contentType && contentType.indexOf('application/json') !== -1) {\n          const errorInfo = await response.json();\n          err = new ServiceError(\n            response,\n            errorInfo.message,\n            errorInfo.code || errorInfo.errorcode\n          );\n        } else {\n          const message = await response.text();\n          err = new ServiceError(response, message);\n        }\n\n        if (\n          result.retryCount + 1 < this.maxRetries() &&\n          this.retryableError(err)\n        ) {\n          const retryAfterHeader = response.headers.get('retry-after');\n          const retryAfterMS = retryAfterHeader\n            ? parseInt(retryAfterHeader, 10) * 1000\n            : 0;\n          const ms = this.calculateRetryDelay(result.retryCount) + retryAfterMS;\n          await delay(ms);\n          result.retryCount++;\n        } else {\n          throw err;\n        }\n      }\n    } while (true);\n  }\n\n  /**\n   * The lowest level fetch function wrapper.\n   * This method allows to separte the step of actual sending of the request in the HTTP pipline and potentially intercept it and fire events.\n   * @param {string} url The url to make request to. Could be absolute or relative to the service base uri.\n   * @param {string} params The HTTP method.\n   * @private\n   */\n  private fetch(url: string, params: any): Promise<Response> {\n    return fetch(url, params);\n  }\n\n  /**\n   * Gets the next range for the rest of the results of response data.\n   * @param {ServiceResponse<D>} response The response for which we get the next range.\n   * @private\n   */\n  private getRange<D>(response: ServiceResponse<D>): Range | undefined {\n    if (!response.response.headers.has('Content-Range')) {\n      return undefined;\n    }\n\n    try {\n      const contentRange = response.response.headers\n        .get('Content-Range')!\n        .split(' ');\n      const rangeTokens = contentRange[1].split('/');\n      const tokens = rangeTokens[0].split('-');\n\n      const start = Number(tokens[0]);\n      const end = Number(tokens[1]);\n      const total = Number(rangeTokens[1]);\n\n      return { start, end, total };\n    } catch {\n      return undefined;\n    }\n  }\n\n  /**\n   * Put a line of text to the log.\n   * @param {string} line The line to log.\n   * @private\n   */\n  private async log(line: string) {\n    const logger = this.config.logger;\n    if (logger !== undefined) {\n      if (typeof logger.log === 'function') {\n        logger.log(line);\n      } else if (typeof logger.write === 'function') {\n        logger.write(line + '\\n');\n      }\n    }\n  }\n}\n"]}