@veho/turvo-integration-sdk 0.1.0-beta.0 → 0.1.0-beta.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,4 +1,6 @@
1
- import { TurvoNotFoundError } from '../types/errors';
1
+ import { DateTime } from 'luxon';
2
+ import { getFacilityMappingForTurvoLocationId, getRetiredFacilityMappingForTurvoLocationId, } from '../db/facilityMapping';
3
+ import { TurvoDateFormatError, TurvoNotFoundError } from '../types/errors';
2
4
  /**
3
5
  * TrackingService orchestrates calls to public and internal APIs
4
6
  * to provide a unified shipment tracking interface.
@@ -43,39 +45,77 @@ export class TrackingService {
43
45
  * Transform raw Turvo API responses into clean ShipmentTracking type
44
46
  *
45
47
  */
46
- transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
48
+ async transformToTrackingDetails(shipment, locationUpdates, internalApiFailed) {
47
49
  const shipmentData = shipment;
50
+ // Filter out deleted entries from turvo data
51
+ const globalRoute = shipmentData.globalRoute?.filter(route => route.deleted !== false);
52
+ const carrierOrder = shipmentData.carrierOrder?.filter(order => order.deleted !== false);
53
+ // Map global route to stops array
54
+ const allStops = await Promise.all(shipment.globalRoute
55
+ .filter(routeStop => !routeStop.deleted)
56
+ .map(async (routeStop) => {
57
+ // Make sure we're actually getting ISO, fail if we don't
58
+ const appointmentDateTime = DateTime.fromISO(routeStop.appointment.date).toUTC().toISO();
59
+ if (!appointmentDateTime) {
60
+ throw new TurvoDateFormatError(`expected ISO date but got ${routeStop.appointment.date}`);
61
+ }
62
+ // Additional check for a retired location mapping to get a facility id,
63
+ // for the case a facility has moved/been given a new entry in turvo.
64
+ const facilityId = (await getFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??
65
+ (await getRetiredFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??
66
+ null;
67
+ return {
68
+ stopNumber: routeStop.sequence,
69
+ appointmentDateTime,
70
+ appointmentLocalTimeZone: routeStop.appointment.timeZone,
71
+ locationName: routeStop.name,
72
+ externalLocationId: routeStop.location.id.toString(),
73
+ facilityId,
74
+ externalStopId: routeStop.id?.toString() ?? null,
75
+ stopServiceType: routeStop.stopType.value === 'Pickup' ? 'pickup' : 'delivery',
76
+ isCompleted: routeStop.state === 'COMPLETED',
77
+ locationId: routeStop.location.id,
78
+ etaToStop: routeStop.etaToStop?.etaValue ?? null,
79
+ milesRemainingToStop: routeStop.etaToStop?.nextMiles ?? null,
80
+ totalMilesToStop: routeStop.distance?.value ?? null,
81
+ // TODO: actual arrival time - time under attributes under global route
82
+ // TODO: actual departure time
83
+ // TODO: address, city, state - address under globalroute
84
+ // TODO: ask about what we want when we have multiple entries for the arrival/departure?
85
+ };
86
+ }));
87
+ const allStopsOrdered = allStops.toSorted((a, b) => a.stopNumber - b.stopNumber);
88
+ const nextStop = allStopsOrdered.find(stop => !stop.isCompleted) ?? null;
48
89
  return {
49
90
  // Identifiers
50
91
  shipmentId: shipmentData.id?.toString() || '',
51
92
  loadReferenceNumber: shipmentData.customId || '',
52
93
  // Status
53
94
  status: shipmentData.status?.code?.value || 'Unknown',
54
- lateBy: shipment.status.runningLate?.lateDurationString ?? null,
95
+ lateByMinutes: shipment.status.runningLate?.lateDuration ?? null,
55
96
  isLate: !!shipment.status.runningLate,
56
97
  // Current location (if tracking available)
57
- currentLocation: shipment.status.location ?? null,
98
+ currentLocation: shipment.status.location ?? null, // TODO: make sure this is the value we actually want, status location may not be it
58
99
  hasGpsTracking: locationUpdates !== null && !internalApiFailed,
59
100
  // Progress
60
- completedStops: 0, // TODO: Count from shipmentData.globalRoute
61
- totalStops: shipmentData.globalRoute?.length || 0, // TODO: parse from shipmentData.globalRoute
101
+ completedStops: allStopsOrdered.filter(stop => stop.isCompleted).length || 0,
102
+ totalStops: globalRoute?.length || 0,
62
103
  // Stops (ordered)
63
- stops: [], // TODO: implment global route parsing
104
+ stops: allStopsOrdered,
64
105
  // 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
106
+ etaUtc: nextStop?.etaToStop ?? null,
107
+ etaTimezone: nextStop?.appointmentLocalTimeZone ?? null,
108
+ milesRemaining: nextStop?.milesRemainingToStop ?? null,
109
+ totalMilesToNextStop: nextStop?.totalMilesToStop ?? null,
69
110
  // GPS Pings (only populated if includeGps: true and hasGpsTracking)
70
111
  locationUpdates: locationUpdates || null, // TODO: implment with scraper
71
112
  pingCount: locationUpdates?.length || null, // TODO: implement with scraper
72
- lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp ||
73
- null, // TODO: implement with scraper
113
+ lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp || null, // TODO: implement with scraper
74
114
  // Shipment Attributes
75
115
  laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,
76
116
  managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,
77
- carrier: shipmentData.carrierOrder?.[0]?.carrier?.name || null,
117
+ carrier: carrierOrder?.[0]?.carrier?.name || null,
78
118
  };
79
119
  }
80
120
  }
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"]}
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"trackingService.js","sourceRoot":"","sources":["../../../src/shipmentTracking/trackingService.ts"],"names":[],"mappings":"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","sourcesContent":["import { DateTime } from 'luxon'\n\nimport { TurvoInternalApi } from '../api/turvoInternalApi'\nimport { TurvoPublicApi } from '../api/turvoPublicApi'\nimport {\n  getFacilityMappingForTurvoLocationId,\n  getRetiredFacilityMappingForTurvoLocationId,\n} from '../db/facilityMapping'\nimport { TurvoShipment } from '../types'\nimport { TurvoDateFormatError, TurvoNotFoundError } from '../types/errors'\nimport { GetTrackingOptions, LocationUpdate, ShipmentTracking, Stop } from '../types/tracking'\n\n/**\n * TrackingService orchestrates calls to public and internal APIs\n * to provide a unified shipment tracking interface.\n */\nexport class TrackingService {\n  constructor(\n    private publicApi: TurvoPublicApi,\n    private internalApi: TurvoInternalApi\n  ) {}\n\n  /**\n   * Main entry point - combines all API data to produce ShipmentTracking\n   *\n   * @param shipmentId - The shipment ID to track\n   * @param options - Tracking options\n   * @returns Complete shipment tracking information\n   */\n  async getTracking(shipmentId: string, options: GetTrackingOptions = {}): Promise<ShipmentTracking> {\n    // 1. Fetch public API data (required)\n    const shipmentResult = await this.publicApi.getShipment({ shipmentId: parseInt(shipmentId) })\n\n    if (!shipmentResult.details) {\n      throw new TurvoNotFoundError(shipmentId)\n    }\n\n    const shipment = shipmentResult.details\n\n    // 2. Fetch internal API data (optional, may fail)\n    let locationUpdates: LocationUpdate[] | null = null\n    let internalApiFailed = false\n\n    if (options.includeGps !== false) {\n      try {\n        locationUpdates = await this.internalApi.getLocationUpdates(shipmentId)\n      } catch (error) {\n        // Log but don't fail - internal API is optional\n        internalApiFailed = true\n      }\n    }\n\n    // 3. Transform to clean response\n    return this.transformToTrackingDetails(shipment, locationUpdates, internalApiFailed)\n  }\n\n  /**\n   * Transform raw Turvo API responses into clean ShipmentTracking type\n   *\n   */\n  private async transformToTrackingDetails(\n    shipment: TurvoShipment,\n    locationUpdates: LocationUpdate[] | null,\n    internalApiFailed: boolean\n  ): Promise<ShipmentTracking> {\n    const shipmentData = shipment as TurvoShipment\n\n    // Filter out deleted entries from turvo data\n    const globalRoute = shipmentData.globalRoute?.filter(route => route.deleted !== false)\n    const carrierOrder = shipmentData.carrierOrder?.filter(order => order.deleted !== false)\n\n    // Map global route to stops array\n    const allStops: Stop[] = await Promise.all(\n      shipment.globalRoute\n        .filter(routeStop => !routeStop.deleted)\n        .map(async routeStop => {\n          // Make sure we're actually getting ISO, fail if we don't\n          const appointmentDateTime = DateTime.fromISO(routeStop.appointment.date).toUTC().toISO()\n          if (!appointmentDateTime) {\n            throw new TurvoDateFormatError(`expected ISO date but got ${routeStop.appointment.date}`)\n          }\n\n          // Additional check for a retired location mapping to get a facility id,\n          // for the case a facility has moved/been given a new entry in turvo.\n          const facilityId =\n            (await getFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??\n            (await getRetiredFacilityMappingForTurvoLocationId(routeStop.location.id))?.facilityId ??\n            null\n\n          return {\n            stopNumber: routeStop.sequence,\n            appointmentDateTime,\n            appointmentLocalTimeZone: routeStop.appointment.timeZone,\n            locationName: routeStop.name,\n            externalLocationId: routeStop.location.id.toString(),\n            facilityId,\n            externalStopId: routeStop.id?.toString() ?? null,\n            stopServiceType: routeStop.stopType.value === 'Pickup' ? 'pickup' : 'delivery',\n            isCompleted: routeStop.state === 'COMPLETED',\n            locationId: routeStop.location.id,\n            etaToStop: routeStop.etaToStop?.etaValue ?? null,\n            milesRemainingToStop: routeStop.etaToStop?.nextMiles ?? null,\n            totalMilesToStop: routeStop.distance?.value ?? null,\n            // TODO: actual arrival time - time under attributes under global route\n            // TODO: actual departure time\n            // TODO: address, city, state -  address under globalroute\n            // TODO: ask about what we want when we have multiple entries for the arrival/departure?\n          }\n        })\n    )\n\n    const allStopsOrdered = allStops.toSorted((a, b) => a.stopNumber - b.stopNumber)\n    const nextStop = allStopsOrdered.find(stop => !stop.isCompleted) ?? null\n\n    return {\n      // Identifiers\n      shipmentId: (shipmentData.id as number)?.toString() || '',\n      loadReferenceNumber: (shipmentData.customId as string) || '',\n\n      // Status\n      status: (shipmentData.status?.code?.value as string) || 'Unknown',\n      lateByMinutes: shipment.status.runningLate?.lateDuration ?? null,\n      isLate: !!shipment.status.runningLate,\n\n      // Current location (if tracking available)\n      currentLocation: shipment.status.location ?? null, // TODO: make sure this is the value we actually want, status location may not be it\n      hasGpsTracking: locationUpdates !== null && !internalApiFailed,\n\n      // Progress\n      completedStops: allStopsOrdered.filter(stop => stop.isCompleted).length || 0,\n      totalStops: globalRoute?.length || 0,\n\n      // Stops (ordered)\n      stops: allStopsOrdered,\n\n      // ETA & Distance\n      etaUtc: nextStop?.etaToStop ?? null,\n      etaTimezone: nextStop?.appointmentLocalTimeZone ?? null,\n      milesRemaining: nextStop?.milesRemainingToStop ?? null,\n      totalMilesToNextStop: nextStop?.totalMilesToStop ?? null,\n\n      // GPS Pings (only populated if includeGps: true and hasGpsTracking)\n      locationUpdates: locationUpdates || null, // TODO: implment with scraper\n      pingCount: locationUpdates?.length || null, // TODO: implement with scraper\n      lastPingAt: locationUpdates?.[(locationUpdates?.length || 0) - 1]?.timestamp || null, // TODO: implement with scraper\n\n      // Shipment Attributes\n      laneType: shipment.flexAttributes?.find(attribute => attribute.name === 'Lane Type')?.value ?? null,\n      managementType: shipment.flexAttributes?.find(attribute => attribute.name === 'Management Type')?.value ?? null,\n      carrier: carrierOrder?.[0]?.carrier?.name || null,\n    }\n  }\n}\n"]}
@@ -33,3 +33,9 @@ export declare class TurvoPartialDataError extends Error {
33
33
  export declare class NoSecretError extends Error {
34
34
  constructor(path: string);
35
35
  }
36
+ /**
37
+ * Error thrown when a date format from Turvo is invalid
38
+ */
39
+ export declare class TurvoDateFormatError extends Error {
40
+ constructor(message: string);
41
+ }
@@ -52,4 +52,13 @@ export class NoSecretError extends Error {
52
52
  this.name = 'NoSecretError';
53
53
  }
54
54
  }
55
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILE1BQU0sT0FBTyxrQkFBbUIsU0FBUSxLQUFLO0lBQ2Y7SUFBNUIsWUFBNEIsVUFBa0I7UUFDNUMsS0FBSyxDQUFDLGdDQUFnQyxVQUFVLEVBQUUsQ0FBQyxDQUFBO1FBRHpCLGVBQVUsR0FBVixVQUFVLENBQVE7UUFFNUMsSUFBSSxDQUFDLElBQUksR0FBRyxvQkFBb0IsQ0FBQTtJQUNsQyxDQUFDO0NBQ0Y7QUFFRDs7R0FFRztBQUNILE1BQU0sT0FBTyxjQUFlLFNBQVEsS0FBSztJQUN2QyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLGdDQUFnQyxPQUFPLEVBQUUsQ0FBQyxDQUFBO1FBQ2hELElBQUksQ0FBQyxJQUFJLEdBQUcsZ0JBQWdCLENBQUE7SUFDOUIsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sbUJBQW9CLFNBQVEsS0FBSztJQUNoQjtJQUE1QixZQUE0QixVQUFtQjtRQUM3QyxLQUFLLENBQUMsd0JBQXdCLFVBQVUsQ0FBQyxDQUFDLENBQUMsaUJBQWlCLFVBQVUsVUFBVSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRDlELGVBQVUsR0FBVixVQUFVLENBQVM7UUFFN0MsSUFBSSxDQUFDLElBQUksR0FBRyxxQkFBcUIsQ0FBQTtJQUNuQyxDQUFDO0NBQ0Y7QUFFRDs7O0dBR0c7QUFDSCxNQUFNLE9BQU8scUJBQXNCLFNBQVEsS0FBSztJQUU1QjtJQUNBO0lBRmxCLFlBQ2tCLFdBQW9CLEVBQ3BCLFVBQW9CO1FBRXBDLEtBQUssQ0FBQyxrQ0FBa0MsVUFBVSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsMkJBQTJCLENBQUMsQ0FBQTtRQUh6RSxnQkFBVyxHQUFYLFdBQVcsQ0FBUztRQUNwQixlQUFVLEdBQVYsVUFBVSxDQUFVO1FBR3BDLElBQUksQ0FBQyxJQUFJLEdBQUcsdUJBQXVCLENBQUE7SUFDckMsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sYUFBYyxTQUFRLEtBQUs7SUFDdEMsWUFBWSxJQUFZO1FBQ3RCLEtBQUssQ0FBQyw0QkFBNEIsSUFBSSxFQUFFLENBQUMsQ0FBQTtRQUN6QyxJQUFJLENBQUMsSUFBSSxHQUFHLGVBQWUsQ0FBQTtJQUM3QixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNoaXBtZW50IG5vdCBmb3VuZCBpbiBUdXJ2b1xuICovXG5leHBvcnQgY2xhc3MgVHVydm9Ob3RGb3VuZEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgc2hpcG1lbnRJZDogc3RyaW5nKSB7XG4gICAgc3VwZXIoYFNoaXBtZW50IG5vdCBmb3VuZCBpbiBUdXJ2bzogJHtzaGlwbWVudElkfWApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvTm90Rm91bmRFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIEF1dGhlbnRpY2F0aW9uIGZhaWxlZCAoaW52YWxpZCBjcmVkZW50aWFscywgZXhwaXJlZCB0b2tlbilcbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvQXV0aEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihtZXNzYWdlOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgVHVydm8gYXV0aGVudGljYXRpb24gZmFpbGVkOiAke21lc3NhZ2V9YClcbiAgICB0aGlzLm5hbWUgPSAnVHVydm9BdXRoRXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBSYXRlIGxpbWl0ZWQgYnkgVHVydm9cbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvUmF0ZUxpbWl0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSByZXRyeUFmdGVyPzogbnVtYmVyKSB7XG4gICAgc3VwZXIoYFJhdGUgbGltaXRlZCBieSBUdXJ2byR7cmV0cnlBZnRlciA/IGAuIFJldHJ5IGFmdGVyICR7cmV0cnlBZnRlcn0gc2Vjb25kc2AgOiAnJ31gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b1JhdGVMaW1pdEVycm9yJ1xuICB9XG59XG5cbi8qKlxuICogSW50ZXJuYWwgQVBJIGZhaWxlZCBidXQgcHVibGljIEFQSSBzdWNjZWVkZWQuXG4gKiBDb250YWlucyBwYXJ0aWFsIGRhdGEgaW4gYHBhcnRpYWxEYXRhYCBmaWVsZC5cbiAqL1xuZXhwb3J0IGNsYXNzIFR1cnZvUGFydGlhbERhdGFFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHJlYWRvbmx5IHBhcnRpYWxEYXRhOiB1bmtub3duLFxuICAgIHB1YmxpYyByZWFkb25seSBmYWlsZWRBcGlzOiBzdHJpbmdbXVxuICApIHtcbiAgICBzdXBlcihgVHVydm8gaW50ZXJuYWwgQVBJIGZhaWxlZCBmb3I6ICR7ZmFpbGVkQXBpcy5qb2luKCcsICcpfS4gUGFydGlhbCBkYXRhIGF2YWlsYWJsZS5gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b1BhcnRpYWxEYXRhRXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBUdXJ2byBzZWNyZXRzIGNhbm5vdCBiZSByZXRyaWV2ZWQgZnJvbSBBV1MgU2VjcmV0cyBNYW5hZ2VyXG4gKi9cbmV4cG9ydCBjbGFzcyBOb1NlY3JldEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihwYXRoOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgTm8gc2VjcmV0IGZvdW5kIGF0IHBhdGg6ICR7cGF0aH1gKVxuICAgIHRoaXMubmFtZSA9ICdOb1NlY3JldEVycm9yJ1xuICB9XG59XG4iXX0=
55
+ /**
56
+ * Error thrown when a date format from Turvo is invalid
57
+ */
58
+ export class TurvoDateFormatError extends Error {
59
+ constructor(message) {
60
+ super(message);
61
+ this.name = 'TurvoDateFormatError';
62
+ }
63
+ }
64
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXJyb3JzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2Vycm9ycy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7R0FFRztBQUNILE1BQU0sT0FBTyxrQkFBbUIsU0FBUSxLQUFLO0lBQ2Y7SUFBNUIsWUFBNEIsVUFBa0I7UUFDNUMsS0FBSyxDQUFDLGdDQUFnQyxVQUFVLEVBQUUsQ0FBQyxDQUFBO1FBRHpCLGVBQVUsR0FBVixVQUFVLENBQVE7UUFFNUMsSUFBSSxDQUFDLElBQUksR0FBRyxvQkFBb0IsQ0FBQTtJQUNsQyxDQUFDO0NBQ0Y7QUFFRDs7R0FFRztBQUNILE1BQU0sT0FBTyxjQUFlLFNBQVEsS0FBSztJQUN2QyxZQUFZLE9BQWU7UUFDekIsS0FBSyxDQUFDLGdDQUFnQyxPQUFPLEVBQUUsQ0FBQyxDQUFBO1FBQ2hELElBQUksQ0FBQyxJQUFJLEdBQUcsZ0JBQWdCLENBQUE7SUFDOUIsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sbUJBQW9CLFNBQVEsS0FBSztJQUNoQjtJQUE1QixZQUE0QixVQUFtQjtRQUM3QyxLQUFLLENBQUMsd0JBQXdCLFVBQVUsQ0FBQyxDQUFDLENBQUMsaUJBQWlCLFVBQVUsVUFBVSxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFBO1FBRDlELGVBQVUsR0FBVixVQUFVLENBQVM7UUFFN0MsSUFBSSxDQUFDLElBQUksR0FBRyxxQkFBcUIsQ0FBQTtJQUNuQyxDQUFDO0NBQ0Y7QUFFRDs7O0dBR0c7QUFDSCxNQUFNLE9BQU8scUJBQXNCLFNBQVEsS0FBSztJQUU1QjtJQUNBO0lBRmxCLFlBQ2tCLFdBQW9CLEVBQ3BCLFVBQW9CO1FBRXBDLEtBQUssQ0FBQyxrQ0FBa0MsVUFBVSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsMkJBQTJCLENBQUMsQ0FBQTtRQUh6RSxnQkFBVyxHQUFYLFdBQVcsQ0FBUztRQUNwQixlQUFVLEdBQVYsVUFBVSxDQUFVO1FBR3BDLElBQUksQ0FBQyxJQUFJLEdBQUcsdUJBQXVCLENBQUE7SUFDckMsQ0FBQztDQUNGO0FBRUQ7O0dBRUc7QUFDSCxNQUFNLE9BQU8sYUFBYyxTQUFRLEtBQUs7SUFDdEMsWUFBWSxJQUFZO1FBQ3RCLEtBQUssQ0FBQyw0QkFBNEIsSUFBSSxFQUFFLENBQUMsQ0FBQTtRQUN6QyxJQUFJLENBQUMsSUFBSSxHQUFHLGVBQWUsQ0FBQTtJQUM3QixDQUFDO0NBQ0Y7QUFFRDs7R0FFRztBQUNILE1BQU0sT0FBTyxvQkFBcUIsU0FBUSxLQUFLO0lBQzdDLFlBQVksT0FBZTtRQUN6QixLQUFLLENBQUMsT0FBTyxDQUFDLENBQUE7UUFDZCxJQUFJLENBQUMsSUFBSSxHQUFHLHNCQUFzQixDQUFBO0lBQ3BDLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU2hpcG1lbnQgbm90IGZvdW5kIGluIFR1cnZvXG4gKi9cbmV4cG9ydCBjbGFzcyBUdXJ2b05vdEZvdW5kRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSBzaGlwbWVudElkOiBzdHJpbmcpIHtcbiAgICBzdXBlcihgU2hpcG1lbnQgbm90IGZvdW5kIGluIFR1cnZvOiAke3NoaXBtZW50SWR9YClcbiAgICB0aGlzLm5hbWUgPSAnVHVydm9Ob3RGb3VuZEVycm9yJ1xuICB9XG59XG5cbi8qKlxuICogQXV0aGVudGljYXRpb24gZmFpbGVkIChpbnZhbGlkIGNyZWRlbnRpYWxzLCBleHBpcmVkIHRva2VuKVxuICovXG5leHBvcnQgY2xhc3MgVHVydm9BdXRoRXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKGBUdXJ2byBhdXRoZW50aWNhdGlvbiBmYWlsZWQ6ICR7bWVzc2FnZX1gKVxuICAgIHRoaXMubmFtZSA9ICdUdXJ2b0F1dGhFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIFJhdGUgbGltaXRlZCBieSBUdXJ2b1xuICovXG5leHBvcnQgY2xhc3MgVHVydm9SYXRlTGltaXRFcnJvciBleHRlbmRzIEVycm9yIHtcbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHJldHJ5QWZ0ZXI/OiBudW1iZXIpIHtcbiAgICBzdXBlcihgUmF0ZSBsaW1pdGVkIGJ5IFR1cnZvJHtyZXRyeUFmdGVyID8gYC4gUmV0cnkgYWZ0ZXIgJHtyZXRyeUFmdGVyfSBzZWNvbmRzYCA6ICcnfWApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvUmF0ZUxpbWl0RXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBJbnRlcm5hbCBBUEkgZmFpbGVkIGJ1dCBwdWJsaWMgQVBJIHN1Y2NlZWRlZC5cbiAqIENvbnRhaW5zIHBhcnRpYWwgZGF0YSBpbiBgcGFydGlhbERhdGFgIGZpZWxkLlxuICovXG5leHBvcnQgY2xhc3MgVHVydm9QYXJ0aWFsRGF0YUVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgcmVhZG9ubHkgcGFydGlhbERhdGE6IHVua25vd24sXG4gICAgcHVibGljIHJlYWRvbmx5IGZhaWxlZEFwaXM6IHN0cmluZ1tdXG4gICkge1xuICAgIHN1cGVyKGBUdXJ2byBpbnRlcm5hbCBBUEkgZmFpbGVkIGZvcjogJHtmYWlsZWRBcGlzLmpvaW4oJywgJyl9LiBQYXJ0aWFsIGRhdGEgYXZhaWxhYmxlLmApXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvUGFydGlhbERhdGFFcnJvcidcbiAgfVxufVxuXG4vKipcbiAqIEVycm9yIHRocm93biB3aGVuIFR1cnZvIHNlY3JldHMgY2Fubm90IGJlIHJldHJpZXZlZCBmcm9tIEFXUyBTZWNyZXRzIE1hbmFnZXJcbiAqL1xuZXhwb3J0IGNsYXNzIE5vU2VjcmV0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKHBhdGg6IHN0cmluZykge1xuICAgIHN1cGVyKGBObyBzZWNyZXQgZm91bmQgYXQgcGF0aDogJHtwYXRofWApXG4gICAgdGhpcy5uYW1lID0gJ05vU2VjcmV0RXJyb3InXG4gIH1cbn1cblxuLyoqXG4gKiBFcnJvciB0aHJvd24gd2hlbiBhIGRhdGUgZm9ybWF0IGZyb20gVHVydm8gaXMgaW52YWxpZFxuICovXG5leHBvcnQgY2xhc3MgVHVydm9EYXRlRm9ybWF0RXJyb3IgZXh0ZW5kcyBFcnJvciB7XG4gIGNvbnN0cnVjdG9yKG1lc3NhZ2U6IHN0cmluZykge1xuICAgIHN1cGVyKG1lc3NhZ2UpXG4gICAgdGhpcy5uYW1lID0gJ1R1cnZvRGF0ZUZvcm1hdEVycm9yJ1xuICB9XG59XG4iXX0=
@@ -0,0 +1,9 @@
1
+ /**
2
+ * Facility mapping types for Turvo location to Veho facility mapping
3
+ */
4
+ export interface VehoTurvoLocationMap {
5
+ facilityCode?: string | null;
6
+ facilityId: string;
7
+ shortCode?: string | null;
8
+ turvoLocationId?: number | null;
9
+ }
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmFjaWxpdHlNYXBwaW5nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3R5cGVzL2ZhY2lsaXR5TWFwcGluZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBGYWNpbGl0eSBtYXBwaW5nIHR5cGVzIGZvciBUdXJ2byBsb2NhdGlvbiB0byBWZWhvIGZhY2lsaXR5IG1hcHBpbmdcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBWZWhvVHVydm9Mb2NhdGlvbk1hcCB7XG4gIGZhY2lsaXR5Q29kZT86IHN0cmluZyB8IG51bGxcbiAgZmFjaWxpdHlJZDogc3RyaW5nXG4gIHNob3J0Q29kZT86IHN0cmluZyB8IG51bGxcbiAgdHVydm9Mb2NhdGlvbklkPzogbnVtYmVyIHwgbnVsbFxufVxuIl19
@@ -19,6 +19,16 @@ export type TurvoGlobalRoute = {
19
19
  carrierOrderSourceId: number;
20
20
  }[];
21
21
  appointment: TurvoAppointment;
22
+ etaToStop: {
23
+ etaValue: string;
24
+ nextMiles: number;
25
+ } | null;
26
+ distance: {
27
+ value: number;
28
+ units: {
29
+ value: string;
30
+ };
31
+ } | null;
22
32
  }[];
23
33
  export type TurvoLane = {
24
34
  start: string;
@@ -271,6 +281,7 @@ export type TurvoCarrierOrder = {
271
281
  payable: true;
272
282
  }[];
273
283
  };
284
+ deleted?: boolean;
274
285
  };
275
286
  export type TurvoAppointment = {
276
287
  date: string;
@@ -40,4 +40,4 @@ export var TurvoFlexAttributeManagementTypeValue;
40
40
  TurvoFlexAttributeManagementTypeValue["VehoManaged"] = "Veho Managed";
41
41
  TurvoFlexAttributeManagementTypeValue["Insourced"] = "Insourced";
42
42
  })(TurvoFlexAttributeManagementTypeValue || (TurvoFlexAttributeManagementTypeValue = {}));
43
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shipment.js","sourceRoot":"","sources":["../../../src/types/shipment.ts"],"names":[],"mappings":"AA0EA,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,8BAA8B,GAAwB;IACjE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,UAAU;KAClB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,iCAAiC,GAAwB;IACpE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,aAAa;KACrB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,qBAAqB,GAAG,kBAAkB,CAAA;AA+BvD,MAAM,CAAN,IAAY,+BAGX;AAHD,WAAY,+BAA+B;IACzC,2DAAwB,CAAA;IACxB,6DAA0B,CAAA;AAC5B,CAAC,EAHW,+BAA+B,KAA/B,+BAA+B,QAG1C;AAED,MAAM,CAAN,IAAY,qCAIX;AAJD,WAAY,qCAAqC;IAC/C,yEAAgC,CAAA;IAChC,qEAA4B,CAAA;IAC5B,gEAAuB,CAAA;AACzB,CAAC,EAJW,qCAAqC,KAArC,qCAAqC,QAIhD","sourcesContent":["import { TurvoDateWithTimezone, TurvoDateWithTimezoneLowercase, TurvoLookup } from './common'\n\nexport type TurvoGlobalRoute = {\n  id: number\n  name: string\n  globalShipLocationSourceId?: string\n  schedulingType: TurvoLookup\n  stopType: TurvoLookup\n  location: {\n    id: number\n  }\n  deleted?: boolean\n  sequence: number\n  segmentSequence: number\n  state: 'OPEN' | 'CLOSED' | 'COMPLETED'\n  customerOrder?: TurvoCustomerOrder[]\n  carrierOrder?: { carrierId: number; id: number; carrierOrderSourceId: number }[]\n  appointment: TurvoAppointment\n}[]\n\nexport type TurvoLane = {\n  start: string\n  end: string\n}\n\nexport type TurvoShipmentService = TurvoLookup &\n  (\n    | {\n        key: '21204'\n        value: 'Team'\n      }\n    | {\n        key: '21214'\n        value: 'Drop trailer'\n      }\n  )\n\nexport type TurvoShipmentStatus = {\n  notes?: string\n  attributes?: {\n    etaValUtc?: string\n    next_eta_cal_val_timezone?: string\n  }\n  description?: string\n  statusDate?: TurvoDateWithTimezoneLowercase\n  updatedBy?: {\n    id?: number\n    name?: string\n  }\n  location?: TurvoShipmentStatusLocation\n  runningLate?: TurvoShipmentStatusRunningLate\n  code:\n    | { key: '2100'; value: 'Quote active' }\n    | { key: '2101'; value: 'Tendered' }\n    | { key: '2102'; value: 'Covered' }\n    | { key: '2103'; value: 'Dispatched' }\n    | { key: '2104'; value: 'At pickup' }\n    | { key: '2105'; value: 'En route' }\n    | { key: '2106'; value: 'At delivery' }\n    | { key: '2116'; value: 'Route Complete' }\n    | { key: '2107'; value: 'Delivered' }\n    | { key: '2108'; value: 'Ready for billing' }\n    | { key: '2109'; value: 'Processing' }\n    | { key: '2110'; value: 'Carrier paid' }\n    | { key: '2111'; value: 'Customer paid' }\n    | { key: '2112'; value: 'Completed' }\n    | { key: '2113'; value: 'Canceled' }\n    | { key: '2117'; value: 'Tender - offered' }\n    | { key: '2118'; value: 'Tender - accepted' }\n    | { key: '2119'; value: 'Tender - rejected' }\n    | { key: '2114'; value: 'Quote inactive' }\n    | { key: '2115'; value: 'Picked up' }\n}\n\nexport const TURVO_SHIPMENT_STATUS_DELIVERED: TurvoShipmentStatus = {\n  code: {\n    key: '2107',\n    value: 'Delivered',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_PICKED_UP: TurvoShipmentStatus = {\n  code: {\n    key: '2115',\n    value: 'Picked up',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_AT_PICKUP: TurvoShipmentStatus = {\n  code: {\n    key: '2104',\n    value: 'At pickup',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_EN_ROUTE: TurvoShipmentStatus = {\n  code: {\n    key: '2105',\n    value: 'En route',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_AT_DELIVERY: TurvoShipmentStatus = {\n  code: {\n    key: '2106',\n    value: 'At delivery',\n  },\n}\n\nexport const TURVO_SYSTEM_BOT_USER = 'Turvo System Bot'\n\nexport type TurvoWebhookStatus = {\n  code:\n    | { key: '2100'; value: 'Quote active' }\n    | { key: '2101'; value: 'Tendered' }\n    | { key: '2102'; value: 'Covered' }\n    | { key: '2103'; value: 'Dispatched' }\n    | { key: '2104'; value: 'At pickup' }\n    | { key: '2105'; value: 'En route' }\n    | { key: '2106'; value: 'At delivery' }\n    | { key: '2116'; value: 'Route Complete' }\n    | { key: '2107'; value: 'Delivered' }\n    | { key: '2108'; value: 'Ready for billing' }\n    | { key: '2109'; value: 'Processing' }\n    | { key: '2110'; value: 'Carrier paid' }\n    | { key: '2111'; value: 'Customer paid' }\n    | { key: '2112'; value: 'Completed' }\n    | { key: '2113'; value: 'Canceled' }\n    | { key: '2117'; value: 'Tender - offered' }\n    | { key: '2118'; value: 'Tender - accepted' }\n    | { key: '2119'; value: 'Tender - rejected' }\n    | { key: '2114'; value: 'Quote inactive' }\n    | { key: '2115'; value: 'Picked up' }\n  description?: string\n  updatedBy?: {\n    id?: number\n    name?: string\n  }\n}\n\nexport enum TurvoFlexAttributeLaneTypeValue {\n  FirstMile = 'First Mile',\n  MiddleMile = 'Middle Mile',\n}\n\nexport enum TurvoFlexAttributeManagementTypeValue {\n  ClientManaged = 'Client Managed',\n  VehoManaged = 'Veho Managed',\n  Insourced = 'Insourced',\n}\n\nexport type TurvoFlexAttribute = {\n  deleted: false\n  shareable: false\n} & (\n  | {\n      name: 'Lane Type'\n      type: {\n        value: 'DROPDOWN'\n        key: 'VehoTech_Lane_Type'\n      }\n      value: TurvoFlexAttributeLaneTypeValue\n    }\n  | {\n      name: 'Management Type'\n      type: {\n        value: 'DROPDOWN'\n        key: 'VehoTech_Management_Type'\n      }\n      value: TurvoFlexAttributeManagementTypeValue\n    }\n)\n\nexport type TurvoEquipment = {\n  type: TurvoLookup\n  size: TurvoLookup | null\n}[]\n\nexport type TurvoCustomer = {\n  id: number\n  name: string\n}\n\nexport type TurvoCustomerOrder = {\n  id?: number\n  customerId?: number\n  customer: TurvoCustomer\n  customerOrderSourceId: number\n  externalIds: {\n    type: { key: '1400'; value: 'Purchase order #' } | { key: '1401'; value: 'Reference #' }\n    value: string\n    copyToCarrierOrder: true\n  }[]\n  items?: {\n    name: string\n    pickupLocation: { id: number; globalShipLocationSourceId?: number; globalShipLocationId?: number; name: string }[]\n    deliveryLocation: { id: number; globalShipLocationSourceId?: number; globalShipLocationId?: number; name: string }[]\n    qty?: number\n    unit?: TurvoLookup\n  }[]\n}\n\nexport type TurvoCarrierOrder = {\n  _operation: 0\n  carrier: {\n    id: number\n    name: string\n  }\n  carrierOrderSourceId: number\n  costs?: {\n    totalAmount: number\n    lineItem: {\n      id?: number\n      code: { key: '1600'; value: 'Freight - flat' }\n      qty: 1\n      price: number\n      amount: number\n      payable: true\n    }[]\n  }\n}\n\nexport type TurvoAppointment = {\n  date: string\n  flex?: number // Apparently required but doesn't make sense I think, and the shipment creates without this\n  timeZone: string\n  hasTime: true\n}\n\nexport type TurvoShipment = {\n  id: number\n  customId: string\n  ltlShipment: boolean\n  startDate: TurvoDateWithTimezone\n  endDate: TurvoDateWithTimezone & { flex: number }\n  status: TurvoShipmentStatus\n  services?: TurvoShipmentService[]\n  equipment: TurvoEquipment\n  flexAttributes?: TurvoFlexAttribute[]\n  lane: TurvoLane\n  globalRoute: TurvoGlobalRoute\n  customerOrder: TurvoCustomerOrder[]\n  carrierOrder: TurvoCarrierOrder[]\n}\n\nexport type TurvoShipmentUpdate = {\n  id: number\n  customId: string\n  globalRoute: TurvoGlobalRoute\n  customerOrder: TurvoCustomerOrder[]\n  status: TurvoWebhookStatus\n}\n\nexport interface TurvoShipmentStatusApiResponse {\n  customId: string\n  status: {\n    location: {\n      lon: number\n      lat: number\n      nextArrivalTime: number\n      nextMiles: number\n      city: string\n    }\n    statusDate: {\n      date: string\n      timezone: string\n    }\n    runningLate?: {\n      nextLocationName: string\n    }\n    nextLocationId: number\n  }\n  globalRoute: TurvoGlobalRoute\n  lane: {\n    start: string\n    end: string\n  }\n}\n\nexport type TurvoShipmentStatusLocation = {\n  city: string\n  state: string\n  countryCode: string\n  countryName: string\n  lon: number\n  lat: number\n  nextEta?: string\n  nextEtaCalVal?: string\n  nextArrivalTime?: number\n  nextMiles?: number\n  currentDate?: string\n}\n\nexport type TurvoShipmentStatusRunningLate = {\n  lateType: string\n  lateDuration: number\n  nextLocationType: string\n  lateDurationString: string\n  nextLocationEta: string\n  nextLocationTimeZone: string\n  nextLocationName: string\n  nextLocationEtaDate: string\n  isApproachingLate: boolean\n  statusKey: string\n  nextLocationAppointmentDate: string\n  nextLocationAppointment: string\n  lastUpdatedOn: string\n  nextLocationId: number\n}\n\nexport type TurvoShipmentStatusUpdate = {\n  id: number\n  status: TurvoWebhookStatus\n}\n\n// Result types for specific operations\nexport type TurvoCreateShipmentApiResult = { id: number }\nexport type TurvoCancelShipmentApiResult = { respMsg: string }\nexport type TurvoUpdateShipmentStatusApiResult = { id: number }\nexport type TurvoAssociateTagsApiResult = {\n  tags: { id: string; tagName: string; contextType: string; contextId: number }[]\n}\n"]}
43
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"shipment.js","sourceRoot":"","sources":["../../../src/types/shipment.ts"],"names":[],"mappings":"AAkFA,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,+BAA+B,GAAwB;IAClE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,WAAW;KACnB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,8BAA8B,GAAwB;IACjE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,UAAU;KAClB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,iCAAiC,GAAwB;IACpE,IAAI,EAAE;QACJ,GAAG,EAAE,MAAM;QACX,KAAK,EAAE,aAAa;KACrB;CACF,CAAA;AAED,MAAM,CAAC,MAAM,qBAAqB,GAAG,kBAAkB,CAAA;AA+BvD,MAAM,CAAN,IAAY,+BAGX;AAHD,WAAY,+BAA+B;IACzC,2DAAwB,CAAA;IACxB,6DAA0B,CAAA;AAC5B,CAAC,EAHW,+BAA+B,KAA/B,+BAA+B,QAG1C;AAED,MAAM,CAAN,IAAY,qCAIX;AAJD,WAAY,qCAAqC;IAC/C,yEAAgC,CAAA;IAChC,qEAA4B,CAAA;IAC5B,gEAAuB,CAAA;AACzB,CAAC,EAJW,qCAAqC,KAArC,qCAAqC,QAIhD","sourcesContent":["import { TurvoDateWithTimezone, TurvoDateWithTimezoneLowercase, TurvoLookup } from './common'\n\nexport type TurvoGlobalRoute = {\n  id: number\n  name: string\n  globalShipLocationSourceId?: string\n  schedulingType: TurvoLookup\n  stopType: TurvoLookup\n  location: {\n    id: number\n  }\n  deleted?: boolean\n  sequence: number\n  segmentSequence: number\n  state: 'OPEN' | 'CLOSED' | 'COMPLETED'\n  customerOrder?: TurvoCustomerOrder[]\n  carrierOrder?: { carrierId: number; id: number; carrierOrderSourceId: number }[]\n  appointment: TurvoAppointment\n  etaToStop: {\n    etaValue: string\n    nextMiles: number\n  } | null\n  distance: {\n    value: number\n    units: { value: string }\n  } | null\n}[]\n\nexport type TurvoLane = {\n  start: string\n  end: string\n}\n\nexport type TurvoShipmentService = TurvoLookup &\n  (\n    | {\n        key: '21204'\n        value: 'Team'\n      }\n    | {\n        key: '21214'\n        value: 'Drop trailer'\n      }\n  )\n\nexport type TurvoShipmentStatus = {\n  notes?: string\n  attributes?: {\n    etaValUtc?: string\n    next_eta_cal_val_timezone?: string\n  }\n  description?: string\n  statusDate?: TurvoDateWithTimezoneLowercase\n  updatedBy?: {\n    id?: number\n    name?: string\n  }\n  location?: TurvoShipmentStatusLocation\n  runningLate?: TurvoShipmentStatusRunningLate\n  code:\n    | { key: '2100'; value: 'Quote active' }\n    | { key: '2101'; value: 'Tendered' }\n    | { key: '2102'; value: 'Covered' }\n    | { key: '2103'; value: 'Dispatched' }\n    | { key: '2104'; value: 'At pickup' }\n    | { key: '2105'; value: 'En route' }\n    | { key: '2106'; value: 'At delivery' }\n    | { key: '2116'; value: 'Route Complete' }\n    | { key: '2107'; value: 'Delivered' }\n    | { key: '2108'; value: 'Ready for billing' }\n    | { key: '2109'; value: 'Processing' }\n    | { key: '2110'; value: 'Carrier paid' }\n    | { key: '2111'; value: 'Customer paid' }\n    | { key: '2112'; value: 'Completed' }\n    | { key: '2113'; value: 'Canceled' }\n    | { key: '2117'; value: 'Tender - offered' }\n    | { key: '2118'; value: 'Tender - accepted' }\n    | { key: '2119'; value: 'Tender - rejected' }\n    | { key: '2114'; value: 'Quote inactive' }\n    | { key: '2115'; value: 'Picked up' }\n}\n\nexport const TURVO_SHIPMENT_STATUS_DELIVERED: TurvoShipmentStatus = {\n  code: {\n    key: '2107',\n    value: 'Delivered',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_PICKED_UP: TurvoShipmentStatus = {\n  code: {\n    key: '2115',\n    value: 'Picked up',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_AT_PICKUP: TurvoShipmentStatus = {\n  code: {\n    key: '2104',\n    value: 'At pickup',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_EN_ROUTE: TurvoShipmentStatus = {\n  code: {\n    key: '2105',\n    value: 'En route',\n  },\n}\n\nexport const TURVO_SHIPMENT_STATUS_AT_DELIVERY: TurvoShipmentStatus = {\n  code: {\n    key: '2106',\n    value: 'At delivery',\n  },\n}\n\nexport const TURVO_SYSTEM_BOT_USER = 'Turvo System Bot'\n\nexport type TurvoWebhookStatus = {\n  code:\n    | { key: '2100'; value: 'Quote active' }\n    | { key: '2101'; value: 'Tendered' }\n    | { key: '2102'; value: 'Covered' }\n    | { key: '2103'; value: 'Dispatched' }\n    | { key: '2104'; value: 'At pickup' }\n    | { key: '2105'; value: 'En route' }\n    | { key: '2106'; value: 'At delivery' }\n    | { key: '2116'; value: 'Route Complete' }\n    | { key: '2107'; value: 'Delivered' }\n    | { key: '2108'; value: 'Ready for billing' }\n    | { key: '2109'; value: 'Processing' }\n    | { key: '2110'; value: 'Carrier paid' }\n    | { key: '2111'; value: 'Customer paid' }\n    | { key: '2112'; value: 'Completed' }\n    | { key: '2113'; value: 'Canceled' }\n    | { key: '2117'; value: 'Tender - offered' }\n    | { key: '2118'; value: 'Tender - accepted' }\n    | { key: '2119'; value: 'Tender - rejected' }\n    | { key: '2114'; value: 'Quote inactive' }\n    | { key: '2115'; value: 'Picked up' }\n  description?: string\n  updatedBy?: {\n    id?: number\n    name?: string\n  }\n}\n\nexport enum TurvoFlexAttributeLaneTypeValue {\n  FirstMile = 'First Mile',\n  MiddleMile = 'Middle Mile',\n}\n\nexport enum TurvoFlexAttributeManagementTypeValue {\n  ClientManaged = 'Client Managed',\n  VehoManaged = 'Veho Managed',\n  Insourced = 'Insourced',\n}\n\nexport type TurvoFlexAttribute = {\n  deleted: false\n  shareable: false\n} & (\n  | {\n      name: 'Lane Type'\n      type: {\n        value: 'DROPDOWN'\n        key: 'VehoTech_Lane_Type'\n      }\n      value: TurvoFlexAttributeLaneTypeValue\n    }\n  | {\n      name: 'Management Type'\n      type: {\n        value: 'DROPDOWN'\n        key: 'VehoTech_Management_Type'\n      }\n      value: TurvoFlexAttributeManagementTypeValue\n    }\n)\n\nexport type TurvoEquipment = {\n  type: TurvoLookup\n  size: TurvoLookup | null\n}[]\n\nexport type TurvoCustomer = {\n  id: number\n  name: string\n}\n\nexport type TurvoCustomerOrder = {\n  id?: number\n  customerId?: number\n  customer: TurvoCustomer\n  customerOrderSourceId: number\n  externalIds: {\n    type: { key: '1400'; value: 'Purchase order #' } | { key: '1401'; value: 'Reference #' }\n    value: string\n    copyToCarrierOrder: true\n  }[]\n  items?: {\n    name: string\n    pickupLocation: { id: number; globalShipLocationSourceId?: number; globalShipLocationId?: number; name: string }[]\n    deliveryLocation: { id: number; globalShipLocationSourceId?: number; globalShipLocationId?: number; name: string }[]\n    qty?: number\n    unit?: TurvoLookup\n  }[]\n}\n\nexport type TurvoCarrierOrder = {\n  _operation: 0\n  carrier: {\n    id: number\n    name: string\n  }\n  carrierOrderSourceId: number\n  costs?: {\n    totalAmount: number\n    lineItem: {\n      id?: number\n      code: { key: '1600'; value: 'Freight - flat' }\n      qty: 1\n      price: number\n      amount: number\n      payable: true\n    }[]\n  }\n  deleted?: boolean\n}\n\nexport type TurvoAppointment = {\n  date: string\n  flex?: number // Apparently required but doesn't make sense I think, and the shipment creates without this\n  timeZone: string\n  hasTime: true\n}\n\nexport type TurvoShipment = {\n  id: number\n  customId: string\n  ltlShipment: boolean\n  startDate: TurvoDateWithTimezone\n  endDate: TurvoDateWithTimezone & { flex: number }\n  status: TurvoShipmentStatus\n  services?: TurvoShipmentService[]\n  equipment: TurvoEquipment\n  flexAttributes?: TurvoFlexAttribute[]\n  lane: TurvoLane\n  globalRoute: TurvoGlobalRoute\n  customerOrder: TurvoCustomerOrder[]\n  carrierOrder: TurvoCarrierOrder[]\n}\n\nexport type TurvoShipmentUpdate = {\n  id: number\n  customId: string\n  globalRoute: TurvoGlobalRoute\n  customerOrder: TurvoCustomerOrder[]\n  status: TurvoWebhookStatus\n}\n\nexport interface TurvoShipmentStatusApiResponse {\n  customId: string\n  status: {\n    location: {\n      lon: number\n      lat: number\n      nextArrivalTime: number\n      nextMiles: number\n      city: string\n    }\n    statusDate: {\n      date: string\n      timezone: string\n    }\n    runningLate?: {\n      nextLocationName: string\n    }\n    nextLocationId: number\n  }\n  globalRoute: TurvoGlobalRoute\n  lane: {\n    start: string\n    end: string\n  }\n}\n\nexport type TurvoShipmentStatusLocation = {\n  city: string\n  state: string\n  countryCode: string\n  countryName: string\n  lon: number\n  lat: number\n  nextEta?: string\n  nextEtaCalVal?: string\n  nextArrivalTime?: number\n  nextMiles?: number\n  currentDate?: string\n}\n\nexport type TurvoShipmentStatusRunningLate = {\n  lateType: string\n  lateDuration: number\n  nextLocationType: string\n  lateDurationString: string\n  nextLocationEta: string\n  nextLocationTimeZone: string\n  nextLocationName: string\n  nextLocationEtaDate: string\n  isApproachingLate: boolean\n  statusKey: string\n  nextLocationAppointmentDate: string\n  nextLocationAppointment: string\n  lastUpdatedOn: string\n  nextLocationId: number\n}\n\nexport type TurvoShipmentStatusUpdate = {\n  id: number\n  status: TurvoWebhookStatus\n}\n\n// Result types for specific operations\nexport type TurvoCreateShipmentApiResult = { id: number }\nexport type TurvoCancelShipmentApiResult = { respMsg: string }\nexport type TurvoUpdateShipmentStatusApiResult = { id: number }\nexport type TurvoAssociateTagsApiResult = {\n  tags: { id: string; tagName: string; contextType: string; contextId: number }[]\n}\n"]}
@@ -27,16 +27,18 @@ export interface LocationUpdate {
27
27
  * Individual stop in a shipment route
28
28
  */
29
29
  export interface Stop {
30
- sequence: number;
31
- stopType: string;
30
+ stopNumber: number;
32
31
  isCompleted: boolean;
33
- facilityCode: string | null;
34
- address: string | null;
35
- city: string | null;
36
- state: string | null;
37
- appointmentTimeUtc: string | null;
38
- actualArrivalUtc: string | null;
39
- actualDepartureUtc: string | null;
32
+ appointmentDateTime: string;
33
+ appointmentLocalTimeZone: string;
34
+ locationName: string;
35
+ externalLocationId: string;
36
+ facilityId: string | null;
37
+ externalStopId: string | null;
38
+ stopServiceType: string;
39
+ etaToStop: string | null;
40
+ milesRemainingToStop: number | null;
41
+ totalMilesToStop: number | null;
40
42
  }
41
43
  /**
42
44
  * Complete shipment tracking information combining public and internal API data
@@ -45,7 +47,7 @@ export interface ShipmentTracking {
45
47
  shipmentId: string;
46
48
  loadReferenceNumber: string;
47
49
  status: string;
48
- lateBy: string | null;
50
+ lateByMinutes: number | null;
49
51
  isLate: boolean;
50
52
  currentLocation: TurvoShipmentStatusLocation | null;
51
53
  hasGpsTracking: boolean;
@@ -55,7 +57,7 @@ export interface ShipmentTracking {
55
57
  etaUtc: string | null;
56
58
  etaTimezone: string | null;
57
59
  milesRemaining: number | null;
58
- totalMiles: number | null;
60
+ totalMilesToNextStop: number | null;
59
61
  locationUpdates: LocationUpdate[] | null;
60
62
  pingCount: number | null;
61
63
  lastPingAt: string | null;
@@ -1,2 +1,2 @@
1
1
  export {};
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2tpbmcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdHlwZXMvdHJhY2tpbmcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFR1cnZvU2hpcG1lbnRTdGF0dXNMb2NhdGlvbiB9IGZyb20gJy4vc2hpcG1lbnQnXG5cbi8qKlxuICogT3B0aW9ucyBmb3IgZmV0Y2hpbmcgc2hpcG1lbnQgdHJhY2tpbmcgZGF0YVxuICovXG5leHBvcnQgaW50ZXJmYWNlIEdldFRyYWNraW5nT3B0aW9ucyB7XG4gIC8qKlxuICAgKiBJbmNsdWRlIEdQUyBwaW5nIGhpc3RvcnkgaW4gcmVzcG9uc2UuXG4gICAqIFJlcXVpcmVzIGludGVybmFsIEFQSSBjYWxsIC0gbWF5IGJlIHNsb3dlci5cbiAgICogRGVmYXVsdDogdHJ1ZVxuICAgKi9cbiAgaW5jbHVkZUdwcz86IGJvb2xlYW5cblxuICAvKipcbiAgICogT3ZlcnJpZGUgdGhlIGRlZmF1bHQgc2VjcmV0IHBhdGggKC90dXJ2by9hcGkpLlxuICAgKiBVc2VmdWwgZm9yIHRlc3Rpbmcgb3IgZGlmZmVyZW50IGVudmlyb25tZW50cy5cbiAgICovXG4gIHNlY3JldFBhdGg/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiBHUFMgbG9jYXRpb24gdXBkYXRlIGZyb20gc2hpcG1lbnQgdHJhY2tpbmdcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBMb2NhdGlvblVwZGF0ZSB7XG4gIGxhdGl0dWRlOiBudW1iZXJcbiAgbG9uZ2l0dWRlOiBudW1iZXJcbiAgdGltZXN0YW1wOiBzdHJpbmdcbn1cblxuLyoqXG4gKiBJbmRpdmlkdWFsIHN0b3AgaW4gYSBzaGlwbWVudCByb3V0ZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN0b3Age1xuICBzZXF1ZW5jZTogbnVtYmVyXG4gIHN0b3BUeXBlOiBzdHJpbmcgLy8gXCJQaWNrdXBcIiwgXCJEZWxpdmVyeVwiXG4gIGlzQ29tcGxldGVkOiBib29sZWFuXG4gIGZhY2lsaXR5Q29kZTogc3RyaW5nIHwgbnVsbCAvLyBcIlBITCAwMVwiLCBcIkRGVyAwMVwiXG4gIGFkZHJlc3M6IHN0cmluZyB8IG51bGxcbiAgY2l0eTogc3RyaW5nIHwgbnVsbFxuICBzdGF0ZTogc3RyaW5nIHwgbnVsbFxuICBhcHBvaW50bWVudFRpbWVVdGM6IHN0cmluZyB8IG51bGxcbiAgYWN0dWFsQXJyaXZhbFV0Yzogc3RyaW5nIHwgbnVsbFxuICBhY3R1YWxEZXBhcnR1cmVVdGM6IHN0cmluZyB8IG51bGxcbn1cblxuLyoqXG4gKiBDb21wbGV0ZSBzaGlwbWVudCB0cmFja2luZyBpbmZvcm1hdGlvbiBjb21iaW5pbmcgcHVibGljIGFuZCBpbnRlcm5hbCBBUEkgZGF0YVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFNoaXBtZW50VHJhY2tpbmcge1xuICAvLyBJZGVudGlmaWVyc1xuICBzaGlwbWVudElkOiBzdHJpbmdcbiAgbG9hZFJlZmVyZW5jZU51bWJlcjogc3RyaW5nXG5cbiAgLy8gU3RhdHVzXG4gIHN0YXR1czogc3RyaW5nIC8vIFwiRW4gcm91dGVcIiwgXCJBdCBwaWNrdXBcIiwgXCJDb21wbGV0ZVwiXG4gIGxhdGVCeTogc3RyaW5nIHwgbnVsbCAvLyBcIjIgaG91cnNcIiwgXCIxNSBtaW5zXCIgKG51bGwgaWYgb24gdGltZSlcbiAgaXNMYXRlOiBib29sZWFuXG5cbiAgLy8gQ3VycmVudCBsb2NhdGlvbiAoaWYgdHJhY2tpbmcgYXZhaWxhYmxlKVxuICBjdXJyZW50TG9jYXRpb246IFR1cnZvU2hpcG1lbnRTdGF0dXNMb2NhdGlvbiB8IG51bGxcbiAgaGFzR3BzVHJhY2tpbmc6IGJvb2xlYW5cblxuICAvLyBQcm9ncmVzc1xuICBjb21wbGV0ZWRTdG9wczogbnVtYmVyXG4gIHRvdGFsU3RvcHM6IG51bWJlclxuXG4gIC8vIFN0b3BzIChvcmRlcmVkKVxuICBzdG9wczogU3RvcFtdXG5cbiAgLy8gRVRBICYgRGlzdGFuY2VcbiAgZXRhVXRjOiBzdHJpbmcgfCBudWxsXG4gIGV0YVRpbWV6b25lOiBzdHJpbmcgfCBudWxsXG4gIG1pbGVzUmVtYWluaW5nOiBudW1iZXIgfCBudWxsXG4gIHRvdGFsTWlsZXM6IG51bWJlciB8IG51bGxcblxuICAvLyBHUFMgUGluZ3MgKG9ubHkgcG9wdWxhdGVkIGlmIGluY2x1ZGVHcHM6IHRydWUgYW5kIGhhc0dwc1RyYWNraW5nKVxuICBsb2NhdGlvblVwZGF0ZXM6IExvY2F0aW9uVXBkYXRlW10gfCBudWxsXG4gIHBpbmdDb3VudDogbnVtYmVyIHwgbnVsbFxuICBsYXN0UGluZ0F0OiBzdHJpbmcgfCBudWxsXG5cbiAgLy8gU2hpcG1lbnQgQXR0cmlidXRlc1xuICBsYW5lVHlwZTogc3RyaW5nIHwgbnVsbCAvLyBcIk1pZGRsZSBNaWxlXCIsIFwiRmlyc3QgTWlsZVwiLCBcIlJldHVyblwiXG4gIG1hbmFnZW1lbnRUeXBlOiBzdHJpbmcgfCBudWxsIC8vIFwiVmVobyBNYW5hZ2VkXCIsIFwiQ2xpZW50IE1hbmFnZWRcIlxuICBjYXJyaWVyOiBzdHJpbmcgfCBudWxsXG59XG4iXX0=
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJhY2tpbmcuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvdHlwZXMvdHJhY2tpbmcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFR1cnZvU2hpcG1lbnRTdGF0dXNMb2NhdGlvbiB9IGZyb20gJy4vc2hpcG1lbnQnXG5cbi8qKlxuICogT3B0aW9ucyBmb3IgZmV0Y2hpbmcgc2hpcG1lbnQgdHJhY2tpbmcgZGF0YVxuICovXG5leHBvcnQgaW50ZXJmYWNlIEdldFRyYWNraW5nT3B0aW9ucyB7XG4gIC8qKlxuICAgKiBJbmNsdWRlIEdQUyBwaW5nIGhpc3RvcnkgaW4gcmVzcG9uc2UuXG4gICAqIFJlcXVpcmVzIGludGVybmFsIEFQSSBjYWxsIC0gbWF5IGJlIHNsb3dlci5cbiAgICogRGVmYXVsdDogdHJ1ZVxuICAgKi9cbiAgaW5jbHVkZUdwcz86IGJvb2xlYW5cblxuICAvKipcbiAgICogT3ZlcnJpZGUgdGhlIGRlZmF1bHQgc2VjcmV0IHBhdGggKC90dXJ2by9hcGkpLlxuICAgKiBVc2VmdWwgZm9yIHRlc3Rpbmcgb3IgZGlmZmVyZW50IGVudmlyb25tZW50cy5cbiAgICovXG4gIHNlY3JldFBhdGg/OiBzdHJpbmdcbn1cblxuLyoqXG4gKiBHUFMgbG9jYXRpb24gdXBkYXRlIGZyb20gc2hpcG1lbnQgdHJhY2tpbmdcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBMb2NhdGlvblVwZGF0ZSB7XG4gIGxhdGl0dWRlOiBudW1iZXJcbiAgbG9uZ2l0dWRlOiBudW1iZXJcbiAgdGltZXN0YW1wOiBzdHJpbmdcbn1cblxuLyoqXG4gKiBJbmRpdmlkdWFsIHN0b3AgaW4gYSBzaGlwbWVudCByb3V0ZVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN0b3Age1xuICBzdG9wTnVtYmVyOiBudW1iZXJcbiAgaXNDb21wbGV0ZWQ6IGJvb2xlYW5cbiAgYXBwb2ludG1lbnREYXRlVGltZTogc3RyaW5nXG4gIGFwcG9pbnRtZW50TG9jYWxUaW1lWm9uZTogc3RyaW5nXG4gIGxvY2F0aW9uTmFtZTogc3RyaW5nXG4gIGV4dGVybmFsTG9jYXRpb25JZDogc3RyaW5nXG4gIGZhY2lsaXR5SWQ6IHN0cmluZyB8IG51bGxcbiAgZXh0ZXJuYWxTdG9wSWQ6IHN0cmluZyB8IG51bGxcbiAgc3RvcFNlcnZpY2VUeXBlOiBzdHJpbmcgLy8gXCJwaWNrdXBcIiBvciBcImRlbGl2ZXJ5XCJcbiAgZXRhVG9TdG9wOiBzdHJpbmcgfCBudWxsXG4gIG1pbGVzUmVtYWluaW5nVG9TdG9wOiBudW1iZXIgfCBudWxsXG4gIHRvdGFsTWlsZXNUb1N0b3A6IG51bWJlciB8IG51bGxcbn1cblxuLyoqXG4gKiBDb21wbGV0ZSBzaGlwbWVudCB0cmFja2luZyBpbmZvcm1hdGlvbiBjb21iaW5pbmcgcHVibGljIGFuZCBpbnRlcm5hbCBBUEkgZGF0YVxuICovXG5leHBvcnQgaW50ZXJmYWNlIFNoaXBtZW50VHJhY2tpbmcge1xuICAvLyBJZGVudGlmaWVyc1xuICBzaGlwbWVudElkOiBzdHJpbmdcbiAgbG9hZFJlZmVyZW5jZU51bWJlcjogc3RyaW5nXG5cbiAgLy8gU3RhdHVzXG4gIHN0YXR1czogc3RyaW5nIC8vIFwiRW4gcm91dGVcIiwgXCJBdCBwaWNrdXBcIiwgXCJDb21wbGV0ZVwiXG4gIGxhdGVCeU1pbnV0ZXM6IG51bWJlciB8IG51bGxcbiAgaXNMYXRlOiBib29sZWFuXG5cbiAgLy8gQ3VycmVudCBsb2NhdGlvbiAoaWYgdHJhY2tpbmcgYXZhaWxhYmxlKVxuICBjdXJyZW50TG9jYXRpb246IFR1cnZvU2hpcG1lbnRTdGF0dXNMb2NhdGlvbiB8IG51bGxcbiAgaGFzR3BzVHJhY2tpbmc6IGJvb2xlYW5cblxuICAvLyBQcm9ncmVzc1xuICBjb21wbGV0ZWRTdG9wczogbnVtYmVyXG4gIHRvdGFsU3RvcHM6IG51bWJlclxuXG4gIC8vIFN0b3BzIChvcmRlcmVkKVxuICBzdG9wczogU3RvcFtdXG5cbiAgLy8gRVRBICYgRGlzdGFuY2VcbiAgZXRhVXRjOiBzdHJpbmcgfCBudWxsXG4gIGV0YVRpbWV6b25lOiBzdHJpbmcgfCBudWxsXG4gIG1pbGVzUmVtYWluaW5nOiBudW1iZXIgfCBudWxsXG4gIHRvdGFsTWlsZXNUb05leHRTdG9wOiBudW1iZXIgfCBudWxsXG5cbiAgLy8gR1BTIFBpbmdzIChvbmx5IHBvcHVsYXRlZCBpZiBpbmNsdWRlR3BzOiB0cnVlIGFuZCBoYXNHcHNUcmFja2luZylcbiAgbG9jYXRpb25VcGRhdGVzOiBMb2NhdGlvblVwZGF0ZVtdIHwgbnVsbFxuICBwaW5nQ291bnQ6IG51bWJlciB8IG51bGxcbiAgbGFzdFBpbmdBdDogc3RyaW5nIHwgbnVsbFxuXG4gIC8vIFNoaXBtZW50IEF0dHJpYnV0ZXNcbiAgbGFuZVR5cGU6IHN0cmluZyB8IG51bGwgLy8gXCJNaWRkbGUgTWlsZVwiLCBcIkZpcnN0IE1pbGVcIiwgXCJSZXR1cm5cIlxuICBtYW5hZ2VtZW50VHlwZTogc3RyaW5nIHwgbnVsbCAvLyBcIlZlaG8gTWFuYWdlZFwiLCBcIkNsaWVudCBNYW5hZ2VkXCJcbiAgY2Fycmllcjogc3RyaW5nIHwgbnVsbFxufVxuIl19
@@ -1 +1 @@
1
- {"root":["../src/constants.ts","../src/index.ts","../src/api/turvointernalapi.ts","../src/api/turvopublicapi.ts","../src/client/turvoclient.ts","../src/shipmenttracking/index.ts","../src/shipmenttracking/trackingservice.ts","../src/types/common.ts","../src/types/config.ts","../src/types/errors.ts","../src/types/index.ts","../src/types/shipment.ts","../src/types/tracking.ts"],"version":"5.9.3"}
1
+ {"root":["../src/constants.ts","../src/index.ts","../src/api/turvointernalapi.ts","../src/api/turvopublicapi.ts","../src/client/turvoclient.ts","../src/db/client.ts","../src/db/facilitymapping.ts","../src/shipmenttracking/index.ts","../src/shipmenttracking/trackingservice.ts","../src/types/common.ts","../src/types/config.ts","../src/types/errors.ts","../src/types/facilitymapping.ts","../src/types/index.ts","../src/types/shipment.ts","../src/types/tracking.ts"],"version":"5.9.3"}
@@ -1 +1 @@
1
- {"root":["../src/constants.ts","../src/index.ts","../src/api/turvointernalapi.ts","../src/api/turvopublicapi.ts","../src/client/turvoclient.ts","../src/shipmenttracking/index.ts","../src/shipmenttracking/trackingservice.ts","../src/types/common.ts","../src/types/config.ts","../src/types/errors.ts","../src/types/index.ts","../src/types/shipment.ts","../src/types/tracking.ts"],"version":"5.9.3"}
1
+ {"root":["../src/constants.ts","../src/index.ts","../src/api/turvointernalapi.ts","../src/api/turvopublicapi.ts","../src/client/turvoclient.ts","../src/db/client.ts","../src/db/facilitymapping.ts","../src/shipmenttracking/index.ts","../src/shipmenttracking/trackingservice.ts","../src/types/common.ts","../src/types/config.ts","../src/types/errors.ts","../src/types/facilitymapping.ts","../src/types/index.ts","../src/types/shipment.ts","../src/types/tracking.ts"],"version":"5.9.3"}
package/package.json CHANGED
@@ -34,6 +34,7 @@
34
34
  },
35
35
  "devDependencies": {
36
36
  "@types/jest": "^30.0.0",
37
+ "@types/luxon": "^3.4.2",
37
38
  "@types/node": "^22",
38
39
  "@types/node-fetch": "^2.6.11",
39
40
  "@typescript-eslint/eslint-plugin": "^6",
@@ -57,7 +58,10 @@
57
58
  "undici-types": "^6.21.0"
58
59
  },
59
60
  "dependencies": {
61
+ "@aws-sdk/client-dynamodb": "^3.574.0",
62
+ "@aws-sdk/lib-dynamodb": "^3.574.0",
60
63
  "lambda-params-secrets": "^1.0.3",
64
+ "luxon": "^3.5.0",
61
65
  "node-fetch": "^2.7.0",
62
66
  "p-retry": "^6.2.0",
63
67
  "p-throttle": "^6.1.0",
@@ -68,7 +72,7 @@
68
72
  },
69
73
  "main": "lib/cjs/index.js",
70
74
  "license": "UNLICENSED",
71
- "version": "0.1.0-beta.0",
75
+ "version": "0.1.0-beta.1",
72
76
  "jest": {
73
77
  "coverageProvider": "v8",
74
78
  "transform": {