Digital Odyssey
0
Return Home
Home

The following PHP code is in relation to one of my recent videos which provides an overview on how to work with the Canada Post shipping rates API – you can watch the video posted in the sidebar. The code is pretty much broken down into three key steps outlined below:

Step 1:

Here i define the necessary variables required for the API and then pass those values into the XML request.

$env = "sandbox";
$customer_number = "000392384";  //Fake, replace with your CA Post company number
$username = $env == "sandbox" ? "998234kjsdfsdf" : "dfad9098234kj"; //Fake, replace with your CA Post API credentials
$password = $env == "sandbox" ? "2adfpoijksd9889234" : "kksd023lalsdf02398sdf";  //Fake, replace with your CA Post API credentials
$service_url = 'https://ct.soa-gw.canadapost.ca/rs/ship/price';

$origin_postal_code = 'L4L9Bk'; 
$destination_postal_code = 'L4B2B9';

$xmlRequest = <<<XML
<?xml version="1.0" encoding="UTF-8"?>
<mailing-scenario xmlns="http://www.canadapost.ca/ws/ship/rate-v4">
<customer-number>{$customer_number}</customer-number>
<parcel-characteristics>
    <weight>1</weight>
    <dimensions>
        <length>13</length>
        <width>13</width>
        <height>5</height>
    </dimensions>
</parcel-characteristics>
<origin-postal-code>{$origin_postal_code}</origin-postal-code>
<destination>
    <domestic>
    <postal-code>{$destination_postal_code}</postal-code>
    </domestic>
</destination>
</mailing-scenario>
XML;
Step 2:

The XML request is parsed through curl which sends the request to the service url and then retrieves the response from the Canada Post API.

$curl = curl_init($service_url); // Create REST Request
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);
//curl_setopt($curl, CURLOPT_CAINFO, realpath(dirname($_SERVER['SCRIPT_FILENAME'])) . '/cert/cacert.pem');
curl_setopt($curl, CURLOPT_POST, true);
curl_setopt($curl, CURLOPT_POSTFIELDS, $xmlRequest);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);
curl_setopt($curl, CURLOPT_USERPWD, $username . ':' . $password);
curl_setopt($curl, CURLOPT_HTTPHEADER, array('Content-Type: application/vnd.cpc.ship.rate-v4+xml', 'Accept: application/vnd.cpc.ship.rate-v4+xml'));
$curl_response = curl_exec($curl); // Execute REST Request
if(curl_errno($curl)){
    throw new Exception("Curl Error encountered.");
}

curl_close($curl);
Step 3:

Here i use SimpleXML to parse the XML response – if the response returns false i throw a new exception else i continue with parsing the response to retrieve the data that i require and then i send that data back to Javascript for further parsing.

//Use SimpleXML to parse xml response
libxml_use_internal_errors(true);
$xml = simplexml_load_string('<root>' . preg_replace('/<\?xml.*\?>/','', $curl_response) . '</root>');
if (!$xml) {
    throw new Exception("XML Parsing Failed");
} else {
    $ratesArray = [];

    if ($xml->{'price-quotes'} ) {
        $priceQuotes = $xml->{'price-quotes'}->children('http://www.canadapost.ca/ws/ship/rate-v4');
        if ( $priceQuotes->{'price-quote'} ) {
            foreach ( $priceQuotes as $priceQuote ) {  
                $ratesArray[] = ["service" => $priceQuote->{'service-name'}, "price" => $priceQuote->{'price-details'}->{'due'}];
            }
        }
    }
    if ($xml->{'messages'} ) {                    
        $messages = $xml->{'messages'}->children('http://www.canadapost.ca/ws/messages');        
    }
    
    //Rates have been fetched, continue to process $ratesArray data anyway you wish

}
Putting it all together:

The final code is executed through a try/catch block as seen below.

try {

    $env = "sandbox";
    $customer_number = "000392384";  //Fake, replace with your CA Post company number
    $username = $env == "sandbox" ? "998234kjsdfsdf" : "dfad9098234kj"; //Fake, replace with your CA Post API credentials
    $password = $env == "sandbox" ? "2adfpoijksd9889234" : "kksd023lalsdf02398sdf";  //Fake, replace with your CA Post API credentials
    $service_url = 'https://ct.soa-gw.canadapost.ca/rs/ship/price';
    
    $origin_postal_code = 'L4L9Bk'; 
    $destination_postal_code = 'L4B2B9';
    
    $xmlRequest = <<<XML
    <?xml version="1.0" encoding="UTF-8"?>
    <mailing-scenario xmlns="http://www.canadapost.ca/ws/ship/rate-v4">
    <customer-number>{$customer_number}</customer-number>
    <parcel-characteristics>
        <weight>1</weight>
        <dimensions>
            <length>13</length>
            <width>13</width>
            <height>5</height>
        </dimensions>
    </parcel-characteristics>
    <origin-postal-code>{$origin_postal_code}</origin-postal-code>
    <destination>
        <domestic>
        <postal-code>{$destination_postal_code}</postal-code>
        </domestic>
    </destination>
    </mailing-scenario>
    XML;
    
    $curl = curl_init($service_url); // Create REST Request
    curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);
    curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);
    //curl_setopt($curl, CURLOPT_CAINFO, realpath(dirname($_SERVER['SCRIPT_FILENAME'])) . '/cert/cacert.pem');
    curl_setopt($curl, CURLOPT_POST, true);
    curl_setopt($curl, CURLOPT_POSTFIELDS, $xmlRequest);
    curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
    curl_setopt($curl, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);
    curl_setopt($curl, CURLOPT_USERPWD, $username . ':' . $password);
    curl_setopt($curl, CURLOPT_HTTPHEADER, array('Content-Type: application/vnd.cpc.ship.rate-v4+xml', 'Accept: application/vnd.cpc.ship.rate-v4+xml'));
    $curl_response = curl_exec($curl); // Execute REST Request
    if(curl_errno($curl)){
        throw new Exception("Curl Error encountered.");
    }
    
    curl_close($curl);
    
    //Use SimpleXML to parse xml response
    libxml_use_internal_errors(true);
    $xml = simplexml_load_string('<root>' . preg_replace('/<\?xml.*\?>/','', $curl_response) . '</root>');
    if (!$xml) {
        throw new Exception("XML Parsing Failed");
    } else {
        $ratesArray = [];
    
        if ($xml->{'price-quotes'} ) {
            $priceQuotes = $xml->{'price-quotes'}->children('http://www.canadapost.ca/ws/ship/rate-v4');
            if ( $priceQuotes->{'price-quote'} ) {
                foreach ( $priceQuotes as $priceQuote ) {  
                    $ratesArray[] = ["service" => $priceQuote->{'service-name'}, "price" => $priceQuote->{'price-details'}->{'due'}];
                }
            }
        }
        if ($xml->{'messages'} ) {                    
            $messages = $xml->{'messages'}->children('http://www.canadapost.ca/ws/messages');        
        }
        
        //Rates have been fetched, continue to process $ratesArray data anyway you wish
        
        //Optional: This can be used to send data back to JS if required
        echo json_encode(array(
            'status' => 'success', 
            'rates' => $ratesArray,
        ));
    
    }

} catch(Exception $e) {
    //Catch errors here
    
    //Optional: This can be used to send data back to JS if required
    echo json_encode(array(
        'status' => 'error', 
        'error' => $e->getMessage()
    ));
}

And that’s all you pretty much need if you require live Canada Post shipping rates. If you are working with another programming language such as Java or C sharp then you can reference the Canada Post API docs here.

If you have any questions or require assistance with your web project feel free to reach me at leo@digitalodyssey.ca

Cheers.

  • Tagged in:

38 Tutorial Views

How to work with queries... Creating a simple Stripe checkout...
Get in Touch

Let's Work Together

Consultation request

Need help with your project but not sure how to get started?
Use the form below for a free project assessment. All information and files submitted will be held in strict confidentiality.

Please enter your full name
Please enter a valid email address
Please enter your instructions

PSD, PNG, AI, PDF, EPS, JPEG, TIF or Zip file only. Please include custom fonts if necessary.
Multiple files supported. Max size per file is 256 MB.

    Files are being uploaded, please wait.

    Submit

    Thank you!
    Your submission has been received and will be reviewed in the next 24-48 hours.

    Drop Files Here

    This website uses cookies to improve your experience. We'll assume you're ok with this, but you can opt-out if you wish.

    Accept Decline