@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.
Files changed (57) hide show
  1. package/README.md +473 -0
  2. package/index.ts +1 -0
  3. package/lib/cjs/api/turvoInternalApi.d.ts +23 -0
  4. package/lib/cjs/api/turvoInternalApi.js +36 -0
  5. package/lib/cjs/api/turvoPublicApi.d.ts +99 -0
  6. package/lib/cjs/api/turvoPublicApi.js +103 -0
  7. package/lib/cjs/client/turvoClient.d.ts +49 -0
  8. package/lib/cjs/client/turvoClient.js +180 -0
  9. package/lib/cjs/constants.d.ts +29 -0
  10. package/lib/cjs/constants.js +33 -0
  11. package/lib/cjs/index.d.ts +6 -0
  12. package/lib/cjs/index.js +37 -0
  13. package/lib/cjs/shipmentTracking/index.d.ts +22 -0
  14. package/lib/cjs/shipmentTracking/index.js +39 -0
  15. package/lib/cjs/shipmentTracking/trackingService.d.ts +25 -0
  16. package/lib/cjs/shipmentTracking/trackingService.js +85 -0
  17. package/lib/cjs/types/common.d.ts +64 -0
  18. package/lib/cjs/types/common.js +27 -0
  19. package/lib/cjs/types/config.d.ts +13 -0
  20. package/lib/cjs/types/config.js +3 -0
  21. package/lib/cjs/types/errors.d.ts +35 -0
  22. package/lib/cjs/types/errors.js +63 -0
  23. package/lib/cjs/types/index.d.ts +5 -0
  24. package/lib/cjs/types/index.js +27 -0
  25. package/lib/cjs/types/shipment.d.ts +379 -0
  26. package/lib/cjs/types/shipment.js +46 -0
  27. package/lib/cjs/types/tracking.d.ts +65 -0
  28. package/lib/cjs/types/tracking.js +3 -0
  29. package/lib/esm/api/turvoInternalApi.d.ts +23 -0
  30. package/lib/esm/api/turvoInternalApi.js +32 -0
  31. package/lib/esm/api/turvoPublicApi.d.ts +99 -0
  32. package/lib/esm/api/turvoPublicApi.js +99 -0
  33. package/lib/esm/client/turvoClient.d.ts +49 -0
  34. package/lib/esm/client/turvoClient.js +172 -0
  35. package/lib/esm/constants.d.ts +29 -0
  36. package/lib/esm/constants.js +30 -0
  37. package/lib/esm/index.d.ts +6 -0
  38. package/lib/esm/index.js +11 -0
  39. package/lib/esm/shipmentTracking/index.d.ts +22 -0
  40. package/lib/esm/shipmentTracking/index.js +36 -0
  41. package/lib/esm/shipmentTracking/trackingService.d.ts +25 -0
  42. package/lib/esm/shipmentTracking/trackingService.js +81 -0
  43. package/lib/esm/types/common.d.ts +64 -0
  44. package/lib/esm/types/common.js +23 -0
  45. package/lib/esm/types/config.d.ts +13 -0
  46. package/lib/esm/types/config.js +2 -0
  47. package/lib/esm/types/errors.d.ts +35 -0
  48. package/lib/esm/types/errors.js +55 -0
  49. package/lib/esm/types/index.d.ts +5 -0
  50. package/lib/esm/types/index.js +11 -0
  51. package/lib/esm/types/shipment.d.ts +379 -0
  52. package/lib/esm/types/shipment.js +43 -0
  53. package/lib/esm/types/tracking.d.ts +65 -0
  54. package/lib/esm/types/tracking.js +2 -0
  55. package/lib/tsconfig.cjs.tsbuildinfo +1 -0
  56. package/lib/tsconfig.esm.tsbuildinfo +1 -0
  57. 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":"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","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';
@@ -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"]}