active_shipping 0.9.15 → 0.10.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -31,12 +31,14 @@ module ActiveMerchant
31
31
  API_CODES = {
32
32
  :us_rates => 'RateV4',
33
33
  :world_rates => 'IntlRateV2',
34
- :test => 'CarrierPickupAvailability'
34
+ :test => 'CarrierPickupAvailability',
35
+ :track => 'TrackV2'
35
36
  }
36
37
  USE_SSL = {
37
38
  :us_rates => false,
38
39
  :world_rates => false,
39
- :test => true
40
+ :test => true,
41
+ :track => false
40
42
  }
41
43
  CONTAINERS = {
42
44
  :envelope => 'Flat Rate Envelope',
@@ -48,6 +50,7 @@ module ActiveMerchant
48
50
  :matter_for_the_blind => 'Matter for the blind',
49
51
  :envelope => 'Envelope'
50
52
  }
53
+
51
54
  PACKAGE_PROPERTIES = {
52
55
  'ZipOrigination' => :origin_zip,
53
56
  'ZipDestination' => :destination_zip,
@@ -72,10 +75,20 @@ module ActiveMerchant
72
75
  :parcel => 'PARCEL',
73
76
  :media => 'MEDIA',
74
77
  :library => 'LIBRARY',
78
+ :online => 'ONLINE',
75
79
  :all => 'ALL'
76
80
  }
81
+ FIRST_CLASS_MAIL_TYPES = {
82
+ :letter => 'LETTER',
83
+ :flat => 'FLAT',
84
+ :parcel => 'PARCEL',
85
+ :post_card => 'POSTCARD',
86
+ :package_service => 'PACKAGESERVICE'
87
+ }
88
+
89
+ # Array of U.S. possessions according to USPS: https://www.usps.com/ship/official-abbreviations.htm
90
+ US_POSSESSIONS = ["AS", "FM", "GU", "MH", "MP", "PW", "PR", "VI"]
77
91
 
78
- # TODO: get rates for "U.S. possessions and Trust Territories" like Guam, etc. via domestic rates API: http://www.usps.com/ncsc/lookups/abbr_state.txt
79
92
  # TODO: figure out how USPS likes to say "Ivory Coast"
80
93
  #
81
94
  # Country names:
@@ -112,6 +125,13 @@ module ActiveMerchant
112
125
  "WS" => "Western Samoa"
113
126
  }
114
127
 
128
+ def find_tracking_info(tracking_number, options={})
129
+ options = @options.update(options)
130
+ tracking_request = build_tracking_request(tracking_number, options)
131
+ response = commit(:track, tracking_request, (options[:test] || false))
132
+ parse_tracking_response(response, options)
133
+ end
134
+
115
135
  def self.size_code_for(package)
116
136
  if package.inches(:max) <= 12
117
137
  'REGULAR'
@@ -149,10 +169,10 @@ module ActiveMerchant
149
169
 
150
170
  #raise ArgumentError.new("USPS packages must originate in the U.S.") unless ['US',nil].include?(origin.country_code(:alpha2))
151
171
 
152
-
153
172
  # domestic or international?
154
173
 
155
- response = if ['US',nil].include?(destination.country_code(:alpha2))
174
+ domestic_codes = US_POSSESSIONS + ['US', nil]
175
+ response = if domestic_codes.include?(destination.country_code(:alpha2))
156
176
  us_rates(origin, destination, packages, options)
157
177
  else
158
178
  world_rates(origin, destination, packages, options)
@@ -169,7 +189,86 @@ module ActiveMerchant
169
189
  end
170
190
 
171
191
  protected
172
-
192
+ def response_success?(xml)
193
+ xml.get_text('/*/Response/ResponseStatusCode').to_s == '1'
194
+ end
195
+
196
+ def response_message(xml)
197
+ xml.get_text('/*/Response/Error/ErrorDescription | /*/Response/ResponseStatusDescription').to_s
198
+ end
199
+
200
+ def parse_tracking_response(response, options={})
201
+ xml = REXML::Document.new(response)
202
+ success = response_success?(xml)
203
+ message = response_message(xml)
204
+
205
+ if success
206
+ tracking_number, origin, destination = nil
207
+ shipment_events = []
208
+
209
+ first_shipment = xml.elements['/*/Shipment']
210
+ first_package = first_shipment.elements['Package']
211
+ tracking_number = first_shipment.get_text('ShipmentIdentificationNumber | Package/TrackingNumber').to_s
212
+
213
+ origin, destination = %w{Shipper ShipTo}.map do |location|
214
+ location_from_address_node(first_shipment.elements["#{location}/Address"])
215
+ end
216
+
217
+ activities = first_package.get_elements('Activity')
218
+ unless activities.empty?
219
+ shipment_events = activities.map do |activity|
220
+ description = activity.get_text('Status/StatusType/Description').to_s
221
+ zoneless_time = if (time = activity.get_text('Time')) &&
222
+ (date = activity.get_text('Date'))
223
+ time, date = time.to_s, date.to_s
224
+ hour, minute, second = time.scan(/\d{2}/)
225
+ year, month, day = date[0..3], date[4..5], date[6..7]
226
+ Time.utc(year, month, day, hour, minute, second)
227
+ end
228
+ location = location_from_address_node(activity.elements['ActivityLocation/Address'])
229
+ ShipmentEvent.new(description, zoneless_time, location)
230
+ end
231
+
232
+ shipment_events = shipment_events.sort_by(&:time)
233
+
234
+ if origin
235
+ first_event = shipment_events[0]
236
+ same_country = origin.country_code(:alpha2) == first_event.location.country_code(:alpha2)
237
+ same_or_blank_city = first_event.location.city.blank? or first_event.location.city == origin.city
238
+ origin_event = ShipmentEvent.new(first_event.name, first_event.time, origin)
239
+ if same_country and same_or_blank_city
240
+ shipment_events[0] = origin_event
241
+ else
242
+ shipment_events.unshift(origin_event)
243
+ end
244
+ end
245
+ if shipment_events.last.name.downcase == 'delivered'
246
+ shipment_events[-1] = ShipmentEvent.new(shipment_events.last.name, shipment_events.last.time, destination)
247
+ end
248
+ end
249
+
250
+ end
251
+ TrackingResponse.new(success, message, Hash.from_xml(response).values.first,
252
+ :xml => response,
253
+ :request => last_request,
254
+ :shipment_events => shipment_events,
255
+ :origin => origin,
256
+ :destination => destination,
257
+ :tracking_number => tracking_number)
258
+ end
259
+
260
+
261
+
262
+
263
+
264
+
265
+ def build_tracking_request(tracking_number, options={})
266
+ xml_request = XmlNode.new('TrackRequest', 'USERID' => @options[:login]) do |root_node|
267
+ root_node << XmlNode.new('TrackID', :ID => tracking_number)
268
+ end
269
+ URI.encode(xml_request.to_s)
270
+ end
271
+
173
272
  def us_rates(origin, destination, packages, options={})
174
273
  request = build_us_rate_request(packages, origin.zip, destination.zip, options)
175
274
  # never use test mode; rate requests just won't work on test servers
@@ -204,7 +303,15 @@ module ActiveMerchant
204
303
  request = XmlNode.new('RateV4Request', :USERID => @options[:login]) do |rate_request|
205
304
  packages.each_with_index do |p,id|
206
305
  rate_request << XmlNode.new('Package', :ID => id.to_s) do |package|
207
- package << XmlNode.new('Service', US_SERVICES[options[:service] || :all])
306
+ if @options[:commercial_base] == true
307
+ raise ArgumentError.new("Commercial Base rates are only provided with the :online method.") if !options[:service].blank? && options[:service] != :online
308
+ default_service = :online
309
+ else
310
+ default_service = :all
311
+ end
312
+
313
+ package << XmlNode.new('Service', US_SERVICES[options[:service] || default_service])
314
+ package << XmlNode.new('FirstClassMailType', FIRST_CLASS_MAIL_TYPES[options[:first_class_mail_type]])
208
315
  package << XmlNode.new('ZipOrigination', strip_zip(origin_zip))
209
316
  package << XmlNode.new('ZipDestination', strip_zip(destination_zip))
210
317
  package << XmlNode.new('Pounds', 0)
@@ -264,6 +371,7 @@ module ActiveMerchant
264
371
  package << XmlNode.new('Length', "%0.2f" % [p.inches(:length), 0.01].max)
265
372
  package << XmlNode.new('Height', "%0.2f" % [p.inches(:height), 0.01].max)
266
373
  package << XmlNode.new('Girth', "%0.2f" % [p.inches(:girth), 0.01].max)
374
+ package << XmlNode.new('CommercialFlag', 'Y') if @options[:commercial_base]
267
375
  end
268
376
  end
269
377
  end
@@ -318,8 +426,13 @@ module ActiveMerchant
318
426
  return false unless (root_node = response_node.elements['/IntlRateV2Response | /RateV4Response'])
319
427
  domestic = (root_node.name == 'RateV4Response')
320
428
 
321
- domestic_elements = ['Postage', 'CLASSID', 'MailService', 'Rate']
322
- international_elements = ['Service', 'ID', 'SvcDescription', 'Postage']
429
+ if @options[:commercial_base]
430
+ domestic_elements = ['Postage', 'CLASSID', 'MailService', 'CommercialRate']
431
+ international_elements = ['Service', 'ID', 'SvcDescription', 'CommercialPostage']
432
+ else
433
+ domestic_elements = ['Postage', 'CLASSID', 'MailService', 'Rate']
434
+ international_elements = ['Service', 'ID', 'SvcDescription', 'Postage']
435
+ end
323
436
  service_node, service_code_node, service_name_node, rate_node = domestic ? domestic_elements : international_elements
324
437
 
325
438
  root_node.each_element('Package') do |package_node|
@@ -420,6 +533,69 @@ module ActiveMerchant
420
533
 
421
534
  return valid
422
535
  end
536
+
537
+ def parse_tracking_response(response, options)
538
+ xml = REXML::Document.new(response)
539
+ root_node = xml.elements['TrackResponse']
540
+
541
+ success = response_success?(xml)
542
+ message = response_message(xml)
543
+
544
+ if success
545
+ tracking_number, origin, destination = nil
546
+ shipment_events = []
547
+ tracking_details = xml.elements.collect('*/*/TrackDetail'){ |e| e }
548
+
549
+ tracking_summary = xml.elements.collect('*/*/TrackSummary'){ |e| e }.first
550
+ tracking_details << tracking_summary
551
+
552
+ tracking_number = root_node.elements['TrackInfo'].attributes['ID'].to_s
553
+
554
+ tracking_details.each do |event|
555
+ location = nil
556
+ timestamp = nil
557
+ description = nil
558
+ if event.get_text.to_s =~ /^(.*), (\w+ \d\d, \d{4}, \d{1,2}:\d\d [ap]m), (.*), (\w\w) (\d{5})$/i ||
559
+ event.get_text.to_s =~ /^Your item \w{2,3} (out for delivery|delivered) at (\d{1,2}:\d\d [ap]m on \w+ \d\d, \d{4}) in (.*), (\w\w) (\d{5})\.$/i
560
+ description = $1.upcase
561
+ timestamp = $2
562
+ city = $3
563
+ state = $4
564
+ zip_code = $5
565
+ location = Location.new(:city => city, :state => state, :postal_code => zip_code, :country => 'USA')
566
+ end
567
+ if location
568
+ time = Time.parse(timestamp)
569
+ zoneless_time = Time.utc(time.year, time.month, time.mday, time.hour, time.min, time.sec)
570
+ shipment_events << ShipmentEvent.new(description, zoneless_time, location)
571
+ end
572
+ end
573
+ shipment_events = shipment_events.sort_by(&:time)
574
+ end
575
+
576
+ TrackingResponse.new(success, message, Hash.from_xml(response),
577
+ :carrier => @@name,
578
+ :xml => response,
579
+ :request => last_request,
580
+ :shipment_events => shipment_events,
581
+ :destination => destination,
582
+ :tracking_number => tracking_number
583
+ )
584
+ end
585
+
586
+ def response_status_node(document)
587
+ document.elements['*/*/TrackSummary'] || document.elements['Error/Description']
588
+ end
589
+
590
+ def response_success?(document)
591
+ summary = response_status_node(document).get_text.to_s
592
+ !(summary =~ /There is no record of that mail item/ || summary =~ /This Information has not been included in this Test Server\./)
593
+ end
594
+
595
+ def response_message(document)
596
+ response_node = response_status_node(document)
597
+ response_status_node(document).get_text.to_s
598
+ end
423
599
 
424
600
  def commit(action, request, test = false)
425
601
  ssl_get(request_url(action, request, test))
@@ -0,0 +1,616 @@
1
+ # -*- encoding: utf-8 -*-
2
+ require 'cgi'
3
+
4
+ module ActiveMerchant
5
+ module Shipping
6
+
7
+ # After getting an API login from USPS (looks like '123YOURNAME456'),
8
+ # run the following test:
9
+ #
10
+ # usps = USPS.new(:login => '123YOURNAME456', :test => true)
11
+ # usps.valid_credentials?
12
+ #
13
+ # This will send a test request to the USPS test servers, which they ask you
14
+ # to do before they put your API key in production mode.
15
+ class USPS < Carrier
16
+ self.retry_safe = true
17
+
18
+ cattr_reader :name
19
+ @@name = "USPS"
20
+
21
+ LIVE_DOMAIN = 'production.shippingapis.com'
22
+ LIVE_RESOURCE = 'ShippingAPI.dll'
23
+
24
+ TEST_DOMAINS = { #indexed by security; e.g. TEST_DOMAINS[USE_SSL[:rates]]
25
+ true => 'secure.shippingapis.com',
26
+ false => 'testing.shippingapis.com'
27
+ }
28
+
29
+ TEST_RESOURCE = 'ShippingAPITest.dll'
30
+
31
+ API_CODES = {
32
+ :us_rates => 'RateV4',
33
+ :world_rates => 'IntlRateV2',
34
+ :test => 'CarrierPickupAvailability',
35
+ :track => 'TrackV2'
36
+ }
37
+ USE_SSL = {
38
+ :us_rates => false,
39
+ :world_rates => false,
40
+ :test => true,
41
+ :track => false
42
+ }
43
+ CONTAINERS = {
44
+ :envelope => 'Flat Rate Envelope',
45
+ :box => 'Flat Rate Box'
46
+ }
47
+ MAIL_TYPES = {
48
+ :package => 'Package',
49
+ :postcard => 'Postcards or aerogrammes',
50
+ :matter_for_the_blind => 'Matter for the blind',
51
+ :envelope => 'Envelope'
52
+ }
53
+
54
+ PACKAGE_PROPERTIES = {
55
+ 'ZipOrigination' => :origin_zip,
56
+ 'ZipDestination' => :destination_zip,
57
+ 'Pounds' => :pounds,
58
+ 'Ounces' => :ounces,
59
+ 'Container' => :container,
60
+ 'Size' => :size,
61
+ 'Machinable' => :machinable,
62
+ 'Zone' => :zone,
63
+ 'Postage' => :postage,
64
+ 'Restrictions' => :restrictions
65
+ }
66
+ POSTAGE_PROPERTIES = {
67
+ 'MailService' => :service,
68
+ 'Rate' => :rate
69
+ }
70
+ US_SERVICES = {
71
+ :first_class => 'FIRST CLASS',
72
+ :priority => 'PRIORITY',
73
+ :express => 'EXPRESS',
74
+ :bpm => 'BPM',
75
+ :parcel => 'PARCEL',
76
+ :media => 'MEDIA',
77
+ :library => 'LIBRARY',
78
+ :all => 'ALL'
79
+ }
80
+ FIRST_CLASS_MAIL_TYPES = {
81
+ :letter => 'LETTER',
82
+ :flat => 'FLAT',
83
+ :parcel => 'PARCEL',
84
+ :post_card => 'POSTCARD',
85
+ :package_service => 'PACKAGESERVICE'
86
+ }
87
+
88
+ # TODO: get rates for "U.S. possessions and Trust Territories" like Guam, etc. via domestic rates API: http://www.usps.com/ncsc/lookups/abbr_state.txt
89
+ # TODO: figure out how USPS likes to say "Ivory Coast"
90
+ #
91
+ # Country names:
92
+ # http://pe.usps.gov/text/Imm/immctry.htm
93
+ COUNTRY_NAME_CONVERSIONS = {
94
+ "BA" => "Bosnia-Herzegovina",
95
+ "CD" => "Congo, Democratic Republic of the",
96
+ "CG" => "Congo (Brazzaville),Republic of the",
97
+ "CI" => "Côte d'Ivoire (Ivory Coast)",
98
+ "CK" => "Cook Islands (New Zealand)",
99
+ "FK" => "Falkland Islands",
100
+ "GB" => "Great Britain and Northern Ireland",
101
+ "GE" => "Georgia, Republic of",
102
+ "IR" => "Iran",
103
+ "KN" => "Saint Kitts (St. Christopher and Nevis)",
104
+ "KP" => "North Korea (Korea, Democratic People's Republic of)",
105
+ "KR" => "South Korea (Korea, Republic of)",
106
+ "LA" => "Laos",
107
+ "LY" => "Libya",
108
+ "MC" => "Monaco (France)",
109
+ "MD" => "Moldova",
110
+ "MK" => "Macedonia, Republic of",
111
+ "MM" => "Burma",
112
+ "PN" => "Pitcairn Island",
113
+ "RU" => "Russia",
114
+ "SK" => "Slovak Republic",
115
+ "TK" => "Tokelau (Union) Group (Western Samoa)",
116
+ "TW" => "Taiwan",
117
+ "TZ" => "Tanzania",
118
+ "VA" => "Vatican City",
119
+ "VG" => "British Virgin Islands",
120
+ "VN" => "Vietnam",
121
+ "WF" => "Wallis and Futuna Islands",
122
+ "WS" => "Western Samoa"
123
+ }
124
+
125
+ def find_tracking_info(tracking_number, options={})
126
+ options = @options.update(options)
127
+ tracking_request = build_tracking_request(tracking_number, options)
128
+ response = commit(:track, tracking_request, (options[:test] || false))
129
+ parse_tracking_response(response, options)
130
+ end
131
+
132
+ def self.size_code_for(package)
133
+ if package.inches(:max) <= 12
134
+ 'REGULAR'
135
+ else
136
+ 'LARGE'
137
+ end
138
+ end
139
+
140
+ # from info at http://www.usps.com/businessmail101/mailcharacteristics/parcels.htm
141
+ #
142
+ # package.options[:books] -- 25 lb. limit instead of 35 for books or other printed matter.
143
+ # Defaults to false.
144
+ def self.package_machinable?(package, options={})
145
+ at_least_minimum = package.inches(:length) >= 6.0 &&
146
+ package.inches(:width) >= 3.0 &&
147
+ package.inches(:height) >= 0.25 &&
148
+ package.ounces >= 6.0
149
+ at_most_maximum = package.inches(:length) <= 34.0 &&
150
+ package.inches(:width) <= 17.0 &&
151
+ package.inches(:height) <= 17.0 &&
152
+ package.pounds <= (package.options[:books] ? 25.0 : 35.0)
153
+ at_least_minimum && at_most_maximum
154
+ end
155
+
156
+ def requirements
157
+ [:login]
158
+ end
159
+
160
+ def find_rates(origin, destination, packages, options = {})
161
+ options = @options.merge(options)
162
+
163
+ origin = Location.from(origin)
164
+ destination = Location.from(destination)
165
+ packages = Array(packages)
166
+
167
+ #raise ArgumentError.new("USPS packages must originate in the U.S.") unless ['US',nil].include?(origin.country_code(:alpha2))
168
+
169
+
170
+ # domestic or international?
171
+
172
+ response = if ['US',nil].include?(destination.country_code(:alpha2))
173
+ us_rates(origin, destination, packages, options)
174
+ else
175
+ world_rates(origin, destination, packages, options)
176
+ end
177
+ end
178
+
179
+ def valid_credentials?
180
+ # Cannot test with find_rates because USPS doesn't allow that in test mode
181
+ test_mode? ? canned_address_verification_works? : super
182
+ end
183
+
184
+ def maximum_weight
185
+ Mass.new(70, :pounds)
186
+ end
187
+ <<<<<<< HEAD
188
+
189
+
190
+ def find_tracking_info(tracking_number, options={})
191
+ options = @options.update(options)
192
+ tracking_request = build_tracking_request(tracking_number, options)
193
+ response = commit(save_request(tracking_request), (options[:test] || false)).gsub(/<(\/)?.*?\:(.*?)>/, '<\1\2>')
194
+ parse_tracking_response(response, options)
195
+ end
196
+
197
+
198
+ =======
199
+
200
+ >>>>>>> d1ef4f7... further cleanup prior to pull request
201
+ protected
202
+
203
+
204
+
205
+ def response_success?(xml)
206
+ xml.get_text('/*/Response/ResponseStatusCode').to_s == '1'
207
+ end
208
+
209
+ def response_message(xml)
210
+ xml.get_text('/*/Response/Error/ErrorDescription | /*/Response/ResponseStatusDescription').to_s
211
+ end
212
+
213
+ def parse_tracking_response(response, options={})
214
+ xml = REXML::Document.new(response)
215
+ success = response_success?(xml)
216
+ message = response_message(xml)
217
+
218
+ if success
219
+ tracking_number, origin, destination = nil
220
+ shipment_events = []
221
+
222
+ first_shipment = xml.elements['/*/Shipment']
223
+ first_package = first_shipment.elements['Package']
224
+ tracking_number = first_shipment.get_text('ShipmentIdentificationNumber | Package/TrackingNumber').to_s
225
+
226
+ origin, destination = %w{Shipper ShipTo}.map do |location|
227
+ location_from_address_node(first_shipment.elements["#{location}/Address"])
228
+ end
229
+
230
+ activities = first_package.get_elements('Activity')
231
+ unless activities.empty?
232
+ shipment_events = activities.map do |activity|
233
+ description = activity.get_text('Status/StatusType/Description').to_s
234
+ zoneless_time = if (time = activity.get_text('Time')) &&
235
+ (date = activity.get_text('Date'))
236
+ time, date = time.to_s, date.to_s
237
+ hour, minute, second = time.scan(/\d{2}/)
238
+ year, month, day = date[0..3], date[4..5], date[6..7]
239
+ Time.utc(year, month, day, hour, minute, second)
240
+ end
241
+ location = location_from_address_node(activity.elements['ActivityLocation/Address'])
242
+ ShipmentEvent.new(description, zoneless_time, location)
243
+ end
244
+
245
+ shipment_events = shipment_events.sort_by(&:time)
246
+
247
+ if origin
248
+ first_event = shipment_events[0]
249
+ same_country = origin.country_code(:alpha2) == first_event.location.country_code(:alpha2)
250
+ same_or_blank_city = first_event.location.city.blank? or first_event.location.city == origin.city
251
+ origin_event = ShipmentEvent.new(first_event.name, first_event.time, origin)
252
+ if same_country and same_or_blank_city
253
+ shipment_events[0] = origin_event
254
+ else
255
+ shipment_events.unshift(origin_event)
256
+ end
257
+ end
258
+ if shipment_events.last.name.downcase == 'delivered'
259
+ shipment_events[-1] = ShipmentEvent.new(shipment_events.last.name, shipment_events.last.time, destination)
260
+ end
261
+ end
262
+
263
+ end
264
+ TrackingResponse.new(success, message, Hash.from_xml(response).values.first,
265
+ :xml => response,
266
+ :request => last_request,
267
+ :shipment_events => shipment_events,
268
+ :origin => origin,
269
+ :destination => destination,
270
+ :tracking_number => tracking_number)
271
+ end
272
+
273
+
274
+
275
+
276
+
277
+
278
+ def build_tracking_request(tracking_number, options={})
279
+ xml_request = XmlNode.new('TrackRequest', 'USERID' => @options[:login]) do |root_node|
280
+ root_node << XmlNode.new('TrackID', :ID => tracking_number)
281
+ end
282
+ URI.encode(xml_request.to_s)
283
+ end
284
+
285
+ def us_rates(origin, destination, packages, options={})
286
+ request = build_us_rate_request(packages, origin.zip, destination.zip, options)
287
+ # never use test mode; rate requests just won't work on test servers
288
+ parse_rate_response origin, destination, packages, commit(:us_rates,request,false), options
289
+ end
290
+
291
+ def world_rates(origin, destination, packages, options={})
292
+ request = build_world_rate_request(packages, destination)
293
+ # never use test mode; rate requests just won't work on test servers
294
+ parse_rate_response origin, destination, packages, commit(:world_rates,request,false), options
295
+ end
296
+
297
+ # Once the address verification API is implemented, remove this and have valid_credentials? build the request using that instead.
298
+ def canned_address_verification_works?
299
+ request = "%3CCarrierPickupAvailabilityRequest%20USERID=%22#{URI.encode(@options[:login])}%22%3E%20%0A%3CFirmName%3EABC%20Corp.%3C/FirmName%3E%20%0A%3CSuiteOrApt%3ESuite%20777%3C/SuiteOrApt%3E%20%0A%3CAddress2%3E1390%20Market%20Street%3C/Address2%3E%20%0A%3CUrbanization%3E%3C/Urbanization%3E%20%0A%3CCity%3EHouston%3C/City%3E%20%0A%3CState%3ETX%3C/State%3E%20%0A%3CZIP5%3E77058%3C/ZIP5%3E%20%0A%3CZIP4%3E1234%3C/ZIP4%3E%20%0A%3C/CarrierPickupAvailabilityRequest%3E%0A"
300
+ # expected_hash = {"CarrierPickupAvailabilityResponse"=>{"City"=>"HOUSTON", "Address2"=>"1390 Market Street", "FirmName"=>"ABC Corp.", "State"=>"TX", "Date"=>"3/1/2004", "DayOfWeek"=>"Monday", "Urbanization"=>nil, "ZIP4"=>"1234", "ZIP5"=>"77058", "CarrierRoute"=>"C", "SuiteOrApt"=>"Suite 777"}}
301
+ xml = REXML::Document.new(commit(:test, request, true))
302
+ xml.get_text('/CarrierPickupAvailabilityResponse/City').to_s == 'HOUSTON' &&
303
+ xml.get_text('/CarrierPickupAvailabilityResponse/Address2').to_s == '1390 Market Street'
304
+ end
305
+
306
+ # options[:service] -- One of [:first_class, :priority, :express, :bpm, :parcel,
307
+ # :media, :library, :all]. defaults to :all.
308
+ # options[:container] -- One of [:envelope, :box]. defaults to neither (this field has
309
+ # special meaning in the USPS API).
310
+ # options[:books] -- Either true or false. Packages of books or other printed matter
311
+ # have a lower weight limit to be considered machinable.
312
+ # package.options[:machinable] -- Either true or false. Overrides the detection of
313
+ # "machinability" entirely.
314
+ def build_us_rate_request(packages, origin_zip, destination_zip, options={})
315
+ packages = Array(packages)
316
+ request = XmlNode.new('RateV4Request', :USERID => @options[:login]) do |rate_request|
317
+ packages.each_with_index do |p,id|
318
+ rate_request << XmlNode.new('Package', :ID => id.to_s) do |package|
319
+ package << XmlNode.new('Service', US_SERVICES[options[:service] || :all])
320
+ package << XmlNode.new('FirstClassMailType', FIRST_CLASS_MAIL_TYPES[options[:first_class_mail_type]])
321
+ package << XmlNode.new('ZipOrigination', strip_zip(origin_zip))
322
+ package << XmlNode.new('ZipDestination', strip_zip(destination_zip))
323
+ package << XmlNode.new('Pounds', 0)
324
+ package << XmlNode.new('Ounces', "%0.1f" % [p.ounces,1].max)
325
+ package << XmlNode.new('Container', CONTAINERS[p.options[:container]])
326
+ package << XmlNode.new('Size', USPS.size_code_for(p))
327
+ package << XmlNode.new('Width', "%0.2f" % p.inches(:width))
328
+ package << XmlNode.new('Length', "%0.2f" % p.inches(:length))
329
+ package << XmlNode.new('Height', "%0.2f" % p.inches(:height))
330
+ package << XmlNode.new('Girth', "%0.2f" % p.inches(:girth))
331
+ is_machinable = if p.options.has_key?(:machinable)
332
+ p.options[:machinable] ? true : false
333
+ else
334
+ USPS.package_machinable?(p)
335
+ end
336
+ package << XmlNode.new('Machinable', is_machinable.to_s.upcase)
337
+ end
338
+ end
339
+ end
340
+ URI.encode(save_request(request.to_s))
341
+ end
342
+
343
+ # important difference with international rate requests:
344
+ # * services are not given in the request
345
+ # * package sizes are not given in the request
346
+ # * services are returned in the response along with restrictions of size
347
+ # * the size restrictions are returned AS AN ENGLISH SENTENCE (!?)
348
+ #
349
+ #
350
+ # package.options[:mail_type] -- one of [:package, :postcard, :matter_for_the_blind, :envelope].
351
+ # Defaults to :package.
352
+ def build_world_rate_request(packages, destination)
353
+ country = COUNTRY_NAME_CONVERSIONS[destination.country.code(:alpha2).value] || destination.country.name
354
+ request = XmlNode.new('IntlRateV2Request', :USERID => @options[:login]) do |rate_request|
355
+ packages.each_index do |id|
356
+ p = packages[id]
357
+ rate_request << XmlNode.new('Package', :ID => id.to_s) do |package|
358
+ package << XmlNode.new('Pounds', 0)
359
+ package << XmlNode.new('Ounces', [p.ounces,1].max.ceil) #takes an integer for some reason, must be rounded UP
360
+ package << XmlNode.new('MailType', MAIL_TYPES[p.options[:mail_type]] || 'Package')
361
+ package << XmlNode.new('GXG') do |gxg|
362
+ gxg << XmlNode.new('POBoxFlag', destination.po_box? ? 'Y' : 'N')
363
+ gxg << XmlNode.new('GiftFlag', p.gift? ? 'Y' : 'N')
364
+ end
365
+ value = if p.value && p.value > 0 && p.currency && p.currency != 'USD'
366
+ 0.0
367
+ else
368
+ (p.value || 0) / 100.0
369
+ end
370
+ package << XmlNode.new('ValueOfContents', value)
371
+ package << XmlNode.new('Country') do |node|
372
+ node.cdata = country
373
+ end
374
+ package << XmlNode.new('Container', p.cylinder? ? 'NONRECTANGULAR' : 'RECTANGULAR')
375
+ package << XmlNode.new('Size', USPS.size_code_for(p))
376
+ package << XmlNode.new('Width', "%0.2f" % [p.inches(:width), 0.01].max)
377
+ package << XmlNode.new('Length', "%0.2f" % [p.inches(:length), 0.01].max)
378
+ package << XmlNode.new('Height', "%0.2f" % [p.inches(:height), 0.01].max)
379
+ package << XmlNode.new('Girth', "%0.2f" % [p.inches(:girth), 0.01].max)
380
+ end
381
+ end
382
+ end
383
+ URI.encode(save_request(request.to_s))
384
+ end
385
+
386
+ def parse_rate_response(origin, destination, packages, response, options={})
387
+ success = true
388
+ message = ''
389
+ rate_hash = {}
390
+
391
+ xml = REXML::Document.new(response)
392
+
393
+ if error = xml.elements['/Error']
394
+ success = false
395
+ message = error.elements['Description'].text
396
+ else
397
+ xml.elements.each('/*/Package') do |package|
398
+ if package.elements['Error']
399
+ success = false
400
+ message = package.get_text('Error/Description').to_s
401
+ break
402
+ end
403
+ end
404
+
405
+ if success
406
+ rate_hash = rates_from_response_node(xml, packages)
407
+ unless rate_hash
408
+ success = false
409
+ message = "Unknown root node in XML response: '#{xml.root.name}'"
410
+ end
411
+ end
412
+
413
+ end
414
+
415
+ if success
416
+ rate_estimates = rate_hash.keys.map do |service_name|
417
+ RateEstimate.new(origin,destination,@@name,"USPS #{service_name}",
418
+ :package_rates => rate_hash[service_name][:package_rates],
419
+ :service_code => rate_hash[service_name][:service_code],
420
+ :currency => 'USD')
421
+ end
422
+ rate_estimates.reject! {|e| e.package_count != packages.length}
423
+ rate_estimates = rate_estimates.sort_by(&:total_price)
424
+ end
425
+
426
+ RateResponse.new(success, message, Hash.from_xml(response), :rates => rate_estimates, :xml => response, :request => last_request)
427
+ end
428
+
429
+ def rates_from_response_node(response_node, packages)
430
+ rate_hash = {}
431
+ return false unless (root_node = response_node.elements['/IntlRateV2Response | /RateV4Response'])
432
+ domestic = (root_node.name == 'RateV4Response')
433
+
434
+ domestic_elements = ['Postage', 'CLASSID', 'MailService', 'Rate']
435
+ international_elements = ['Service', 'ID', 'SvcDescription', 'Postage']
436
+ service_node, service_code_node, service_name_node, rate_node = domestic ? domestic_elements : international_elements
437
+
438
+ root_node.each_element('Package') do |package_node|
439
+ this_package = packages[package_node.attributes['ID'].to_i]
440
+
441
+ package_node.each_element(service_node) do |service_response_node|
442
+ service_name = service_response_node.get_text(service_name_node).to_s
443
+
444
+ # strips the double-escaped HTML for trademark symbols from service names
445
+ service_name.gsub!(/&amp;lt;\S*&amp;gt;/,'')
446
+ # ...leading "USPS"
447
+ service_name.gsub!(/^USPS/,'')
448
+ # ...trailing asterisks
449
+ service_name.gsub!(/\*+$/,'')
450
+ # ...surrounding spaces
451
+ service_name.strip!
452
+
453
+ # aggregate specific package rates into a service-centric RateEstimate
454
+ # first package with a given service name will initialize these;
455
+ # later packages with same service will add to them
456
+ this_service = rate_hash[service_name] ||= {}
457
+ this_service[:service_code] ||= service_response_node.attributes[service_code_node]
458
+ package_rates = this_service[:package_rates] ||= []
459
+ this_package_rate = {:package => this_package,
460
+ :rate => Package.cents_from(service_response_node.get_text(rate_node).to_s.to_f)}
461
+
462
+ package_rates << this_package_rate if package_valid_for_service(this_package,service_response_node)
463
+ end
464
+ end
465
+ rate_hash
466
+ end
467
+
468
+ def package_valid_for_service(package, service_node)
469
+ return true if service_node.elements['MaxWeight'].nil?
470
+ max_weight = service_node.get_text('MaxWeight').to_s.to_f
471
+ name = service_node.get_text('SvcDescription | MailService').to_s.downcase
472
+
473
+ if name =~ /flat.rate.box/ #domestic or international flat rate box
474
+ # flat rate dimensions from http://www.usps.com/shipping/flatrate.htm
475
+ return (package_valid_for_max_dimensions(package,
476
+ :weight => max_weight, #domestic apparently has no weight restriction
477
+ :length => 11.0,
478
+ :width => 8.5,
479
+ :height => 5.5) or
480
+ package_valid_for_max_dimensions(package,
481
+ :weight => max_weight,
482
+ :length => 13.625,
483
+ :width => 11.875,
484
+ :height => 3.375))
485
+ elsif name =~ /flat.rate.envelope/
486
+ return package_valid_for_max_dimensions(package,
487
+ :weight => max_weight,
488
+ :length => 12.5,
489
+ :width => 9.5,
490
+ :height => 0.75)
491
+ elsif service_node.elements['MailService'] # domestic non-flat rates
492
+ return true
493
+ else #international non-flat rates
494
+ # Some sample english that this is required to parse:
495
+ #
496
+ # 'Max. length 46", width 35", height 46" and max. length plus girth 108"'
497
+ # 'Max. length 24", Max. length, height, depth combined 36"'
498
+ #
499
+ sentence = CGI.unescapeHTML(service_node.get_text('MaxDimensions').to_s)
500
+ tokens = sentence.downcase.split(/[^\d]*"/).reject {|t| t.empty?}
501
+ max_dimensions = {:weight => max_weight}
502
+ single_axis_values = []
503
+ tokens.each do |token|
504
+ axis_sum = [/length/,/width/,/height/,/depth/].sum {|regex| (token =~ regex) ? 1 : 0}
505
+ unless axis_sum == 0
506
+ value = token[/\d+$/].to_f
507
+ if axis_sum == 3
508
+ max_dimensions[:length_plus_width_plus_height] = value
509
+ elsif token =~ /girth/ and axis_sum == 1
510
+ max_dimensions[:length_plus_girth] = value
511
+ else
512
+ single_axis_values << value
513
+ end
514
+ end
515
+ end
516
+ single_axis_values.sort!.reverse!
517
+ [:length, :width, :height].each_with_index do |axis,i|
518
+ max_dimensions[axis] = single_axis_values[i] if single_axis_values[i]
519
+ end
520
+ return package_valid_for_max_dimensions(package, max_dimensions)
521
+ end
522
+ end
523
+
524
+ def package_valid_for_max_dimensions(package,dimensions)
525
+ valid = ((not ([:length,:width,:height].map {|dim| dimensions[dim].nil? || dimensions[dim].to_f >= package.inches(dim).to_f}.include?(false))) and
526
+ (dimensions[:weight].nil? || dimensions[:weight] >= package.pounds) and
527
+ (dimensions[:length_plus_girth].nil? or
528
+ dimensions[:length_plus_girth].to_f >=
529
+ package.inches(:length) + package.inches(:girth)) and
530
+ (dimensions[:length_plus_width_plus_height].nil? or
531
+ dimensions[:length_plus_width_plus_height].to_f >=
532
+ package.inches(:length) + package.inches(:width) + package.inches(:height)))
533
+
534
+ return valid
535
+ end
536
+
537
+ def parse_tracking_response(response, options)
538
+ xml = REXML::Document.new(response)
539
+ root_node = xml.elements['TrackResponse']
540
+
541
+ success = response_success?(xml)
542
+ message = response_message(xml)
543
+
544
+ if success
545
+ tracking_number, origin, destination = nil
546
+ shipment_events = []
547
+ tracking_details = xml.elements.collect('*/*/TrackDetail'){ |e| e }
548
+
549
+ tracking_summary = xml.elements.collect('*/*/TrackSummary'){ |e| e }.first
550
+ tracking_details << tracking_summary
551
+
552
+ tracking_number = root_node.elements['TrackInfo'].attributes['ID'].to_s
553
+
554
+ tracking_details.each do |event|
555
+ location = nil
556
+ timestamp = nil
557
+ description = nil
558
+ if event.get_text.to_s =~ /^(.*), (\w+ \d\d, \d{4}, \d{1,2}:\d\d [ap]m), (.*), (\w\w) (\d{5})$/i ||
559
+ event.get_text.to_s =~ /^Your item \w{2,3} (out for delivery|delivered) at (\d{1,2}:\d\d [ap]m on \w+ \d\d, \d{4}) in (.*), (\w\w) (\d{5})\.$/i
560
+ description = $1.upcase
561
+ timestamp = $2
562
+ city = $3
563
+ state = $4
564
+ zip_code = $5
565
+ location = Location.new(:city => city, :state => state, :postal_code => zip_code, :country => 'USA')
566
+ end
567
+ if location
568
+ time = Time.parse(timestamp)
569
+ zoneless_time = Time.utc(time.year, time.month, time.mday, time.hour, time.min, time.sec)
570
+ shipment_events << ShipmentEvent.new(description, zoneless_time, location)
571
+ end
572
+ end
573
+ shipment_events = shipment_events.sort_by(&:time)
574
+ end
575
+
576
+ TrackingResponse.new(success, message, Hash.from_xml(response),
577
+ :xml => response,
578
+ :request => last_request,
579
+ :shipment_events => shipment_events,
580
+ :destination => destination,
581
+ :tracking_number => tracking_number
582
+ )
583
+ end
584
+
585
+ def response_status_node(document)
586
+ document.elements['*/*/TrackSummary'] || document.elements['Error/Description']
587
+ end
588
+
589
+ def response_success?(document)
590
+ summary = response_status_node(document).get_text.to_s
591
+ !(summary =~ /There is no record of that mail item/ || summary =~ /This Information has not been included in this Test Server\./)
592
+ end
593
+
594
+ def response_message(document)
595
+ response_node = response_status_node(document)
596
+ response_status_node(document).get_text.to_s
597
+ end
598
+
599
+ def commit(action, request, test = false)
600
+ ssl_get(request_url(action, request, test))
601
+ end
602
+
603
+ def request_url(action, request, test)
604
+ scheme = USE_SSL[action] ? 'https://' : 'http://'
605
+ host = test ? TEST_DOMAINS[USE_SSL[action]] : LIVE_DOMAIN
606
+ resource = test ? TEST_RESOURCE : LIVE_RESOURCE
607
+ "#{scheme}#{host}/#{resource}?API=#{API_CODES[action]}&XML=#{request}"
608
+ end
609
+
610
+ def strip_zip(zip)
611
+ zip.to_s.scan(/\d{5}/).first || zip
612
+ end
613
+
614
+ end
615
+ end
616
+ end