aws-rotate 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,42 @@
1
+ module AwsRotate
2
+ class Keys < Base
3
+ def run
4
+ list = List.new(@options)
5
+ list.profiles.each do |profile|
6
+ next unless filter_match?(profile)
7
+
8
+ ENV['AWS_PROFILE'] = profile
9
+ update_key
10
+ end
11
+ end
12
+
13
+ def filter_match?(profile)
14
+ return true if @options[:select].nil? && @options[:reject].nil?
15
+
16
+ unless @options[:reject].nil?
17
+ reject_list = @options[:reject]
18
+ reject_list.map! { |f| Regexp.new(f) }
19
+ rejected = !!reject_list.detect do |regexp|
20
+ profile =~ regexp
21
+ end
22
+ return false if rejected
23
+ end
24
+
25
+ return true if @options[:select].nil?
26
+
27
+ select_list = @options[:select]
28
+ select_list.map! { |f| Regexp.new(f) }
29
+ selected = !!select_list.detect do |regexp|
30
+ profile =~ regexp
31
+ end
32
+ selected
33
+ end
34
+
35
+ def update_key
36
+ Key.new(@options).run
37
+ rescue Key::GetIamUserError
38
+ message = @options[:noop] ? "Will not be able to update key" : "Unable to update key"
39
+ puts "WARN: #{message} for AWS_PROFILE=#{@profile}".color(:yellow)
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,20 @@
1
+ module AwsRotate
2
+ class List < Base
3
+ def run
4
+ puts "AWS Profiles:"
5
+ puts profiles
6
+ end
7
+
8
+ def profiles
9
+ lines = IO.readlines(@credentials_path)
10
+ profiles = []
11
+ lines.each do |line|
12
+ next if line =~ /^\s*#/ # ignore comments
13
+
14
+ md = line.match(/\[(.*)\]/)
15
+ profiles << md[1] if md
16
+ end
17
+ profiles
18
+ end
19
+ end
20
+ end
@@ -0,0 +1,3 @@
1
+ module AwsRotate
2
+ VERSION = "0.1.0"
3
+ end
@@ -0,0 +1,15 @@
1
+ [profile default]
2
+ output = json
3
+ region = us-west-2
4
+
5
+ [profile parent-account]
6
+ output = json
7
+ region = us-west-2
8
+
9
+ [profile child-account]
10
+ output = json
11
+ region = us-west-2
12
+
13
+ [profile iam-account]
14
+ output = json
15
+ region = us-east-1
@@ -0,0 +1,15 @@
1
+ [default]
2
+ role_arn = arn:aws:iam::112233445566:role/Admin
3
+ source_profile = parent-account
4
+
5
+ [parent-account]
6
+ aws_access_key_id=AAAEXAMPLEAABBCCDDEE
7
+ aws_secret_access_key=AAAEXAMPLEAABBCCDDEEFFGGHHIIJJKKLLMMNNOO
8
+
9
+ [child-account]
10
+ role_arn = arn:aws:iam::112233445566:role/Admin
11
+ source_profile = parent-account
12
+
13
+ [iam-account]
14
+ aws_access_key_id=BBBEXAMPLEAABBCCDDEE
15
+ aws_secret_access_key=BBBEXAMPLEAABBCCDDEEFFGGHHIIJJKKLLMMNNOO
@@ -0,0 +1,12 @@
1
+ describe AwsRotate::CLI do
2
+ before(:all) do
3
+ @args = ""
4
+ end
5
+
6
+ describe "aws-rotate" do
7
+ it "list" do
8
+ out = execute("exe/aws-rotate list #{@args}")
9
+ expect(out).to include("AWS Profiles:")
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,52 @@
1
+ describe AwsRotate::Key do
2
+ let(:rotater) do
3
+ rotater = AwsRotate::Key.new
4
+ # The methods that are commented out have stubs at lower-levels.
5
+ # allow(rotater).to receive(:get_iam_user).and_return('tung')
6
+ allow(rotater).to receive(:check_max_keys_limit).and_return(null)
7
+ allow(rotater).to receive(:cache_access_key).and_return(cache_access_key)
8
+ # allow(rotater).to receive(:create_access_key).and_return(create_access_key)
9
+ allow(rotater).to receive(:update_aws_credentials_file).and_return(null)
10
+ # allow(rotater).to receive(:delete_old_access_key).and_return(null)
11
+
12
+ # stub out aws configure calls
13
+ allow(rotater).to receive(:aws_configure_get).and_return(null)
14
+ allow(rotater).to receive(:aws_configure_set).and_return(null)
15
+
16
+ # stub out aws clients
17
+ allow(rotater).to receive(:sts).and_return(sts)
18
+ allow(rotater).to receive(:iam).and_return(iam)
19
+
20
+ rotater
21
+ end
22
+ let(:null) { double(:null).as_null_object }
23
+ let(:cache_access_key) { nil }
24
+ let(:create_access_key) { null }
25
+
26
+ let(:iam) do
27
+ iam = double(:null).as_null_object
28
+ resp = OpenStruct.new(
29
+ access_key: OpenStruct.new(
30
+ access_key_id: "FAKE-KEY-ID",
31
+ secret_access_key: "FAKE-SECRET-ACCESS-KEY",
32
+ )
33
+ )
34
+
35
+ allow(iam).to receive(:create_access_key).and_return(resp)
36
+ iam
37
+ end
38
+
39
+ let(:sts) do
40
+ sts = double(:null).as_null_object
41
+ resp = OpenStruct.new(arn: 'arn:aws:iam::112233445566:user/tung')
42
+ allow(sts).to receive(:get_caller_identity).and_return(resp)
43
+ sts
44
+ end
45
+
46
+ describe "rotater" do
47
+ it "run" do
48
+ completed = rotater.run
49
+ expect(completed).to be true
50
+ end
51
+ end
52
+ end
@@ -0,0 +1,60 @@
1
+ describe AwsRotate::Key do
2
+ let(:keys) do
3
+ AwsRotate::Keys.new(options)
4
+ end
5
+ let(:options) { {select: select, reject: reject } }
6
+ let(:select) { nil }
7
+ let(:reject) { nil }
8
+
9
+ context "no filters" do
10
+ it "always returns true" do
11
+ result = keys.filter_match?("my-account")
12
+ expect(result).to be true
13
+ result = keys.filter_match?("whatever")
14
+ expect(result).to be true
15
+ end
16
+ end
17
+
18
+ context "select filter string match" do
19
+ let(:select) { ["my"] }
20
+ it "test" do
21
+ result = keys.filter_match?("my-account")
22
+ expect(result).to be true
23
+ result = keys.filter_match?("whatever")
24
+ expect(result).to be false
25
+ end
26
+ end
27
+
28
+ context "select filter regexp match" do
29
+ let(:select) { ["^my"] }
30
+ it "test" do
31
+ result = keys.filter_match?("my-account")
32
+ expect(result).to be true
33
+ result = keys.filter_match?("test-my-test")
34
+ expect(result).to be false
35
+ end
36
+ end
37
+
38
+ context "reject filter" do
39
+ let(:reject) { ["my"] }
40
+ it "test" do
41
+ result = keys.filter_match?("my-account")
42
+ expect(result).to be false
43
+ result = keys.filter_match?("whatever")
44
+ expect(result).to be true
45
+ end
46
+ end
47
+
48
+ context "both select and reject filters" do
49
+ let(:select) { ["my"] }
50
+ let(:reject) { ["test"] }
51
+ it "test" do
52
+ result = keys.filter_match?("my-account")
53
+ expect(result).to be true
54
+ result = keys.filter_match?("my-test-account")
55
+ expect(result).to be false
56
+ result = keys.filter_match?("whatever")
57
+ expect(result).to be false
58
+ end
59
+ end
60
+ end
@@ -0,0 +1,33 @@
1
+ ENV["TEST"] = "1"
2
+ # Ensures aws api never called. Fixture home does not contain ~/.aws/credentials
3
+ ENV['HOME'] = "spec/fixtures/home"
4
+
5
+ # CodeClimate test coverage: https://docs.codeclimate.com/docs/configuring-test-coverage
6
+ # require 'simplecov'
7
+ # SimpleCov.start
8
+
9
+ require "pp"
10
+ require "byebug"
11
+ root = File.expand_path("../", File.dirname(__FILE__))
12
+ require "#{root}/lib/aws-rotate"
13
+
14
+ require 'webmock/rspec'
15
+
16
+ module Helper
17
+ def execute(cmd)
18
+ puts "Running: #{cmd}" if show_command?
19
+ out = `#{cmd}`
20
+ puts out if show_command?
21
+ out
22
+ end
23
+
24
+ # Added SHOW_COMMAND because DEBUG is also used by other libraries like
25
+ # bundler and it shows its internal debugging logging also.
26
+ def show_command?
27
+ ENV['DEBUG'] || ENV['SHOW_COMMAND']
28
+ end
29
+ end
30
+
31
+ RSpec.configure do |c|
32
+ c.include Helper
33
+ end
metadata ADDED
@@ -0,0 +1,243 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: aws-rotate
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - Tung Nguyen
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2019-08-14 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: activesupport
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '0'
20
+ type: :runtime
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: aws-sdk-iam
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'
41
+ - !ruby/object:Gem::Dependency
42
+ name: aws-sdk-core
43
+ requirement: !ruby/object:Gem::Requirement
44
+ requirements:
45
+ - - ">="
46
+ - !ruby/object:Gem::Version
47
+ version: '0'
48
+ type: :runtime
49
+ prerelease: false
50
+ version_requirements: !ruby/object:Gem::Requirement
51
+ requirements:
52
+ - - ">="
53
+ - !ruby/object:Gem::Version
54
+ version: '0'
55
+ - !ruby/object:Gem::Dependency
56
+ name: rainbow
57
+ requirement: !ruby/object:Gem::Requirement
58
+ requirements:
59
+ - - ">="
60
+ - !ruby/object:Gem::Version
61
+ version: '0'
62
+ type: :runtime
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: thor
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :runtime
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: zeitwerk
85
+ requirement: !ruby/object:Gem::Requirement
86
+ requirements:
87
+ - - ">="
88
+ - !ruby/object:Gem::Version
89
+ version: '0'
90
+ type: :runtime
91
+ prerelease: false
92
+ version_requirements: !ruby/object:Gem::Requirement
93
+ requirements:
94
+ - - ">="
95
+ - !ruby/object:Gem::Version
96
+ version: '0'
97
+ - !ruby/object:Gem::Dependency
98
+ name: bundler
99
+ requirement: !ruby/object:Gem::Requirement
100
+ requirements:
101
+ - - ">="
102
+ - !ruby/object:Gem::Version
103
+ version: '0'
104
+ type: :development
105
+ prerelease: false
106
+ version_requirements: !ruby/object:Gem::Requirement
107
+ requirements:
108
+ - - ">="
109
+ - !ruby/object:Gem::Version
110
+ version: '0'
111
+ - !ruby/object:Gem::Dependency
112
+ name: byebug
113
+ requirement: !ruby/object:Gem::Requirement
114
+ requirements:
115
+ - - ">="
116
+ - !ruby/object:Gem::Version
117
+ version: '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'
125
+ - !ruby/object:Gem::Dependency
126
+ name: cli_markdown
127
+ requirement: !ruby/object:Gem::Requirement
128
+ requirements:
129
+ - - ">="
130
+ - !ruby/object:Gem::Version
131
+ version: '0'
132
+ type: :development
133
+ prerelease: false
134
+ version_requirements: !ruby/object:Gem::Requirement
135
+ requirements:
136
+ - - ">="
137
+ - !ruby/object:Gem::Version
138
+ version: '0'
139
+ - !ruby/object:Gem::Dependency
140
+ name: rake
141
+ requirement: !ruby/object:Gem::Requirement
142
+ requirements:
143
+ - - ">="
144
+ - !ruby/object:Gem::Version
145
+ version: '0'
146
+ type: :development
147
+ prerelease: false
148
+ version_requirements: !ruby/object:Gem::Requirement
149
+ requirements:
150
+ - - ">="
151
+ - !ruby/object:Gem::Version
152
+ version: '0'
153
+ - !ruby/object:Gem::Dependency
154
+ name: rspec
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
+ description:
168
+ email:
169
+ - tongueroo@gmail.com
170
+ executables:
171
+ - aws-rotate
172
+ extensions: []
173
+ extra_rdoc_files: []
174
+ files:
175
+ - ".gitignore"
176
+ - ".rspec"
177
+ - CHANGELOG.md
178
+ - Gemfile
179
+ - Gemfile.lock
180
+ - Guardfile
181
+ - LICENSE.txt
182
+ - README.md
183
+ - Rakefile
184
+ - aws-rotate.gemspec
185
+ - exe/aws-rotate
186
+ - lib/aws-rotate.rb
187
+ - lib/aws_rotate.rb
188
+ - lib/aws_rotate/autoloader.rb
189
+ - lib/aws_rotate/aws_services.rb
190
+ - lib/aws_rotate/backup.rb
191
+ - lib/aws_rotate/base.rb
192
+ - lib/aws_rotate/cache_key.rb
193
+ - lib/aws_rotate/cli.rb
194
+ - lib/aws_rotate/command.rb
195
+ - lib/aws_rotate/completer.rb
196
+ - lib/aws_rotate/completer/script.rb
197
+ - lib/aws_rotate/completer/script.sh
198
+ - lib/aws_rotate/help.rb
199
+ - lib/aws_rotate/help/completion.md
200
+ - lib/aws_rotate/help/completion_script.md
201
+ - lib/aws_rotate/help/key.md
202
+ - lib/aws_rotate/help/keys.md
203
+ - lib/aws_rotate/help/list.md
204
+ - lib/aws_rotate/key.rb
205
+ - lib/aws_rotate/keys.rb
206
+ - lib/aws_rotate/list.rb
207
+ - lib/aws_rotate/version.rb
208
+ - spec/fixtures/home/.aws/config
209
+ - spec/fixtures/home/.aws/credentials
210
+ - spec/lib/cli_spec.rb
211
+ - spec/lib/key_spec.rb
212
+ - spec/lib/keys_spec.rb
213
+ - spec/spec_helper.rb
214
+ homepage: https://github.com/tongueroo/aws-rotate
215
+ licenses:
216
+ - MIT
217
+ metadata: {}
218
+ post_install_message:
219
+ rdoc_options: []
220
+ require_paths:
221
+ - lib
222
+ required_ruby_version: !ruby/object:Gem::Requirement
223
+ requirements:
224
+ - - ">="
225
+ - !ruby/object:Gem::Version
226
+ version: '0'
227
+ required_rubygems_version: !ruby/object:Gem::Requirement
228
+ requirements:
229
+ - - ">="
230
+ - !ruby/object:Gem::Version
231
+ version: '0'
232
+ requirements: []
233
+ rubygems_version: 3.0.3
234
+ signing_key:
235
+ specification_version: 4
236
+ summary: Easy way to rotate all your AWS keys in your ~/.aws/credentials
237
+ test_files:
238
+ - spec/fixtures/home/.aws/config
239
+ - spec/fixtures/home/.aws/credentials
240
+ - spec/lib/cli_spec.rb
241
+ - spec/lib/key_spec.rb
242
+ - spec/lib/keys_spec.rb
243
+ - spec/spec_helper.rb