This is a PHP helper class for creating custom shipping rates for the FoxyCart Custom Shipping Endpoint functionality added in version 2.0
- PHP 5.3+
- FoxyCart 2.0
FoxyCart POSTs to your endpoint script a JSON payload containing information about the customers cart and address. As an example of a simple endpoint using the ShippingResponse helper class:
<?php
require('ShippingResponse.php');
$rawPost = file_get_contents('php://input');
$cart_details = json_decode($rawPost, true);
$rates = new ShippingResponse($cart_details);
$rates->add(10001, 5, 'FoxyPost', 'Standard');
$rates->add(10002, 15.50, 'FoxyPost', 'Express');
header('Content-Type: application/json');
print $rates->output();
For an example of the payload that is sent to your endpoint, please refer to the example on the FoxyCart wiki.
To start creating your custom rates, you first need to create the ShippingResponse
object. The requires that you've included the ShippingResponse.php
class file in your script, and you pass the JSON object of the cart data payload your endpoint received when creating the object:
require('ShippingResponse.php');
$rawPost = file_get_contents('php://input');
$cart_details = json_decode($rawPost, true);
$rates = new ShippingResponse($cart_details);
The ShippingResponse
class adds a number of functions you can utilise to execute your custom shipping logic:
Adds an additional shipping option to the checkout.
service_id
(Number) - Minimum of 10000. Can not be the same as another rateprice
(Number) - A number to two decimal placesmethod
(String)service_name
(String)add_flat_rate
(Boolean) - Optional, default:true
- If true, any applicable flat rates will be added onto the price of this rateadd_handling
(Boolean) - Optional, default:true
- If true, any applicable handling fees will be added onto the price of this rate
$rates->add(10001, 5, 'FoxyPost', 'Standard');
$rates->add(10002, 0, false, 'Free Shipping', false, false);
- If the
service_id
provided is less than 10000, the provided rate will be added to it (eg: if 125 is provided as theservice_id
, the resulting id will actually be 10125). - Make sure that the
service_id
you use doesn't duplicate a id for an existing rate. - You don't have to provide both the carrier and the service parameters - but at least one of them is required.
Hides one or many existing shipping options.
selector
(Number, String or Array) - Can be the service id of a rate, a string containing the carrier or the service (or a combination of) or an array of service ids.
$rates->hide(10001); // Will hide rate 10001
$rates->hide('all'); // Will hide all rates
$rates->hide('FedEx'); // Will hide all rates for FedEx
$rates->hide('Overnight'); // Will hide all rates with a service name that contains 'Overnight'
$rates->hide('USPS Express'); // Will hide any rates from USPS that contain the word 'Express'
$rates->hide([10001,10005,10007]); // Will hide rates with codes 10001, 10005 and 10007
- Any rates that are still hidden at the end of the custom logic block will be removed and not passed back to the checkout in the response.
- If you hide all rates without adding or showing at least one before outputting the rates, it will result in a generic error
Shows one or many existing shipping options that have previously been hidden.
selector
(Number, String or Array) - Can be the service id of a rate, a string containing the carrier or the service (or a combination of) or an array of service ids.
$rates->show(10001); // Will show rate 10001
$rates->show('all'); // Will show all rates
$rates->show('FedEx'); // Will show all rates for FedEx
$rates->show('Overnight'); // Will show all rates with a service name that contains 'Overnight'
$rates->show('USPS Express'); // Will show any rates from USPS that contain the word 'Express'
$rates->show([10001,10005,10007]); // Will show rates with codes 10001, 10005 and 10007
Updates one or many existing shipping options.
selector
(Number, String or Array) - Can be the service id of a rate, a string containing the carrier or the service (or a combination of) or an array of service ids.modifier
(String or Number) - Can either be a number (which sets the price to match) or a string containing the operator and a number eg+20
,-10
,*2
,/2.5
,=15
. You can also append the string with a%
sign to make the operation based on a percentage, eg+20%
- add 20%,-20%
- less 20%,/20%
- divide by 20%,*20%
- multiply by 20%.method
(String) - Optional, if provided replaces the current methodservice_name
(String) - Optional, if provided replaces the current service name
$rates->update(10001, 5); // Will set rate 10001 to be $5
$rates->update('all', '*2'); // Will set all current rates to double their current cost
$rates->update('FedEx', '+5'); // Will set all rates for FedEx to be $5 more than what they are currently
$rates->update('Overnight', '-5'); // Will set all rates with a service name that contains 'Overnight' to be $5 less than currently set
$rates->update('USPS Express', '=6'); // Will set any rates from USPS that contain the word 'Express' to be $6
$rates->update([10001,10005,10007], '/2'); // Will set rates with codes 10001, 10005 and 10007 to be half their current cost
$rates->update('USPS', '+20%'); // Will add 20% of the current rate to each of the USPS rates
$rates->update('USPS Ground', false, false, 'Super Saver'); // Will change “USPS Ground” to be called “USPS Super Saver”
Resets the shipping results to be empty and clears out any error message.
$rates->reset();
Set an error response for the rates. If both an error message and valid shipping rates are present, the error message will take precendence.
message
(String) - If not passed, any existing error message will be cleared
$rates->error("Sorry, we can't ship to Canada");
Returns the current rates or error message as a JSON encoded string
output_as_string
(Boolean) - Optional, default:true
- If true, the output will be passed throughjson_encode()
to return the JSON object as a string.
print $rates->output();
The following are examples of the custom logic for common shipping requirements. They all assume a basic endpoint with the following code, and that the ShippingReponse.php
class file exists in the same directory:
<?php
require('ShippingResponse.php');
$rawPost = file_get_contents('php://input');
$cart_details = json_decode($rawPost, true);
$rates = new ShippingResponse($cart_details);
// Custom logic placed here
header('Content-Type: application/json');
print $rates->output();
- Free shipping if the customer orders $40 or more, otherwise it's $5 flat rate
$rates->add(10001, 5, 'FoxyPost', 'Standard');
if ($cart_details['_embedded']['fx:shipment']['total_item_price'] >= 40) {
$rates->update(10001, 0);
}
- 3 default shipping options: standard, priority and express.
- If the total weight of the cart is greater that 10, adjust the shipping costs.
- If there are more than 5 products, remove the express option.
$rates->add(10001, 5, 'FoxyPost', 'Standard');
$rates->add(10002, 9.45, 'FoxyPost', 'Priority');
$rates->add(10003, 10, 'FoxyPost', 'Express (Next Day)');
if ($cart_details['_embedded']['fx:shipment']['total_weight'] > 10) {
$rates->update(10001, 6);
$rates->update(10002, 10);
$rates->update(10003, 11.99);
}
if ($cart_details['_embedded']['fx:shipment']['item_count'] > 5) {
$rates->hide(10003);
}
- Postage is calculated as a base price per product, with each subsequent product adding an additional cost.
- Two different groups of shipping options are presented, one for local delivery within the US, and one for international addresses based off of the shipping country.
$item_count = $cart_details['_embedded']['fx:shipment']['item_count'];
if ($cart_details['_embedded']['fx:shipment']['country'] == "US") {
$postage = 10 + (($item_count - 1) * 0.50);
$rates->add(10001, $postage, 'FoxyPost', 'Standard');
$postage = 12 + (($item_count - 1) * 1.50);
$rates->add(10002, $postage, 'FoxyPost', 'Express');
} else {
$postage = 15 + (($item_count - 1) * 2);
$rates->add(10003, $postage, 'FoxyPost', 'International');
}
- Postage is assigned per category.
- If there is a product from CategoryA in the cart, then present express option
- If there is only a product from CategoryB in the cart, provide free shipping as an option
- Ensure that any existing flat rate and handling fees don't get added on to the free shipping option
$hasCategoryA = false;
$hasCategoryB = false;
foreach($cart_details['_embedded']['fx:items'] as $item) {
switch ($item['_embedded']['fx:item_category']['code']) {
case "CategoryA":
$hasCategoryA = true;
break;
case "CategoryB":
$hasCategoryB = true;
break;
}
}
if ($hasCategoryB && !$hasCategoryA) {
$rates->add(10001, 0, '', 'Free Ground Shipping', false, false);
} else if ($hasCategoryA) {
$rates->add(10002, 5.99, 'FoxyPost', 'Express');
}
- Postage is flat rate, but free if “free shipping” coupon is present
- Allow free shipping only if a certain coupon code is present. In this example, one with a code of
freeshipping
.
$rates->add(10001, 5, "FoxyPost", "Standard");
foreach ($cart_details['_embedded']['fx:discounts'] as $discount) {
if ($discount['code'] == "freeshipping") {
$rates->update(10001, 0);
}
}
- Pricing tiers, one for the UK, one for Europe and then the rest of the world
$tier1 = array('GB');
$tier2 = array('AL', 'AD', 'AM', 'AT', 'BY', 'BE', 'BA', 'BG', 'CH', 'CY', 'CZ', 'DE', 'DK', 'EE', 'ES', 'FO', 'FI', 'FR', 'GB', 'GE', 'GI', 'GR', 'HU', 'HR', 'IE', 'IS', 'IT', 'LT', 'LU', 'LV', 'MC', 'MK', 'MT', 'NO', 'NL', 'PL', 'PT', 'RO', 'RU', 'SE', 'SI', 'SK', 'SM', 'TR', 'UA', 'VA');
$country = $cart_details['_embedded']['fx:shipment']['country'];
if (in_array($country, $tier1)) {
// United Kingdom
$rates->add(1, 10, 'FoxyPost', 'Standard');
} else if (in_array($country, $tier2)) {
// Europe
$rates->add(2, 20, 'FoxyPost', 'International');
} else {
// Rest of world
$rates->add(3, 30, 'FoxyPost', 'International');
}
"Error: This store has not been setup correctly to calculate shipping to this location with this weight."
If you receive this error in your store, that means that either no rates were returned from your custom endpoint, or it didn't return a valid JSON object.
- Firstly ensure that your endpoint is set to return at least one rate or an error for all requests.
- If you are - ensure that only the JSON output is printed on the page. You can't have any other text on the page.
If you're using PHP 5.6, there is a known issue with that related to the always_populate_raw_post_data
setting in PHP. It defaults to 0
in that version, and when receiving a JSON POST to the page, will output an error warning on the page like this:
PHP Deprecated: Automatically populating $HTTP_RAW_POST_DATA is deprecated and will be removed in a future version. To avoid this warning set 'always_populate_raw_post_data' to '-1' in php.ini and use the php://input stream instead. in Unknown on line 0
To work around this, you could use a more recent version of PHP or you can set the always_populate_raw_post_data
to -1
if you have access to your PHP settings. Alternatively you can also set display_errors
to Off
to prevent it outputting the warning. On your production environment, display_errors
should be off anyway for security - but this may affect your development environments too.
Copyright (c) Foxy.io
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.