/
Fynd Shipment Accept/ Reject Flow

Fynd Shipment Accept/ Reject Flow

 

Step 1: Eshopbox Shipment Service will listen to the WMS Update Shipment Event from PubSub.

{ "resource": "shipment", "eventType": "PUT", "eventSubType": "accepted", "accountSlug": "winspiring", "accountId": null, "actor": "SYSTEM", "version": "v1", "request_data": [], "response_data": { "customerOrderNumber": "4405473214658", "vendorOrderNumber": "3229789520066", "externalShipmentID": "4405473214658-846-20", "externalWarehouseID": "ESBMUM_112", "externalChannelID": "CH0846", "integrationType": "", "vendorPartyID": null, "partner_as2_id": null, "defaultWarehouseCode": "ESBMUM_112", "orderDate": "2021-02-03 18:58:51", "portal_id": 27, "paymentType": "Cash on Delivery (COD)", "expectedShipDate": null, "dispatchAfterDate": null, "externalManifestNumber": null, "channelManifestNumber": null, "order_id": 5331324, "channel_id": 846, "warehouse_id": 72, "channel_account_id": 158, "account_id": 149, "connectionId": 5, "locationId": "59469168834", "region": "", "picklistCode": null, "invoiceNumber": "12345678", "boxType": "UNKNOWN", "isPriorityShipment": "0", "isGift": "0", "invoice_url": "", "invoiceDate": null, "label_url": "", "labels": "", "shippingInfo": [], "boxAdditionalRecommendation": [], "dimension_length": "0", "dimension_width": "0", "dimension_height": "0", "weight": "0", "trackingID": "12345678", "packageID": "", "barcode": "", "taxAmount": 0, "shipChargeAmount": 30, "courierName": "EK", "created_at": "2021-02-03 18:59:07", "updated_at": "2021-02-03 18:59:07", "status": "created", "remarks": "", "customerName": "Joe", "customerContactNumber": "0123466666", "email": "prateek.kaushik@eshopbox.com", "channelSlug": "meraki", "status_updated_at": "2021-02-03 18:59:07", "status_log": { "created": "2021-02-03 18:59:07" }, "orderExternalCreatedAt": "2021-02-03 18:59:03", "shippingAddress": { "customerName": "Joe", "addressLine1": "Pune", "addressLine2": "Pune", "city": "Pune", "state": "Haryana", "postalCode": "122001", "countryCode": "IN", "countryName": "INDIA", "contactPhone": "0123466666", "email": "prateek.kaushik@eshopbox.com" }, "id": 5252341, "isCOD": "1", "track_payload": [], "items": [ { "lineItemSequenceNumber": 9136087335106, "orderItemID": "4405473214658-13893324", "itemID": "40125724983490", "sku": "", "asin": "", "productName": "X", "quantity": 1, "customerPrice": 200, "lineItemTotal": 236, "invoiceTotal": 246, "discount": 0, "taxRate": 0, "taxAmount": 0, "giftMessage": "", "giftLabelContent": "", "lineItemOrderStatus": "", "orderItemIDs": [ "4405473214658-13893324" ], "productIm "transportMode": "SUR", "deliveryMode": "HD"ageUrl": null, "productAdditionalInfo": { }, "expectedDeliveryDate": null, "shippingCharges": 10, "productUrl": null, "originalOrderItemId": null }, { "lineItemSequenceNumber": 9136087335106, "orderItemID": "4405473214658-13893325", "itemID": "40125724983490", "sku": "", "asin": "", "productName": "X", "quantity": 1, "customerPrice": 200, "lineItemTotal": 236, "invoiceTotal": 246, "discount": 0, "taxRate": 0, "taxAmount": 0, "giftMessage": "", "giftLabelContent": "", "lineItemOrderStatus": "", "orderItemIDs": [ "4405473214658-13893325" ], "productImageUrl": null, "productAdditionalInfo": { "transportMode": "SUR", "deliveryMode": "HD" }, "expectedDeliveryDate": null, "shippingCharges": 10, "productUrl": null, "originalOrderItemId": null }, { "lineItemSequenceNumber": 9136087335106, "orderItemID": "4405473214658-13893326", "itemID": "40125724983490", "sku": "", "asin": "", "productName": "X", "quantity": 1, "customerPrice": 200, "lineItemTotal": 236, "invoiceTotal": 246, "discount": 0, "taxRate": 0, "taxAmount": 0, "giftMessage": "", "giftLabelContent": "", "lineItemOrderStatus": "", "orderItemIDs": [ "4405473214658-13893326" ], "productImageUrl": null, "productAdditionalInfo": { "transportMode": "SUR", "deliveryMode": "HD" }, "expectedDeliveryDate": null, "shippingCharges": 10, "productUrl": null, "originalOrderItemId": null } ] }, "previous_data": [], "resource_type": "shipment.update", "account_slug": "winspiring", "custom": [] }

Step 2: From this WMS Event extract the “integrationType" and “resource”. Before moving on, make sure that the integrationType is “Fynd“ and the resource is “shipment”. If these conditions are met, and the eventSubType is either “accepted” or “rejected”, push the update shipment request into the TaskQueue.

Step 3: After that tasks are listened one by one from the taskQueue and requestBody is formed then.

Step 4: RequestBody Formation: 

Check the eventSubType in the WMS event request body

3.1 If the eventSubType is “rejected”, Mark the status as “bag_not_confirmed”.

3.2 If the eventSubType is “accepted”, Mark the status as “Confirmed”.

Step 5: In Case of order eventSubType : “rejected”. Post Order Cancelation API by following Request Body.

{{oms_host}}/api/v2/{{aggregator}}/shipment/cancel

Request Body:

[ { "orderId": "String", "orderItems": [ { "orderItemId": "String", "reasonCode": "String" }, {...}, {...} ] }, {...}, {...} ]

Parameters:

Key

Value

Key

Value

OrderId

response_data.vendorOrderNumber

OrderItemId

response_data.items.orderItemID

reasonCode

“ “

Reason

Code

price mismatch

8

out of stock

6

damaged product

9

product not as per description

7

note:

  • Must provide cancellation reason_code on bag level.

  • Partial shipment cancellation is allowed before confirming and never after confirming.

  • Complete shipment cancellation is allowed until the bag is not picked by the delivery partner.

Response :

HTTP status: 200

{ "data": { "payloadCount": 100, "successCount": 100, "failedCount": 0, "status": "success", "errors": [] } }

Step 6: If the eventSubType is “accepted”:

6.1  It try to fetch the access token and inputFields(to get sellerCode) from the cache

6.2.  If it is not found in the cache, then it retrieves it from the Database (ie_appinstall_connection

         table), using the connectionId fetched from WMS event body.

SELECT ie_appinstall_connection.accessToken AS accessToken, ie_appinstall_connection.inputFields AS inputFields FROM ie_appinstall_connection WHERE ie_appinstall_connection.id =:connectionId

It is then stored in the cache memory in the following format:

{ "connectionData_{{connectionId}}": { "accessToken":"{{accessToken}}", "inputFields":"{{inputFields}}" } }

6.3 Now we have accessToken.

STEPS TO CHECK CUSTOMER CANCELLATION BEFORE ACKNOWLEDMENT

  1. will call a new method to check the customer cancellation status by passing parameters accesstoken, shipmentevent.

  2. In the method at first i will retrive customer order number from shipment event. Prepare a request body to call fynd get Shipment API

  3. Now add a header in the request as : ApiKey <{{accessToken}}>

  4. call the get shipment API :

Request Body:

curl --location --request GET 'https://api.fyndx1.de/hogwarts/aggregators/api/v2/eshopbox/shipment' \ --header 'ApiKey: NjA4MTAzODFhMzk1ZTE3NjNiMDlhYWVlLm5GYm40ZTd1VS4zNzI=' \ --header 'Content-Type: application/json' \ --data-raw '[ { "orderId": "16735117911231844514" } ]'

Response Body :

{ "orders": [ { "id": "16735117911231844514", "forward_id": "16735117911231844514", "marketplaceOrderId": "FY63BFC36E0D0103D196", "marketplaceReturnId": null, "shippingAddress": { "phone": "8638915874", "addressLine1": "234", "state": "Maharashtra", "country": "India", "city": "Mumbai", "email": "bulta0932@gmail.com", "pincode": "400051", "name": "Sudip Das" }, "thirdPartyShipping": true, "additionalInfo": "you can add additional info here", "orderPrice": { "currency": "INR", "totalPrepaidAmount": 1499.0 }, "orderItems": [ { "shippingMethodCode": "STD", "shipNode": "EB1", "product_identifiers": { "ean": "NOTEBOOK" }, "quantity": 1, "marketplaceOrderItemId": "598962", "variantId": "S", "orderItemId": "598961", "orderItemPrice": { "totalPrice": 1499.0, "shippingCharges": 0, "sellingPrice": 3499.0, "transferPrice": 0, "currency": "INR", "cashOnDeliveryCharges": 0, "discount": 2000.0 }, "giftWrap": { "giftWrapMessage": "", "giftWrapCharges": 0 }, "onHold": false, "status": "CANCELLED", "sku": "NOTEBOOK", "productId": "EPB1", "gst_details": { "CGST": 114.33, "SGST": 114.33 }, "packetNumber": 0, "title": "Eshopbox" }, { "shippingMethodCode": "STD", "shipNode": "EB1", "product_identifiers": { "ean": "KEYBOARD" }, "quantity": 1, "marketplaceOrderItemId": "598962", "variantId": "FREE", "orderItemId": "598962", "orderItemPrice": { "totalPrice": 1499.0, "shippingCharges": 0, "sellingPrice": 3499.0, "transferPrice": 0, "currency": "INR", "cashOnDeliveryCharges": 0, "discount": 2000.0 }, "giftWrap": { "giftWrapMessage": "", "giftWrapCharges": 0 }, "onHold": false, "status": "CREATED", "sku": "KEYBOARD", "productId": "EPB1", "gst_details": { "CGST": 114.33, "SGST": 114.33 }, "packetNumber": 0, "title": "Eshopbox" } ], "orderStatus": "CREATED", "orderDate": "2023-01-12T13:53:11", "paymentType": "COD", "cFormProvided": false, "taxExempted": false, "gstin": "", "sla": "2023-01-14T13:55:20.754409+05:30", "orderingChannel": "ESHOPBOX", "fyndInvoiceUrl": "https://fynd-staging-assets-private.s3-accelerate.amazonaws.com/addsale/documents/invoice_a4/PDFs/16735117911231844514_invoice_a4.pdf?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=AKIAJUADR2WMPQT6ZJ2Q%2F20230112%2Fap-south-1%2Fs3%2Faws4_request&X-Amz-Date=20230112T082520Z&X-Amz-Expires=604800&X-Amz-Signature=0d9c6fa9343e6b61a8b3aba39d110f3be42e63b9a3e1074389de26300defcb08&X-Amz-SignedHeaders=host", "fyndInvoiceUrlA6": "https://fynd-staging-assets-private.s3-accelerate.amazonaws.com/addsale/documents/invoice_a6/PDFs/16735117911231844514_invoice_a6.pdf?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=AKIAJUADR2WMPQT6ZJ2Q%2F20230112%2Fap-south-1%2Fs3%2Faws4_request&X-Amz-Date=20230112T082520Z&X-Amz-Expires=604800&X-Amz-Signature=1e472e95b458be313fb4703dadb6644c8fd3cb805df434ddf41f80e2e6a1a770&X-Amz-SignedHeaders=host", "fyndLabelUrl": "https://fynd-staging-assets-private.s3-accelerate.amazonaws.com/addsale/documents/label_a4/PDFs/16735117911231844514_label_a4.pdf?X-Amz-Algorithm=AWS4-HMAC-SHA256&X-Amz-Credential=AKIAJUADR2WMPQT6ZJ2Q%2F20230112%2Fap-south-1%2Fs3%2Faws4_request&X-Amz-Date=20230112T082520Z&X-Amz-Expires=604800&X-Amz-Signature=9d4684145120a676a1dfb176f822c2091532ace22efa2cd88e68182b661b1bf0&X-Amz-SignedHeaders=host", "invoice_id": null, "return_invoice_id": null, "marketPlaceInvoiceUrl": "", "marketPlaceLabelUrl": "", "purchaseOrderNumberPDF": "", "irn_number": "", "invoice_qr_code": "", "signed_invoice": "", "B2BData": { "name": "", "phone": "", "address": "", "state": "", "pincode": "", "gstin": "" }, "lockStatus": false, "fulfilmentPriority": 10, "journeyType": "Forward", "billingAddress": { "phone": "8638915874", "addressLine1": "234", "state": "Maharashtra", "country": "India", "city": "Mumbai", "email": "bulta0932@gmail.com", "pincode": "400051", "name": "Sudip Das" } } ] }

5. From the response I will fetch list of orderItemId whose line item status =”CANCELLED”.

6. Declare a variable bool.

7. If size of List[orderItemId]>0 then simply will put bool = false and prepare wms cancel request body using the list and shipment event.

8. call the wms cancel order API to cancel the customer cancelled items in WMS:

curl --location --request POST 'https://eshopbox-wms-api-dot-eshopbox-wms-staging.el.r.appspot.com/api/cancel-order' \ --header 'Authorization: Bearer eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6IlJVVXdSREZCUVRSRFFqQkdORFUxTVVZeE16ZEdPRFJHTnpORk5EaEJSVEU0TVVORk5qVTJOdyJ9.eyJodHRwczovL2FwcERhdGEiOnsiYXBwQ2xpZW50SWQiOiJjMDQzY2RhNWVlZDQzY2YxM2E5M2ZiYTI3OWRiN2I0MiJ9LCJodHRwczovL2FjY291bnRTbHVnIjoidGVzdF90ZXN0X2FqZWV0IiwiaHR0cHM6Ly91c2VyRGV0YWlscyI6eyJpZCI6MjY0MywidXNlclR5cGUiOiJkZWZhdWx0IiwiZW1haWwiOiJhcHB1c2VyNDg1QGVzaG9wYm94LmNvbSJ9LCJodHRwczovL2FjY291bnRzIjpbInRlc3RfdGVzdF9hamVldCJdLCJodHRwczovL3dhcmVob3VzZVdvcmtzcGFjZXMiOltdLCJodHRwczovL3dhcmVob3VzZXMiOltdLCJodHRwczovL3BhcnRuZXJzIjpbXSwiaXNzIjoiaHR0cHM6Ly9lc2hvcGJveC1wYXltZW50LXJlY28uYXV0aDAuY29tLyIsInN1YiI6ImF1dGgwfDYzNjBiODI4NzFiZjg2MDNlZWU3MjFjZiIsImF1ZCI6Imh0dHBzOi8vZXNob3Bib3gtcG9ydGFsLWRldi5hcHBzcG90LmNvbSIsImlhdCI6MTY3MjkyMTQxNSwiZXhwIjoxNjc1NTEzNDE1LCJhenAiOiJ3OHZZWGtmTElmZjhZdjJqNE8zVDlKZWFqMVMzcThyZSIsInNjb3BlIjoicmVhZDpwcm9kdWN0cyByZWFkOmludmVudG9yeSByZWFkOnByb2R1Y3RfbGlzdGluZ3MiLCJndHkiOiJwYXNzd29yZCJ9.HQ7KLMaur80xyBgmssIP0wKEi6RR-BLrIF-HjfqnqDD5awE_AzynfZDgDn7LFbqBJGH1tiMXlbKgBdEm1bV3odi6V1koXHzpCPvlG6i6zXYTDrIHpiu-7Nf1j0oQYoKQVCStbRiaim9hRyZqtEqen8WNDMdw5CgoxWPm9QHdgBQN_82xOkAIzjlIMVGYyAb-94e1PGW6FDtlXdcH80wxgP3ww4a17u3vjItiZrvhnzQEKvSnwdmQ9NC8hfO8QQGnINYc5HG0ng3_Gv05StcsqbjJKiLJH56DqrfA5_Sl43MS09EZwgyLxsuGkEL_TsjCoNj6NvbKyjB-Aqu3Y9a4ww' \ --header 'Content-Type: application/json' \ --data-raw '{ "reason": "fulfillment_cancel", "externalChannelID": "CH5853", "customerOrderNumber": "16724060948331180111", "items": [ { "itemID": "BAG101", "quantity": 1, "orderItemID": "16724060948331180111-2392223", "lineItemSequenceNumber": 552855, "remark": "fulfillment_cancel" } ] }

9. else will put bool = true and return.

10. in the shipment accepted event service if bool = true we will continue from the step (6.4) below, else we will skip the entire flow.

Steps to check Unfullilable item in shipment accepted event

  1. call a method and check for the particular customer Number if for any Item Shipment ID is 0 or not

  2. check if for any item shipment ID = 0

SELECT orders.customerOrderNumber AS customerOrderNumber, order_items.shipment_id AS shipment_id ,order_items.itemID AS itemID ,order_items.lineItemSequenceNumber AS lineItemSequenceNumber, order_items.orderItemID AS orderItemID, order_items.quantity AS quantity \r\n" + " FROM orders LEFT JOIN order_items ON order_items.order_id = orders.id \r\n" + " LEFT JOIN channels ON channels.id = orders.channel_id \r\n" + " WHERE channels.externalChannelID = :externalChannelID AND orders.customerOrderNumber = :customerOrderNumber AND order_items.shipment_id = '0'\r\n")

3. for the items whose shipment id=0 , we just reject the item in Fynd (STEP 5) and reject that in WMS.

4. else we will continue from Step (6.4) shipment accept flow.

 

6.4 Now make order Status Update In API by replacing the ApiKey with the accessToken we retrieved.

6.5 Now add a header in the request as : ApiKey <{{accessToken}}>

Step 7: Call the Post Order Confirmation status Update In API to update the shipment status with requestBody:

{{oms_host}}/api/v2/{{aggregator}}/shipment

Request Body:

[ { "orderId": "String", "orderItems": [ { "orderItemId": "String" }, {...}, {...} ] }, {...}, {...} ]

Note:

  • In the shipment accept case also we can partially Reject “if for some item Shipment_id=0” (ignore those items in the request body) and Post Shipment Confirmation.

  • In case of order confirmation is raised for shipment was part item is cancelled by the customer, confirm non-cancelled item and consider the request as a success.

  • Limit 100 orders per transaction

  • Mark status as “confirmed”

  • In case of order confirmation is raised for shipment where the entire shipment is cancelled by the customer, fail the shipment update.

Response:

Situation 1: Request successful
http status: 200

{ "data": { "payloadCount": 100, "successCount": 100, "failedCount": 0, "status": "success", "errors": [] } }

Situation 2: Request partially successful
http status: 200

{ "data": { "payloadCount": 100, "successCount": 99, "failedCount": 1, "status": "warning", "errors": [ { "orderId": "string" "SH-20191112-5DCA4C130-193C954AA", "orderItems": [ { "orderItemId": "string", "quantity": 1 "A301", "reasonCode": "", "message": "Cancelled by customer/seller" } ] } ] } }

Situation 3: Request failed
http status: 200

{ "data": { "payloadCount": 2, "successCount": 0, "failedCount": 2, "status": "failed", "errors": [{ "orderId": "SH-20191112-5DCA68F60-1E7E5EE6F", "orderItems": [ { "orderItemId": "B100", "reasonCode": "", "message" : "invalid orderId" } ] },{ "orderId": "SH-20191112-5DCA81050-175D13F80", "orderItems": [ { "orderItemId": "B101", "reasonCode": "", "message" : "cancelled by customer/seller" } ] }] } }

Reason

code

description

cancelled by customer/seller

O400

If orderId or orderItemId is cancelled

invalid orderId

O422

If orderId doesn't belong to the requested location

1.Invalid Token Error Handling Flow : In case of any error due to invalid token - call refresh token API

  1. get connection Detail using the below API

url - http:{}/api/fynd/connection/{connectionId}

2. fetch existing connection detail from database using connectionId.

SELECT `id`,'account_id','channel_id','additionalDetails' FROM `channels` WHERE `ConnectionId` = :ConnectionId

3. call the fynd refresh token api using the ConnctionId details

API Method: POST

url- http://{}/api/fynd/refresh/connection/{connectionId}

Request:

connectionDetails{ id:"string" account_id:"string" channel_id:"string" additionalDetails:"string" ConnctionId:"string" }

Response:

connectionDetails{ id:"new_string" account_id:"new_string" channel_id:"new_string" additionalDetails:"new_string" ConnctionId:"new_string" }

4. update the new details in database against the connectionId.

UPDATE INTO 'channel' SET `:id`,':account_id',':channel_id',':additionalDetails' WHERE 'connectionId'=:connectionId

5. get the existing cache detail (save in a variable existingCacheDetailMap)

6. Clear old cache and create new cache from new detail of db

newConnectionDetailMap= existingCacheDetailMap+connectionId

 

Related content