Weight discrepency : Rate Calculator
What is the purpose of this article?
This article aims to develop an API that would offer clients a means to calculate the shipping charges between their pickup postal code (from the warehouse's location) and the drop-off postal code (the customer's location) based on the types of couriers.
Based on the shipping charges, clients can decide the courier they want to ship the order.
Rate Calculator Figma
WIP | June 2023
Examples: Courier_Rate_Calculator_.xlsx
DB Models
Table name - fees_types
Column name | id | type | journey_type |
Data Type | int(5) | varchar(100) | enum('forward','return','both') |
Null/ Not null |
|
|
|
Unsigned | YES |
|
|
|
|
|
|
| 1 | Express Charge | forward |
| 2 | COD Rate | forward |
| 3 | COD percent | forward |
| 4 | CIR Charges | return |
| 5 | Fuel charges | both |
| 6 | QC charges | return |
coumn name | Data Type | Null/ Not null | Unsigned |
id | int(5) | Not null | Yes |
type | varchar(100) |
|
|
journry_type | enum('forward','return','both') |
|
|
What are the steps to achieving this task?
Note - New Endpoint will be made to achieve this task in the below project
Project - Inventory engine Client portal
service - shipping service
API Path(Staging) = GET https://tab.auperator.co/api/v1/routingrules/combinations
API Path(Prod) = GET https://blackberrys.eshopbox.com/api/v1/routingrules//combinations
The user fills in the details on a workspace and clicks on the calculate button.
Then POST /ratecalculator
API gets hit with the below request body from frontend{ "journeyType": "forward", "pickupPincode": "110001", "dropPincode": "835210", "pickupPincodeState" : "RN", "dropPincodeState" : "DL", "orderWeight": 500.5, "length": 100.5, "breadth": 50.5, "height": 10.5. "paymentMethod": "Prepaid", }
From the endpoint, Call a function calculateRatesByPincodes. This function will be developed in service layer.
In this function, perform the following steps:-
Create a model class (baseRatesModel) for the table base_rates_model which contains the data about what will be the charges based on zones and order weight for every account (workspace).
Based on pickup pincode and drop pincode, calculate the zone (use the same logic as used in inventory allocation)
Fetch the details from memchache or by hibernate query in baseRatesModel class and store the result in baseRatesModelList.
Call a function prepareSequenceListByOrderWeight. This function will be developed in helper layer. Inside this function perform the below task.
Set a minimumWeight int value = 0
Define two empty lists - chargeableWeightSequenceList<Double> and chargeableAmountList<Integer>
Iterate over the baseRatesModelList
Divide the (max weight - minimum weight) by per weight and store the result in a new variable int slabs.
Add values in the above-defined lists limited to the value of slabs.
chargeableWeightSequenceList contains the weight sequence which will be taken into consideration for calculating the charges. Here the value of per_weight will be added.
chargeableAmountList contains the chargeable amount. The chargeable amount should be only for the calculated zone. Here the value of rate of the zone will be added.
Update minimum weight variable by the value of max weight
Logic with Example :- (500 - 0) / 500 = 1 ===> [500], [22] (2000 - 500) / 500 = 3 ===> [500, 500, 500, 500], [22, 20, 20, 20] (5000 - 2000) / 1000 = 3 ===> [500, 500, 500, 500, 1000, 1000, 1000], [22, 20, 20, 20, 22, 22, 22]
Return both the prepared list in a map.
Once the two lists are prepared, calculateCourierwiseRatesByWeight function. This function will be prepared in helper layer. Perform the below tasks in this function :-
Get the orderWeight from the request body.
Define a variable int totalChargeableWeight = 0 and intTotalAmount = 0
Iterate over the chargeableWeightSequenceList
If orderWeight > totalChargeableWeight , then break
Update totalChargeableWeight = totalChargeableWeight + chargeableWeightSequenceList.index value of iteration
Update totalAmount = totalAmount + chargeableAmountList.index value of iteration
and for each value keep adding the amount from the chargeableAmountList till the time the orderWeight is not greater than the total of chargeableWeightSequenceList iteration.
Once the amount is calculated, get feesTypes details and accountAddionalFees details from the Memcache or by hibernate query from fees_types and account_additional_fees respectively.
Prepare a Map<String, Dto> for accountAddionalFees where the key will be fee_types_id and value will be accountAddionalFees model row.
Get the journeyType from the request body
Iterate over the feesTypes details
if feesTypesDetails.resource = journeyType
Get the accountAddionalFees model row by fee_types_id from the prepared Map.
If Calculation_type = flat then directly add the fees in the totalAmount.
If calculation_type = percent the calculate the percentage and add the value in the total amount.
Return the courierwise response from the function
Add the request DTO along with the prepared courierwise response
Return the response to the frontend.
{
"journeyType": "forward",
"pickupPincode": "110001",
"dropPincode": "835210",
"pickupPincodeState" : "RN",
"dropPincodeState" : "DL",
"orderWeight": 500.5,
"length": 100.5,
"breadth": 50.5,
"height": 10.5,
"paymentMethod": "Prepaid",
"totalAmount": 1500.5,
"Standard": {
"shippingCharge": 100.5,
"GST": 10.5,
"TotalCharge": 111,
"expectedDeliveryInDays": 3
},
"Express": {
"shippingCharge": 50.5,
"GST": 5.5,
"TotalCharge": 56,
"expectedDeliveryInDays": 5
},
"Priority": {
"shippingCharge": 200.5,
"GST": 20.5,
"TotalCharge": 221,
"expectedDeliveryBy": 7
}
}