envied 0.9.1 → 0.9.2.rc1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +5 -5
- data/.gitignore +13 -17
- data/.rspec +1 -2
- data/.travis.yml +5 -6
- data/CHANGELOG.md +5 -0
- data/Gemfile +0 -3
- data/README.md +5 -6
- data/Rakefile +2 -6
- data/bin/console +14 -0
- data/bin/envied +4 -0
- data/bin/setup +8 -0
- data/envied.gemspec +5 -4
- data/lib/envied.rb +14 -17
- data/lib/envied/cli.rb +4 -11
- data/lib/envied/coercer.rb +7 -11
- data/lib/envied/coercer/envied_string.rb +51 -2
- data/lib/envied/configuration.rb +14 -17
- data/lib/envied/env_proxy.rb +22 -21
- data/lib/envied/env_var_extractor.rb +2 -2
- data/lib/envied/variable.rb +1 -1
- data/lib/envied/version.rb +1 -1
- data/spec/coercer_spec.rb +201 -49
- data/spec/configuration_spec.rb +78 -15
- data/spec/env_var_extractor_spec.rb +1 -3
- data/spec/envied_spec.rb +129 -170
- data/spec/gemspec_spec.rb +17 -0
- data/spec/spec_helper.rb +73 -5
- data/spec/variable_spec.rb +17 -7
- metadata +18 -27
data/spec/envied_spec.rb
CHANGED
@@ -1,112 +1,93 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
describe ENVied do
|
1
|
+
RSpec.describe ENVied do
|
4
2
|
describe 'class' do
|
5
3
|
subject { described_class }
|
6
|
-
|
7
4
|
it { is_expected.to respond_to :require }
|
8
5
|
end
|
9
6
|
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
reset_envied_config
|
16
|
-
reset_configuration
|
17
|
-
end
|
18
|
-
|
19
|
-
def reset_configuration
|
20
|
-
@config = ENVied::Configuration.new
|
21
|
-
end
|
22
|
-
|
23
|
-
def reset_env
|
24
|
-
ENVied.instance_eval { @env = nil }
|
25
|
-
end
|
26
|
-
|
27
|
-
def reset_envied_config
|
28
|
-
ENVied.instance_eval { @config = nil }
|
7
|
+
def reset_envied
|
8
|
+
ENVied.instance_eval do
|
9
|
+
@env = nil
|
10
|
+
@config = nil
|
11
|
+
end
|
29
12
|
end
|
30
13
|
|
31
14
|
context 'configured' do
|
32
15
|
|
33
|
-
|
34
|
-
|
35
|
-
self
|
16
|
+
before do
|
17
|
+
reset_envied
|
36
18
|
end
|
37
19
|
|
38
|
-
def
|
39
|
-
|
20
|
+
def set_ENV(env = {})
|
21
|
+
stub_const("ENV", env)
|
40
22
|
end
|
41
23
|
|
42
|
-
def configure(options
|
24
|
+
def configure(**options, &block)
|
43
25
|
@config = ENVied::Configuration.new(options, &block)
|
44
|
-
self
|
45
26
|
end
|
46
27
|
|
47
|
-
def configured_with(hash
|
48
|
-
@config = ENVied::Configuration.new.tap do |
|
28
|
+
def configured_with(**hash)
|
29
|
+
@config = ENVied::Configuration.new.tap do |config|
|
49
30
|
hash.each do |name, type|
|
50
|
-
|
31
|
+
config.variable(name, type)
|
51
32
|
end
|
52
33
|
end
|
53
|
-
self
|
54
|
-
end
|
55
|
-
|
56
|
-
def and_ENV(env = {})
|
57
|
-
stub_const("ENV", env)
|
58
|
-
described_class
|
59
34
|
end
|
60
35
|
|
61
|
-
def
|
62
|
-
|
36
|
+
def envied_require(*args, **options)
|
37
|
+
options[:config] = @config || ENVied::Configuration.new # Prevent `Configuration.load` from being called
|
38
|
+
ENVied.require(*args, **options)
|
63
39
|
end
|
64
40
|
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
ENVied.require(*args, options)
|
70
|
-
end
|
71
|
-
|
72
|
-
it 'responds to configured variables' do
|
73
|
-
configured_with(a: :integer).and_ENV({'a' => '1'})
|
41
|
+
it 'does respond to configured variables' do
|
42
|
+
set_ENV('A' => '1')
|
43
|
+
configured_with(A: :integer)
|
74
44
|
envied_require
|
75
45
|
|
76
|
-
expect(described_class).to respond_to :
|
46
|
+
expect(described_class).to respond_to :A
|
77
47
|
end
|
78
48
|
|
79
|
-
it '
|
80
|
-
|
49
|
+
it 'does not respond to unconfigured variables' do
|
50
|
+
set_ENV('A' => '1')
|
51
|
+
configured_with
|
81
52
|
envied_require
|
82
53
|
|
83
54
|
expect(described_class).to_not respond_to :B
|
84
55
|
end
|
85
56
|
|
86
57
|
it 'sets ENVied.config' do
|
87
|
-
|
58
|
+
set_ENV('A' => '1')
|
59
|
+
configured_with(A: :integer)
|
88
60
|
envied_require
|
89
61
|
|
90
|
-
expect(ENVied.config).
|
62
|
+
expect(ENVied.config).to be_instance_of(ENVied::Configuration)
|
91
63
|
end
|
92
64
|
|
93
65
|
context 'ENV contains not all configured variables' do
|
94
|
-
before
|
66
|
+
before do
|
67
|
+
set_ENV
|
68
|
+
configured_with(A: :integer)
|
69
|
+
end
|
95
70
|
|
96
71
|
specify do
|
97
72
|
expect {
|
98
73
|
envied_require
|
99
|
-
}.to raise_error(RuntimeError, 'The following environment variables should be set:
|
74
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: A.')
|
100
75
|
end
|
101
76
|
end
|
102
77
|
|
103
78
|
context 'ENV variables are not coercible' do
|
104
|
-
before
|
79
|
+
before do
|
80
|
+
set_ENV('A' => 'NaN', 'B' => 'invalid')
|
81
|
+
configured_with(A: :integer, B: :boolean)
|
82
|
+
end
|
105
83
|
|
106
84
|
specify do
|
107
85
|
expect {
|
108
86
|
envied_require
|
109
|
-
}.to raise_error(
|
87
|
+
}.to raise_error(
|
88
|
+
RuntimeError,
|
89
|
+
'The following environment variables are not coercible: A with "NaN" (integer), B with "invalid" (boolean).'
|
90
|
+
)
|
110
91
|
end
|
111
92
|
end
|
112
93
|
|
@@ -114,16 +95,17 @@ describe ENVied do
|
|
114
95
|
it 'raises error when configuring variable of unknown type' do
|
115
96
|
expect {
|
116
97
|
configured_with(A: :fixnum)
|
117
|
-
}.to raise_error(ArgumentError,
|
98
|
+
}.to raise_error(ArgumentError, ":fixnum is not a supported type. Should be one of #{ENVied::Coercer.supported_types}")
|
118
99
|
end
|
119
100
|
end
|
120
101
|
|
102
|
+
# TODO: review needed, doesn't make sense?
|
121
103
|
context 'bug: default value "false" is not coercible' do
|
122
|
-
before
|
104
|
+
before do
|
123
105
|
configure(enable_defaults: true) do
|
124
106
|
variable :FORCE_SSL, :boolean, default: true
|
125
107
|
end
|
126
|
-
|
108
|
+
end
|
127
109
|
|
128
110
|
specify do
|
129
111
|
expect {
|
@@ -133,41 +115,8 @@ describe ENVied do
|
|
133
115
|
end
|
134
116
|
|
135
117
|
describe 'defaults' do
|
136
|
-
describe 'setting' do
|
137
|
-
subject { config }
|
138
|
-
|
139
|
-
it 'is disabled by default' do
|
140
|
-
expect(subject.defaults_enabled?).to_not be
|
141
|
-
end
|
142
|
-
|
143
|
-
it 'yields ENV["ENVIED_ENABLE_DEFAULTS"] if not set otherwise' do
|
144
|
-
stub_const("ENV", {'ENVIED_ENABLE_DEFAULTS' => '1'})
|
145
|
-
configure
|
146
|
-
|
147
|
-
expect(subject.defaults_enabled?).to be
|
148
|
-
end
|
149
|
-
|
150
|
-
it 'can be enabled via #configure' do
|
151
|
-
configure(enable_defaults: true){ }
|
152
|
-
|
153
|
-
expect(subject.defaults_enabled?).to be
|
154
|
-
end
|
155
|
-
|
156
|
-
it 'can be enabled via a configure-block' do
|
157
|
-
configure { self.enable_defaults! }
|
158
|
-
|
159
|
-
expect(subject.defaults_enabled?).to be
|
160
|
-
end
|
161
|
-
|
162
|
-
it 'can be assigned a Proc' do
|
163
|
-
configure { self.enable_defaults! { true } }
|
164
|
-
|
165
|
-
expect(subject.defaults_enabled?).to be
|
166
|
-
end
|
167
|
-
end
|
168
|
-
|
169
118
|
describe 'assigning' do
|
170
|
-
it '
|
119
|
+
it 'sets a default value for ENV variable' do
|
171
120
|
configure(enable_defaults: true) do
|
172
121
|
variable :A, :integer, default: '1'
|
173
122
|
end
|
@@ -176,7 +125,7 @@ describe ENVied do
|
|
176
125
|
expect(described_class.A).to eq 1
|
177
126
|
end
|
178
127
|
|
179
|
-
it '
|
128
|
+
it 'sets a default value from calling a proc for ENV variable' do
|
180
129
|
configure(enable_defaults: true) do
|
181
130
|
variable :A, :integer, default: proc { "1" }
|
182
131
|
end
|
@@ -185,30 +134,34 @@ describe ENVied do
|
|
185
134
|
expect(described_class.A).to eq 1
|
186
135
|
end
|
187
136
|
|
188
|
-
it '
|
137
|
+
it 'ignores setting defaults if they are disabled' do
|
138
|
+
set_ENV
|
189
139
|
configure(enable_defaults: false) do
|
190
140
|
variable :A, :integer, default: "1"
|
191
|
-
|
141
|
+
variable :B, :integer, default: "1"
|
142
|
+
end
|
192
143
|
|
193
144
|
expect {
|
194
145
|
envied_require
|
195
|
-
}.to raise_error(RuntimeError, 'The following environment variables should be set: A.')
|
146
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: A, B.')
|
196
147
|
end
|
197
148
|
|
198
|
-
it '
|
149
|
+
it 'ignores a default value if ENV variable is already provided' do
|
150
|
+
set_ENV('A' => '2')
|
199
151
|
configure(enable_defaults: true) do
|
200
152
|
variable :A, :integer, default: "1"
|
201
|
-
end
|
153
|
+
end
|
202
154
|
envied_require
|
203
155
|
|
204
156
|
expect(described_class.A).to eq 2
|
205
157
|
end
|
206
158
|
|
207
|
-
it 'can
|
159
|
+
it 'can set a default value via a proc that references another ENV variable' do
|
160
|
+
set_ENV('A' => '1')
|
208
161
|
configure(enable_defaults: true) do
|
209
162
|
variable :A, :integer
|
210
163
|
variable :B, :integer, default: proc {|env| env.A * 2 }
|
211
|
-
end
|
164
|
+
end
|
212
165
|
envied_require
|
213
166
|
|
214
167
|
expect(described_class.B).to eq 2
|
@@ -216,8 +169,9 @@ describe ENVied do
|
|
216
169
|
end
|
217
170
|
end
|
218
171
|
|
219
|
-
describe "
|
220
|
-
|
172
|
+
describe ".required?" do
|
173
|
+
# TODO: change to always return boolean
|
174
|
+
it 'returns true-ish if `ENVied.require` was called' do
|
221
175
|
expect {
|
222
176
|
envied_require
|
223
177
|
}.to change { ENVied.required? }.from(nil).to(anything)
|
@@ -225,154 +179,159 @@ describe ENVied do
|
|
225
179
|
end
|
226
180
|
|
227
181
|
describe "groups" do
|
228
|
-
describe 'requiring' do
|
229
|
-
|
230
|
-
it 'yields :default when nothing passed to require' do
|
231
|
-
envied_require
|
232
|
-
expect(ENVied.env.groups).to eq [:default]
|
233
|
-
end
|
234
|
-
|
235
|
-
it 'takes ENV["ENVIED_GROUPS"] into account when nothing passed to require' do
|
236
|
-
and_ENV('ENVIED_GROUPS' => 'baz')
|
237
|
-
envied_require
|
238
|
-
expect(ENVied.env.groups).to eq [:baz]
|
239
|
-
end
|
240
|
-
|
241
|
-
it 'yields groupnames passed to it as string' do
|
242
|
-
envied_require('bar')
|
243
|
-
expect(ENVied.env.groups).to eq [:bar]
|
244
|
-
end
|
245
|
-
|
246
|
-
it 'yields groupnames passed to it as symbols' do
|
247
|
-
envied_require(:foo)
|
248
|
-
expect(ENVied.env.groups).to eq [:foo]
|
249
|
-
end
|
250
|
-
|
251
|
-
it 'yields the groups passed via a string with groupnames' do
|
252
|
-
envied_require('foo,bar')
|
253
|
-
expect(ENVied.env.groups).to eq [:foo, :bar]
|
254
|
-
end
|
255
|
-
end
|
256
|
-
|
257
182
|
context 'a variable in a group' do
|
258
183
|
before do
|
184
|
+
set_ENV
|
259
185
|
configure do
|
260
|
-
variable :
|
186
|
+
variable :MORE
|
261
187
|
|
262
188
|
group :foo do
|
263
|
-
variable :
|
189
|
+
variable :BAR
|
264
190
|
end
|
265
|
-
|
191
|
+
group :moo do
|
192
|
+
variable :BAT
|
193
|
+
end
|
194
|
+
end
|
266
195
|
end
|
267
196
|
|
268
|
-
it 'is required when requiring the
|
197
|
+
it 'is required when requiring the groups passed as a delimited string' do
|
269
198
|
expect {
|
270
|
-
envied_require(
|
271
|
-
}.to raise_error(
|
199
|
+
envied_require('foo,moo')
|
200
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: BAR, BAT.')
|
201
|
+
end
|
202
|
+
|
203
|
+
it 'is required when requiring the group' do
|
204
|
+
[:foo, 'foo'].each do |group|
|
205
|
+
expect {
|
206
|
+
envied_require(group)
|
207
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: BAR.')
|
208
|
+
end
|
272
209
|
end
|
273
210
|
|
274
211
|
it 'is not required when requiring another group' do
|
275
|
-
|
276
|
-
|
277
|
-
|
212
|
+
[:bat, 'bat'].each do |group|
|
213
|
+
expect {
|
214
|
+
envied_require(group)
|
215
|
+
}.to_not raise_error
|
216
|
+
end
|
278
217
|
end
|
279
218
|
|
280
|
-
it '
|
281
|
-
|
219
|
+
it 'will not define variables not part of the default group' do
|
220
|
+
set_ENV('MORE' => 'yes')
|
282
221
|
envied_require(:default)
|
283
222
|
|
284
223
|
expect {
|
285
|
-
described_class.
|
224
|
+
described_class.BAR
|
286
225
|
}.to raise_error(NoMethodError)
|
287
226
|
end
|
288
227
|
|
228
|
+
it 'takes ENV["ENVIED_GROUPS"] into account when nothing is passed to require' do
|
229
|
+
set_ENV('ENVIED_GROUPS' => 'foo')
|
230
|
+
expect {
|
231
|
+
envied_require
|
232
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: BAR.')
|
233
|
+
end
|
234
|
+
|
235
|
+
it 'will define variables in the default group when nothing is passed to require' do
|
236
|
+
set_ENV('MORE' => 'yes')
|
237
|
+
envied_require
|
238
|
+
|
239
|
+
expect(described_class.MORE).to eq 'yes'
|
240
|
+
end
|
241
|
+
|
289
242
|
it 'requires variables without a group when requiring the default group' do
|
290
|
-
[:default, 'default'].each do |
|
243
|
+
[:default, 'default'].each do |group|
|
291
244
|
expect {
|
292
|
-
envied_require(
|
293
|
-
}.to raise_error(
|
245
|
+
envied_require(group)
|
246
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: MORE.')
|
294
247
|
end
|
295
248
|
end
|
296
249
|
end
|
297
250
|
|
298
251
|
context 'a variable in multiple groups' do
|
299
252
|
before do
|
253
|
+
set_ENV
|
300
254
|
configure do
|
301
|
-
variable :
|
255
|
+
variable :MORE
|
302
256
|
|
303
257
|
group :foo, :moo do
|
304
|
-
variable :
|
258
|
+
variable :BAR
|
305
259
|
end
|
306
|
-
end
|
260
|
+
end
|
307
261
|
end
|
308
262
|
|
309
263
|
it 'is required when requiring any of the groups' do
|
310
264
|
expect {
|
311
265
|
envied_require(:foo)
|
312
|
-
}.to raise_error(
|
266
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: BAR.')
|
313
267
|
|
314
268
|
expect {
|
315
269
|
envied_require(:moo)
|
316
|
-
}.to raise_error(
|
270
|
+
}.to raise_error(RuntimeError, 'The following environment variables should be set: BAR.')
|
317
271
|
end
|
318
272
|
end
|
319
273
|
|
320
274
|
describe 'Hashable' do
|
321
275
|
before do
|
276
|
+
set_ENV('FOO' => 'a=1&b=&c', 'BAR' => '')
|
322
277
|
configure do
|
323
|
-
variable :
|
324
|
-
variable :
|
325
|
-
end
|
278
|
+
variable :FOO, :hash
|
279
|
+
variable :BAR, :hash
|
280
|
+
end
|
326
281
|
envied_require
|
327
282
|
end
|
328
283
|
|
329
284
|
it 'yields hash from string' do
|
330
|
-
expect(ENVied.
|
285
|
+
expect(ENVied.FOO).to eq({ 'a' => '1', 'b' => '', 'c' => nil })
|
331
286
|
end
|
332
287
|
|
333
288
|
it 'yields hash from an empty string' do
|
334
|
-
expect(ENVied.
|
289
|
+
expect(ENVied.BAR).to eq({})
|
335
290
|
end
|
336
291
|
|
337
292
|
context 'with defaults enabled' do
|
338
293
|
before do
|
294
|
+
set_ENV
|
339
295
|
configure(enable_defaults: true) do
|
340
|
-
variable :
|
341
|
-
end
|
296
|
+
variable :BAZ, :hash
|
297
|
+
end
|
342
298
|
end
|
343
299
|
|
344
300
|
it 'has no default by default' do
|
345
301
|
# fixes a bug where variables of type :Hash had a default even
|
346
302
|
# when none was configured.
|
347
|
-
expect { envied_require }.to raise_error(RuntimeError, 'The following environment variables should be set:
|
303
|
+
expect { envied_require }.to raise_error(RuntimeError, 'The following environment variables should be set: BAZ.')
|
348
304
|
end
|
349
305
|
end
|
350
306
|
end
|
351
307
|
|
352
308
|
describe 'Arrayable' do
|
353
309
|
before do
|
310
|
+
set_ENV('MORE' => 'a, b, and\, c')
|
354
311
|
configure do
|
355
|
-
variable :
|
356
|
-
end
|
312
|
+
variable :MORE, :array
|
313
|
+
end
|
357
314
|
envied_require
|
358
315
|
end
|
359
316
|
|
360
317
|
it 'yields array from string' do
|
361
|
-
expect(ENVied.
|
318
|
+
expect(ENVied.MORE).to eq ['a',' b',' and, c']
|
362
319
|
end
|
363
320
|
end
|
364
321
|
|
365
322
|
describe 'URIable' do
|
366
323
|
before do
|
324
|
+
set_ENV('SITE_URL' => 'https://www.google.com')
|
367
325
|
configure do
|
368
|
-
variable :
|
369
|
-
end
|
326
|
+
variable :SITE_URL, :uri
|
327
|
+
end
|
370
328
|
envied_require
|
371
329
|
end
|
372
330
|
|
373
331
|
it 'yields a URI from string' do
|
374
|
-
expect(ENVied.
|
375
|
-
expect(ENVied.
|
332
|
+
expect(ENVied.SITE_URL).to be_a URI
|
333
|
+
expect(ENVied.SITE_URL.scheme).to eq 'https'
|
334
|
+
expect(ENVied.SITE_URL.host).to eq 'www.google.com'
|
376
335
|
end
|
377
336
|
end
|
378
337
|
end
|