patches 2.0.0 → 2.4.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/.gitignore +1 -0
- data/docs/usage.md +49 -5
- data/lib/generators/patches/patch_generator.rb +3 -0
- data/lib/generators/patches/templates/patch_spec.rb.erb +8 -0
- data/lib/patches.rb +3 -1
- data/lib/patches/config.rb +47 -8
- data/lib/patches/notifier.rb +34 -6
- data/lib/patches/tenant_run_concern.rb +8 -0
- data/lib/patches/tenant_runner.rb +12 -8
- data/lib/patches/tenant_worker.rb +12 -0
- data/lib/patches/version.rb +1 -1
- data/lib/patches/worker.rb +1 -1
- data/lib/tasks/patches.rake +1 -1
- data/patches.gemspec +4 -0
- data/script/buildkite.sh +3 -1
- metadata +62 -3
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
|
-
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 44dfdb47ded3bbc59e4f3e6e60608368cd63018a2424bcc209800f882444e0d9
|
4
|
+
data.tar.gz: 4f197d97c612321564aba0b1e15fbfefa7eb288affdd20fdeb556d6bea8a1de3
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 5ff6266eee49006debf847b631eebe295b33c9bf3d5f50c528fb19db5e2d137d1158ee89a21fc7f108b623e236a0dacbdaefae444ce3944f3bc951b9420fdc2a
|
7
|
+
data.tar.gz: dd761de25dd47a74c9c7123352fda506a38117278ed59c254e4511573a2748172702be682a3fc7e1fe9b54f854052827b428f4c01e563771f222d3322db33cf1
|
data/.gitignore
CHANGED
data/docs/usage.md
CHANGED
@@ -5,7 +5,7 @@
|
|
5
5
|
Add patches to the project Gemfile
|
6
6
|
|
7
7
|
```ruby
|
8
|
-
gem 'patches', '~> 2.
|
8
|
+
gem 'patches', '~> 2.4.0'
|
9
9
|
```
|
10
10
|
|
11
11
|
Install the database migration
|
@@ -22,7 +22,9 @@ bundle exec rake db:migrate
|
|
22
22
|
|
23
23
|
## Configuration
|
24
24
|
|
25
|
-
If you would like to run the patches asynchronously, or would like them to notify
|
25
|
+
If you would like to run the patches asynchronously, or would like them to notify
|
26
|
+
your hipchat room or Slack channel when they fail or succeed, you need to set up
|
27
|
+
an initializer to set those options.
|
26
28
|
|
27
29
|
```Ruby
|
28
30
|
Patches::Config.configure do |config|
|
@@ -32,11 +34,34 @@ Patches::Config.configure do |config|
|
|
32
34
|
config.hipchat_options = {
|
33
35
|
api_token: ENV['HIPCHAT_TOKEN'],
|
34
36
|
room: ENV['HIPCHAT_ROOM'],
|
35
|
-
user: ENV['HIPCHAT_USERNAME'] #maximum of 15 characters
|
37
|
+
user: ENV['HIPCHAT_USERNAME'], # maximum of 15 characters
|
38
|
+
api_version: 'v1', # optional
|
36
39
|
}
|
40
|
+
|
41
|
+
config.use_slack = true
|
42
|
+
config.slack_options = {
|
43
|
+
webhook_url: ENV['SLACK_WEBHOOK_URL'],
|
44
|
+
channel: ENV['SLACK_CHANNEL'],
|
45
|
+
username: ENV['SLACK_USER']
|
46
|
+
}
|
47
|
+
end
|
48
|
+
```
|
49
|
+
|
50
|
+
### Running patches in parallel for tenants
|
51
|
+
|
52
|
+
If you are using the Apartment gem, you can run the patches for each tenant in parallel.
|
53
|
+
Just set the config ```sidekiq_parallel``` to ```true``` and you're good to go.
|
54
|
+
|
55
|
+
```
|
56
|
+
Patches::Config.configure do |config|
|
57
|
+
config.use_sidekiq = true
|
58
|
+
config.sidekiq_parallel = true
|
37
59
|
end
|
38
60
|
```
|
39
61
|
|
62
|
+
*Note:* Make sure your sidekiq queue is able to process concurrent jobs.
|
63
|
+
You can use ```config.sidekiq_options``` to customise it.
|
64
|
+
|
40
65
|
## Creating Patches
|
41
66
|
|
42
67
|
Generate a patch
|
@@ -57,6 +82,12 @@ end
|
|
57
82
|
|
58
83
|
update the run method and then execute
|
59
84
|
|
85
|
+
Generate patch with specs
|
86
|
+
|
87
|
+
```
|
88
|
+
bundle exec rails g patches:patch PreferenceUpdate --specs=true
|
89
|
+
```
|
90
|
+
|
60
91
|
|
61
92
|
```bash
|
62
93
|
bundle exec rake patches:run
|
@@ -76,8 +107,21 @@ And then in your deploy.rb
|
|
76
107
|
after 'last_task_you_want_to_run' 'patches:run'
|
77
108
|
```
|
78
109
|
|
79
|
-
If you are using sidekiq and restarting the sidekiq process on the box
|
110
|
+
If you are using sidekiq and restarting the sidekiq process on the box
|
111
|
+
as a part of the deploy process, please make sure that the patches run task runs
|
112
|
+
after sidekiq restarts, otherwise there is no guarentee the tasks will run.
|
113
|
+
|
114
|
+
## File Download
|
115
|
+
|
116
|
+
If a patch requires data assets, you could use S3 to store the file.
|
117
|
+
If credentials are defined in env vars, as per https://docs.aws.amazon.com/cli/latest/topic/config-vars.html#id1
|
118
|
+
|
119
|
+
```
|
120
|
+
require 'aws-sdk-s3'
|
121
|
+
Aws::S3::Client.new.get_object(bucket: @bucket_name, key: filename, response_target: destination)
|
122
|
+
```
|
80
123
|
|
81
124
|
## Multitenant
|
82
125
|
|
83
|
-
Patches will detect if `Apartment` gem is installed and if there are any tenants
|
126
|
+
Patches will detect if `Apartment` gem is installed and if there are any tenants
|
127
|
+
and run the patches for each tenant
|
@@ -4,8 +4,11 @@ module Patches
|
|
4
4
|
desc 'Adds an empty patch'
|
5
5
|
source_root File.expand_path('../templates', __FILE__)
|
6
6
|
|
7
|
+
class_option :specs, type: :boolean, default: false, description: 'Generates a rspec file for the patch'
|
8
|
+
|
7
9
|
def generate_patch
|
8
10
|
template "patch.rb.erb", "db/patches/#{file_name}.rb"
|
11
|
+
template "patch_spec.rb.erb", "spec/patches/#{file_name}_spec.rb" if options['specs']
|
9
12
|
end
|
10
13
|
|
11
14
|
private
|
data/lib/patches.rb
CHANGED
@@ -20,11 +20,13 @@ module Patches
|
|
20
20
|
end
|
21
21
|
|
22
22
|
require "patches/base"
|
23
|
+
require "patches/config"
|
24
|
+
require "patches/tenant_run_concern"
|
25
|
+
require "patches/tenant_worker" if defined?(Sidekiq)
|
23
26
|
require "patches/engine" if defined?(Rails)
|
24
27
|
require "patches/patch"
|
25
28
|
require "patches/pending"
|
26
29
|
require "patches/runner"
|
27
30
|
require "patches/tenant_runner"
|
28
|
-
require "patches/config"
|
29
31
|
require "patches/notifier"
|
30
32
|
require "patches/worker" if defined?(Sidekiq)
|
data/lib/patches/config.rb
CHANGED
@@ -1,19 +1,58 @@
|
|
1
1
|
module Patches
|
2
2
|
class Config
|
3
3
|
class << self
|
4
|
-
|
5
|
-
|
6
|
-
def sidekiq_queue
|
7
|
-
@sidekiq_queue ||= 'default'
|
4
|
+
def configuration
|
5
|
+
@configuration ||= Configuration.new
|
8
6
|
end
|
9
7
|
|
10
|
-
def
|
11
|
-
@
|
8
|
+
def configuration=(config)
|
9
|
+
@configuration = config
|
12
10
|
end
|
13
11
|
|
14
12
|
def configure
|
15
|
-
yield
|
13
|
+
yield configuration
|
14
|
+
end
|
15
|
+
|
16
|
+
class Configuration
|
17
|
+
attr_accessor :use_sidekiq, :sidekiq_queue, :sidekiq_options, :use_hipchat,
|
18
|
+
:hipchat_options, :sidekiq_parallel, :use_slack, :slack_options
|
19
|
+
|
20
|
+
def initialize
|
21
|
+
@sidekiq_queue = 'default'
|
22
|
+
end
|
23
|
+
|
24
|
+
def sidekiq_options
|
25
|
+
@sidekiq_options ||= { retry: false, queue: sidekiq_queue }
|
26
|
+
end
|
27
|
+
|
28
|
+
def hipchat_api_token
|
29
|
+
hipchat_options[:api_token]
|
30
|
+
end
|
31
|
+
|
32
|
+
def hipchat_init_options
|
33
|
+
hipchat_options.except(:api_token, :room, :user)
|
34
|
+
end
|
35
|
+
|
36
|
+
def hipchat_room
|
37
|
+
hipchat_options[:room]
|
38
|
+
end
|
39
|
+
|
40
|
+
def hipchat_user
|
41
|
+
hipchat_options[:user]
|
42
|
+
end
|
43
|
+
|
44
|
+
def slack_channel
|
45
|
+
slack_options[:channel]
|
46
|
+
end
|
47
|
+
|
48
|
+
def slack_username
|
49
|
+
slack_options[:username]
|
50
|
+
end
|
51
|
+
|
52
|
+
def slack_webhook_url
|
53
|
+
slack_options[:webhook_url]
|
54
|
+
end
|
16
55
|
end
|
17
56
|
end
|
18
57
|
end
|
19
|
-
end
|
58
|
+
end
|
data/lib/patches/notifier.rb
CHANGED
@@ -1,34 +1,62 @@
|
|
1
|
+
require 'slack-notifier'
|
2
|
+
|
1
3
|
class Patches::Notifier
|
2
4
|
class << self
|
3
5
|
def notify_success(patches)
|
4
6
|
send_hipchat_message(success_message(patches), color: 'green')
|
7
|
+
send_slack_message(success_message(patches), 'good')
|
5
8
|
end
|
6
9
|
|
7
10
|
def notify_failure(patch_path, error)
|
8
11
|
send_hipchat_message(failure_message(patch_path, error), color: 'red')
|
12
|
+
send_slack_message(failure_message(patch_path, error), 'danger')
|
9
13
|
end
|
10
14
|
|
11
15
|
def success_message(patches)
|
12
|
-
message = "#{patches.count} patches succeeded"
|
16
|
+
message = "#{environment_prefix}#{patches.count} patches succeeded"
|
13
17
|
append_tenant_message(message)
|
14
18
|
end
|
15
19
|
|
16
20
|
def failure_message(patch_path, error)
|
17
|
-
|
21
|
+
details = "#{Pathname.new(patch_path).basename} failed with error: #{error}"
|
22
|
+
message = "#{environment_prefix}Error applying patch: #{details}"
|
18
23
|
append_tenant_message(message)
|
19
24
|
end
|
20
25
|
|
26
|
+
def environment_prefix
|
27
|
+
"[#{Rails.env.upcase}] " if defined?(Rails)
|
28
|
+
end
|
29
|
+
|
21
30
|
def append_tenant_message(message)
|
22
31
|
message = message + " for tenant: #{Apartment::Tenant.current}" if defined?(Apartment)
|
23
32
|
message
|
24
33
|
end
|
25
34
|
|
35
|
+
def send_hipchat_message(message, options)
|
36
|
+
return unless defined?(HipChat) && config.use_hipchat
|
37
|
+
|
38
|
+
client = HipChat::Client.new(config.hipchat_api_token, config.hipchat_init_options)
|
39
|
+
room = client[config.hipchat_room]
|
40
|
+
room.send(config.hipchat_user, message, options)
|
41
|
+
end
|
42
|
+
|
43
|
+
def send_slack_message(message, color)
|
44
|
+
return unless defined?(Slack) && config.use_slack
|
45
|
+
|
46
|
+
notifier = Slack::Notifier.new(
|
47
|
+
config.slack_webhook_url,
|
48
|
+
channel: config.slack_channel,
|
49
|
+
username: config.slack_username)
|
50
|
+
|
51
|
+
payload = { attachments: [{ color: color, text: message }] }
|
52
|
+
|
53
|
+
notifier.post payload
|
54
|
+
end
|
55
|
+
|
26
56
|
private
|
27
57
|
|
28
|
-
def
|
29
|
-
|
30
|
-
HipChat::Client.new(Patches::Config.hipchat_options[:api_token])[Patches::Config.hipchat_options[:room]].send(Patches::Config.hipchat_options[:user], message, options)
|
31
|
-
end
|
58
|
+
def config
|
59
|
+
Patches::Config.configuration
|
32
60
|
end
|
33
61
|
end
|
34
62
|
end
|
@@ -1,4 +1,5 @@
|
|
1
1
|
class Patches::TenantRunner
|
2
|
+
include Patches::TenantRunConcern
|
2
3
|
attr_accessor :path
|
3
4
|
|
4
5
|
def initialize(path: nil, tenants: nil)
|
@@ -8,19 +9,22 @@ class Patches::TenantRunner
|
|
8
9
|
|
9
10
|
def perform
|
10
11
|
Patches.logger.info("Patches tenant runner for: #{tenants.join(',')}")
|
11
|
-
|
12
12
|
tenants.each do |tenant|
|
13
|
-
|
14
|
-
|
15
|
-
|
13
|
+
if parallel?
|
14
|
+
Patches::TenantWorker.perform_async(tenant, path)
|
15
|
+
else
|
16
|
+
run(tenant, path)
|
17
|
+
end
|
16
18
|
end
|
17
19
|
end
|
18
20
|
|
19
|
-
def build
|
20
|
-
Patches::Runner.new(path)
|
21
|
-
end
|
22
|
-
|
23
21
|
def tenants
|
24
22
|
@tenants ||= (Apartment.tenant_names || [])
|
25
23
|
end
|
24
|
+
|
25
|
+
private
|
26
|
+
|
27
|
+
def parallel?
|
28
|
+
Patches::Config.configuration.sidekiq_parallel
|
29
|
+
end
|
26
30
|
end
|
data/lib/patches/version.rb
CHANGED
data/lib/patches/worker.rb
CHANGED
data/lib/tasks/patches.rake
CHANGED
data/patches.gemspec
CHANGED
@@ -20,6 +20,7 @@ Gem::Specification.new do |spec|
|
|
20
20
|
spec.require_paths = ["lib"]
|
21
21
|
|
22
22
|
spec.add_dependency "railties", ">= 3.2"
|
23
|
+
spec.add_dependency "slack-notifier"
|
23
24
|
|
24
25
|
spec.add_development_dependency "bundler", "~> 1.8"
|
25
26
|
spec.add_development_dependency "rake", "~> 10.0"
|
@@ -34,4 +35,7 @@ Gem::Specification.new do |spec|
|
|
34
35
|
spec.add_development_dependency "codeclimate-test-reporter", "~> 0.4"
|
35
36
|
spec.add_development_dependency "pry"
|
36
37
|
spec.add_development_dependency "sidekiq", "~> 3.4.1"
|
38
|
+
spec.add_development_dependency "hipchat"
|
39
|
+
spec.add_development_dependency "webmock"
|
40
|
+
spec.add_development_dependency "byebug"
|
37
41
|
end
|
data/script/buildkite.sh
CHANGED
@@ -2,7 +2,9 @@
|
|
2
2
|
set -e
|
3
3
|
|
4
4
|
echo '--- setting ruby version'
|
5
|
-
rbenv
|
5
|
+
cd /var/lib/buildkite-agent/.rbenv/plugins/ruby-build && git pull && cd -
|
6
|
+
rbenv install 2.3.7 -s
|
7
|
+
rbenv local 2.3.7
|
6
8
|
|
7
9
|
echo '--- setting up env'
|
8
10
|
REVISION=https://github.com/$BUILDBOX_PROJECT_SLUG/commit/$BUILDBOX_COMMIT
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: patches
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 2.
|
4
|
+
version: 2.4.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- John D'Agostino
|
8
8
|
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2018-09-17 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: railties
|
@@ -24,6 +24,20 @@ dependencies:
|
|
24
24
|
- - ">="
|
25
25
|
- !ruby/object:Gem::Version
|
26
26
|
version: '3.2'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: slack-notifier
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - ">="
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '0'
|
34
|
+
type: :runtime
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - ">="
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '0'
|
27
41
|
- !ruby/object:Gem::Dependency
|
28
42
|
name: bundler
|
29
43
|
requirement: !ruby/object:Gem::Requirement
|
@@ -206,6 +220,48 @@ dependencies:
|
|
206
220
|
- - "~>"
|
207
221
|
- !ruby/object:Gem::Version
|
208
222
|
version: 3.4.1
|
223
|
+
- !ruby/object:Gem::Dependency
|
224
|
+
name: hipchat
|
225
|
+
requirement: !ruby/object:Gem::Requirement
|
226
|
+
requirements:
|
227
|
+
- - ">="
|
228
|
+
- !ruby/object:Gem::Version
|
229
|
+
version: '0'
|
230
|
+
type: :development
|
231
|
+
prerelease: false
|
232
|
+
version_requirements: !ruby/object:Gem::Requirement
|
233
|
+
requirements:
|
234
|
+
- - ">="
|
235
|
+
- !ruby/object:Gem::Version
|
236
|
+
version: '0'
|
237
|
+
- !ruby/object:Gem::Dependency
|
238
|
+
name: webmock
|
239
|
+
requirement: !ruby/object:Gem::Requirement
|
240
|
+
requirements:
|
241
|
+
- - ">="
|
242
|
+
- !ruby/object:Gem::Version
|
243
|
+
version: '0'
|
244
|
+
type: :development
|
245
|
+
prerelease: false
|
246
|
+
version_requirements: !ruby/object:Gem::Requirement
|
247
|
+
requirements:
|
248
|
+
- - ">="
|
249
|
+
- !ruby/object:Gem::Version
|
250
|
+
version: '0'
|
251
|
+
- !ruby/object:Gem::Dependency
|
252
|
+
name: byebug
|
253
|
+
requirement: !ruby/object:Gem::Requirement
|
254
|
+
requirements:
|
255
|
+
- - ">="
|
256
|
+
- !ruby/object:Gem::Version
|
257
|
+
version: '0'
|
258
|
+
type: :development
|
259
|
+
prerelease: false
|
260
|
+
version_requirements: !ruby/object:Gem::Requirement
|
261
|
+
requirements:
|
262
|
+
- - ">="
|
263
|
+
- !ruby/object:Gem::Version
|
264
|
+
version: '0'
|
209
265
|
description: A simple gem for one off tasks for example database patches
|
210
266
|
email:
|
211
267
|
- johnd@jobready.com.au
|
@@ -228,6 +284,7 @@ files:
|
|
228
284
|
- lib/generators/patches.rb
|
229
285
|
- lib/generators/patches/patch_generator.rb
|
230
286
|
- lib/generators/patches/templates/patch.rb.erb
|
287
|
+
- lib/generators/patches/templates/patch_spec.rb.erb
|
231
288
|
- lib/patches.rb
|
232
289
|
- lib/patches/base.rb
|
233
290
|
- lib/patches/capistrano.rb
|
@@ -238,7 +295,9 @@ files:
|
|
238
295
|
- lib/patches/patch.rb
|
239
296
|
- lib/patches/pending.rb
|
240
297
|
- lib/patches/runner.rb
|
298
|
+
- lib/patches/tenant_run_concern.rb
|
241
299
|
- lib/patches/tenant_runner.rb
|
300
|
+
- lib/patches/tenant_worker.rb
|
242
301
|
- lib/patches/version.rb
|
243
302
|
- lib/patches/worker.rb
|
244
303
|
- lib/tasks/patches.rake
|
@@ -264,7 +323,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
264
323
|
version: '0'
|
265
324
|
requirements: []
|
266
325
|
rubyforge_project:
|
267
|
-
rubygems_version: 2.
|
326
|
+
rubygems_version: 2.7.6
|
268
327
|
signing_key:
|
269
328
|
specification_version: 4
|
270
329
|
summary: A simple gem for one off tasks
|