Skip to content

Commit

Permalink
feat: resolve unspecified as default error code if value in json does…
Browse files Browse the repository at this point in the history
… not match existing enum
  • Loading branch information
jpill committed Apr 23, 2024
1 parent 5e58a5b commit f5169aa
Show file tree
Hide file tree
Showing 8 changed files with 217 additions and 22 deletions.
Original file line number Diff line number Diff line change
@@ -0,0 +1,146 @@
{
"rate_response": {
"rates": [],
"invalid_rates": [],
"rate_request_id": "se-139005222",
"shipment_id": "se-1506668211",
"created_at": "2024-04-23T16:09:08.1763483Z",
"status": "error",
"errors": [
{
"error_source": "carrier",
"error_type": "business_rules",
"error_code": "this_code_doesn't_exist",
"message": "A shipping carrier error occurred: Authentication failed.",
"carrier_id": "se-325186",
"carrier_code": "stamps_com",
"carrier_name": "Stamps.com"
}
]
},
"shipment_id": "se-1506668211",
"carrier_id": "se-325186",
"service_code": "usps_priority_mail_international",
"external_shipment_id": null,
"shipment_number": null,
"ship_date": "2024-04-23T00:00:00Z",
"created_at": "2024-04-23T16:09:08.037Z",
"modified_at": "2024-04-23T16:09:08.02Z",
"shipment_status": "pending",
"ship_to": {
"geolocation": null,
"instructions": "",
"name": "Test Person",
"phone": "8889997777",
"email": null,
"company_name": "Test Company",
"address_line1": "123 Fake St",
"address_line2": "",
"address_line3": null,
"city_locality": "NEW YORK",
"state_province": "NY",
"postal_code": "10011-4758",
"country_code": "US",
"address_residential_indicator": "unknown"
},
"ship_from": {
"instructions": null,
"name": "Test Person",
"phone": "8889997777",
"email": null,
"company_name": null,
"address_line1": "321 Fake St.",
"address_line2": null,
"address_line3": null,
"city_locality": "Brooklyn",
"state_province": "NY",
"postal_code": "11207",
"country_code": "US",
"address_residential_indicator": "no"
},
"warehouse_id": null,
"return_to": {
"instructions": null,
"name": "Test Person",
"phone": "8889997777",
"email": null,
"company_name": null,
"address_line1": "321 Fake St.",
"address_line2": null,
"address_line3": null,
"city_locality": "Brooklyn",
"state_province": "NY",
"postal_code": "11207",
"country_code": "US",
"address_residential_indicator": "no"
},
"is_return": false,
"confirmation": "none",
"customs": null,
"external_order_id": null,
"order_source_code": null,
"advanced_options": {
"bill_to_account": null,
"bill_to_country_code": null,
"bill_to_party": null,
"bill_to_postal_code": null,
"contains_alcohol": false,
"delivered_duty_paid": false,
"non_machinable": false,
"saturday_delivery": false,
"dry_ice": false,
"dry_ice_weight": null,
"fedex_freight": null,
"third_party_consignee": false,
"ancillary_endorsements_option": null,
"freight_class": null,
"custom_field1": null,
"custom_field2": null,
"custom_field3": null,
"collect_on_delivery": null,
"return_pickup_attempts": null,
"additional_handling": false,
"own_document_upload": false,
"limited_quantity": false,
"event_notification": false
},
"comparison_rate_type": null,
"shipping_rule_id": null,
"insurance_provider": "none",
"tags": [],
"packages": [
{
"shipment_package_id": "se-645844381",
"package_id": "se-3",
"package_code": "package",
"package_name": "Package",
"weight": {
"value": 3.83,
"unit": "pound"
},
"dimensions": {
"unit": "inch",
"length": 0.00,
"width": 0.00,
"height": 0.00
},
"insured_value": {
"currency": "usd",
"amount": 0.00
},
"label_messages": {
"reference1": null,
"reference2": null,
"reference3": null
},
"external_package_id": null,
"content_description": null,
"products": null
}
],
"total_weight": {
"value": 3.83,
"unit": "pound"
},
"items": []
}
2 changes: 1 addition & 1 deletion ShipEngine.Tests/ShipEngine.Tests.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,6 @@
<ProjectReference Include="..\ShipEngine\ShipEngine.csproj" />
</ItemGroup>

<ProjectExtensions><VisualStudio><UserProperties httpresponsemocks_4createimplicitmanifest200response_1json__JsonSchema="{" /></VisualStudio></ProjectExtensions>
<ProjectExtensions><VisualStudio><UserProperties /></VisualStudio></ProjectExtensions>

</Project>
18 changes: 18 additions & 0 deletions ShipEngine.Tests/ShipEngineMethodTests/GetRatesFromShipmentTest.cs
Original file line number Diff line number Diff line change
Expand Up @@ -249,5 +249,23 @@ public async void InvalidRetriesInMethodCall()
Assert.Equal("Retries must be greater than zero.", ex.Message);
Assert.Null(ex.RequestId);
}

[Fact]
public async void MissingErrorCodeEnumResolvesToUnspecified()
{
var apiKeyString = "TEST_bTYAskEX6tD7vv6u/cZ/M4LaUSWBJ219+8S1jgFcnkk";
var config = new Config(apiKey: apiKeyString);
var mockShipEngineFixture = new MockShipEngineFixture(config);

string json = File.ReadAllText(Path.Combine(Directory.GetCurrentDirectory(), "../../../HttpResponseMocks/GetRatesWithShipmentDetailsMissingErrorCodeEnum200Response.json"));

mockShipEngineFixture.StubRequest(HttpMethod.Post, "/v1/rates", System.Net.HttpStatusCode.OK, json);

var result = await mockShipEngineFixture.ShipEngine.GetRatesWithShipmentDetails(RatesParameters);

var errorCode = result.RateResponse.Errors.FirstOrDefault().ErrorCode;

Assert.Equal(ErrorCode.Unspecified, errorCode);
}
}
}
21 changes: 21 additions & 0 deletions ShipEngine/Converters/ErrorCodeEnumConverter.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
using Newtonsoft.Json.Converters;
using Newtonsoft.Json;
using System;

namespace ShipEngineSDK.Converters;
public class ErrorCodeEnumConverter : StringEnumConverter
{
public ErrorCode DefaultValue { get; set; }

public override object ReadJson(JsonReader reader, Type objectType, object existingValue, JsonSerializer serializer)
{
try
{
return base.ReadJson(reader, objectType, existingValue, serializer);
}
catch (JsonSerializationException)
{
return DefaultValue;
}
}
}
26 changes: 13 additions & 13 deletions ShipEngine/Enums/ErrorCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -27,10 +27,10 @@ public enum ErrorCode
[EnumMember(Value = "carrier_conflict")]
CarrierConflict,

// <summary>
// You have selected to insure a package, however the carrier you are attempting to create a shipping label
// for does not support declaring insurance on shipments.
// </summary>
/// <summary>
/// You have selected to insure a package, however the carrier you are attempting to create a shipping label
/// for does not support declaring insurance on shipments.
/// </summary>
[EnumMember(Value = "carrier_insurance_not_supported")]
CarrierInsuranceNotSupported,

Expand All @@ -54,12 +54,12 @@ public enum ErrorCode
[EnumMember(Value = "confirmation_not_supported")]
ConfirmationNotSupported,

// <summary>
// When shipping internationally, you must add a customs declaration which outlines the items you are shipping.
// This is helpful to customs agents when your package enters it's destination country. This error arises when you
// do not add a customs declaration to your shipment before attempting to create a shipping label for an
// international shipmet.
// </summary>
/// <summary>
/// When shipping internationally, you must add a customs declaration which outlines the items you are shipping.
/// This is helpful to customs agents when your package enters it's destination country. This error arises when you
/// do not add a customs declaration to your shipment before attempting to create a shipping label for an
/// international shipment.
/// </summary>
[EnumMember(Value = "customs_items_required")]
CustomsItemsRequired,

Expand Down Expand Up @@ -122,9 +122,9 @@ public enum ErrorCode
[EnumMember(Value = "invalid_charge_event")]
InvalidChargeEvent,

// <summary>
// When creating a label, if you set the date to an invalid date e.g. A date in the past
// </summary>
/// <summary>
/// When creating a label, if you set the date to an invalid date e.g. A date in the past
/// </summary>
[EnumMember(Value = "invalid_date")]
InvalidDate,

Expand Down
5 changes: 4 additions & 1 deletion ShipEngine/Models/Dto/Common/ShipEngineAPIError.cs
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,12 @@
using Newtonsoft.Json;
using Newtonsoft.Json.Converters;
using System.Collections.Generic;
using System.ComponentModel;

namespace ShipEngineSDK.Common
{
using Converters;

/// <summary>
/// Error object returned by the ShipEngine API when an error occurs.
/// <see href="https://www.shipengine.com/docs/errors/"/>
Expand Down Expand Up @@ -46,7 +49,7 @@ public class Error
/// <summary>
/// The error code specified for the failed API Call
/// </summary>
[JsonConverter(typeof(StringEnumConverter))]
[JsonConverter(typeof(ErrorCodeEnumConverter))]
public ErrorCode ErrorCode { get; set; }

/// <summary>
Expand Down
6 changes: 5 additions & 1 deletion ShipEngine/ShipEngine.csproj
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
<Project Sdk="Microsoft.NET.Sdk">
<Project Sdk="Microsoft.NET.Sdk">

<PropertyGroup>
<PackageId>ShipEngine</PackageId>
Expand Down Expand Up @@ -31,4 +31,8 @@
<None Include="../LICENSE" Pack="true" PackagePath="" />
</ItemGroup>

<ItemGroup>
<Folder Include="Converters\" />
</ItemGroup>

</Project>
15 changes: 9 additions & 6 deletions ShipEngine/ShipEngineClient.cs
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,8 @@

namespace ShipEngineSDK
{
using Converters;

/// <summary>
/// ShipEngine Client is used for handling generic calls and settings that
/// are needed for all ShipEngine API calls.
Expand Down Expand Up @@ -101,14 +103,15 @@ private async Task<T> DeserializedResultOrThrow<T>(HttpResponseMessage response)

}

var result = JsonConvert.DeserializeObject<T>(contentString, new JsonSerializerSettings()
var settings = new JsonSerializerSettings()
{
NullValueHandling = NullValueHandling.Include,
ContractResolver = new DefaultContractResolver
{
NamingStrategy = new SnakeCaseNamingStrategy()
}
});
ContractResolver = new DefaultContractResolver {NamingStrategy = new SnakeCaseNamingStrategy()}
};

settings.Converters.Add(new ErrorCodeEnumConverter { DefaultValue = ErrorCode.Unspecified });

var result = JsonConvert.DeserializeObject<T>(contentString, settings);

if (result != null)
{
Expand Down

0 comments on commit f5169aa

Please sign in to comment.