ledger_sync-xero 0.1.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.
@@ -0,0 +1,20 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Contact
6
+ module Operations
7
+ class Update < Xero::Operation::Update
8
+ class Contract < LedgerSync::Ledgers::Contract
9
+ params do
10
+ required(:external_id).filled(:string)
11
+ required(:ledger_id).filled(:string)
12
+ required(:Name).maybe(:string)
13
+ required(:EmailAddress).maybe(:string)
14
+ end
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,14 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Contact
6
+ class Serializer < Xero::Serializer
7
+ attribute 'ContactID',
8
+ resource_attribute: :ledger_id
9
+ attribute :Name
10
+ attribute :EmailAddress
11
+ end
12
+ end
13
+ end
14
+ end
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Deserializer < LedgerSync::Deserializer
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,100 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class OAuthClient
6
+ AUTHORIZE_URL = 'https://login.xero.com/identity/connect/authorize'
7
+ RESPONSE_TYPE = 'code'
8
+ SITE_URL = 'https://api.xero.com/api.xro/2.0'
9
+ TOKEN_URL = 'https://identity.xero.com/connect/token'
10
+ SCOPE = %w[
11
+ offline_access
12
+ openid
13
+ profile
14
+ email
15
+ accounting.transactions
16
+ accounting.contacts
17
+ ].freeze
18
+
19
+ class RedirectURIParser
20
+ attr_reader :uri
21
+
22
+ def initialize(uri:)
23
+ @uri = uri
24
+ end
25
+
26
+ def code
27
+ @code ||= query.fetch('code').first
28
+ end
29
+
30
+ def parsed_uri
31
+ @parsed_uri = URI.parse(uri)
32
+ end
33
+
34
+ def query
35
+ @query ||= CGI.parse(parsed_uri.query)
36
+ end
37
+
38
+ def realm_id
39
+ @realm_id ||= query.fetch('realmId').first
40
+ end
41
+
42
+ def redirect_uri
43
+ @redirect_uri ||= begin
44
+ use_uri = parsed_uri.dup
45
+ use_uri.fragment = nil
46
+ use_uri.query = nil
47
+ ret = use_uri.to_s
48
+ ret = ret[0..-2] if ret.end_with?('/')
49
+ ret
50
+ end
51
+ end
52
+ end
53
+
54
+ attr_reader :client_id,
55
+ :client_secret
56
+
57
+ def initialize(client_id:, client_secret:)
58
+ @client_id = client_id
59
+ @client_secret = client_secret
60
+ end
61
+
62
+ def authorization_url(redirect_uri:)
63
+ client.auth_code.authorize_url(
64
+ redirect_uri: redirect_uri,
65
+ response_type: RESPONSE_TYPE,
66
+ state: SecureRandom.hex(12),
67
+ scope: SCOPE.join(' ')
68
+ )
69
+ end
70
+
71
+ def auth_code
72
+ client.auth_code
73
+ end
74
+
75
+ def client
76
+ @client ||= OAuth2::Client.new(
77
+ client_id,
78
+ client_secret,
79
+ authorize_url: AUTHORIZE_URL,
80
+ site: SITE_URL,
81
+ token_url: TOKEN_URL
82
+ )
83
+ end
84
+
85
+ def get_token(code:, redirect_uri:)
86
+ auth_code.get_token(
87
+ code,
88
+ redirect_uri: redirect_uri
89
+ )
90
+ end
91
+
92
+ def self.new_from_env
93
+ new(
94
+ client_id: ENV.fetch('XERO_CLIENT_ID'),
95
+ client_secret: ENV.fetch('XERO_CLIENT_SECRET')
96
+ )
97
+ end
98
+ end
99
+ end
100
+ end
@@ -0,0 +1,57 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Operation
6
+ module Mixin
7
+ def self.included(base)
8
+ base.include Ledgers::Operation::Mixin
9
+ base.include InstanceMethods # To ensure these override parent methods
10
+ end
11
+
12
+ module InstanceMethods
13
+ def deserialized_resource(response:)
14
+ deserializer.deserialize(
15
+ hash: response,
16
+ resource: resource
17
+ )
18
+ end
19
+
20
+ def ledger_resource_path
21
+ @ledger_resource_path ||= "#{ledger_resource_type_for_path}/#{resource.ledger_id}"
22
+ end
23
+
24
+ def ledger_resource_type_for_path
25
+ Util::StringHelpers.camelcase(xero_resource_type.pluralize)
26
+ end
27
+
28
+ def response_to_operation_result(response:)
29
+ if response.success?
30
+ success(
31
+ resource: deserialized_resource(
32
+ response: response.body[ledger_resource_type_for_path.to_s.capitalize]&.first
33
+ ),
34
+ response: response
35
+ )
36
+ else
37
+ failure
38
+ # TODO: implement failure handler
39
+ end
40
+ end
41
+
42
+ def perform
43
+ super
44
+ rescue LedgerSync::Error::OperationError, OAuth2::Error => e
45
+ failure(e)
46
+ ensure
47
+ client.update_secrets_in_dotenv
48
+ end
49
+
50
+ def xero_resource_type
51
+ @xero_resource_type ||= client.class.ledger_resource_type_for(resource_class: resource.class)
52
+ end
53
+ end
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative '../operation'
4
+
5
+ module LedgerSync
6
+ module Xero
7
+ class Operation
8
+ class Create
9
+ include Xero::Operation::Mixin
10
+
11
+ private
12
+
13
+ def operate
14
+ response_to_operation_result(
15
+ response: client.post(
16
+ path: ledger_resource_type_for_path,
17
+ payload: [
18
+ serializer.serialize(resource: resource)
19
+ ]
20
+ )
21
+ )
22
+ end
23
+ end
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,23 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative '../operation'
4
+
5
+ module LedgerSync
6
+ module Xero
7
+ class Operation
8
+ class Find
9
+ include Xero::Operation::Mixin
10
+
11
+ private
12
+
13
+ def operate
14
+ response_to_operation_result(
15
+ response: client.find(
16
+ path: ledger_resource_path
17
+ )
18
+ )
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,26 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative '../operation'
4
+
5
+ module LedgerSync
6
+ module Xero
7
+ class Operation
8
+ class Update
9
+ include Xero::Operation::Mixin
10
+
11
+ private
12
+
13
+ def operate
14
+ response_to_operation_result(
15
+ response: client.post(
16
+ path: ledger_resource_type_for_path,
17
+ payload: [
18
+ serializer.serialize(resource: resource)
19
+ ]
20
+ )
21
+ )
22
+ end
23
+ end
24
+ end
25
+ end
26
+ end
@@ -0,0 +1,57 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Request < Ledgers::Request
6
+ attr_reader :client
7
+
8
+ def initialize(*args, client:, **keywords)
9
+ @client = client
10
+ super(*args, **keywords)
11
+ end
12
+
13
+ def perform
14
+ raise 'Request already performed' if performed?
15
+
16
+ @response = generate_response(
17
+ body: body,
18
+ headers: headers,
19
+ method: method,
20
+ url: url
21
+ )
22
+ ensure
23
+ @performed = true
24
+ end
25
+
26
+ def refresh!
27
+ oauth.refresh!
28
+ rescue OAuth2::Error => e
29
+ raise e # parse_error(error: e)
30
+ end
31
+
32
+ private
33
+
34
+ def generate_response(body:, headers:, method:, url:)
35
+ oauth_response = oauth.send(
36
+ method,
37
+ url,
38
+ body: (body.to_json unless body.nil?),
39
+ headers: headers
40
+ )
41
+
42
+ LedgerSync::Ledgers::Response.new_from_oauth_response(
43
+ oauth_response: oauth_response,
44
+ request: self
45
+ )
46
+ end
47
+
48
+ def oauth
49
+ client.oauth
50
+ end
51
+
52
+ def oauth_client
53
+ client.oauth_client
54
+ end
55
+ end
56
+ end
57
+ end
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Resource < LedgerSync::Resource
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,10 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Contact < Xero::Resource
6
+ attribute :Name, type: Type::String
7
+ attribute :EmailAddress, type: Type::String
8
+ end
9
+ end
10
+ end
@@ -0,0 +1,8 @@
1
+ # frozen_string_literal: true
2
+
3
+ module LedgerSync
4
+ module Xero
5
+ class Serializer < LedgerSync::Serializer
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ # :nocov:
4
+ module LedgerSync
5
+ module Xero
6
+ VERSION = '0.1.0'
7
+
8
+ def self.version(args = {})
9
+ pre = args.fetch(:pre, false)
10
+
11
+ if !pre && (!ENV['TRAVIS'] || ENV.fetch('TRAVIS_TAG', '') != '')
12
+ VERSION
13
+ else
14
+ "#{VERSION}.pre.#{ENV['TRAVIS_BUILD_NUMBER']}"
15
+ end
16
+ end
17
+ end
18
+ end
19
+ # :nocov:
metadata ADDED
@@ -0,0 +1,301 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: ledger_sync-xero
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Modern Treasury
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2020-11-06 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: awesome_print
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :development
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '0'
27
+ - !ruby/object:Gem::Dependency
28
+ name: bump
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: 0.9.0
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: 0.9.0
41
+ - !ruby/object:Gem::Dependency
42
+ name: bundler
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - "~>"
46
+ - !ruby/object:Gem::Version
47
+ version: '2.1'
48
+ type: :development
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - "~>"
53
+ - !ruby/object:Gem::Version
54
+ version: '2.1'
55
+ - !ruby/object:Gem::Dependency
56
+ name: byebug
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :development
63
+ prerelease: false
64
+ version_requirements: !ruby/object:Gem::Requirement
65
+ requirements:
66
+ - - ">="
67
+ - !ruby/object:Gem::Version
68
+ version: '0'
69
+ - !ruby/object:Gem::Dependency
70
+ name: climate_control
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: coveralls
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - "~>"
88
+ - !ruby/object:Gem::Version
89
+ version: 0.8.23
90
+ type: :development
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - "~>"
95
+ - !ruby/object:Gem::Version
96
+ version: 0.8.23
97
+ - !ruby/object:Gem::Dependency
98
+ name: factory_bot
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - "~>"
102
+ - !ruby/object:Gem::Version
103
+ version: 6.1.0
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - "~>"
109
+ - !ruby/object:Gem::Version
110
+ version: 6.1.0
111
+ - !ruby/object:Gem::Dependency
112
+ name: overcommit
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - "~>"
116
+ - !ruby/object:Gem::Version
117
+ version: 0.57.0
118
+ type: :development
119
+ prerelease: false
120
+ version_requirements: !ruby/object:Gem::Requirement
121
+ requirements:
122
+ - - "~>"
123
+ - !ruby/object:Gem::Version
124
+ version: 0.57.0
125
+ - !ruby/object:Gem::Dependency
126
+ name: rake
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - "~>"
130
+ - !ruby/object:Gem::Version
131
+ version: '13.0'
132
+ type: :development
133
+ prerelease: false
134
+ version_requirements: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - "~>"
137
+ - !ruby/object:Gem::Version
138
+ version: '13.0'
139
+ - !ruby/object:Gem::Dependency
140
+ name: rspec
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - "~>"
144
+ - !ruby/object:Gem::Version
145
+ version: '3.2'
146
+ type: :development
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - "~>"
151
+ - !ruby/object:Gem::Version
152
+ version: '3.2'
153
+ - !ruby/object:Gem::Dependency
154
+ name: rubocop
155
+ requirement: !ruby/object:Gem::Requirement
156
+ requirements:
157
+ - - ">="
158
+ - !ruby/object:Gem::Version
159
+ version: '0'
160
+ type: :development
161
+ prerelease: false
162
+ version_requirements: !ruby/object:Gem::Requirement
163
+ requirements:
164
+ - - ">="
165
+ - !ruby/object:Gem::Version
166
+ version: '0'
167
+ - !ruby/object:Gem::Dependency
168
+ name: webmock
169
+ requirement: !ruby/object:Gem::Requirement
170
+ requirements:
171
+ - - ">="
172
+ - !ruby/object:Gem::Version
173
+ version: '0'
174
+ type: :development
175
+ prerelease: false
176
+ version_requirements: !ruby/object:Gem::Requirement
177
+ requirements:
178
+ - - ">="
179
+ - !ruby/object:Gem::Version
180
+ version: '0'
181
+ - !ruby/object:Gem::Dependency
182
+ name: dotenv
183
+ requirement: !ruby/object:Gem::Requirement
184
+ requirements:
185
+ - - ">="
186
+ - !ruby/object:Gem::Version
187
+ version: '0'
188
+ type: :runtime
189
+ prerelease: false
190
+ version_requirements: !ruby/object:Gem::Requirement
191
+ requirements:
192
+ - - ">="
193
+ - !ruby/object:Gem::Version
194
+ version: '0'
195
+ - !ruby/object:Gem::Dependency
196
+ name: ledger_sync
197
+ requirement: !ruby/object:Gem::Requirement
198
+ requirements:
199
+ - - ">="
200
+ - !ruby/object:Gem::Version
201
+ version: '0'
202
+ type: :runtime
203
+ prerelease: false
204
+ version_requirements: !ruby/object:Gem::Requirement
205
+ requirements:
206
+ - - ">="
207
+ - !ruby/object:Gem::Version
208
+ version: '0'
209
+ - !ruby/object:Gem::Dependency
210
+ name: nokogiri
211
+ requirement: !ruby/object:Gem::Requirement
212
+ requirements:
213
+ - - ">="
214
+ - !ruby/object:Gem::Version
215
+ version: '0'
216
+ type: :runtime
217
+ prerelease: false
218
+ version_requirements: !ruby/object:Gem::Requirement
219
+ requirements:
220
+ - - ">="
221
+ - !ruby/object:Gem::Version
222
+ version: '0'
223
+ - !ruby/object:Gem::Dependency
224
+ name: oauth2
225
+ requirement: !ruby/object:Gem::Requirement
226
+ requirements:
227
+ - - ">="
228
+ - !ruby/object:Gem::Version
229
+ version: '0'
230
+ type: :runtime
231
+ prerelease: false
232
+ version_requirements: !ruby/object:Gem::Requirement
233
+ requirements:
234
+ - - ">="
235
+ - !ruby/object:Gem::Version
236
+ version: '0'
237
+ description: LedgerSync is a simple library that allows you to sync common objects
238
+ to popular accounting software like QuickBooks Online, Xero, NetSuite, etc.
239
+ email:
240
+ - ledgersync@moderntreasury.com
241
+ executables: []
242
+ extensions: []
243
+ extra_rdoc_files: []
244
+ files:
245
+ - ".coveralls.yml"
246
+ - ".env.test"
247
+ - ".gitignore"
248
+ - ".overcommit.yml"
249
+ - ".rubocop.yml"
250
+ - ".rubocop_todo.yml"
251
+ - ".travis.yml"
252
+ - Gemfile
253
+ - Gemfile.lock
254
+ - README.md
255
+ - Rakefile
256
+ - bin/console
257
+ - bin/setup
258
+ - bin/xero_oauth_server.rb
259
+ - ledger_sync-xero.gemspec
260
+ - lib/ledger_sync/xero.rb
261
+ - lib/ledger_sync/xero/client.rb
262
+ - lib/ledger_sync/xero/config.rb
263
+ - lib/ledger_sync/xero/contact/deserializer.rb
264
+ - lib/ledger_sync/xero/contact/operations/create.rb
265
+ - lib/ledger_sync/xero/contact/operations/find.rb
266
+ - lib/ledger_sync/xero/contact/operations/update.rb
267
+ - lib/ledger_sync/xero/contact/serializer.rb
268
+ - lib/ledger_sync/xero/deserializer.rb
269
+ - lib/ledger_sync/xero/oauth_client.rb
270
+ - lib/ledger_sync/xero/operation.rb
271
+ - lib/ledger_sync/xero/operation/create.rb
272
+ - lib/ledger_sync/xero/operation/find.rb
273
+ - lib/ledger_sync/xero/operation/update.rb
274
+ - lib/ledger_sync/xero/request.rb
275
+ - lib/ledger_sync/xero/resource.rb
276
+ - lib/ledger_sync/xero/resources/contact.rb
277
+ - lib/ledger_sync/xero/serializer.rb
278
+ - lib/ledger_sync/xero/version.rb
279
+ homepage: https://www.ledgersync.dev
280
+ licenses: []
281
+ metadata: {}
282
+ post_install_message:
283
+ rdoc_options: []
284
+ require_paths:
285
+ - lib
286
+ required_ruby_version: !ruby/object:Gem::Requirement
287
+ requirements:
288
+ - - ">="
289
+ - !ruby/object:Gem::Version
290
+ version: 2.5.8
291
+ required_rubygems_version: !ruby/object:Gem::Requirement
292
+ requirements:
293
+ - - ">="
294
+ - !ruby/object:Gem::Version
295
+ version: '0'
296
+ requirements: []
297
+ rubygems_version: 3.0.3
298
+ signing_key:
299
+ specification_version: 4
300
+ summary: Sync common objects to accounting software.
301
+ test_files: []