@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,99 @@
|
|
|
1
|
+
import { TurvoApiMethod, } from '../types/common';
|
|
2
|
+
/**
|
|
3
|
+
* TurvoPublicApi provides access to documented Turvo API endpoints.
|
|
4
|
+
* All shipment management operations (create, update, cancel, query) go through this class.
|
|
5
|
+
*/
|
|
6
|
+
export class TurvoPublicApi {
|
|
7
|
+
client;
|
|
8
|
+
constructor(client) {
|
|
9
|
+
this.client = client;
|
|
10
|
+
}
|
|
11
|
+
/**
|
|
12
|
+
* GET /v1/shipments/{id}
|
|
13
|
+
* Returns shipment details, stops, status
|
|
14
|
+
*/
|
|
15
|
+
async getShipment(params) {
|
|
16
|
+
const { shipmentId } = params;
|
|
17
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
18
|
+
return httpClient.sendRequest(`/shipments/${shipmentId}`, TurvoApiMethod.GET, {});
|
|
19
|
+
}
|
|
20
|
+
/**
|
|
21
|
+
* POST /v1/shipments
|
|
22
|
+
* Upload properly formatted shipment to Turvo
|
|
23
|
+
*/
|
|
24
|
+
async uploadShipment(params) {
|
|
25
|
+
const { shipment } = params;
|
|
26
|
+
// Note: vehoLoadId is kept in the params for logging purposes by consumers
|
|
27
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
28
|
+
return httpClient.sendRequest('/shipments', TurvoApiMethod.POST, {
|
|
29
|
+
body: shipment,
|
|
30
|
+
});
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* PUT /v1/tags/attach/shipment/{id}
|
|
34
|
+
* Associate a list of tags with a shipment by ID
|
|
35
|
+
*/
|
|
36
|
+
async associateTagsToShipment(params) {
|
|
37
|
+
const { turvoShipmentId, tags } = params;
|
|
38
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
39
|
+
return httpClient.sendRequest(`/tags/attach/shipment/${turvoShipmentId}`, TurvoApiMethod.PUT, {
|
|
40
|
+
body: {
|
|
41
|
+
tagNames: tags,
|
|
42
|
+
},
|
|
43
|
+
});
|
|
44
|
+
}
|
|
45
|
+
/**
|
|
46
|
+
* PUT /v1/shipments/status/{id}
|
|
47
|
+
* Set shipment status to Cancelled in Turvo
|
|
48
|
+
*/
|
|
49
|
+
async cancelShipment(params) {
|
|
50
|
+
const { turvoShipmentId } = params;
|
|
51
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
52
|
+
const body = {
|
|
53
|
+
id: turvoShipmentId,
|
|
54
|
+
status: {
|
|
55
|
+
code: { key: '2113', value: 'Canceled' },
|
|
56
|
+
},
|
|
57
|
+
};
|
|
58
|
+
return httpClient.sendRequest(`/shipments/status/${turvoShipmentId}`, TurvoApiMethod.PUT, {
|
|
59
|
+
body,
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* PUT /v1/shipments/status/{id}
|
|
64
|
+
* Update shipment status at a specific stop
|
|
65
|
+
*/
|
|
66
|
+
async updateShipmentStatus(params) {
|
|
67
|
+
const { turvoShipmentId, turvoStopId, turvoStatusCode, statusDate, statusTimezone } = params;
|
|
68
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
69
|
+
const body = {
|
|
70
|
+
id: turvoShipmentId,
|
|
71
|
+
status: {
|
|
72
|
+
globalShipLocationId: turvoStopId,
|
|
73
|
+
code: turvoStatusCode.code,
|
|
74
|
+
timezone: statusTimezone,
|
|
75
|
+
statusDate: {
|
|
76
|
+
date: statusDate,
|
|
77
|
+
timezone: statusTimezone,
|
|
78
|
+
},
|
|
79
|
+
},
|
|
80
|
+
};
|
|
81
|
+
return httpClient.sendRequest(`/shipments/status/${turvoShipmentId}`, TurvoApiMethod.PUT, { body });
|
|
82
|
+
}
|
|
83
|
+
/**
|
|
84
|
+
* GET /v1/shipments/list
|
|
85
|
+
* Query shipments from Turvo with pagination support
|
|
86
|
+
*/
|
|
87
|
+
async filterShipments(params) {
|
|
88
|
+
const { turvoLocationId, pickupDateStart, start = 0 } = params;
|
|
89
|
+
const httpClient = await this.client.getAuthenticatedClient();
|
|
90
|
+
return httpClient.sendRequest('/shipments/list', TurvoApiMethod.GET, {
|
|
91
|
+
query: {
|
|
92
|
+
'locationId[eq]': turvoLocationId.toString(),
|
|
93
|
+
'pickupDate[gte]': pickupDateStart,
|
|
94
|
+
start: start.toString(),
|
|
95
|
+
},
|
|
96
|
+
});
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
//# 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"]}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
import { TurvoApiMethod, TurvoApiResult } from '../types/common';
|
|
2
|
+
import { TurvoCredentials } from '../types/config';
|
|
3
|
+
/**
|
|
4
|
+
* Clear the cached access token (useful for testing or when token expires)
|
|
5
|
+
*/
|
|
6
|
+
export declare const clearCachedAccessToken: () => void;
|
|
7
|
+
/**
|
|
8
|
+
* HTTP client interface for making authenticated Turvo API requests
|
|
9
|
+
*/
|
|
10
|
+
export interface AuthenticatedHttpClient {
|
|
11
|
+
/**
|
|
12
|
+
* Send a request to the Turvo API
|
|
13
|
+
* @param apiRoute - API route path (e.g., '/shipments')
|
|
14
|
+
* @param method - HTTP method
|
|
15
|
+
* @param params - Request parameters (body, headers, query)
|
|
16
|
+
* @returns Typed API result
|
|
17
|
+
*/
|
|
18
|
+
sendRequest<T>(apiRoute: string, method: TurvoApiMethod, params: {
|
|
19
|
+
body?: object;
|
|
20
|
+
headers?: Record<string, string>;
|
|
21
|
+
query?: Record<string, string>;
|
|
22
|
+
}): Promise<TurvoApiResult<T>>;
|
|
23
|
+
}
|
|
24
|
+
/**
|
|
25
|
+
* Main client for interacting with Turvo APIs.
|
|
26
|
+
* Handles authentication, token management, and HTTP requests.
|
|
27
|
+
*
|
|
28
|
+
* @example
|
|
29
|
+
* ```typescript
|
|
30
|
+
* const client = new TurvoClient()
|
|
31
|
+
* const httpClient = await client.getAuthenticatedClient()
|
|
32
|
+
* const result = await httpClient.sendRequest('/shipments/123', TurvoApiMethod.GET, {})
|
|
33
|
+
* ```
|
|
34
|
+
*/
|
|
35
|
+
export declare class TurvoClient {
|
|
36
|
+
private secretPath;
|
|
37
|
+
private credentials?;
|
|
38
|
+
private httpClient?;
|
|
39
|
+
constructor(secretPath?: string);
|
|
40
|
+
/**
|
|
41
|
+
* Get authenticated HTTP client.
|
|
42
|
+
* Handles token refresh automatically.
|
|
43
|
+
*/
|
|
44
|
+
getAuthenticatedClient(): Promise<AuthenticatedHttpClient>;
|
|
45
|
+
/**
|
|
46
|
+
* Get raw secrets (for internal API auth which uses different flow)
|
|
47
|
+
*/
|
|
48
|
+
getCredentials(): Promise<TurvoCredentials>;
|
|
49
|
+
}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { Client } from 'lambda-params-secrets';
|
|
2
|
+
import fetch from 'node-fetch';
|
|
3
|
+
import pRetry from 'p-retry';
|
|
4
|
+
import pThrottle from 'p-throttle';
|
|
5
|
+
import { DEFAULT_ALLOWED_RETRIES, DEFAULT_API_ROUTE_PREFIX, DEFAULT_RATE_LIMIT_RPS, TURVO_SECRET_PATH, } from '../constants';
|
|
6
|
+
import { TurvoApiError, TurvoApiResponseStatus } from '../types/common';
|
|
7
|
+
import { NoSecretError, TurvoAuthError } from '../types/errors';
|
|
8
|
+
let secretClient = null;
|
|
9
|
+
/**
|
|
10
|
+
* Get or create the secrets client instance
|
|
11
|
+
*/
|
|
12
|
+
const getSecretClientInstance = () => {
|
|
13
|
+
if (!secretClient) {
|
|
14
|
+
secretClient = new Client();
|
|
15
|
+
}
|
|
16
|
+
return secretClient;
|
|
17
|
+
};
|
|
18
|
+
/**
|
|
19
|
+
* Fetch Turvo API credentials from AWS Secrets Manager
|
|
20
|
+
*/
|
|
21
|
+
const getTurvoSecrets = async (secretsPath = TURVO_SECRET_PATH) => {
|
|
22
|
+
const fetchedSecret = await getSecretClientInstance().stringSecret(secretsPath);
|
|
23
|
+
if (!fetchedSecret) {
|
|
24
|
+
throw new NoSecretError(secretsPath);
|
|
25
|
+
}
|
|
26
|
+
return JSON.parse(fetchedSecret);
|
|
27
|
+
};
|
|
28
|
+
/**
|
|
29
|
+
* Cached access token to avoid repeated OAuth requests
|
|
30
|
+
*/
|
|
31
|
+
let cachedAccessToken = null;
|
|
32
|
+
let tokenExpiry = null;
|
|
33
|
+
/**
|
|
34
|
+
* Fetch an access token from Turvo using OAuth2 password grant
|
|
35
|
+
*/
|
|
36
|
+
const getTurvoAccessToken = async (credentials, apiRoutePrefix = DEFAULT_API_ROUTE_PREFIX, allowedRetries = DEFAULT_ALLOWED_RETRIES) => {
|
|
37
|
+
// Check if we have a valid cached token
|
|
38
|
+
if (cachedAccessToken && tokenExpiry && tokenExpiry > new Date()) {
|
|
39
|
+
return cachedAccessToken;
|
|
40
|
+
}
|
|
41
|
+
const fullUrl = `${credentials.urlBase + apiRoutePrefix}/oauth/token`;
|
|
42
|
+
return pRetry(async () => {
|
|
43
|
+
const url = new URL(fullUrl);
|
|
44
|
+
url.searchParams.set('client_id', 'publicapi');
|
|
45
|
+
url.searchParams.set('client_secret', 'secret');
|
|
46
|
+
const response = await fetch(url, {
|
|
47
|
+
method: 'POST',
|
|
48
|
+
headers: {
|
|
49
|
+
'Content-Type': 'application/json',
|
|
50
|
+
'x-api-key': credentials.apiKey,
|
|
51
|
+
},
|
|
52
|
+
body: JSON.stringify({
|
|
53
|
+
grant_type: 'password',
|
|
54
|
+
scope: 'read+trust+write',
|
|
55
|
+
username: credentials.username,
|
|
56
|
+
password: credentials.password,
|
|
57
|
+
type: 'business',
|
|
58
|
+
}),
|
|
59
|
+
});
|
|
60
|
+
if (!response.ok) {
|
|
61
|
+
throw new TurvoAuthError(`Failed to authenticate: ${response.status} ${response.statusText}`);
|
|
62
|
+
}
|
|
63
|
+
const data = (await response.json());
|
|
64
|
+
const { access_token, expires_in } = data;
|
|
65
|
+
if (!access_token) {
|
|
66
|
+
throw new TurvoAuthError('No access token returned from Turvo');
|
|
67
|
+
}
|
|
68
|
+
cachedAccessToken = access_token;
|
|
69
|
+
// Set token expiry (expires_in is in seconds, default to 1 hour if not provided)
|
|
70
|
+
const expiresInMs = (expires_in || 3600) * 1000;
|
|
71
|
+
tokenExpiry = new Date(Date.now() + expiresInMs - 60000); // Refresh 1 minute early
|
|
72
|
+
return access_token;
|
|
73
|
+
}, { retries: allowedRetries });
|
|
74
|
+
};
|
|
75
|
+
/**
|
|
76
|
+
* Clear the cached access token (useful for testing or when token expires)
|
|
77
|
+
*/
|
|
78
|
+
export const clearCachedAccessToken = () => {
|
|
79
|
+
cachedAccessToken = null;
|
|
80
|
+
tokenExpiry = null;
|
|
81
|
+
};
|
|
82
|
+
/**
|
|
83
|
+
* Main client for interacting with Turvo APIs.
|
|
84
|
+
* Handles authentication, token management, and HTTP requests.
|
|
85
|
+
*
|
|
86
|
+
* @example
|
|
87
|
+
* ```typescript
|
|
88
|
+
* const client = new TurvoClient()
|
|
89
|
+
* const httpClient = await client.getAuthenticatedClient()
|
|
90
|
+
* const result = await httpClient.sendRequest('/shipments/123', TurvoApiMethod.GET, {})
|
|
91
|
+
* ```
|
|
92
|
+
*/
|
|
93
|
+
export class TurvoClient {
|
|
94
|
+
secretPath;
|
|
95
|
+
credentials;
|
|
96
|
+
httpClient;
|
|
97
|
+
constructor(secretPath = TURVO_SECRET_PATH) {
|
|
98
|
+
this.secretPath = secretPath;
|
|
99
|
+
}
|
|
100
|
+
/**
|
|
101
|
+
* Get authenticated HTTP client.
|
|
102
|
+
* Handles token refresh automatically.
|
|
103
|
+
*/
|
|
104
|
+
async getAuthenticatedClient() {
|
|
105
|
+
if (this.httpClient) {
|
|
106
|
+
return this.httpClient;
|
|
107
|
+
}
|
|
108
|
+
// Fetch credentials if not already loaded
|
|
109
|
+
if (!this.credentials) {
|
|
110
|
+
this.credentials = await getTurvoSecrets(this.secretPath);
|
|
111
|
+
}
|
|
112
|
+
// Create HTTP client with rate limiting and retries
|
|
113
|
+
const apiRoutePrefix = DEFAULT_API_ROUTE_PREFIX;
|
|
114
|
+
const allowedRetries = DEFAULT_ALLOWED_RETRIES;
|
|
115
|
+
const rateLimitRequestsPerSecond = DEFAULT_RATE_LIMIT_RPS;
|
|
116
|
+
// Create throttled version of the send function
|
|
117
|
+
const throttle = pThrottle({
|
|
118
|
+
limit: rateLimitRequestsPerSecond,
|
|
119
|
+
interval: 1000,
|
|
120
|
+
});
|
|
121
|
+
/**
|
|
122
|
+
* Internal function to send API request (with retry logic)
|
|
123
|
+
*/
|
|
124
|
+
const sendToTurvoAPI = async (apiRoute, method, params) => {
|
|
125
|
+
const accessToken = await getTurvoAccessToken(this.credentials, apiRoutePrefix, allowedRetries);
|
|
126
|
+
const { body, headers, query } = params;
|
|
127
|
+
let fullRoute = `${this.credentials.urlBase + apiRoutePrefix + apiRoute}`;
|
|
128
|
+
if (query && Object.keys(query).length > 0) {
|
|
129
|
+
fullRoute = `${fullRoute}?${new URLSearchParams(query)}`;
|
|
130
|
+
}
|
|
131
|
+
return pRetry(async () => {
|
|
132
|
+
return fetch(fullRoute, {
|
|
133
|
+
method,
|
|
134
|
+
headers: {
|
|
135
|
+
Authorization: `Bearer ${accessToken}`,
|
|
136
|
+
'Content-Type': 'application/json',
|
|
137
|
+
'x-api-key': this.credentials.apiKey,
|
|
138
|
+
...headers,
|
|
139
|
+
},
|
|
140
|
+
...(body && { body: JSON.stringify(body) }),
|
|
141
|
+
}).then(async (res) => {
|
|
142
|
+
const result = await res.json();
|
|
143
|
+
// Handle rate limiting (429)
|
|
144
|
+
if (res.status === 429) {
|
|
145
|
+
throw new TurvoApiError(res.status.toString(), result.message || 'Rate limited');
|
|
146
|
+
}
|
|
147
|
+
// Check for API-level errors
|
|
148
|
+
if (result.Status !== TurvoApiResponseStatus.SUCCESS) {
|
|
149
|
+
throw new TurvoApiError(result.details?.errorCode || 'UNKNOWN', result.details?.errorMessage || 'API error');
|
|
150
|
+
}
|
|
151
|
+
return result;
|
|
152
|
+
});
|
|
153
|
+
}, { retries: allowedRetries });
|
|
154
|
+
};
|
|
155
|
+
// Throttle the send function
|
|
156
|
+
const throttledSend = throttle(sendToTurvoAPI);
|
|
157
|
+
this.httpClient = {
|
|
158
|
+
sendRequest: throttledSend,
|
|
159
|
+
};
|
|
160
|
+
return this.httpClient;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Get raw secrets (for internal API auth which uses different flow)
|
|
164
|
+
*/
|
|
165
|
+
async getCredentials() {
|
|
166
|
+
if (!this.credentials) {
|
|
167
|
+
this.credentials = await getTurvoSecrets(this.secretPath);
|
|
168
|
+
}
|
|
169
|
+
return this.credentials;
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turvoClient.js","sourceRoot":"","sources":["../../../src/client/turvoClient.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,uBAAuB,CAAA;AAC9C,OAAO,KAAK,MAAM,YAAY,CAAA;AAC9B,OAAO,MAAM,MAAM,SAAS,CAAA;AAC5B,OAAO,SAAS,MAAM,YAAY,CAAA;AAElC,OAAO,EACL,uBAAuB,EACvB,wBAAwB,EACxB,sBAAsB,EACtB,iBAAiB,GAClB,MAAM,cAAc,CAAA;AACrB,OAAO,EAAE,aAAa,EAAkB,sBAAsB,EAAkB,MAAM,iBAAiB,CAAA;AAEvG,OAAO,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,iBAAiB,CAAA;AAE/D,IAAI,YAAY,GAAkB,IAAI,CAAA;AAEtC;;GAEG;AACH,MAAM,uBAAuB,GAAG,GAAW,EAAE;IAC3C,IAAI,CAAC,YAAY,EAAE,CAAC;QAClB,YAAY,GAAG,IAAI,MAAM,EAAE,CAAA;IAC7B,CAAC;IACD,OAAO,YAAY,CAAA;AACrB,CAAC,CAAA;AAED;;GAEG;AACH,MAAM,eAAe,GAAG,KAAK,EAAE,cAAsB,iBAAiB,EAA6B,EAAE;IACnG,MAAM,aAAa,GAAG,MAAM,uBAAuB,EAAE,CAAC,YAAY,CAAC,WAAW,CAAC,CAAA;IAE/E,IAAI,CAAC,aAAa,EAAE,CAAC;QACnB,MAAM,IAAI,aAAa,CAAC,WAAW,CAAC,CAAA;IACtC,CAAC;IAED,OAAO,IAAI,CAAC,KAAK,CAAC,aAAa,CAAqB,CAAA;AACtD,CAAC,CAAA;AAED;;GAEG;AACH,IAAI,iBAAiB,GAAkB,IAAI,CAAA;AAC3C,IAAI,WAAW,GAAgB,IAAI,CAAA;AAEnC;;GAEG;AACH,MAAM,mBAAmB,GAAG,KAAK,EAC/B,WAA6B,EAC7B,iBAAyB,wBAAwB,EACjD,iBAAyB,uBAAuB,EAC/B,EAAE;IACnB,wCAAwC;IACxC,IAAI,iBAAiB,IAAI,WAAW,IAAI,WAAW,GAAG,IAAI,IAAI,EAAE,EAAE,CAAC;QACjE,OAAO,iBAAiB,CAAA;IAC1B,CAAC;IAED,MAAM,OAAO,GAAG,GAAG,WAAW,CAAC,OAAO,GAAG,cAAc,cAAc,CAAA;IAErE,OAAO,MAAM,CACX,KAAK,IAAI,EAAE;QACT,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,OAAO,CAAC,CAAA;QAC5B,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,WAAW,EAAE,WAAW,CAAC,CAAA;QAC9C,GAAG,CAAC,YAAY,CAAC,GAAG,CAAC,eAAe,EAAE,QAAQ,CAAC,CAAA;QAE/C,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;YAChC,MAAM,EAAE,MAAM;YACd,OAAO,EAAE;gBACP,cAAc,EAAE,kBAAkB;gBAClC,WAAW,EAAE,WAAW,CAAC,MAAM;aAChC;YACD,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC;gBACnB,UAAU,EAAE,UAAU;gBACtB,KAAK,EAAE,kBAAkB;gBACzB,QAAQ,EAAE,WAAW,CAAC,QAAQ;gBAC9B,QAAQ,EAAE,WAAW,CAAC,QAAQ;gBAC9B,IAAI,EAAE,UAAU;aACjB,CAAC;SACH,CAAC,CAAA;QAEF,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,IAAI,cAAc,CAAC,2BAA2B,QAAQ,CAAC,MAAM,IAAI,QAAQ,CAAC,UAAU,EAAE,CAAC,CAAA;QAC/F,CAAC;QAED,MAAM,IAAI,GAAG,CAAC,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAkD,CAAA;QACrF,MAAM,EAAE,YAAY,EAAE,UAAU,EAAE,GAAG,IAAI,CAAA;QAEzC,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,IAAI,cAAc,CAAC,qCAAqC,CAAC,CAAA;QACjE,CAAC;QAED,iBAAiB,GAAG,YAAY,CAAA;QAEhC,iFAAiF;QACjF,MAAM,WAAW,GAAG,CAAC,UAAU,IAAI,IAAI,CAAC,GAAG,IAAI,CAAA;QAC/C,WAAW,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,WAAW,GAAG,KAAK,CAAC,CAAA,CAAC,yBAAyB;QAElF,OAAO,YAAY,CAAA;IACrB,CAAC,EACD,EAAE,OAAO,EAAE,cAAc,EAAE,CAC5B,CAAA;AACH,CAAC,CAAA;AAED;;GAEG;AACH,MAAM,CAAC,MAAM,sBAAsB,GAAG,GAAS,EAAE;IAC/C,iBAAiB,GAAG,IAAI,CAAA;IACxB,WAAW,GAAG,IAAI,CAAA;AACpB,CAAC,CAAA;AAoBD;;;;;;;;;;GAUG;AACH,MAAM,OAAO,WAAW;IAIF;IAHZ,WAAW,CAAmB;IAC9B,UAAU,CAA0B;IAE5C,YAAoB,aAAqB,iBAAiB;QAAtC,eAAU,GAAV,UAAU,CAA4B;IAAG,CAAC;IAE9D;;;OAGG;IACH,KAAK,CAAC,sBAAsB;QAC1B,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,UAAU,CAAA;QACxB,CAAC;QAED,0CAA0C;QAC1C,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,IAAI,CAAC,WAAW,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;QAC3D,CAAC;QAED,oDAAoD;QACpD,MAAM,cAAc,GAAG,wBAAwB,CAAA;QAC/C,MAAM,cAAc,GAAG,uBAAuB,CAAA;QAC9C,MAAM,0BAA0B,GAAG,sBAAsB,CAAA;QAEzD,gDAAgD;QAChD,MAAM,QAAQ,GAAG,SAAS,CAAC;YACzB,KAAK,EAAE,0BAA0B;YACjC,QAAQ,EAAE,IAAI;SACf,CAAC,CAAA;QAEF;;WAEG;QACH,MAAM,cAAc,GAAG,KAAK,EAC1B,QAAgB,EAChB,MAAsB,EACtB,MAA2F,EAC/D,EAAE;YAC9B,MAAM,WAAW,GAAG,MAAM,mBAAmB,CAAC,IAAI,CAAC,WAAY,EAAE,cAAc,EAAE,cAAc,CAAC,CAAA;YAChG,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,GAAG,MAAM,CAAA;YACvC,IAAI,SAAS,GAAG,GAAG,IAAI,CAAC,WAAY,CAAC,OAAO,GAAG,cAAc,GAAG,QAAQ,EAAE,CAAA;YAE1E,IAAI,KAAK,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC3C,SAAS,GAAG,GAAG,SAAS,IAAI,IAAI,eAAe,CAAC,KAAK,CAAC,EAAE,CAAA;YAC1D,CAAC;YAED,OAAO,MAAM,CACX,KAAK,IAAI,EAAE;gBACT,OAAO,KAAK,CAAC,SAAS,EAAE;oBACtB,MAAM;oBACN,OAAO,EAAE;wBACP,aAAa,EAAE,UAAU,WAAW,EAAE;wBACtC,cAAc,EAAE,kBAAkB;wBAClC,WAAW,EAAE,IAAI,CAAC,WAAY,CAAC,MAAM;wBACrC,GAAG,OAAO;qBACX;oBACD,GAAG,CAAC,IAAI,IAAI,EAAE,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC;iBAC5C,CAAC,CAAC,IAAI,CAAC,KAAK,EAAC,GAAG,EAAC,EAAE;oBAClB,MAAM,MAAM,GAAG,MAAM,GAAG,CAAC,IAAI,EAAE,CAAA;oBAE/B,6BAA6B;oBAC7B,IAAI,GAAG,CAAC,MAAM,KAAK,GAAG,EAAE,CAAC;wBACvB,MAAM,IAAI,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,MAAM,CAAC,OAAO,IAAI,cAAc,CAAC,CAAA;oBAClF,CAAC;oBAED,6BAA6B;oBAC7B,IAAI,MAAM,CAAC,MAAM,KAAK,sBAAsB,CAAC,OAAO,EAAE,CAAC;wBACrD,MAAM,IAAI,aAAa,CACrB,MAAM,CAAC,OAAO,EAAE,SAAS,IAAI,SAAS,EACtC,MAAM,CAAC,OAAO,EAAE,YAAY,IAAI,WAAW,CAC5C,CAAA;oBACH,CAAC;oBAED,OAAO,MAAM,CAAA;gBACf,CAAC,CAAC,CAAA;YACJ,CAAC,EACD,EAAE,OAAO,EAAE,cAAc,EAAE,CAC5B,CAAA;QACH,CAAC,CAAA;QAED,6BAA6B;QAC7B,MAAM,aAAa,GAAG,QAAQ,CAAC,cAAc,CAAC,CAAA;QAE9C,IAAI,CAAC,UAAU,GAAG;YAChB,WAAW,EAAE,aAAa;SAC3B,CAAA;QAED,OAAO,IAAI,CAAC,UAAU,CAAA;IACxB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,cAAc;QAClB,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;YACtB,IAAI,CAAC,WAAW,GAAG,MAAM,eAAe,CAAC,IAAI,CAAC,UAAU,CAAC,CAAA;QAC3D,CAAC;QACD,OAAO,IAAI,CAAC,WAAW,CAAA;IACzB,CAAC;CACF","sourcesContent":["import { Client } from 'lambda-params-secrets'\nimport fetch from 'node-fetch'\nimport pRetry from 'p-retry'\nimport pThrottle from 'p-throttle'\n\nimport {\n  DEFAULT_ALLOWED_RETRIES,\n  DEFAULT_API_ROUTE_PREFIX,\n  DEFAULT_RATE_LIMIT_RPS,\n  TURVO_SECRET_PATH,\n} from '../constants'\nimport { TurvoApiError, TurvoApiMethod, TurvoApiResponseStatus, TurvoApiResult } from '../types/common'\nimport { TurvoCredentials } from '../types/config'\nimport { NoSecretError, TurvoAuthError } from '../types/errors'\n\nlet secretClient: Client | null = null\n\n/**\n * Get or create the secrets client instance\n */\nconst getSecretClientInstance = (): Client => {\n  if (!secretClient) {\n    secretClient = new Client()\n  }\n  return secretClient\n}\n\n/**\n * Fetch Turvo API credentials from AWS Secrets Manager\n */\nconst getTurvoSecrets = async (secretsPath: string = TURVO_SECRET_PATH): Promise<TurvoCredentials> => {\n  const fetchedSecret = await getSecretClientInstance().stringSecret(secretsPath)\n\n  if (!fetchedSecret) {\n    throw new NoSecretError(secretsPath)\n  }\n\n  return JSON.parse(fetchedSecret) as TurvoCredentials\n}\n\n/**\n * Cached access token to avoid repeated OAuth requests\n */\nlet cachedAccessToken: string | null = null\nlet tokenExpiry: Date | null = null\n\n/**\n * Fetch an access token from Turvo using OAuth2 password grant\n */\nconst getTurvoAccessToken = async (\n  credentials: TurvoCredentials,\n  apiRoutePrefix: string = DEFAULT_API_ROUTE_PREFIX,\n  allowedRetries: number = DEFAULT_ALLOWED_RETRIES\n): Promise<string> => {\n  // Check if we have a valid cached token\n  if (cachedAccessToken && tokenExpiry && tokenExpiry > new Date()) {\n    return cachedAccessToken\n  }\n\n  const fullUrl = `${credentials.urlBase + apiRoutePrefix}/oauth/token`\n\n  return pRetry(\n    async () => {\n      const url = new URL(fullUrl)\n      url.searchParams.set('client_id', 'publicapi')\n      url.searchParams.set('client_secret', 'secret')\n\n      const response = await fetch(url, {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          'x-api-key': credentials.apiKey,\n        },\n        body: JSON.stringify({\n          grant_type: 'password',\n          scope: 'read+trust+write',\n          username: credentials.username,\n          password: credentials.password,\n          type: 'business',\n        }),\n      })\n\n      if (!response.ok) {\n        throw new TurvoAuthError(`Failed to authenticate: ${response.status} ${response.statusText}`)\n      }\n\n      const data = (await response.json()) as { access_token: string; expires_in?: number }\n      const { access_token, expires_in } = data\n\n      if (!access_token) {\n        throw new TurvoAuthError('No access token returned from Turvo')\n      }\n\n      cachedAccessToken = access_token\n\n      // Set token expiry (expires_in is in seconds, default to 1 hour if not provided)\n      const expiresInMs = (expires_in || 3600) * 1000\n      tokenExpiry = new Date(Date.now() + expiresInMs - 60000) // Refresh 1 minute early\n\n      return access_token\n    },\n    { retries: allowedRetries }\n  )\n}\n\n/**\n * Clear the cached access token (useful for testing or when token expires)\n */\nexport const clearCachedAccessToken = (): void => {\n  cachedAccessToken = null\n  tokenExpiry = null\n}\n\n/**\n * HTTP client interface for making authenticated Turvo API requests\n */\nexport interface AuthenticatedHttpClient {\n  /**\n   * Send a request to the Turvo API\n   * @param apiRoute - API route path (e.g., '/shipments')\n   * @param method - HTTP method\n   * @param params - Request parameters (body, headers, query)\n   * @returns Typed API result\n   */\n  sendRequest<T>(\n    apiRoute: string,\n    method: TurvoApiMethod,\n    params: { body?: object; headers?: Record<string, string>; query?: Record<string, string> }\n  ): Promise<TurvoApiResult<T>>\n}\n\n/**\n * Main client for interacting with Turvo APIs.\n * Handles authentication, token management, and HTTP requests.\n *\n * @example\n * ```typescript\n * const client = new TurvoClient()\n * const httpClient = await client.getAuthenticatedClient()\n * const result = await httpClient.sendRequest('/shipments/123', TurvoApiMethod.GET, {})\n * ```\n */\nexport class TurvoClient {\n  private credentials?: TurvoCredentials\n  private httpClient?: AuthenticatedHttpClient\n\n  constructor(private secretPath: string = TURVO_SECRET_PATH) {}\n\n  /**\n   * Get authenticated HTTP client.\n   * Handles token refresh automatically.\n   */\n  async getAuthenticatedClient(): Promise<AuthenticatedHttpClient> {\n    if (this.httpClient) {\n      return this.httpClient\n    }\n\n    // Fetch credentials if not already loaded\n    if (!this.credentials) {\n      this.credentials = await getTurvoSecrets(this.secretPath)\n    }\n\n    // Create HTTP client with rate limiting and retries\n    const apiRoutePrefix = DEFAULT_API_ROUTE_PREFIX\n    const allowedRetries = DEFAULT_ALLOWED_RETRIES\n    const rateLimitRequestsPerSecond = DEFAULT_RATE_LIMIT_RPS\n\n    // Create throttled version of the send function\n    const throttle = pThrottle({\n      limit: rateLimitRequestsPerSecond,\n      interval: 1000,\n    })\n\n    /**\n     * Internal function to send API request (with retry logic)\n     */\n    const sendToTurvoAPI = async <T>(\n      apiRoute: string,\n      method: TurvoApiMethod,\n      params: { body?: object; headers?: Record<string, string>; query?: Record<string, string> }\n    ): Promise<TurvoApiResult<T>> => {\n      const accessToken = await getTurvoAccessToken(this.credentials!, apiRoutePrefix, allowedRetries)\n      const { body, headers, query } = params\n      let fullRoute = `${this.credentials!.urlBase + apiRoutePrefix + apiRoute}`\n\n      if (query && Object.keys(query).length > 0) {\n        fullRoute = `${fullRoute}?${new URLSearchParams(query)}`\n      }\n\n      return pRetry(\n        async () => {\n          return fetch(fullRoute, {\n            method,\n            headers: {\n              Authorization: `Bearer ${accessToken}`,\n              'Content-Type': 'application/json',\n              'x-api-key': this.credentials!.apiKey,\n              ...headers,\n            },\n            ...(body && { body: JSON.stringify(body) }),\n          }).then(async res => {\n            const result = await res.json()\n\n            // Handle rate limiting (429)\n            if (res.status === 429) {\n              throw new TurvoApiError(res.status.toString(), result.message || 'Rate limited')\n            }\n\n            // Check for API-level errors\n            if (result.Status !== TurvoApiResponseStatus.SUCCESS) {\n              throw new TurvoApiError(\n                result.details?.errorCode || 'UNKNOWN',\n                result.details?.errorMessage || 'API error'\n              )\n            }\n\n            return result\n          })\n        },\n        { retries: allowedRetries }\n      )\n    }\n\n    // Throttle the send function\n    const throttledSend = throttle(sendToTurvoAPI)\n\n    this.httpClient = {\n      sendRequest: throttledSend,\n    }\n\n    return this.httpClient\n  }\n\n  /**\n   * Get raw secrets (for internal API auth which uses different flow)\n   */\n  async getCredentials(): Promise<TurvoCredentials> {\n    if (!this.credentials) {\n      this.credentials = await getTurvoSecrets(this.secretPath)\n    }\n    return this.credentials\n  }\n}\n"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default secret path for Turvo credentials in AWS Secrets Manager.
|
|
3
|
+
* Export this so consumers can grant IAM permissions.
|
|
4
|
+
*
|
|
5
|
+
* @example
|
|
6
|
+
* ```typescript
|
|
7
|
+
* import { TURVO_SECRET_PATH } from '@veho/turvo-sdk'
|
|
8
|
+
*
|
|
9
|
+
* // In CDK stack
|
|
10
|
+
* myLambda.addToRolePolicy(new iam.PolicyStatement({
|
|
11
|
+
* actions: ['secretsmanager:GetSecretValue'],
|
|
12
|
+
* resources: [`arn:aws:secretsmanager:*:*:secret:${TURVO_SECRET_PATH}*`],
|
|
13
|
+
* }))
|
|
14
|
+
* ```
|
|
15
|
+
*/
|
|
16
|
+
export declare const TURVO_SECRET_PATH = "/turvo/api";
|
|
17
|
+
/**
|
|
18
|
+
* Default API route prefix for Turvo endpoints.
|
|
19
|
+
* Currently, Turvo uses a common prefix of "/v1" for all documented API routes.
|
|
20
|
+
*/
|
|
21
|
+
export declare const DEFAULT_API_ROUTE_PREFIX = "/v1";
|
|
22
|
+
/**
|
|
23
|
+
* Default number of retry attempts for failed requests.
|
|
24
|
+
*/
|
|
25
|
+
export declare const DEFAULT_ALLOWED_RETRIES = 2;
|
|
26
|
+
/**
|
|
27
|
+
* Default rate limit (requests per second) for Turvo API calls.
|
|
28
|
+
*/
|
|
29
|
+
export declare const DEFAULT_RATE_LIMIT_RPS = 1;
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default secret path for Turvo credentials in AWS Secrets Manager.
|
|
3
|
+
* Export this so consumers can grant IAM permissions.
|
|
4
|
+
*
|
|
5
|
+
* @example
|
|
6
|
+
* ```typescript
|
|
7
|
+
* import { TURVO_SECRET_PATH } from '@veho/turvo-sdk'
|
|
8
|
+
*
|
|
9
|
+
* // In CDK stack
|
|
10
|
+
* myLambda.addToRolePolicy(new iam.PolicyStatement({
|
|
11
|
+
* actions: ['secretsmanager:GetSecretValue'],
|
|
12
|
+
* resources: [`arn:aws:secretsmanager:*:*:secret:${TURVO_SECRET_PATH}*`],
|
|
13
|
+
* }))
|
|
14
|
+
* ```
|
|
15
|
+
*/
|
|
16
|
+
export const TURVO_SECRET_PATH = '/turvo/api';
|
|
17
|
+
/**
|
|
18
|
+
* Default API route prefix for Turvo endpoints.
|
|
19
|
+
* Currently, Turvo uses a common prefix of "/v1" for all documented API routes.
|
|
20
|
+
*/
|
|
21
|
+
export const DEFAULT_API_ROUTE_PREFIX = '/v1';
|
|
22
|
+
/**
|
|
23
|
+
* Default number of retry attempts for failed requests.
|
|
24
|
+
*/
|
|
25
|
+
export const DEFAULT_ALLOWED_RETRIES = 2;
|
|
26
|
+
/**
|
|
27
|
+
* Default rate limit (requests per second) for Turvo API calls.
|
|
28
|
+
*/
|
|
29
|
+
export const DEFAULT_RATE_LIMIT_RPS = 1;
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRztBQUNILE1BQU0sQ0FBQyxNQUFNLGlCQUFpQixHQUFHLFlBQVksQ0FBQTtBQUU3Qzs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSx3QkFBd0IsR0FBRyxLQUFLLENBQUE7QUFFN0M7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSx1QkFBdUIsR0FBRyxDQUFDLENBQUE7QUFFeEM7O0dBRUc7QUFDSCxNQUFNLENBQUMsTUFBTSxzQkFBc0IsR0FBRyxDQUFDLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIERlZmF1bHQgc2VjcmV0IHBhdGggZm9yIFR1cnZvIGNyZWRlbnRpYWxzIGluIEFXUyBTZWNyZXRzIE1hbmFnZXIuXG4gKiBFeHBvcnQgdGhpcyBzbyBjb25zdW1lcnMgY2FuIGdyYW50IElBTSBwZXJtaXNzaW9ucy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHlwZXNjcmlwdFxuICogaW1wb3J0IHsgVFVSVk9fU0VDUkVUX1BBVEggfSBmcm9tICdAdmVoby90dXJ2by1zZGsnXG4gKlxuICogLy8gSW4gQ0RLIHN0YWNrXG4gKiBteUxhbWJkYS5hZGRUb1JvbGVQb2xpY3kobmV3IGlhbS5Qb2xpY3lTdGF0ZW1lbnQoe1xuICogICBhY3Rpb25zOiBbJ3NlY3JldHNtYW5hZ2VyOkdldFNlY3JldFZhbHVlJ10sXG4gKiAgIHJlc291cmNlczogW2Bhcm46YXdzOnNlY3JldHNtYW5hZ2VyOio6KjpzZWNyZXQ6JHtUVVJWT19TRUNSRVRfUEFUSH0qYF0sXG4gKiB9KSlcbiAqIGBgYFxuICovXG5leHBvcnQgY29uc3QgVFVSVk9fU0VDUkVUX1BBVEggPSAnL3R1cnZvL2FwaSdcblxuLyoqXG4gKiBEZWZhdWx0IEFQSSByb3V0ZSBwcmVmaXggZm9yIFR1cnZvIGVuZHBvaW50cy5cbiAqIEN1cnJlbnRseSwgVHVydm8gdXNlcyBhIGNvbW1vbiBwcmVmaXggb2YgXCIvdjFcIiBmb3IgYWxsIGRvY3VtZW50ZWQgQVBJIHJvdXRlcy5cbiAqL1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfQVBJX1JPVVRFX1BSRUZJWCA9ICcvdjEnXG5cbi8qKlxuICogRGVmYXVsdCBudW1iZXIgb2YgcmV0cnkgYXR0ZW1wdHMgZm9yIGZhaWxlZCByZXF1ZXN0cy5cbiAqL1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfQUxMT1dFRF9SRVRSSUVTID0gMlxuXG4vKipcbiAqIERlZmF1bHQgcmF0ZSBsaW1pdCAocmVxdWVzdHMgcGVyIHNlY29uZCkgZm9yIFR1cnZvIEFQSSBjYWxscy5cbiAqL1xuZXhwb3J0IGNvbnN0IERFRkFVTFRfUkFURV9MSU1JVF9SUFMgPSAxXG4iXX0=
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
export { getShipmentTracking } from './shipmentTracking';
|
|
2
|
+
export { TurvoInternalApi } from './api/turvoInternalApi';
|
|
3
|
+
export { TurvoPublicApi } from './api/turvoPublicApi';
|
|
4
|
+
export { clearCachedAccessToken, TurvoClient } from './client/turvoClient';
|
|
5
|
+
export * from './types';
|
|
6
|
+
export { DEFAULT_ALLOWED_RETRIES, DEFAULT_API_ROUTE_PREFIX, DEFAULT_RATE_LIMIT_RPS, TURVO_SECRET_PATH, } from './constants';
|
package/lib/esm/index.js
ADDED
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
// Main tracking function
|
|
2
|
+
export { getShipmentTracking } from './shipmentTracking';
|
|
3
|
+
// Client classes (for advanced usage)
|
|
4
|
+
export { TurvoInternalApi } from './api/turvoInternalApi';
|
|
5
|
+
export { TurvoPublicApi } from './api/turvoPublicApi';
|
|
6
|
+
export { clearCachedAccessToken, TurvoClient } from './client/turvoClient';
|
|
7
|
+
// All types
|
|
8
|
+
export * from './types';
|
|
9
|
+
// Constants
|
|
10
|
+
export { DEFAULT_ALLOWED_RETRIES, DEFAULT_API_ROUTE_PREFIX, DEFAULT_RATE_LIMIT_RPS, TURVO_SECRET_PATH, } from './constants';
|
|
11
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEseUJBQXlCO0FBQ3pCLE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLG9CQUFvQixDQUFBO0FBRXhELHNDQUFzQztBQUN0QyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQTtBQUN6RCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sc0JBQXNCLENBQUE7QUFDckQsT0FBTyxFQUFFLHNCQUFzQixFQUFFLFdBQVcsRUFBRSxNQUFNLHNCQUFzQixDQUFBO0FBRTFFLFlBQVk7QUFDWixjQUFjLFNBQVMsQ0FBQTtBQUV2QixZQUFZO0FBQ1osT0FBTyxFQUNMLHVCQUF1QixFQUN2Qix3QkFBd0IsRUFDeEIsc0JBQXNCLEVBQ3RCLGlCQUFpQixHQUNsQixNQUFNLGFBQWEsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8vIE1haW4gdHJhY2tpbmcgZnVuY3Rpb25cbmV4cG9ydCB7IGdldFNoaXBtZW50VHJhY2tpbmcgfSBmcm9tICcuL3NoaXBtZW50VHJhY2tpbmcnXG5cbi8vIENsaWVudCBjbGFzc2VzIChmb3IgYWR2YW5jZWQgdXNhZ2UpXG5leHBvcnQgeyBUdXJ2b0ludGVybmFsQXBpIH0gZnJvbSAnLi9hcGkvdHVydm9JbnRlcm5hbEFwaSdcbmV4cG9ydCB7IFR1cnZvUHVibGljQXBpIH0gZnJvbSAnLi9hcGkvdHVydm9QdWJsaWNBcGknXG5leHBvcnQgeyBjbGVhckNhY2hlZEFjY2Vzc1Rva2VuLCBUdXJ2b0NsaWVudCB9IGZyb20gJy4vY2xpZW50L3R1cnZvQ2xpZW50J1xuXG4vLyBBbGwgdHlwZXNcbmV4cG9ydCAqIGZyb20gJy4vdHlwZXMnXG5cbi8vIENvbnN0YW50c1xuZXhwb3J0IHtcbiAgREVGQVVMVF9BTExPV0VEX1JFVFJJRVMsXG4gIERFRkFVTFRfQVBJX1JPVVRFX1BSRUZJWCxcbiAgREVGQVVMVF9SQVRFX0xJTUlUX1JQUyxcbiAgVFVSVk9fU0VDUkVUX1BBVEgsXG59IGZyb20gJy4vY29uc3RhbnRzJ1xuIl19
|
|
@@ -0,0 +1,22 @@
|
|
|
1
|
+
import { GetTrackingOptions, ShipmentTracking } from '../types/tracking';
|
|
2
|
+
/**
|
|
3
|
+
* Main function to fetch shipment tracking data from Turvo.
|
|
4
|
+
* Combines public API (shipment details) with internal API (GPS tracking).
|
|
5
|
+
*
|
|
6
|
+
* @param shipmentId - The Turvo shipment ID to track
|
|
7
|
+
* @param options - Optional configuration for tracking request
|
|
8
|
+
* @returns Complete shipment tracking information
|
|
9
|
+
*
|
|
10
|
+
* @example
|
|
11
|
+
* ```typescript
|
|
12
|
+
* import { getShipmentTracking } from '@veho/turvo-sdk'
|
|
13
|
+
*
|
|
14
|
+
* const tracking = await getShipmentTracking('12345', {
|
|
15
|
+
* includeGps: true,
|
|
16
|
+
* secretPath: '/turvo/api'
|
|
17
|
+
* })
|
|
18
|
+
*
|
|
19
|
+
* console.log(tracking.status, tracking.etaUtc, tracking.isLate)
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
export declare function getShipmentTracking(shipmentId: string, options?: GetTrackingOptions): Promise<ShipmentTracking>;
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
import { TurvoInternalApi } from '../api/turvoInternalApi';
|
|
2
|
+
import { TurvoPublicApi } from '../api/turvoPublicApi';
|
|
3
|
+
import { TurvoClient } from '../client/turvoClient';
|
|
4
|
+
import { TrackingService } from './trackingService';
|
|
5
|
+
/**
|
|
6
|
+
* Main function to fetch shipment tracking data from Turvo.
|
|
7
|
+
* Combines public API (shipment details) with internal API (GPS tracking).
|
|
8
|
+
*
|
|
9
|
+
* @param shipmentId - The Turvo shipment ID to track
|
|
10
|
+
* @param options - Optional configuration for tracking request
|
|
11
|
+
* @returns Complete shipment tracking information
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* ```typescript
|
|
15
|
+
* import { getShipmentTracking } from '@veho/turvo-sdk'
|
|
16
|
+
*
|
|
17
|
+
* const tracking = await getShipmentTracking('12345', {
|
|
18
|
+
* includeGps: true,
|
|
19
|
+
* secretPath: '/turvo/api'
|
|
20
|
+
* })
|
|
21
|
+
*
|
|
22
|
+
* console.log(tracking.status, tracking.etaUtc, tracking.isLate)
|
|
23
|
+
* ```
|
|
24
|
+
*/
|
|
25
|
+
export async function getShipmentTracking(shipmentId, options = {}) {
|
|
26
|
+
// Create client with optional custom secret path
|
|
27
|
+
const client = new TurvoClient(options.secretPath);
|
|
28
|
+
// Create API instances
|
|
29
|
+
const publicApi = new TurvoPublicApi(client);
|
|
30
|
+
const internalApi = new TurvoInternalApi(client);
|
|
31
|
+
// Create tracking service
|
|
32
|
+
const trackingService = new TrackingService(publicApi, internalApi);
|
|
33
|
+
// Fetch and return tracking data
|
|
34
|
+
return trackingService.getTracking(shipmentId, options);
|
|
35
|
+
}
|
|
36
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2hpcG1lbnRUcmFja2luZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQTtBQUMxRCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sdUJBQXVCLENBQUE7QUFDdEQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLHVCQUF1QixDQUFBO0FBRW5ELE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxtQkFBbUIsQ0FBQTtBQUVuRDs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQW1CRztBQUNILE1BQU0sQ0FBQyxLQUFLLFVBQVUsbUJBQW1CLENBQ3ZDLFVBQWtCLEVBQ2xCLFVBQThCLEVBQUU7SUFFaEMsaURBQWlEO0lBQ2pELE1BQU0sTUFBTSxHQUFHLElBQUksV0FBVyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQTtJQUVsRCx1QkFBdUI7SUFDdkIsTUFBTSxTQUFTLEdBQUcsSUFBSSxjQUFjLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDNUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxnQkFBZ0IsQ0FBQyxNQUFNLENBQUMsQ0FBQTtJQUVoRCwwQkFBMEI7SUFDMUIsTUFBTSxlQUFlLEdBQUcsSUFBSSxlQUFlLENBQUMsU0FBUyxFQUFFLFdBQVcsQ0FBQyxDQUFBO0lBRW5FLGlDQUFpQztJQUNqQyxPQUFPLGVBQWUsQ0FBQyxXQUFXLENBQUMsVUFBVSxFQUFFLE9BQU8sQ0FBQyxDQUFBO0FBQ3pELENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBUdXJ2b0ludGVybmFsQXBpIH0gZnJvbSAnLi4vYXBpL3R1cnZvSW50ZXJuYWxBcGknXG5pbXBvcnQgeyBUdXJ2b1B1YmxpY0FwaSB9IGZyb20gJy4uL2FwaS90dXJ2b1B1YmxpY0FwaSdcbmltcG9ydCB7IFR1cnZvQ2xpZW50IH0gZnJvbSAnLi4vY2xpZW50L3R1cnZvQ2xpZW50J1xuaW1wb3J0IHsgR2V0VHJhY2tpbmdPcHRpb25zLCBTaGlwbWVudFRyYWNraW5nIH0gZnJvbSAnLi4vdHlwZXMvdHJhY2tpbmcnXG5pbXBvcnQgeyBUcmFja2luZ1NlcnZpY2UgfSBmcm9tICcuL3RyYWNraW5nU2VydmljZSdcblxuLyoqXG4gKiBNYWluIGZ1bmN0aW9uIHRvIGZldGNoIHNoaXBtZW50IHRyYWNraW5nIGRhdGEgZnJvbSBUdXJ2by5cbiAqIENvbWJpbmVzIHB1YmxpYyBBUEkgKHNoaXBtZW50IGRldGFpbHMpIHdpdGggaW50ZXJuYWwgQVBJIChHUFMgdHJhY2tpbmcpLlxuICpcbiAqIEBwYXJhbSBzaGlwbWVudElkIC0gVGhlIFR1cnZvIHNoaXBtZW50IElEIHRvIHRyYWNrXG4gKiBAcGFyYW0gb3B0aW9ucyAtIE9wdGlvbmFsIGNvbmZpZ3VyYXRpb24gZm9yIHRyYWNraW5nIHJlcXVlc3RcbiAqIEByZXR1cm5zIENvbXBsZXRlIHNoaXBtZW50IHRyYWNraW5nIGluZm9ybWF0aW9uXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGltcG9ydCB7IGdldFNoaXBtZW50VHJhY2tpbmcgfSBmcm9tICdAdmVoby90dXJ2by1zZGsnXG4gKlxuICogY29uc3QgdHJhY2tpbmcgPSBhd2FpdCBnZXRTaGlwbWVudFRyYWNraW5nKCcxMjM0NScsIHtcbiAqICAgaW5jbHVkZUdwczogdHJ1ZSxcbiAqICAgc2VjcmV0UGF0aDogJy90dXJ2by9hcGknXG4gKiB9KVxuICpcbiAqIGNvbnNvbGUubG9nKHRyYWNraW5nLnN0YXR1cywgdHJhY2tpbmcuZXRhVXRjLCB0cmFja2luZy5pc0xhdGUpXG4gKiBgYGBcbiAqL1xuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFNoaXBtZW50VHJhY2tpbmcoXG4gIHNoaXBtZW50SWQ6IHN0cmluZyxcbiAgb3B0aW9uczogR2V0VHJhY2tpbmdPcHRpb25zID0ge31cbik6IFByb21pc2U8U2hpcG1lbnRUcmFja2luZz4ge1xuICAvLyBDcmVhdGUgY2xpZW50IHdpdGggb3B0aW9uYWwgY3VzdG9tIHNlY3JldCBwYXRoXG4gIGNvbnN0IGNsaWVudCA9IG5ldyBUdXJ2b0NsaWVudChvcHRpb25zLnNlY3JldFBhdGgpXG5cbiAgLy8gQ3JlYXRlIEFQSSBpbnN0YW5jZXNcbiAgY29uc3QgcHVibGljQXBpID0gbmV3IFR1cnZvUHVibGljQXBpKGNsaWVudClcbiAgY29uc3QgaW50ZXJuYWxBcGkgPSBuZXcgVHVydm9JbnRlcm5hbEFwaShjbGllbnQpXG5cbiAgLy8gQ3JlYXRlIHRyYWNraW5nIHNlcnZpY2VcbiAgY29uc3QgdHJhY2tpbmdTZXJ2aWNlID0gbmV3IFRyYWNraW5nU2VydmljZShwdWJsaWNBcGksIGludGVybmFsQXBpKVxuXG4gIC8vIEZldGNoIGFuZCByZXR1cm4gdHJhY2tpbmcgZGF0YVxuICByZXR1cm4gdHJhY2tpbmdTZXJ2aWNlLmdldFRyYWNraW5nKHNoaXBtZW50SWQsIG9wdGlvbnMpXG59XG4iXX0=
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { TurvoInternalApi } from '../api/turvoInternalApi';
|
|
2
|
+
import { TurvoPublicApi } from '../api/turvoPublicApi';
|
|
3
|
+
import { GetTrackingOptions, ShipmentTracking } from '../types/tracking';
|
|
4
|
+
/**
|
|
5
|
+
* TrackingService orchestrates calls to public and internal APIs
|
|
6
|
+
* to provide a unified shipment tracking interface.
|
|
7
|
+
*/
|
|
8
|
+
export declare class TrackingService {
|
|
9
|
+
private publicApi;
|
|
10
|
+
private internalApi;
|
|
11
|
+
constructor(publicApi: TurvoPublicApi, internalApi: TurvoInternalApi);
|
|
12
|
+
/**
|
|
13
|
+
* Main entry point - combines all API data to produce ShipmentTracking
|
|
14
|
+
*
|
|
15
|
+
* @param shipmentId - The shipment ID to track
|
|
16
|
+
* @param options - Tracking options
|
|
17
|
+
* @returns Complete shipment tracking information
|
|
18
|
+
*/
|
|
19
|
+
getTracking(shipmentId: string, options?: GetTrackingOptions): Promise<ShipmentTracking>;
|
|
20
|
+
/**
|
|
21
|
+
* Transform raw Turvo API responses into clean ShipmentTracking type
|
|
22
|
+
*
|
|
23
|
+
*/
|
|
24
|
+
private transformToTrackingDetails;
|
|
25
|
+
}
|
|
@@ -0,0 +1,81 @@
|
|
|
1
|
+
import { TurvoNotFoundError } from '../types/errors';
|
|
2
|
+
/**
|
|
3
|
+
* TrackingService orchestrates calls to public and internal APIs
|
|
4
|
+
* to provide a unified shipment tracking interface.
|
|
5
|
+
*/
|
|
6
|
+
export class TrackingService {
|
|
7
|
+
publicApi;
|
|
8
|
+
internalApi;
|
|
9
|
+
constructor(publicApi, internalApi) {
|
|
10
|
+
this.publicApi = publicApi;
|
|
11
|
+
this.internalApi = internalApi;
|
|
12
|
+
}
|
|
13
|
+
/**
|
|
14
|
+
* Main entry point - combines all API data to produce ShipmentTracking
|
|
15
|
+
*
|
|
16
|
+
* @param shipmentId - The shipment ID to track
|
|
17
|
+
* @param options - Tracking options
|
|
18
|
+
* @returns Complete shipment tracking information
|
|
19
|
+
*/
|
|
20
|
+
async getTracking(shipmentId, options = {}) {
|
|
21
|
+
// 1. Fetch public API data (required)
|
|
22
|
+
const shipmentResult = await this.publicApi.getShipment({ shipmentId: parseInt(shipmentId) });
|
|
23
|
+
if (!shipmentResult.details) {
|
|
24
|
+
throw new TurvoNotFoundError(shipmentId);
|
|
25
|
+
}
|
|
26
|
+
const shipment = shipmentResult.details;
|
|
27
|
+
// 2. Fetch internal API data (optional, may fail)
|
|
28
|
+
let locationUpdates = null;
|
|
29
|
+
let internalApiFailed = false;
|
|
30
|
+
if (options.includeGps !== false) {
|
|
31
|
+
try {
|
|
32
|
+
locationUpdates = await this.internalApi.getLocationUpdates(shipmentId);
|
|
33
|
+
}
|
|
34
|
+
catch (error) {
|
|
35
|
+
// Log but don't fail - internal API is optional
|
|
36
|
+
internalApiFailed = true;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
// 3. Transform to clean response
|
|
40
|
+
return this.transformToTrackingDetails(shipment, locationUpdates, internalApiFailed);
|
|
41
|
+
}
|
|
42
|
+
/**
|
|
43
|
+
* Transform raw Turvo API responses into clean ShipmentTracking type
|
|
44
|
+
*
|
|
45
|
+
*/
|
|
46
|
+
transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
|
|
47
|
+
const shipmentData = shipment;
|
|
48
|
+
return {
|
|
49
|
+
// Identifiers
|
|
50
|
+
shipmentId: shipmentData.id?.toString() || '',
|
|
51
|
+
loadReferenceNumber: shipmentData.customId || '',
|
|
52
|
+
// Status
|
|
53
|
+
status: shipmentData.status?.code?.value || 'Unknown',
|
|
54
|
+
lateBy: shipment.status.runningLate?.lateDurationString ?? null,
|
|
55
|
+
isLate: !!shipment.status.runningLate,
|
|
56
|
+
// Current location (if tracking available)
|
|
57
|
+
currentLocation: shipment.status.location ?? null,
|
|
58
|
+
hasGpsTracking: locationUpdates !== null && !internalApiFailed,
|
|
59
|
+
// Progress
|
|
60
|
+
completedStops: 0, // TODO: Count from shipmentData.globalRoute
|
|
61
|
+
totalStops: shipmentData.globalRoute?.length || 0, // TODO: parse from shipmentData.globalRoute
|
|
62
|
+
// Stops (ordered)
|
|
63
|
+
stops: [], // TODO: implment global route parsing
|
|
64
|
+
// ETA & Distance
|
|
65
|
+
etaUtc: shipment.status.attributes?.etaValUtc ?? null,
|
|
66
|
+
etaTimezone: shipment.status.attributes?.next_eta_cal_val_timezone ?? null,
|
|
67
|
+
milesRemaining: shipment.status.location?.nextMiles ?? null,
|
|
68
|
+
totalMiles: null, // TODO: Derive from global route -- find current stop, and figure out total distance leading to it
|
|
69
|
+
// GPS Pings (only populated if includeGps: true and hasGpsTracking)
|
|
70
|
+
locationUpdates: locationUpdates || null, // TODO: implment with scraper
|
|
71
|
+
pingCount: locationUpdates?.length || null, // TODO: implement with scraper
|
|
72
|
+
lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp ||
|
|
73
|
+
null, // TODO: implement with scraper
|
|
74
|
+
// Shipment Attributes
|
|
75
|
+
laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,
|
|
76
|
+
managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,
|
|
77
|
+
carrier: shipmentData.carrierOrder?.[0]?.carrier?.name || null,
|
|
78
|
+
};
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
//# 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"]}
|