@veho/turvo-integration-sdk 0.1.0-beta.0
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/README.md +473 -0
- package/index.ts +1 -0
- package/lib/cjs/api/turvoInternalApi.d.ts +23 -0
- package/lib/cjs/api/turvoInternalApi.js +36 -0
- package/lib/cjs/api/turvoPublicApi.d.ts +99 -0
- package/lib/cjs/api/turvoPublicApi.js +103 -0
- package/lib/cjs/client/turvoClient.d.ts +49 -0
- package/lib/cjs/client/turvoClient.js +180 -0
- package/lib/cjs/constants.d.ts +29 -0
- package/lib/cjs/constants.js +33 -0
- package/lib/cjs/index.d.ts +6 -0
- package/lib/cjs/index.js +37 -0
- package/lib/cjs/shipmentTracking/index.d.ts +22 -0
- package/lib/cjs/shipmentTracking/index.js +39 -0
- package/lib/cjs/shipmentTracking/trackingService.d.ts +25 -0
- package/lib/cjs/shipmentTracking/trackingService.js +85 -0
- package/lib/cjs/types/common.d.ts +64 -0
- package/lib/cjs/types/common.js +27 -0
- package/lib/cjs/types/config.d.ts +13 -0
- package/lib/cjs/types/config.js +3 -0
- package/lib/cjs/types/errors.d.ts +35 -0
- package/lib/cjs/types/errors.js +63 -0
- package/lib/cjs/types/index.d.ts +5 -0
- package/lib/cjs/types/index.js +27 -0
- package/lib/cjs/types/shipment.d.ts +379 -0
- package/lib/cjs/types/shipment.js +46 -0
- package/lib/cjs/types/tracking.d.ts +65 -0
- package/lib/cjs/types/tracking.js +3 -0
- package/lib/esm/api/turvoInternalApi.d.ts +23 -0
- package/lib/esm/api/turvoInternalApi.js +32 -0
- package/lib/esm/api/turvoPublicApi.d.ts +99 -0
- package/lib/esm/api/turvoPublicApi.js +99 -0
- package/lib/esm/client/turvoClient.d.ts +49 -0
- package/lib/esm/client/turvoClient.js +172 -0
- package/lib/esm/constants.d.ts +29 -0
- package/lib/esm/constants.js +30 -0
- package/lib/esm/index.d.ts +6 -0
- package/lib/esm/index.js +11 -0
- package/lib/esm/shipmentTracking/index.d.ts +22 -0
- package/lib/esm/shipmentTracking/index.js +36 -0
- package/lib/esm/shipmentTracking/trackingService.d.ts +25 -0
- package/lib/esm/shipmentTracking/trackingService.js +81 -0
- package/lib/esm/types/common.d.ts +64 -0
- package/lib/esm/types/common.js +23 -0
- package/lib/esm/types/config.d.ts +13 -0
- package/lib/esm/types/config.js +2 -0
- package/lib/esm/types/errors.d.ts +35 -0
- package/lib/esm/types/errors.js +55 -0
- package/lib/esm/types/index.d.ts +5 -0
- package/lib/esm/types/index.js +11 -0
- package/lib/esm/types/shipment.d.ts +379 -0
- package/lib/esm/types/shipment.js +43 -0
- package/lib/esm/types/tracking.d.ts +65 -0
- package/lib/esm/types/tracking.js +2 -0
- package/lib/tsconfig.cjs.tsbuildinfo +1 -0
- package/lib/tsconfig.esm.tsbuildinfo +1 -0
- package/package.json +126 -0
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TrackingService = void 0;
|
|
4
|
+
const errors_1 = require("../types/errors");
|
|
5
|
+
/**
|
|
6
|
+
* TrackingService orchestrates calls to public and internal APIs
|
|
7
|
+
* to provide a unified shipment tracking interface.
|
|
8
|
+
*/
|
|
9
|
+
class TrackingService {
|
|
10
|
+
publicApi;
|
|
11
|
+
internalApi;
|
|
12
|
+
constructor(publicApi, internalApi) {
|
|
13
|
+
this.publicApi = publicApi;
|
|
14
|
+
this.internalApi = internalApi;
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Main entry point - combines all API data to produce ShipmentTracking
|
|
18
|
+
*
|
|
19
|
+
* @param shipmentId - The shipment ID to track
|
|
20
|
+
* @param options - Tracking options
|
|
21
|
+
* @returns Complete shipment tracking information
|
|
22
|
+
*/
|
|
23
|
+
async getTracking(shipmentId, options = {}) {
|
|
24
|
+
// 1. Fetch public API data (required)
|
|
25
|
+
const shipmentResult = await this.publicApi.getShipment({ shipmentId: parseInt(shipmentId) });
|
|
26
|
+
if (!shipmentResult.details) {
|
|
27
|
+
throw new errors_1.TurvoNotFoundError(shipmentId);
|
|
28
|
+
}
|
|
29
|
+
const shipment = shipmentResult.details;
|
|
30
|
+
// 2. Fetch internal API data (optional, may fail)
|
|
31
|
+
let locationUpdates = null;
|
|
32
|
+
let internalApiFailed = false;
|
|
33
|
+
if (options.includeGps !== false) {
|
|
34
|
+
try {
|
|
35
|
+
locationUpdates = await this.internalApi.getLocationUpdates(shipmentId);
|
|
36
|
+
}
|
|
37
|
+
catch (error) {
|
|
38
|
+
// Log but don't fail - internal API is optional
|
|
39
|
+
internalApiFailed = true;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
// 3. Transform to clean response
|
|
43
|
+
return this.transformToTrackingDetails(shipment, locationUpdates, internalApiFailed);
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* Transform raw Turvo API responses into clean ShipmentTracking type
|
|
47
|
+
*
|
|
48
|
+
*/
|
|
49
|
+
transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
|
|
50
|
+
const shipmentData = shipment;
|
|
51
|
+
return {
|
|
52
|
+
// Identifiers
|
|
53
|
+
shipmentId: shipmentData.id?.toString() || '',
|
|
54
|
+
loadReferenceNumber: shipmentData.customId || '',
|
|
55
|
+
// Status
|
|
56
|
+
status: shipmentData.status?.code?.value || 'Unknown',
|
|
57
|
+
lateBy: shipment.status.runningLate?.lateDurationString ?? null,
|
|
58
|
+
isLate: !!shipment.status.runningLate,
|
|
59
|
+
// Current location (if tracking available)
|
|
60
|
+
currentLocation: shipment.status.location ?? null,
|
|
61
|
+
hasGpsTracking: locationUpdates !== null && !internalApiFailed,
|
|
62
|
+
// Progress
|
|
63
|
+
completedStops: 0, // TODO: Count from shipmentData.globalRoute
|
|
64
|
+
totalStops: shipmentData.globalRoute?.length || 0, // TODO: parse from shipmentData.globalRoute
|
|
65
|
+
// Stops (ordered)
|
|
66
|
+
stops: [], // TODO: implment global route parsing
|
|
67
|
+
// ETA & Distance
|
|
68
|
+
etaUtc: shipment.status.attributes?.etaValUtc ?? null,
|
|
69
|
+
etaTimezone: shipment.status.attributes?.next_eta_cal_val_timezone ?? null,
|
|
70
|
+
milesRemaining: shipment.status.location?.nextMiles ?? null,
|
|
71
|
+
totalMiles: null, // TODO: Derive from global route -- find current stop, and figure out total distance leading to it
|
|
72
|
+
// GPS Pings (only populated if includeGps: true and hasGpsTracking)
|
|
73
|
+
locationUpdates: locationUpdates || null, // TODO: implment with scraper
|
|
74
|
+
pingCount: locationUpdates?.length || null, // TODO: implement with scraper
|
|
75
|
+
lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp ||
|
|
76
|
+
null, // TODO: implement with scraper
|
|
77
|
+
// Shipment Attributes
|
|
78
|
+
laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,
|
|
79
|
+
managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,
|
|
80
|
+
carrier: shipmentData.carrierOrder?.[0]?.carrier?.name || null,
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
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"]}
|
|
@@ -0,0 +1,64 @@
|
|
|
1
|
+
export declare class TurvoApiError extends Error {
|
|
2
|
+
errorCode: string;
|
|
3
|
+
errorMessage: string;
|
|
4
|
+
constructor(errorCode: string, errorMessage: string);
|
|
5
|
+
}
|
|
6
|
+
export type TurvoApiResult<T = void> = {
|
|
7
|
+
Status: 'SUCCESS' | string;
|
|
8
|
+
details: T;
|
|
9
|
+
};
|
|
10
|
+
export declare enum TurvoApiResponseStatus {
|
|
11
|
+
SUCCESS = "SUCCESS",
|
|
12
|
+
ERROR = "ERROR"
|
|
13
|
+
}
|
|
14
|
+
export declare enum TurvoApiMethod {
|
|
15
|
+
GET = "GET",
|
|
16
|
+
POST = "POST",
|
|
17
|
+
PUT = "PUT",
|
|
18
|
+
DELETE = "DELETE"
|
|
19
|
+
}
|
|
20
|
+
export type TurvoPagedResult<K extends string, T extends Record<any, unknown>> = {
|
|
21
|
+
pagination: {
|
|
22
|
+
start: number;
|
|
23
|
+
pageSize: number;
|
|
24
|
+
totalRecordsInPage: number;
|
|
25
|
+
moreAvailable: boolean;
|
|
26
|
+
};
|
|
27
|
+
} & (Record<K, T[]> | Record<K, undefined>);
|
|
28
|
+
export type TurvoFilterShipmentsResultItem = {
|
|
29
|
+
id: number;
|
|
30
|
+
customId: string;
|
|
31
|
+
lastUpdatedOn: string;
|
|
32
|
+
updated: string;
|
|
33
|
+
createdDate: string;
|
|
34
|
+
created: string;
|
|
35
|
+
status: {
|
|
36
|
+
code: TurvoLookup;
|
|
37
|
+
};
|
|
38
|
+
customerOrder: {
|
|
39
|
+
id: number;
|
|
40
|
+
customer: {
|
|
41
|
+
id: number;
|
|
42
|
+
name: string;
|
|
43
|
+
};
|
|
44
|
+
}[];
|
|
45
|
+
carrierOrder: {
|
|
46
|
+
id: number;
|
|
47
|
+
carrier: {
|
|
48
|
+
name: string;
|
|
49
|
+
id: number;
|
|
50
|
+
};
|
|
51
|
+
}[];
|
|
52
|
+
};
|
|
53
|
+
export type TurvoDateWithTimezone = {
|
|
54
|
+
date: string;
|
|
55
|
+
timeZone: string;
|
|
56
|
+
};
|
|
57
|
+
export type TurvoDateWithTimezoneLowercase = {
|
|
58
|
+
date: string;
|
|
59
|
+
timezone: string;
|
|
60
|
+
};
|
|
61
|
+
export type TurvoLookup = {
|
|
62
|
+
key: string;
|
|
63
|
+
value: string;
|
|
64
|
+
};
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TurvoApiMethod = exports.TurvoApiResponseStatus = exports.TurvoApiError = void 0;
|
|
4
|
+
/* eslint-disable @typescript-eslint/no-explicit-any */
|
|
5
|
+
class TurvoApiError extends Error {
|
|
6
|
+
errorCode;
|
|
7
|
+
errorMessage;
|
|
8
|
+
constructor(errorCode, errorMessage) {
|
|
9
|
+
super(`Turvo Api Response Error ${errorCode}: ${errorMessage}`);
|
|
10
|
+
this.errorCode = errorCode;
|
|
11
|
+
this.errorMessage = errorMessage;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
exports.TurvoApiError = TurvoApiError;
|
|
15
|
+
var TurvoApiResponseStatus;
|
|
16
|
+
(function (TurvoApiResponseStatus) {
|
|
17
|
+
TurvoApiResponseStatus["SUCCESS"] = "SUCCESS";
|
|
18
|
+
TurvoApiResponseStatus["ERROR"] = "ERROR";
|
|
19
|
+
})(TurvoApiResponseStatus || (exports.TurvoApiResponseStatus = TurvoApiResponseStatus = {}));
|
|
20
|
+
var TurvoApiMethod;
|
|
21
|
+
(function (TurvoApiMethod) {
|
|
22
|
+
TurvoApiMethod["GET"] = "GET";
|
|
23
|
+
TurvoApiMethod["POST"] = "POST";
|
|
24
|
+
TurvoApiMethod["PUT"] = "PUT";
|
|
25
|
+
TurvoApiMethod["DELETE"] = "DELETE";
|
|
26
|
+
})(TurvoApiMethod || (exports.TurvoApiMethod = TurvoApiMethod = {}));
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29tbW9uLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2NvbW1vbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSx1REFBdUQ7QUFDdkQsTUFBYSxhQUFjLFNBQVEsS0FBSztJQUN0QyxTQUFTLENBQVE7SUFDakIsWUFBWSxDQUFRO0lBRXBCLFlBQVksU0FBaUIsRUFBRSxZQUFvQjtRQUNqRCxLQUFLLENBQUMsNEJBQTRCLFNBQVMsS0FBSyxZQUFZLEVBQUUsQ0FBQyxDQUFBO1FBQy9ELElBQUksQ0FBQyxTQUFTLEdBQUcsU0FBUyxDQUFBO1FBQzFCLElBQUksQ0FBQyxZQUFZLEdBQUcsWUFBWSxDQUFBO0lBQ2xDLENBQUM7Q0FDRjtBQVRELHNDQVNDO0FBT0QsSUFBWSxzQkFHWDtBQUhELFdBQVksc0JBQXNCO0lBQ2hDLDZDQUFtQixDQUFBO0lBQ25CLHlDQUFlLENBQUE7QUFDakIsQ0FBQyxFQUhXLHNCQUFzQixzQ0FBdEIsc0JBQXNCLFFBR2pDO0FBRUQsSUFBWSxjQUtYO0FBTEQsV0FBWSxjQUFjO0lBQ3hCLDZCQUFXLENBQUE7SUFDWCwrQkFBYSxDQUFBO0lBQ2IsNkJBQVcsQ0FBQTtJQUNYLG1DQUFpQixDQUFBO0FBQ25CLENBQUMsRUFMVyxjQUFjLDhCQUFkLGNBQWMsUUFLekIiLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tZXhwbGljaXQtYW55ICovXG5leHBvcnQgY2xhc3MgVHVydm9BcGlFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgZXJyb3JDb2RlOiBzdHJpbmdcbiAgZXJyb3JNZXNzYWdlOiBzdHJpbmdcblxuICBjb25zdHJ1Y3RvcihlcnJvckNvZGU6IHN0cmluZywgZXJyb3JNZXNzYWdlOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgVHVydm8gQXBpIFJlc3BvbnNlIEVycm9yICR7ZXJyb3JDb2RlfTogJHtlcnJvck1lc3NhZ2V9YClcbiAgICB0aGlzLmVycm9yQ29kZSA9IGVycm9yQ29kZVxuICAgIHRoaXMuZXJyb3JNZXNzYWdlID0gZXJyb3JNZXNzYWdlXG4gIH1cbn1cblxuZXhwb3J0IHR5cGUgVHVydm9BcGlSZXN1bHQ8VCA9IHZvaWQ+ID0ge1xuICBTdGF0dXM6ICdTVUNDRVNTJyB8IHN0cmluZ1xuICBkZXRhaWxzOiBUXG59XG5cbmV4cG9ydCBlbnVtIFR1cnZvQXBpUmVzcG9uc2VTdGF0dXMge1xuICBTVUNDRVNTID0gJ1NVQ0NFU1MnLFxuICBFUlJPUiA9ICdFUlJPUicsXG59XG5cbmV4cG9ydCBlbnVtIFR1cnZvQXBpTWV0aG9kIHtcbiAgR0VUID0gJ0dFVCcsXG4gIFBPU1QgPSAnUE9TVCcsXG4gIFBVVCA9ICdQVVQnLFxuICBERUxFVEUgPSAnREVMRVRFJyxcbn1cblxuZXhwb3J0IHR5cGUgVHVydm9QYWdlZFJlc3VsdDxLIGV4dGVuZHMgc3RyaW5nLCBUIGV4dGVuZHMgUmVjb3JkPGFueSwgdW5rbm93bj4+ID0ge1xuICBwYWdpbmF0aW9uOiB7XG4gICAgc3RhcnQ6IG51bWJlclxuICAgIHBhZ2VTaXplOiBudW1iZXJcbiAgICB0b3RhbFJlY29yZHNJblBhZ2U6IG51bWJlclxuICAgIG1vcmVBdmFpbGFibGU6IGJvb2xlYW5cbiAgfVxufSAmIChSZWNvcmQ8SywgVFtdPiB8IFJlY29yZDxLLCB1bmRlZmluZWQ+KVxuXG5leHBvcnQgdHlwZSBUdXJ2b0ZpbHRlclNoaXBtZW50c1Jlc3VsdEl0ZW0gPSB7XG4gIGlkOiBudW1iZXJcbiAgY3VzdG9tSWQ6IHN0cmluZ1xuICBsYXN0VXBkYXRlZE9uOiBzdHJpbmdcbiAgdXBkYXRlZDogc3RyaW5nXG4gIGNyZWF0ZWREYXRlOiBzdHJpbmdcbiAgY3JlYXRlZDogc3RyaW5nXG4gIHN0YXR1czogeyBjb2RlOiBUdXJ2b0xvb2t1cCB9XG4gIGN1c3RvbWVyT3JkZXI6IHtcbiAgICBpZDogbnVtYmVyXG4gICAgY3VzdG9tZXI6IHtcbiAgICAgIGlkOiBudW1iZXJcbiAgICAgIG5hbWU6IHN0cmluZ1xuICAgIH1cbiAgfVtdXG4gIGNhcnJpZXJPcmRlcjoge1xuICAgIGlkOiBudW1iZXJcbiAgICBjYXJyaWVyOiB7XG4gICAgICBuYW1lOiBzdHJpbmdcbiAgICAgIGlkOiBudW1iZXJcbiAgICB9XG4gIH1bXVxufVxuXG5leHBvcnQgdHlwZSBUdXJ2b0RhdGVXaXRoVGltZXpvbmUgPSB7XG4gIGRhdGU6IHN0cmluZ1xuICB0aW1lWm9uZTogc3RyaW5nXG59XG5cbmV4cG9ydCB0eXBlIFR1cnZvRGF0ZVdpdGhUaW1lem9uZUxvd2VyY2FzZSA9IHtcbiAgZGF0ZTogc3RyaW5nXG4gIHRpbWV6b25lOiBzdHJpbmdcbn1cblxuZXhwb3J0IHR5cGUgVHVydm9Mb29rdXAgPSB7XG4gIGtleTogc3RyaW5nXG4gIHZhbHVlOiBzdHJpbmdcbn1cbiJdfQ==
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Turvo API credentials
|
|
3
|
+
*/
|
|
4
|
+
export interface TurvoCredentials {
|
|
5
|
+
/** Turvo API key (x-api-key header) */
|
|
6
|
+
apiKey: string;
|
|
7
|
+
/** OAuth username */
|
|
8
|
+
username: string;
|
|
9
|
+
/** OAuth password */
|
|
10
|
+
password: string;
|
|
11
|
+
/** Turvo API base URL */
|
|
12
|
+
urlBase: string;
|
|
13
|
+
}
|
|
@@ -0,0 +1,3 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uZmlnLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2NvbmZpZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBUdXJ2byBBUEkgY3JlZGVudGlhbHNcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBUdXJ2b0NyZWRlbnRpYWxzIHtcbiAgLyoqIFR1cnZvIEFQSSBrZXkgKHgtYXBpLWtleSBoZWFkZXIpICovXG4gIGFwaUtleTogc3RyaW5nXG4gIC8qKiBPQXV0aCB1c2VybmFtZSAqL1xuICB1c2VybmFtZTogc3RyaW5nXG4gIC8qKiBPQXV0aCBwYXNzd29yZCAqL1xuICBwYXNzd29yZDogc3RyaW5nXG4gIC8qKiBUdXJ2byBBUEkgYmFzZSBVUkwgKi9cbiAgdXJsQmFzZTogc3RyaW5nXG59XG4iXX0=
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Shipment not found in Turvo
|
|
3
|
+
*/
|
|
4
|
+
export declare class TurvoNotFoundError extends Error {
|
|
5
|
+
readonly shipmentId: string;
|
|
6
|
+
constructor(shipmentId: string);
|
|
7
|
+
}
|
|
8
|
+
/**
|
|
9
|
+
* Authentication failed (invalid credentials, expired token)
|
|
10
|
+
*/
|
|
11
|
+
export declare class TurvoAuthError extends Error {
|
|
12
|
+
constructor(message: string);
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* Rate limited by Turvo
|
|
16
|
+
*/
|
|
17
|
+
export declare class TurvoRateLimitError extends Error {
|
|
18
|
+
readonly retryAfter?: number | undefined;
|
|
19
|
+
constructor(retryAfter?: number | undefined);
|
|
20
|
+
}
|
|
21
|
+
/**
|
|
22
|
+
* Internal API failed but public API succeeded.
|
|
23
|
+
* Contains partial data in `partialData` field.
|
|
24
|
+
*/
|
|
25
|
+
export declare class TurvoPartialDataError extends Error {
|
|
26
|
+
readonly partialData: unknown;
|
|
27
|
+
readonly failedApis: string[];
|
|
28
|
+
constructor(partialData: unknown, failedApis: string[]);
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Error thrown when Turvo secrets cannot be retrieved from AWS Secrets Manager
|
|
32
|
+
*/
|
|
33
|
+
export declare class NoSecretError extends Error {
|
|
34
|
+
constructor(path: string);
|
|
35
|
+
}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.NoSecretError = exports.TurvoPartialDataError = exports.TurvoRateLimitError = exports.TurvoAuthError = exports.TurvoNotFoundError = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* Shipment not found in Turvo
|
|
6
|
+
*/
|
|
7
|
+
class TurvoNotFoundError extends Error {
|
|
8
|
+
shipmentId;
|
|
9
|
+
constructor(shipmentId) {
|
|
10
|
+
super(`Shipment not found in Turvo: ${shipmentId}`);
|
|
11
|
+
this.shipmentId = shipmentId;
|
|
12
|
+
this.name = 'TurvoNotFoundError';
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
exports.TurvoNotFoundError = TurvoNotFoundError;
|
|
16
|
+
/**
|
|
17
|
+
* Authentication failed (invalid credentials, expired token)
|
|
18
|
+
*/
|
|
19
|
+
class TurvoAuthError extends Error {
|
|
20
|
+
constructor(message) {
|
|
21
|
+
super(`Turvo authentication failed: ${message}`);
|
|
22
|
+
this.name = 'TurvoAuthError';
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
exports.TurvoAuthError = TurvoAuthError;
|
|
26
|
+
/**
|
|
27
|
+
* Rate limited by Turvo
|
|
28
|
+
*/
|
|
29
|
+
class TurvoRateLimitError extends Error {
|
|
30
|
+
retryAfter;
|
|
31
|
+
constructor(retryAfter) {
|
|
32
|
+
super(`Rate limited by Turvo${retryAfter ? `. Retry after ${retryAfter} seconds` : ''}`);
|
|
33
|
+
this.retryAfter = retryAfter;
|
|
34
|
+
this.name = 'TurvoRateLimitError';
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
exports.TurvoRateLimitError = TurvoRateLimitError;
|
|
38
|
+
/**
|
|
39
|
+
* Internal API failed but public API succeeded.
|
|
40
|
+
* Contains partial data in `partialData` field.
|
|
41
|
+
*/
|
|
42
|
+
class TurvoPartialDataError extends Error {
|
|
43
|
+
partialData;
|
|
44
|
+
failedApis;
|
|
45
|
+
constructor(partialData, failedApis) {
|
|
46
|
+
super(`Turvo internal API failed for: ${failedApis.join(', ')}. Partial data available.`);
|
|
47
|
+
this.partialData = partialData;
|
|
48
|
+
this.failedApis = failedApis;
|
|
49
|
+
this.name = 'TurvoPartialDataError';
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
exports.TurvoPartialDataError = TurvoPartialDataError;
|
|
53
|
+
/**
|
|
54
|
+
* Error thrown when Turvo secrets cannot be retrieved from AWS Secrets Manager
|
|
55
|
+
*/
|
|
56
|
+
class NoSecretError extends Error {
|
|
57
|
+
constructor(path) {
|
|
58
|
+
super(`No secret found at path: ${path}`);
|
|
59
|
+
this.name = 'NoSecretError';
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
exports.NoSecretError = NoSecretError;
|
|
63
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQTs7R0FFRztBQUNILE1BQWEsa0JBQW1CLFNBQVEsS0FBSztJQUNmO0lBQTVCLFlBQTRCLFVBQWtCO1FBQzVDLEtBQUssQ0FBQyxnQ0FBZ0MsVUFBVSxFQUFFLENBQUMsQ0FBQTtRQUR6QixlQUFVLEdBQVYsVUFBVSxDQUFRO1FBRTVDLElBQUksQ0FBQyxJQUFJLEdBQUcsb0JBQW9CLENBQUE7SUFDbEMsQ0FBQztDQUNGO0FBTEQsZ0RBS0M7QUFFRDs7R0FFRztBQUNILE1BQWEsY0FBZSxTQUFRLEtBQUs7SUFDdkMsWUFBWSxPQUFlO1FBQ3pCLEtBQUssQ0FBQyxnQ0FBZ0MsT0FBTyxFQUFFLENBQUMsQ0FBQTtRQUNoRCxJQUFJLENBQUMsSUFBSSxHQUFHLGdCQUFnQixDQUFBO0lBQzlCLENBQUM7Q0FDRjtBQUxELHdDQUtDO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLG1CQUFvQixTQUFRLEtBQUs7SUFDaEI7SUFBNUIsWUFBNEIsVUFBbUI7UUFDN0MsS0FBSyxDQUFDLHdCQUF3QixVQUFVLENBQUMsQ0FBQyxDQUFDLGlCQUFpQixVQUFVLFVBQVUsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQTtRQUQ5RCxlQUFVLEdBQVYsVUFBVSxDQUFTO1FBRTdDLElBQUksQ0FBQyxJQUFJLEdBQUcscUJBQXFCLENBQUE7SUFDbkMsQ0FBQztDQUNGO0FBTEQsa0RBS0M7QUFFRDs7O0dBR0c7QUFDSCxNQUFhLHFCQUFzQixTQUFRLEtBQUs7SUFFNUI7SUFDQTtJQUZsQixZQUNrQixXQUFvQixFQUNwQixVQUFvQjtRQUVwQyxLQUFLLENBQUMsa0NBQWtDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLDJCQUEyQixDQUFDLENBQUE7UUFIekUsZ0JBQVcsR0FBWCxXQUFXLENBQVM7UUFDcEIsZUFBVSxHQUFWLFVBQVUsQ0FBVTtRQUdwQyxJQUFJLENBQUMsSUFBSSxHQUFHLHVCQUF1QixDQUFBO0lBQ3JDLENBQUM7Q0FDRjtBQVJELHNEQVFDO0FBRUQ7O0dBRUc7QUFDSCxNQUFhLGFBQWMsU0FBUSxLQUFLO0lBQ3RDLFlBQVksSUFBWTtRQUN0QixLQUFLLENBQUMsNEJBQTRCLElBQUksRUFBRSxDQUFDLENBQUE7UUFDekMsSUFBSSxDQUFDLElBQUksR0FBRyxlQUFlLENBQUE7SUFDN0IsQ0FBQztDQUNGO0FBTEQsc0NBS0MiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNoaXBtZW50IG5vdCBmb3VuZCBpbiBUdXJ2b1xuICovXG5leHBvcnQgY2xhc3MgVHVydm9Ob3RGb3VuZEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgc2hpcG1lbnRJZDogc3RyaW5nKSB7XG4gICAgc3VwZXIoYFNoaXBtZW50IG5vdCBmb3VuZCBpbiBUdXJ2bzogJHtzaGlwbWVudElkfWApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvTm90Rm91bmRFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIEF1dGhlbnRpY2F0aW9uIGZhaWxlZCAoaW52YWxpZCBjcmVkZW50aWFscywgZXhwaXJlZCB0b2tlbilcbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvQXV0aEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihtZXNzYWdlOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgVHVydm8gYXV0aGVudGljYXRpb24gZmFpbGVkOiAke21lc3NhZ2V9YClcbiAgICB0aGlzLm5hbWUgPSAnVHVydm9BdXRoRXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBSYXRlIGxpbWl0ZWQgYnkgVHVydm9cbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvUmF0ZUxpbWl0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSByZXRyeUFmdGVyPzogbnVtYmVyKSB7XG4gICAgc3VwZXIoYFJhdGUgbGltaXRlZCBieSBUdXJ2byR7cmV0cnlBZnRlciA/IGAuIFJldHJ5IGFmdGVyICR7cmV0cnlBZnRlcn0gc2Vjb25kc2AgOiAnJ31gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b1JhdGVMaW1pdEVycm9yJ1xuICB9XG59XG5cbi8qKlxuICogSW50ZXJuYWwgQVBJIGZhaWxlZCBidXQgcHVibGljIEFQSSBzdWNjZWVkZWQuXG4gKiBDb250YWlucyBwYXJ0aWFsIGRhdGEgaW4gYHBhcnRpYWxEYXRhYCBmaWVsZC5cbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvUGFydGlhbERhdGFFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHJlYWRvbmx5IHBhcnRpYWxEYXRhOiB1bmtub3duLFxuICAgIHB1YmxpYyByZWFkb25seSBmYWlsZWRBcGlzOiBzdHJpbmdbXVxuICApIHtcbiAgICBzdXBlcihgVHVydm8gaW50ZXJuYWwgQVBJIGZhaWxlZCBmb3I6ICR7ZmFpbGVkQXBpcy5qb2luKCcsICcpfS4gUGFydGlhbCBkYXRhIGF2YWlsYWJsZS5gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b1BhcnRpYWxEYXRhRXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBUdXJ2byBzZWNyZXRzIGNhbm5vdCBiZSByZXRyaWV2ZWQgZnJvbSBBV1MgU2VjcmV0cyBNYW5hZ2VyXG4gKi9cbmV4cG9ydCBjbGFzcyBOb1NlY3JldEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwYXRoOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgTm8gc2VjcmV0IGZvdW5kIGF0IHBhdGg6ICR7cGF0aH1gKVxuICAgIHRoaXMubmFtZSA9ICdOb1NlY3JldEVycm9yJ1xuICB9XG59XG4iXX0=
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
// Config types
|
|
18
|
+
__exportStar(require("./config"), exports);
|
|
19
|
+
// Common API types
|
|
20
|
+
__exportStar(require("./common"), exports);
|
|
21
|
+
// Shipment types
|
|
22
|
+
__exportStar(require("./shipment"), exports);
|
|
23
|
+
// Tracking types
|
|
24
|
+
__exportStar(require("./tracking"), exports);
|
|
25
|
+
// Error types
|
|
26
|
+
__exportStar(require("./errors"), exports);
|
|
27
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdHlwZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7OztBQUFBLGVBQWU7QUFDZiwyQ0FBd0I7QUFFeEIsbUJBQW1CO0FBQ25CLDJDQUF3QjtBQUV4QixpQkFBaUI7QUFDakIsNkNBQTBCO0FBRTFCLGlCQUFpQjtBQUNqQiw2Q0FBMEI7QUFFMUIsY0FBYztBQUNkLDJDQUF3QiIsInNvdXJjZXNDb250ZW50IjpbIi8vIENvbmZpZyB0eXBlc1xuZXhwb3J0ICogZnJvbSAnLi9jb25maWcnXG5cbi8vIENvbW1vbiBBUEkgdHlwZXNcbmV4cG9ydCAqIGZyb20gJy4vY29tbW9uJ1xuXG4vLyBTaGlwbWVudCB0eXBlc1xuZXhwb3J0ICogZnJvbSAnLi9zaGlwbWVudCdcblxuLy8gVHJhY2tpbmcgdHlwZXNcbmV4cG9ydCAqIGZyb20gJy4vdHJhY2tpbmcnXG5cbi8vIEVycm9yIHR5cGVzXG5leHBvcnQgKiBmcm9tICcuL2Vycm9ycydcbiJdfQ==
|