apispree_core 0.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (890) hide show
  1. data/LICENSE +26 -0
  2. data/README.md +15 -0
  3. data/app/controllers/admin/adjustments_controller.rb +10 -0
  4. data/app/controllers/admin/base_controller.rb +63 -0
  5. data/app/controllers/admin/configurations_controller.rb +2 -0
  6. data/app/controllers/admin/general_settings_controller.rb +22 -0
  7. data/app/controllers/admin/images_controller.rb +49 -0
  8. data/app/controllers/admin/inventory_settings_controller.rb +13 -0
  9. data/app/controllers/admin/inventory_units_controller.rb +2 -0
  10. data/app/controllers/admin/line_items_controller.rb +60 -0
  11. data/app/controllers/admin/mail_methods_controller.rb +8 -0
  12. data/app/controllers/admin/mail_settings_controller.rb +14 -0
  13. data/app/controllers/admin/option_types_controller.rb +65 -0
  14. data/app/controllers/admin/orders_controller.rb +136 -0
  15. data/app/controllers/admin/overview_controller.rb +14 -0
  16. data/app/controllers/admin/payment_methods_controller.rb +43 -0
  17. data/app/controllers/admin/payments_controller.rb +107 -0
  18. data/app/controllers/admin/product_groups_controller.rb +46 -0
  19. data/app/controllers/admin/product_properties_controller.rb +10 -0
  20. data/app/controllers/admin/product_scopes_controller.rb +35 -0
  21. data/app/controllers/admin/products_controller.rb +113 -0
  22. data/app/controllers/admin/products_controller.rb~ +114 -0
  23. data/app/controllers/admin/properties_controller.rb +11 -0
  24. data/app/controllers/admin/prototypes_controller.rb +26 -0
  25. data/app/controllers/admin/reports_controller.rb +50 -0
  26. data/app/controllers/admin/resource_controller.rb +232 -0
  27. data/app/controllers/admin/return_authorizations_controller.rb +17 -0
  28. data/app/controllers/admin/shipments_controller.rb +117 -0
  29. data/app/controllers/admin/shipping_categories_controller.rb +2 -0
  30. data/app/controllers/admin/shipping_methods_controller.rb +14 -0
  31. data/app/controllers/admin/states_controller.rb +25 -0
  32. data/app/controllers/admin/tax_categories_controller.rb +2 -0
  33. data/app/controllers/admin/tax_rates_controller.rb +22 -0
  34. data/app/controllers/admin/tax_settings_controller.rb +13 -0
  35. data/app/controllers/admin/taxonomies_controller.rb +20 -0
  36. data/app/controllers/admin/taxons_controller.rb +161 -0
  37. data/app/controllers/admin/trackers_controller.rb +2 -0
  38. data/app/controllers/admin/users_controller.rb +65 -0
  39. data/app/controllers/admin/variants_controller.rb +51 -0
  40. data/app/controllers/admin/zones_controller.rb +23 -0
  41. data/app/controllers/checkout_controller.rb +97 -0
  42. data/app/controllers/content_controller.rb +21 -0
  43. data/app/controllers/locale_controller.rb +14 -0
  44. data/app/controllers/orders_controller.rb +64 -0
  45. data/app/controllers/products_controller.rb +39 -0
  46. data/app/controllers/spree/base_controller.rb +4 -0
  47. data/app/controllers/states_controller.rb +16 -0
  48. data/app/controllers/taxons_controller.rb +22 -0
  49. data/app/helpers/account_helper.rb +2 -0
  50. data/app/helpers/admin/base_helper.rb +194 -0
  51. data/app/helpers/admin/inventory_settings_helper.rb +5 -0
  52. data/app/helpers/admin/navigation_helper.rb +162 -0
  53. data/app/helpers/admin/orders_helper.rb +22 -0
  54. data/app/helpers/admin/overview_helper.rb +12 -0
  55. data/app/helpers/admin/payments_helper.rb +7 -0
  56. data/app/helpers/admin/product_groups_helper.rb +11 -0
  57. data/app/helpers/admin/product_properties_helper.rb +21 -0
  58. data/app/helpers/admin/products_helper.rb +11 -0
  59. data/app/helpers/admin/taxons_helper.rb +18 -0
  60. data/app/helpers/admin/users_helper.rb +8 -0
  61. data/app/helpers/admin/zones_helper.rb +11 -0
  62. data/app/helpers/checkout_helper.rb +31 -0
  63. data/app/helpers/hook_helper.rb +20 -0
  64. data/app/helpers/orders_helper.rb +3 -0
  65. data/app/helpers/products_helper.rb +30 -0
  66. data/app/helpers/search_helper.rb +10 -0
  67. data/app/helpers/spree/base_helper.rb +119 -0
  68. data/app/helpers/taxons_helper.rb +32 -0
  69. data/app/helpers/trackers_helper.rb +2 -0
  70. data/app/mailers/order_mailer.rb +19 -0
  71. data/app/mailers/shipment_mailer.rb +11 -0
  72. data/app/models/address.rb +115 -0
  73. data/app/models/adjustment.rb +47 -0
  74. data/app/models/app_configuration.rb +44 -0
  75. data/app/models/asset.rb +4 -0
  76. data/app/models/billing_integration.rb +23 -0
  77. data/app/models/calculator.rb +46 -0
  78. data/app/models/calculator/flat_percent_item_total.rb +19 -0
  79. data/app/models/calculator/flat_rate.rb +16 -0
  80. data/app/models/calculator/flexi_rate.rb +32 -0
  81. data/app/models/calculator/per_item.rb +16 -0
  82. data/app/models/calculator/price_bucket.rb +30 -0
  83. data/app/models/calculator/sales_tax.rb +48 -0
  84. data/app/models/calculator/vat.rb +63 -0
  85. data/app/models/configuration.rb +3 -0
  86. data/app/models/country.rb +16 -0
  87. data/app/models/creditcard.rb +267 -0
  88. data/app/models/gateway.rb +48 -0
  89. data/app/models/gateway/authorize_net.rb +19 -0
  90. data/app/models/gateway/authorize_net_cim.rb +132 -0
  91. data/app/models/gateway/beanstream.rb +187 -0
  92. data/app/models/gateway/bogus.rb +80 -0
  93. data/app/models/gateway/braintree.rb +88 -0
  94. data/app/models/gateway/eway.rb +11 -0
  95. data/app/models/gateway/linkpoint.rb +8 -0
  96. data/app/models/gateway/pay_pal.rb +11 -0
  97. data/app/models/gateway/sage_pay.rb +9 -0
  98. data/app/models/image.rb +31 -0
  99. data/app/models/inventory_unit.rb +119 -0
  100. data/app/models/line_item.rb +95 -0
  101. data/app/models/log_entry.rb +3 -0
  102. data/app/models/mail_method.rb +23 -0
  103. data/app/models/option_type.rb +9 -0
  104. data/app/models/option_value.rb +5 -0
  105. data/app/models/order.rb +484 -0
  106. data/app/models/payment.rb +119 -0
  107. data/app/models/payment_method.rb +56 -0
  108. data/app/models/payment_method/check.rb +27 -0
  109. data/app/models/preference.rb +51 -0
  110. data/app/models/product.rb +258 -0
  111. data/app/models/product_group.rb +216 -0
  112. data/app/models/product_option_type.rb +5 -0
  113. data/app/models/product_property.rb +15 -0
  114. data/app/models/product_scope.rb +78 -0
  115. data/app/models/property.rb +18 -0
  116. data/app/models/prototype.rb +5 -0
  117. data/app/models/return_authorization.rb +79 -0
  118. data/app/models/role.rb +3 -0
  119. data/app/models/shipment.rb +157 -0
  120. data/app/models/shipping_category.rb +3 -0
  121. data/app/models/shipping_method.rb +25 -0
  122. data/app/models/spree/alert.rb +13 -0
  123. data/app/models/state.rb +16 -0
  124. data/app/models/state_event.rb +16 -0
  125. data/app/models/tax_category.rb +15 -0
  126. data/app/models/tax_rate.rb +16 -0
  127. data/app/models/taxon.rb +52 -0
  128. data/app/models/taxonomy.rb +20 -0
  129. data/app/models/taxonomy.rb~ +22 -0
  130. data/app/models/tracker.rb +5 -0
  131. data/app/models/user.rb +21 -0
  132. data/app/models/variant.rb +104 -0
  133. data/app/models/zone.rb +84 -0
  134. data/app/models/zone_member.rb +10 -0
  135. data/app/stylesheets/_buttons.less +80 -0
  136. data/app/stylesheets/_cart.less +24 -0
  137. data/app/stylesheets/_checkout.less +76 -0
  138. data/app/stylesheets/_checkout_progress.less +62 -0
  139. data/app/stylesheets/_colors.less +16 -0
  140. data/app/stylesheets/_forms.less +50 -0
  141. data/app/stylesheets/_layout.less +77 -0
  142. data/app/stylesheets/_messages.less +32 -0
  143. data/app/stylesheets/_mixins.less +71 -0
  144. data/app/stylesheets/_navigation.less +130 -0
  145. data/app/stylesheets/_prices.less +18 -0
  146. data/app/stylesheets/_product_details.less +56 -0
  147. data/app/stylesheets/_product_lists.less +66 -0
  148. data/app/stylesheets/_registration.less +24 -0
  149. data/app/stylesheets/_reset.less +69 -0
  150. data/app/stylesheets/_typography.less +158 -0
  151. data/app/stylesheets/screen.less +13 -0
  152. data/app/views/admin/adjustments/_adjustments_table.html.erb +24 -0
  153. data/app/views/admin/adjustments/_form.html.erb +12 -0
  154. data/app/views/admin/adjustments/edit.html.erb +13 -0
  155. data/app/views/admin/adjustments/index.html.erb +4 -0
  156. data/app/views/admin/adjustments/new.html.erb +14 -0
  157. data/app/views/admin/configurations/index.html.erb +66 -0
  158. data/app/views/admin/extensions/index.html.erb +23 -0
  159. data/app/views/admin/general_settings/edit.html.erb +46 -0
  160. data/app/views/admin/general_settings/show.html.erb +36 -0
  161. data/app/views/admin/images/_form.html.erb +16 -0
  162. data/app/views/admin/images/edit.html.erb +19 -0
  163. data/app/views/admin/images/index.html.erb +69 -0
  164. data/app/views/admin/images/new.html.erb +22 -0
  165. data/app/views/admin/images/new.html.erb~ +22 -0
  166. data/app/views/admin/inventory_settings/edit.html.erb +27 -0
  167. data/app/views/admin/inventory_settings/show.html.erb +9 -0
  168. data/app/views/admin/inventory_units/adjust.html.erb +33 -0
  169. data/app/views/admin/mail_methods/_form.html.erb +96 -0
  170. data/app/views/admin/mail_methods/edit.html.erb +9 -0
  171. data/app/views/admin/mail_methods/index.html.erb +41 -0
  172. data/app/views/admin/mail_methods/new.html.erb +9 -0
  173. data/app/views/admin/option_types/_available.html.erb +27 -0
  174. data/app/views/admin/option_types/_form.html.erb +11 -0
  175. data/app/views/admin/option_types/_option_value_fields.html.erb +5 -0
  176. data/app/views/admin/option_types/_selected.html.erb +26 -0
  177. data/app/views/admin/option_types/available.js.erb +2 -0
  178. data/app/views/admin/option_types/edit.html.erb +32 -0
  179. data/app/views/admin/option_types/index.html.erb +33 -0
  180. data/app/views/admin/option_types/new.erb +11 -0
  181. data/app/views/admin/option_types/select.js.erb +3 -0
  182. data/app/views/admin/option_types/selected.html.erb +6 -0
  183. data/app/views/admin/orders/_add_product.html.erb +20 -0
  184. data/app/views/admin/orders/_form.html.erb +67 -0
  185. data/app/views/admin/orders/_line_item.html.erb +17 -0
  186. data/app/views/admin/orders/_user_form.html.erb +42 -0
  187. data/app/views/admin/orders/edit.html.erb +31 -0
  188. data/app/views/admin/orders/history.html.erb +29 -0
  189. data/app/views/admin/orders/index.html.erb +103 -0
  190. data/app/views/admin/orders/new.html.erb +20 -0
  191. data/app/views/admin/orders/show.html.erb +28 -0
  192. data/app/views/admin/orders/user.html.erb +24 -0
  193. data/app/views/admin/payment_methods/_form.html.erb +50 -0
  194. data/app/views/admin/payment_methods/edit.html.erb +10 -0
  195. data/app/views/admin/payment_methods/index.html.erb +49 -0
  196. data/app/views/admin/payment_methods/new.html.erb +11 -0
  197. data/app/views/admin/payments/_bill_address_form.html.erb +9 -0
  198. data/app/views/admin/payments/_form.html.erb +15 -0
  199. data/app/views/admin/payments/_list.html.erb +22 -0
  200. data/app/views/admin/payments/_transaction_list.html.erb +24 -0
  201. data/app/views/admin/payments/credit.html.erb +14 -0
  202. data/app/views/admin/payments/index.html.erb +22 -0
  203. data/app/views/admin/payments/new.html.erb +19 -0
  204. data/app/views/admin/payments/show.html.erb +10 -0
  205. data/app/views/admin/payments/source_forms/_check.html.erb +0 -0
  206. data/app/views/admin/payments/source_forms/_gateway.html.erb +43 -0
  207. data/app/views/admin/payments/source_views/_check.html.erb +0 -0
  208. data/app/views/admin/payments/source_views/_gateway.html.erb +34 -0
  209. data/app/views/admin/product_groups/_preview.html.erb +34 -0
  210. data/app/views/admin/product_groups/_product_scope.html.erb +24 -0
  211. data/app/views/admin/product_groups/edit.html.erb +56 -0
  212. data/app/views/admin/product_groups/index.html.erb +37 -0
  213. data/app/views/admin/product_groups/new.html.erb +12 -0
  214. data/app/views/admin/product_groups/show.html.erb +32 -0
  215. data/app/views/admin/product_groups/update.js.erb +8 -0
  216. data/app/views/admin/product_properties/_product_property_fields.html.erb +11 -0
  217. data/app/views/admin/product_properties/index.html.erb +55 -0
  218. data/app/views/admin/product_scopes/_form.html.erb +106 -0
  219. data/app/views/admin/product_scopes/create.js.erb +3 -0
  220. data/app/views/admin/product_scopes/destroy.js.erb +3 -0
  221. data/app/views/admin/product_scopes/new.html.erb +1 -0
  222. data/app/views/admin/products/_form.html.erb +112 -0
  223. data/app/views/admin/products/_option_types.html.erb +40 -0
  224. data/app/views/admin/products/_properties_form.erb +7 -0
  225. data/app/views/admin/products/edit.html.erb +10 -0
  226. data/app/views/admin/products/edit.html.erb~ +10 -0
  227. data/app/views/admin/products/index.html.erb +81 -0
  228. data/app/views/admin/products/index.html.erb~ +81 -0
  229. data/app/views/admin/products/new.erb +58 -0
  230. data/app/views/admin/products/new.erb~ +53 -0
  231. data/app/views/admin/products/show.html.erb +4 -0
  232. data/app/views/admin/products/show.html.erb~ +4 -0
  233. data/app/views/admin/properties/_form.html.erb +12 -0
  234. data/app/views/admin/properties/edit.html.erb +10 -0
  235. data/app/views/admin/properties/filtered.html.erb +1 -0
  236. data/app/views/admin/properties/index.html.erb +40 -0
  237. data/app/views/admin/properties/new.erb +11 -0
  238. data/app/views/admin/prototypes/_form.html.erb +44 -0
  239. data/app/views/admin/prototypes/available.html.erb +22 -0
  240. data/app/views/admin/prototypes/edit.html.erb +9 -0
  241. data/app/views/admin/prototypes/index.html.erb +37 -0
  242. data/app/views/admin/prototypes/new.erb +11 -0
  243. data/app/views/admin/prototypes/select.js.erb +4 -0
  244. data/app/views/admin/reports/index.html.erb +19 -0
  245. data/app/views/admin/reports/sales_total.html.erb +23 -0
  246. data/app/views/admin/return_authorizations/_form.html.erb +78 -0
  247. data/app/views/admin/return_authorizations/edit.html.erb +27 -0
  248. data/app/views/admin/return_authorizations/index.html.erb +45 -0
  249. data/app/views/admin/return_authorizations/new.html.erb +13 -0
  250. data/app/views/admin/shared/_additional_field.html.erb +5 -0
  251. data/app/views/admin/shared/_address.html.erb +5 -0
  252. data/app/views/admin/shared/_address_form.html.erb +79 -0
  253. data/app/views/admin/shared/_alert.html.erb +6 -0
  254. data/app/views/admin/shared/_calculator_fields.html.erb +19 -0
  255. data/app/views/admin/shared/_configuration_menu.html.erb +22 -0
  256. data/app/views/admin/shared/_destroy.js.erb +16 -0
  257. data/app/views/admin/shared/_edit_resource_links.html.erb +4 -0
  258. data/app/views/admin/shared/_group_from_products_form.html.erb +12 -0
  259. data/app/views/admin/shared/_new_adjustment_button.html.erb +8 -0
  260. data/app/views/admin/shared/_new_resource_links.html.erb +4 -0
  261. data/app/views/admin/shared/_order_details.html.erb +3 -0
  262. data/app/views/admin/shared/_order_sub_menu.html.erb +0 -0
  263. data/app/views/admin/shared/_order_tabs.html.erb +64 -0
  264. data/app/views/admin/shared/_product_sub_menu.html.erb +11 -0
  265. data/app/views/admin/shared/_product_tabs.html.erb +34 -0
  266. data/app/views/admin/shared/_report_criteria.html.erb +15 -0
  267. data/app/views/admin/shared/_show_resource_links.html.erb +5 -0
  268. data/app/views/admin/shared/_tabs.html.erb +6 -0
  269. data/app/views/admin/shared/_update_order_state.js +6 -0
  270. data/app/views/admin/shipments/_form.html.erb +77 -0
  271. data/app/views/admin/shipments/edit.html.erb +40 -0
  272. data/app/views/admin/shipments/index.html.erb +47 -0
  273. data/app/views/admin/shipments/new.html.erb +18 -0
  274. data/app/views/admin/shipping_categories/_form.html.erb +6 -0
  275. data/app/views/admin/shipping_categories/edit.html.erb +6 -0
  276. data/app/views/admin/shipping_categories/index.html.erb +29 -0
  277. data/app/views/admin/shipping_categories/new.html.erb +8 -0
  278. data/app/views/admin/shipping_methods/_form.html.erb +26 -0
  279. data/app/views/admin/shipping_methods/edit.html.erb +16 -0
  280. data/app/views/admin/shipping_methods/index.html.erb +49 -0
  281. data/app/views/admin/shipping_methods/new.html.erb +17 -0
  282. data/app/views/admin/states/_form.html.erb +9 -0
  283. data/app/views/admin/states/_state_list.html.erb +34 -0
  284. data/app/views/admin/states/edit.html.erb +10 -0
  285. data/app/views/admin/states/index.html.erb +16 -0
  286. data/app/views/admin/states/new.erb +8 -0
  287. data/app/views/admin/tax_categories/_form.html.erb +12 -0
  288. data/app/views/admin/tax_categories/edit.html.erb +10 -0
  289. data/app/views/admin/tax_categories/index.html.erb +44 -0
  290. data/app/views/admin/tax_categories/new.html.erb +10 -0
  291. data/app/views/admin/tax_categories/show.html.erb +1 -0
  292. data/app/views/admin/tax_rates/_form.html.erb +16 -0
  293. data/app/views/admin/tax_rates/edit.html.erb +6 -0
  294. data/app/views/admin/tax_rates/index.html.erb +34 -0
  295. data/app/views/admin/tax_rates/new.html.erb +8 -0
  296. data/app/views/admin/tax_settings/edit.html.erb +17 -0
  297. data/app/views/admin/tax_settings/show.html.erb +14 -0
  298. data/app/views/admin/taxonomies/_form.html.erb +10 -0
  299. data/app/views/admin/taxonomies/_form.html.erb~ +11 -0
  300. data/app/views/admin/taxonomies/_js_head.html.erb +9 -0
  301. data/app/views/admin/taxonomies/_list.html.erb +19 -0
  302. data/app/views/admin/taxonomies/_list.html.erb~ +20 -0
  303. data/app/views/admin/taxonomies/_taxon.html.erb +12 -0
  304. data/app/views/admin/taxonomies/edit.erb +49 -0
  305. data/app/views/admin/taxonomies/get_children.json.erb +9 -0
  306. data/app/views/admin/taxonomies/index.html.erb +16 -0
  307. data/app/views/admin/taxonomies/new.html.erb +12 -0
  308. data/app/views/admin/taxons/_form.html.erb +23 -0
  309. data/app/views/admin/taxons/_taxon_table.html.erb +23 -0
  310. data/app/views/admin/taxons/available.js.erb +33 -0
  311. data/app/views/admin/taxons/edit.html.erb +11 -0
  312. data/app/views/admin/taxons/remove.html.erb +1 -0
  313. data/app/views/admin/taxons/select.js.erb +2 -0
  314. data/app/views/admin/taxons/selected.html.erb +43 -0
  315. data/app/views/admin/trackers/_form.html.erb +26 -0
  316. data/app/views/admin/trackers/edit.html.erb +8 -0
  317. data/app/views/admin/trackers/index.html.erb +41 -0
  318. data/app/views/admin/trackers/new.html.erb +9 -0
  319. data/app/views/admin/users/_form.html.erb +35 -0
  320. data/app/views/admin/users/edit.html.erb +14 -0
  321. data/app/views/admin/users/index.html.erb +63 -0
  322. data/app/views/admin/users/new.html.erb +14 -0
  323. data/app/views/admin/users/show.html.erb +21 -0
  324. data/app/views/admin/variants/_form.html.erb +40 -0
  325. data/app/views/admin/variants/edit.html.erb +13 -0
  326. data/app/views/admin/variants/index.html.erb +66 -0
  327. data/app/views/admin/variants/new.erb +11 -0
  328. data/app/views/admin/zones/_country_member.html.erb +5 -0
  329. data/app/views/admin/zones/_form.html.erb +29 -0
  330. data/app/views/admin/zones/_member_type.html.erb +15 -0
  331. data/app/views/admin/zones/_state_member.html.erb +5 -0
  332. data/app/views/admin/zones/_zone_member.html.erb +5 -0
  333. data/app/views/admin/zones/edit.html.erb +10 -0
  334. data/app/views/admin/zones/index.html.erb +41 -0
  335. data/app/views/admin/zones/new.html.erb +10 -0
  336. data/app/views/checkout/_address.html.erb +163 -0
  337. data/app/views/checkout/_confirm.html.erb +11 -0
  338. data/app/views/checkout/_delivery.html.erb +25 -0
  339. data/app/views/checkout/_payment.html.erb +30 -0
  340. data/app/views/checkout/_summary.html.erb +21 -0
  341. data/app/views/checkout/edit.html.erb +20 -0
  342. data/app/views/checkout/payment/_check.html.erb +0 -0
  343. data/app/views/checkout/payment/_gateway.html.erb +31 -0
  344. data/app/views/checkout/registration.html.erb +20 -0
  345. data/app/views/content/cvv.html.erb +13 -0
  346. data/app/views/layouts/admin.html.erb +75 -0
  347. data/app/views/layouts/spree_application.html.erb +52 -0
  348. data/app/views/order_mailer/cancel_email.text.erb +16 -0
  349. data/app/views/order_mailer/confirm_email.text.erb +19 -0
  350. data/app/views/orders/_form.html.erb +19 -0
  351. data/app/views/orders/_line_item.html.erb +41 -0
  352. data/app/views/orders/edit.html.erb +41 -0
  353. data/app/views/orders/new.html.erb +9 -0
  354. data/app/views/orders/show.html.erb +14 -0
  355. data/app/views/products/_cart_form.html.erb +54 -0
  356. data/app/views/products/_cart_form.html.erb~ +54 -0
  357. data/app/views/products/_image.html.erb +5 -0
  358. data/app/views/products/_image.html.erb~ +5 -0
  359. data/app/views/products/_properties.html.erb +8 -0
  360. data/app/views/products/_taxons.html.erb +16 -0
  361. data/app/views/products/_thumbnails.html.erb +40 -0
  362. data/app/views/products/index.html.erb +37 -0
  363. data/app/views/products/show.html.erb +31 -0
  364. data/app/views/shared/_additional_field.html.erb +7 -0
  365. data/app/views/shared/_admin_head.html.erb +34 -0
  366. data/app/views/shared/_basic_layout.html.erb +44 -0
  367. data/app/views/shared/_edit_resource_links.html.erb +1 -0
  368. data/app/views/shared/_end_soon.html.erb +43 -0
  369. data/app/views/shared/_error_messages.html.erb +11 -0
  370. data/app/views/shared/_filters.html.erb +27 -0
  371. data/app/views/shared/_footer.html.erb +13 -0
  372. data/app/views/shared/_google_analytics.html.erb +40 -0
  373. data/app/views/shared/_head.html.erb +10 -0
  374. data/app/views/shared/_method_error_message.html.erb +4 -0
  375. data/app/views/shared/_nav_bar.html.erb +5 -0
  376. data/app/views/shared/_new_release.html.erb +42 -0
  377. data/app/views/shared/_new_resource_links.html.erb +1 -0
  378. data/app/views/shared/_order_details.html.erb +50 -0
  379. data/app/views/shared/_products.html.erb +42 -0
  380. data/app/views/shared/_search.html.erb +9 -0
  381. data/app/views/shared/_show_resource_links.html.erb +3 -0
  382. data/app/views/shared/_store_menu.html.erb +3 -0
  383. data/app/views/shared/_taxonomies.html.erb +19 -0
  384. data/app/views/shared/_test_hook.html.erb +1 -0
  385. data/app/views/shared/_upcoming_products.html.erb +42 -0
  386. data/app/views/shipment_mailer/shipped_email.text.erb +15 -0
  387. data/app/views/states/index.js.erb +1 -0
  388. data/app/views/taxons/_taxon.html.erb +4 -0
  389. data/app/views/taxons/show.html.erb +19 -0
  390. data/config/:cart +0 -0
  391. data/config/:checkout +0 -0
  392. data/config/:checkout_state +0 -0
  393. data/config/:collection +0 -0
  394. data/config/:empty_cart +0 -0
  395. data/config/:get, +0 -0
  396. data/config/:index +0 -0
  397. data/config/:nested_taxons +0 -0
  398. data/config/:pg_search +0 -0
  399. data/config/:put, +0 -0
  400. data/config/:simple_search +0 -0
  401. data/config/:taxons_pg_search +0 -0
  402. data/config/:taxons_search +0 -0
  403. data/config/:update_cart +0 -0
  404. data/config/:update_checkout +0 -0
  405. data/config/[:index, +0 -0
  406. data/config/address, +0 -0
  407. data/config/checkout#edit, +0 -0
  408. data/config/checkout#update, +0 -0
  409. data/config/content#cvv +0 -0
  410. data/config/content#show +0 -0
  411. data/config/cucumber.yml +10 -0
  412. data/config/initializers/form_builder.rb +19 -0
  413. data/config/initializers/load_unobtrusive_date_picker.rb +8 -0
  414. data/config/initializers/spree.rb +29 -0
  415. data/config/initializers/workarounds_for_ruby19.rb +72 -0
  416. data/config/locale#set +0 -0
  417. data/config/locales/en.yml +1015 -0
  418. data/config/orders#edit, +0 -0
  419. data/config/orders#empty, +0 -0
  420. data/config/orders#update, +0 -0
  421. data/config/products#index +0 -0
  422. data/config/products#index, +0 -0
  423. data/config/routes.rb +207 -0
  424. data/config/taxons#show, +0 -0
  425. data/db/default/countries.yml +1583 -0
  426. data/db/default/roles.yml +7 -0
  427. data/db/default/states.yml +256 -0
  428. data/db/default/zone_members.yml +169 -0
  429. data/db/default/zones.yml +13 -0
  430. data/db/migrate/20090823005402_spree_zero_nine_zero.rb +442 -0
  431. data/db/migrate/20090904192342_create_indexes_for_inventory_units.rb +12 -0
  432. data/db/migrate/20090923100315_add_count_on_hand_to_variants_and_products.rb +36 -0
  433. data/db/migrate/20091007134354_change_taxons_to_nested_set.rb +40 -0
  434. data/db/migrate/20091008091614_move_to_configurable_gateways.rb +55 -0
  435. data/db/migrate/20091012120519_product_groups_and_scopes.rb +25 -0
  436. data/db/migrate/20091015110842_add_open_id_authentication_tables.rb +20 -0
  437. data/db/migrate/20091015153048_add_openid_field_to_users.rb +19 -0
  438. data/db/migrate/20091016174634_change_preference_value_type.rb +10 -0
  439. data/db/migrate/20091017175558_create_billing_integrations.rb +16 -0
  440. data/db/migrate/20091021133257_charge_refactoring.rb +35 -0
  441. data/db/migrate/20091104151730_add_some_indexes.rb +21 -0
  442. data/db/migrate/20091126190904_checkout_state_machine.rb +13 -0
  443. data/db/migrate/20091209153045_state_for_shipments.rb +9 -0
  444. data/db/migrate/20091209202200_make_state_events_polymorphic.rb +12 -0
  445. data/db/migrate/20091211203813_ship_address_id_for_checkouts.rb +9 -0
  446. data/db/migrate/20091212161118_shipping_method_id_for_checkouts.rb +9 -0
  447. data/db/migrate/20091213222815_creditcard_last_four_digits.rb +19 -0
  448. data/db/migrate/20091214183826_populate_legacy_shipment_state.rb +19 -0
  449. data/db/migrate/20100105090147_add_cost_price.rb +9 -0
  450. data/db/migrate/20100105132138_shipment_id_for_inventory_units.rb +21 -0
  451. data/db/migrate/20100111205525_cim_fields_for_creditcards.rb +11 -0
  452. data/db/migrate/20100112151511_create_return_authorizations.rb +16 -0
  453. data/db/migrate/20100113090919_add_return_authorization_to_inventory_units.rb +9 -0
  454. data/db/migrate/20100113203104_create_trackers.rb +14 -0
  455. data/db/migrate/20100121160010_creditcard_id_for_creditcard_txns.rb +9 -0
  456. data/db/migrate/20100121183934_original_creditcard_txn_id_for_creditcard_txns.rb +9 -0
  457. data/db/migrate/20100125145351_add_test_mode_to_billing_integration.rb +11 -0
  458. data/db/migrate/20100126103714_create_products_product_groups.rb +12 -0
  459. data/db/migrate/20100209025806_create_payment_methods.rb +20 -0
  460. data/db/migrate/20100209144531_polymorphic_payments.rb +29 -0
  461. data/db/migrate/20100213103131_change_payments_payment_method_to_belongs_to.rb +11 -0
  462. data/db/migrate/20100214212536_assign_creditcard_txns_to_payment.rb +16 -0
  463. data/db/migrate/20100223170312_sti_for_transactions.rb +14 -0
  464. data/db/migrate/20100223183812_drop_billing_integrations.rb +16 -0
  465. data/db/migrate/20100224153127_deleted_at_for_payment_methods.rb +13 -0
  466. data/db/migrate/20100301163454_add_adjustments_index.rb +10 -0
  467. data/db/migrate/20100306153445_fix_by_popularity.rb +8 -0
  468. data/db/migrate/20100317120946_add_alt_text_to_images.rb +9 -0
  469. data/db/migrate/20100427121301_add_display_to_payment_methods.rb +9 -0
  470. data/db/migrate/20100504142133_add_addresses_checkouts_indexes.rb +16 -0
  471. data/db/migrate/20100506180619_add_icon_to_taxons.rb +18 -0
  472. data/db/migrate/20100506185838_add_description_to_taxons.rb +11 -0
  473. data/db/migrate/20100528155333_index_for_shipments_number.rb +9 -0
  474. data/db/migrate/20100528185820_add_index_on_users_persistence_token.rb +9 -0
  475. data/db/migrate/20100605152042_add_default_to_tax_categories.rb +9 -0
  476. data/db/migrate/20100624110730_add_display_to_shipping_methods.rb +9 -0
  477. data/db/migrate/20100624123336_rename_payment_method_display.rb +9 -0
  478. data/db/migrate/20100624175547_rename_preferences_field.rb +8 -0
  479. data/db/migrate/20100811163637_add_guest_flag.rb +13 -0
  480. data/db/migrate/20100811205836_drop_order_token.rb +11 -0
  481. data/db/migrate/20100812162326_payments_state_and_assigned_to_order_only.rb +11 -0
  482. data/db/migrate/20100813023502_create_address_keys_for_order.rb +15 -0
  483. data/db/migrate/20100813185745_payment_total_for_orders.rb +9 -0
  484. data/db/migrate/20100816212146_shipping_method_id_for_orders.rb +9 -0
  485. data/db/migrate/20100817152723_add_shipment_and_payment_state.rb +15 -0
  486. data/db/migrate/20100819170125_refactor_adjustments.rb +19 -0
  487. data/db/migrate/20100820135707_response_code_and_avs_response_for_payments.rb +11 -0
  488. data/db/migrate/20100901171814_change_guest_flag_to_anonymous.rb +13 -0
  489. data/db/migrate/20100903203949_email_for_orders.rb +9 -0
  490. data/db/migrate/20100923162011_create_mail_methods.rb +12 -0
  491. data/db/migrate/20100929151905_rename_frozen_to_locked.rb +8 -0
  492. data/db/migrate/20101008190536_move_special_instructions_to_orders.rb +11 -0
  493. data/db/migrate/20101026184700_create_log_entries.rb +15 -0
  494. data/db/migrate/20101026184714_migrate_transactions_to_payment_state.rb +94 -0
  495. data/db/migrate/20101026184746_delete_in_progress_orders.rb +18 -0
  496. data/db/migrate/20101026184808_migrate_checkout_to_orders.rb +27 -0
  497. data/db/migrate/20101026184833_migrate_adjustments.rb +9 -0
  498. data/db/migrate/20101026184855_remove_shipped_state.rb +14 -0
  499. data/db/migrate/20101026184916_prevent_nil_payment_total.rb +8 -0
  500. data/db/migrate/20101026184932_prevent_nil_email.rb +9 -0
  501. data/db/migrate/20101026184959_generate_anonymous_users.rb +14 -0
  502. data/db/migrate/20101026185022_update_order_state.rb +8 -0
  503. data/db/migrate/20101026192225_cleanup_legacy_tables.rb +11 -0
  504. data/db/migrate/20101028151745_remove_number_and_cvv_from_credicard.rb +9 -0
  505. data/db/migrate/20101103212716_drop_anonymous_field_for_user.rb +8 -0
  506. data/db/migrate/20101111133551_renamed_rma_cancelled_state.rb +10 -0
  507. data/db/migrate/20101117031806_fix_problematic_index_names.rb +13 -0
  508. data/db/migrate/20101223215658_add_position_to_variants.rb +9 -0
  509. data/db/migrate/20110110130847_add_next_state_to_state_events.rb +9 -0
  510. data/db/migrate/20110111122537_add_position_to_option_types.rb +9 -0
  511. data/db/migrate/20110314192118_remove_trailing_slashes_in_taxon_permalinks.rb +17 -0
  512. data/db/sample/users.rb +61 -0
  513. data/db/seeds.rb +3 -0
  514. data/lib/apispree_core.rb +94 -0
  515. data/lib/custom_fixtures.rb +7 -0
  516. data/lib/generators/spree_core/upgrade_generator.rb +23 -0
  517. data/lib/generators/templates/config/initializers/%file_name%.rb.tt +1 -0
  518. data/lib/generators/templates/config/locales/%current_locale%.yml.tt +2 -0
  519. data/lib/generators/templates/config/routes.rb +3 -0
  520. data/lib/middleware/seo_assist.rb +44 -0
  521. data/lib/product_filters.rb +179 -0
  522. data/lib/redirect_legacy_product_url.rb +13 -0
  523. data/lib/scopes.rb +69 -0
  524. data/lib/scopes/dynamic.rb +31 -0
  525. data/lib/scopes/product.rb +266 -0
  526. data/lib/scopes/variant.rb +18 -0
  527. data/lib/spree/calculated_adjustments.rb +69 -0
  528. data/lib/spree/config.rb +33 -0
  529. data/lib/spree/current_order.rb +30 -0
  530. data/lib/spree/file_utilz.rb +73 -0
  531. data/lib/spree/gateway_error.rb +3 -0
  532. data/lib/spree/mail_interceptor.rb +23 -0
  533. data/lib/spree/mail_settings.rb +33 -0
  534. data/lib/spree/preference_access.rb +27 -0
  535. data/lib/spree/search/base.rb +64 -0
  536. data/lib/spree_base.rb +116 -0
  537. data/lib/spree_core/action_callbacks.rb +26 -0
  538. data/lib/spree_core/authorize_net_cim_hack.rb +871 -0
  539. data/lib/spree_core/delegate_belongs_to.rb +89 -0
  540. data/lib/spree_core/enumerable_constants.rb +209 -0
  541. data/lib/spree_core/ext/active_record.rb +15 -0
  542. data/lib/spree_core/ext/array.rb +14 -0
  543. data/lib/spree_core/ext/hash.rb +75 -0
  544. data/lib/spree_core/ext/string.rb +10 -0
  545. data/lib/spree_core/preferences/model_hooks.rb +293 -0
  546. data/lib/spree_core/preferences/preference_definition.rb +53 -0
  547. data/lib/spree_core/railtie.rb +63 -0
  548. data/lib/spree_core/spree_custom_responder.rb +29 -0
  549. data/lib/spree_core/spree_respond_with.rb +57 -0
  550. data/lib/spree_core/ssl_requirement.rb +104 -0
  551. data/lib/spree_core/testing_support/factories.rb +13 -0
  552. data/lib/spree_core/testing_support/factories/address_factory.rb +20 -0
  553. data/lib/spree_core/testing_support/factories/adjustment_factory.rb +6 -0
  554. data/lib/spree_core/testing_support/factories/calculator_factory.rb +5 -0
  555. data/lib/spree_core/testing_support/factories/configuraion_factory.rb +4 -0
  556. data/lib/spree_core/testing_support/factories/country_factory.rb +7 -0
  557. data/lib/spree_core/testing_support/factories/creditcard_factory.rb +11 -0
  558. data/lib/spree_core/testing_support/factories/inventory_unit_factory.rb +7 -0
  559. data/lib/spree_core/testing_support/factories/line_item_factory.rb +8 -0
  560. data/lib/spree_core/testing_support/factories/mail_method_factory.rb +4 -0
  561. data/lib/spree_core/testing_support/factories/options_factory.rb +10 -0
  562. data/lib/spree_core/testing_support/factories/order_factory.rb +18 -0
  563. data/lib/spree_core/testing_support/factories/payment_factory.rb +26 -0
  564. data/lib/spree_core/testing_support/factories/payment_method_factory.rb +17 -0
  565. data/lib/spree_core/testing_support/factories/product_factory.rb +16 -0
  566. data/lib/spree_core/testing_support/factories/product_group_factory.rb +3 -0
  567. data/lib/spree_core/testing_support/factories/product_option_type_factory.rb +4 -0
  568. data/lib/spree_core/testing_support/factories/product_property_factory.rb +4 -0
  569. data/lib/spree_core/testing_support/factories/product_scope_factory.rb +6 -0
  570. data/lib/spree_core/testing_support/factories/property_factory.rb +4 -0
  571. data/lib/spree_core/testing_support/factories/prototype_factory.rb +4 -0
  572. data/lib/spree_core/testing_support/factories/return_authorization_factory.rb +8 -0
  573. data/lib/spree_core/testing_support/factories/role_factory.rb +9 -0
  574. data/lib/spree_core/testing_support/factories/shipment_factory.rb +9 -0
  575. data/lib/spree_core/testing_support/factories/shipping_category_factory.rb +5 -0
  576. data/lib/spree_core/testing_support/factories/shipping_method_factory.rb +11 -0
  577. data/lib/spree_core/testing_support/factories/state_factory.rb +11 -0
  578. data/lib/spree_core/testing_support/factories/tax_category_factory.rb +8 -0
  579. data/lib/spree_core/testing_support/factories/tax_rate_factory.rb +5 -0
  580. data/lib/spree_core/testing_support/factories/taxon_factory.rb +5 -0
  581. data/lib/spree_core/testing_support/factories/taxonomy_factory.rb +3 -0
  582. data/lib/spree_core/testing_support/factories/tracker_factory.rb +5 -0
  583. data/lib/spree_core/testing_support/factories/user_factory.rb +15 -0
  584. data/lib/spree_core/testing_support/factories/variant_factory.rb +14 -0
  585. data/lib/spree_core/testing_support/factories/zone_factory.rb +18 -0
  586. data/lib/spree_core/theme_support.rb +4 -0
  587. data/lib/spree_core/theme_support/hook.rb +51 -0
  588. data/lib/spree_core/theme_support/hook_listener.rb +68 -0
  589. data/lib/spree_core/theme_support/hook_modifier.rb +35 -0
  590. data/lib/spree_core/theme_support/more_patches.rb +118 -0
  591. data/lib/spree_core/version.rb +5 -0
  592. data/lib/store_helpers.rb +10 -0
  593. data/lib/tasks/common.rb +30 -0
  594. data/lib/tasks/core.rake +105 -0
  595. data/lib/tasks/install.rake +26 -0
  596. data/lib/tasks/rake_util.rb +19 -0
  597. data/lib/tasks/taxon.rake +14 -0
  598. data/lib/tasks/themes.rake +34 -0
  599. data/public/images/add-to-cart.png +0 -0
  600. data/public/images/admin/bg/active-tab.png +0 -0
  601. data/public/images/admin/bg/admin_tab_back.png +0 -0
  602. data/public/images/admin/bg/admin_tab_selected_back.png +0 -0
  603. data/public/images/admin/bg/content-back-blue.png +0 -0
  604. data/public/images/admin/bg/content-back-green.png +0 -0
  605. data/public/images/admin/bg/content-back.png +0 -0
  606. data/public/images/admin/bg/flash-error.png +0 -0
  607. data/public/images/admin/bg/flash-notice.png +0 -0
  608. data/public/images/admin/bg/green-stripes.gif +0 -0
  609. data/public/images/admin/bg/green-stripes.png +0 -0
  610. data/public/images/admin/bg/grid_header_back.png +0 -0
  611. data/public/images/admin/bg/grid_header_back_green.png +0 -0
  612. data/public/images/admin/bg/header-bg.png +0 -0
  613. data/public/images/admin/bg/header.png +0 -0
  614. data/public/images/admin/bg/header_bg.jpg +0 -0
  615. data/public/images/admin/bg/menu-current.png +0 -0
  616. data/public/images/admin/bg/red-stripes.gif +0 -0
  617. data/public/images/admin/bg/red-stripes.png +0 -0
  618. data/public/images/admin/bg/spree_50.png +0 -0
  619. data/public/images/admin/bg/subnav-divider.png +0 -0
  620. data/public/images/admin/bg/subnav.png +0 -0
  621. data/public/images/admin/bg/tab-back.png +0 -0
  622. data/public/images/admin/buttons/blue/left_01.png +0 -0
  623. data/public/images/admin/buttons/blue/right_01.png +0 -0
  624. data/public/images/admin/buttons/drag-handle-green.png +0 -0
  625. data/public/images/admin/buttons/green/left_01.png +0 -0
  626. data/public/images/admin/buttons/green/right_01.png +0 -0
  627. data/public/images/admin/buttons/left_01.png +0 -0
  628. data/public/images/admin/buttons/left_01_small.png +0 -0
  629. data/public/images/admin/buttons/orange/left_03.png +0 -0
  630. data/public/images/admin/buttons/orange/right_03.png +0 -0
  631. data/public/images/admin/buttons/right_01.png +0 -0
  632. data/public/images/admin/buttons/right_01_small.png +0 -0
  633. data/public/images/admin/icons/16x16/1.png +0 -0
  634. data/public/images/admin/icons/16x16/10.png +0 -0
  635. data/public/images/admin/icons/16x16/2.png +0 -0
  636. data/public/images/admin/icons/16x16/3.png +0 -0
  637. data/public/images/admin/icons/16x16/4.png +0 -0
  638. data/public/images/admin/icons/16x16/5.png +0 -0
  639. data/public/images/admin/icons/16x16/6.png +0 -0
  640. data/public/images/admin/icons/16x16/7.png +0 -0
  641. data/public/images/admin/icons/16x16/8.png +0 -0
  642. data/public/images/admin/icons/16x16/9.png +0 -0
  643. data/public/images/admin/icons/32x32/1.png +0 -0
  644. data/public/images/admin/icons/32x32/10.png +0 -0
  645. data/public/images/admin/icons/32x32/11.png +0 -0
  646. data/public/images/admin/icons/32x32/2.png +0 -0
  647. data/public/images/admin/icons/32x32/3.png +0 -0
  648. data/public/images/admin/icons/32x32/4.png +0 -0
  649. data/public/images/admin/icons/32x32/5.png +0 -0
  650. data/public/images/admin/icons/32x32/6.png +0 -0
  651. data/public/images/admin/icons/32x32/7.png +0 -0
  652. data/public/images/admin/icons/32x32/8.png +0 -0
  653. data/public/images/admin/icons/32x32/9.png +0 -0
  654. data/public/images/admin/icons/accept.png +0 -0
  655. data/public/images/admin/icons/add.gif +0 -0
  656. data/public/images/admin/icons/add.png +0 -0
  657. data/public/images/admin/icons/arrow-down.gif +0 -0
  658. data/public/images/admin/icons/cross.png +0 -0
  659. data/public/images/admin/icons/delete.gif +0 -0
  660. data/public/images/admin/icons/delete.png +0 -0
  661. data/public/images/admin/icons/drag.gif +0 -0
  662. data/public/images/admin/icons/edit.gif +0 -0
  663. data/public/images/admin/icons/edit.png +0 -0
  664. data/public/images/admin/icons/email.png +0 -0
  665. data/public/images/admin/icons/error.png +0 -0
  666. data/public/images/admin/icons/exclamation.png +0 -0
  667. data/public/images/admin/icons/feed.png +0 -0
  668. data/public/images/admin/icons/pdf.png +0 -0
  669. data/public/images/admin/icons/reorder.gif +0 -0
  670. data/public/images/admin/icons/search.gif +0 -0
  671. data/public/images/admin/icons/send-email.png +0 -0
  672. data/public/images/admin/icons/stop.png +0 -0
  673. data/public/images/admin/icons/tick.png +0 -0
  674. data/public/images/admin/icons/up.gif +0 -0
  675. data/public/images/admin/icons/xls.png +0 -0
  676. data/public/images/admin/tabs/off-left.png +0 -0
  677. data/public/images/admin/tabs/off-right.png +0 -0
  678. data/public/images/admin/tabs/on-left.png +0 -0
  679. data/public/images/admin/tabs/on-right.png +0 -0
  680. data/public/images/ajax_loader.gif +0 -0
  681. data/public/images/amex_cid.gif +0 -0
  682. data/public/images/bg-button-hover.png +0 -0
  683. data/public/images/bg-button-pressed.png +0 -0
  684. data/public/images/bg-button.gif +0 -0
  685. data/public/images/bg-button.png +0 -0
  686. data/public/images/blue/left_01.png +0 -0
  687. data/public/images/blue/right_01.png +0 -0
  688. data/public/images/body-back.png +0 -0
  689. data/public/images/bottom_shine.png +0 -0
  690. data/public/images/breadcrumb.gif +0 -0
  691. data/public/images/button-dark-hover.png +0 -0
  692. data/public/images/button-dark.png +0 -0
  693. data/public/images/buttons/bg-button-hover.png +0 -0
  694. data/public/images/buttons/bg-button-pressed.png +0 -0
  695. data/public/images/buttons/bg-button.gif +0 -0
  696. data/public/images/buttons/bg-button.png +0 -0
  697. data/public/images/buttons/blue/left_01.png +0 -0
  698. data/public/images/buttons/blue/right_01.png +0 -0
  699. data/public/images/buttons/button-dark-hover.png +0 -0
  700. data/public/images/buttons/button-dark.png +0 -0
  701. data/public/images/buttons/drag-handle-green.png +0 -0
  702. data/public/images/buttons/green/left_01.png +0 -0
  703. data/public/images/buttons/green/right_01.png +0 -0
  704. data/public/images/buttons/left_01.png +0 -0
  705. data/public/images/buttons/left_01_small.png +0 -0
  706. data/public/images/buttons/orange/left_03.png +0 -0
  707. data/public/images/buttons/orange/right_03.png +0 -0
  708. data/public/images/buttons/right_01.png +0 -0
  709. data/public/images/buttons/right_01_small.png +0 -0
  710. data/public/images/buttons/sxsw-ribbon-v1.png +0 -0
  711. data/public/images/buttons/top-shine.png +0 -0
  712. data/public/images/calendar_date_select/calendar.gif +0 -0
  713. data/public/images/cart-empty.png +0 -0
  714. data/public/images/cart-empty_x32.png +0 -0
  715. data/public/images/cart-full.png +0 -0
  716. data/public/images/cart-full_x32.png +0 -0
  717. data/public/images/checkout.png +0 -0
  718. data/public/images/creditcard.gif +0 -0
  719. data/public/images/datepicker/backstripes.gif +0 -0
  720. data/public/images/datepicker/bg_header.jpg +0 -0
  721. data/public/images/datepicker/bullet1.gif +0 -0
  722. data/public/images/datepicker/bullet2.gif +0 -0
  723. data/public/images/datepicker/cal.gif +0 -0
  724. data/public/images/datepicker/gradient-e5e5e5-ffffff.gif +0 -0
  725. data/public/images/discover_cid.gif +0 -0
  726. data/public/images/drag-handle-green.png +0 -0
  727. data/public/images/favicon.ico +0 -0
  728. data/public/images/green/left_01.png +0 -0
  729. data/public/images/green/right_01.png +0 -0
  730. data/public/images/grid.png +0 -0
  731. data/public/images/left_01.png +0 -0
  732. data/public/images/left_01_small.png +0 -0
  733. data/public/images/master_cid.jpg +0 -0
  734. data/public/images/menu-current.png +0 -0
  735. data/public/images/menu-hover.png +0 -0
  736. data/public/images/noimage/mini.jpg +0 -0
  737. data/public/images/noimage/product.jpg +0 -0
  738. data/public/images/noimage/small.jpg +0 -0
  739. data/public/images/openid-inputicon.gif +0 -0
  740. data/public/images/orange/left_03.png +0 -0
  741. data/public/images/orange/right_03.png +0 -0
  742. data/public/images/progress.gif +0 -0
  743. data/public/images/reorder.jpg +0 -0
  744. data/public/images/right_01.png +0 -0
  745. data/public/images/right_01_small.png +0 -0
  746. data/public/images/separator.png +0 -0
  747. data/public/images/shadow-top.png +0 -0
  748. data/public/images/shadow_top.png +0 -0
  749. data/public/images/spinner.gif +0 -0
  750. data/public/images/spree.jpg +0 -0
  751. data/public/images/spree/progress.gif +0 -0
  752. data/public/images/spree/spinner.gif +0 -0
  753. data/public/images/spree/spree.jpg +0 -0
  754. data/public/images/step-progress/completed-completed.gif +0 -0
  755. data/public/images/step-progress/completed-current.gif +0 -0
  756. data/public/images/step-progress/completed-first.gif +0 -0
  757. data/public/images/step-progress/current-first.gif +0 -0
  758. data/public/images/step-progress/current-incomplete.gif +0 -0
  759. data/public/images/step-progress/current-right.gif +0 -0
  760. data/public/images/step-progress/incomplete-incomplete.gif +0 -0
  761. data/public/images/step-progress/incomplete-right.gif +0 -0
  762. data/public/images/steps/1.png +0 -0
  763. data/public/images/steps/1_small.png +0 -0
  764. data/public/images/steps/2.png +0 -0
  765. data/public/images/steps/2_small.png +0 -0
  766. data/public/images/steps/3.png +0 -0
  767. data/public/images/steps/3_small.png +0 -0
  768. data/public/images/steps/4.png +0 -0
  769. data/public/images/steps/4_small.png +0 -0
  770. data/public/images/steps/5.png +0 -0
  771. data/public/images/steps/5_small.png +0 -0
  772. data/public/images/steps/6.png +0 -0
  773. data/public/images/steps/6_small.png +0 -0
  774. data/public/images/sxsw-ribbon-v1.png +0 -0
  775. data/public/images/tab_bottom.gif +0 -0
  776. data/public/images/tile-header.png +0 -0
  777. data/public/images/tile-slider.png +0 -0
  778. data/public/images/top-shine.png +0 -0
  779. data/public/images/tree-nav-icons/bullet.gif +0 -0
  780. data/public/images/tree-nav-icons/minus.gif +0 -0
  781. data/public/images/tree-nav-icons/plus.gif +0 -0
  782. data/public/images/tree-nav-icons/treeview-loading.gif +0 -0
  783. data/public/images/tree-nav-icons/treeview-sprite.gif +0 -0
  784. data/public/images/update.png +0 -0
  785. data/public/images/visa_cid.gif +0 -0
  786. data/public/images/wrapper-back-2.png +0 -0
  787. data/public/images/wrapper-back.png +0 -0
  788. data/public/images/yui-menubaritem_submenuindicator.png +0 -0
  789. data/public/images/yui-menubaritem_submenuindicator_disabled.png +0 -0
  790. data/public/images/yui-menuitem_checkbox.png +0 -0
  791. data/public/images/yui-menuitem_checkbox_disabled.png +0 -0
  792. data/public/images/yui-menuitem_submenuindicator.png +0 -0
  793. data/public/images/yui-menuitem_submenuindicator_disabled.png +0 -0
  794. data/public/images/yui-sprite.png +0 -0
  795. data/public/javascripts/additional-methods.js +236 -0
  796. data/public/javascripts/admin.js +232 -0
  797. data/public/javascripts/admin/address_states.js +25 -0
  798. data/public/javascripts/admin/checkouts/edit.js +130 -0
  799. data/public/javascripts/admin/orders/edit.js +23 -0
  800. data/public/javascripts/admin/orders/edit_form.js +16 -0
  801. data/public/javascripts/admin/payments/new.js +9 -0
  802. data/public/javascripts/admin/unobtrusive_handlers.js +43 -0
  803. data/public/javascripts/application.js +12 -0
  804. data/public/javascripts/calculator.js +15 -0
  805. data/public/javascripts/checkout.js +75 -0
  806. data/public/javascripts/datepicker.js +1445 -0
  807. data/public/javascripts/gateway.js +13 -0
  808. data/public/javascripts/jquery-1.4.2.min.js +154 -0
  809. data/public/javascripts/jquery-ui.js +188 -0
  810. data/public/javascripts/jquery.alerts/images/help.gif +0 -0
  811. data/public/javascripts/jquery.alerts/images/important.gif +0 -0
  812. data/public/javascripts/jquery.alerts/images/info.gif +0 -0
  813. data/public/javascripts/jquery.alerts/images/title.gif +0 -0
  814. data/public/javascripts/jquery.alerts/jquery.alerts.css +57 -0
  815. data/public/javascripts/jquery.alerts/jquery.alerts.js +235 -0
  816. data/public/javascripts/jquery.alerts/jquery.alerts.spree.css +30 -0
  817. data/public/javascripts/jquery.autocomplete.min.js +13 -0
  818. data/public/javascripts/jquery.cookie.js +96 -0
  819. data/public/javascripts/jquery.delayedobserver.js +35 -0
  820. data/public/javascripts/jquery.suggest.js +276 -0
  821. data/public/javascripts/jquery.template.js +255 -0
  822. data/public/javascripts/jquery.tokeninput.js +618 -0
  823. data/public/javascripts/jquery.validate.min.js +16 -0
  824. data/public/javascripts/jsTree/jquery.jstree.js +3510 -0
  825. data/public/javascripts/jsTree/themes/apple/bg.jpg +0 -0
  826. data/public/javascripts/jsTree/themes/apple/d.png +0 -0
  827. data/public/javascripts/jsTree/themes/apple/dot_for_ie.gif +0 -0
  828. data/public/javascripts/jsTree/themes/apple/style.css +60 -0
  829. data/public/javascripts/jsTree/themes/apple/throbber.gif +0 -0
  830. data/public/javascripts/lang/af.js +40 -0
  831. data/public/javascripts/lang/ar.js +50 -0
  832. data/public/javascripts/lang/de.js +40 -0
  833. data/public/javascripts/lang/du.js +40 -0
  834. data/public/javascripts/lang/en.js +42 -0
  835. data/public/javascripts/lang/es.js +41 -0
  836. data/public/javascripts/lang/fi.js +40 -0
  837. data/public/javascripts/lang/fr.js +44 -0
  838. data/public/javascripts/lang/gr.js +40 -0
  839. data/public/javascripts/lang/he.js +49 -0
  840. data/public/javascripts/lang/it.js +13 -0
  841. data/public/javascripts/lang/nl.js +40 -0
  842. data/public/javascripts/lang/no.js +40 -0
  843. data/public/javascripts/lang/pt.js +50 -0
  844. data/public/javascripts/lang/ro.js +40 -0
  845. data/public/javascripts/lang/ru.js +40 -0
  846. data/public/javascripts/lang/sp.js +40 -0
  847. data/public/javascripts/lang/sv.js +41 -0
  848. data/public/javascripts/lang/ua.js +40 -0
  849. data/public/javascripts/localization/messages_cn.js +24 -0
  850. data/public/javascripts/localization/messages_cs.js +23 -0
  851. data/public/javascripts/localization/messages_da.js +21 -0
  852. data/public/javascripts/localization/messages_de.js +21 -0
  853. data/public/javascripts/localization/messages_es.js +24 -0
  854. data/public/javascripts/localization/messages_fr.js +23 -0
  855. data/public/javascripts/localization/messages_hu.js +21 -0
  856. data/public/javascripts/localization/messages_it.js +26 -0
  857. data/public/javascripts/localization/messages_kk.js +23 -0
  858. data/public/javascripts/localization/messages_nl.js +23 -0
  859. data/public/javascripts/localization/messages_no.js +23 -0
  860. data/public/javascripts/localization/messages_pl.js +23 -0
  861. data/public/javascripts/localization/messages_ptbr.js +30 -0
  862. data/public/javascripts/localization/messages_ro.js +24 -0
  863. data/public/javascripts/localization/messages_ru.js +23 -0
  864. data/public/javascripts/localization/messages_se.js +23 -0
  865. data/public/javascripts/localization/messages_sk.js +21 -0
  866. data/public/javascripts/localization/messages_tr.js +24 -0
  867. data/public/javascripts/localization/messages_tw.js +24 -0
  868. data/public/javascripts/localization/messages_ua.js +24 -0
  869. data/public/javascripts/nested-attribute.js +26 -0
  870. data/public/javascripts/open_id.js +15 -0
  871. data/public/javascripts/product.js +49 -0
  872. data/public/javascripts/rails.js +127 -0
  873. data/public/javascripts/taxonomy.js +201 -0
  874. data/public/javascripts/zone.js +39 -0
  875. data/public/stylesheets/admin/admin-forms.css +154 -0
  876. data/public/stylesheets/admin/admin-reset.css +69 -0
  877. data/public/stylesheets/admin/admin-tables.css +39 -0
  878. data/public/stylesheets/admin/admin-typography.css +117 -0
  879. data/public/stylesheets/admin/admin.css +614 -0
  880. data/public/stylesheets/admin/autocomplete.css +73 -0
  881. data/public/stylesheets/admin/dashboard.css +143 -0
  882. data/public/stylesheets/admin/edit_checkouts.css +57 -0
  883. data/public/stylesheets/admin/grids.css +314 -0
  884. data/public/stylesheets/admin/reset-fonts-grids-2-6-0.css +7 -0
  885. data/public/stylesheets/admin/token-input.css +109 -0
  886. data/public/stylesheets/admin/yui-includes.css +14 -0
  887. data/public/stylesheets/datepicker.css +263 -0
  888. data/public/stylesheets/jquery.autocomplete.css +48 -0
  889. data/public/stylesheets/screen.css +916 -0
  890. metadata +1099 -0
@@ -0,0 +1,33 @@
1
+ module Spree
2
+ # Singleton class to access the configuration object (AppConfiguration.first by default) and its preferences.
3
+ #
4
+ # Usage:
5
+ # Spree::Config[:foo] # Returns the +foo+ preference
6
+ # Spree::Config[] # Returns a Hash with all the application preferences
7
+ # Spree::Config.instance # Returns the configuration object (AppConfiguration.first)
8
+ # Spree::Config.set(preferences_hash) # Set the application preferences as especified in +preference_hash+
9
+ # Spree::Config.searcher/searcher= # get/set the default product search implementation
10
+ class Config
11
+ include Singleton
12
+ include Spree::PreferenceAccess
13
+
14
+ class << self
15
+ def instance
16
+ return @configuration if @configuration
17
+ return nil unless ActiveRecord::Base.connection.tables.include?('configurations')
18
+ @configuration ||= AppConfiguration.find_or_create_by_name("Default configuration")
19
+ @configuration
20
+ end
21
+
22
+ # searcher_class allows spree extension writers to provide their own Search class
23
+ def searcher_class
24
+ @searcher_class ||= Spree::Search::Base
25
+ end
26
+
27
+ def searcher_class=(sclass)
28
+ @searcher_class = sclass
29
+ end
30
+
31
+ end
32
+ end
33
+ end
@@ -0,0 +1,30 @@
1
+ module Spree
2
+ module CurrentOrder
3
+
4
+ # This should be overridden by an auth-related extension which would then have the
5
+ # opportunity to associate the new order with the # current user before saving.
6
+ def before_save_new_order
7
+ end
8
+
9
+ # This should be overridden by an auth-related extension which would then have the
10
+ # opporutnity to store tokens, etc. in the session # after saving.
11
+ def after_save_new_order
12
+ end
13
+
14
+ # The current incomplete order from the session for use in cart and during checkout
15
+ def current_order(create_order_if_necessary = false)
16
+ return @current_order if @current_order
17
+ if session[:order_id]
18
+ @current_order = Order.find_by_id(session[:order_id], :include => :adjustments)
19
+ end
20
+ if create_order_if_necessary and (@current_order.nil? or @current_order.completed?)
21
+ @current_order = Order.new
22
+ before_save_new_order
23
+ @current_order.save!
24
+ after_save_new_order
25
+ end
26
+ session[:order_id] = @current_order ? @current_order.id : nil
27
+ @current_order
28
+ end
29
+ end
30
+ end
@@ -0,0 +1,73 @@
1
+ require 'fileutils'
2
+
3
+ module Spree
4
+ class FileUtilz
5
+ # A general purpose method to mirror a directory (+source+) into a destination
6
+ # directory, including all files and subdirectories. Files will not be mirrored
7
+ # if they are identical already (checked via FileUtils#identical?).
8
+ #
9
+ # Copyright (c) 2008 James Adam (The MIT License)
10
+
11
+ # added: do mirroring with backup saving turned on
12
+ def self.mirror_with_backup(source, destination)
13
+ self.mirror_files(source, destination, true)
14
+ end
15
+
16
+ def self.mirror_files(source, destination, create_backups = false)
17
+ # TODO: use Rake::FileList#pathmap?
18
+ if File.directory?(source)
19
+ source_files = Dir[source + "/**/*"]
20
+ source_dirs = source_files.select { |d| File.directory?(d) }
21
+ source_files -= source_dirs
22
+ elsif File.exist? source
23
+ source_dirs = []
24
+ source_files = [source]
25
+ source = File.dirname source
26
+ else
27
+ puts "Could not mirror #{source} - entity does not exist"
28
+ return
29
+ end
30
+
31
+ unless source_files.empty?
32
+ base_target_dir = File.join(destination)
33
+ base_target_dir = File.dirname base_target_dir unless File.directory? base_target_dir
34
+ FileUtils.mkdir_p(base_target_dir)
35
+ end
36
+
37
+ base_target_dir ||= File.join(destination)
38
+
39
+ source_dirs.each do |dir|
40
+ # strip down these paths so we have simple, relative paths we can
41
+ # add to the destination
42
+ target_dir = File.join(base_target_dir, dir.gsub(source, ''))
43
+ begin
44
+ FileUtils.mkdir_p(target_dir)
45
+ rescue Exception => e
46
+ raise "Could not create directory #{target_dir}: \n" + e
47
+ end
48
+ end
49
+
50
+ source_files.each do |file|
51
+ begin
52
+ target = File.join(base_target_dir, file.gsub(source, ''))
53
+ unless File.exist?(target) && self.same_contents(file, target)
54
+ # WAS FileUtils.identical?(file, target), but we want to test contents too
55
+ if create_backups && File.exist?(target)
56
+ File.rename(target, target + '~')
57
+ end
58
+ FileUtils.cp(file, target)
59
+ end
60
+ rescue Exception => e
61
+ raise "Could not copy #{file} to #{target}: \n" + e
62
+ end
63
+ end
64
+ end
65
+
66
+ # for windows users...
67
+ def self.same_contents(source,destination)
68
+ File.read(source) == File.read(destination)
69
+ end
70
+
71
+ end
72
+ end
73
+
@@ -0,0 +1,3 @@
1
+ module Spree
2
+ class GatewayError < RuntimeError; end
3
+ end
@@ -0,0 +1,23 @@
1
+ # Allows us to intercept any outbound mail message and make last minute changes (such as specifying a "from" address or
2
+ # sending to a test email account.)
3
+ #
4
+ # See http://railscasts.com/episodes/206-action-mailer-in-rails-3 for more details.
5
+ module Spree
6
+ class MailInterceptor
7
+
8
+ def self.delivering_email(message)
9
+ return unless mail_method = MailMethod.current
10
+ message.from ||= mail_method.preferred_mails_from
11
+
12
+ if mail_method.preferred_intercept_email.present?
13
+ message.subject = "[#{message.to}] #{message.subject}"
14
+ message.to = mail_method.preferred_intercept_email
15
+ end
16
+
17
+ if mail_method.preferred_mail_bcc.present?
18
+ message.bcc = mail_method.preferred_mail_bcc
19
+ end
20
+ end
21
+
22
+ end
23
+ end
@@ -0,0 +1,33 @@
1
+ module Spree
2
+ module MailSettings
3
+
4
+ # Override the Rails application mail settings based on preference.
5
+ # This makes it possible to configure the mail settings
6
+ # through an admin interface instead of requiring changes to the Rails envrionment file.
7
+ def self.init
8
+ return unless mail_method = MailMethod.current
9
+ if mail_method.prefers_enable_mail_delivery?
10
+ mail_server_settings = {
11
+ :address => mail_method.preferred_mail_host,
12
+ :domain => mail_method.preferred_mail_domain,
13
+ :port => mail_method.preferred_mail_port,
14
+ :authentication => mail_method.preferred_mail_auth_type
15
+ }
16
+
17
+ if mail_method.preferred_mail_auth_type != 'none'
18
+ mail_server_settings[:user_name] = mail_method.preferred_smtp_username
19
+ mail_server_settings[:password] = mail_method.preferred_smtp_password
20
+ end
21
+
22
+ mail_server_settings[:enable_starttls_auto] = (mail_method.preferred_secure_connection_type == 'TLS')
23
+
24
+ ActionMailer::Base.smtp_settings = mail_server_settings
25
+ ActionMailer::Base.perform_deliveries = true
26
+ else
27
+ #logger.warn "NOTICE: Mail not enabled"
28
+ ActionMailer::Base.perform_deliveries = false
29
+ end
30
+ end
31
+
32
+ end
33
+ end
@@ -0,0 +1,27 @@
1
+ module Spree::PreferenceAccess
2
+
3
+ def self.included(base)
4
+ class << base
5
+ def get(key = nil)
6
+ key = key.to_s if key.is_a?(Symbol)
7
+ return nil unless config = self.instance
8
+ # preferences will be cached under the name of the class including this module (ex. Spree::Config)
9
+ prefs = Rails.cache.fetch("configuration_#{config.class.name}".to_sym) { config.preferences }
10
+ return prefs if key.nil?
11
+ prefs[key]
12
+ end
13
+
14
+ # Set the preferences as specified in a hash (like params[:preferences] from a post request)
15
+ def set(preferences={})
16
+ config = self.instance
17
+ preferences.each do |key, value|
18
+ config.set_preference(key, value)
19
+ end
20
+ config.save
21
+ Rails.cache.delete("configuration_#{config.class.name}".to_sym)
22
+ end
23
+
24
+ alias_method :[], :get
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,64 @@
1
+ module Spree::Search
2
+ class Base
3
+ attr_accessor :properties
4
+
5
+ def initialize(params)
6
+ @properties = {}
7
+ prepare(params)
8
+ end
9
+
10
+ def retrieve_products
11
+ base_scope = get_base_scope
12
+ @products_scope = @product_group.apply_on(base_scope)
13
+ curr_page = manage_pagination && keywords ? 1 : page
14
+
15
+ @products = @products_scope.paginate({
16
+ :include => [:images, :master],
17
+ :per_page => per_page,
18
+ :page => curr_page
19
+ })
20
+ end
21
+
22
+ def method_missing(name)
23
+ @properties[name]
24
+ end
25
+
26
+ protected
27
+ def get_base_scope
28
+ base_scope = @cached_product_group ? @cached_product_group.products.active : Product.active
29
+ base_scope = base_scope.in_taxon(taxon) unless taxon.blank?
30
+ base_scope = get_products_conditions_for(base_scope, keywords) unless keywords.blank?
31
+
32
+ base_scope = base_scope.on_hand unless Spree::Config[:show_zero_stock_products]
33
+ base_scope = base_scope.group_by_products_id if @product_group.product_scopes.size > 1
34
+ base_scope
35
+ end
36
+
37
+ # method should return new scope based on base_scope
38
+ def get_products_conditions_for(base_scope, query)
39
+ base_scope.like_any([:name, :description], query.split)
40
+ end
41
+
42
+ def prepare(params)
43
+ @properties[:taxon] = params[:taxon].blank? ? nil : Taxon.find(params[:taxon])
44
+ @properties[:keywords] = params[:keywords]
45
+
46
+ per_page = params[:per_page].to_i
47
+ @properties[:per_page] = per_page > 0 ? per_page : Spree::Config[:products_per_page]
48
+ @properties[:page] = (params[:page].to_i <= 0) ? 1 : params[:page].to_i
49
+
50
+ if !params[:order_by_price].blank?
51
+ @product_group = ProductGroup.new.from_route([params[:order_by_price]+"_by_master_price"])
52
+ elsif params[:product_group_name]
53
+ @cached_product_group = ProductGroup.find_by_permalink(params[:product_group_name])
54
+ @product_group = ProductGroup.new
55
+ elsif params[:product_group_query]
56
+ @product_group = ProductGroup.new.from_route(params[:product_group_query].split("/"))
57
+ else
58
+ @product_group = ProductGroup.new
59
+ end
60
+ @product_group = @product_group.from_search(params[:search]) if params[:search]
61
+
62
+ end
63
+ end
64
+ end
@@ -0,0 +1,116 @@
1
+ module SpreeBase
2
+ module InstanceMethods
3
+ def access_forbidden
4
+ render :text => 'Access Forbidden', :layout => true, :status => 401
5
+ end
6
+
7
+ # can be used in views as well as controllers.
8
+ # e.g. <% title = 'This is a custom title for this view' %>
9
+ attr_writer :title
10
+
11
+ def title
12
+ title_string = @title.blank? ? accurate_title : @title
13
+ if title_string.blank?
14
+ default_title
15
+ else
16
+ if Spree::Config[:always_put_site_name_in_title]
17
+ [default_title, title_string].join(' - ')
18
+ else
19
+ title_string
20
+ end
21
+ end
22
+ end
23
+
24
+ protected
25
+
26
+ def default_title
27
+ Spree::Config[:site_name]
28
+ end
29
+
30
+ # this is a hook for subclasses to provide title
31
+ def accurate_title
32
+ Spree::Config[:default_seo_title]
33
+ end
34
+
35
+ # def reject_unknown_object
36
+ # # workaround to catch problems with loading errors for permalink ids (reconsider RC permalink hack elsewhere?)
37
+ # begin
38
+ # load_object
39
+ # rescue Exception => e
40
+ # @object = nil
41
+ # end
42
+ # the_object = instance_variable_get "@#{object_name}"
43
+ # the_object = nil if (the_object.respond_to?(:deleted?) && the_object.deleted?)
44
+ # unless params[:id].blank? || the_object
45
+ # if self.respond_to? :object_missing
46
+ # self.object_missing(params[:id])
47
+ # else
48
+ # render_404(Exception.new("missing object in #{self.class.to_s}"))
49
+ # end
50
+ # end
51
+ # true
52
+ # end
53
+
54
+ def render_404(exception = nil)
55
+ respond_to do |type|
56
+ type.html { render :status => :not_found, :file => "#{Rails.root}/public/404.html", :layout => nil}
57
+ type.all { render :status => :not_found, :nothing => true }
58
+ end
59
+ end
60
+
61
+ private
62
+
63
+ def redirect_back_or_default(default)
64
+ redirect_to(session["user_return_to"] || default)
65
+ session["user_return_to"] = nil
66
+ end
67
+
68
+ def instantiate_controller_and_action_names
69
+ @current_action = action_name
70
+ @current_controller = controller_name
71
+ end
72
+
73
+ def get_taxonomies
74
+ @taxonomies ||= Taxonomy.includes(:root => :children)
75
+ @taxonomies.reject { |t| t.root.nil? }
76
+ end
77
+
78
+ def current_gateway
79
+ @current_gateway ||= Gateway.current
80
+ end
81
+
82
+ #RAILS 3 TODO
83
+ # # Load all models using STI to fix associations such as @order.credits giving no results and resulting in incorrect order totals
84
+ # def touch_sti_subclasses
85
+ # if Rails.env == 'development'
86
+ # load(File.join(SPREE_ROOT,'config/initializers/touch.rb'))
87
+ # end
88
+ # end
89
+
90
+ def set_user_language
91
+ locale = session[:locale] || Spree::Config[:default_locale]
92
+ locale = I18n.default_locale unless locale && I18n.available_locales.include?(locale.to_sym)
93
+ I18n.locale = locale.to_sym
94
+ end
95
+ end
96
+
97
+ def self.included(receiver)
98
+ #receiver.extend ClassMethods
99
+ receiver.send :include, InstanceMethods
100
+ receiver.send :layout, 'spree_application'
101
+ receiver.send :helper, 'hook'
102
+ receiver.send :before_filter, 'instantiate_controller_and_action_names'
103
+ # #RAILS 3 TODO
104
+ # #before_filter :touch_sti_subclasses
105
+ receiver.send :before_filter, 'set_user_language'
106
+
107
+ receiver.send :helper_method, 'title'
108
+ receiver.send :helper_method, 'title='
109
+ receiver.send :helper_method, 'accurate_title'
110
+ receiver.send :helper_method, 'get_taxonomies'
111
+ receiver.send :helper_method, 'current_gateway'
112
+ receiver.send :helper_method, 'current_order'
113
+ receiver.send :include, SslRequirement
114
+ receiver.send :include, Spree::CurrentOrder
115
+ end
116
+ end
@@ -0,0 +1,26 @@
1
+ module Spree
2
+ class ActionCallbacks
3
+ attr_reader :before_methods
4
+ attr_reader :after_methods
5
+ attr_reader :fails_methods
6
+
7
+ def initialize
8
+ @before_methods = []
9
+ @after_methods = []
10
+ @fails_methods = []
11
+ end
12
+
13
+ def before(method)
14
+ @before_methods << method
15
+ end
16
+
17
+ def after(method)
18
+ @after_methods << method
19
+ end
20
+
21
+ def fails(method)
22
+ @fails_methods << method
23
+ end
24
+ end
25
+
26
+ end
@@ -0,0 +1,871 @@
1
+ module ActiveMerchant #:nodoc:
2
+ module Billing #:nodoc:
3
+ # ==== Customer Information Manager (CIM)
4
+ #
5
+ # The Authorize.Net Customer Information Manager (CIM) is an optional additional service that allows you to store sensitive payment information on
6
+ # Authorize.Net's servers, simplifying payments for returning customers and recurring transactions. It can also help with Payment Card Industry (PCI)
7
+ # Data Security Standard compliance, since customer data is no longer stored locally.
8
+ #
9
+ # To use the AuthorizeNetCimGateway CIM must be enabled for your account.
10
+ #
11
+ # Information about CIM is available on the {Authorize.Net website}[http://www.authorize.net/solutions/merchantsolutions/merchantservices/cim/].
12
+ # Information about the CIM API is available at the {Authorize.Net Integration Center}[http://developer.authorize.net/]
13
+ #
14
+ # ==== Login and Password
15
+ #
16
+ # The login and password are not the username and password you use to
17
+ # login to the Authorize.Net Merchant Interface. Instead, you will
18
+ # use the API Login ID as the login and Transaction Key as the
19
+ # password.
20
+ #
21
+ # ==== How to Get Your API Login ID and Transaction Key
22
+ #
23
+ # 1. Log into the Merchant Interface
24
+ # 2. Select Settings from the Main Menu
25
+ # 3. Click on API Login ID and Transaction Key in the Security section
26
+ # 4. Type in the answer to the secret question configured on setup
27
+ # 5. Click Submit
28
+ class AuthorizeNetCimGateway < Gateway
29
+
30
+ class_attribute :test_url, :live_url
31
+
32
+ self.test_url = 'https://apitest.authorize.net/xml/v1/request.api'
33
+ self.live_url = 'https://api.authorize.net/xml/v1/request.api'
34
+
35
+ AUTHORIZE_NET_CIM_NAMESPACE = 'AnetApi/xml/v1/schema/AnetApiSchema.xsd'
36
+
37
+ CIM_ACTIONS = {
38
+ :create_customer_profile => 'createCustomerProfile',
39
+ :create_customer_payment_profile => 'createCustomerPaymentProfile',
40
+ :create_customer_shipping_address => 'createCustomerShippingAddress',
41
+ :get_customer_profile => 'getCustomerProfile',
42
+ :get_customer_payment_profile => 'getCustomerPaymentProfile',
43
+ :get_customer_shipping_address => 'getCustomerShippingAddress',
44
+ :delete_customer_profile => 'deleteCustomerProfile',
45
+ :delete_customer_payment_profile => 'deleteCustomerPaymentProfile',
46
+ :delete_customer_shipping_address => 'deleteCustomerShippingAddress',
47
+ :update_customer_profile => 'updateCustomerProfile',
48
+ :update_customer_payment_profile => 'updateCustomerPaymentProfile',
49
+ :update_customer_shipping_address => 'updateCustomerShippingAddress',
50
+ :create_customer_profile_transaction => 'createCustomerProfileTransaction',
51
+ :validate_customer_payment_profile => 'validateCustomerPaymentProfile'
52
+ }
53
+
54
+ CIM_TRANSACTION_TYPES = {
55
+ :auth_capture => 'profileTransAuthCapture',
56
+ :auth_only => 'profileTransAuthOnly',
57
+ :capture_only => 'profileTransCaptureOnly',
58
+ :prior_auth_capture => 'profileTransPriorAuthCapture',
59
+ :refund => 'profileTransRefund',
60
+ :void => 'profileTransVoid'
61
+ }
62
+
63
+ CIM_VALIDATION_MODES = {
64
+ :none => 'none',
65
+ :test => 'testMode',
66
+ :live => 'liveMode'
67
+ }
68
+
69
+ BANK_ACCOUNT_TYPES = {
70
+ :checking => 'checking',
71
+ :savings => 'savings',
72
+ :business_checking => 'businessChecking'
73
+ }
74
+
75
+ ECHECK_TYPES = {
76
+ :ccd => 'CCD',
77
+ :ppd => 'PPD'
78
+ }
79
+
80
+ self.homepage_url = 'http://www.authorize.net/'
81
+ self.display_name = 'Authorize.Net CIM'
82
+ self.supported_countries = ['US']
83
+ self.supported_cardtypes = [:visa, :master, :american_express, :discover]
84
+
85
+ # Creates a new AuthorizeNetCimGateway
86
+ #
87
+ # The gateway requires that a valid API Login ID and Transaction Key be passed
88
+ # in the +options+ hash.
89
+ #
90
+ # ==== Options
91
+ #
92
+ # * <tt>:login</tt> -- The Authorize.Net API Login ID (REQUIRED)
93
+ # * <tt>:password</tt> -- The Authorize.Net Transaction Key. (REQUIRED)
94
+ # * <tt>:test</tt> -- +true+ or +false+. If true, perform transactions against the test server.
95
+ # Otherwise, perform transactions against the production server.
96
+ def initialize(options = {})
97
+ requires!(options, :login, :password)
98
+ @options = options
99
+ super
100
+ end
101
+
102
+ # Creates a new customer profile along with any customer payment profiles and customer shipping addresses
103
+ # for the customer profile.
104
+ #
105
+ # Returns a Response with the Customer Profile ID of the new customer profile in the authorization field.
106
+ # It is *CRITICAL* that you save this ID. There is no way to retrieve this through the API. You will not
107
+ # be able to create another Customer Profile with the same information.
108
+ #
109
+ # ==== Options
110
+ #
111
+ # TODO
112
+ def create_customer_profile(options)
113
+ # TODO Add requires
114
+ request = build_request(:create_customer_profile, options)
115
+ commit(:create_customer_profile, request)
116
+ end
117
+
118
+ # Creates a new customer payment profile for an existing customer profile.
119
+ #
120
+ # ==== Options
121
+ #
122
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer the payment profile will be added to. (REQUIRED)
123
+ # * <tt>:payment_profile</tt> -- A hash containing the elements of the new payment profile (REQUIRED)
124
+ #
125
+ # ==== Payment Profile
126
+ #
127
+ # * <tt>:payment</tt> -- A hash containing information on payment. Either :credit_card or :bank_account (REQUIRED)
128
+ def create_customer_payment_profile(options)
129
+ requires!(options, :customer_profile_id)
130
+ requires!(options, :payment_profile)
131
+ requires!(options[:payment_profile], :payment)
132
+
133
+ request = build_request(:create_customer_payment_profile, options)
134
+ commit(:create_customer_payment_profile, request)
135
+ end
136
+
137
+ # Creates a new customer shipping address for an existing customer profile.
138
+ #
139
+ # ==== Options
140
+ #
141
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer the payment profile will be added to. (REQUIRED)
142
+ # * <tt>:address</tt> -- A hash containing the elements of the shipping address (REQUIRED)
143
+ def create_customer_shipping_address(options)
144
+ requires!(options, :customer_profile_id)
145
+ requires!(options, :address)
146
+
147
+ request = build_request(:create_customer_shipping_address, options)
148
+ commit(:create_customer_shipping_address, request)
149
+ end
150
+
151
+ # Deletes an existing customer profile along with all associated customer payment profiles and customer shipping addresses.
152
+ #
153
+ # ==== Options
154
+ #
155
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to be deleted. (REQUIRED)
156
+ def delete_customer_profile(options)
157
+ requires!(options, :customer_profile_id)
158
+
159
+ request = build_request(:delete_customer_profile, options)
160
+ commit(:delete_customer_profile, request)
161
+ end
162
+
163
+ # Deletes a customer payment profile from an existing customer profile.
164
+ #
165
+ # ==== Options
166
+ #
167
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be deleted. (REQUIRED)
168
+ # * <tt>:customer_payment_profile_id</tt> -- The Payment Profile ID of the payment profile to be deleted. (REQUIRED)
169
+ def delete_customer_payment_profile(options)
170
+ requires!(options, :customer_profile_id)
171
+ requires!(options, :customer_payment_profile_id)
172
+
173
+ request = build_request(:delete_customer_payment_profile, options)
174
+ commit(:delete_customer_payment_profile, request)
175
+ end
176
+
177
+ # Deletes a customer shipping address from an existing customer profile.
178
+ #
179
+ # ==== Options
180
+ #
181
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be deleted. (REQUIRED)
182
+ # * <tt>:customer_address_id</tt> -- The Shipping Address ID of the shipping address to be deleted. (REQUIRED)
183
+ def delete_customer_shipping_address(options)
184
+ requires!(options, :customer_profile_id)
185
+ requires!(options, :customer_address_id)
186
+
187
+ request = build_request(:delete_customer_shipping_address, options)
188
+ commit(:delete_customer_shipping_address, request)
189
+ end
190
+
191
+ # Retrieves an existing customer profile along with all the associated customer payment profiles and customer shipping addresses.
192
+ #
193
+ # Returns a Response whose params hash contains all the profile information.
194
+ #
195
+ # ==== Options
196
+ #
197
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to retrieve. (REQUIRED)
198
+ def get_customer_profile(options)
199
+ requires!(options, :customer_profile_id)
200
+
201
+ request = build_request(:get_customer_profile, options)
202
+ commit(:get_customer_profile, request)
203
+ end
204
+
205
+ # Retrieve a customer payment profile for an existing customer profile.
206
+ #
207
+ # Returns a Response whose params hash contains all the payment profile information. Sensitive information such as credit card
208
+ # numbers will be masked.
209
+ #
210
+ # ==== Options
211
+ #
212
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be retrieved. (REQUIRED)
213
+ # * <tt>:customer_payment_profile_id</tt> -- The Payment Profile ID of the payment profile to be retrieved. (REQUIRED)
214
+ def get_customer_payment_profile(options)
215
+ requires!(options, :customer_profile_id)
216
+ requires!(options, :customer_payment_profile_id)
217
+
218
+ request = build_request(:get_customer_payment_profile, options)
219
+ commit(:get_customer_payment_profile, request)
220
+ end
221
+
222
+ # Retrieve a customer shipping address for an existing customer profile.
223
+ #
224
+ # Returns a Response whose params hash contains all the shipping address information.
225
+ #
226
+ # ==== Options
227
+ #
228
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be retrieved. (REQUIRED)
229
+ # * <tt>:customer_address_id</tt> -- The Shipping Address ID of the shipping address to be retrieved. (REQUIRED)
230
+ def get_customer_shipping_address(options)
231
+ requires!(options, :customer_profile_id)
232
+ requires!(options, :customer_address_id)
233
+
234
+ request = build_request(:get_customer_shipping_address, options)
235
+ commit(:get_customer_shipping_address, request)
236
+ end
237
+
238
+ # Updates an existing customer profile.
239
+ #
240
+ # Warning: if you do not provide a parameter in the <tt>:payment_profile</tt> hash, it is automatically set to nil at
241
+ # Authorize.Net. You will most likely want to first get the profile hash using get_customer_profile and then only change the
242
+ # elements you wish to change.
243
+ #
244
+ # ==== Options
245
+ #
246
+ # * <tt>:profile</tt> -- A hash containing the values the Customer Profile should be updated to. (REQUIRED)
247
+ #
248
+ # ==== Profile
249
+ #
250
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer profile to update. (REQUIRED)
251
+ def update_customer_profile(options)
252
+ requires!(options, :profile)
253
+ requires!(options[:profile], :customer_profile_id)
254
+
255
+ request = build_request(:update_customer_profile, options)
256
+ commit(:update_customer_profile, request)
257
+ end
258
+
259
+ # Updates a customer payment profile for an existing customer profile.
260
+ #
261
+ # Warning: if you do not provide a parameter in the <tt>:payment_profile</tt> hash, it is automatically set to nil at
262
+ # Authorize.Net. You will most likely want to first get the profile hash using get_customer_payment_profile and then only
263
+ # change the elements you wish to change.
264
+ #
265
+ # ==== Options
266
+ #
267
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be updated. (REQUIRED)
268
+ # * <tt>:payment_profile</tt> -- A hash containing the values the Customer Payment Profile should be updated to. (REQUIRED)
269
+ #
270
+ # ==== Payment Profile
271
+ #
272
+ # * <tt>:customer_payment_profile_id</tt> -- The Customer Payment Profile ID of the Customer Payment Profile to update. (REQUIRED)
273
+ def update_customer_payment_profile(options)
274
+ requires!(options, :customer_profile_id, :payment_profile)
275
+ requires!(options[:payment_profile], :customer_payment_profile_id)
276
+
277
+ request = build_request(:update_customer_payment_profile, options)
278
+ commit(:update_customer_payment_profile, request)
279
+ end
280
+
281
+ # Updates a customer shipping address for an existing customer profile.
282
+ #
283
+ # Warning: if you do not provide a parameter in the <tt>:address</tt> hash, it is automatically set to nil at
284
+ # Authorize.Net. You will most likely want to first get the profile hash using get_customer_shipping_address and then only
285
+ # change the elements you wish to change.
286
+ #
287
+ # ==== Options
288
+ #
289
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer with the payment profile to be updated. (REQUIRED)
290
+ # * <tt>:address</tt> -- A hash containing the values the Customer Shipping Address should be updated to. (REQUIRED)
291
+ #
292
+ # ==== Address
293
+ #
294
+ # * <tt>:customer_address_id</tt> -- The Customer Address ID of the Customer Payment Profile to update. (REQUIRED)
295
+ def update_customer_shipping_address(options)
296
+ requires!(options, :customer_profile_id, :address)
297
+ requires!(options[:address], :customer_address_id)
298
+
299
+ request = build_request(:update_customer_shipping_address, options)
300
+ commit(:update_customer_shipping_address, request)
301
+ end
302
+
303
+ # Creates a new payment transaction from an existing customer profile
304
+ #
305
+ # This is what is used to charge a customer whose information you have stored in a Customer Profile.
306
+ #
307
+ # Returns a Response object that contains the result of the transaction in <tt>params['direct_response']</tt>
308
+ #
309
+ # ==== Options
310
+ #
311
+ # * <tt>:transaction</tt> -- A hash containing information on the transaction that is being requested. (REQUIRED)
312
+ #
313
+ # ==== Transaction
314
+ #
315
+ # * <tt>:type</tt> -- The type of transaction. Can be either <tt>:auth_only</tt>, <tt>:capture_only</tt>, <tt>:auth_capture</tt>, <tt>:prior_auth_capture</tt>, <tt>:refund</tt> or <tt>:void</tt>. (REQUIRED)
316
+ # * <tt>:amount</tt> -- The amount for the tranaction. Formatted with a decimal. For example "4.95" (CONDITIONAL)
317
+ # - :type == :void (NOT USED)
318
+ # - :type == (:refund, :auth_only, :capture_only, :auth_capture, :prior_auth_capture) (REQUIRED)
319
+ #
320
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to use in this transaction. (CONDITIONAL)
321
+ # - :type == (:void, :prior_auth_capture) (OPTIONAL)
322
+ # - :type == :refund (CONDITIONAL - required if masked information is not being submitted [see below])
323
+ # - :type == (:auth_only, :capture_only, :auth_capture) (REQUIRED)
324
+ #
325
+ # * <tt>:customer_payment_profile_id</tt> -- The Customer Payment Profile ID of the Customer Payment Profile to use in this transaction. (CONDITIONAL)
326
+ # - :type == (:void, :prior_auth_capture) (OPTIONAL)
327
+ # - :type == :refund (CONDITIONAL - required if masked information is not being submitted [see below])
328
+ # - :type == (:auth_only, :capture_only, :auth_capture) (REQUIRED)
329
+ #
330
+ # * <tt>:trans_id</tt> -- The payment gateway assigned transaction ID of the original transaction (CONDITIONAL):
331
+ # - :type = (:void, :refund, :prior_auth_capture) (REQUIRED)
332
+ # - :type = (:auth_only, :capture_only, :auth_capture) (NOT USED)
333
+ #
334
+ # * <tt>customer_shipping_address_id</tt> -- Payment gateway assigned ID associated with the customer shipping address (CONDITIONAL)
335
+ # - :type = (:void, :refund) (OPTIONAL)
336
+ # - :type = (:auth_only, :capture_only, :auth_capture) (NOT USED)
337
+ # - :type = (:prior_auth_capture) (OPTIONAL)
338
+ #
339
+ # ==== For :type == :refund only
340
+ # * <tt>:credit_card_number_masked</tt> -- (CONDITIONAL - requied for credit card refunds is :customer_profile_id AND :customer_payment_profile_id are missing)
341
+ # * <tt>:bank_routing_number_masked && :bank_account_number_masked</tt> -- (CONDITIONAL - requied for electronic check refunds is :customer_profile_id AND :customer_payment_profile_id are missing) (NOT ABLE TO TEST - I keep getting "ACH transactions are not accepted by this merchant." when trying to make a payment and, until that's possible I can't refund (wiseleyb@gmail.com))
342
+ def create_customer_profile_transaction(options)
343
+ requires!(options, :transaction)
344
+ requires!(options[:transaction], :type)
345
+ case options[:transaction][:type]
346
+ when :void
347
+ requires!(options[:transaction], :trans_id)
348
+ when :refund
349
+ requires!(options[:transaction], :trans_id) &&
350
+ (
351
+ (options[:transaction][:customer_profile_id] && options[:transaction][:customer_payment_profile_id]) ||
352
+ options[:transaction][:credit_card_number_masked] ||
353
+ (options[:transaction][:bank_routing_number_masked] && options[:transaction][:bank_account_number_masked])
354
+ )
355
+ when :prior_auth_capture
356
+ requires!(options[:transaction], :amount, :trans_id)
357
+ else
358
+ requires!(options[:transaction], :amount, :customer_profile_id, :customer_payment_profile_id)
359
+ end
360
+ request = build_request(:create_customer_profile_transaction, options)
361
+ commit(:create_customer_profile_transaction, request)
362
+ end
363
+
364
+ # Creates a new payment transaction for refund from an existing customer profile
365
+ #
366
+ # This is what is used to refund a transaction you have stored in a Customer Profile.
367
+ #
368
+ # Returns a Response object that contains the result of the transaction in <tt>params['direct_response']</tt>
369
+ #
370
+ # ==== Options
371
+ #
372
+ # * <tt>:transaction</tt> -- A hash containing information on the transaction that is being requested. (REQUIRED)
373
+ #
374
+ # ==== Transaction
375
+ #
376
+ # * <tt>:amount</tt> -- The total amount to be refunded (REQUIRED)
377
+ #
378
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to use in this transaction. (CONDITIONAL :customer_payment_profile_id must be included if used)
379
+ # * <tt>:customer_payment_profile_id</tt> -- The Customer Payment Profile ID of the Customer Payment Profile to use in this transaction. (CONDITIONAL :customer_profile_id must be included if used)
380
+ #
381
+ # * <tt>:credit_card_number_masked</tt> -- Four Xs follwed by the last four digits of the credit card (CONDITIONAL - used if customer_profile_id and customer_payment_profile_id aren't given)
382
+ #
383
+ # * <tt>:bank_routing_number_masked</tt> -- The last four gidits of the routing number to be refunded (CONDITIONAL - must be used with :bank_account_number_masked)
384
+ # * <tt>:bank_account_number_masked</tt> -- The last four digis of the bank account number to be refunded, Ex. XXXX1234 (CONDITIONAL - must be used with :bank_routing_number_masked)
385
+ def create_customer_profile_transaction_for_refund(options)
386
+ requires!(options, :transaction)
387
+ options[:transaction][:type] = :refund
388
+ requires!(options[:transaction], :trans_id)
389
+ requires!(options[:transaction], :amount)
390
+ request = build_request(:create_customer_profile_transaction, options)
391
+ commit(:create_customer_profile_transaction, request)
392
+ end
393
+
394
+ # Creates a new payment transaction for void from an existing customer profile
395
+ #
396
+ # This is what is used to void a transaction you have stored in a Customer Profile.
397
+ #
398
+ # Returns a Response object that contains the result of the transaction in <tt>params['direct_response']</tt>
399
+ #
400
+ # ==== Options
401
+ #
402
+ # * <tt>:transaction</tt> -- A hash containing information on the transaction that is being requested. (REQUIRED)
403
+ #
404
+ # ==== Transaction
405
+ #
406
+ # * <tt>:trans_id</tt> -- The payment gateway assigned transaction id of the original transaction. (REQUIRED)
407
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to use in this transaction.
408
+ # * <tt>:customer_payment_profile_id</tt> -- The Customer Payment Profile ID of the Customer Payment Profile to use in this transaction.
409
+ # * <tt>:customer_shipping_address_id</tt> -- Payment gateway assigned ID associated with the customer shipping address.
410
+ def create_customer_profile_transaction_for_void(options)
411
+ requires!(options, :transaction)
412
+ options[:transaction][:type] = :void
413
+ requires!(options[:transaction], :trans_id)
414
+ request = build_request(:create_customer_profile_transaction, options)
415
+ commit(:create_customer_profile_transaction, request)
416
+ end
417
+
418
+ # Verifies an existing customer payment profile by generating a test transaction
419
+ #
420
+ # Returns a Response object that contains the result of the transaction in <tt>params['direct_response']</tt>
421
+ #
422
+ # ==== Options
423
+ #
424
+ # * <tt>:customer_profile_id</tt> -- The Customer Profile ID of the customer to use in this transaction. (REQUIRED)
425
+ # * <tt>:customer_payment_profile_id</tt> -- The Customer Payment Profile ID of the Customer Payment Profile to be verified. (REQUIRED)
426
+ # * <tt>:customer_address_id</tt> -- The Customer Address ID of the Customer Shipping Address to be verified.
427
+ # * <tt>:validation_mode</tt> -- <tt>:live</tt> or <tt>:test</tt> In Test Mode, only field validation is performed.
428
+ # In Live Mode, a transaction is generated and submitted to the processor with the amount of $0.01. If successful, the transaction is immediately voided. (REQUIRED)
429
+ def validate_customer_payment_profile(options)
430
+ requires!(options, :customer_profile_id, :customer_payment_profile_id, :validation_mode)
431
+
432
+ request = build_request(:validate_customer_payment_profile, options)
433
+ commit(:validate_customer_payment_profile, request)
434
+ end
435
+
436
+ private
437
+
438
+ def expdate(credit_card)
439
+ sprintf('%04d-%02d', credit_card.year, credit_card.month)
440
+ end
441
+
442
+ def build_request(action, options = {})
443
+ unless CIM_ACTIONS.include?(action)
444
+ raise StandardError, "Invalid Customer Information Manager Action: #{action}"
445
+ end
446
+
447
+ xml = Builder::XmlMarkup.new(:indent => 2)
448
+ xml.instruct!(:xml, :version => '1.0', :encoding => 'utf-8')
449
+ xml.tag!("#{CIM_ACTIONS[action]}Request", :xmlns => AUTHORIZE_NET_CIM_NAMESPACE) do
450
+ add_merchant_authentication(xml)
451
+ # Merchant-assigned reference ID for the request
452
+ xml.tag!('refId', options[:ref_id]) if options[:ref_id]
453
+ send("build_#{action}_request", xml, options)
454
+ end
455
+ end
456
+
457
+ # Contains the merchant’s payment gateway account authentication information
458
+ def add_merchant_authentication(xml)
459
+ xml.tag!('merchantAuthentication') do
460
+ xml.tag!('name', @options[:login])
461
+ xml.tag!('transactionKey', @options[:password])
462
+ end
463
+ end
464
+
465
+ def build_create_customer_profile_request(xml, options)
466
+ add_profile(xml, options[:profile])
467
+
468
+ xml.tag!('validationMode',CIM_VALIDATION_MODES[options[:validation_mode]]) if options[:validation_mode]
469
+
470
+ xml.target!
471
+ end
472
+
473
+ def build_create_customer_payment_profile_request(xml, options)
474
+ xml.tag!('customerProfileId', options[:customer_profile_id])
475
+
476
+ xml.tag!('paymentProfile') do
477
+ add_payment_profile(xml, options[:payment_profile])
478
+ end
479
+
480
+ xml.tag!('validationMode', CIM_VALIDATION_MODES[options[:validation_mode]]) if options[:validation_mode]
481
+
482
+ xml.target!
483
+ end
484
+
485
+ def build_create_customer_shipping_address_request(xml, options)
486
+ xml.tag!('customerProfileId', options[:customer_profile_id])
487
+
488
+ xml.tag!('address') do
489
+ add_address(xml, options[:address])
490
+ end
491
+
492
+ xml.target!
493
+ end
494
+
495
+ def build_delete_customer_profile_request(xml, options)
496
+ xml.tag!('customerProfileId', options[:customer_profile_id])
497
+ xml.target!
498
+ end
499
+
500
+ def build_delete_customer_payment_profile_request(xml, options)
501
+ xml.tag!('customerProfileId', options[:customer_profile_id])
502
+ xml.tag!('customerPaymentProfileId', options[:customer_payment_profile_id])
503
+ xml.target!
504
+ end
505
+
506
+ def build_delete_customer_shipping_address_request(xml, options)
507
+ xml.tag!('customerProfileId', options[:customer_profile_id])
508
+ xml.tag!('customerAddressId', options[:customer_address_id])
509
+ xml.target!
510
+ end
511
+
512
+ def build_get_customer_profile_request(xml, options)
513
+ xml.tag!('customerProfileId', options[:customer_profile_id])
514
+ xml.target!
515
+ end
516
+
517
+ def build_get_customer_payment_profile_request(xml, options)
518
+ xml.tag!('customerProfileId', options[:customer_profile_id])
519
+ xml.tag!('customerPaymentProfileId', options[:customer_payment_profile_id])
520
+ xml.target!
521
+ end
522
+
523
+ def build_get_customer_shipping_address_request(xml, options)
524
+ xml.tag!('customerProfileId', options[:customer_profile_id])
525
+ xml.tag!('customerAddressId', options[:customer_address_id])
526
+ xml.target!
527
+ end
528
+
529
+ def build_update_customer_profile_request(xml, options)
530
+ add_profile(xml, options[:profile], true)
531
+
532
+ xml.target!
533
+ end
534
+
535
+ def build_update_customer_payment_profile_request(xml, options)
536
+ xml.tag!('customerProfileId', options[:customer_profile_id])
537
+
538
+ xml.tag!('paymentProfile') do
539
+ add_payment_profile(xml, options[:payment_profile])
540
+ end
541
+
542
+ xml.target!
543
+ end
544
+
545
+ def build_update_customer_shipping_address_request(xml, options)
546
+ xml.tag!('customerProfileId', options[:customer_profile_id])
547
+
548
+ xml.tag!('address') do
549
+ add_address(xml, options[:address])
550
+ end
551
+
552
+ xml.target!
553
+ end
554
+
555
+ def build_create_customer_profile_transaction_request(xml, options)
556
+ add_transaction(xml, options[:transaction])
557
+ xml.tag!('extraOptions', "x_test_request=TRUE") if @options[:test]
558
+
559
+ xml.target!
560
+ end
561
+
562
+ def build_validate_customer_payment_profile_request(xml, options)
563
+ xml.tag!('customerProfileId', options[:customer_profile_id])
564
+ xml.tag!('customerPaymentProfileId', options[:customer_payment_profile_id])
565
+ xml.tag!('customerShippingAddressId', options[:customer_address_id]) if options[:customer_address_id]
566
+ xml.tag!('validationMode', CIM_VALIDATION_MODES[options[:validation_mode]]) if options[:validation_mode]
567
+
568
+ xml.target!
569
+ end
570
+
571
+ # :merchant_customer_id (Optional)
572
+ # :description (Optional)
573
+ # :email (Optional)
574
+ # :payment_profiles (Optional)
575
+ def add_profile(xml, profile, update = false)
576
+ xml.tag!('profile') do
577
+ # Merchant assigned ID for the customer. Up to 20 characters. (optional)
578
+ xml.tag!('merchantCustomerId', profile[:merchant_customer_id]) if profile[:merchant_customer_id]
579
+ # Description of the customer. Up to 255 Characters (optional)
580
+ xml.tag!('description', profile[:description]) if profile[:description]
581
+ # Email Address for the customer. Up to 255 Characters (optional)
582
+ xml.tag!('email', profile[:email]) if profile[:email]
583
+
584
+ if update
585
+ xml.tag!('customerProfileId', profile[:customer_profile_id])
586
+ else
587
+ add_payment_profiles(xml, profile[:payment_profiles]) if profile[:payment_profiles]
588
+ add_ship_to_list(xml, profile[:ship_to_list]) if profile[:ship_to_list]
589
+ end
590
+ end
591
+ end
592
+
593
+ def add_transaction(xml, transaction)
594
+ unless CIM_TRANSACTION_TYPES.include?(transaction[:type])
595
+ raise StandardError, "Invalid Customer Information Manager Transaction Type: #{transaction[:type]}"
596
+ end
597
+
598
+ xml.tag!('transaction') do
599
+ xml.tag!(CIM_TRANSACTION_TYPES[transaction[:type]]) do
600
+ # The amount to be billed to the customer
601
+ case transaction[:type]
602
+ when :void
603
+ tag_unless_blank(xml,'customerProfileId', transaction[:customer_profile_id])
604
+ tag_unless_blank(xml,'customerPaymentProfileId', transaction[:customer_payment_profile_id])
605
+ tag_unless_blank(xml,'customerShippingAddressId', transaction[:customer_shipping_address_id])
606
+ xml.tag!('transId', transaction[:trans_id])
607
+ when :refund
608
+ #TODO - add support for all the other options fields
609
+ xml.tag!('amount', transaction[:amount])
610
+ tag_unless_blank(xml, 'customerProfileId', transaction[:customer_profile_id])
611
+ tag_unless_blank(xml, 'customerPaymentProfileId', transaction[:customer_payment_profile_id])
612
+ tag_unless_blank(xml, 'customerShippingAddressId', transaction[:customer_shipping_address_id])
613
+ tag_unless_blank(xml, 'creditCardNumberMasked', transaction[:credit_card_number_masked])
614
+ tag_unless_blank(xml, 'bankRoutingNumberMasked', transaction[:bank_routing_number_masked])
615
+ tag_unless_blank(xml, 'bankAccountNumberMasked', transaction[:bank_account_number_masked])
616
+ xml.tag!('transId', transaction[:trans_id])
617
+ when :prior_auth_capture
618
+ xml.tag!('amount', transaction[:amount])
619
+ xml.tag!('transId', transaction[:trans_id])
620
+ else
621
+ xml.tag!('amount', transaction[:amount])
622
+ xml.tag!('customerProfileId', transaction[:customer_profile_id])
623
+ xml.tag!('customerPaymentProfileId', transaction[:customer_payment_profile_id])
624
+ xml.tag!('approvalCode', transaction[:approval_code]) if transaction[:type] == :capture_only
625
+ end
626
+ add_order(xml, transaction[:order]) if transaction[:order]
627
+ end
628
+ end
629
+ end
630
+
631
+ def add_order(xml, order)
632
+ xml.tag!('order') do
633
+ xml.tag!('invoiceNumber', order[:invoice_number]) if order[:invoice_number]
634
+ xml.tag!('description', order[:description]) if order[:description]
635
+ xml.tag!('purchaseOrderNumber', order[:purchase_order_number]) if order[:purchase_order_number]
636
+ end
637
+ end
638
+
639
+ def add_payment_profiles(xml, payment_profiles)
640
+ xml.tag!('paymentProfiles') do
641
+ add_payment_profile(xml, payment_profiles)
642
+ end
643
+ end
644
+
645
+ # :customer_type => 'individual or business', # Optional
646
+ # :bill_to => @address,
647
+ # :payment => @payment
648
+ def add_payment_profile(xml, payment_profile)
649
+ # 'individual' or 'business' (optional)
650
+ xml.tag!('customerType', payment_profile[:customer_type]) if payment_profile[:customer_type]
651
+
652
+ if payment_profile[:bill_to]
653
+ xml.tag!('billTo') do
654
+ add_address(xml, payment_profile[:bill_to])
655
+ end
656
+ end
657
+
658
+ if payment_profile[:payment]
659
+ xml.tag!('payment') do
660
+ add_credit_card(xml, payment_profile[:payment][:credit_card]) if payment_profile[:payment].has_key?(:credit_card)
661
+ add_bank_account(xml, payment_profile[:payment][:bank_account]) if payment_profile[:payment].has_key?(:bank_account)
662
+ add_drivers_license(xml, payment_profile[:payment][:drivers_license]) if payment_profile[:payment].has_key?(:drivers_license)
663
+ # This element is only required for Wells Fargo SecureSource eCheck.Net merchants
664
+ # The customer's Social Security Number or Tax ID
665
+ xml.tag!('taxId', payment_profile[:payment]) if payment_profile[:payment].has_key?(:tax_id)
666
+ end
667
+ end
668
+
669
+ xml.tag!('customerPaymentProfileId', payment_profile[:customer_payment_profile_id]) if payment_profile[:customer_payment_profile_id]
670
+ end
671
+
672
+ def add_ship_to_list(xml, ship_to_list)
673
+ xml.tag!('shipToList') do
674
+ add_address(xml, ship_to_list)
675
+ end
676
+ end
677
+
678
+ def add_address(xml, address)
679
+ xml.tag!('firstName', address[:first_name])
680
+ xml.tag!('lastName', address[:last_name])
681
+ xml.tag!('company', address[:company])
682
+ xml.tag!('address', address[:address1]) if address[:address1]
683
+ xml.tag!('address', address[:address]) if address[:address]
684
+ xml.tag!('city', address[:city])
685
+ xml.tag!('state', address[:state])
686
+ xml.tag!('zip', address[:zip])
687
+ xml.tag!('country', address[:country])
688
+ xml.tag!('phoneNumber', address[:phone_number]) if address[:phone_number]
689
+ xml.tag!('faxNumber', address[:fax_number]) if address[:fax_number]
690
+
691
+ xml.tag!('customerAddressId', address[:customer_address_id]) if address[:customer_address_id]
692
+ end
693
+
694
+ # Adds customer’s credit card information
695
+ # Note: This element should only be included
696
+ # when the payment method is credit card.
697
+ def add_credit_card(xml, credit_card)
698
+ return unless credit_card
699
+ xml.tag!('creditCard') do
700
+ # The credit card number used for payment of the subscription
701
+ xml.tag!('cardNumber', credit_card.number)
702
+ # The expiration date of the credit card used for the subscription
703
+ xml.tag!('expirationDate', expdate(credit_card))
704
+ xml.tag!('cardCode', credit_card.verification_value) if credit_card.verification_value?
705
+ end
706
+ end
707
+
708
+ # Adds customer’s bank account information
709
+ # Note: This element should only be included
710
+ # when the payment method is bank account.
711
+ def add_bank_account(xml, bank_account)
712
+ raise StandardError, "Invalid Bank Account Type: #{bank_account[:account_type]}" unless BANK_ACCOUNT_TYPES.include?(bank_account[:account_type])
713
+ raise StandardError, "Invalid eCheck Type: #{bank_account[:echeck_type]}" unless ECHECK_TYPES.include?(bank_account[:echeck_type])
714
+
715
+ xml.tag!('bankAccount') do
716
+ # The type of bank account
717
+ xml.tag!('accountType', BANK_ACCOUNT_TYPES[bank_account[:account_type]])
718
+ # The routing number of the customer’s bank
719
+ xml.tag!('routingNumber', bank_account[:routing_number])
720
+ # The bank account number
721
+ xml.tag!('accountNumber', bank_account[:account_number])
722
+ # The full name of the individual associated
723
+ # with the bank account number
724
+ xml.tag!('nameOnAccount', bank_account[:name_on_account])
725
+ # The type of electronic check transaction
726
+ xml.tag!('echeckType', ECHECK_TYPES[bank_account[:echeck_type]])
727
+ # The full name of the individual associated
728
+ # with the bank account number (optional)
729
+ xml.tag!('bankName', bank_account[:bank_name]) if bank_account[:bank_name]
730
+ end
731
+ end
732
+
733
+ # Adds customer’s driver's license information
734
+ # Note: This element is only required for
735
+ # Wells Fargo SecureSource eCheck.Net merchants
736
+ def add_drivers_license(xml, drivers_license)
737
+ xml.tag!('driversLicense') do
738
+ # The state of the customer's driver's license
739
+ # A valid two character state code
740
+ xml.tag!('state', drivers_license[:state])
741
+ # The customer’s driver's license number
742
+ xml.tag!('number', drivers_license[:number])
743
+ # The date of birth listed on the customer's driver's license
744
+ # YYYY-MM-DD
745
+ xml.tag!('dateOfBirth', drivers_license[:date_of_birth])
746
+ end
747
+ end
748
+
749
+ def commit(action, request)
750
+ url = test? ? test_url : live_url
751
+ xml = ssl_post(url, request, "Content-Type" => "text/xml")
752
+ response_params = parse(action, xml)
753
+ message = response_params['messages']['message']
754
+
755
+ if message.class.is_a? Array
756
+ text = "#{message[0]['code']} - #{message[0]['text']}"
757
+ else
758
+ text = "#{message['code']} - #{message['text']}"
759
+ end
760
+ test_mode = test? || text =~ /Test Mode/
761
+ success = response_params['messages']['result_code'] == 'Ok'
762
+
763
+ Rails.logger.error("Gateway Error: #{text}") unless success
764
+
765
+ direct_response_params = parse_direct_response(response_params['direct_response'])
766
+
767
+ response = Response.new(success, text, response_params,
768
+ :test => test_mode,
769
+ :authorization => response_params['customer_profile_id'] || (response_params['profile'] ? response_params['profile']['customer_profile_id'] : direct_response_params['transaction_id']),
770
+ :avs_result => {:code => direct_response_params['avs_response']}
771
+ )
772
+ response.params['direct_response'] = direct_response_params
773
+
774
+ # transaction_id from direct response should be used for the authorization option on response
775
+ if response.authorization.nil? and response.params['direct_response']
776
+ if !response.params['direct_response']['transaction_id'].blank? and response.params['direct_response']['transaction_id'] != '0'
777
+ response.authorization = response.params['direct_response']['transaction_id']
778
+ end
779
+ end
780
+
781
+ response
782
+ end
783
+
784
+ def tag_unless_blank(xml, tag_name, data)
785
+ xml.tag!(tag_name, data) unless data.blank? || data.nil?
786
+ end
787
+
788
+ def parse_direct_response(raw)
789
+ return {} if raw.blank?
790
+ direct_response_fields = raw.split(',')
791
+ {
792
+ 'raw' => raw,
793
+ 'response_code' => direct_response_fields[0],
794
+ 'response_subcode' => direct_response_fields[1],
795
+ 'response_reason_code' => direct_response_fields[2],
796
+ 'message' => direct_response_fields[3],
797
+ 'approval_code' => direct_response_fields[4],
798
+ 'avs_response' => direct_response_fields[5],
799
+ 'transaction_id' => direct_response_fields[6],
800
+ 'invoice_number' => direct_response_fields[7],
801
+ 'order_description' => direct_response_fields[8],
802
+ 'amount' => direct_response_fields[9],
803
+ 'method' => direct_response_fields[10],
804
+ 'transaction_type' => direct_response_fields[11],
805
+ 'customer_id' => direct_response_fields[12],
806
+ 'first_name' => direct_response_fields[13],
807
+ 'last_name' => direct_response_fields[14],
808
+ 'company' => direct_response_fields[15],
809
+ 'address' => direct_response_fields[16],
810
+ 'city' => direct_response_fields[17],
811
+ 'state' => direct_response_fields[18],
812
+ 'zip_code' => direct_response_fields[19],
813
+ 'country' => direct_response_fields[20],
814
+ 'phone' => direct_response_fields[21],
815
+ 'fax' => direct_response_fields[22],
816
+ 'email_address' => direct_response_fields[23],
817
+ 'ship_to_first_name' => direct_response_fields[24],
818
+ 'ship_to_last_name' => direct_response_fields[25],
819
+ 'ship_to_company' => direct_response_fields[26],
820
+ 'ship_to_address' => direct_response_fields[27],
821
+ 'ship_to_city' => direct_response_fields[28],
822
+ 'ship_to_state' => direct_response_fields[29],
823
+ 'ship_to_zip_code' => direct_response_fields[30],
824
+ 'ship_to_country' => direct_response_fields[31],
825
+ 'tax' => direct_response_fields[32],
826
+ 'duty' => direct_response_fields[33],
827
+ 'freight' => direct_response_fields[34],
828
+ 'tax_exempt' => direct_response_fields[35],
829
+ 'purchase_order_number' => direct_response_fields[36],
830
+ 'md5_hash' => direct_response_fields[37],
831
+ 'card_code' => direct_response_fields[38],
832
+ 'cardholder_authentication_verification_response' => direct_response_fields[39]
833
+ }
834
+ end
835
+
836
+ def parse(action, xml)
837
+ xml = REXML::Document.new(xml)
838
+ root = REXML::XPath.first(xml, "//#{CIM_ACTIONS[action]}Response") ||
839
+ REXML::XPath.first(xml, "//ErrorResponse")
840
+ if root
841
+ response = parse_element(root)
842
+ end
843
+
844
+ response
845
+ end
846
+
847
+ def parse_element(node)
848
+ if node.has_elements?
849
+ response = {}
850
+ node.elements.each{ |e|
851
+ key = e.name.underscore
852
+ value = parse_element(e)
853
+ if response.has_key?(key)
854
+ if response[key].is_a?(Array)
855
+ response[key].push(value)
856
+ else
857
+ response[key] = [response[key], value]
858
+ end
859
+ else
860
+ response[key] = parse_element(e)
861
+ end
862
+ }
863
+ else
864
+ response = node.text
865
+ end
866
+
867
+ response
868
+ end
869
+ end
870
+ end
871
+ end