JSPM

@rytass/logistics

0.0.13
  • ESM via JSPM
  • ES Module Entrypoint
  • Export Map
  • Keywords
  • License
  • Repository URL
  • TypeScript Types
  • README
  • Created
  • Published
  • Downloads 11
  • Score
    100M100P100Q50711F
  • License MIT

Rytass Utils Logistics

Package Exports

  • @rytass/logistics
  • @rytass/logistics/index.js

This package does not declare an exports field, so the exports above have been automatically detected and optimized by JSPM instead. If any package subpath is missing, it is recommended to post an issue to the original package (@rytass/logistics) to support the "exports" field. If that is not possible, create a JSPM override to customize the exports field for this package.

Readme

Rytass Utils - Logistics

A unified interface for Taiwan logistics tracking services. Provides a standardized way to track packages across different logistics providers including TCAT (黑貓宅急便) and CTC (宅配通).

Features

  • Unified logistics tracking interface
  • Multiple logistics provider support
  • TypeScript with strict typing
  • Standardized status tracking
  • Batch tracking support
  • Status history tracking
  • Error handling

Installation

npm install @rytass/logistics
# or
yarn add @rytass/logistics

Available Adapters

Basic Usage

Using TCAT Adapter

import { TCATLogisticsService } from '@rytass/logistics-adapter-tcat';

const tcat = new TCATLogisticsService();

// Track single package
const results = await tcat.trace('123456789');
console.log('Tracking results:', results);

// Track multiple packages
const batchResults = await tcat.trace(['123456789', '987654321']);
console.log('Batch results:', batchResults);

Using CTC Adapter

import { CTCLogisticsService } from '@rytass/logistics-adapter-ctc';

const ctc = new CTCLogisticsService();

// Track package
const results = await ctc.trace('123456789');
console.log('CTC tracking:', results);

Core Concepts

LogisticsInterface

interface LogisticsInterface<T = LogisticsBaseStatus> {
  reference?: T;
  url: string;
}

LogisticsService

interface LogisticsService<LogisticsType extends LogisticsInterface<LogisticsStatus<LogisticsType>>> {
  trace(request: string): Promise<LogisticsTraceResponse<LogisticsType>[]>;
  trace(request: string[]): Promise<LogisticsTraceResponse<LogisticsType>[]>;
}

Status Types

type LogisticsBaseStatus = 'DELIVERED' | 'DELIVERING' | 'SHELVED';

Tracking Response

interface LogisticsTraceResponse<K extends LogisticsInterface<LogisticsStatus<K>>> {
  logisticsId: string;
  statusHistory: LogisticsStatusHistory<K['reference']>[];
}

interface LogisticsStatusHistory<T> {
  date: string;
  status: T;
}

Advanced Usage

Custom Logistics Service

import { LogisticsService, LogisticsInterface, LogisticsTraceResponse, LogisticsStatus } from '@rytass/logistics';

interface CustomLogisticsInterface extends LogisticsInterface<'PENDING' | 'SHIPPED' | 'DELIVERED'> {
  reference: 'PENDING' | 'SHIPPED' | 'DELIVERED';
  url: string;
}

class CustomLogisticsService implements LogisticsService<CustomLogisticsInterface> {
  async trace(request: string | string[]): Promise<LogisticsTraceResponse<CustomLogisticsInterface>[]> {
    const trackingNumbers = Array.isArray(request) ? request : [request];

    const results = await Promise.all(
      trackingNumbers.map(async trackingNumber => {
        // Implement your tracking logic here
        return {
          logisticsId: trackingNumber,
          statusHistory: [
            {
              date: '2024-01-01',
              status: 'PENDING' as const,
            },
            {
              date: '2024-01-02',
              status: 'SHIPPED' as const,
            },
          ],
        };
      }),
    );

    return results;
  }
}

Multi-Provider Tracking

import { TCATLogisticsService } from '@rytass/logistics-adapter-tcat';
import { CTCLogisticsService } from '@rytass/logistics-adapter-ctc';

class MultiProviderTracker {
  private tcat = new TCATLogisticsService();
  private ctc = new CTCLogisticsService();

  async trackPackage(trackingNumber: string, provider: 'TCAT' | 'CTC') {
    try {
      switch (provider) {
        case 'TCAT':
          return await this.tcat.trace(trackingNumber);
        case 'CTC':
          return await this.ctc.trace(trackingNumber);
        default:
          throw new Error(`Unsupported provider: ${provider}`);
      }
    } catch (error) {
      console.error(`Failed to track with ${provider}:`, error);
      throw error;
    }
  }

  async trackWithAllProviders(trackingNumber: string) {
    const results = await Promise.allSettled([
      this.tcat.trace(trackingNumber).catch(err => ({ provider: 'TCAT', error: err })),
      this.ctc.trace(trackingNumber).catch(err => ({ provider: 'CTC', error: err })),
    ]);

    return results.map((result, index) => ({
      provider: index === 0 ? 'TCAT' : 'CTC',
      result: result.status === 'fulfilled' ? result.value : result.reason,
    }));
  }
}

Integration Examples

Express.js API

import express from 'express';
import { TCATLogisticsService } from '@rytass/logistics-adapter-tcat';
import { CTCLogisticsService } from '@rytass/logistics-adapter-ctc';

const app = express();
const tcat = new TCATLogisticsService();
const ctc = new CTCLogisticsService();

app.get('/track/:provider/:trackingNumber', async (req, res) => {
  try {
    const { provider, trackingNumber } = req.params;

    let results;
    switch (provider.toUpperCase()) {
      case 'TCAT':
        results = await tcat.trace(trackingNumber);
        break;
      case 'CTC':
        results = await ctc.trace(trackingNumber);
        break;
      default:
        return res.status(400).json({ error: 'Unsupported provider' });
    }

    res.json({
      success: true,
      trackingNumber,
      provider: provider.toUpperCase(),
      results,
    });
  } catch (error) {
    res.status(500).json({
      success: false,
      error: error.message,
    });
  }
});

app.post('/track/batch', async (req, res) => {
  try {
    const { provider, trackingNumbers } = req.body;

    let results;
    switch (provider.toUpperCase()) {
      case 'TCAT':
        results = await tcat.trace(trackingNumbers);
        break;
      case 'CTC':
        results = await ctc.trace(trackingNumbers);
        break;
      default:
        return res.status(400).json({ error: 'Unsupported provider' });
    }

    res.json({
      success: true,
      provider: provider.toUpperCase(),
      results,
    });
  } catch (error) {
    res.status(500).json({
      success: false,
      error: error.message,
    });
  }
});

NestJS Service

import { Injectable } from '@nestjs/common';
import { TCATLogisticsService } from '@rytass/logistics-adapter-tcat';
import { CTCLogisticsService } from '@rytass/logistics-adapter-ctc';

@Injectable()
export class TrackingService {
  private tcat = new TCATLogisticsService();
  private ctc = new CTCLogisticsService();

  async trackPackage(trackingNumber: string, provider: string) {
    switch (provider.toUpperCase()) {
      case 'TCAT':
        return await this.tcat.trace(trackingNumber);
      case 'CTC':
        return await this.ctc.trace(trackingNumber);
      default:
        throw new Error(`Unsupported provider: ${provider}`);
    }
  }

  async batchTrack(trackingNumbers: string[], provider: string) {
    switch (provider.toUpperCase()) {
      case 'TCAT':
        return await this.tcat.trace(trackingNumbers);
      case 'CTC':
        return await this.ctc.trace(trackingNumbers);
      default:
        throw new Error(`Unsupported provider: ${provider}`);
    }
  }

  async getLatestStatus(trackingNumber: string, provider: string) {
    const results = await this.trackPackage(trackingNumber, provider);

    if (results.length === 0) {
      throw new Error('No tracking information found');
    }

    const latestResult = results[0];
    const latestStatus = latestResult.statusHistory[latestResult.statusHistory.length - 1];

    return {
      trackingNumber: latestResult.logisticsId,
      currentStatus: latestStatus.status,
      lastUpdated: latestStatus.date,
      fullHistory: latestResult.statusHistory,
    };
  }
}

Error Handling

import { LogisticsErrorInterface } from '@rytass/logistics';

async function safeTrack(trackingNumber: string) {
  try {
    const tcat = new TCATLogisticsService();
    const results = await tcat.trace(trackingNumber);
    return { success: true, results };
  } catch (error) {
    if (error as LogisticsErrorInterface) {
      return {
        success: false,
        error: {
          code: error.code,
          message: error.message,
        },
      };
    }

    return {
      success: false,
      error: {
        code: 'UNKNOWN_ERROR',
        message: error.message,
      },
    };
  }
}

API Reference

LogisticsService Interface

interface LogisticsService<LogisticsType> {
  trace(request: string): Promise<LogisticsTraceResponse<LogisticsType>[]>;
  trace(request: string[]): Promise<LogisticsTraceResponse<LogisticsType>[]>;
}

Types

Type Description
LogisticsBaseStatus 'DELIVERED' | 'DELIVERING' | 'SHELVED'
LogisticsInterface<T> Base interface for logistics providers
LogisticsTraceResponse<K> Tracking response with status history
LogisticsStatusHistory<T> Individual status entry with date
LogisticsErrorInterface Error interface with code and message

Best Practices

Error Handling

  • Always wrap tracking calls in try-catch blocks
  • Implement retry logic for transient failures
  • Validate tracking numbers before making API calls
  • Handle provider-specific error codes appropriately

Performance

  • Use batch tracking for multiple packages when possible
  • Implement caching for frequently tracked packages
  • Consider rate limiting to avoid provider API limits
  • Use appropriate timeouts for network requests

Data Management

  • Store tracking results with timestamps
  • Implement data retention policies
  • Consider privacy implications of tracking data
  • Validate and sanitize tracking numbers

Taiwan Logistics Providers

TCAT (黑貓宅急便)

  • One of the largest logistics providers in Taiwan
  • Supports package tracking via tracking number
  • Provides detailed status updates

CTC (宅配通)

  • Major Taiwan logistics and delivery service
  • Comprehensive tracking capabilities
  • Integration with e-commerce platforms

License

MIT