fortnox-api 0.7.2 → 0.9.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.codeclimate.yml +1 -0
- data/.env.template +7 -0
- data/.env.test +11 -3
- data/.gitignore +7 -1
- data/.rubocop.yml +18 -2
- data/.tool-versions +1 -0
- data/.travis.yml +15 -19
- data/CHANGELOG.md +66 -1
- data/CONTRIBUTE.md +21 -9
- data/DEVELOPER_README.md +72 -0
- data/Guardfile +13 -4
- data/README.md +226 -61
- data/Rakefile +133 -0
- data/bin/get_tokens +79 -0
- data/bin/renew_tokens +28 -0
- data/fortnox-api.gemspec +31 -25
- data/lib/fortnox/api/mappers/article.rb +1 -1
- data/lib/fortnox/api/mappers/base/from_json.rb +7 -6
- data/lib/fortnox/api/mappers/base/to_json.rb +4 -5
- data/lib/fortnox/api/mappers/base.rb +3 -3
- data/lib/fortnox/api/mappers/customer.rb +1 -1
- data/lib/fortnox/api/mappers/default_delivery_types.rb +1 -1
- data/lib/fortnox/api/mappers/default_templates.rb +1 -1
- data/lib/fortnox/api/mappers/edi_information.rb +1 -1
- data/lib/fortnox/api/mappers/email_information.rb +1 -1
- data/lib/fortnox/api/mappers/invoice.rb +4 -4
- data/lib/fortnox/api/mappers/invoice_row.rb +1 -1
- data/lib/fortnox/api/mappers/order.rb +4 -4
- data/lib/fortnox/api/mappers/order_row.rb +1 -1
- data/lib/fortnox/api/mappers/project.rb +1 -1
- data/lib/fortnox/api/mappers/terms_of_payment.rb +1 -1
- data/lib/fortnox/api/mappers/unit.rb +1 -1
- data/lib/fortnox/api/mappers/value/country_string.rb +1 -1
- data/lib/fortnox/api/mappers.rb +18 -18
- data/lib/fortnox/api/models/article.rb +2 -2
- data/lib/fortnox/api/models/base.rb +23 -21
- data/lib/fortnox/api/models/customer.rb +57 -57
- data/lib/fortnox/api/models/document.rb +5 -2
- data/lib/fortnox/api/models/invoice.rb +2 -2
- data/lib/fortnox/api/models/label.rb +3 -3
- data/lib/fortnox/api/models/order.rb +2 -2
- data/lib/fortnox/api/models/project.rb +2 -2
- data/lib/fortnox/api/models/terms_of_payment.rb +2 -2
- data/lib/fortnox/api/models/unit.rb +2 -2
- data/lib/fortnox/api/models.rb +7 -7
- data/lib/fortnox/api/repositories/article.rb +3 -3
- data/lib/fortnox/api/repositories/authentication.rb +61 -0
- data/lib/fortnox/api/repositories/base/savers.rb +3 -1
- data/lib/fortnox/api/repositories/base.rb +25 -38
- data/lib/fortnox/api/repositories/customer.rb +3 -3
- data/lib/fortnox/api/repositories/invoice.rb +3 -3
- data/lib/fortnox/api/repositories/order.rb +3 -3
- data/lib/fortnox/api/repositories/project.rb +3 -3
- data/lib/fortnox/api/repositories/terms_of_payment.rb +3 -3
- data/lib/fortnox/api/repositories/unit.rb +3 -3
- data/lib/fortnox/api/repositories.rb +8 -7
- data/lib/fortnox/api/request_handling.rb +30 -18
- data/lib/fortnox/api/types/default_delivery_types.rb +0 -2
- data/lib/fortnox/api/types/default_templates.rb +0 -2
- data/lib/fortnox/api/types/document_row.rb +3 -3
- data/lib/fortnox/api/types/edi_information.rb +0 -2
- data/lib/fortnox/api/types/email_information.rb +0 -2
- data/lib/fortnox/api/types/enums.rb +54 -10
- data/lib/fortnox/api/types/invoice_row.rb +1 -1
- data/lib/fortnox/api/types/model.rb +5 -9
- data/lib/fortnox/api/types/nullable.rb +13 -9
- data/lib/fortnox/api/types/order_row.rb +1 -1
- data/lib/fortnox/api/types/required.rb +3 -3
- data/lib/fortnox/api/types/sized.rb +4 -4
- data/lib/fortnox/api/types.rb +37 -24
- data/lib/fortnox/api/version.rb +1 -1
- data/lib/fortnox/api.rb +21 -39
- data/spec/fortnox/api/mappers/base/canonical_name_sym_spec.rb +13 -11
- data/spec/fortnox/api/mappers/base/from_json_spec.rb +10 -12
- data/spec/fortnox/api/mappers/base/to_json_spec.rb +48 -57
- data/spec/fortnox/api/mappers/base_spec.rb +4 -7
- data/spec/fortnox/api/mappers/contexts/json_conversion.rb +38 -33
- data/spec/fortnox/api/mappers/default_delivery_types_spec.rb +1 -1
- data/spec/fortnox/api/mappers/examples/mapper.rb +1 -1
- data/spec/fortnox/api/mappers/unit_spec.rb +3 -4
- data/spec/fortnox/api/models/base_spec.rb +33 -22
- data/spec/fortnox/api/models/unit_spec.rb +5 -3
- data/spec/fortnox/api/repositories/article_spec.rb +14 -9
- data/spec/fortnox/api/repositories/authentication_spec.rb +103 -0
- data/spec/fortnox/api/repositories/base_spec.rb +105 -326
- data/spec/fortnox/api/repositories/customer_spec.rb +37 -7
- data/spec/fortnox/api/repositories/examples/all.rb +0 -1
- data/spec/fortnox/api/repositories/examples/find.rb +5 -8
- data/spec/fortnox/api/repositories/examples/only.rb +4 -13
- data/spec/fortnox/api/repositories/examples/save.rb +32 -18
- data/spec/fortnox/api/repositories/examples/save_with_nested_model.rb +0 -5
- data/spec/fortnox/api/repositories/examples/save_with_specially_named_attribute.rb +1 -4
- data/spec/fortnox/api/repositories/examples/search.rb +4 -7
- data/spec/fortnox/api/repositories/invoice_spec.rb +72 -29
- data/spec/fortnox/api/repositories/order_spec.rb +11 -9
- data/spec/fortnox/api/repositories/project_spec.rb +7 -6
- data/spec/fortnox/api/repositories/terms_of_payment_spec.rb +9 -7
- data/spec/fortnox/api/repositories/unit_spec.rb +13 -11
- data/spec/fortnox/api/types/country_spec.rb +1 -1
- data/spec/fortnox/api/types/email_spec.rb +2 -2
- data/spec/fortnox/api/types/enums_spec.rb +1 -0
- data/spec/fortnox/api/types/examples/document_row.rb +3 -3
- data/spec/fortnox/api/types/examples/enum.rb +4 -4
- data/spec/fortnox/api/types/examples/types.rb +1 -3
- data/spec/fortnox/api/types/housework_types_spec.rb +124 -43
- data/spec/fortnox/api/types/model_spec.rb +13 -23
- data/spec/fortnox/api/types/nullable_spec.rb +30 -10
- data/spec/fortnox/api/types/order_row_spec.rb +2 -2
- data/spec/fortnox/api/types/required_spec.rb +7 -15
- data/spec/fortnox/api/types/sales_account_spec.rb +57 -0
- data/spec/fortnox/api_spec.rb +19 -124
- data/spec/spec_helper.rb +0 -14
- data/spec/support/helpers/configuration_helper.rb +30 -3
- data/spec/support/helpers.rb +1 -1
- data/spec/support/matchers/type/attribute_matcher.rb +2 -2
- data/spec/support/matchers/type/enum_matcher.rb +1 -1
- data/spec/support/matchers/type/have_account_number_matcher.rb +1 -1
- data/spec/support/matchers/type/have_email_matcher.rb +1 -1
- data/spec/support/matchers/type/have_nullable_date_matcher.rb +7 -5
- data/spec/support/matchers/type/have_nullable_matcher.rb +1 -1
- data/spec/support/matchers/type/have_nullable_string_matcher.rb +5 -5
- data/spec/support/matchers/type/require_attribute_matcher.rb +5 -5
- data/spec/support/matchers/type/type_matcher.rb +1 -1
- data/spec/support/vcr_setup.rb +16 -0
- data/spec/vcr_cassettes/articles/all.yml +41 -42
- data/spec/vcr_cassettes/articles/find_by_hash_failure.yml +36 -19
- data/spec/vcr_cassettes/articles/find_failure.yml +36 -19
- data/spec/vcr_cassettes/articles/find_id_1.yml +38 -20
- data/spec/vcr_cassettes/articles/find_new.yml +39 -22
- data/spec/vcr_cassettes/articles/multi_param_find_by_hash.yml +38 -21
- data/spec/vcr_cassettes/articles/save_new.yml +37 -19
- data/spec/vcr_cassettes/articles/save_old.yml +39 -22
- data/spec/vcr_cassettes/articles/save_with_specially_named_attribute.yml +37 -19
- data/spec/vcr_cassettes/articles/search_by_name.yml +41 -21
- data/spec/vcr_cassettes/articles/search_miss.yml +36 -19
- data/spec/vcr_cassettes/articles/search_with_special_char.yml +36 -19
- data/spec/vcr_cassettes/articles/single_param_find_by_hash.yml +38 -32
- data/spec/vcr_cassettes/authentication/expired_token.yml +54 -0
- data/spec/vcr_cassettes/authentication/invalid_authorization.yml +57 -0
- data/spec/vcr_cassettes/authentication/invalid_refresh_token.yml +58 -0
- data/spec/vcr_cassettes/authentication/valid_request.yml +63 -0
- data/spec/vcr_cassettes/customers/all.yml +44 -132
- data/spec/vcr_cassettes/customers/find_by_hash_failure.yml +36 -19
- data/spec/vcr_cassettes/customers/find_failure.yml +36 -19
- data/spec/vcr_cassettes/customers/find_id_1.yml +39 -21
- data/spec/vcr_cassettes/customers/find_new.yml +38 -21
- data/spec/vcr_cassettes/customers/find_with_sales_account.yml +63 -0
- data/spec/vcr_cassettes/customers/multi_param_find_by_hash.yml +38 -21
- data/spec/vcr_cassettes/customers/save_new.yml +36 -18
- data/spec/vcr_cassettes/customers/save_new_with_country_code_SE.yml +32 -24
- data/spec/vcr_cassettes/customers/save_new_with_sales_account.yml +63 -0
- data/spec/vcr_cassettes/customers/save_old.yml +38 -21
- data/spec/vcr_cassettes/customers/save_with_specially_named_attribute.yml +36 -18
- data/spec/vcr_cassettes/customers/search_by_name.yml +38 -48
- data/spec/vcr_cassettes/customers/search_miss.yml +36 -19
- data/spec/vcr_cassettes/customers/search_with_special_char.yml +36 -19
- data/spec/vcr_cassettes/customers/single_param_find_by_hash.yml +39 -22
- data/spec/vcr_cassettes/invoices/all.yml +71 -114
- data/spec/vcr_cassettes/invoices/filter_hit.yml +39 -24
- data/spec/vcr_cassettes/invoices/filter_invalid.yml +35 -17
- data/spec/vcr_cassettes/invoices/find_by_hash_failure.yml +36 -19
- data/spec/vcr_cassettes/invoices/find_failure.yml +36 -19
- data/spec/vcr_cassettes/invoices/find_id_1.yml +40 -22
- data/spec/vcr_cassettes/invoices/find_new.yml +41 -24
- data/spec/vcr_cassettes/invoices/multi_param_find_by_hash.yml +38 -21
- data/spec/vcr_cassettes/invoices/row_description_limit.yml +65 -0
- data/spec/vcr_cassettes/invoices/save_new.yml +39 -21
- data/spec/vcr_cassettes/invoices/save_new_with_comments.yml +39 -21
- data/spec/vcr_cassettes/invoices/save_new_with_country.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_country_GB.yml +36 -27
- data/spec/vcr_cassettes/invoices/save_new_with_country_Norge.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_country_Norway.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_country_Sverige.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_country_VA.yml +36 -27
- data/spec/vcr_cassettes/invoices/save_new_with_country_VI.yml +36 -27
- data/spec/vcr_cassettes/invoices/save_new_with_country_empty_string.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_country_nil.yml +35 -26
- data/spec/vcr_cassettes/invoices/save_new_with_unsaved_parent.yml +65 -0
- data/spec/vcr_cassettes/invoices/save_old.yml +41 -24
- data/spec/vcr_cassettes/invoices/save_old_with_empty_comments.yml +41 -24
- data/spec/vcr_cassettes/invoices/save_old_with_empty_country.yml +37 -29
- data/spec/vcr_cassettes/invoices/save_old_with_nil_comments.yml +41 -24
- data/spec/vcr_cassettes/invoices/save_old_with_nil_country.yml +37 -29
- data/spec/vcr_cassettes/invoices/save_with_nested_model.yml +40 -21
- data/spec/vcr_cassettes/invoices/save_with_specially_named_attribute.yml +39 -20
- data/spec/vcr_cassettes/invoices/search_by_name.yml +38 -27
- data/spec/vcr_cassettes/invoices/search_miss.yml +36 -19
- data/spec/vcr_cassettes/invoices/search_with_special_char.yml +36 -19
- data/spec/vcr_cassettes/invoices/single_param_find_by_hash.yml +39 -22
- data/spec/vcr_cassettes/orders/all.yml +44 -119
- data/spec/vcr_cassettes/orders/filter_hit.yml +39 -26
- data/spec/vcr_cassettes/orders/filter_invalid.yml +35 -17
- data/spec/vcr_cassettes/orders/find_by_hash_failure.yml +36 -19
- data/spec/vcr_cassettes/orders/find_failure.yml +36 -19
- data/spec/vcr_cassettes/orders/find_id_1.yml +42 -23
- data/spec/vcr_cassettes/orders/find_new.yml +41 -24
- data/spec/vcr_cassettes/orders/housework_invalid_tax_reduction_type.yml +61 -0
- data/spec/vcr_cassettes/orders/housework_othercoses_invalid.yml +61 -0
- data/spec/vcr_cassettes/orders/housework_type_babysitting.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_cleaning.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_construction.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_cooking.yml +36 -18
- data/spec/vcr_cassettes/orders/housework_type_electricity.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_gardening.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_glassmetalwork.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_grounddrainagework.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_hvac.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_itservices.yml +65 -0
- data/spec/vcr_cassettes/orders/housework_type_majorappliancerepair.yml +65 -0
- data/spec/vcr_cassettes/orders/housework_type_masonry.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_movingservices.yml +65 -0
- data/spec/vcr_cassettes/orders/housework_type_othercare.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_othercosts.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_paintingwallpapering.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_snowplowing.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_textileclothing.yml +40 -21
- data/spec/vcr_cassettes/orders/housework_type_tutoring.yml +36 -18
- data/spec/vcr_cassettes/orders/multi_param_find_by_hash.yml +38 -21
- data/spec/vcr_cassettes/orders/save_new.yml +40 -22
- data/spec/vcr_cassettes/orders/save_old.yml +41 -24
- data/spec/vcr_cassettes/orders/save_with_nested_model.yml +40 -21
- data/spec/vcr_cassettes/orders/search_by_name.yml +38 -23
- data/spec/vcr_cassettes/orders/search_miss.yml +36 -19
- data/spec/vcr_cassettes/orders/search_with_special_char.yml +36 -19
- data/spec/vcr_cassettes/orders/single_param_find_by_hash.yml +39 -22
- data/spec/vcr_cassettes/projects/all.yml +39 -37
- data/spec/vcr_cassettes/projects/find_by_hash_failure.yml +36 -19
- data/spec/vcr_cassettes/projects/find_failure.yml +36 -19
- data/spec/vcr_cassettes/projects/find_id_1.yml +38 -21
- data/spec/vcr_cassettes/projects/find_new.yml +39 -22
- data/spec/vcr_cassettes/projects/multi_param_find_by_hash.yml +40 -22
- data/spec/vcr_cassettes/projects/save_new.yml +37 -19
- data/spec/vcr_cassettes/projects/save_old.yml +39 -22
- data/spec/vcr_cassettes/projects/single_param_find_by_hash.yml +38 -21
- data/spec/vcr_cassettes/termsofpayments/all.yml +43 -29
- data/spec/vcr_cassettes/termsofpayments/find_failure.yml +36 -19
- data/spec/vcr_cassettes/termsofpayments/find_id_1.yml +38 -22
- data/spec/vcr_cassettes/termsofpayments/find_new.yml +38 -21
- data/spec/vcr_cassettes/termsofpayments/save_new.yml +37 -19
- data/spec/vcr_cassettes/termsofpayments/save_old.yml +38 -21
- data/spec/vcr_cassettes/units/all.yml +38 -26
- data/spec/vcr_cassettes/units/find_failure.yml +36 -19
- data/spec/vcr_cassettes/units/find_id_1.yml +38 -21
- data/spec/vcr_cassettes/units/find_new.yml +38 -21
- data/spec/vcr_cassettes/units/save_new.yml +37 -19
- data/spec/vcr_cassettes/units/save_old.yml +38 -21
- data/spec/vcr_cassettes/units/save_with_specially_named_attribute.yml +37 -19
- metadata +133 -252
- data/lib/fortnox/api/circular_queue.rb +0 -39
- data/spec/fortnox/api/circular_queue_spec.rb +0 -52
- data/spec/support/helpers/dummy_class_helper.rb +0 -38
- data/spec/support/helpers/when_performing_helper.rb +0 -7
- data/spec/vcr_cassettes/invoices/save_new_with_country_KR.yml +0 -57
- data/temp.txt +0 -1
@@ -7,62 +7,143 @@ require 'fortnox/api/repositories/order'
|
|
7
7
|
require 'fortnox/api/models/order'
|
8
8
|
require 'fortnox/api/types/order_row'
|
9
9
|
|
10
|
-
# rubocop:disable RSpec/DescribeClass
|
11
10
|
describe 'HouseworkTypes', integration: true do
|
12
11
|
include Helpers::Configuration
|
12
|
+
include Helpers::Repositories
|
13
13
|
|
14
|
-
before
|
14
|
+
before do
|
15
|
+
set_api_test_configuration
|
16
|
+
stub_const('TYPE_ROT', Fortnox::API::Types::TaxReductionTypes['rot'])
|
17
|
+
stub_const('TYPE_RUT', Fortnox::API::Types::TaxReductionTypes['rut'])
|
18
|
+
end
|
15
19
|
|
16
20
|
let(:repository) { Fortnox::API::Repository::Order.new }
|
17
|
-
let(:valid_model) do
|
18
|
-
Fortnox::API::Model::Order.new(customer_number: '1', order_rows: [order_row])
|
19
|
-
end
|
20
|
-
let(:order_row) do
|
21
|
-
Fortnox::API::Types::OrderRow.new(ordered_quantity: 1,
|
22
|
-
article_number: '0000',
|
23
|
-
housework_type: housework_type)
|
24
|
-
end
|
25
21
|
|
26
|
-
shared_examples_for 'housework type' do |type, legacy: false|
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
22
|
+
shared_examples_for 'housework type' do |type, tax_reduction_type, legacy: false, housework: true|
|
23
|
+
let(:document) do
|
24
|
+
Fortnox::API::Model::Order.new(
|
25
|
+
customer_number: '1',
|
26
|
+
tax_reduction_type: Object.const_get(tax_reduction_type),
|
27
|
+
order_rows: [
|
28
|
+
Fortnox::API::Types::OrderRow.new(
|
29
|
+
ordered_quantity: 1,
|
30
|
+
article_number: '101',
|
31
|
+
housework_type: Fortnox::API::Types::HouseworkTypes[type],
|
32
|
+
housework: housework
|
33
|
+
)
|
34
|
+
]
|
35
|
+
)
|
36
|
+
end
|
37
|
+
let(:save) do
|
38
|
+
-> { VCR.use_cassette("orders/housework_type_#{type.downcase}") { repository.save(document) } }
|
33
39
|
end
|
34
|
-
|
35
|
-
let(:error_message) { 'Skattereduktion för den valda typen av husarbete har upphört.' }
|
36
|
-
let(:housework_type) { Fortnox::API::Types::HouseworkTypes[type] }
|
37
40
|
|
38
41
|
context "when creating an OrderRow with housework_type set to #{type}" do
|
39
42
|
if legacy
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
+
let(:error_message) { /Skattereduktion för den valda typen av husarbete har upphört/ }
|
44
|
+
|
45
|
+
specify { expect { save.call }.to raise_error(Fortnox::API::RemoteServerError, error_message) }
|
43
46
|
else
|
44
|
-
|
47
|
+
specify { expect { save.call }.not_to raise_error }
|
45
48
|
end
|
46
49
|
end
|
47
50
|
end
|
48
51
|
|
49
|
-
it_behaves_like 'housework type', 'CONSTRUCTION'
|
50
|
-
it_behaves_like 'housework type', 'ELECTRICITY'
|
51
|
-
it_behaves_like 'housework type', 'GLASSMETALWORK'
|
52
|
-
it_behaves_like 'housework type', 'GROUNDDRAINAGEWORK'
|
53
|
-
it_behaves_like 'housework type', 'MASONRY'
|
54
|
-
it_behaves_like 'housework type', 'PAINTINGWALLPAPERING'
|
55
|
-
it_behaves_like 'housework type', 'HVAC'
|
56
|
-
it_behaves_like 'housework type', '
|
57
|
-
|
58
|
-
it_behaves_like 'housework type', '
|
59
|
-
it_behaves_like 'housework type', '
|
60
|
-
it_behaves_like 'housework type', '
|
61
|
-
it_behaves_like 'housework type', '
|
62
|
-
it_behaves_like 'housework type', '
|
63
|
-
it_behaves_like 'housework type', '
|
64
|
-
|
65
|
-
it_behaves_like 'housework type', '
|
66
|
-
it_behaves_like 'housework type', '
|
52
|
+
it_behaves_like 'housework type', 'CONSTRUCTION', 'TYPE_ROT'
|
53
|
+
it_behaves_like 'housework type', 'ELECTRICITY', 'TYPE_ROT'
|
54
|
+
it_behaves_like 'housework type', 'GLASSMETALWORK', 'TYPE_ROT'
|
55
|
+
it_behaves_like 'housework type', 'GROUNDDRAINAGEWORK', 'TYPE_ROT'
|
56
|
+
it_behaves_like 'housework type', 'MASONRY', 'TYPE_ROT'
|
57
|
+
it_behaves_like 'housework type', 'PAINTINGWALLPAPERING', 'TYPE_ROT'
|
58
|
+
it_behaves_like 'housework type', 'HVAC', 'TYPE_ROT'
|
59
|
+
it_behaves_like 'housework type', 'OTHERCOSTS', 'TYPE_ROT', housework: false
|
60
|
+
|
61
|
+
it_behaves_like 'housework type', 'MAJORAPPLIANCEREPAIR', 'TYPE_RUT'
|
62
|
+
it_behaves_like 'housework type', 'MOVINGSERVICES', 'TYPE_RUT'
|
63
|
+
it_behaves_like 'housework type', 'ITSERVICES', 'TYPE_RUT'
|
64
|
+
it_behaves_like 'housework type', 'CLEANING', 'TYPE_RUT'
|
65
|
+
it_behaves_like 'housework type', 'TEXTILECLOTHING', 'TYPE_RUT'
|
66
|
+
it_behaves_like 'housework type', 'SNOWPLOWING', 'TYPE_RUT'
|
67
|
+
it_behaves_like 'housework type', 'GARDENING', 'TYPE_RUT'
|
68
|
+
it_behaves_like 'housework type', 'BABYSITTING', 'TYPE_RUT'
|
69
|
+
it_behaves_like 'housework type', 'OTHERCARE', 'TYPE_RUT'
|
70
|
+
it_behaves_like 'housework type', 'OTHERCOSTS', 'TYPE_RUT', housework: false
|
71
|
+
|
72
|
+
# rubocop:disable RSpec/RepeatedExample
|
73
|
+
# rubocop:disable RSpec/RepeatedDescription
|
74
|
+
it 'will soon be added' do
|
75
|
+
pending 'todo'
|
76
|
+
raise StandardError, 'Will be supported 2021-01-01'
|
77
|
+
# it_behaves_like 'housework type', 'HOMEMAINTENANCE', TYPE_RUT
|
78
|
+
# it_behaves_like 'housework type', 'FURNISHING', TYPE_RUT
|
79
|
+
# it_behaves_like 'housework type', 'TRANSPORTATIONSERVICES', TYPE_RUT
|
80
|
+
# it_behaves_like 'housework type', 'WASHINGANDCAREOFCLOTHING', TYPE_RUT
|
81
|
+
end
|
82
|
+
|
83
|
+
it 'will soon be added' do
|
84
|
+
pending 'todo'
|
85
|
+
raise StandardError, 'Will be supported 2021-01-01'
|
86
|
+
# it_behaves_like 'housework type', 'SOLARCELLS', TYPE_GREEN
|
87
|
+
# it_behaves_like 'housework type', 'STORAGESELFPRODUCEDELECTRICTY', TYPE_GREEN
|
88
|
+
# it_behaves_like 'housework type', 'CHARGINGSTATIONELECTRICVEHICLE', TYPE_GREEN
|
89
|
+
# it_behaves_like 'housework type', 'OTHERCOSTS', TYPE_GREEN
|
90
|
+
end
|
91
|
+
# rubocop:enable RSpec/RepeatedExample
|
92
|
+
# rubocop:enable RSpec/RepeatedDescription
|
93
|
+
|
94
|
+
it_behaves_like 'housework type', 'COOKING', 'TYPE_RUT', legacy: true
|
95
|
+
it_behaves_like 'housework type', 'TUTORING', 'TYPE_RUT', legacy: true
|
96
|
+
|
97
|
+
describe 'with OTHERCOSTS' do
|
98
|
+
let(:document) do
|
99
|
+
Fortnox::API::Model::Order.new(
|
100
|
+
customer_number: '1',
|
101
|
+
tax_reduction_type: TYPE_ROT,
|
102
|
+
order_rows: [
|
103
|
+
Fortnox::API::Types::OrderRow.new(
|
104
|
+
ordered_quantity: 1,
|
105
|
+
article_number: '101',
|
106
|
+
housework_type: Fortnox::API::Types::HouseworkTypes['OTHERCOSTS'],
|
107
|
+
housework: true
|
108
|
+
)
|
109
|
+
]
|
110
|
+
)
|
111
|
+
end
|
112
|
+
|
113
|
+
let(:error_message) { /Kan inte sätta typen övrig kostnad på en rad markerad som husarbete/ }
|
114
|
+
|
115
|
+
it "can't have housework set to true" do
|
116
|
+
expect do
|
117
|
+
VCR.use_cassette('orders/housework_othercoses_invalid') { repository.save(document) }
|
118
|
+
end.to raise_error(Fortnox::API::RemoteServerError, error_message)
|
119
|
+
end
|
120
|
+
end
|
121
|
+
|
122
|
+
describe 'with wrong tax reduction type' do
|
123
|
+
let(:type) { 'CONSTRUCTION' }
|
124
|
+
let(:document) do
|
125
|
+
Fortnox::API::Model::Order.new(
|
126
|
+
customer_number: '1',
|
127
|
+
tax_reduction_type: TYPE_RUT,
|
128
|
+
order_rows: [
|
129
|
+
Fortnox::API::Types::OrderRow.new(
|
130
|
+
ordered_quantity: 1,
|
131
|
+
article_number: '101',
|
132
|
+
housework_type: Fortnox::API::Types::HouseworkTypes[type],
|
133
|
+
housework: true
|
134
|
+
)
|
135
|
+
]
|
136
|
+
)
|
137
|
+
end
|
138
|
+
|
139
|
+
let(:error_message) do
|
140
|
+
/Dokument med skattereduktionstypen '#{TYPE_RUT}' får inte innehålla rader med husarbetestypen '#{type}'/
|
141
|
+
end
|
142
|
+
|
143
|
+
it 'raises an error' do
|
144
|
+
expect do
|
145
|
+
VCR.use_cassette('orders/housework_invalid_tax_reduction_type') { repository.save(document) }
|
146
|
+
end.to raise_error(Fortnox::API::RemoteServerError, error_message)
|
147
|
+
end
|
148
|
+
end
|
67
149
|
end
|
68
|
-
# rubocop:enable RSpec/DescribeClass
|
@@ -6,22 +6,18 @@ require 'fortnox/api/types/model'
|
|
6
6
|
|
7
7
|
RSpec.describe Fortnox::API::Types::Model do
|
8
8
|
shared_examples_for 'raises error' do |error|
|
9
|
-
|
10
|
-
|
11
|
-
include Dry::Types.module
|
9
|
+
before do
|
10
|
+
stub_const('Types::Age', Dry::Types['strict.integer'].constrained(gt: 18).is(:required))
|
12
11
|
|
13
|
-
|
14
|
-
end
|
15
|
-
|
16
|
-
class TypesModelUser < Fortnox::API::Types::Model
|
12
|
+
types_model_user_class = Class.new(Fortnox::API::Types::Model) do
|
17
13
|
attribute :age, Types::Age
|
18
14
|
end
|
15
|
+
|
16
|
+
stub_const('TypesModelUser', types_model_user_class)
|
19
17
|
end
|
20
18
|
|
21
19
|
describe "User inheriting from #{described_class}" do
|
22
|
-
|
23
|
-
|
24
|
-
it { is_expected.to raise_error(error) }
|
20
|
+
specify { expect { TypesModelUser.new(args) }.to raise_error(error) }
|
25
21
|
end
|
26
22
|
end
|
27
23
|
|
@@ -32,28 +28,22 @@ RSpec.describe Fortnox::API::Types::Model do
|
|
32
28
|
end
|
33
29
|
|
34
30
|
context 'when omitting optional keys' do
|
35
|
-
|
36
|
-
|
37
|
-
include Dry::Types.module
|
31
|
+
before do
|
32
|
+
stub_const('Types::Nullable::String', Dry::Types['strict.string'].optional)
|
38
33
|
|
39
|
-
|
40
|
-
String = Types::Strict::String.optional
|
41
|
-
end
|
42
|
-
end
|
43
|
-
|
44
|
-
class User < Fortnox::API::Types::Model
|
34
|
+
user_class = Class.new(Fortnox::API::Types::Model) do
|
45
35
|
attribute :optional_string, Types::Nullable::String
|
46
36
|
end
|
47
|
-
end
|
48
37
|
|
49
|
-
|
38
|
+
stub_const('User', user_class)
|
39
|
+
end
|
50
40
|
|
51
|
-
|
41
|
+
specify { expect { User.new }.not_to raise_error }
|
52
42
|
|
53
43
|
describe 'optional attribute' do
|
54
44
|
subject { User.new.optional_string }
|
55
45
|
|
56
|
-
it { is_expected.to
|
46
|
+
it { is_expected.to be_nil }
|
57
47
|
end
|
58
48
|
end
|
59
49
|
end
|
@@ -8,52 +8,72 @@ describe Fortnox::API::Types::Nullable, type: :type do
|
|
8
8
|
subject { TestStruct }
|
9
9
|
|
10
10
|
describe 'String' do
|
11
|
-
|
12
|
-
|
11
|
+
before do
|
12
|
+
test_struct_class = Class.new(Dry::Struct) do
|
13
13
|
attribute :string, Fortnox::API::Types::Nullable::String
|
14
14
|
end
|
15
|
+
|
16
|
+
stub_const('TestStruct', test_struct_class)
|
15
17
|
end
|
16
18
|
|
17
19
|
it { is_expected.to have_nullable(:string, 'A simple message', 0, '0') }
|
18
20
|
end
|
19
21
|
|
20
22
|
describe 'Float' do
|
21
|
-
|
22
|
-
|
23
|
+
before do
|
24
|
+
test_struct_class = Class.new(Dry::Struct) do
|
23
25
|
attribute :float, Fortnox::API::Types::Nullable::Float
|
24
26
|
end
|
27
|
+
|
28
|
+
stub_const('TestStruct', test_struct_class)
|
25
29
|
end
|
26
30
|
|
27
31
|
it { is_expected.to have_nullable(:float, 14.0, 'Not a Float!', 0.0) }
|
28
32
|
end
|
29
33
|
|
30
34
|
describe 'Integer' do
|
31
|
-
|
32
|
-
|
35
|
+
before do
|
36
|
+
test_struct_class = Class.new(Dry::Struct) do
|
33
37
|
attribute :integer, Fortnox::API::Types::Nullable::Integer
|
34
38
|
end
|
39
|
+
|
40
|
+
stub_const('TestStruct', test_struct_class)
|
35
41
|
end
|
36
42
|
|
37
43
|
it { is_expected.to have_nullable(:integer, 14, 14.0, 14) }
|
38
44
|
end
|
39
45
|
|
40
46
|
describe 'Boolean' do
|
41
|
-
|
42
|
-
|
47
|
+
before do
|
48
|
+
test_struct_class = Class.new(Dry::Struct) do
|
43
49
|
attribute :boolean, Fortnox::API::Types::Nullable::Boolean
|
44
50
|
end
|
51
|
+
|
52
|
+
stub_const('TestStruct', test_struct_class)
|
45
53
|
end
|
46
54
|
|
47
55
|
it { is_expected.to have_nullable(:boolean, true, 'Not a Boolean!', false) }
|
48
56
|
end
|
49
57
|
|
50
58
|
describe 'Date' do
|
51
|
-
|
52
|
-
|
59
|
+
before do
|
60
|
+
test_struct_class = Class.new(Dry::Struct) do
|
53
61
|
attribute :date, Fortnox::API::Types::Nullable::Date
|
54
62
|
end
|
63
|
+
|
64
|
+
stub_const('TestStruct', test_struct_class)
|
55
65
|
end
|
56
66
|
|
57
67
|
it { is_expected.to have_nullable_date(:date, Date.new(2016, 1, 1), 'Not a Date!') }
|
68
|
+
|
69
|
+
context 'with empty string' do
|
70
|
+
subject { TestStruct.new(date: '').date }
|
71
|
+
|
72
|
+
it { is_expected.to be_nil }
|
73
|
+
end
|
74
|
+
|
75
|
+
context 'with invalid date' do
|
76
|
+
it { expect { TestStruct.new(date: 'Not a Date!') }.to raise_error(Fortnox::API::AttributeError, /invalid date/) }
|
77
|
+
end
|
58
78
|
end
|
59
79
|
end
|
@@ -5,10 +5,10 @@ require 'fortnox/api/types/order_row'
|
|
5
5
|
require 'fortnox/api/types/examples/document_row'
|
6
6
|
|
7
7
|
RSpec.describe Fortnox::API::Types::OrderRow, type: :type do
|
8
|
-
valid_hash = { ordered_quantity: 10.5 }
|
9
|
-
|
10
8
|
subject { described_class }
|
11
9
|
|
10
|
+
valid_hash = { ordered_quantity: 10.5 }
|
11
|
+
|
12
12
|
it { is_expected.to require_attribute(:ordered_quantity, valid_hash) }
|
13
13
|
|
14
14
|
it_behaves_like 'DocumentRow', valid_hash
|
@@ -5,28 +5,19 @@ require 'fortnox/api/types'
|
|
5
5
|
require 'fortnox/api/types/required'
|
6
6
|
|
7
7
|
describe Fortnox::API::Types::Required, type: :type do
|
8
|
-
|
9
|
-
class TestClass < Dry::Struct
|
10
|
-
end
|
11
|
-
end
|
8
|
+
before { stub_const('TestClass', Class.new(Dry::Struct)) }
|
12
9
|
|
13
10
|
shared_examples_for 'required attribute' do |_type|
|
14
|
-
|
15
|
-
|
16
|
-
let(:error_message) do
|
17
|
-
"[#{TestClass}.new] #{:required_attribute.inspect} is missing in Hash input"
|
18
|
-
end
|
19
|
-
|
20
|
-
it 'raises an error' do
|
21
|
-
is_expected.to raise_error(Dry::Struct::Error, error_message)
|
22
|
-
end
|
11
|
+
let(:error_message) { "[#{TestClass}.new] #{:required_attribute.inspect} is missing in Hash input" }
|
12
|
+
specify { expect { TestClass.new({}) }.to raise_error(Dry::Struct::Error, error_message) }
|
23
13
|
end
|
24
14
|
|
25
15
|
describe 'String' do
|
26
16
|
before do
|
27
|
-
|
17
|
+
test_class = Class.new(TestClass) do
|
28
18
|
attribute :required_attribute, Fortnox::API::Types::Required::String
|
29
19
|
end
|
20
|
+
stub_const('TestClass', test_class)
|
30
21
|
end
|
31
22
|
|
32
23
|
include_examples 'required attribute', String
|
@@ -34,9 +25,10 @@ describe Fortnox::API::Types::Required, type: :type do
|
|
34
25
|
|
35
26
|
describe 'Float' do
|
36
27
|
before do
|
37
|
-
|
28
|
+
test_class = Class.new(TestClass) do
|
38
29
|
attribute :required_attribute, Fortnox::API::Types::Required::Float
|
39
30
|
end
|
31
|
+
stub_const('TestClass', test_class)
|
40
32
|
end
|
41
33
|
|
42
34
|
include_examples 'required attribute', Float
|
@@ -0,0 +1,57 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'spec_helper'
|
4
|
+
require 'fortnox/api/types'
|
5
|
+
require 'fortnox/api/types/examples/types'
|
6
|
+
|
7
|
+
describe Fortnox::API::Types do
|
8
|
+
let(:klass) { Fortnox::API::Types::SalesAccount }
|
9
|
+
|
10
|
+
describe 'SalesAccount' do
|
11
|
+
context 'when created with nil' do
|
12
|
+
subject { klass[nil] }
|
13
|
+
|
14
|
+
it { is_expected.to be_nil }
|
15
|
+
end
|
16
|
+
|
17
|
+
context 'when created with empty string' do
|
18
|
+
subject { klass[''] }
|
19
|
+
|
20
|
+
it { is_expected.to be_nil }
|
21
|
+
end
|
22
|
+
|
23
|
+
[-12_345, -1_234, -123, -1, 0, 1, 12, 123, 1_234.0, 12_345].each do |invalid_number|
|
24
|
+
context "with #{invalid_number}" do
|
25
|
+
it 'raises ConstraintError' do
|
26
|
+
expect { klass[invalid_number] }.to raise_error(Dry::Types::ConstraintError)
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
context 'with 4 digits' do
|
32
|
+
subject { klass[1_234] }
|
33
|
+
|
34
|
+
it { is_expected.to eq('1234') }
|
35
|
+
end
|
36
|
+
|
37
|
+
context 'when created with valid string' do
|
38
|
+
include_examples 'equals input', '1234'
|
39
|
+
end
|
40
|
+
|
41
|
+
context 'when created with a too many digits' do
|
42
|
+
include_examples 'raises ConstraintError', '12345'
|
43
|
+
end
|
44
|
+
|
45
|
+
context 'when created with too few digits' do
|
46
|
+
include_examples 'raises ConstraintError', '123'
|
47
|
+
end
|
48
|
+
|
49
|
+
context 'when created with a negative number' do
|
50
|
+
include_examples 'raises ConstraintError', '-1234'
|
51
|
+
end
|
52
|
+
|
53
|
+
context 'when created with characters' do
|
54
|
+
include_examples 'raises ConstraintError', 'abcd'
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
data/spec/fortnox/api_spec.rb
CHANGED
@@ -4,14 +4,7 @@ require 'spec_helper'
|
|
4
4
|
require 'fortnox/api'
|
5
5
|
|
6
6
|
describe Fortnox::API do
|
7
|
-
before
|
8
|
-
module Fortnox
|
9
|
-
module API
|
10
|
-
class TestBase
|
11
|
-
end
|
12
|
-
end
|
13
|
-
end
|
14
|
-
end
|
7
|
+
before { stub_const('Fortnox::API::TestBase', Class.new) }
|
15
8
|
|
16
9
|
describe 'configuration defaults' do
|
17
10
|
subject(:config_value) { described_class.config[config_key] }
|
@@ -21,37 +14,13 @@ describe Fortnox::API do
|
|
21
14
|
describe 'base_url' do
|
22
15
|
let(:config_key) { :base_url }
|
23
16
|
|
24
|
-
it { is_expected.to
|
25
|
-
end
|
26
|
-
|
27
|
-
describe 'client_secret' do
|
28
|
-
let(:config_key) { :client_secret }
|
29
|
-
|
30
|
-
it { is_expected.to be_nil }
|
31
|
-
end
|
32
|
-
|
33
|
-
describe 'access_token' do
|
34
|
-
let(:config_key) { :access_token }
|
35
|
-
|
36
|
-
it { is_expected.to be_nil }
|
37
|
-
end
|
38
|
-
|
39
|
-
describe 'access_tokens' do
|
40
|
-
let(:config_key) { :access_tokens }
|
41
|
-
|
42
|
-
it { is_expected.to be_nil }
|
17
|
+
it { is_expected.to eq 'https://api.fortnox.se/3/' }
|
43
18
|
end
|
44
19
|
|
45
|
-
describe '
|
46
|
-
let(:config_key) { :
|
47
|
-
|
48
|
-
it { is_expected.to eql({}) }
|
49
|
-
end
|
50
|
-
|
51
|
-
describe 'debugging' do
|
52
|
-
let(:config_key) { :debugging }
|
20
|
+
describe 'token_url' do
|
21
|
+
let(:config_key) { :token_url }
|
53
22
|
|
54
|
-
it { is_expected.to
|
23
|
+
it { is_expected.to eq 'https://apps.fortnox.se/oauth-v1/token' }
|
55
24
|
end
|
56
25
|
|
57
26
|
describe 'logger' do
|
@@ -67,104 +36,30 @@ describe Fortnox::API do
|
|
67
36
|
end
|
68
37
|
end
|
69
38
|
|
70
|
-
describe 'access_token' do
|
71
|
-
context 'when set to a String' do
|
72
|
-
subject { described_class.config.access_token }
|
73
|
-
|
74
|
-
before { described_class.configure { |config| config.access_token = value } }
|
75
|
-
let(:value) { '12345' }
|
76
|
-
|
77
|
-
it { is_expected.to eql value }
|
78
|
-
end
|
79
|
-
|
80
|
-
shared_examples_for 'invalid argument' do
|
81
|
-
subject { -> { described_class.configure { |config| config.access_token = value } } }
|
82
|
-
|
83
|
-
it { is_expected.to raise_error(ArgumentError, /expected a String/) }
|
84
|
-
end
|
85
|
-
|
86
|
-
context 'when set to a Hash' do
|
87
|
-
include_examples 'invalid argument' do
|
88
|
-
let(:value) { { a: '123' } }
|
89
|
-
end
|
90
|
-
end
|
91
|
-
|
92
|
-
context 'when set to an Array' do
|
93
|
-
include_examples 'invalid argument' do
|
94
|
-
let(:value) { %w[123 456] }
|
95
|
-
end
|
96
|
-
end
|
97
|
-
end
|
98
|
-
|
99
|
-
describe 'access_tokens' do
|
100
|
-
context 'when set to a String' do
|
101
|
-
subject { -> { described_class.configure { |config| config.access_tokens = '12345' } } }
|
102
|
-
|
103
|
-
it { is_expected.to raise_error(ArgumentError, /expected a Hash or an Array/) }
|
104
|
-
end
|
105
|
-
|
106
|
-
shared_examples_for 'valid argument' do
|
107
|
-
subject { described_class.config.access_tokens }
|
108
|
-
|
109
|
-
before { described_class.configure { |config| config.access_tokens = value } }
|
110
|
-
|
111
|
-
it { is_expected.to eql(value) }
|
112
|
-
end
|
113
|
-
|
114
|
-
context 'when set to a Hash' do
|
115
|
-
include_examples 'valid argument' do
|
116
|
-
let(:value) { { a: '123', b: '456' } }
|
117
|
-
end
|
118
|
-
end
|
119
|
-
|
120
|
-
context 'when set to an Array' do
|
121
|
-
include_examples 'valid argument' do
|
122
|
-
let(:value) { %w[123 456] }
|
123
|
-
end
|
124
|
-
end
|
125
|
-
end
|
126
|
-
|
127
|
-
describe 'token_store' do
|
128
|
-
subject { described_class.config[:token_store] }
|
129
|
-
|
130
|
-
context 'when access_token set' do
|
131
|
-
before { described_class.configure { |config| config.access_token = access_token } }
|
132
|
-
let(:access_token) { '12345' }
|
133
|
-
|
134
|
-
it { is_expected.to eql(default: access_token) }
|
135
|
-
end
|
136
|
-
|
137
|
-
context 'when access_tokens is' do
|
138
|
-
before { described_class.configure { |config| config.access_tokens = access_tokens } }
|
139
|
-
|
140
|
-
describe 'a Hash' do
|
141
|
-
let(:access_tokens) { { a: '123', b: '456' } }
|
142
|
-
|
143
|
-
it { is_expected.to eql(access_tokens) }
|
144
|
-
end
|
145
|
-
|
146
|
-
describe 'an Array' do
|
147
|
-
let(:access_tokens) { %w[123 456] }
|
148
|
-
|
149
|
-
it { is_expected.to eql(default: access_tokens) }
|
150
|
-
end
|
151
|
-
end
|
152
|
-
end
|
153
|
-
|
154
39
|
describe 'readers for' do
|
155
40
|
describe 'debugging' do
|
156
|
-
subject { described_class.debugging }
|
41
|
+
subject(:debugging) { described_class.debugging }
|
157
42
|
|
158
43
|
it 'is available' do
|
159
|
-
|
44
|
+
expect(debugging).to be false
|
160
45
|
end
|
161
46
|
end
|
162
47
|
|
163
48
|
describe 'logger' do
|
164
|
-
subject { described_class.logger }
|
49
|
+
subject(:logger) { described_class.logger }
|
165
50
|
|
166
51
|
it 'is available' do
|
167
|
-
|
52
|
+
expect(logger).to be_a Logger
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
describe '#access_token' do
|
58
|
+
context 'when token is set' do
|
59
|
+
before { described_class.access_token = 'an_access_token' }
|
60
|
+
|
61
|
+
it 'returns the token' do
|
62
|
+
expect(described_class.access_token).to eq('an_access_token')
|
168
63
|
end
|
169
64
|
end
|
170
65
|
end
|
data/spec/spec_helper.rb
CHANGED
@@ -20,26 +20,12 @@ RSpec.configure do |config|
|
|
20
20
|
config.extend Helpers # Allow access to helpers in describe and context blocks
|
21
21
|
config.include Helpers # Allow access to helpers in it and let blocks
|
22
22
|
|
23
|
-
config.include Helpers::Repositories, integration: true
|
24
23
|
config.include Matchers::Type, type: :type
|
25
24
|
|
26
25
|
config.order = 'random'
|
27
26
|
|
28
27
|
WebMock.disable_net_connect!(allow: 'codeclimate.com')
|
29
28
|
|
30
|
-
config.before do
|
31
|
-
module Test
|
32
|
-
def self.remove_constants
|
33
|
-
constants.each { |const| remove_const(const) }
|
34
|
-
self
|
35
|
-
end
|
36
|
-
end
|
37
|
-
end
|
38
|
-
|
39
|
-
config.after do
|
40
|
-
Object.send(:remove_const, Test.remove_constants.name)
|
41
|
-
end
|
42
|
-
|
43
29
|
# Reset configuration after each test run
|
44
30
|
config.after do
|
45
31
|
Fortnox::API::DEFAULT_CONFIGURATION.each do |key, value|
|