@veho/turvo-integration-sdk 0.1.0-beta.0 → 0.1.0-beta.1
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/CHANGELOG.md +22 -0
- package/lib/cjs/api/turvoPublicApi.d.ts +3 -3
- package/lib/cjs/api/turvoPublicApi.js +2 -2
- package/lib/cjs/db/client.d.ts +9 -0
- package/lib/cjs/db/client.js +37 -0
- package/lib/cjs/db/facilityMapping.d.ts +27 -0
- package/lib/cjs/db/facilityMapping.js +178 -0
- package/lib/cjs/shipmentTracking/trackingService.js +54 -14
- package/lib/cjs/types/errors.d.ts +6 -0
- package/lib/cjs/types/errors.js +12 -2
- package/lib/cjs/types/facilityMapping.d.ts +9 -0
- package/lib/cjs/types/facilityMapping.js +3 -0
- package/lib/cjs/types/shipment.d.ts +11 -0
- package/lib/cjs/types/shipment.js +1 -1
- package/lib/cjs/types/tracking.d.ts +13 -11
- package/lib/cjs/types/tracking.js +1 -1
- package/lib/esm/api/turvoPublicApi.d.ts +3 -3
- package/lib/esm/api/turvoPublicApi.js +2 -2
- package/lib/esm/db/client.d.ts +9 -0
- package/lib/esm/db/client.js +26 -0
- package/lib/esm/db/facilityMapping.d.ts +27 -0
- package/lib/esm/db/facilityMapping.js +171 -0
- package/lib/esm/shipmentTracking/trackingService.js +55 -15
- package/lib/esm/types/errors.d.ts +6 -0
- package/lib/esm/types/errors.js +10 -1
- package/lib/esm/types/facilityMapping.d.ts +9 -0
- package/lib/esm/types/facilityMapping.js +2 -0
- package/lib/esm/types/shipment.d.ts +11 -0
- package/lib/esm/types/shipment.js +1 -1
- package/lib/esm/types/tracking.d.ts +13 -11
- package/lib/esm/types/tracking.js +1 -1
- package/lib/tsconfig.cjs.tsbuildinfo +1 -1
- package/lib/tsconfig.esm.tsbuildinfo +1 -1
- package/package.json +5 -1
package/CHANGELOG.md
ADDED
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
# Changelog
|
|
2
|
+
|
|
3
|
+
All notable changes to this project will be documented in this file.
|
|
4
|
+
|
|
5
|
+
## [0.1.0-beta.1] - 2026-01-29
|
|
6
|
+
|
|
7
|
+
### Added
|
|
8
|
+
- Initial SDK implementation for Turvo API integration
|
|
9
|
+
- TurvoClient for managing API interactions
|
|
10
|
+
- Turvo Public API client with shipment operations
|
|
11
|
+
- Turvo Internal API client for additional functionality
|
|
12
|
+
- Facility mapping functionality with DynamoDB integration
|
|
13
|
+
- Shipment tracking service with detailed tracking information
|
|
14
|
+
- Comprehensive TypeScript types for Turvo entities
|
|
15
|
+
- Support for both CommonJS and ESM modules
|
|
16
|
+
|
|
17
|
+
### Features
|
|
18
|
+
- Create, update, and query shipments
|
|
19
|
+
- Retrieve shipment tracking details
|
|
20
|
+
- Manage facility code mappings
|
|
21
|
+
- Handle authentication and rate limiting
|
|
22
|
+
- Error handling with custom error types
|
|
@@ -32,10 +32,10 @@ export interface CancelShipmentParams {
|
|
|
32
32
|
export interface UpdateShipmentStatusParams {
|
|
33
33
|
/** The Turvo shipment ID */
|
|
34
34
|
turvoShipmentId: number;
|
|
35
|
-
/** The Turvo stop ID */
|
|
36
|
-
turvoStopId
|
|
35
|
+
/** The Turvo stop ID (optional, depending on type of status update) */
|
|
36
|
+
turvoStopId?: number;
|
|
37
37
|
/** The status code to set */
|
|
38
|
-
turvoStatusCode: TurvoShipmentStatus;
|
|
38
|
+
turvoStatusCode: TurvoShipmentStatus['code'];
|
|
39
39
|
/** The status date in ISO format */
|
|
40
40
|
statusDate: string;
|
|
41
41
|
/** The timezone for the status date */
|
|
@@ -73,7 +73,7 @@ class TurvoPublicApi {
|
|
|
73
73
|
id: turvoShipmentId,
|
|
74
74
|
status: {
|
|
75
75
|
globalShipLocationId: turvoStopId,
|
|
76
|
-
code: turvoStatusCode
|
|
76
|
+
code: turvoStatusCode,
|
|
77
77
|
timezone: statusTimezone,
|
|
78
78
|
statusDate: {
|
|
79
79
|
date: statusDate,
|
|
@@ -100,4 +100,4 @@ class TurvoPublicApi {
|
|
|
100
100
|
}
|
|
101
101
|
}
|
|
102
102
|
exports.TurvoPublicApi = TurvoPublicApi;
|
|
103
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turvoPublicApi.js","sourceRoot":"","sources":["../../../src/api/turvoPublicApi.ts"],"names":[],"mappings":"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","sourcesContent":["import { TurvoClient } from '../client/turvoClient'\nimport {\n  TurvoApiMethod,\n  TurvoApiResult,\n  TurvoFilterShipmentsResultItem,\n  TurvoLookup,\n  TurvoPagedResult,\n} from '../types/common'\nimport {\n  TurvoAssociateTagsApiResult,\n  TurvoCancelShipmentApiResult,\n  TurvoCreateShipmentApiResult,\n  TurvoShipment,\n  TurvoShipmentStatus,\n  TurvoUpdateShipmentStatusApiResult,\n} from '../types/shipment'\n\n/**\n * Parameters for uploading a shipment to Turvo\n */\nexport interface UploadShipmentParams {\n  /** The formatted shipment to upload */\n  shipment: TurvoShipment\n  /** The Load ID, solely for logging purposes */\n  vehoLoadId: string\n}\n\n/**\n * Parameters for associating tags to a shipment\n */\nexport interface AssociateTagsParams {\n  /** The shipment ID to associate tags with */\n  turvoShipmentId: number\n  /** List of tag names to assign */\n  tags: string[]\n}\n\n/**\n * Parameters for cancelling a shipment\n */\nexport interface CancelShipmentParams {\n  /** The Turvo system Shipment ID to cancel */\n  turvoShipmentId: number\n}\n\n/**\n * Parameters for updating shipment status\n */\nexport interface UpdateShipmentStatusParams {\n  /** The Turvo shipment ID */\n  turvoShipmentId: number\n  /** The Turvo stop ID */\n  turvoStopId: number\n  /** The status code to set */\n  turvoStatusCode: TurvoShipmentStatus\n  /** The status date in ISO format */\n  statusDate: string\n  /** The timezone for the status date */\n  statusTimezone: string\n}\n\n/**\n * Parameters for retrieving a shipment\n */\nexport interface GetShipmentParams {\n  /** The shipment ID to retrieve */\n  shipmentId: number\n}\n\n/**\n * Parameters for filtering shipments\n */\nexport interface FilterShipmentsParams {\n  /** Turvo location ID to filter by */\n  turvoLocationId: number\n  /** Pickup date start filter (ISO format) */\n  pickupDateStart: string\n  /** Optional pagination start offset */\n  start?: number\n}\n\n/**\n * TurvoPublicApi provides access to documented Turvo API endpoints.\n * All shipment management operations (create, update, cancel, query) go through this class.\n */\nexport class TurvoPublicApi {\n  constructor(private client: TurvoClient) {}\n\n  /**\n   * GET /v1/shipments/{id}\n   * Returns shipment details, stops, status\n   */\n  async getShipment(params: GetShipmentParams): Promise<TurvoApiResult<TurvoShipment>> {\n    const { shipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoShipment>(`/shipments/${shipmentId}`, TurvoApiMethod.GET, {})\n  }\n\n  /**\n   * POST /v1/shipments\n   * Upload properly formatted shipment to Turvo\n   */\n  async uploadShipment(params: UploadShipmentParams): Promise<TurvoApiResult<TurvoCreateShipmentApiResult>> {\n    const { shipment } = params\n    // Note: vehoLoadId is kept in the params for logging purposes by consumers\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoCreateShipmentApiResult>('/shipments', TurvoApiMethod.POST, {\n      body: shipment,\n    })\n  }\n\n  /**\n   * PUT /v1/tags/attach/shipment/{id}\n   * Associate a list of tags with a shipment by ID\n   */\n  async associateTagsToShipment(params: AssociateTagsParams): Promise<TurvoApiResult<TurvoAssociateTagsApiResult>> {\n    const { turvoShipmentId, tags } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoAssociateTagsApiResult>(\n      `/tags/attach/shipment/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body: {\n          tagNames: tags,\n        },\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Set shipment status to Cancelled in Turvo\n   */\n  async cancelShipment(params: CancelShipmentParams): Promise<TurvoApiResult<TurvoCancelShipmentApiResult>> {\n    const { turvoShipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        code: { key: '2113', value: 'Canceled' } satisfies TurvoLookup,\n      },\n    }\n\n    return httpClient.sendRequest<TurvoCancelShipmentApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body,\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Update shipment status at a specific stop\n   */\n  async updateShipmentStatus(\n    params: UpdateShipmentStatusParams\n  ): Promise<TurvoApiResult<TurvoUpdateShipmentStatusApiResult>> {\n    const { turvoShipmentId, turvoStopId, turvoStatusCode, statusDate, statusTimezone } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        globalShipLocationId: turvoStopId,\n        code: turvoStatusCode.code,\n        timezone: statusTimezone,\n        statusDate: {\n          date: statusDate,\n          timezone: statusTimezone,\n        },\n      },\n    }\n\n    return httpClient.sendRequest<TurvoUpdateShipmentStatusApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      { body }\n    )\n  }\n\n  /**\n   * GET /v1/shipments/list\n   * Query shipments from Turvo with pagination support\n   */\n  async filterShipments(\n    params: FilterShipmentsParams\n  ): Promise<TurvoApiResult<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>> {\n    const { turvoLocationId, pickupDateStart, start = 0 } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>(\n      '/shipments/list',\n      TurvoApiMethod.GET,\n      {\n        query: {\n          'locationId[eq]': turvoLocationId.toString(),\n          'pickupDate[gte]': pickupDateStart,\n          start: start.toString(),\n        },\n      }\n    )\n  }\n}\n"]}
|
|
103
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turvoPublicApi.js","sourceRoot":"","sources":["../../../src/api/turvoPublicApi.ts"],"names":[],"mappings":"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","sourcesContent":["import { TurvoClient } from '../client/turvoClient'\nimport {\n  TurvoApiMethod,\n  TurvoApiResult,\n  TurvoFilterShipmentsResultItem,\n  TurvoLookup,\n  TurvoPagedResult,\n} from '../types/common'\nimport {\n  TurvoAssociateTagsApiResult,\n  TurvoCancelShipmentApiResult,\n  TurvoCreateShipmentApiResult,\n  TurvoShipment,\n  TurvoShipmentStatus,\n  TurvoUpdateShipmentStatusApiResult,\n} from '../types/shipment'\n\n/**\n * Parameters for uploading a shipment to Turvo\n */\nexport interface UploadShipmentParams {\n  /** The formatted shipment to upload */\n  shipment: TurvoShipment\n  /** The Load ID, solely for logging purposes */\n  vehoLoadId: string\n}\n\n/**\n * Parameters for associating tags to a shipment\n */\nexport interface AssociateTagsParams {\n  /** The shipment ID to associate tags with */\n  turvoShipmentId: number\n  /** List of tag names to assign */\n  tags: string[]\n}\n\n/**\n * Parameters for cancelling a shipment\n */\nexport interface CancelShipmentParams {\n  /** The Turvo system Shipment ID to cancel */\n  turvoShipmentId: number\n}\n\n/**\n * Parameters for updating shipment status\n */\nexport interface UpdateShipmentStatusParams {\n  /** The Turvo shipment ID */\n  turvoShipmentId: number\n  /** The Turvo stop ID (optional, depending on type of status update) */\n  turvoStopId?: number\n  /** The status code to set */\n  turvoStatusCode: TurvoShipmentStatus['code']\n  /** The status date in ISO format */\n  statusDate: string\n  /** The timezone for the status date */\n  statusTimezone: string\n}\n\n/**\n * Parameters for retrieving a shipment\n */\nexport interface GetShipmentParams {\n  /** The shipment ID to retrieve */\n  shipmentId: number\n}\n\n/**\n * Parameters for filtering shipments\n */\nexport interface FilterShipmentsParams {\n  /** Turvo location ID to filter by */\n  turvoLocationId: number\n  /** Pickup date start filter (ISO format) */\n  pickupDateStart: string\n  /** Optional pagination start offset */\n  start?: number\n}\n\n/**\n * TurvoPublicApi provides access to documented Turvo API endpoints.\n * All shipment management operations (create, update, cancel, query) go through this class.\n */\nexport class TurvoPublicApi {\n  constructor(private client: TurvoClient) {}\n\n  /**\n   * GET /v1/shipments/{id}\n   * Returns shipment details, stops, status\n   */\n  async getShipment(params: GetShipmentParams): Promise<TurvoApiResult<TurvoShipment>> {\n    const { shipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoShipment>(`/shipments/${shipmentId}`, TurvoApiMethod.GET, {})\n  }\n\n  /**\n   * POST /v1/shipments\n   * Upload properly formatted shipment to Turvo\n   */\n  async uploadShipment(params: UploadShipmentParams): Promise<TurvoApiResult<TurvoCreateShipmentApiResult>> {\n    const { shipment } = params\n    // Note: vehoLoadId is kept in the params for logging purposes by consumers\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoCreateShipmentApiResult>('/shipments', TurvoApiMethod.POST, {\n      body: shipment,\n    })\n  }\n\n  /**\n   * PUT /v1/tags/attach/shipment/{id}\n   * Associate a list of tags with a shipment by ID\n   */\n  async associateTagsToShipment(params: AssociateTagsParams): Promise<TurvoApiResult<TurvoAssociateTagsApiResult>> {\n    const { turvoShipmentId, tags } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoAssociateTagsApiResult>(\n      `/tags/attach/shipment/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body: {\n          tagNames: tags,\n        },\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Set shipment status to Cancelled in Turvo\n   */\n  async cancelShipment(params: CancelShipmentParams): Promise<TurvoApiResult<TurvoCancelShipmentApiResult>> {\n    const { turvoShipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        code: { key: '2113', value: 'Canceled' } satisfies TurvoLookup,\n      },\n    }\n\n    return httpClient.sendRequest<TurvoCancelShipmentApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body,\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Update shipment status at a specific stop\n   */\n  async updateShipmentStatus(\n    params: UpdateShipmentStatusParams\n  ): Promise<TurvoApiResult<TurvoUpdateShipmentStatusApiResult>> {\n    const { turvoShipmentId, turvoStopId, turvoStatusCode, statusDate, statusTimezone } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        globalShipLocationId: turvoStopId,\n        code: turvoStatusCode,\n        timezone: statusTimezone,\n        statusDate: {\n          date: statusDate,\n          timezone: statusTimezone,\n        },\n      },\n    }\n\n    return httpClient.sendRequest<TurvoUpdateShipmentStatusApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      { body }\n    )\n  }\n\n  /**\n   * GET /v1/shipments/list\n   * Query shipments from Turvo with pagination support\n   */\n  async filterShipments(\n    params: FilterShipmentsParams\n  ): Promise<TurvoApiResult<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>> {\n    const { turvoLocationId, pickupDateStart, start = 0 } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>(\n      '/shipments/list',\n      TurvoApiMethod.GET,\n      {\n        query: {\n          'locationId[eq]': turvoLocationId.toString(),\n          'pickupDate[gte]': pickupDateStart,\n          start: start.toString(),\n        },\n      }\n    )\n  }\n}\n"]}
|
|
@@ -0,0 +1,9 @@
|
|
|
1
|
+
import { DynamoDBDocumentClient } from '@aws-sdk/lib-dynamodb';
|
|
2
|
+
export declare const createDocClient: () => DynamoDBDocumentClient;
|
|
3
|
+
export declare const getDocClient: () => DynamoDBDocumentClient;
|
|
4
|
+
export declare const getIndexNameByTurvoLocationId: () => string | undefined;
|
|
5
|
+
export declare const getTableName: () => string | undefined;
|
|
6
|
+
export declare const getIndexNameByShortCode: () => string | undefined;
|
|
7
|
+
export declare const getRetiredFacilityMappingTableName: () => string | undefined;
|
|
8
|
+
export declare const getFacilityCodeMapTable: () => string | undefined;
|
|
9
|
+
export declare const getIndexNameByFacilityCode: () => string | undefined;
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getIndexNameByFacilityCode = exports.getFacilityCodeMapTable = exports.getRetiredFacilityMappingTableName = exports.getIndexNameByShortCode = exports.getTableName = exports.getIndexNameByTurvoLocationId = exports.getDocClient = exports.createDocClient = void 0;
|
|
4
|
+
const client_dynamodb_1 = require("@aws-sdk/client-dynamodb");
|
|
5
|
+
const lib_dynamodb_1 = require("@aws-sdk/lib-dynamodb");
|
|
6
|
+
// Create DynamoDB client without X-Ray tracing for SDK usage
|
|
7
|
+
const createDocClient = () => {
|
|
8
|
+
return lib_dynamodb_1.DynamoDBDocumentClient.from(new client_dynamodb_1.DynamoDBClient({}), {
|
|
9
|
+
marshallOptions: {
|
|
10
|
+
convertEmptyValues: true,
|
|
11
|
+
},
|
|
12
|
+
});
|
|
13
|
+
};
|
|
14
|
+
exports.createDocClient = createDocClient;
|
|
15
|
+
// Singleton instance
|
|
16
|
+
let docClientInstance = null;
|
|
17
|
+
const getDocClient = () => {
|
|
18
|
+
if (!docClientInstance) {
|
|
19
|
+
docClientInstance = (0, exports.createDocClient)();
|
|
20
|
+
}
|
|
21
|
+
return docClientInstance;
|
|
22
|
+
};
|
|
23
|
+
exports.getDocClient = getDocClient;
|
|
24
|
+
// Environment-based table names and indexes
|
|
25
|
+
const getIndexNameByTurvoLocationId = () => process.env.INDEX_NAME_BY_TURVO_LOCATION_ID;
|
|
26
|
+
exports.getIndexNameByTurvoLocationId = getIndexNameByTurvoLocationId;
|
|
27
|
+
const getTableName = () => process.env.TABLE_NAME;
|
|
28
|
+
exports.getTableName = getTableName;
|
|
29
|
+
const getIndexNameByShortCode = () => process.env.INDEX_NAME_BY_SHORT_CODE;
|
|
30
|
+
exports.getIndexNameByShortCode = getIndexNameByShortCode;
|
|
31
|
+
const getRetiredFacilityMappingTableName = () => process.env.RETIRED_FACILITY_MAPPINGS_TABLE_NAME;
|
|
32
|
+
exports.getRetiredFacilityMappingTableName = getRetiredFacilityMappingTableName;
|
|
33
|
+
const getFacilityCodeMapTable = () => process.env.FACILITY_CODE_MAP_TABLE;
|
|
34
|
+
exports.getFacilityCodeMapTable = getFacilityCodeMapTable;
|
|
35
|
+
const getIndexNameByFacilityCode = () => process.env.INDEX_NAME_BY_FACILITY_CODE;
|
|
36
|
+
exports.getIndexNameByFacilityCode = getIndexNameByFacilityCode;
|
|
37
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xpZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2RiL2NsaWVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSw4REFBeUQ7QUFDekQsd0RBQThEO0FBRTlELDZEQUE2RDtBQUN0RCxNQUFNLGVBQWUsR0FBRyxHQUFHLEVBQUU7SUFDbEMsT0FBTyxxQ0FBc0IsQ0FBQyxJQUFJLENBQUMsSUFBSSxnQ0FBYyxDQUFDLEVBQUUsQ0FBQyxFQUFFO1FBQ3pELGVBQWUsRUFBRTtZQUNmLGtCQUFrQixFQUFFLElBQUk7U0FDekI7S0FDRixDQUFDLENBQUE7QUFDSixDQUFDLENBQUE7QUFOWSxRQUFBLGVBQWUsbUJBTTNCO0FBRUQscUJBQXFCO0FBQ3JCLElBQUksaUJBQWlCLEdBQWtDLElBQUksQ0FBQTtBQUVwRCxNQUFNLFlBQVksR0FBRyxHQUEyQixFQUFFO0lBQ3ZELElBQUksQ0FBQyxpQkFBaUIsRUFBRSxDQUFDO1FBQ3ZCLGlCQUFpQixHQUFHLElBQUEsdUJBQWUsR0FBRSxDQUFBO0lBQ3ZDLENBQUM7SUFDRCxPQUFPLGlCQUFpQixDQUFBO0FBQzFCLENBQUMsQ0FBQTtBQUxZLFFBQUEsWUFBWSxnQkFLeEI7QUFFRCw0Q0FBNEM7QUFDckMsTUFBTSw2QkFBNkIsR0FBRyxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLCtCQUErQixDQUFBO0FBQWpGLFFBQUEsNkJBQTZCLGlDQUFvRDtBQUN2RixNQUFNLFlBQVksR0FBRyxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQTtBQUEzQyxRQUFBLFlBQVksZ0JBQStCO0FBQ2pELE1BQU0sdUJBQXVCLEdBQUcsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQyx3QkFBd0IsQ0FBQTtBQUFwRSxRQUFBLHVCQUF1QiwyQkFBNkM7QUFDMUUsTUFBTSxrQ0FBa0MsR0FBRyxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLG9DQUFvQyxDQUFBO0FBQTNGLFFBQUEsa0NBQWtDLHNDQUF5RDtBQUNqRyxNQUFNLHVCQUF1QixHQUFHLEdBQUcsRUFBRSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsdUJBQXVCLENBQUE7QUFBbkUsUUFBQSx1QkFBdUIsMkJBQTRDO0FBQ3pFLE1BQU0sMEJBQTBCLEdBQUcsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEdBQUcsQ0FBQywyQkFBMkIsQ0FBQTtBQUExRSxRQUFBLDBCQUEwQiw4QkFBZ0QiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEeW5hbW9EQkNsaWVudCB9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1keW5hbW9kYidcbmltcG9ydCB7IER5bmFtb0RCRG9jdW1lbnRDbGllbnQgfSBmcm9tICdAYXdzLXNkay9saWItZHluYW1vZGInXG5cbi8vIENyZWF0ZSBEeW5hbW9EQiBjbGllbnQgd2l0aG91dCBYLVJheSB0cmFjaW5nIGZvciBTREsgdXNhZ2VcbmV4cG9ydCBjb25zdCBjcmVhdGVEb2NDbGllbnQgPSAoKSA9PiB7XG4gIHJldHVybiBEeW5hbW9EQkRvY3VtZW50Q2xpZW50LmZyb20obmV3IER5bmFtb0RCQ2xpZW50KHt9KSwge1xuICAgIG1hcnNoYWxsT3B0aW9uczoge1xuICAgICAgY29udmVydEVtcHR5VmFsdWVzOiB0cnVlLFxuICAgIH0sXG4gIH0pXG59XG5cbi8vIFNpbmdsZXRvbiBpbnN0YW5jZVxubGV0IGRvY0NsaWVudEluc3RhbmNlOiBEeW5hbW9EQkRvY3VtZW50Q2xpZW50IHwgbnVsbCA9IG51bGxcblxuZXhwb3J0IGNvbnN0IGdldERvY0NsaWVudCA9ICgpOiBEeW5hbW9EQkRvY3VtZW50Q2xpZW50ID0+IHtcbiAgaWYgKCFkb2NDbGllbnRJbnN0YW5jZSkge1xuICAgIGRvY0NsaWVudEluc3RhbmNlID0gY3JlYXRlRG9jQ2xpZW50KClcbiAgfVxuICByZXR1cm4gZG9jQ2xpZW50SW5zdGFuY2Vcbn1cblxuLy8gRW52aXJvbm1lbnQtYmFzZWQgdGFibGUgbmFtZXMgYW5kIGluZGV4ZXNcbmV4cG9ydCBjb25zdCBnZXRJbmRleE5hbWVCeVR1cnZvTG9jYXRpb25JZCA9ICgpID0+IHByb2Nlc3MuZW52LklOREVYX05BTUVfQllfVFVSVk9fTE9DQVRJT05fSURcbmV4cG9ydCBjb25zdCBnZXRUYWJsZU5hbWUgPSAoKSA9PiBwcm9jZXNzLmVudi5UQUJMRV9OQU1FXG5leHBvcnQgY29uc3QgZ2V0SW5kZXhOYW1lQnlTaG9ydENvZGUgPSAoKSA9PiBwcm9jZXNzLmVudi5JTkRFWF9OQU1FX0JZX1NIT1JUX0NPREVcbmV4cG9ydCBjb25zdCBnZXRSZXRpcmVkRmFjaWxpdHlNYXBwaW5nVGFibGVOYW1lID0gKCkgPT4gcHJvY2Vzcy5lbnYuUkVUSVJFRF9GQUNJTElUWV9NQVBQSU5HU19UQUJMRV9OQU1FXG5leHBvcnQgY29uc3QgZ2V0RmFjaWxpdHlDb2RlTWFwVGFibGUgPSAoKSA9PiBwcm9jZXNzLmVudi5GQUNJTElUWV9DT0RFX01BUF9UQUJMRVxuZXhwb3J0IGNvbnN0IGdldEluZGV4TmFtZUJ5RmFjaWxpdHlDb2RlID0gKCkgPT4gcHJvY2Vzcy5lbnYuSU5ERVhfTkFNRV9CWV9GQUNJTElUWV9DT0RFXG4iXX0=
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Facility mapping functions for Turvo location to Veho facility mapping
|
|
3
|
+
*
|
|
4
|
+
* This file contains methods for mapping between Turvo locations and Veho facilities.
|
|
5
|
+
* It abstracts over three different tables:
|
|
6
|
+
* 1. FACILITY_CODE_MAP_TABLE: stores the map using the current facilityCode naming scheme
|
|
7
|
+
* 2. SHORT_CODE_MAP_TABLE: stores the map using the old shortCode naming scheme
|
|
8
|
+
* 3. RETIRED_FACILITY_MAPPING_TABLE_NAME: stores retired facility mappings
|
|
9
|
+
*/
|
|
10
|
+
import { VehoTurvoLocationMap } from '../types/facilityMapping';
|
|
11
|
+
/**
|
|
12
|
+
* Get facility mapping for a Turvo location ID
|
|
13
|
+
* Checks both facility code and short code mappings
|
|
14
|
+
*/
|
|
15
|
+
export declare const getFacilityMappingForTurvoLocationId: (turvoLocationId: number) => Promise<VehoTurvoLocationMap | null>;
|
|
16
|
+
/**
|
|
17
|
+
* Get retired facility mapping for a Turvo location ID
|
|
18
|
+
*/
|
|
19
|
+
export declare const getRetiredFacilityMappingForTurvoLocationId: (turvoLocationId: number) => Promise<VehoTurvoLocationMap | null>;
|
|
20
|
+
/**
|
|
21
|
+
* Get retired facility mapping by human identifier (facility code or short code)
|
|
22
|
+
*/
|
|
23
|
+
export declare const getRetiredFacilityMappingByHumanIdentifier: (humanIdentifier: string) => Promise<VehoTurvoLocationMap | null>;
|
|
24
|
+
/**
|
|
25
|
+
* Get facility mapping by human identifier (facility code)
|
|
26
|
+
*/
|
|
27
|
+
export declare const getFacilityMappingByHumanIdentifier: (identifier: string) => Promise<VehoTurvoLocationMap | null>;
|
|
@@ -0,0 +1,178 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Facility mapping functions for Turvo location to Veho facility mapping
|
|
4
|
+
*
|
|
5
|
+
* This file contains methods for mapping between Turvo locations and Veho facilities.
|
|
6
|
+
* It abstracts over three different tables:
|
|
7
|
+
* 1. FACILITY_CODE_MAP_TABLE: stores the map using the current facilityCode naming scheme
|
|
8
|
+
* 2. SHORT_CODE_MAP_TABLE: stores the map using the old shortCode naming scheme
|
|
9
|
+
* 3. RETIRED_FACILITY_MAPPING_TABLE_NAME: stores retired facility mappings
|
|
10
|
+
*/
|
|
11
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
12
|
+
exports.getFacilityMappingByHumanIdentifier = exports.getRetiredFacilityMappingByHumanIdentifier = exports.getRetiredFacilityMappingForTurvoLocationId = exports.getFacilityMappingForTurvoLocationId = void 0;
|
|
13
|
+
const lib_dynamodb_1 = require("@aws-sdk/lib-dynamodb");
|
|
14
|
+
const client_1 = require("./client");
|
|
15
|
+
/**
|
|
16
|
+
* Get facility mapping for a Turvo location ID
|
|
17
|
+
* Checks both facility code and short code mappings
|
|
18
|
+
*/
|
|
19
|
+
const getFacilityMappingForTurvoLocationId = async (turvoLocationId) => {
|
|
20
|
+
if (typeof turvoLocationId !== 'number') {
|
|
21
|
+
return null;
|
|
22
|
+
}
|
|
23
|
+
// Check new FACILITY_CODE_MAP_TABLE first
|
|
24
|
+
const facilityCodeMapping = await getFacilityCodeMappingForTurvoLocationId(turvoLocationId);
|
|
25
|
+
if (facilityCodeMapping) {
|
|
26
|
+
return facilityCodeMapping;
|
|
27
|
+
}
|
|
28
|
+
// Then check the SHORT_CODE_MAP_TABLE
|
|
29
|
+
const shortCodeMapping = await getShortCodeMappingForTurvoLocationId(turvoLocationId);
|
|
30
|
+
if (shortCodeMapping) {
|
|
31
|
+
return shortCodeMapping;
|
|
32
|
+
}
|
|
33
|
+
return null;
|
|
34
|
+
};
|
|
35
|
+
exports.getFacilityMappingForTurvoLocationId = getFacilityMappingForTurvoLocationId;
|
|
36
|
+
/**
|
|
37
|
+
* Get retired facility mapping for a Turvo location ID
|
|
38
|
+
*/
|
|
39
|
+
const getRetiredFacilityMappingForTurvoLocationId = async (turvoLocationId) => {
|
|
40
|
+
if (typeof turvoLocationId !== 'number') {
|
|
41
|
+
return null;
|
|
42
|
+
}
|
|
43
|
+
const tableName = (0, client_1.getRetiredFacilityMappingTableName)();
|
|
44
|
+
const indexName = (0, client_1.getIndexNameByTurvoLocationId)();
|
|
45
|
+
if (!tableName || !indexName) {
|
|
46
|
+
return null;
|
|
47
|
+
}
|
|
48
|
+
const dbQuery = new lib_dynamodb_1.QueryCommand({
|
|
49
|
+
TableName: tableName,
|
|
50
|
+
IndexName: indexName,
|
|
51
|
+
KeyConditionExpression: 'turvoLocationId = :turvoLocationId',
|
|
52
|
+
ExpressionAttributeValues: {
|
|
53
|
+
':turvoLocationId': turvoLocationId,
|
|
54
|
+
},
|
|
55
|
+
});
|
|
56
|
+
const response = await (0, client_1.getDocClient)().send(dbQuery);
|
|
57
|
+
if (!response.Items?.length) {
|
|
58
|
+
return null;
|
|
59
|
+
}
|
|
60
|
+
return response.Items[0];
|
|
61
|
+
};
|
|
62
|
+
exports.getRetiredFacilityMappingForTurvoLocationId = getRetiredFacilityMappingForTurvoLocationId;
|
|
63
|
+
/**
|
|
64
|
+
* Get retired facility mapping by human identifier (facility code or short code)
|
|
65
|
+
*/
|
|
66
|
+
const getRetiredFacilityMappingByHumanIdentifier = async (humanIdentifier) => {
|
|
67
|
+
if (typeof humanIdentifier !== 'string' || humanIdentifier === '') {
|
|
68
|
+
return null;
|
|
69
|
+
}
|
|
70
|
+
const tableName = (0, client_1.getRetiredFacilityMappingTableName)();
|
|
71
|
+
const indexName = (0, client_1.getIndexNameByShortCode)();
|
|
72
|
+
if (!tableName || !indexName) {
|
|
73
|
+
return null;
|
|
74
|
+
}
|
|
75
|
+
const dbQuery = new lib_dynamodb_1.QueryCommand({
|
|
76
|
+
TableName: tableName,
|
|
77
|
+
IndexName: indexName,
|
|
78
|
+
KeyConditionExpression: 'shortCode = :shortCode',
|
|
79
|
+
ExpressionAttributeValues: {
|
|
80
|
+
':shortCode': humanIdentifier,
|
|
81
|
+
},
|
|
82
|
+
});
|
|
83
|
+
const response = await (0, client_1.getDocClient)().send(dbQuery);
|
|
84
|
+
if (!response.Items?.length) {
|
|
85
|
+
return null;
|
|
86
|
+
}
|
|
87
|
+
return response.Items[0];
|
|
88
|
+
};
|
|
89
|
+
exports.getRetiredFacilityMappingByHumanIdentifier = getRetiredFacilityMappingByHumanIdentifier;
|
|
90
|
+
/**
|
|
91
|
+
* Get facility mapping by human identifier (facility code)
|
|
92
|
+
*/
|
|
93
|
+
const getFacilityMappingByHumanIdentifier = async (identifier) => {
|
|
94
|
+
if (typeof identifier !== 'string' || identifier === '') {
|
|
95
|
+
return null;
|
|
96
|
+
}
|
|
97
|
+
return getFacilityMappingByFacilityCode(identifier);
|
|
98
|
+
};
|
|
99
|
+
exports.getFacilityMappingByHumanIdentifier = getFacilityMappingByHumanIdentifier;
|
|
100
|
+
/**
|
|
101
|
+
* Get facility code mapping for a Turvo location ID
|
|
102
|
+
*/
|
|
103
|
+
const getFacilityCodeMappingForTurvoLocationId = async (turvoLocationId) => {
|
|
104
|
+
if (typeof turvoLocationId !== 'number') {
|
|
105
|
+
return null;
|
|
106
|
+
}
|
|
107
|
+
const tableName = (0, client_1.getFacilityCodeMapTable)();
|
|
108
|
+
const indexName = (0, client_1.getIndexNameByTurvoLocationId)();
|
|
109
|
+
if (!tableName || !indexName) {
|
|
110
|
+
return null;
|
|
111
|
+
}
|
|
112
|
+
const dbQuery = new lib_dynamodb_1.QueryCommand({
|
|
113
|
+
TableName: tableName,
|
|
114
|
+
IndexName: indexName,
|
|
115
|
+
KeyConditionExpression: 'turvoLocationId = :turvoLocationId',
|
|
116
|
+
ExpressionAttributeValues: {
|
|
117
|
+
':turvoLocationId': turvoLocationId,
|
|
118
|
+
},
|
|
119
|
+
});
|
|
120
|
+
const response = await (0, client_1.getDocClient)().send(dbQuery);
|
|
121
|
+
if (!response.Items?.length || response.Items.length !== 1) {
|
|
122
|
+
return null;
|
|
123
|
+
}
|
|
124
|
+
return response.Items[0];
|
|
125
|
+
};
|
|
126
|
+
/**
|
|
127
|
+
* Get short code mapping for a Turvo location ID
|
|
128
|
+
*/
|
|
129
|
+
const getShortCodeMappingForTurvoLocationId = async (turvoLocationId) => {
|
|
130
|
+
if (typeof turvoLocationId !== 'number') {
|
|
131
|
+
return null;
|
|
132
|
+
}
|
|
133
|
+
const tableName = (0, client_1.getTableName)();
|
|
134
|
+
const indexName = (0, client_1.getIndexNameByTurvoLocationId)();
|
|
135
|
+
if (!tableName || !indexName) {
|
|
136
|
+
return null;
|
|
137
|
+
}
|
|
138
|
+
const dbQuery = new lib_dynamodb_1.QueryCommand({
|
|
139
|
+
TableName: tableName,
|
|
140
|
+
IndexName: indexName,
|
|
141
|
+
KeyConditionExpression: 'turvoLocationId = :turvoLocationId',
|
|
142
|
+
ExpressionAttributeValues: {
|
|
143
|
+
':turvoLocationId': turvoLocationId,
|
|
144
|
+
},
|
|
145
|
+
});
|
|
146
|
+
const response = await (0, client_1.getDocClient)().send(dbQuery);
|
|
147
|
+
if (!response.Items?.length) {
|
|
148
|
+
return null;
|
|
149
|
+
}
|
|
150
|
+
return response.Items[0];
|
|
151
|
+
};
|
|
152
|
+
/**
|
|
153
|
+
* Get facility mapping by facility code
|
|
154
|
+
*/
|
|
155
|
+
const getFacilityMappingByFacilityCode = async (facilityCode) => {
|
|
156
|
+
if (typeof facilityCode !== 'string' || facilityCode === '') {
|
|
157
|
+
return null;
|
|
158
|
+
}
|
|
159
|
+
const tableName = (0, client_1.getFacilityCodeMapTable)();
|
|
160
|
+
const indexName = (0, client_1.getIndexNameByFacilityCode)();
|
|
161
|
+
if (!tableName || !indexName) {
|
|
162
|
+
return null;
|
|
163
|
+
}
|
|
164
|
+
const dbQuery = new lib_dynamodb_1.QueryCommand({
|
|
165
|
+
TableName: tableName,
|
|
166
|
+
IndexName: indexName,
|
|
167
|
+
KeyConditionExpression: 'facilityCode = :facilityCode',
|
|
168
|
+
ExpressionAttributeValues: {
|
|
169
|
+
':facilityCode': facilityCode,
|
|
170
|
+
},
|
|
171
|
+
});
|
|
172
|
+
const response = await (0, client_1.getDocClient)().send(dbQuery);
|
|
173
|
+
if (!response.Items?.length) {
|
|
174
|
+
return null;
|
|
175
|
+
}
|
|
176
|
+
return response.Items[0];
|
|
177
|
+
};
|
|
178
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"facilityMapping.js","sourceRoot":"","sources":["../../../src/db/facilityMapping.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Facility mapping functions for Turvo location to Veho facility mapping\n *\n * This file contains methods for mapping between Turvo locations and Veho facilities.\n * It abstracts over three different tables:\n * 1. FACILITY_CODE_MAP_TABLE: stores the map using the current facilityCode naming scheme\n * 2. SHORT_CODE_MAP_TABLE: stores the map using the old shortCode naming scheme\n * 3. RETIRED_FACILITY_MAPPING_TABLE_NAME: stores retired facility mappings\n */\n\nimport { QueryCommand } from '@aws-sdk/lib-dynamodb'\n\nimport { VehoTurvoLocationMap } from '../types/facilityMapping'\nimport {\n  getDocClient,\n  getFacilityCodeMapTable,\n  getIndexNameByFacilityCode,\n  getIndexNameByShortCode,\n  getIndexNameByTurvoLocationId,\n  getRetiredFacilityMappingTableName,\n  getTableName,\n} from './client'\n\n/**\n * Get facility mapping for a Turvo location ID\n * Checks both facility code and short code mappings\n */\nexport const getFacilityMappingForTurvoLocationId = async (\n  turvoLocationId: number\n): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof turvoLocationId !== 'number') {\n    return null\n  }\n\n  // Check new FACILITY_CODE_MAP_TABLE first\n  const facilityCodeMapping = await getFacilityCodeMappingForTurvoLocationId(turvoLocationId)\n  if (facilityCodeMapping) {\n    return facilityCodeMapping\n  }\n\n  // Then check the SHORT_CODE_MAP_TABLE\n  const shortCodeMapping = await getShortCodeMappingForTurvoLocationId(turvoLocationId)\n  if (shortCodeMapping) {\n    return shortCodeMapping\n  }\n\n  return null\n}\n\n/**\n * Get retired facility mapping for a Turvo location ID\n */\nexport const getRetiredFacilityMappingForTurvoLocationId = async (\n  turvoLocationId: number\n): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof turvoLocationId !== 'number') {\n    return null\n  }\n\n  const tableName = getRetiredFacilityMappingTableName()\n  const indexName = getIndexNameByTurvoLocationId()\n\n  if (!tableName || !indexName) {\n    return null\n  }\n\n  const dbQuery = new QueryCommand({\n    TableName: tableName,\n    IndexName: indexName,\n    KeyConditionExpression: 'turvoLocationId = :turvoLocationId',\n    ExpressionAttributeValues: {\n      ':turvoLocationId': turvoLocationId,\n    },\n  })\n\n  const response = await getDocClient().send(dbQuery)\n  if (!response.Items?.length) {\n    return null\n  }\n  return response.Items[0] as VehoTurvoLocationMap\n}\n\n/**\n * Get retired facility mapping by human identifier (facility code or short code)\n */\nexport const getRetiredFacilityMappingByHumanIdentifier = async (\n  humanIdentifier: string\n): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof humanIdentifier !== 'string' || humanIdentifier === '') {\n    return null\n  }\n\n  const tableName = getRetiredFacilityMappingTableName()\n  const indexName = getIndexNameByShortCode()\n\n  if (!tableName || !indexName) {\n    return null\n  }\n\n  const dbQuery = new QueryCommand({\n    TableName: tableName,\n    IndexName: indexName,\n    KeyConditionExpression: 'shortCode = :shortCode',\n    ExpressionAttributeValues: {\n      ':shortCode': humanIdentifier,\n    },\n  })\n\n  const response = await getDocClient().send(dbQuery)\n  if (!response.Items?.length) {\n    return null\n  }\n  return response.Items[0] as VehoTurvoLocationMap\n}\n\n/**\n * Get facility mapping by human identifier (facility code)\n */\nexport const getFacilityMappingByHumanIdentifier = async (identifier: string): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof identifier !== 'string' || identifier === '') {\n    return null\n  }\n\n  return getFacilityMappingByFacilityCode(identifier)\n}\n\n/**\n * Get facility code mapping for a Turvo location ID\n */\nconst getFacilityCodeMappingForTurvoLocationId = async (\n  turvoLocationId: number\n): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof turvoLocationId !== 'number') {\n    return null\n  }\n\n  const tableName = getFacilityCodeMapTable()\n  const indexName = getIndexNameByTurvoLocationId()\n\n  if (!tableName || !indexName) {\n    return null\n  }\n\n  const dbQuery = new QueryCommand({\n    TableName: tableName,\n    IndexName: indexName,\n    KeyConditionExpression: 'turvoLocationId = :turvoLocationId',\n    ExpressionAttributeValues: {\n      ':turvoLocationId': turvoLocationId,\n    },\n  })\n\n  const response = await getDocClient().send(dbQuery)\n  if (!response.Items?.length || response.Items.length !== 1) {\n    return null\n  }\n  return response.Items[0] as VehoTurvoLocationMap\n}\n\n/**\n * Get short code mapping for a Turvo location ID\n */\nconst getShortCodeMappingForTurvoLocationId = async (turvoLocationId: number): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof turvoLocationId !== 'number') {\n    return null\n  }\n\n  const tableName = getTableName()\n  const indexName = getIndexNameByTurvoLocationId()\n\n  if (!tableName || !indexName) {\n    return null\n  }\n\n  const dbQuery = new QueryCommand({\n    TableName: tableName,\n    IndexName: indexName,\n    KeyConditionExpression: 'turvoLocationId = :turvoLocationId',\n    ExpressionAttributeValues: {\n      ':turvoLocationId': turvoLocationId,\n    },\n  })\n\n  const response = await getDocClient().send(dbQuery)\n  if (!response.Items?.length) {\n    return null\n  }\n  return response.Items[0] as VehoTurvoLocationMap\n}\n\n/**\n * Get facility mapping by facility code\n */\nconst getFacilityMappingByFacilityCode = async (facilityCode: string): Promise<VehoTurvoLocationMap | null> => {\n  if (typeof facilityCode !== 'string' || facilityCode === '') {\n    return null\n  }\n\n  const tableName = getFacilityCodeMapTable()\n  const indexName = getIndexNameByFacilityCode()\n\n  if (!tableName || !indexName) {\n    return null\n  }\n\n  const dbQuery = new QueryCommand({\n    TableName: tableName,\n    IndexName: indexName,\n    KeyConditionExpression: 'facilityCode = :facilityCode',\n    ExpressionAttributeValues: {\n      ':facilityCode': facilityCode,\n    },\n  })\n\n  const response = await getDocClient().send(dbQuery)\n  if (!response.Items?.length) {\n    return null\n  }\n  return response.Items[0] as VehoTurvoLocationMap\n}\n"]}
|
|
@@ -1,6 +1,8 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.TrackingService = void 0;
|
|
4
|
+
const luxon_1 = require("luxon");
|
|
5
|
+
const facilityMapping_1 = require("../db/facilityMapping");
|
|
4
6
|
const errors_1 = require("../types/errors");
|
|
5
7
|
/**
|
|
6
8
|
* TrackingService orchestrates calls to public and internal APIs
|
|
@@ -46,40 +48,78 @@ class TrackingService {
|
|
|
46
48
|
* Transform raw Turvo API responses into clean ShipmentTracking type
|
|
47
49
|
*
|
|
48
50
|
*/
|
|
49
|
-
transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
|
|
51
|
+
async transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
|
|
50
52
|
const shipmentData = shipment;
|
|
53
|
+
// Filter out deleted entries from turvo data
|
|
54
|
+
const globalRoute = shipmentData.globalRoute?.filter(route => route.deleted !== false);
|
|
55
|
+
const carrierOrder = shipmentData.carrierOrder?.filter(order => order.deleted !== false);
|
|
56
|
+
// Map global route to stops array
|
|
57
|
+
const allStops = await Promise.all(shipment.globalRoute
|
|
58
|
+
.filter(routeStop => !routeStop.deleted)
|
|
59
|
+
.map(async (routeStop) => {
|
|
60
|
+
// Make sure we're actually getting ISO, fail if we don't
|
|
61
|
+
const appointmentDateTime = luxon_1.DateTime.fromISO(routeStop.appointment.date).toUTC().toISO();
|
|
62
|
+
if (!appointmentDateTime) {
|
|
63
|
+
throw new errors_1.TurvoDateFormatError(`expected ISO date but got ${routeStop.appointment.date}`);
|
|
64
|
+
}
|
|
65
|
+
// Additional check for a retired location mapping to get a facility id,
|
|
66
|
+
// for the case a facility has moved/been given a new entry in turvo.
|
|
67
|
+
const facilityId = (await (0, facilityMapping_1.getFacilityMappingForTurvoLocationId)(routeStop.location.id))?.facilityId ??
|
|
68
|
+
(await (0, facilityMapping_1.getRetiredFacilityMappingForTurvoLocationId)(routeStop.location.id))?.facilityId ??
|
|
69
|
+
null;
|
|
70
|
+
return {
|
|
71
|
+
stopNumber: routeStop.sequence,
|
|
72
|
+
appointmentDateTime,
|
|
73
|
+
appointmentLocalTimeZone: routeStop.appointment.timeZone,
|
|
74
|
+
locationName: routeStop.name,
|
|
75
|
+
externalLocationId: routeStop.location.id.toString(),
|
|
76
|
+
facilityId,
|
|
77
|
+
externalStopId: routeStop.id?.toString() ?? null,
|
|
78
|
+
stopServiceType: routeStop.stopType.value === 'Pickup' ? 'pickup' : 'delivery',
|
|
79
|
+
isCompleted: routeStop.state === 'COMPLETED',
|
|
80
|
+
locationId: routeStop.location.id,
|
|
81
|
+
etaToStop: routeStop.etaToStop?.etaValue ?? null,
|
|
82
|
+
milesRemainingToStop: routeStop.etaToStop?.nextMiles ?? null,
|
|
83
|
+
totalMilesToStop: routeStop.distance?.value ?? null,
|
|
84
|
+
// TODO: actual arrival time - time under attributes under global route
|
|
85
|
+
// TODO: actual departure time
|
|
86
|
+
// TODO: address, city, state - address under globalroute
|
|
87
|
+
// TODO: ask about what we want when we have multiple entries for the arrival/departure?
|
|
88
|
+
};
|
|
89
|
+
}));
|
|
90
|
+
const allStopsOrdered = allStops.toSorted((a, b) => a.stopNumber - b.stopNumber);
|
|
91
|
+
const nextStop = allStopsOrdered.find(stop => !stop.isCompleted) ?? null;
|
|
51
92
|
return {
|
|
52
93
|
// Identifiers
|
|
53
94
|
shipmentId: shipmentData.id?.toString() || '',
|
|
54
95
|
loadReferenceNumber: shipmentData.customId || '',
|
|
55
96
|
// Status
|
|
56
97
|
status: shipmentData.status?.code?.value || 'Unknown',
|
|
57
|
-
|
|
98
|
+
lateByMinutes: shipment.status.runningLate?.lateDuration ?? null,
|
|
58
99
|
isLate: !!shipment.status.runningLate,
|
|
59
100
|
// Current location (if tracking available)
|
|
60
|
-
currentLocation: shipment.status.location ?? null,
|
|
101
|
+
currentLocation: shipment.status.location ?? null, // TODO: make sure this is the value we actually want, status location may not be it
|
|
61
102
|
hasGpsTracking: locationUpdates !== null && !internalApiFailed,
|
|
62
103
|
// Progress
|
|
63
|
-
completedStops:
|
|
64
|
-
totalStops:
|
|
104
|
+
completedStops: allStopsOrdered.filter(stop => stop.isCompleted).length || 0,
|
|
105
|
+
totalStops: globalRoute?.length || 0,
|
|
65
106
|
// Stops (ordered)
|
|
66
|
-
stops:
|
|
107
|
+
stops: allStopsOrdered,
|
|
67
108
|
// ETA & Distance
|
|
68
|
-
etaUtc:
|
|
69
|
-
etaTimezone:
|
|
70
|
-
milesRemaining:
|
|
71
|
-
|
|
109
|
+
etaUtc: nextStop?.etaToStop ?? null,
|
|
110
|
+
etaTimezone: nextStop?.appointmentLocalTimeZone ?? null,
|
|
111
|
+
milesRemaining: nextStop?.milesRemainingToStop ?? null,
|
|
112
|
+
totalMilesToNextStop: nextStop?.totalMilesToStop ?? null,
|
|
72
113
|
// GPS Pings (only populated if includeGps: true and hasGpsTracking)
|
|
73
114
|
locationUpdates: locationUpdates || null, // TODO: implment with scraper
|
|
74
115
|
pingCount: locationUpdates?.length || null, // TODO: implement with scraper
|
|
75
|
-
lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp ||
|
|
76
|
-
null, // TODO: implement with scraper
|
|
116
|
+
lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp || null, // TODO: implement with scraper
|
|
77
117
|
// Shipment Attributes
|
|
78
118
|
laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,
|
|
79
119
|
managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,
|
|
80
|
-
carrier:
|
|
120
|
+
carrier: carrierOrder?.[0]?.carrier?.name || null,
|
|
81
121
|
};
|
|
82
122
|
}
|
|
83
123
|
}
|
|
84
124
|
exports.TrackingService = TrackingService;
|
|
85
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"trackingService.js","sourceRoot":"","sources":["../../../src/shipmentTracking/trackingService.ts"],"names":[],"mappings":"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","sourcesContent":["import { TurvoInternalApi } from '../api/turvoInternalApi'\nimport { TurvoPublicApi } from '../api/turvoPublicApi'\nimport { TurvoShipment } from '../types'\nimport { TurvoNotFoundError } from '../types/errors'\nimport { GetTrackingOptions, LocationUpdate, ShipmentTracking } from '../types/tracking'\n\n/**\n * TrackingService orchestrates calls to public and internal APIs\n * to provide a unified shipment tracking interface.\n */\nexport class TrackingService {\n  constructor(\n    private publicApi: TurvoPublicApi,\n    private internalApi: TurvoInternalApi\n  ) {}\n\n  /**\n   * Main entry point - combines all API data to produce ShipmentTracking\n   *\n   * @param shipmentId - The shipment ID to track\n   * @param options - Tracking options\n   * @returns Complete shipment tracking information\n   */\n  async getTracking(shipmentId: string, options: GetTrackingOptions = {}): Promise<ShipmentTracking> {\n    // 1. Fetch public API data (required)\n    const shipmentResult = await this.publicApi.getShipment({ shipmentId: parseInt(shipmentId) })\n\n    if (!shipmentResult.details) {\n      throw new TurvoNotFoundError(shipmentId)\n    }\n\n    const shipment = shipmentResult.details\n\n    // 2. Fetch internal API data (optional, may fail)\n    let locationUpdates: LocationUpdate[] | null = null\n    let internalApiFailed = false\n\n    if (options.includeGps !== false) {\n      try {\n        locationUpdates = await this.internalApi.getLocationUpdates(shipmentId)\n      } catch (error) {\n        // Log but don't fail - internal API is optional\n        internalApiFailed = true\n      }\n    }\n\n    // 3. Transform to clean response\n    return this.transformToTrackingDetails(shipment, locationUpdates, internalApiFailed)\n  }\n\n  /**\n   * Transform raw Turvo API responses into clean ShipmentTracking type\n   *\n   */\n  private transformToTrackingDetails(\n    shipment: TurvoShipment,\n    locationUpdates: LocationUpdate[] | null,\n    internalApiFailed: boolean\n  ): ShipmentTracking {\n    const shipmentData = shipment as TurvoShipment\n    return {\n      // Identifiers\n      shipmentId: (shipmentData.id as number)?.toString() || '',\n      loadReferenceNumber: (shipmentData.customId as string) || '',\n\n      // Status\n      status: (shipmentData.status?.code?.value as string) || 'Unknown',\n      lateBy: shipment.status.runningLate?.lateDurationString ?? null,\n      isLate: !!shipment.status.runningLate,\n\n      // Current location (if tracking available)\n      currentLocation: shipment.status.location ?? null,\n      hasGpsTracking: locationUpdates !== null && !internalApiFailed,\n\n      // Progress\n      completedStops: 0, // TODO: Count from shipmentData.globalRoute\n      totalStops: (shipmentData.globalRoute as unknown[])?.length || 0, // TODO: parse from shipmentData.globalRoute\n\n      // Stops (ordered)\n      stops: [], // TODO: implment global route parsing\n\n      // ETA & Distance\n      etaUtc: shipment.status.attributes?.etaValUtc ?? null,\n      etaTimezone: shipment.status.attributes?.next_eta_cal_val_timezone ?? null,\n      milesRemaining: shipment.status.location?.nextMiles ?? null,\n      totalMiles: null, // TODO: Derive from global route -- find current stop, and figure out total distance leading to it\n\n      // GPS Pings (only populated if includeGps: true and hasGpsTracking)\n      locationUpdates: (locationUpdates as LocationUpdate[]) || null, // TODO: implment with scraper\n      pingCount: (locationUpdates as LocationUpdate[])?.length || null, // TODO: implement with scraper\n      lastPingAt:\n        (locationUpdates as LocationUpdate[])?.[((locationUpdates as LocationUpdate[])?.length || 0) - 1]?.timestamp ||\n        null, // TODO: implement with scraper\n\n      // Shipment Attributes\n      laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,\n      managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,\n      carrier:\n        ((\n          ((shipmentData.carrierOrder as unknown[])?.[0] as Record<string, unknown>)?.carrier as Record<string, unknown>\n        )?.name as string) || null,\n    }\n  }\n}\n"]}
|
|
125
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"trackingService.js","sourceRoot":"","sources":["../../../src/shipmentTracking/trackingService.ts"],"names":[],"mappings":"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","sourcesContent":["import { DateTime } from 'luxon'\n\nimport { TurvoInternalApi } from '../api/turvoInternalApi'\nimport { TurvoPublicApi } from '../api/turvoPublicApi'\nimport {\n  getFacilityMappingForTurvoLocationId,\n  getRetiredFacilityMappingForTurvoLocationId,\n} from '../db/facilityMapping'\nimport { TurvoShipment } from '../types'\nimport { TurvoDateFormatError, TurvoNotFoundError } from '../types/errors'\nimport { GetTrackingOptions, LocationUpdate, ShipmentTracking, Stop } from '../types/tracking'\n\n/**\n * TrackingService orchestrates calls to public and internal APIs\n * to provide a unified shipment tracking interface.\n */\nexport class TrackingService {\n  constructor(\n    private publicApi: TurvoPublicApi,\n    private internalApi: TurvoInternalApi\n  ) {}\n\n  /**\n   * Main entry point - combines all API data to produce ShipmentTracking\n   *\n   * @param shipmentId - The shipment ID to track\n   * @param options - Tracking options\n   * @returns Complete shipment tracking information\n   */\n  async getTracking(shipmentId: string, options: GetTrackingOptions = {}): Promise<ShipmentTracking> {\n    // 1. Fetch public API data (required)\n    const shipmentResult = await this.publicApi.getShipment({ shipmentId: parseInt(shipmentId) })\n\n    if (!shipmentResult.details) {\n      throw new TurvoNotFoundError(shipmentId)\n    }\n\n    const shipment = shipmentResult.details\n\n    // 2. Fetch internal API data (optional, may fail)\n    let locationUpdates: LocationUpdate[] | null = null\n    let internalApiFailed = false\n\n    if (options.includeGps !== false) {\n      try {\n        locationUpdates = await this.internalApi.getLocationUpdates(shipmentId)\n      } catch (error) {\n        // Log but don't fail - internal API is optional\n        internalApiFailed = true\n      }\n    }\n\n    // 3. Transform to clean response\n    return this.transformToTrackingDetails(shipment, locationUpdates, internalApiFailed)\n  }\n\n  /**\n   * Transform raw Turvo API responses into clean ShipmentTracking type\n   *\n   */\n  private async transformToTrackingDetails(\n    shipment: TurvoShipment,\n    locationUpdates: LocationUpdate[] | null,\n    internalApiFailed: boolean\n  ): Promise<ShipmentTracking> {\n    const shipmentData = shipment as TurvoShipment\n\n    // Filter out deleted entries from turvo data\n    const globalRoute = shipmentData.globalRoute?.filter(route => route.deleted !== false)\n    const carrierOrder = shipmentData.carrierOrder?.filter(order => order.deleted !== false)\n\n    // Map global route to stops array\n    const allStops: Stop[] = await Promise.all(\n      shipment.globalRoute\n        .filter(routeStop => !routeStop.deleted)\n        .map(async routeStop => {\n          // Make sure we're actually getting ISO, fail if we don't\n          const appointmentDateTime = DateTime.fromISO(routeStop.appointment.date).toUTC().toISO()\n          if (!appointmentDateTime) {\n            throw new TurvoDateFormatError(`expected ISO date but got ${routeStop.appointment.date}`)\n          }\n\n          // Additional check for a retired location mapping to get a facility id,\n          // for the case a facility has moved/been given a new entry in turvo.\n          const facilityId =\n            (await getFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??\n            (await getRetiredFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??\n            null\n\n          return {\n            stopNumber: routeStop.sequence,\n            appointmentDateTime,\n            appointmentLocalTimeZone: routeStop.appointment.timeZone,\n            locationName: routeStop.name,\n            externalLocationId: routeStop.location.id.toString(),\n            facilityId,\n            externalStopId: routeStop.id?.toString() ?? null,\n            stopServiceType: routeStop.stopType.value === 'Pickup' ? 'pickup' : 'delivery',\n            isCompleted: routeStop.state === 'COMPLETED',\n            locationId: routeStop.location.id,\n            etaToStop: routeStop.etaToStop?.etaValue ?? null,\n            milesRemainingToStop: routeStop.etaToStop?.nextMiles ?? null,\n            totalMilesToStop: routeStop.distance?.value ?? null,\n            // TODO: actual arrival time - time under attributes under global route\n            // TODO: actual departure time\n            // TODO: address, city, state -  address under globalroute\n            // TODO: ask about what we want when we have multiple entries for the arrival/departure?\n          }\n        })\n    )\n\n    const allStopsOrdered = allStops.toSorted((a, b) => a.stopNumber - b.stopNumber)\n    const nextStop = allStopsOrdered.find(stop => !stop.isCompleted) ?? null\n\n    return {\n      // Identifiers\n      shipmentId: (shipmentData.id as number)?.toString() || '',\n      loadReferenceNumber: (shipmentData.customId as string) || '',\n\n      // Status\n      status: (shipmentData.status?.code?.value as string) || 'Unknown',\n      lateByMinutes: shipment.status.runningLate?.lateDuration ?? null,\n      isLate: !!shipment.status.runningLate,\n\n      // Current location (if tracking available)\n      currentLocation: shipment.status.location ?? null, // TODO: make sure this is the value we actually want, status location may not be it\n      hasGpsTracking: locationUpdates !== null && !internalApiFailed,\n\n      // Progress\n      completedStops: allStopsOrdered.filter(stop => stop.isCompleted).length || 0,\n      totalStops: globalRoute?.length || 0,\n\n      // Stops (ordered)\n      stops: allStopsOrdered,\n\n      // ETA & Distance\n      etaUtc: nextStop?.etaToStop ?? null,\n      etaTimezone: nextStop?.appointmentLocalTimeZone ?? null,\n      milesRemaining: nextStop?.milesRemainingToStop ?? null,\n      totalMilesToNextStop: nextStop?.totalMilesToStop ?? null,\n\n      // GPS Pings (only populated if includeGps: true and hasGpsTracking)\n      locationUpdates: locationUpdates || null, // TODO: implment with scraper\n      pingCount: locationUpdates?.length || null, // TODO: implement with scraper\n      lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp || null, // TODO: implement with scraper\n\n      // Shipment Attributes\n      laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,\n      managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,\n      carrier: carrierOrder?.[0]?.carrier?.name || null,\n    }\n  }\n}\n"]}
|
|
@@ -33,3 +33,9 @@ export declare class TurvoPartialDataError extends Error {
|
|
|
33
33
|
export declare class NoSecretError extends Error {
|
|
34
34
|
constructor(path: string);
|
|
35
35
|
}
|
|
36
|
+
/**
|
|
37
|
+
* Error thrown when a date format from Turvo is invalid
|
|
38
|
+
*/
|
|
39
|
+
export declare class TurvoDateFormatError extends Error {
|
|
40
|
+
constructor(message: string);
|
|
41
|
+
}
|
package/lib/cjs/types/errors.js
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.NoSecretError = exports.TurvoPartialDataError = exports.TurvoRateLimitError = exports.TurvoAuthError = exports.TurvoNotFoundError = void 0;
|
|
3
|
+
exports.TurvoDateFormatError = exports.NoSecretError = exports.TurvoPartialDataError = exports.TurvoRateLimitError = exports.TurvoAuthError = exports.TurvoNotFoundError = void 0;
|
|
4
4
|
/**
|
|
5
5
|
* Shipment not found in Turvo
|
|
6
6
|
*/
|
|
@@ -60,4 +60,14 @@ class NoSecretError extends Error {
|
|
|
60
60
|
}
|
|
61
61
|
}
|
|
62
62
|
exports.NoSecretError = NoSecretError;
|
|
63
|
-
|
|
63
|
+
/**
|
|
64
|
+
* Error thrown when a date format from Turvo is invalid
|
|
65
|
+
*/
|
|
66
|
+
class TurvoDateFormatError extends Error {
|
|
67
|
+
constructor(message) {
|
|
68
|
+
super(message);
|
|
69
|
+
this.name = 'TurvoDateFormatError';
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
exports.TurvoDateFormatError = TurvoDateFormatError;
|
|
73
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQTs7R0FFRztBQUNILE1BQWEsa0JBQW1CLFNBQVEsS0FBSztJQUNmO0lBQTVCLFlBQTRCLFVBQWtCO1FBQzVDLEtBQUssQ0FBQyxnQ0FBZ0MsVUFBVSxFQUFFLENBQUMsQ0FBQTtRQUR6QixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBRTVDLElBQUksQ0FBQyxJQUFJLEdBQUcsb0JBQW9CLENBQUE7SUFDbEMsQ0FBQztDQUNGO0FBTEQsZ0RBS0M7QUFFRDs7R0FFRztBQUNILE1BQWEsY0FBZSxTQUFRLEtBQUs7SUFDdkMsWUFBWSxPQUFlO1FBQ3pCLEtBQUssQ0FBQyxnQ0FBZ0MsT0FBTyxFQUFFLENBQUMsQ0FBQTtRQUNoRCxJQUFJLENBQUMsSUFBSSxHQUFHLGdCQUFnQixDQUFBO0lBQzlCLENBQUM7Q0FDRjtBQUxELHdDQUtDO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLG1CQUFvQixTQUFRLEtBQUs7SUFDaEI7SUFBNUIsWUFBNEIsVUFBbUI7UUFDN0MsS0FBSyxDQUFDLHdCQUF3QixVQUFVLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixVQUFVLFVBQVUsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUQ5RCxlQUFVLEdBQVYsVUFBVSxDQUFTO1FBRTdDLElBQUksQ0FBQyxJQUFJLEdBQUcscUJBQXFCLENBQUE7SUFDbkMsQ0FBQztDQUNGO0FBTEQsa0RBS0M7QUFFRDs7O0dBR0c7QUFDSCxNQUFhLHFCQUFzQixTQUFRLEtBQUs7SUFFNUI7SUFDQTtJQUZsQixZQUNrQixXQUFvQixFQUNwQixVQUFvQjtRQUVwQyxLQUFLLENBQUMsa0NBQWtDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLDJCQUEyQixDQUFDLENBQUE7UUFIekUsZ0JBQVcsR0FBWCxXQUFXLENBQVM7UUFDcEIsZUFBVSxHQUFWLFVBQVUsQ0FBVTtRQUdwQyxJQUFJLENBQUMsSUFBSSxHQUFHLHVCQUF1QixDQUFBO0lBQ3JDLENBQUM7Q0FDRjtBQVJELHNEQVFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLGFBQWMsU0FBUSxLQUFLO0lBQ3RDLFlBQVksSUFBWTtRQUN0QixLQUFLLENBQUMsNEJBQTRCLElBQUksRUFBRSxDQUFDLENBQUE7UUFDekMsSUFBSSxDQUFDLElBQUksR0FBRyxlQUFlLENBQUE7SUFDN0IsQ0FBQztDQUNGO0FBTEQsc0NBS0M7QUFFRDs7R0FFRztBQUNILE1BQWEsb0JBQXFCLFNBQVEsS0FBSztJQUM3QyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFBO1FBQ2QsSUFBSSxDQUFDLElBQUksR0FBRyxzQkFBc0IsQ0FBQTtJQUNwQyxDQUFDO0NBQ0Y7QUFMRCxvREFLQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU2hpcG1lbnQgbm90IGZvdW5kIGluIFR1cnZvXG4gKi9cbmV4cG9ydCBjbGFzcyBUdXJ2b05vdEZvdW5kRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSBzaGlwbWVudElkOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgU2hpcG1lbnQgbm90IGZvdW5kIGluIFR1cnZvOiAke3NoaXBtZW50SWR9YClcbiAgICB0aGlzLm5hbWUgPSAnVHVydm9Ob3RGb3VuZEVycm9yJ1xuICB9XG59XG5cbi8qKlxuICogQXV0aGVudGljYXRpb24gZmFpbGVkIChpbnZhbGlkIGNyZWRlbnRpYWxzLCBleHBpcmVkIHRva2VuKVxuICovXG5leHBvcnQgY2xhc3MgVHVydm9BdXRoRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBUdXJ2byBhdXRoZW50aWNhdGlvbiBmYWlsZWQ6ICR7bWVzc2FnZX1gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b0F1dGhFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIFJhdGUgbGltaXRlZCBieSBUdXJ2b1xuICovXG5leHBvcnQgY2xhc3MgVHVydm9SYXRlTGltaXRFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHJldHJ5QWZ0ZXI/OiBudW1iZXIpIHtcbiAgICBzdXBlcihgUmF0ZSBsaW1pdGVkIGJ5IFR1cnZvJHtyZXRyeUFmdGVyID8gYC4gUmV0cnkgYWZ0ZXIgJHtyZXRyeUFmdGVyfSBzZWNvbmRzYCA6ICcnfWApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvUmF0ZUxpbWl0RXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBJbnRlcm5hbCBBUEkgZmFpbGVkIGJ1dCBwdWJsaWMgQVBJIHN1Y2NlZWRlZC5cbiAqIENvbnRhaW5zIHBhcnRpYWwgZGF0YSBpbiBgcGFydGlhbERhdGFgIGZpZWxkLlxuICovXG5leHBvcnQgY2xhc3MgVHVydm9QYXJ0aWFsRGF0YUVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgcmVhZG9ubHkgcGFydGlhbERhdGE6IHVua25vd24sXG4gICAgcHVibGljIHJlYWRvbmx5IGZhaWxlZEFwaXM6IHN0cmluZ1tdXG4gICkge1xuICAgIHN1cGVyKGBUdXJ2byBpbnRlcm5hbCBBUEkgZmFpbGVkIGZvcjogJHtmYWlsZWRBcGlzLmpvaW4oJywgJyl9LiBQYXJ0aWFsIGRhdGEgYXZhaWxhYmxlLmApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvUGFydGlhbERhdGFFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIEVycm9yIHRocm93biB3aGVuIFR1cnZvIHNlY3JldHMgY2Fubm90IGJlIHJldHJpZXZlZCBmcm9tIEFXUyBTZWNyZXRzIE1hbmFnZXJcbiAqL1xuZXhwb3J0IGNsYXNzIE5vU2VjcmV0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHBhdGg6IHN0cmluZykge1xuICAgIHN1cGVyKGBObyBzZWNyZXQgZm91bmQgYXQgcGF0aDogJHtwYXRofWApXG4gICAgdGhpcy5uYW1lID0gJ05vU2VjcmV0RXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBhIGRhdGUgZm9ybWF0IGZyb20gVHVydm8gaXMgaW52YWxpZFxuICovXG5leHBvcnQgY2xhc3MgVHVydm9EYXRlRm9ybWF0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKG1lc3NhZ2UpXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvRGF0ZUZvcm1hdEVycm9yJ1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFjaWxpdHlNYXBwaW5nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2ZhY2lsaXR5TWFwcGluZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBGYWNpbGl0eSBtYXBwaW5nIHR5cGVzIGZvciBUdXJ2byBsb2NhdGlvbiB0byBWZWhvIGZhY2lsaXR5IG1hcHBpbmdcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBWZWhvVHVydm9Mb2NhdGlvbk1hcCB7XG4gIGZhY2lsaXR5Q29kZT86IHN0cmluZyB8IG51bGxcbiAgZmFjaWxpdHlJZDogc3RyaW5nXG4gIHNob3J0Q29kZT86IHN0cmluZyB8IG51bGxcbiAgdHVydm9Mb2NhdGlvbklkPzogbnVtYmVyIHwgbnVsbFxufVxuIl19
|