Skip to main content
Solved

Easypost International Shipments Customs Documentation


Browsing the Forums, it seems many people have Easypost working with Acumatica to generate their Customs Documentation automatically from Acumatica.  I’ve installed the Easypost customization and have Fedex and UPS working, but I can’t determine which fields on the shipment I’m supposed to enter the required Customs information (Commodity type, Customs Value, Harmonized Code, etc.) to generate the commercial Invoice.

It seems easy enough to create the Customs Object for the Easypost API, but I can’t find any documentation on the Easypost customization project (that I believe Acumatica maintains) that shows what fields map to what.  The project does add some extra fields to the shipment.  

 

thanks!

14 replies

Userlevel 5
Badge +2

The part which is specific to EasyPost is on page 9 of EasyPost documentation (see attachment at the bottom of this message):

Support for supplying Exemption and Exclusion Legend (EEL) code or a Proof of Filing Citation (PFC) for international destination When shipping outside the US, you would need to provide either an Exemption and Exclusion Legend (EEL) code or a Proof of Filing Citation (PFC). Which you need is based on the value of the goods being shipped. Acumatica allows to specify value for EEL/PFC in Shipment → Packages’ EEL/PFC (EasyPost) field for each package in a shipment. Per EasyPost customs-guide: • If the value of the goods is less than $2,500, then you pass the following EEL code: "NOEEI 30.37(a)" • If the value of the goods is greater than $2,500, you need to get an Automated Export System (AES) Internal Transaction Number (ITN) for your shipment. ITN will look like “AES X20120502123456”. To get an ITN, go to the AESDirect website. • An ITN is required for any international ship

 

There are a few standard item configuration fields that are used for international shipping.
 


This code snippet shows these values used by EasyPost integration:
shipment.customs_info = new CustomsInfo();
shipment.customs_info.customs_certify = "true";
shipment.customs_info.customs_signer = fromAddress.name;

shipment.customs_info.contents_type = ContentType.GetEasyPostContentType(carrierBox.CustomsData.ContentType);
shipment.customs_info.non_delivery_option = _settings.InternationalNonDeliveryOption;
shipment.customs_info.contents_explanation = carrierBox.CustomsData.OtherContentTypeDesc ?? shipment.customs_info.contents_type;
shipment.customs_info.eel_pfc = carrierBox.CustomsData.CertificateNumber;
shipment.customs_info.customs_items = new List<CustomsItem>();

foreach (var cline in carrierBox.CustomsData.DetailedLines)
{
    CustomsItem line = new CustomsItem();
    line.code = cline.InventoryCode;
    line.description = cline.InventoryDescr;
    line.quantity = Convert.ToInt32(cline.InventoryQty);
    line.hs_tariff_number = cline.InventoryHSTariffCode;
    line.value = Convert.ToDouble(cline.InventoryValue * cline.InventoryQty);
    line.origin_country = cline.InventoryCountryOfOrigin;
    line.currency = curCode;
    line.weight = GetBoxWeightForEasyPost(cline.InventoryWeight, cRequest.Units) * line.quantity;
    shipment.customs_info.customs_items.Add(line);
}

Userlevel 3
Badge +1

Thanks Hughes!! This certainly helps me get closer!

 

Sometimes our customers provide us with information to ship internationally.  Creating shipments through Fedex gives us the option for the Declared Value to be different from the Customs Value.  Is this an option?  Is it pulling any Value from the Sales Order or is it all from the ‘Declared Value’ on the Package?  Since we sell through distributors the value on the SO does not reflect the Customs or Declared Value.

 

 Also, our products are used for different purposes, I’m guessing there is no override for the tariff/harmonized code and that has to be on the stock item? 

 

Is the projects souce code available?  That way I could dig around and try to figure some of this out myself.

 

Thanks again, it really is appreciated!

Userlevel 5
Badge +2

 

>  Creating shipments through Fedex gives us the option for the Declared Value to be different from the Customs Value.

Box declared value is used for the insurance options.

> Is it pulling any Value from the Sales Order or is it all from the ‘Declared Value’ on the Package? 

The customs value is based on the SOLine unit price value. The computation from there to the carrier is very long and complex. The last field in the chain to contain the custom value is SOShipLineSplitPackage.UnitPriceFactor:

protected virtual IList<InternationalCustomsInfoLine> GetCustomsInfoDetailLines(SOPackageDetailEx box, CarrierPlugin plugin)
{
    var packageSplits = SelectFrom<SOShipLineSplitPackage>.
        Where<SOShipLineSplitPackage.shipmentNbr.IsEqual<@P.AsString>.
        And<SOShipLineSplitPackage.packageLineNbr.IsEqual<@P.AsInt>>>
        .View.Select(Base, box.ShipmentNbr, box.LineNbr)
        .RowCast<SOShipLineSplitPackage>()
        .ToList();

    var detailLines = new List<InternationalCustomsInfoLine>();
    foreach (SOShipLineSplitPackage packageSplit in packageSplits)
    {
        SOShipLine shipLine = PXSelect<SOShipLine,
                            Where<SOShipLine.shipmentNbr, Equal<Required<SOShipLine.shipmentNbr>>,
                                And<SOShipLine.lineNbr, Equal<Required<SOShipLine.lineNbr>>>>>
            .SelectWindowed(Base, 0, 1, packageSplit.ShipmentNbr, packageSplit.ShipmentLineNbr);

        InventoryItem shipLineItem = InventoryItem.PK.Find(Base, shipLine.InventoryID);
        InventoryItem packedItem = shipLine.InventoryID == packageSplit.InventoryID
            ? shipLineItem
            : InventoryItem.PK.Find(Base, packageSplit.InventoryID);

        decimal itemWeightFactor = (packageSplit.WeightFactor ?? 0m) != 0m ? packageSplit.WeightFactor.Value : 1m;
        decimal weightInStandardUnit = Base.CarrierRatesExt.ConvertWeightValue((shipLineItem.BaseWeight ?? 0) / itemWeightFactor, plugin);

        var customsInfoLine = new InternationalCustomsInfoLine();
        customsInfoLine.InventoryCode = packedItem.InventoryCD;
        customsInfoLine.InventoryDescr = !string.IsNullOrWhiteSpace(shipLine.TranDesc)
            ? shipLine.TranDesc
            : !string.IsNullOrWhiteSpace(shipLineItem.Descr) ? shipLineItem.Descr : shipLineItem.InventoryCD;
        SOShipLine.unitPrice
        customsInfoLine.InventoryValue = packageSplit.UnitPriceFactor ?? 0m;
        customsInfoLine.InventoryQty = packageSplit.PackedQty ?? 0m;
        customsInfoLine.InventoryCountryOfOrigin = packedItem.CountryOfOrigin;
        customsInfoLine.InventoryHSTariffCode = packedItem.HSTariffCode;
        customsInfoLine.InventoryWeight = weightInStandardUnit;

        detailLines.Add(customsInfoLine);
    }

    return detailLines;
}

> Is the projects souce code available?

Yes and no, source code is distributed with the product. You can open the source code page but it doesn’t contain code for all assemblies.

Userlevel 5
Badge +2

> I’m guessing there is no override for the tariff/harmonized code and that has to be on the stock item? 

You would need to customize Acumatica product or create duplicate stock items for this use case.

Userlevel 5
Badge +2

> Since we sell through distributors the value on the SO does not reflect the Customs or Declared Value.

That value appears to be based on SOLine UnitPrice field. So you would have to find a workflow where you can use this value or customize Acumatica product to provide a different value.

Userlevel 3
Badge +1

Incredible reply.  Thank you very much Hughes.  If every topic here got this treatment, end users would be thrilled!

 

> Is the projects souce code available?

Yes and no, source code is distributed with the product. You can open the source code page but it doesn’t contain code for all assemblies.

Great!  I had assumed those dll’s were obfuscated.

 

> I’m guessing there is no override for the tariff/harmonized code and that has to be on the stock item? 

You would need to customize Acumatica product or create duplicate stock items for this use case.

This is unfortunate.  There is different Harmonized (tariff) code for item repair or replaced under warranty.  So anyone offering warranties or doing repairs potentially are going to need at least 2 codes for the same Stock Item.

 

> Since we sell through distributors the value on the SO does not reflect the Customs or Declared Value.

That value appears to be based on SOLine UnitPrice field. So you would have to find a workflow where you can use this value or customize Acumatica product to provide a different value.

This one is even more unfortunate.  Not just for situations like ours where the item on our SO holds a different actual value of the item, but again for warranty repairs where the line item is going to be $0. 

 

Thanks again though.  If possible pass my requests to the customization project authors to make as many of the API calls editable on the Shipment (or any) screen.

Hey @Noah - noticed you weren’t able to get this working and wanted to make you aware of our StarShip Acumatica shipping app.  It connects directly to Acumatica and can pull the item info over but also gives you the ability to: 

  • store additional information about the item in the StarShip database (so it can be a different value than what is on the Acumatica document)
  • edit information on the fly 

We also support other international-related functions like direct ACE integration, paperless invoice, ETD, etc.  Let me know if you would like more details!

Userlevel 3
Badge +1

Thanks Caroline,  I’ll check it out!

Userlevel 3
Badge +1

Sorry to bump an old thread, but finally found some time to start testing Easypost again and I’m getting an error message: “Carrier Service returned error: FedEx:Fedex returned error: Commodity 1 - Customs value is required.” 

Per above, isn’t the customs value supposed to be coming from the Sales Order total?  The structed data from the trace includes value information, but some items have the ‘free item’ box checked. (We’re also having the spacing problem that seems to be everywhere; the inventory id’s include all blank spaces until the end of the length of the numbering sequence, but I doubt it’s this).

 

More importantly, where do I go for help in getting this working?  Easypost? UPS? My VAR?  

 

Thanks for any guidance.

Hello Noah, 

Have you been able to resolve this issue? Usually you will want to work with your VAR, who works directly with EasyPost to resolve these issues.

If you still need it, I am happy to help.

Userlevel 1

@fimprota …. my experience is that EasyPost points you back to Acumatica who is the one that supports the integration between the two. If you’re VAR has no experience with EasyPost, then you’re relying on Acumatica and there is scant details to properly troubleshoot EasyPost integration in my eyes…..it seems to work best only on simple scenarios like ship to customer fully prepaid on your account, or ship to customer fully prepaid on customer’s account.  

@rharvey6 There are two common troubleshooting errors - Set up requirements from the Acumatica shipping UI, or errors to do with label/tracking requests coming from the EasyPost API. We can easily troubleshoot any errors returned by the carriers through our API, but will need help from the VAR or Acumatica to troubleshoot any UI requirements, as our support has limited experience with Acumatica. Most of the issues I’ve been involved in troubleshooting have come from different setup requirements from the shipping UI. If you have any specifics, happy to help. 

Userlevel 1

Thanks @fimprota and certainly will reach out if I require things.

Our major issue is we use Manual Packaging which requires us to manual define a ton of information for EasyPost to operate properly in even the most basic situations, but that’s due to an Acumatica limitation that I hope they fix…..that, and I believe that all Declared Value, COO, Tariff and other customs requirements should be pre-populated by Acumatica (yet editable) vs. having to be manually defined in the Shipment screen - most information is easily available out-of-box, and if not, adding a field (vs. attribute) should solve obtaining whatever information customs requires.

Only other one I believe Acumatica is looking at is the scenario where you ship on Customer (or your) freight account, but you want to bill Customs/Import charges to a 3rd party account which is possible directly in FedEx for instance, but not in the Acumatica-EasyPost setup using the Acumatica UI.

Userlevel 1

@fimprota One additional question would be about currency. We’ve gotten EasyPost waybill & customs form for a shipment from our location cross-border to the US...however, the Customs Form is being printed defined currency as $ CAD vs. $ USD which is the Sales Order and Customer’s currency, and the customer’s profile does not allow for currency to be overridden. 

Reply