laravel-nepal / nepal-can-move
SDK for integration with the Nepal Can Move API in PHP applications.
Fund package maintenance!
achyutkneupane
Patreon
Buy Me A Coffee
buymemomo.com/achyut
Installs: 189
Dependents: 0
Suggesters: 0
Security: 0
Stars: 0
Watchers: 0
Forks: 0
Open Issues: 0
pkg:composer/laravel-nepal/nepal-can-move
Requires
- php: ^8.2|^8.3|^8.4
- guzzlehttp/guzzle: ^7.10
- illuminate/http: ^10.0|^11.0|^12.0
- illuminate/support: ^10.0|^11.0|^12.0
Requires (Dev)
- laravel/pint: ^1.27
- pestphp/pest: ^3.8|^4.3
- phpstan/phpstan: ^2.1
- rector/rector: ^2.3
This package is auto-updated.
Last update: 2026-02-19 10:35:06 UTC
README
A strictly typed, and expressive PHP SDK for integrating with the Nepal Can Move (NCM) logistics API.
This package provides a fluent wrapper around the NCM API, normalizing API responses into clean Data Transfer Objects (DTOs), leveraging PHP 8.1+ Enums for type safety, and handling all authentication and error states automatically.
Requirements
- PHP: 8.2+
- Laravel: 10.x, 11.x, or 12.x
Installation
Install the package via Composer:
composer require laravel-nepal/nepal-can-move
(Optional) Publish the configuration file:
php artisan vendor:publish --tag="nepal-can-move"
Configuration
Add your NCM credentials to your .env file. Enable sandbox mode for testing.
NCM_TOKEN=your_api_token_here NCM_SANDBOX_MODE=true
You can get your token by registering a vendor account then from the portal dashboard.
Usage
Warning
API Limits
Please be mindful of the NCM API limits to avoid IP throttling:
Order Creation: 1,000 per day.
Order Views (Detail/Status): 20,000 per day.
Fetching Branches
Retrieve all available NCM branches as strictly typed Branch objects.
use LaravelNepal\NCM\Facades\NCM; $branches = NCM::getBranches(); $tinkune = $branches->firstWhere('name', 'TINKUNE'); echo $tinkune->phone;
Calculating Shipping Rates
Calculate delivery charges using the DeliveryType enum.
use LaravelNepal\NCM\Enums\DeliveryType; $charge = NCM::getDeliveryCharge( source: $tinkune, destination: $pokhara, deliveryType: DeliveryType::BranchToDoor );
Creating an Order
Use the CreateOrderRequest DTO to ensure all required fields are present.
use LaravelNepal\NCM\Data\CreateOrderRequest; use LaravelNepal\NCM\Enums\DeliveryType; $request = new CreateOrderRequest( name: 'Achyut Neupane', phone: '9800000000', codCharge: '1500', address: 'Lakeside, Pokhara', sourceBranch: 'KATHMANDU', destinationBranch: 'POKHARA', package: 'Books', deliveryType: DeliveryType::DoorToDoor ); $order = NCM::createOrder($request); echo $order->id;
Important
The NCM API uses different naming conventions across its endpoints (e.g., Pickup/Collect vs Door2Door).
This SDK normalizes these into the DeliveryType enum. You should always use the Enum cases; the SDK handles the underlying API string transformations automatically.
Order Management
The Order object exposes rich behavior.
Fetching an Order
$order = NCM::getOrder(12345);
Status History
$history = $order->statusHistory(); foreach ($history as $status) { echo $status->status . ' - ' . $status->addedTime->diffForHumans(); }
Latest Order Status
echo $order->status();
Comments
$order->addComment('Customer requested evening delivery.'); $comments = $order->comments();
Return & Exchange
$order->return('Customer refused delivery'); $order->exchange();
Redirecting an Order
use LaravelNepal\NCM\Data\RedirectOrderRequest; $redirect = new RedirectOrderRequest( orderId: $order->id, name: 'Not Achyut Neupane', phone: '9811111111', address: 'New Address, Kathmandu', orderIdentifier: 'ORD-REF-002', destinationBranchId: 5, codCharge: 1600.00 ); NCM::redirectOrder($redirect);
Support Tickets
General Support
use LaravelNepal\NCM\Enums\TicketType; NCM::createSupportTicket( TicketType::OrderProcessing, "My order is stuck in Pickup status for 3 days." );
COD Transfer Request
$ticketId = NCM::createCODTransferTicket( bankName: 'Nabil Bank', accountHolderName: 'My Company Pvt Ltd', accountNumber: '001001001001' );
Webhooks
The Nepal Can Move SDK allows you to manage your webhook configuration and transform incoming POST data into strictly typed objects.
Configuring Webhooks
You can programmatically set, test, or remove your webhook URL:
// Set the URL where NCM will push updates $ncm->setWebhookUrl('https://your-app.com/api/ncm/webhook'); // Send a test payload to your URL to verify connectivity $ncm->testWebhookUrl('https://your-app.com/api/ncm/webhook'); // Disable webhooks $ncm->removeWebhookUrl();
Handling Webhook Payloads
When NCM sends a status update to your server, use parseWebhook to convert the raw request data into a StatusEvent DTO. This automatically maps technical events to your OrderStatus enums using the toOrderStatus method.
use LaravelNepal\NCM\Exceptions\NCMException; use LaravelNepal\NCM\Enums\OrderStatus; try { $event = $ncm->parseWebhook($payload); echo $event->orderIds; // e.g., [123, 124] echo $event->event->getLabel(); // e.g., "Delivered" // Get the normalized OrderStatus enum $status = $event->getOrderStatus(); if ($status === OrderStatus::Delivered) { // Perform business logic } } catch (NCMException $NCMException) { // Handle unknown event types or malformed data }
Tip
Since NCM webhooks do not currently include a cryptographic signature, it is recommended to add a unique query parameter to your webhook URL (e.g., ?secret=your-random-key) and verify it in your controller to ensure the request is legitimate.
Exception Handling
use LaravelNepal\NCM\Exceptions\NCMException; try { NCM::createOrder($request); } catch (NCMException $e) { return back()->withErrors($e->getMessage()); }
Contributing
Contributions are welcome! Please create a pull request or open an issue if you find any bugs or have feature requests.
License
This package is open-sourced software licensed under the MIT license.
Support
If you find this package useful, please consider starring the repository on GitHub to show your support.