tax_cloud 0.3.0 → 0.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +5 -5
- data/.github/workflows/test.yml +30 -0
- data/.rubocop.yml +9 -25
- data/.rubocop_todo.yml +47 -0
- data/CHANGELOG.md +85 -0
- data/CONTRIBUTORS.txt +3 -0
- data/Gemfile +15 -5
- data/LICENSE.md +21 -0
- data/README.md +200 -0
- data/Rakefile +5 -19
- data/lib/config/locales/en.yml +2 -2
- data/lib/hash.rb +3 -4
- data/lib/tasks/tax_cloud.rake +4 -4
- data/lib/tasks/tax_code_groups.rake +11 -11
- data/lib/tasks/tax_codes.rake +12 -12
- data/lib/tax_cloud/address.rb +10 -5
- data/lib/tax_cloud/cart_item.rb +2 -0
- data/lib/tax_cloud/client.rb +13 -3
- data/lib/tax_cloud/configuration.rb +13 -2
- data/lib/tax_cloud/errors/api_error.rb +3 -3
- data/lib/tax_cloud/errors/missing_config_error.rb +2 -1
- data/lib/tax_cloud/errors/missing_config_option_error.rb +2 -1
- data/lib/tax_cloud/errors/soap_error.rb +6 -6
- data/lib/tax_cloud/errors/tax_cloud_error.rb +5 -4
- data/lib/tax_cloud/errors/unexpected_soap_response_error.rb +3 -3
- data/lib/tax_cloud/errors.rb +2 -0
- data/lib/tax_cloud/record.rb +2 -0
- data/lib/tax_cloud/responses/authorized.rb +3 -1
- data/lib/tax_cloud/responses/authorized_with_capture.rb +3 -1
- data/lib/tax_cloud/responses/base.rb +5 -4
- data/lib/tax_cloud/responses/captured.rb +3 -1
- data/lib/tax_cloud/responses/cart_item.rb +6 -2
- data/lib/tax_cloud/responses/generic.rb +2 -0
- data/lib/tax_cloud/responses/lookup.rb +10 -8
- data/lib/tax_cloud/responses/ping.rb +3 -1
- data/lib/tax_cloud/responses/returned.rb +3 -1
- data/lib/tax_cloud/responses/tax_code_groups.rb +3 -1
- data/lib/tax_cloud/responses/tax_codes.rb +3 -1
- data/lib/tax_cloud/responses/tax_codes_by_group.rb +3 -1
- data/lib/tax_cloud/responses/verify_address.rb +3 -1
- data/lib/tax_cloud/responses.rb +2 -0
- data/lib/tax_cloud/tax_code.rb +2 -0
- data/lib/tax_cloud/tax_code_constants.rb +262 -260
- data/lib/tax_cloud/tax_code_group.rb +3 -1
- data/lib/tax_cloud/tax_code_group_constants.rb +2 -0
- data/lib/tax_cloud/tax_code_groups.rb +3 -1
- data/lib/tax_cloud/tax_codes.rb +3 -1
- data/lib/tax_cloud/transaction.rb +2 -0
- data/lib/tax_cloud/version.rb +3 -1
- data/lib/tax_cloud.rb +8 -3
- data/tax_cloud.gemspec +17 -18
- data/test/cassettes/authorized.yml +6 -6
- data/test/cassettes/authorized_with_capture.yml +6 -6
- data/test/cassettes/authorized_with_localized_time.yml +6 -6
- data/test/cassettes/captured.yml +7 -7
- data/test/cassettes/get_tic_groups.yml +5 -5
- data/test/cassettes/get_tics.yml +5 -5
- data/test/cassettes/get_tics_by_group.yml +5 -5
- data/test/cassettes/invalid_soap_call.yml +5 -5
- data/test/cassettes/lookup.yml +5 -5
- data/test/cassettes/lookup_ny.yml +5 -5
- data/test/cassettes/ping.yml +5 -5
- data/test/cassettes/ping_with_invalid_credentials.yml +5 -5
- data/test/cassettes/ping_with_invalid_response.yml +5 -5
- data/test/cassettes/returned.yml +7 -7
- data/test/cassettes/verify_bad_address.yml +5 -5
- data/test/cassettes/verify_good_address.yml +5 -5
- data/test/helper.rb +3 -1
- data/test/test_address.rb +7 -5
- data/test/test_cart_item.rb +3 -1
- data/test/test_client.rb +4 -2
- data/test/test_configuration_optional_keys.rb +44 -0
- data/test/{test_configuration.rb → test_configuration_required_keys.rb} +6 -4
- data/test/test_lookup_response.rb +22 -0
- data/test/test_setup.rb +3 -1
- data/test/test_soap.rb +3 -1
- data/test/test_tax_code_groups.rb +4 -2
- data/test/test_tax_codes.rb +3 -1
- data/test/test_transaction.rb +3 -1
- data/test/test_transaction_ny.rb +3 -1
- data/test/vcr_setup.rb +2 -0
- metadata +37 -91
- data/.travis.yml +0 -6
- data/CHANGELOG.rdoc +0 -42
- data/LICENSE.rdoc +0 -22
- data/README.rdoc +0 -140
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
|
-
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 55a0579c5212968283b84a2e6a87921e8d741d182e4f23337d480435c26dbec6
|
4
|
+
data.tar.gz: 7cb858801bc88041e2b402962c5dc05d91afb6b52dc3fe793c4a943014b7d1bc
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: efc17316081884f02aebf3d1064e81df78515e5b8ee9355c224d8be53ef7529833648295c692ab332d35018092c940b2c21d2356450fa67761db7cd563b472ca
|
7
|
+
data.tar.gz: b256cb07f74161ad04c869c258e75549fbed0920c2ce3a030e60a126518df9d32111df50afaba455b69682a31fc64f8cf30960b0c9e6c660da64f4b3c72fc994
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# Reference https://github.com/actions/starter-workflows/blob/main/ci/ruby.yml
|
2
|
+
name: test
|
3
|
+
on: [push, pull_request]
|
4
|
+
jobs:
|
5
|
+
lint:
|
6
|
+
name: RuboCop
|
7
|
+
runs-on: ubuntu-latest
|
8
|
+
steps:
|
9
|
+
- uses: actions/checkout@v3
|
10
|
+
- name: Set up Ruby
|
11
|
+
uses: ruby/setup-ruby@v1
|
12
|
+
with:
|
13
|
+
ruby-version: 2.7
|
14
|
+
bundler-cache: true
|
15
|
+
- name: Run RuboCop
|
16
|
+
run: bundle exec rubocop
|
17
|
+
test:
|
18
|
+
runs-on: ubuntu-latest
|
19
|
+
strategy:
|
20
|
+
matrix:
|
21
|
+
ruby-version: ['2.6', '2.7']
|
22
|
+
name: test (${{ matrix.ruby-version }})
|
23
|
+
steps:
|
24
|
+
- uses: actions/checkout@v3
|
25
|
+
- uses: ruby/setup-ruby@v1
|
26
|
+
with:
|
27
|
+
ruby-version: ${{ matrix.ruby-version }}
|
28
|
+
bundler-cache: true # runs 'bundle install' and caches installed gems automatically
|
29
|
+
- run: bundle exec rake test
|
30
|
+
|
data/.rubocop.yml
CHANGED
@@ -1,29 +1,13 @@
|
|
1
|
+
inherit_from: .rubocop_todo.yml
|
2
|
+
|
1
3
|
AllCops:
|
2
|
-
|
4
|
+
TargetRubyVersion: 2.6
|
5
|
+
NewCops: enable
|
6
|
+
Exclude:
|
3
7
|
- vendor/**
|
8
|
+
- vendor/bundle/**/*
|
4
9
|
- bin/**
|
5
10
|
|
6
|
-
|
7
|
-
Enabled:
|
8
|
-
|
9
|
-
MethodLength:
|
10
|
-
Enabled: false
|
11
|
-
|
12
|
-
ClassLength:
|
13
|
-
Enabled: false
|
14
|
-
|
15
|
-
Documentation:
|
16
|
-
# don't require classes to be documented
|
17
|
-
Enabled: false
|
18
|
-
|
19
|
-
NumericLiterals:
|
20
|
-
# don't separate numeric literals with _
|
21
|
-
Enabled: false
|
22
|
-
|
23
|
-
RaiseArgs:
|
24
|
-
# don't need to provide an exception class and message, SoapError takes care of that
|
25
|
-
Enabled: false
|
26
|
-
|
27
|
-
Encoding:
|
28
|
-
# missing UTF-8 comment
|
29
|
-
Enabled: false
|
11
|
+
Style/NumericLiteralPrefix:
|
12
|
+
Enabled: true
|
13
|
+
EnforcedOctalStyle: zero_only
|
data/.rubocop_todo.yml
ADDED
@@ -0,0 +1,47 @@
|
|
1
|
+
# This configuration was generated by
|
2
|
+
# `rubocop --auto-gen-config`
|
3
|
+
# on 2020-10-14 20:47:01 UTC using RuboCop version 0.93.1.
|
4
|
+
# The point is for the user to remove these configuration records
|
5
|
+
# one by one as the offenses are removed from the code base.
|
6
|
+
# Note that changes in the inspected code, or installation of new
|
7
|
+
# versions of RuboCop, may require this file to be generated again.
|
8
|
+
|
9
|
+
# Offense count: 1
|
10
|
+
Lint/MissingSuper:
|
11
|
+
Exclude:
|
12
|
+
- 'test/test_lookup_response.rb'
|
13
|
+
|
14
|
+
# Offense count: 3
|
15
|
+
# Configuration parameters: IgnoredMethods.
|
16
|
+
Metrics/AbcSize:
|
17
|
+
Max: 23
|
18
|
+
|
19
|
+
# Offense count: 5
|
20
|
+
# Configuration parameters: CountComments, CountAsOne, ExcludedMethods.
|
21
|
+
# ExcludedMethods: refine
|
22
|
+
Metrics/BlockLength:
|
23
|
+
Max: 39
|
24
|
+
|
25
|
+
# Offense count: 1
|
26
|
+
# Configuration parameters: CountComments, CountAsOne.
|
27
|
+
Metrics/ClassLength:
|
28
|
+
Max: 261
|
29
|
+
|
30
|
+
# Offense count: 2
|
31
|
+
# Configuration parameters: CountComments, CountAsOne, ExcludedMethods.
|
32
|
+
Metrics/MethodLength:
|
33
|
+
Max: 14
|
34
|
+
|
35
|
+
# Offense count: 1
|
36
|
+
Style/Documentation:
|
37
|
+
Exclude:
|
38
|
+
- 'spec/**/*'
|
39
|
+
- 'test/**/*'
|
40
|
+
- 'lib/tax_cloud/tax_codes.rb'
|
41
|
+
|
42
|
+
# Offense count: 61
|
43
|
+
# Cop supports --auto-correct.
|
44
|
+
# Configuration parameters: AutoCorrect, AllowHeredoc, AllowURI, URISchemes, IgnoreCopDirectives, IgnoredPatterns.
|
45
|
+
# URISchemes: http, https
|
46
|
+
Layout/LineLength:
|
47
|
+
Max: 235
|
data/CHANGELOG.md
ADDED
@@ -0,0 +1,85 @@
|
|
1
|
+
### 0.5.0 (7/21/2023)
|
2
|
+
|
3
|
+
* Update README, CHANGELOG, and LICENSE from RDoc to Markdown -
|
4
|
+
@brchristian.
|
5
|
+
* Migrate from Travis CI to GitHub Actions -
|
6
|
+
@rchekaluk.
|
7
|
+
* Update WSDL endpoint URL as recommended by TaxCloud to
|
8
|
+
https://api.taxcloud.com/1.0/?wsdl - @mrmarcondes.
|
9
|
+
|
10
|
+
|
11
|
+
### 0.4.0 (12/7/2020)
|
12
|
+
|
13
|
+
* Add `open_timeout` and `read_timeout` options into configuration -
|
14
|
+
@ka8725.
|
15
|
+
* Switch to using BigDecimal for `tax_amount` to avoid floating point
|
16
|
+
arithmetic errors - @paulhenrich.
|
17
|
+
* Require ruby 2.6.0+
|
18
|
+
* Upgrade to rubocop latest
|
19
|
+
* Add `TargetRubyVersion: 2.6` to rubocop config (based on
|
20
|
+
https://www.ruby-lang.org/en/downloads/branches/ oldest "normal
|
21
|
+
maintenance" version)
|
22
|
+
* Add `required_ruby_version` to gemspec
|
23
|
+
* addresses rubocop violations
|
24
|
+
|
25
|
+
* Update WSDL endpoint URL as recommended by TaxCloud to
|
26
|
+
https://asmx.taxcloud.com/1.0/?wsdl - @brchristian.
|
27
|
+
|
28
|
+
|
29
|
+
### 0.3.0 (1/9/2014)
|
30
|
+
|
31
|
+
* #19: Support Savon 2 - @drewtempelmeyer.
|
32
|
+
* Implemented Rubocop, Ruby style linter - @dblock.
|
33
|
+
* Ruby 1.8.7 and 1.9.2 are no longer supported - @dblock.
|
34
|
+
|
35
|
+
|
36
|
+
### 0.2.2 (4/29/2013)
|
37
|
+
|
38
|
+
* Relaxed thirdparty gem dependency versions - @mperham.
|
39
|
+
|
40
|
+
|
41
|
+
### 0.2.1 (3/3/2013)
|
42
|
+
|
43
|
+
* Fixed date formatting in API requests with localized applications -
|
44
|
+
@soulcutter.
|
45
|
+
|
46
|
+
|
47
|
+
### 0.2.0 (11/26/2012)
|
48
|
+
|
49
|
+
* The gem is now licensed under the MIT license - @dblock.
|
50
|
+
* Raise specialized configuration and SOAP errors - @dblock.
|
51
|
+
* Added `TaxCloud::Client.ping` - @dblock.
|
52
|
+
* Returning a verified `TaxCloud::Address` from `TaxCloud::Address.verify` -
|
53
|
+
@dblock.
|
54
|
+
* Added `TaxCloud::Address.zip` that returns a 9-digit address zip code,
|
55
|
+
when available - @dblock.
|
56
|
+
* Returning a `TaxCloud::Responses::Lookup` from
|
57
|
+
`TaxCloud::Transaction.lookup` - @dblock.
|
58
|
+
* Returning transaction state, ie. `"OK"` from all other TaxCloud
|
59
|
+
transactions. Exceptions are raised on error - @dblock.
|
60
|
+
* Added support for tax codes and tax code groups lookup with
|
61
|
+
`TaxCloud::TaxCodes` and `TaxCloud::TaxCode::Groups` - @dblock.
|
62
|
+
|
63
|
+
|
64
|
+
### 0.1.5 (5/9/2012)
|
65
|
+
|
66
|
+
* Fixed compatibility with Ruby 1.8.7 and 1.9.3, removed super from
|
67
|
+
constructors for classes which inherit from `Object` - @gfmurphy.
|
68
|
+
|
69
|
+
|
70
|
+
### 0.1.4 (10/20/2011)
|
71
|
+
|
72
|
+
* Upgraded rdoc - @drewtempelmeyer.
|
73
|
+
* Fixed `.gemspec` dependency declaration for savon - @drewtempelmeyer.
|
74
|
+
|
75
|
+
|
76
|
+
### 0.1.3 (9/19/2011)
|
77
|
+
|
78
|
+
* Added support for the "returned" request - @danielmorrison.
|
79
|
+
* Refactoreted tests to use vcr and webmock - @danielmorrison.
|
80
|
+
|
81
|
+
|
82
|
+
### 0.1.0 (8/23/2011)
|
83
|
+
|
84
|
+
* Initial public release - @drewtempelmeyer.
|
85
|
+
|
data/CONTRIBUTORS.txt
CHANGED
data/Gemfile
CHANGED
@@ -1,9 +1,19 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
source 'https://rubygems.org'
|
2
4
|
|
3
5
|
gemspec
|
4
6
|
|
5
|
-
|
6
|
-
gem '
|
7
|
-
|
8
|
-
|
7
|
+
group :development do
|
8
|
+
gem 'rubocop', '~> 0.93.1'
|
9
|
+
end
|
10
|
+
|
11
|
+
group :test do
|
12
|
+
gem 'minitest'
|
13
|
+
gem 'vcr'
|
14
|
+
gem 'webmock'
|
15
|
+
end
|
16
|
+
|
17
|
+
group :development, :test do
|
18
|
+
gem 'rake', '~>12.2.1'
|
9
19
|
end
|
data/LICENSE.md
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
MIT License
|
2
|
+
|
3
|
+
Copyright (c) 2011-2020 Drew Tempelmeyer & Contributors
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
13
|
+
copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
21
|
+
SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,200 @@
|
|
1
|
+
# TaxCloud
|
2
|
+
|
3
|
+
[](https://github.com/txcrb/tax_cloud/actions)
|
4
|
+
|
5
|
+
[TaxCloud](http://www.taxcloud.com) is a free service to calculate sales tax
|
6
|
+
and generate tax reports. The `tax_cloud` gem allows you to easily integrate
|
7
|
+
with TaxCloud's API.
|
8
|
+
|
9
|
+
### Getting Started
|
10
|
+
Create a [TaxCloud](http://www.taxcloud.com) merchant account at
|
11
|
+
http://www.taxcloud.net. Add a website to your account under
|
12
|
+
[Locations](https://taxcloud.net/account/locations). This will generate an API
|
13
|
+
ID and API Key that you will need to use the service.
|
14
|
+
|
15
|
+
[TaxCloud](http://www.taxcloud.com) also offers an optional address
|
16
|
+
verification API. To use it, you need a USPS (United States Postal Service)
|
17
|
+
Address API Username. To obtain your USPS Username:
|
18
|
+
1. Go through the Web Tools API Portal registration process at
|
19
|
+
https://registration.shippingapis.com/
|
20
|
+
2. Once you have registered for your account, you will receive an email with
|
21
|
+
your USPS username and password.
|
22
|
+
3. Enter your USPS username in your TaxCloud initializer (see below).
|
23
|
+
|
24
|
+
|
25
|
+
### Setup
|
26
|
+
Add the gem to your Gemfile.
|
27
|
+
|
28
|
+
```ruby
|
29
|
+
gem 'tax_cloud'
|
30
|
+
```
|
31
|
+
|
32
|
+
Configure your environment. For example, create an initializer in Rails in
|
33
|
+
`config/initializers/tax_cloud.rb`.
|
34
|
+
|
35
|
+
```ruby
|
36
|
+
TaxCloud.configure do |config|
|
37
|
+
config.api_login_id = 'your_tax_cloud_api_login_id'
|
38
|
+
config.api_key = 'your_tax_cloud_api_key'
|
39
|
+
config.usps_username = 'your_usps_username' # optional
|
40
|
+
config.open_timeout = 1 # optional
|
41
|
+
config.read_timeout = 1 # optional
|
42
|
+
end
|
43
|
+
```
|
44
|
+
|
45
|
+
The `open_timeout` and `read_timeout` options are used to specify waiting time
|
46
|
+
for the TaxCloud web service response in seconds. Default values:
|
47
|
+
`open_timeout = 2` and `read_timeout = 2`.
|
48
|
+
|
49
|
+
### Using TaxCloud
|
50
|
+
Define the destination and origin addresses using `TaxCloud::Address`.
|
51
|
+
|
52
|
+
```ruby
|
53
|
+
origin = TaxCloud::Address.new(
|
54
|
+
address1: '162 East Avenue',
|
55
|
+
address2: 'Third Floor',
|
56
|
+
city: 'Norwalk',
|
57
|
+
state: 'CT',
|
58
|
+
zip5: '06851'
|
59
|
+
)
|
60
|
+
destination = TaxCloud::Address.new(
|
61
|
+
address1: '3121 West Government Way',
|
62
|
+
address2: 'Suite 2B',
|
63
|
+
city: 'Seattle',
|
64
|
+
state: 'WA',
|
65
|
+
zip5: '98199'
|
66
|
+
)
|
67
|
+
```
|
68
|
+
|
69
|
+
Create your Transaction and set up your cart items
|
70
|
+
|
71
|
+
```ruby
|
72
|
+
transaction = TaxCloud::Transaction.new(
|
73
|
+
customer_id: '1',
|
74
|
+
cart_id: '1',
|
75
|
+
origin: origin,
|
76
|
+
destination: destination
|
77
|
+
)
|
78
|
+
transaction.cart_items << TaxCloud::CartItem.new(
|
79
|
+
index: 0,
|
80
|
+
item_id: 'SKU-100',
|
81
|
+
tic: TaxCloud::TaxCodes::GENERAL,
|
82
|
+
price: 10.00,
|
83
|
+
quantity: 1
|
84
|
+
)
|
85
|
+
lookup = transaction.lookup # this will return a TaxCloud::Responses::Lookup instance
|
86
|
+
lookup.tax_amount # total tax amount
|
87
|
+
lookup.cart_items.each do |cart_item|
|
88
|
+
cart_item.tax_amount # tax for a single item
|
89
|
+
end
|
90
|
+
```
|
91
|
+
|
92
|
+
After you've authorized and captured the transaction via your merchant
|
93
|
+
account, you should do the same with TaxCloud for maintaining accurate tax
|
94
|
+
information.
|
95
|
+
|
96
|
+
```ruby
|
97
|
+
transaction.order_id = 100
|
98
|
+
transaction.authorized_with_capture # returns "OK" or raises an error
|
99
|
+
```
|
100
|
+
|
101
|
+
Later, you may need to mark some cart items as returned. TaxCloud will ignore
|
102
|
+
any cart items that you don't include.
|
103
|
+
|
104
|
+
```ruby
|
105
|
+
transaction.order_id = 100
|
106
|
+
transaction.cart_items << TaxCloud::CartItem.new(
|
107
|
+
index: 0,
|
108
|
+
item_id: 'SKU-100',
|
109
|
+
tic: TaxCloud::TaxCodes::GENERAL,
|
110
|
+
price: 10.00,
|
111
|
+
quantity: 1
|
112
|
+
)
|
113
|
+
transaction.returned # returns "OK" or raises an error
|
114
|
+
```
|
115
|
+
|
116
|
+
### Verifying Addresses
|
117
|
+
|
118
|
+
[TaxCloud](http://www.taxcloud.com) optionally integrates with the USPS
|
119
|
+
Address API. An address can be verified, which can also yield a 9-digit zip
|
120
|
+
code that helps determine a more accurate tax rate.
|
121
|
+
|
122
|
+
```ruby
|
123
|
+
address = TaxCloud::Address.new({
|
124
|
+
address1: '888 6th Ave',
|
125
|
+
city: 'New York',
|
126
|
+
state: 'New York',
|
127
|
+
zip5: '10001'
|
128
|
+
})
|
129
|
+
|
130
|
+
verified_address = address.verify
|
131
|
+
verified_address.zip5 # 10001
|
132
|
+
verified_address.zip4 # 3502
|
133
|
+
verified_address.zip # 10001-3502
|
134
|
+
```
|
135
|
+
|
136
|
+
### Tax Codes
|
137
|
+
[TaxCloud](http://www.taxcloud.com) maintains a list of all Taxability
|
138
|
+
Information Codes or TICs, which can be found at https://taxcloud.net/tic.
|
139
|
+
|
140
|
+
You can obtain all tax codes as well as lookup a tax code by ID.
|
141
|
+
|
142
|
+
```ruby
|
143
|
+
TaxCloud::TaxCodes.all # a hash of all codes
|
144
|
+
tax_code = TaxCloud::TaxCodes[TaxCloud::TaxCodes::DIRECT_MAIL_RELATED]
|
145
|
+
tax_code.ticid # 11000 or TaxCloud::TaxCodes::DIRECT_MAIL_RELATED
|
146
|
+
tax_code.description # "Direct-mail related"
|
147
|
+
```
|
148
|
+
|
149
|
+
Tax codes are organized in groups.
|
150
|
+
|
151
|
+
```ruby
|
152
|
+
TaxCloud::TaxCode::Groups.all # a hash of all groups
|
153
|
+
tax_code_group = TaxCloud::TaxCode::Groups[TaxCloud::TaxCode::Groups::SCHOOL_RELATED_PRODUCTS]
|
154
|
+
tax_code_group.group_id # 3 or TaxCloud::TaxCode::Groups::SCHOOL_RELATED_PRODUCTS
|
155
|
+
tax_code_group.description # School Related Products
|
156
|
+
tax_code_group.tax_codes # a hash of all codes in this group
|
157
|
+
```
|
158
|
+
|
159
|
+
Tax code constants are defined in `tax_code_constants.rb` and tax code group
|
160
|
+
constants in `tax_code_group_constants.rb`. These files can be generated by
|
161
|
+
running the following rake tasks.
|
162
|
+
|
163
|
+
```
|
164
|
+
TAXCLOUD_API_LOGIN_ID=... TAXCLOUD_API_KEY=... TAXCLOUD_USPS_USERNAME=... tax_cloud:tax_codes
|
165
|
+
TAXCLOUD_API_LOGIN_ID=... TAXCLOUD_API_KEY=... TAXCLOUD_USPS_USERNAME=... tax_cloud:tax_code_groups
|
166
|
+
```
|
167
|
+
|
168
|
+
### Tax States
|
169
|
+
TaxCloud manages a list of states in which you can calculate sales tax. The
|
170
|
+
default setup will only have SSUTA (Streamlined Sales and Use Tax Agreement)
|
171
|
+
states enabled. All other states will return $0 for tax values. To enable
|
172
|
+
other states, go to https://taxcloud.com/go/states-management/. You can find
|
173
|
+
more information about SSUTA
|
174
|
+
[here](http://www.streamlinedsalestax.org/index.php?page=About-Us).
|
175
|
+
|
176
|
+
### Running Tests
|
177
|
+
* VCR and WebMock are used to replay requests and avoid hitting the API each
|
178
|
+
time. To refresh the mocks, simply delete the `test/cassettes` directory.
|
179
|
+
* Run tests.
|
180
|
+
rake test
|
181
|
+
|
182
|
+
* If you need to record new requests against TaxCloud, set your keys first.
|
183
|
+
TAXCLOUD_API_LOGIN_ID=... TAXCLOUD_API_KEY=... TAXCLOUD_USPS_USERNAME=... rake test
|
184
|
+
|
185
|
+
The mocks will filter out your configuration details.
|
186
|
+
|
187
|
+
|
188
|
+
### Bugs, fixes, etc
|
189
|
+
* Fork.
|
190
|
+
* Write test(s).
|
191
|
+
* Fix.
|
192
|
+
* Commit.
|
193
|
+
* Submit pull request.
|
194
|
+
|
195
|
+
|
196
|
+
### License
|
197
|
+
|
198
|
+
Copyright Drew Tempelmeyer and contributors, 2011-2020.
|
199
|
+
|
200
|
+
This gem is licensed under the MIT license.
|
data/Rakefile
CHANGED
@@ -1,8 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'bundler/gem_tasks'
|
2
4
|
require 'rake/testtask'
|
3
|
-
require '
|
4
|
-
|
5
|
-
gemspec = eval(File.read(Dir["*.gemspec"].first))
|
5
|
+
require 'vcr'
|
6
6
|
|
7
7
|
Rake::TestTask.new(:test) do |test|
|
8
8
|
test.libs << 'lib' << 'test'
|
@@ -10,26 +10,12 @@ Rake::TestTask.new(:test) do |test|
|
|
10
10
|
test.verbose = false
|
11
11
|
end
|
12
12
|
|
13
|
-
RDoc::Task.new do |rd|
|
14
|
-
README = 'README.rdoc'
|
15
|
-
rd.main = README
|
16
|
-
rd.rdoc_files.include(README, 'CHANGELOG.rdoc', 'LICENSE.rdoc', "lib/**/*.rb")
|
17
|
-
rd.rdoc_dir = 'doc'
|
18
|
-
rd.title = 'tax_cloud'
|
19
|
-
end
|
20
|
-
|
21
|
-
desc "Validate the gemspec."
|
22
|
-
task :gemspec do
|
23
|
-
puts gemspec.validate
|
24
|
-
end
|
25
|
-
|
26
13
|
load 'lib/tasks/tax_cloud.rake'
|
27
14
|
load 'lib/tasks/tax_codes.rake'
|
28
15
|
load 'lib/tasks/tax_code_groups.rake'
|
29
|
-
|
30
16
|
load 'vcr/tasks/vcr.rake'
|
31
17
|
|
32
18
|
require 'rubocop/rake_task'
|
33
|
-
|
19
|
+
RuboCop::RakeTask.new(:rubocop)
|
34
20
|
|
35
|
-
task default: [
|
21
|
+
task default: %i[rubocop test]
|
data/lib/config/locales/en.yml
CHANGED
@@ -16,7 +16,7 @@ en:
|
|
16
16
|
\_\_end\n"
|
17
17
|
missing_config_option:
|
18
18
|
message: "Missing configuration option: %{name}."
|
19
|
-
summary: "A configuration option was not provided when
|
19
|
+
summary: "A configuration option was not provided when configuring TaxCloud."
|
20
20
|
resolution: "Review your `TaxCloud.configure` code. Are you using an environment
|
21
21
|
variable that has not been set?"
|
22
22
|
soap_error:
|
@@ -31,4 +31,4 @@ en:
|
|
31
31
|
api_error:
|
32
32
|
message: "%{message}"
|
33
33
|
summary: "The TaxCloud server returned an error in the response: `%{raw}`"
|
34
|
-
resolution: "Check the request parameters."
|
34
|
+
resolution: "Check the request parameters."
|
data/lib/hash.rb
CHANGED
@@ -1,9 +1,8 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
class Hash #:nodoc: all
|
2
4
|
# Downcase the keys. Use this because <tt>TaxCloud::Address.verify</tt> requires downcased keys
|
3
5
|
def downcase_keys
|
4
|
-
|
5
|
-
h[k.downcase] = v
|
6
|
-
h
|
7
|
-
end
|
6
|
+
transform_keys(&:downcase)
|
8
7
|
end
|
9
8
|
end
|
data/lib/tasks/tax_cloud.rake
CHANGED
@@ -1,20 +1,20 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
1
3
|
require 'tax_cloud'
|
2
4
|
|
3
5
|
namespace :tax_cloud do
|
4
|
-
|
5
|
-
desc "Configure TaxCloud."
|
6
|
+
desc 'Configure TaxCloud.'
|
6
7
|
task :configure do
|
7
8
|
unless TaxCloud.configured?
|
8
9
|
TaxCloud.configure do |config|
|
9
10
|
config.api_login_id = ENV['TAXCLOUD_API_LOGIN_ID']
|
10
11
|
config.api_key = ENV['TAXCLOUD_API_KEY']
|
11
12
|
config.usps_username = ENV['TAXCLOUD_USPS_USERNAME']
|
12
|
-
end
|
13
|
+
end
|
13
14
|
Savon.configure do |config|
|
14
15
|
config.log = false
|
15
16
|
end
|
16
17
|
HTTPI.log = false
|
17
18
|
end
|
18
19
|
end
|
19
|
-
|
20
20
|
end
|
@@ -1,13 +1,14 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
4
|
-
|
5
|
-
|
3
|
+
namespace :tax_cloud do
|
4
|
+
desc 'Generate tax code groups.'
|
5
|
+
task tax_code_groups: :configure do
|
6
|
+
filename = 'lib/tax_cloud/tax_code_group_constants.rb'
|
6
7
|
|
7
8
|
begin
|
8
9
|
groups = TaxCloud::TaxCode::Groups.all.values
|
9
10
|
|
10
|
-
File.open filename,
|
11
|
+
File.open filename, 'wt' do |f|
|
11
12
|
f.write "module TaxCloud\n"
|
12
13
|
f.write " class TaxCode\n"
|
13
14
|
f.write " # Tax Code Groups.\n"
|
@@ -16,9 +17,9 @@ namespace :tax_cloud do
|
|
16
17
|
groups.each do |group|
|
17
18
|
puts " #{group.description}"
|
18
19
|
code = group.description.upcase
|
19
|
-
code.gsub!
|
20
|
-
code.gsub!
|
21
|
-
code.gsub!
|
20
|
+
code.gsub!(/[^A-Z0-9]/, '_')
|
21
|
+
code.gsub!(/_$/, '')
|
22
|
+
code.gsub!(/_+/, '_')
|
22
23
|
f.write " \# #{group.description}\n"
|
23
24
|
f.write " #{code} = #{group.group_id}\n"
|
24
25
|
end
|
@@ -29,11 +30,10 @@ namespace :tax_cloud do
|
|
29
30
|
end
|
30
31
|
|
31
32
|
puts "Done, #{filename}."
|
32
|
-
rescue => e
|
33
|
-
puts
|
33
|
+
rescue StandardError => e
|
34
|
+
puts 'ERROR: Unable to generate a new list of tax codes.'
|
34
35
|
puts e.message
|
35
36
|
raise e
|
36
37
|
end
|
37
38
|
end
|
38
|
-
|
39
39
|
end
|
data/lib/tasks/tax_codes.rake
CHANGED
@@ -1,15 +1,16 @@
|
|
1
|
-
|
1
|
+
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
4
|
-
|
5
|
-
|
3
|
+
namespace :tax_cloud do
|
4
|
+
desc 'Generate tax codes.'
|
5
|
+
task tax_codes: :configure do
|
6
|
+
filename = 'lib/tax_cloud/tax_code_constants.rb'
|
6
7
|
|
7
8
|
begin
|
8
9
|
groups = TaxCloud::TaxCode::Groups.all.values
|
9
10
|
groups_and_tax_codes = Hash[groups.map do |group|
|
10
|
-
[
|
11
|
+
[group, group.tax_codes.values]
|
11
12
|
end]
|
12
|
-
File.open filename,
|
13
|
+
File.open filename, 'wt' do |f|
|
13
14
|
f.write "module TaxCloud\n"
|
14
15
|
f.write " # Tax Codes.\n"
|
15
16
|
f.write " class TaxCodes\n"
|
@@ -20,9 +21,9 @@ namespace :tax_cloud do
|
|
20
21
|
f.write " \# #{group.description}\n\n"
|
21
22
|
tax_codes.each do |tax_code|
|
22
23
|
code = tax_code.description.upcase
|
23
|
-
code.gsub!
|
24
|
-
code.gsub!
|
25
|
-
code.gsub!
|
24
|
+
code.gsub!(/[^A-Z0-9]/, '_')
|
25
|
+
code.gsub!(/_+$/, '')
|
26
|
+
code.gsub!(/_+/, '_')
|
26
27
|
# avoid duplicates
|
27
28
|
code_id = codes[code] ? "#{code}_#{codes[code]}" : code
|
28
29
|
codes[code] = (codes[code] || 0) + 1
|
@@ -35,11 +36,10 @@ namespace :tax_cloud do
|
|
35
36
|
f.write "end\n"
|
36
37
|
end
|
37
38
|
puts "Done, #{filename}."
|
38
|
-
rescue => e
|
39
|
-
puts
|
39
|
+
rescue StandardError => e
|
40
|
+
puts 'ERROR: Unable to generate a new list of tax codes.'
|
40
41
|
puts e.message
|
41
42
|
raise e
|
42
43
|
end
|
43
|
-
|
44
44
|
end
|
45
45
|
end
|