@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,103 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.TurvoPublicApi = void 0;
4
+ const common_1 = require("../types/common");
5
+ /**
6
+ * TurvoPublicApi provides access to documented Turvo API endpoints.
7
+ * All shipment management operations (create, update, cancel, query) go through this class.
8
+ */
9
+ class TurvoPublicApi {
10
+ client;
11
+ constructor(client) {
12
+ this.client = client;
13
+ }
14
+ /**
15
+ * GET /v1/shipments/{id}
16
+ * Returns shipment details, stops, status
17
+ */
18
+ async getShipment(params) {
19
+ const { shipmentId } = params;
20
+ const httpClient = await this.client.getAuthenticatedClient();
21
+ return httpClient.sendRequest(`/shipments/${shipmentId}`, common_1.TurvoApiMethod.GET, {});
22
+ }
23
+ /**
24
+ * POST /v1/shipments
25
+ * Upload properly formatted shipment to Turvo
26
+ */
27
+ async uploadShipment(params) {
28
+ const { shipment } = params;
29
+ // Note: vehoLoadId is kept in the params for logging purposes by consumers
30
+ const httpClient = await this.client.getAuthenticatedClient();
31
+ return httpClient.sendRequest('/shipments', common_1.TurvoApiMethod.POST, {
32
+ body: shipment,
33
+ });
34
+ }
35
+ /**
36
+ * PUT /v1/tags/attach/shipment/{id}
37
+ * Associate a list of tags with a shipment by ID
38
+ */
39
+ async associateTagsToShipment(params) {
40
+ const { turvoShipmentId, tags } = params;
41
+ const httpClient = await this.client.getAuthenticatedClient();
42
+ return httpClient.sendRequest(`/tags/attach/shipment/${turvoShipmentId}`, common_1.TurvoApiMethod.PUT, {
43
+ body: {
44
+ tagNames: tags,
45
+ },
46
+ });
47
+ }
48
+ /**
49
+ * PUT /v1/shipments/status/{id}
50
+ * Set shipment status to Cancelled in Turvo
51
+ */
52
+ async cancelShipment(params) {
53
+ const { turvoShipmentId } = params;
54
+ const httpClient = await this.client.getAuthenticatedClient();
55
+ const body = {
56
+ id: turvoShipmentId,
57
+ status: {
58
+ code: { key: '2113', value: 'Canceled' },
59
+ },
60
+ };
61
+ return httpClient.sendRequest(`/shipments/status/${turvoShipmentId}`, common_1.TurvoApiMethod.PUT, {
62
+ body,
63
+ });
64
+ }
65
+ /**
66
+ * PUT /v1/shipments/status/{id}
67
+ * Update shipment status at a specific stop
68
+ */
69
+ async updateShipmentStatus(params) {
70
+ const { turvoShipmentId, turvoStopId, turvoStatusCode, statusDate, statusTimezone } = params;
71
+ const httpClient = await this.client.getAuthenticatedClient();
72
+ const body = {
73
+ id: turvoShipmentId,
74
+ status: {
75
+ globalShipLocationId: turvoStopId,
76
+ code: turvoStatusCode.code,
77
+ timezone: statusTimezone,
78
+ statusDate: {
79
+ date: statusDate,
80
+ timezone: statusTimezone,
81
+ },
82
+ },
83
+ };
84
+ return httpClient.sendRequest(`/shipments/status/${turvoShipmentId}`, common_1.TurvoApiMethod.PUT, { body });
85
+ }
86
+ /**
87
+ * GET /v1/shipments/list
88
+ * Query shipments from Turvo with pagination support
89
+ */
90
+ async filterShipments(params) {
91
+ const { turvoLocationId, pickupDateStart, start = 0 } = params;
92
+ const httpClient = await this.client.getAuthenticatedClient();
93
+ return httpClient.sendRequest('/shipments/list', common_1.TurvoApiMethod.GET, {
94
+ query: {
95
+ 'locationId[eq]': turvoLocationId.toString(),
96
+ 'pickupDate[gte]': pickupDateStart,
97
+ start: start.toString(),
98
+ },
99
+ });
100
+ }
101
+ }
102
+ exports.TurvoPublicApi = TurvoPublicApi;
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"turvoPublicApi.js","sourceRoot":"","sources":["../../../src/api/turvoPublicApi.ts"],"names":[],"mappings":"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","sourcesContent":["import { TurvoClient } from '../client/turvoClient'\nimport {\n  TurvoApiMethod,\n  TurvoApiResult,\n  TurvoFilterShipmentsResultItem,\n  TurvoLookup,\n  TurvoPagedResult,\n} from '../types/common'\nimport {\n  TurvoAssociateTagsApiResult,\n  TurvoCancelShipmentApiResult,\n  TurvoCreateShipmentApiResult,\n  TurvoShipment,\n  TurvoShipmentStatus,\n  TurvoUpdateShipmentStatusApiResult,\n} from '../types/shipment'\n\n/**\n * Parameters for uploading a shipment to Turvo\n */\nexport interface UploadShipmentParams {\n  /** The formatted shipment to upload */\n  shipment: TurvoShipment\n  /** The Load ID, solely for logging purposes */\n  vehoLoadId: string\n}\n\n/**\n * Parameters for associating tags to a shipment\n */\nexport interface AssociateTagsParams {\n  /** The shipment ID to associate tags with */\n  turvoShipmentId: number\n  /** List of tag names to assign */\n  tags: string[]\n}\n\n/**\n * Parameters for cancelling a shipment\n */\nexport interface CancelShipmentParams {\n  /** The Turvo system Shipment ID to cancel */\n  turvoShipmentId: number\n}\n\n/**\n * Parameters for updating shipment status\n */\nexport interface UpdateShipmentStatusParams {\n  /** The Turvo shipment ID */\n  turvoShipmentId: number\n  /** The Turvo stop ID */\n  turvoStopId: number\n  /** The status code to set */\n  turvoStatusCode: TurvoShipmentStatus\n  /** The status date in ISO format */\n  statusDate: string\n  /** The timezone for the status date */\n  statusTimezone: string\n}\n\n/**\n * Parameters for retrieving a shipment\n */\nexport interface GetShipmentParams {\n  /** The shipment ID to retrieve */\n  shipmentId: number\n}\n\n/**\n * Parameters for filtering shipments\n */\nexport interface FilterShipmentsParams {\n  /** Turvo location ID to filter by */\n  turvoLocationId: number\n  /** Pickup date start filter (ISO format) */\n  pickupDateStart: string\n  /** Optional pagination start offset */\n  start?: number\n}\n\n/**\n * TurvoPublicApi provides access to documented Turvo API endpoints.\n * All shipment management operations (create, update, cancel, query) go through this class.\n */\nexport class TurvoPublicApi {\n  constructor(private client: TurvoClient) {}\n\n  /**\n   * GET /v1/shipments/{id}\n   * Returns shipment details, stops, status\n   */\n  async getShipment(params: GetShipmentParams): Promise<TurvoApiResult<TurvoShipment>> {\n    const { shipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoShipment>(`/shipments/${shipmentId}`, TurvoApiMethod.GET, {})\n  }\n\n  /**\n   * POST /v1/shipments\n   * Upload properly formatted shipment to Turvo\n   */\n  async uploadShipment(params: UploadShipmentParams): Promise<TurvoApiResult<TurvoCreateShipmentApiResult>> {\n    const { shipment } = params\n    // Note: vehoLoadId is kept in the params for logging purposes by consumers\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoCreateShipmentApiResult>('/shipments', TurvoApiMethod.POST, {\n      body: shipment,\n    })\n  }\n\n  /**\n   * PUT /v1/tags/attach/shipment/{id}\n   * Associate a list of tags with a shipment by ID\n   */\n  async associateTagsToShipment(params: AssociateTagsParams): Promise<TurvoApiResult<TurvoAssociateTagsApiResult>> {\n    const { turvoShipmentId, tags } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoAssociateTagsApiResult>(\n      `/tags/attach/shipment/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body: {\n          tagNames: tags,\n        },\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Set shipment status to Cancelled in Turvo\n   */\n  async cancelShipment(params: CancelShipmentParams): Promise<TurvoApiResult<TurvoCancelShipmentApiResult>> {\n    const { turvoShipmentId } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        code: { key: '2113', value: 'Canceled' } satisfies TurvoLookup,\n      },\n    }\n\n    return httpClient.sendRequest<TurvoCancelShipmentApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      {\n        body,\n      }\n    )\n  }\n\n  /**\n   * PUT /v1/shipments/status/{id}\n   * Update shipment status at a specific stop\n   */\n  async updateShipmentStatus(\n    params: UpdateShipmentStatusParams\n  ): Promise<TurvoApiResult<TurvoUpdateShipmentStatusApiResult>> {\n    const { turvoShipmentId, turvoStopId, turvoStatusCode, statusDate, statusTimezone } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    const body = {\n      id: turvoShipmentId,\n      status: {\n        globalShipLocationId: turvoStopId,\n        code: turvoStatusCode.code,\n        timezone: statusTimezone,\n        statusDate: {\n          date: statusDate,\n          timezone: statusTimezone,\n        },\n      },\n    }\n\n    return httpClient.sendRequest<TurvoUpdateShipmentStatusApiResult>(\n      `/shipments/status/${turvoShipmentId}`,\n      TurvoApiMethod.PUT,\n      { body }\n    )\n  }\n\n  /**\n   * GET /v1/shipments/list\n   * Query shipments from Turvo with pagination support\n   */\n  async filterShipments(\n    params: FilterShipmentsParams\n  ): Promise<TurvoApiResult<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>> {\n    const { turvoLocationId, pickupDateStart, start = 0 } = params\n    const httpClient = await this.client.getAuthenticatedClient()\n\n    return httpClient.sendRequest<TurvoPagedResult<'shipments', TurvoFilterShipmentsResultItem>>(\n      '/shipments/list',\n      TurvoApiMethod.GET,\n      {\n        query: {\n          'locationId[eq]': turvoLocationId.toString(),\n          'pickupDate[gte]': pickupDateStart,\n          start: start.toString(),\n        },\n      }\n    )\n  }\n}\n"]}
@@ -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,180 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.TurvoClient = exports.clearCachedAccessToken = void 0;
7
+ const lambda_params_secrets_1 = require("lambda-params-secrets");
8
+ const node_fetch_1 = __importDefault(require("node-fetch"));
9
+ const p_retry_1 = __importDefault(require("p-retry"));
10
+ const p_throttle_1 = __importDefault(require("p-throttle"));
11
+ const constants_1 = require("../constants");
12
+ const common_1 = require("../types/common");
13
+ const errors_1 = require("../types/errors");
14
+ let secretClient = null;
15
+ /**
16
+ * Get or create the secrets client instance
17
+ */
18
+ const getSecretClientInstance = () => {
19
+ if (!secretClient) {
20
+ secretClient = new lambda_params_secrets_1.Client();
21
+ }
22
+ return secretClient;
23
+ };
24
+ /**
25
+ * Fetch Turvo API credentials from AWS Secrets Manager
26
+ */
27
+ const getTurvoSecrets = async (secretsPath = constants_1.TURVO_SECRET_PATH) => {
28
+ const fetchedSecret = await getSecretClientInstance().stringSecret(secretsPath);
29
+ if (!fetchedSecret) {
30
+ throw new errors_1.NoSecretError(secretsPath);
31
+ }
32
+ return JSON.parse(fetchedSecret);
33
+ };
34
+ /**
35
+ * Cached access token to avoid repeated OAuth requests
36
+ */
37
+ let cachedAccessToken = null;
38
+ let tokenExpiry = null;
39
+ /**
40
+ * Fetch an access token from Turvo using OAuth2 password grant
41
+ */
42
+ const getTurvoAccessToken = async (credentials, apiRoutePrefix = constants_1.DEFAULT_API_ROUTE_PREFIX, allowedRetries = constants_1.DEFAULT_ALLOWED_RETRIES) => {
43
+ // Check if we have a valid cached token
44
+ if (cachedAccessToken && tokenExpiry && tokenExpiry > new Date()) {
45
+ return cachedAccessToken;
46
+ }
47
+ const fullUrl = `${credentials.urlBase + apiRoutePrefix}/oauth/token`;
48
+ return (0, p_retry_1.default)(async () => {
49
+ const url = new URL(fullUrl);
50
+ url.searchParams.set('client_id', 'publicapi');
51
+ url.searchParams.set('client_secret', 'secret');
52
+ const response = await (0, node_fetch_1.default)(url, {
53
+ method: 'POST',
54
+ headers: {
55
+ 'Content-Type': 'application/json',
56
+ 'x-api-key': credentials.apiKey,
57
+ },
58
+ body: JSON.stringify({
59
+ grant_type: 'password',
60
+ scope: 'read+trust+write',
61
+ username: credentials.username,
62
+ password: credentials.password,
63
+ type: 'business',
64
+ }),
65
+ });
66
+ if (!response.ok) {
67
+ throw new errors_1.TurvoAuthError(`Failed to authenticate: ${response.status} ${response.statusText}`);
68
+ }
69
+ const data = (await response.json());
70
+ const { access_token, expires_in } = data;
71
+ if (!access_token) {
72
+ throw new errors_1.TurvoAuthError('No access token returned from Turvo');
73
+ }
74
+ cachedAccessToken = access_token;
75
+ // Set token expiry (expires_in is in seconds, default to 1 hour if not provided)
76
+ const expiresInMs = (expires_in || 3600) * 1000;
77
+ tokenExpiry = new Date(Date.now() + expiresInMs - 60000); // Refresh 1 minute early
78
+ return access_token;
79
+ }, { retries: allowedRetries });
80
+ };
81
+ /**
82
+ * Clear the cached access token (useful for testing or when token expires)
83
+ */
84
+ const clearCachedAccessToken = () => {
85
+ cachedAccessToken = null;
86
+ tokenExpiry = null;
87
+ };
88
+ exports.clearCachedAccessToken = clearCachedAccessToken;
89
+ /**
90
+ * Main client for interacting with Turvo APIs.
91
+ * Handles authentication, token management, and HTTP requests.
92
+ *
93
+ * @example
94
+ * ```typescript
95
+ * const client = new TurvoClient()
96
+ * const httpClient = await client.getAuthenticatedClient()
97
+ * const result = await httpClient.sendRequest('/shipments/123', TurvoApiMethod.GET, {})
98
+ * ```
99
+ */
100
+ class TurvoClient {
101
+ secretPath;
102
+ credentials;
103
+ httpClient;
104
+ constructor(secretPath = constants_1.TURVO_SECRET_PATH) {
105
+ this.secretPath = secretPath;
106
+ }
107
+ /**
108
+ * Get authenticated HTTP client.
109
+ * Handles token refresh automatically.
110
+ */
111
+ async getAuthenticatedClient() {
112
+ if (this.httpClient) {
113
+ return this.httpClient;
114
+ }
115
+ // Fetch credentials if not already loaded
116
+ if (!this.credentials) {
117
+ this.credentials = await getTurvoSecrets(this.secretPath);
118
+ }
119
+ // Create HTTP client with rate limiting and retries
120
+ const apiRoutePrefix = constants_1.DEFAULT_API_ROUTE_PREFIX;
121
+ const allowedRetries = constants_1.DEFAULT_ALLOWED_RETRIES;
122
+ const rateLimitRequestsPerSecond = constants_1.DEFAULT_RATE_LIMIT_RPS;
123
+ // Create throttled version of the send function
124
+ const throttle = (0, p_throttle_1.default)({
125
+ limit: rateLimitRequestsPerSecond,
126
+ interval: 1000,
127
+ });
128
+ /**
129
+ * Internal function to send API request (with retry logic)
130
+ */
131
+ const sendToTurvoAPI = async (apiRoute, method, params) => {
132
+ const accessToken = await getTurvoAccessToken(this.credentials, apiRoutePrefix, allowedRetries);
133
+ const { body, headers, query } = params;
134
+ let fullRoute = `${this.credentials.urlBase + apiRoutePrefix + apiRoute}`;
135
+ if (query && Object.keys(query).length > 0) {
136
+ fullRoute = `${fullRoute}?${new URLSearchParams(query)}`;
137
+ }
138
+ return (0, p_retry_1.default)(async () => {
139
+ return (0, node_fetch_1.default)(fullRoute, {
140
+ method,
141
+ headers: {
142
+ Authorization: `Bearer ${accessToken}`,
143
+ 'Content-Type': 'application/json',
144
+ 'x-api-key': this.credentials.apiKey,
145
+ ...headers,
146
+ },
147
+ ...(body && { body: JSON.stringify(body) }),
148
+ }).then(async (res) => {
149
+ const result = await res.json();
150
+ // Handle rate limiting (429)
151
+ if (res.status === 429) {
152
+ throw new common_1.TurvoApiError(res.status.toString(), result.message || 'Rate limited');
153
+ }
154
+ // Check for API-level errors
155
+ if (result.Status !== common_1.TurvoApiResponseStatus.SUCCESS) {
156
+ throw new common_1.TurvoApiError(result.details?.errorCode || 'UNKNOWN', result.details?.errorMessage || 'API error');
157
+ }
158
+ return result;
159
+ });
160
+ }, { retries: allowedRetries });
161
+ };
162
+ // Throttle the send function
163
+ const throttledSend = throttle(sendToTurvoAPI);
164
+ this.httpClient = {
165
+ sendRequest: throttledSend,
166
+ };
167
+ return this.httpClient;
168
+ }
169
+ /**
170
+ * Get raw secrets (for internal API auth which uses different flow)
171
+ */
172
+ async getCredentials() {
173
+ if (!this.credentials) {
174
+ this.credentials = await getTurvoSecrets(this.secretPath);
175
+ }
176
+ return this.credentials;
177
+ }
178
+ }
179
+ exports.TurvoClient = TurvoClient;
180
+ //# 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,33 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.DEFAULT_RATE_LIMIT_RPS = exports.DEFAULT_ALLOWED_RETRIES = exports.DEFAULT_API_ROUTE_PREFIX = exports.TURVO_SECRET_PATH = void 0;
4
+ /**
5
+ * Default secret path for Turvo credentials in AWS Secrets Manager.
6
+ * Export this so consumers can grant IAM permissions.
7
+ *
8
+ * @example
9
+ * ```typescript
10
+ * import { TURVO_SECRET_PATH } from '@veho/turvo-sdk'
11
+ *
12
+ * // In CDK stack
13
+ * myLambda.addToRolePolicy(new iam.PolicyStatement({
14
+ * actions: ['secretsmanager:GetSecretValue'],
15
+ * resources: [`arn:aws:secretsmanager:*:*:secret:${TURVO_SECRET_PATH}*`],
16
+ * }))
17
+ * ```
18
+ */
19
+ exports.TURVO_SECRET_PATH = '/turvo/api';
20
+ /**
21
+ * Default API route prefix for Turvo endpoints.
22
+ * Currently, Turvo uses a common prefix of "/v1" for all documented API routes.
23
+ */
24
+ exports.DEFAULT_API_ROUTE_PREFIX = '/v1';
25
+ /**
26
+ * Default number of retry attempts for failed requests.
27
+ */
28
+ exports.DEFAULT_ALLOWED_RETRIES = 2;
29
+ /**
30
+ * Default rate limit (requests per second) for Turvo API calls.
31
+ */
32
+ exports.DEFAULT_RATE_LIMIT_RPS = 1;
33
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29uc3RhbnRzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2NvbnN0YW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRztBQUNVLFFBQUEsaUJBQWlCLEdBQUcsWUFBWSxDQUFBO0FBRTdDOzs7R0FHRztBQUNVLFFBQUEsd0JBQXdCLEdBQUcsS0FBSyxDQUFBO0FBRTdDOztHQUVHO0FBQ1UsUUFBQSx1QkFBdUIsR0FBRyxDQUFDLENBQUE7QUFFeEM7O0dBRUc7QUFDVSxRQUFBLHNCQUFzQixHQUFHLENBQUMsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogRGVmYXVsdCBzZWNyZXQgcGF0aCBmb3IgVHVydm8gY3JlZGVudGlhbHMgaW4gQVdTIFNlY3JldHMgTWFuYWdlci5cbiAqIEV4cG9ydCB0aGlzIHNvIGNvbnN1bWVycyBjYW4gZ3JhbnQgSUFNIHBlcm1pc3Npb25zLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBUVVJWT19TRUNSRVRfUEFUSCB9IGZyb20gJ0B2ZWhvL3R1cnZvLXNkaydcbiAqXG4gKiAvLyBJbiBDREsgc3RhY2tcbiAqIG15TGFtYmRhLmFkZFRvUm9sZVBvbGljeShuZXcgaWFtLlBvbGljeVN0YXRlbWVudCh7XG4gKiAgIGFjdGlvbnM6IFsnc2VjcmV0c21hbmFnZXI6R2V0U2VjcmV0VmFsdWUnXSxcbiAqICAgcmVzb3VyY2VzOiBbYGFybjphd3M6c2VjcmV0c21hbmFnZXI6KjoqOnNlY3JldDoke1RVUlZPX1NFQ1JFVF9QQVRIfSpgXSxcbiAqIH0pKVxuICogYGBgXG4gKi9cbmV4cG9ydCBjb25zdCBUVVJWT19TRUNSRVRfUEFUSCA9ICcvdHVydm8vYXBpJ1xuXG4vKipcbiAqIERlZmF1bHQgQVBJIHJvdXRlIHByZWZpeCBmb3IgVHVydm8gZW5kcG9pbnRzLlxuICogQ3VycmVudGx5LCBUdXJ2byB1c2VzIGEgY29tbW9uIHByZWZpeCBvZiBcIi92MVwiIGZvciBhbGwgZG9jdW1lbnRlZCBBUEkgcm91dGVzLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9BUElfUk9VVEVfUFJFRklYID0gJy92MSdcblxuLyoqXG4gKiBEZWZhdWx0IG51bWJlciBvZiByZXRyeSBhdHRlbXB0cyBmb3IgZmFpbGVkIHJlcXVlc3RzLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9BTExPV0VEX1JFVFJJRVMgPSAyXG5cbi8qKlxuICogRGVmYXVsdCByYXRlIGxpbWl0IChyZXF1ZXN0cyBwZXIgc2Vjb25kKSBmb3IgVHVydm8gQVBJIGNhbGxzLlxuICovXG5leHBvcnQgY29uc3QgREVGQVVMVF9SQVRFX0xJTUlUX1JQUyA9IDFcbiJdfQ==
@@ -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,37 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __exportStar = (this && this.__exportStar) || function(m, exports) {
14
+ for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
15
+ };
16
+ Object.defineProperty(exports, "__esModule", { value: true });
17
+ exports.TURVO_SECRET_PATH = exports.DEFAULT_RATE_LIMIT_RPS = exports.DEFAULT_API_ROUTE_PREFIX = exports.DEFAULT_ALLOWED_RETRIES = exports.TurvoClient = exports.clearCachedAccessToken = exports.TurvoPublicApi = exports.TurvoInternalApi = exports.getShipmentTracking = void 0;
18
+ // Main tracking function
19
+ var shipmentTracking_1 = require("./shipmentTracking");
20
+ Object.defineProperty(exports, "getShipmentTracking", { enumerable: true, get: function () { return shipmentTracking_1.getShipmentTracking; } });
21
+ // Client classes (for advanced usage)
22
+ var turvoInternalApi_1 = require("./api/turvoInternalApi");
23
+ Object.defineProperty(exports, "TurvoInternalApi", { enumerable: true, get: function () { return turvoInternalApi_1.TurvoInternalApi; } });
24
+ var turvoPublicApi_1 = require("./api/turvoPublicApi");
25
+ Object.defineProperty(exports, "TurvoPublicApi", { enumerable: true, get: function () { return turvoPublicApi_1.TurvoPublicApi; } });
26
+ var turvoClient_1 = require("./client/turvoClient");
27
+ Object.defineProperty(exports, "clearCachedAccessToken", { enumerable: true, get: function () { return turvoClient_1.clearCachedAccessToken; } });
28
+ Object.defineProperty(exports, "TurvoClient", { enumerable: true, get: function () { return turvoClient_1.TurvoClient; } });
29
+ // All types
30
+ __exportStar(require("./types"), exports);
31
+ // Constants
32
+ var constants_1 = require("./constants");
33
+ Object.defineProperty(exports, "DEFAULT_ALLOWED_RETRIES", { enumerable: true, get: function () { return constants_1.DEFAULT_ALLOWED_RETRIES; } });
34
+ Object.defineProperty(exports, "DEFAULT_API_ROUTE_PREFIX", { enumerable: true, get: function () { return constants_1.DEFAULT_API_ROUTE_PREFIX; } });
35
+ Object.defineProperty(exports, "DEFAULT_RATE_LIMIT_RPS", { enumerable: true, get: function () { return constants_1.DEFAULT_RATE_LIMIT_RPS; } });
36
+ Object.defineProperty(exports, "TURVO_SECRET_PATH", { enumerable: true, get: function () { return constants_1.TURVO_SECRET_PATH; } });
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQSx5QkFBeUI7QUFDekIsdURBQXdEO0FBQS9DLHVIQUFBLG1CQUFtQixPQUFBO0FBRTVCLHNDQUFzQztBQUN0QywyREFBeUQ7QUFBaEQsb0hBQUEsZ0JBQWdCLE9BQUE7QUFDekIsdURBQXFEO0FBQTVDLGdIQUFBLGNBQWMsT0FBQTtBQUN2QixvREFBMEU7QUFBakUscUhBQUEsc0JBQXNCLE9BQUE7QUFBRSwwR0FBQSxXQUFXLE9BQUE7QUFFNUMsWUFBWTtBQUNaLDBDQUF1QjtBQUV2QixZQUFZO0FBQ1oseUNBS29CO0FBSmxCLG9IQUFBLHVCQUF1QixPQUFBO0FBQ3ZCLHFIQUFBLHdCQUF3QixPQUFBO0FBQ3hCLG1IQUFBLHNCQUFzQixPQUFBO0FBQ3RCLDhHQUFBLGlCQUFpQixPQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLy8gTWFpbiB0cmFja2luZyBmdW5jdGlvblxuZXhwb3J0IHsgZ2V0U2hpcG1lbnRUcmFja2luZyB9IGZyb20gJy4vc2hpcG1lbnRUcmFja2luZydcblxuLy8gQ2xpZW50IGNsYXNzZXMgKGZvciBhZHZhbmNlZCB1c2FnZSlcbmV4cG9ydCB7IFR1cnZvSW50ZXJuYWxBcGkgfSBmcm9tICcuL2FwaS90dXJ2b0ludGVybmFsQXBpJ1xuZXhwb3J0IHsgVHVydm9QdWJsaWNBcGkgfSBmcm9tICcuL2FwaS90dXJ2b1B1YmxpY0FwaSdcbmV4cG9ydCB7IGNsZWFyQ2FjaGVkQWNjZXNzVG9rZW4sIFR1cnZvQ2xpZW50IH0gZnJvbSAnLi9jbGllbnQvdHVydm9DbGllbnQnXG5cbi8vIEFsbCB0eXBlc1xuZXhwb3J0ICogZnJvbSAnLi90eXBlcydcblxuLy8gQ29uc3RhbnRzXG5leHBvcnQge1xuICBERUZBVUxUX0FMTE9XRURfUkVUUklFUyxcbiAgREVGQVVMVF9BUElfUk9VVEVfUFJFRklYLFxuICBERUZBVUxUX1JBVEVfTElNSVRfUlBTLFxuICBUVVJWT19TRUNSRVRfUEFUSCxcbn0gZnJvbSAnLi9jb25zdGFudHMnXG4iXX0=
@@ -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,39 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.getShipmentTracking = getShipmentTracking;
4
+ const turvoInternalApi_1 = require("../api/turvoInternalApi");
5
+ const turvoPublicApi_1 = require("../api/turvoPublicApi");
6
+ const turvoClient_1 = require("../client/turvoClient");
7
+ const trackingService_1 = require("./trackingService");
8
+ /**
9
+ * Main function to fetch shipment tracking data from Turvo.
10
+ * Combines public API (shipment details) with internal API (GPS tracking).
11
+ *
12
+ * @param shipmentId - The Turvo shipment ID to track
13
+ * @param options - Optional configuration for tracking request
14
+ * @returns Complete shipment tracking information
15
+ *
16
+ * @example
17
+ * ```typescript
18
+ * import { getShipmentTracking } from '@veho/turvo-sdk'
19
+ *
20
+ * const tracking = await getShipmentTracking('12345', {
21
+ * includeGps: true,
22
+ * secretPath: '/turvo/api'
23
+ * })
24
+ *
25
+ * console.log(tracking.status, tracking.etaUtc, tracking.isLate)
26
+ * ```
27
+ */
28
+ async function getShipmentTracking(shipmentId, options = {}) {
29
+ // Create client with optional custom secret path
30
+ const client = new turvoClient_1.TurvoClient(options.secretPath);
31
+ // Create API instances
32
+ const publicApi = new turvoPublicApi_1.TurvoPublicApi(client);
33
+ const internalApi = new turvoInternalApi_1.TurvoInternalApi(client);
34
+ // Create tracking service
35
+ const trackingService = new trackingService_1.TrackingService(publicApi, internalApi);
36
+ // Fetch and return tracking data
37
+ return trackingService.getTracking(shipmentId, options);
38
+ }
39
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvc2hpcG1lbnRUcmFja2luZy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOztBQTBCQSxrREFnQkM7QUExQ0QsOERBQTBEO0FBQzFELDBEQUFzRDtBQUN0RCx1REFBbUQ7QUFFbkQsdURBQW1EO0FBRW5EOzs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBbUJHO0FBQ0ksS0FBSyxVQUFVLG1CQUFtQixDQUN2QyxVQUFrQixFQUNsQixVQUE4QixFQUFFO0lBRWhDLGlEQUFpRDtJQUNqRCxNQUFNLE1BQU0sR0FBRyxJQUFJLHlCQUFXLENBQUMsT0FBTyxDQUFDLFVBQVUsQ0FBQyxDQUFBO0lBRWxELHVCQUF1QjtJQUN2QixNQUFNLFNBQVMsR0FBRyxJQUFJLCtCQUFjLENBQUMsTUFBTSxDQUFDLENBQUE7SUFDNUMsTUFBTSxXQUFXLEdBQUcsSUFBSSxtQ0FBZ0IsQ0FBQyxNQUFNLENBQUMsQ0FBQTtJQUVoRCwwQkFBMEI7SUFDMUIsTUFBTSxlQUFlLEdBQUcsSUFBSSxpQ0FBZSxDQUFDLFNBQVMsRUFBRSxXQUFXLENBQUMsQ0FBQTtJQUVuRSxpQ0FBaUM7SUFDakMsT0FBTyxlQUFlLENBQUMsV0FBVyxDQUFDLFVBQVUsRUFBRSxPQUFPLENBQUMsQ0FBQTtBQUN6RCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgVHVydm9JbnRlcm5hbEFwaSB9IGZyb20gJy4uL2FwaS90dXJ2b0ludGVybmFsQXBpJ1xuaW1wb3J0IHsgVHVydm9QdWJsaWNBcGkgfSBmcm9tICcuLi9hcGkvdHVydm9QdWJsaWNBcGknXG5pbXBvcnQgeyBUdXJ2b0NsaWVudCB9IGZyb20gJy4uL2NsaWVudC90dXJ2b0NsaWVudCdcbmltcG9ydCB7IEdldFRyYWNraW5nT3B0aW9ucywgU2hpcG1lbnRUcmFja2luZyB9IGZyb20gJy4uL3R5cGVzL3RyYWNraW5nJ1xuaW1wb3J0IHsgVHJhY2tpbmdTZXJ2aWNlIH0gZnJvbSAnLi90cmFja2luZ1NlcnZpY2UnXG5cbi8qKlxuICogTWFpbiBmdW5jdGlvbiB0byBmZXRjaCBzaGlwbWVudCB0cmFja2luZyBkYXRhIGZyb20gVHVydm8uXG4gKiBDb21iaW5lcyBwdWJsaWMgQVBJIChzaGlwbWVudCBkZXRhaWxzKSB3aXRoIGludGVybmFsIEFQSSAoR1BTIHRyYWNraW5nKS5cbiAqXG4gKiBAcGFyYW0gc2hpcG1lbnRJZCAtIFRoZSBUdXJ2byBzaGlwbWVudCBJRCB0byB0cmFja1xuICogQHBhcmFtIG9wdGlvbnMgLSBPcHRpb25hbCBjb25maWd1cmF0aW9uIGZvciB0cmFja2luZyByZXF1ZXN0XG4gKiBAcmV0dXJucyBDb21wbGV0ZSBzaGlwbWVudCB0cmFja2luZyBpbmZvcm1hdGlvblxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0eXBlc2NyaXB0XG4gKiBpbXBvcnQgeyBnZXRTaGlwbWVudFRyYWNraW5nIH0gZnJvbSAnQHZlaG8vdHVydm8tc2RrJ1xuICpcbiAqIGNvbnN0IHRyYWNraW5nID0gYXdhaXQgZ2V0U2hpcG1lbnRUcmFja2luZygnMTIzNDUnLCB7XG4gKiAgIGluY2x1ZGVHcHM6IHRydWUsXG4gKiAgIHNlY3JldFBhdGg6ICcvdHVydm8vYXBpJ1xuICogfSlcbiAqXG4gKiBjb25zb2xlLmxvZyh0cmFja2luZy5zdGF0dXMsIHRyYWNraW5nLmV0YVV0YywgdHJhY2tpbmcuaXNMYXRlKVxuICogYGBgXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRTaGlwbWVudFRyYWNraW5nKFxuICBzaGlwbWVudElkOiBzdHJpbmcsXG4gIG9wdGlvbnM6IEdldFRyYWNraW5nT3B0aW9ucyA9IHt9XG4pOiBQcm9taXNlPFNoaXBtZW50VHJhY2tpbmc+IHtcbiAgLy8gQ3JlYXRlIGNsaWVudCB3aXRoIG9wdGlvbmFsIGN1c3RvbSBzZWNyZXQgcGF0aFxuICBjb25zdCBjbGllbnQgPSBuZXcgVHVydm9DbGllbnQob3B0aW9ucy5zZWNyZXRQYXRoKVxuXG4gIC8vIENyZWF0ZSBBUEkgaW5zdGFuY2VzXG4gIGNvbnN0IHB1YmxpY0FwaSA9IG5ldyBUdXJ2b1B1YmxpY0FwaShjbGllbnQpXG4gIGNvbnN0IGludGVybmFsQXBpID0gbmV3IFR1cnZvSW50ZXJuYWxBcGkoY2xpZW50KVxuXG4gIC8vIENyZWF0ZSB0cmFja2luZyBzZXJ2aWNlXG4gIGNvbnN0IHRyYWNraW5nU2VydmljZSA9IG5ldyBUcmFja2luZ1NlcnZpY2UocHVibGljQXBpLCBpbnRlcm5hbEFwaSlcblxuICAvLyBGZXRjaCBhbmQgcmV0dXJuIHRyYWNraW5nIGRhdGFcbiAgcmV0dXJuIHRyYWNraW5nU2VydmljZS5nZXRUcmFja2luZyhzaGlwbWVudElkLCBvcHRpb25zKVxufVxuIl19
@@ -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
+ }