module ActiveShipping
class ExternalReturnLabelRequest
CAP_STRING_LEN = 100
USPS_EMAIL_REGEX = /^([a-zA-Z0-9_\-\.]+)@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.)|(([a-zA-Z0-9\-]+\.)+))([a-zA-Z]{2,4}|[0-9]{1,3})(\]?)$/
LABEL_FORMAT = {
'Instructions' => 'null',
'No Instructions' => 'NOI',
'Double Label' => 'TWO'
}
SERVICE_TYPE_CODE = [
'044', '019', '596', '020', '597', '022', '024', '017', '018'
]
CALL_CENTER_OR_SELF_SERVICE = ['CallCenter', 'Customer']
LABEL_DEFINITION = ['4X6', 'Zebra-4X6', '4X4', '3X6']
IMAGE_TYPE = ['PDF', 'TIF']
attr_reader :customer_name,
:customer_address1,
:customer_address2,
:customer_city,
:customer_state,
:customer_zipcode,
:customer_urbanization,
:company_name,
:attention,
:label_format,
:label_definition,
:service_type_code,
:merchandise_description,
:insurance_amount,
:address_override_notification,
:packaging_information,
:packaging_information2,
:call_center_or_self_service,
:image_type,
:address_validation,
:sender_name,
:sender_email,
:recipient_name,
:recipient_email,
:recipient_bcc,
:merchant_account_id,
:mid
def initialize(options = {})
options.each do |pair|
self.public_send("#{pair[0]}=".to_sym, pair[1]) if self.respond_to?("#{pair[0]}=".to_sym)
end
verify_or_raise_required
end
def self.from_hash(options = {})
self.new(options)
end
# Sent by the system containing the returns label attachment and message.
def recipient_bcc=(v)
@recipient_bcc = validate_email(v, __method__)
end
# Sent by the system containing the returns label attachment and message.
# Optional.
def recipient_email=(v)
@recipient_email = validate_email(v, __method__)
end
# The name in an email sent by the system containing the returns label attachment.
# Optional.
def recipient_name=(v)
@recipient_name = nil
if (v = sanitize(v)) && v.length > 0
@recipient_name = v
else
raise USPSValidationError, "'#{v}' is not a valid string in #{__method__}"
end
end
# The From address in an email sent by the system containing the returns
# label attachment and message, Defaults to DONOTREPLY@USPSReturns.com
# if a recipient email is entered and a sender email is not.
# Optional.
def sender_email=(v)
@sender_email = validate_email(v, __method__)
end
# The From name in an email sent by the system containing the returns
# label attachment. Defaults to “Merchant Returns” if a recipient name
# is entered and a sender name is not.
# Optional.
def sender_name=(v)
@sender_name = nil
if (v = sanitize(v)) && v.length > 0
@sender_name = v
else
raise USPSValidationError, "'#{v}' is not a valid string in #{__method__}"
end
end
# Used to override the validation of the customer address.
# If true, the address will be validated against WebTools.
# If false, the system will bypass the validation.
# Optional.
def address_validation=(v)
@address_validation = to_bool(v, true)
end
# Used to select the format of the return label.
# Optional.
# * PDF Default.
# * TIF
def image_type=(v)
@image_type = validate_set_inclusion(v.to_s.upcase, IMAGE_TYPE, __method__)
end
# Used to determine if the returns label request is coming from a
# merchant call center agent or an end customer.
# Required.
# [CallCenter]
# [Customer]
def call_center_or_self_service=(v)
@call_center_or_self_service = validate_set_inclusion(v, CALL_CENTER_OR_SELF_SERVICE, __method__)
end
# Package information can be one of three types: RMA, Invoice or
# Order number. This will appear on the second label generated when
# the LabelFormat “TWO” is selected.
# Optional.
def packaging_information2=(v)
@packaging_information2 = validate_string_length(v, 15, __method__)
end
# Package information can be one of three types: RMA, Invoice or
# Order number. This will appear on the generated label.
# Optional.
def packaging_information=(v)
@packaging_information = validate_string_length(v, 15, __method__)
end
# Override address if more address information
# is needed or system cannot find address. If
# the address_override_notification value is
# true then any address error being passed from
# WebTools would be bypassed and a successful
# response will be sent.
# Required.
def address_override_notification=(v)
@address_validation = to_bool(v)
end
# Insured amount of package.
def insurance_amount=(v)
@insurance_amount = nil
if (1..200).include?(v.to_f)
@insurance_amount = v
else
raise USPSValidationError, "#{__method__} must be a numerical value between 1 and 200, found value '#{v}'."
end
end
# Description of the merchandise.
# Optional.
def merchandise_description=(v)
@merchandise_description = validate_string_length(v, 255, __method__)
end
# Service type of the label as specified in the merchant profile setup.
# Required.
# [044] (Parcel Return Service)
# [019] (Priority Mail Returns service)
# [596] (Priority Mail Returns service, Insurance <= $200)
# [020] (First-Class Package Return service)
# [597] (First-Class Package Return service, Insurance <= $200)
# [022] (Ground Return Service)
# [024] (PRS – Full Network)
# [017] (PRS – Full Network, Insurance <=$200)
# [018] (PRS – Full Network, Insurance >$200)
def service_type_code=(v)
@service_type_code = validate_set_inclusion(v, SERVICE_TYPE_CODE, __method__)
end
# Size of the label.
# Required.
# * 4X6
# * Zebra-4X6
# * 4X4
# * 3X6
def label_definition=(v)
@label_definition = validate_set_inclusion(v, LABEL_DEFINITION, __method__)
end
def label_format
@label_format && LABEL_FORMAT[@label_format]
end
# Format in which the label(s) will be printed.
# * null (“Instructions”)
# * NOI (“No Instructions”)
# * TWO (“Double Label”)
def label_format=(v)
@label_format = validate_set_inclusion(v, LABEL_FORMAT.keys, __method__)
end
# The intended recipient of the returned package (e.g. Returns Department).
# Optional.
def attention=(v)
@attention = validate_string_length(v, 38, __method__)
end
# The name of the company to which the package is being returned.
# Optional.
def company_name=(v)
@company_name = validate_string_length(v, 38, __method__)
end
# Required.
def merchant_account_id=(v)
@merchant_account_id = nil
if v.to_i > 0
@merchant_account_id = v
else
raise USPSValidationError, "#{__method__} must be a valid positive integer, found value '#{v}'."
end
end
# Required.
def mid=(v)
@mid = nil
if v.to_s =~ /^\d{6,9}$/
@mid = v
else
raise USPSValidationError, "#{__method__} must be a valid integer between 6 and 9 digits in length, found value '#{v}'."
end
end
# Urbanization of customer returning the package (only applicable to Puerto Rico addresses).
# Optional.
def customer_urbanization=(v)
@customer_urbanization = validate_string_length(v, 32, __method__)
end
# Name of customer returning package.
# Required.
def customer_name=(v)
@customer_name = validate_range(v, 1, 32, __method__)
end
# Address of the customer returning the package.
# Required.
def customer_address1=(v)
@customer_address1 = validate_range(v, 1, 32, __method__)
end
# Secondary address unit designator / number of customer
# returning the package. (such as an apartment or
# suite number, e.g. APT 202, STE 100)
def customer_address2=(v)
@customer_address2 = validate_range(v, 0, 32, __method__)
end
# City of customer returning the package.
# Required.
def customer_city=(v)
@customer_city = validate_range(v, 1, 32, __method__)
end
# State of customer returning the package.
# Required.
def customer_state=(v)
@customer_state = nil
if (v = sanitize(v)) && v =~ /^[a-zA-Z]{2}$/
@customer_state = v
else
raise USPSValidationError, "#{__method__} must be a String 2 chars in length, found value '#{v}'."
end
end
# Zipcode of customer returning the package.
# According to the USPS documentation, Zipcode is optional
# unless address_override_notification is true
# and address_validation is set to false.
# It's probably just easier to require Zipcodes.
# Required.
def customer_zipcode=(v)
@customer_zipcode = nil
if (v = sanitize(v))
v = v[0..4]
if v =~ /^\d{5}$/
@customer_zipcode = v
end
else
raise USPSValidationError, "#{__method__} must be a 5 digit number, found value '#{v}'."
end
end
def verify_or_raise_required
%w(customer_name customer_address1 customer_city customer_state
customer_zipcode label_format label_definition service_type_code
call_center_or_self_service).each do |attr|
raise USPSMissingRequiredTagError.new(attr.camelize, attr) unless send(attr.to_sym)
end
# Safer than using inflection acroynms
raise USPSMissingRequiredTagError.new("MID", "mid") unless mid
raise USPSMissingRequiredTagError.new("MerchantAccountID", "merchant_account_id") unless merchant_account_id
end
def to_xml
xml_builder = Nokogiri::XML::Builder.new do |xml|
xml.ExternalReturnLabelRequest do
xml.CustomerName { xml.text(customer_name) }
xml.CustomerAddress1 { xml.text(customer_address1) }
xml.CustomerAddress2 { xml.text(customer_address2) } if customer_address2
xml.CustomerCity { xml.text(customer_city) }
xml.CustomerState { xml.text(customer_state) }
xml.CustomerZipCode { xml.text(customer_zipcode) } if customer_zipcode
xml.CustomerUrbanization { xml.text(customer_urbanization) } if customer_urbanization
xml.MerchantAccountID { xml.text(merchant_account_id) }
xml.MID { xml.text(mid) }
xml.SenderName { xml.text(sender_name) } if sender_name
xml.SenderEmail { xml.text(sender_email) } if sender_email
xml.RecipientName { xml.text(recipient_name) } if recipient_name
xml.RecipientEmail { xml.text(recipient_email) } if recipient_email
xml.RecipientBcc { xml.text(recipient_bcc) } if recipient_bcc
xml.LabelFormat { xml.text(label_format) } if label_format
xml.LabelDefinition { xml.text(label_definition) } if label_definition
xml.ServiceTypeCode { xml.text(service_type_code) } if service_type_code
xml.CompanyName { xml.text(company_name) } if company_name
xml.Attention { xml.text(attention) } if attention
xml.CallCenterOrSelfService { xml.text(call_center_or_self_service) }
xml.MerchandiseDescription { xml.text(merchandise_description) } if merchandise_description
xml.InsuranceAmount { xml.text(insurance_amount) } if insurance_amount
xml.AddressOverrideNotification { xml.text(!!address_override_notification) }
xml.PackageInformation { xml.text(packaging_information) } if packaging_information
xml.PackageInformation2 { xml.text(packaging_information2) } if packaging_information2
xml.ImageType { xml.text(image_type) } if image_type
xml.AddressValidation { xml.text(!!address_validation) }
end
end
xml_builder.to_xml
end
private
def to_bool(v, default = false)
v = v.to_s
if v =~ (/(true|yes|1)$/i)
true
elsif v =~ (/(false|no|0)$/i)
false
else
default
end
end
def sanitize(v)
if v.is_a?(String)
v.strip!
v[0..CAP_STRING_LEN - 1]
else
nil
end
end
def validate_range(v, min, max, meth)
if (v = sanitize(v).to_s) && ((min.to_i)..(max.to_i)).include?(v.length)
if v.length == 0
nil
else
v
end
else
raise USPSValidationError, "#{meth} must be a String between #{min.to_i} and #{max.to_i} chars in length, found value '#{v}'."
end
end
def validate_string_length(s, max_len, meth)
if (s = sanitize(s)) && s.length <= max_len.to_i
s
else
raise USPSValidationError, "#{meth} must be a String no more than #{max_len} chars in length, found value '#{s}'."
end
end
def validate_set_inclusion(v, set, meth)
if set.respond_to?(:include?) && set.include?(v)
v
else
raise USPSValidationError, "#{v} is not valid in #{meth}, try any of the following: #{(set.respond_to?(:join) && set.join(',')) || ''}"
end
end
def validate_email(v, meth)
if (v = sanitize(v)) && v =~ USPS_EMAIL_REGEX
v
else
raise USPSValidationError, "'#{v}' is not a valid e-mail in #{meth}"
end
end
end
end