Rate Calculator using Fees-Creation
What is the purpose of this article?
This article aims to make the reader understand how the backend logic is used to calculate the shipping charges from the warehouse's location to the customer's location based on:
Shipment weight,
Warehouse and Customer pincodes,
Courier type (Standard, Express or Priority),
Journey type (Forward, Reverse or Return-to-Origin) and
Payment Method
Based on the shipping charges, clients can decide the courier they want to ship the order.
Examples: Courier_Rate_Calculator_.xlsx
Fees Creation
If the user wants to use different fees for a different client, he needs to create fees mentioned below :
If you want a detailed desiption please refer below link
Payments Team
Rate Calculator Figma
Code Implementation Details
GCP Project - Inventory Engine Client Portal
service - shipping service
GitLab Repo - Esb Shipping Service
link: https://gitlab.com/eshopbox-team/orders-and-shipping/esb-shipping-service
API Path(Staging) = POST https://tab.auperator.co/shipping/api/v1/calculate/rate
API Path(Prod) = POST https://blackberrys.myeshopbox.com/shipping/api/v1/calculate/rate
Request Body
{
//"accountId": 5,
"journeyType": "forward",// or return
"pickupPincode": "560034",
"dropPincode": "560034",
"orderWeight": "5",//in grams
"length": "12",//in cms
"width": "12",//in cms
"height": "33",//in cms
"paymentMethod": "Cash on delivery",//or prepaid
"codAmountToBeCollected": 100.0,//in Rs.
//"shippingMode": "eshopboxStandard",
//"zone":"local",
"doorstepQc": false//for return
}
Fetching Data
Verify the request body is correct.
Use fees-types table to create a map in Constants file
FEES_ID_NAME_MAP
to map every fees with correspondinglocalId
as the key.Fetch
portalId
of fromportals
table usingexternalPortalId=“EBSH"
DB: 35.200.149.199 ➤
eshopbox_client_portal_dev
Using
accountId
,portalId
andfeesNameList
fetch fees details fromfees
table andfee_portal_mapping
table using below queryDB: 35.200.133.164 ➤
eshopbox_wms_dev
SELECT fee_portal_mapping.id, fee_portal_mapping.fee_id, fee_portal_mapping.version, fee_portal_mapping.is_inclusive_tax, fee_portal_mapping.inclusive_tax_percentage, fee_portal_mapping.fixedApplicableOn, fee_portal_mapping.percentageApplicableOn, fee_portal_mapping.dependsOn FROM fees LEFT JOIN fee_portal_mapping ON fee_portal_mapping.fee_id = fees.id WHERE fees.portal_id = :portalId AND fees.account_id = :accountId AND (fees.name IN (:feesNameList) OR fees.actual_name IN (:feesNameList))
Using the
portalId
,fee_id
and andversion
from table to fetch latest and actual fees values fromper_fee_rule
table in Mongo DB ➤payment-reco-dev
If the total count of fees is not 25 in either of step 4 or 5, then also fetch fees for
accountId=5
.Use the keys from
FEES_ID_NAME_MAP
and above values from mongo server to create afeesValueMap
map.Sample data structure in Mongo:
{ "_id": { "ObjectId": "660be783d8dc732d617dffba" }, "paymentMethod": "cod", "fixed": 54, "fixedApplicableOn": "Per Shipment", "percentage": 1.5, "percentageApplicableOn": "Per Shipment", "feeRuleConditionId": "66069f550b2fd0251e578ad0", "feeRulesStatus": "", "shippingZone": "local", "weightrange": { "unit": "kg", "min": 0, "max": 0.5 }, "variable": 22, "variableApplicableOn": "Per Shipment", "partitionRange": 0.5, "partitionRangeType": "kg", "feePortalId": 1499, "version": 1.1, "feeId": 315, "portalId": 51 }
We have to use either fixed, percentage or variable key value
Base-Rate-Calculation
Based on pickup pincode and drop pincode, calculate the zone if not present in request body.
Based on the zone and order weight, fetch correct fees from
feesValueMap
Example: If Weight = 3600 gm & Zone='Metro' then fees Type used will be:
From above fees types we know starting keys is 8
Run a outer loop on the fees fetched, till weight reaches in correct weight range and in each iteration:
Calculate number of slabs =
(maxWeight - minWeight) / partitionRange
Run a inner loop on the slabs
add weight for each slab and
add actual value i.e.
variable
in this caseif added_weight >= actual_weight then break the outer loop
Additional Fees Creation
Add fuel and base charges to
shippingCharges
as it is not dependent on any parameterIf courier type = express or priority, then
Calculate and add express charges to
shippingCharges
Set express charges
If courier type =”standard”, then
Set express charges = 0
Add
estimatedDeliveryTimeInDays
based on the courier type and data fromzone_courier_type_tats
table in wms database.If payment method = “cod” or “cash on delivery”, then
Calculate max of either fixed COD charges or percent of shipment invoice (
codAmountToBeCollected
)Add COD charges to
shippingCharges
Set COD charges
If payment method = “prepaid”
Set COD charges = 0
If journey type = “forward” or “return to origin” then
Set Quality Check charges = 0, and
Set Customer initiated return charges = 0
In case of “return to origin”,
substract the cod charges(if added) from
shippingCharges
Double the
shippingCharges
and store inshippingCharges
If journey type = “reverse”, then,
If Quality check at the doorstep is true, then
Calculate and add Quality check charges to
shippingCharges
Set quality check charges
Calculate and add Customer initiated return charges to
shippingCharges
Set Customer initiated return charges
Set
shippingCharges
Calculate 18.0% GST of
shippingCharges
Set Total charges =
shippingCharges
+ GST
Response
{
"zone": "Metro",
"essentialPlan": {
"eshopboxStandard": {
"estimatedDeliveryDays": 5,
"shippingBaseFreight": 0.0,
"expressSurcharge": 0.0,
"CODCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 0.0,
"doorstepQCFees": 0.0,
"GST": 6.3,
"totalShippingCharges": 41.3
}
}
}
Possible response with eshop auth token
{
"zone": "Metro",
"essentialPlan": {
"eshopboxExpress": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxStandard": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxPriority": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
}
},
"premium": {
"eshopboxExpress": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxStandard": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxPriority": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
}
},
"advance": {
"eshopboxExpress": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxStandard": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
},
"eshopboxPriority": {
"fees": {
"shippingBaseFreight": 22.0,
"expressSurcharge": 121.0,
"codCollectionFees": 35.0,
"reverseShippingFees": 0.0,
"fuelSurcharge": 3.3,
"doorstepQcFees": 0.0,
"gst": 11.84,
"totalShippingCharges": 77.64
},
"estimatedDeliveryDays": 2
}
}
}
Error Response
{
"error": {
"errors": [
{
"domain": "global",
"reason": "required",
"message": "Authentication failed : userInfo is null"
}
],
"code": 401,
"message": "Authentication failed : userInfo is null"
}
}
{
"error": {
"errors": [
{
"domain": "global",
"reason": "conditionNotMet",
"message": "Order weight cannot be blank"
}
],
"code": 412,
"message": "Order weight cannot be blank"
}
}
-end-