How to Add a Fulfillment Provider
In this document, you’ll learn how to add a fulfillment provider to a Medusa backend. If you’re unfamiliar with the Shipping architecture in Medusa, make sure to check out the overview first.
Overview
A fulfillment provider is the shipping provider used to fulfill orders and deliver them to customers. An example of a fulfillment provider is FedEx.
By default, a Medusa Backend has a manual
Copy to Clipboard fulfillment provider which has minimal implementation. It allows you to accept orders and fulfill them manually. However, you can integrate any fulfillment provider into Medusa, and your fulfillment provider can interact with third-party shipping providers.
Adding a fulfillment provider is as simple as creating one service file in src/services
Copy to Clipboard. A fulfillment provider is essentially a service that extends the FulfillmentService
Copy to Clipboard. It requires implementing 4 methods:
getFulfillmentOptions
Copy to Clipboard: used to retrieve available fulfillment options provided by this fulfillment provider.validateOption
Copy to Clipboard: used to validate the shipping option when it’s being created by the admin.validateFulfillmentData
Copy to Clipboard: used to validate the shipping method when the customer chooses a shipping option on checkout.createFulfillment
Copy to Clipboard: used to perform any additional actions when fulfillment is being created for an order.
Also, the fulfillment provider class should have a static property identifier
Copy to Clipboard. It is the name that will be used to install and refer to the fulfillment provider throughout Medusa.
Fulfillment providers are loaded and installed on the backend startup.
Create a Fulfillment Provider
The first step is to create a JavaScript or TypeScript file under src/services
Copy to Clipboard. For example, create the file src/services/my-fulfillment.ts
Copy to Clipboard with the following content:
Fulfillment provider services must extend the FulfillmentService
Copy to Clipboard class imported from medusa-interfaces
Copy to Clipboard.
Following the naming convention of Services, the name of the file should be the slug name of the fulfillment provider, and the name of the class should be the camel case name of the fulfillment provider suffixed with “Service”. You can learn more in the service documentation.
Identifier
As mentioned in the overview, fulfillment providers should have a static identifier
Copy to Clipboard property.
The FulfillmentProvider
Copy to Clipboard entity has 2 properties: identifier
Copy to Clipboard and is_installed
Copy to Clipboard. The identifier
Copy to Clipboard property in the class will be used when the fulfillment provider is created in the database.
The value of this property will also be used to reference the fulfillment provider throughout Medusa. For example, it is used to add a fulfillment provider to a region.
constructor
You can use the constructor
Copy to Clipboard of your fulfillment provider to have access to different services in Medusa through dependency injection. You can access any services you create in Medusa in the first parameter.
You can also use the constructor to initialize your integration with the third-party provider. For example, if you use a client to connect to the third-party provider’s APIs, you can initialize it in the constructor and use it in other methods in the service.
Additionally, if you’re creating your fulfillment provider as an external plugin to be installed on any Medusa backend and you want to access the options added for the plugin, you can access it in the constructor. The options are passed as a second parameter.
For example:
getFulfillmentOptions
When the admin is creating shipping options available for customers during checkout, they choose one of the fulfillment options provided by underlying fulfillment providers.
For example, if you’re integrating UPS as a fulfillment provider, you might support two fulfillment options: UPS Express Shipping and UPS Access Point.
These fulfillment options are defined in the getFulfillmentOptions
Copy to Clipboard method. This method should return an array of options.
For example:
When the admin chooses one of those fulfillment options, the data of the chosen fulfillment option is stored in the data
Copy to Clipboard property of the shipping option created. This property is used to add any additional data you need to fulfill the order with the third-party provider.
For that reason, the fulfillment option doesn't have any required structure and can be of any format that works for your integration.
validateOption
Once the admin creates the shipping option, the data will be validated first using this method in the underlying fulfillment provider of that shipping option. This method is called when a POST
Copy to Clipboard request is sent to /admin/shipping-options
Copy to Clipboard.
This method accepts the data
Copy to Clipboard object that is sent in the body of the request. You can use this data to validate the shipping option before it is saved.
This method returns a boolean. If the result is false, an error is thrown and the shipping option will not be saved.
For example, you can use this method to ensure that the id
Copy to Clipboard in the data
Copy to Clipboard object is correct:
If your fulfillment provider does not need to run any validation, you can simply return true
Copy to Clipboard.
validateFulfillmentOption
When the customer chooses a shipping option on checkout, the shipping option and its data are validated before the shipping method is created.
validateFulfillmentOption
Copy to Clipboard is called when a POST
Copy to Clipboard request is sent to /carts/:id/shipping-methods
Copy to Clipboard.
This method accepts three parameters:
- The shipping option data.
- The
data
Copy to Clipboard object passed in the body of the request. - The customer’s cart data.
You can use these parameters to validate the chosen shipping option. For example, you can check if the data
Copy to Clipboard object includes all data needed to fulfill the shipment later on.
If any of the data is invalid, you can throw an error. This error will stop Medusa from creating a shipping method and the error message will be returned as a result to the endpoint.
If everything is valid, this method must return a value that will be stored in the data
Copy to Clipboard property of the shipping method to be created. So, make sure the value you return contains everything you need to fulfill the shipment later on.
For example:
createFulfillment
After an order is placed, it can be fulfilled either manually by the admin or using automation.
This method gives you access to the fulfillment being created as well as other details in case you need to perform any additional actions with the third-party provider.
This method accepts four parameters:
- The data of the shipping method associated with the order.
- An array of items in the order to be fulfilled. The admin can choose all or some of the items to fulfill.
- The data of the order
- The data of the fulfillment being created.
You can use the data
Copy to Clipboard property in the shipping method (first parameter) to access the data specific to the shipping option. This is based on your implementation of previous methods.
Here is a basic implementation of createFulfillment
Copy to Clipboard for a fulfillment provider that does not interact with any third-party provider to create the fulfillment:
This method is also used to create claims and swaps. The fulfillment object has the fields claim_id
Copy to Clipboard, swap_id
Copy to Clipboard, and order_id
Copy to Clipboard. You can check which isn’t null to determine what type of fulfillment is being created.
Useful Methods
The above-detailed methods are the required methods for every fulfillment provider. However, there are additional methods that you can use in your fulfillment provider to customize it further or add additional features.
canCalculate
This method validates whether a shipping option is calculated dynamically or flat rate. It is called if the price_type
Copy to Clipboard of the shipping option being created is set to calculated
Copy to Clipboard.
If this method returns true
Copy to Clipboard, that means that the price should be calculated dynamically. The amount
Copy to Clipboard property of the shipping option will then be set to null
Copy to Clipboard. The amount will be created later when the shipping method is created on checkout using the calculatePrice
Copy to Clipboard method (explained next).
If the method returns false
Copy to Clipboard, an error is thrown as it means the selected shipping option can only be chosen if the price type is set to flat_rate
Copy to Clipboard.
This method receives as a parameter the data
Copy to Clipboard object sent with the request that creates the shipping option. You can use this data to determine whether the shipping option should be calculated or not. This is useful if the fulfillment provider you are integrating has both flat rate and dynamically priced fulfillment options.
For example:
calculatePrice
This method is called on checkout when the shipping method is being created if the price_type
Copy to Clipboard of the selected shipping option is calculated
Copy to Clipboard.
This method receives three parameters:
- The
data
Copy to Clipboard parameter of the selected shipping option. - The
data
Copy to Clipboard parameter sent with the request. - The customer’s cart data.
If your fulfillment provider does not provide any dynamically calculated rates you can keep the function empty:
Otherwise, you can use it to calculate the price with a custom logic. For example:
createReturn
Fulfillment providers can also be used to return products. A shipping option can be used for returns if the is_return
Copy to Clipboard property is true
Copy to Clipboard or if an admin creates a Return Shipping Option from the settings.
This method is called when the admin creates a return request for an order or when the customer creates a return of their order.
It gives you access to the return being created in case you need to perform any additional actions with the third-party provider.
It receives the return created as a parameter. The value it returns is set to the shipping_data
Copy to Clipboard of the return instance.
This is the basic implementation of the method for a fulfillment provider that does not contact with a third-party provider to fulfill the return:
cancelFulfillment
This method is called when a fulfillment is cancelled by the admin. This fulfillment can be for an order, a claim, or a swap.
It gives you access to the fulfillment being canceled in case you need to perform any additional actions with your third-party provider.
This method receives the fulfillment being cancelled as a parameter.
This is the basic implementation of the method for a fulfillment provider that does not interact with a third-party provider to cancel the fulfillment:
See Also
- Example Implementations: Webshipper plugin and the manual fulfillment plugin