nero 0.3.0 → 0.5.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 +4 -4
- data/Appraisals +7 -0
- data/CHANGELOG.md +50 -0
- data/README.md +166 -44
- data/gemfiles/psych_3.gemfile +10 -0
- data/gemfiles/psych_4.gemfile +10 -0
- data/lib/nero/version.rb +1 -1
- data/lib/nero.rb +291 -98
- metadata +19 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 4e28a4b8bcd989c20dd60dc592bfb31039314437b3d8ca5df33b79adefd5cb2e
|
4
|
+
data.tar.gz: f6eb0ec1e9ceb041088a7d2b850494e090f08d98f4e8369f82f0608a4449ab69
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 98dc2c475e0e164f47d1738dbe32d6dfeeba6b531f6db0b396a25d3e29219c312ac0750206fddea1d9ee450e9205425a5f0342c6b5b13d38867a364e1219bed0
|
7
|
+
data.tar.gz: e9daf1ee287679b10d4ea744c41d3f93bf50570394eb261f7c8b3587a1008efe57b58b7957141aeea48229b65efdaa614f23c181fbf08659a0a878a7a944e773
|
data/Appraisals
ADDED
data/CHANGELOG.md
CHANGED
@@ -1,4 +1,54 @@
|
|
1
1
|
## [Unreleased]
|
2
|
+
...
|
3
|
+
|
4
|
+
## [0.5.0] - 2025-03-20
|
5
|
+
|
6
|
+
- tag-classes
|
7
|
+
Added [`Nero::BaseTag`](https://rubydoc.info/github/eval/nero/main/Nero/BaseTag) that is the basis of all existing tags.
|
8
|
+
This means that building upon existing tags is easier and custom tags can be more powerful.
|
9
|
+
|
10
|
+
Create new tags can be done in 3 ways:
|
11
|
+
By block (as before, but slightly changed interface):
|
12
|
+
```ruby
|
13
|
+
Nero.configure do |nero|
|
14
|
+
nero.add_tag("foo") do |tag|
|
15
|
+
# tag of type Nero::BaseTag
|
16
|
+
end
|
17
|
+
end
|
18
|
+
```
|
19
|
+
By re-using existing tags via options:
|
20
|
+
```ruby
|
21
|
+
nero.add_tag("env/upcase", klass: Nero::EnvTag[coerce: :upcase])
|
22
|
+
```
|
23
|
+
Finally, by subclassing [Nero::BaseTag](https://rubydoc.info/github/eval/nero/main/Nero/BaseTag). See the section ["custom tags"](https://github.com/eval/nero?tab=readme-ov-file#custom-tags) from the README.
|
24
|
+
|
25
|
+
- `!env/float` and `!env/float?`
|
26
|
+
- `!env/git_root` and `!env/rails_root`
|
27
|
+
Construct a path relative to some root-path:
|
28
|
+
```yaml
|
29
|
+
asset_path: !path/rails_root [ public/assets ]
|
30
|
+
```
|
31
|
+
Easy to use for your own tags:
|
32
|
+
```ruby
|
33
|
+
config.add_tag("path/project_root", klass: Nero::PathRootTag[containing: '.git']) do |path|
|
34
|
+
# possible post-processing
|
35
|
+
end
|
36
|
+
```
|
37
|
+
- [#2](https://github.com/eval/nero/pull/2) Add irb to gemfile (@dlibanori)
|
38
|
+
- [#3](https://github.com/eval/nero/pull/3) Fix missing require (@dlibanori)
|
39
|
+
|
40
|
+
## [0.4.0] - 2025-02-15
|
41
|
+
|
42
|
+
- Add `!ref`-tag:
|
43
|
+
```ruby
|
44
|
+
Nero.load(<<~YAML)
|
45
|
+
min_threads: !env [MIN_THREADS, !ref [max_threads]]
|
46
|
+
max_threads: 5
|
47
|
+
end
|
48
|
+
# => {min_threads: 5, max_threads: 5}
|
49
|
+
```
|
50
|
+
- Support Psych v3
|
51
|
+
...so it can used with Rails v6
|
2
52
|
|
3
53
|
## [0.3.0] - 2025-02-02
|
4
54
|
|
data/README.md
CHANGED
@@ -2,33 +2,43 @@
|
|
2
2
|
|
3
3
|
[](https://badge.fury.io/rb/nero)
|
4
4
|
|
5
|
-
Nero is a RubyGem that offers
|
5
|
+
Nero is a RubyGem that offers declarative YAML-tags to simplify config files, e.g. for requiring and coercion of env-vars.
|
6
|
+
Additionally, it allows you to create your own.
|
6
7
|
|
7
|
-
|
8
|
+
**Sample:**
|
8
9
|
|
9
10
|
```yaml
|
10
11
|
development:
|
11
|
-
|
12
|
-
# custom logic how to get a boolean...
|
13
|
-
debug?: <%= ENV["DEBUG"] == "true" %>
|
14
|
-
production:
|
15
|
-
# any ENV.fetch in this section would hamper local development...
|
16
|
-
secret: <%= ENV["SECRET"] %>
|
17
|
-
# custom coercion logic
|
18
|
-
max_threads: <%= ENV.fetch("MAX_THREADS", 5).to_i %>
|
19
|
-
```
|
20
|
-
|
21
|
-
...turn it into this:
|
22
|
-
```yaml
|
23
|
-
development:
|
12
|
+
# env-var with default value
|
24
13
|
secret: !env [SECRET, "dummy"]
|
14
|
+
|
15
|
+
# optional env-var with coercion
|
25
16
|
debug?: !env/bool? DEBUG
|
17
|
+
|
26
18
|
production:
|
27
|
-
# required
|
19
|
+
# required env-var (not required during development)
|
28
20
|
secret: !env SECRET
|
21
|
+
|
22
|
+
# coercion
|
29
23
|
max_threads: !env/integer [MAX_THREADS, 5]
|
24
|
+
|
25
|
+
# refer to other keys
|
26
|
+
min_threads: !env/integer [MIN_THREADS, !ref max_threads ]
|
27
|
+
|
28
|
+
# descriptive names
|
29
|
+
asset_folder: !path/rails_root [ public/assets ]
|
30
|
+
|
31
|
+
# easy to add custom tags
|
32
|
+
cache_ttl: !duration [2, hours]
|
30
33
|
```
|
31
34
|
|
35
|
+
## Highlights
|
36
|
+
|
37
|
+
* 💎 declarative YAML-tags for e.g. requiring and coercing env-vars
|
38
|
+
* 🛠️ add custom tags
|
39
|
+
* 🛤️ `Rails.application.config_for` drop-in
|
40
|
+
* ♻️ Zeitwerk-only dependency
|
41
|
+
|
32
42
|
## Installation
|
33
43
|
|
34
44
|
Install the gem and add to the application's Gemfile by executing:
|
@@ -42,6 +52,8 @@ bundle add nero
|
|
42
52
|
> [!WARNING]
|
43
53
|
> It's early days - the API and included tags will certainly change. Check the CHANGELOG when upgrading.
|
44
54
|
|
55
|
+
### loading a config
|
56
|
+
|
45
57
|
Given the following config:
|
46
58
|
```yaml
|
47
59
|
# config/settings.yml
|
@@ -76,6 +88,17 @@ Nero.load_config("config/settings", root: :production)
|
|
76
88
|
# ...with ENV {"SECRET" => "s3cr3t", "MAX_THREADS" => "3"}
|
77
89
|
#=> {secret: "s3cr3t", max_threads: 3}
|
78
90
|
```
|
91
|
+
> [!TIP]
|
92
|
+
> The following configuration would make `Nero.load_config` a drop-in replacement for [Rails.application.config_for](https://api.rubyonrails.org/classes/Rails/Application.html#method-i-config_for):
|
93
|
+
```ruby
|
94
|
+
Nero.configure do |config|
|
95
|
+
config.config_dir = Rails.root / "config"
|
96
|
+
end
|
97
|
+
|
98
|
+
Nero.load_config(:settings, env: Rails.env)
|
99
|
+
```
|
100
|
+
|
101
|
+
### built-in tags
|
79
102
|
|
80
103
|
The following tags are provided:
|
81
104
|
- `!env KEY`, `!env? KEY`
|
@@ -90,10 +113,11 @@ The following tags are provided:
|
|
90
113
|
secret: !env? SECRET
|
91
114
|
```
|
92
115
|
- to coerce env-values:
|
93
|
-
- `env/integer`, `env/integer?`:
|
116
|
+
- `env/integer`, `env/integer?`, `env/float`, `env/float?`:
|
94
117
|
```yaml
|
95
118
|
port: !env/integer [PORT, 3000]
|
96
119
|
threads: !env/integer? THREADS # nil when not provided
|
120
|
+
threshold: !env/float CUTOFF
|
97
121
|
```
|
98
122
|
- `env/bool`, `env/bool?`:
|
99
123
|
```yaml
|
@@ -104,6 +128,11 @@ The following tags are provided:
|
|
104
128
|
# ...or false:
|
105
129
|
debug?: !env/bool? DEBUG
|
106
130
|
```
|
131
|
+
> [!TIP]
|
132
|
+
> Make all env-var's optional by providing `ENV["NERO_ENV_ALL_OPTIONAL"]`, e.g.
|
133
|
+
```shell
|
134
|
+
$ env NERO_ENV_ALL_OPTIONAL=1 SECRET_KEY_BASE_DUMMY=1 rails asset:precompile
|
135
|
+
```
|
107
136
|
- `!path`
|
108
137
|
Create a [Pathname](https://rubyapi.org/3.4/o/pathname):
|
109
138
|
```yaml
|
@@ -113,6 +142,15 @@ The following tags are provided:
|
|
113
142
|
- !env PROJECT_ROOT
|
114
143
|
- /public/assets
|
115
144
|
```
|
145
|
+
- `!path/git_root`, `!path/rails_root`
|
146
|
+
Create a Pathname relative to some root-path.
|
147
|
+
The root-path is expected to be an existing ancestor folder of the yaml-config being parsed.
|
148
|
+
It's found by traversing up and checking for the presence of specific files/folders, e.g. '.git' (`!path/git_root`) or 'config.ru' (`!path/rails_root`).
|
149
|
+
While the root-path needs to exist, the resulting Pathname doesn't need to.
|
150
|
+
```yaml
|
151
|
+
project_root: !path/git_root
|
152
|
+
config_folder: !path/rails_root [ config ]
|
153
|
+
```
|
116
154
|
- `!uri`
|
117
155
|
Create a [URI](https://rubyapi.org/3.4/o/uri):
|
118
156
|
```yaml
|
@@ -128,39 +166,123 @@ The following tags are provided:
|
|
128
166
|
- smtps://%s:%s@smtp.gmail.com
|
129
167
|
- !env SMTP_USER
|
130
168
|
- !env SMTP_PASS
|
131
|
-
|
169
|
+
|
170
|
+
# pass it a map (including a key 'fmt') to use references
|
132
171
|
smtp_url: !str/format
|
133
172
|
fmt: smtps://%<user>s:%<pass>s@smtp.gmail.com
|
134
173
|
user: !env SMTP_USER
|
135
174
|
pass: !env SMTP_PASS
|
136
175
|
```
|
176
|
+
- `!ref`
|
177
|
+
Include values from elsewhere:
|
178
|
+
```yaml
|
179
|
+
# simple
|
180
|
+
min_threads: !env/integer [MIN_THREADS, !ref [max_threads]]
|
181
|
+
max_threads: 5
|
182
|
+
|
183
|
+
# oauth_callback -refs-> base.url -refs-> base.host
|
184
|
+
base:
|
185
|
+
host: !env [HOST]
|
186
|
+
url: !str/format ['https://%s', !ref[base, host]]
|
187
|
+
oauth_callback: !str/format
|
188
|
+
- '%s/oauth/callback'
|
189
|
+
- !ref[base, url]
|
137
190
|
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
162
|
-
|
163
|
-
|
191
|
+
# refs are resolved within the tree of the selected root.
|
192
|
+
# The following config won't work when doing `Nero.load_config(:app, root: :prod)`
|
193
|
+
dev:
|
194
|
+
max_threads: 5
|
195
|
+
prod:
|
196
|
+
max_threads: !env[MAX_THREADS, !ref[dev, max_threads]]
|
197
|
+
```
|
198
|
+
NOTE future version should raise properly over ref-ing a non-existing path.
|
199
|
+
|
200
|
+
### custom tags
|
201
|
+
|
202
|
+
Three ways to do this:
|
203
|
+
|
204
|
+
1. a block
|
205
|
+
```ruby
|
206
|
+
Nero.configure do |nero|
|
207
|
+
nero.add_tag("upcase") do |tag|
|
208
|
+
# `tag` is a `Nero::BaseTag`.
|
209
|
+
# In YAML args are provided as scalar, seq or map:
|
210
|
+
# ---
|
211
|
+
# k: !upcase bar
|
212
|
+
# ---
|
213
|
+
# k: !upcase [bar] # equivalent to:
|
214
|
+
# k: !upcase
|
215
|
+
# - bar
|
216
|
+
# ---
|
217
|
+
# k: !upcase
|
218
|
+
# bar: baz
|
219
|
+
#
|
220
|
+
# Find these args via `tag.args` (Array or Hash):
|
221
|
+
case tag.args
|
222
|
+
when Hash
|
223
|
+
tag.args.each_with_object({}) {|(k,v), acc| acc[k] = v.upcase }
|
224
|
+
else
|
225
|
+
tag.args.map(&:upcase)
|
226
|
+
end
|
227
|
+
|
228
|
+
# NOTE though you might just need one argument, it's helpful to accept a seq nonetheless
|
229
|
+
# as it allows for chaining:
|
230
|
+
# a: !my/inc 4 # scalar suffices
|
231
|
+
# ...but when chaining, it comes as a seq:
|
232
|
+
# a: !my/inc [!my/square 2]
|
233
|
+
end
|
234
|
+
end
|
235
|
+
```
|
236
|
+
1. re-use existing tag-class
|
237
|
+
You can add an existing tag under a better fitting name this way.
|
238
|
+
Also: some tag-classes have options that allow for simple customizations (like `coerce` below):
|
239
|
+
```ruby
|
240
|
+
Nero.configure do |nero|
|
241
|
+
nero.add_tag("env/upcase", klass: Nero::EnvTag[coerce: :upcase])
|
242
|
+
|
243
|
+
# Alias for path/git_root:
|
244
|
+
nero.add_tag("path/project_root", klass: Nero::PathRootTag[containing: '.git'])
|
245
|
+
end
|
246
|
+
```
|
247
|
+
1. custom class
|
248
|
+
```ruby
|
249
|
+
class RotTag < Nero::BaseTag
|
250
|
+
# Configure:
|
251
|
+
# ```
|
252
|
+
# config.add_tag("rot/12", klass: RotTag[n: 12])
|
253
|
+
# config.add_tag("rot/10", klass: RotTag[n: 10]) do |secret|
|
254
|
+
# "#{secret} (try breaking this!)"
|
255
|
+
# end
|
256
|
+
# ```
|
257
|
+
#
|
258
|
+
# Usage in YAML:
|
259
|
+
# ```
|
260
|
+
# secret: !rot/12 some message
|
261
|
+
# very_secret: !rot/10 [ !env [ MSG, some message ] ]
|
262
|
+
# ```
|
263
|
+
# => {secret: "EAyq yqEEmsq", very_secret: "Cywo woCCkqo (try breaking this!)"}
|
264
|
+
|
265
|
+
# By overriding `init_options` we can restrict/require options,
|
266
|
+
# provide default values and do any other setup.
|
267
|
+
# By default an option is available via `options[:foo]`.
|
268
|
+
def init_options(n: 10)
|
269
|
+
super # no specific assignments, so available via `options[:n]`.
|
270
|
+
end
|
271
|
+
|
272
|
+
def chars
|
273
|
+
@chars ||= (('a'..'z').to_a + ('A'..'Z').to_a + ('0'..'9').to_a)
|
274
|
+
end
|
275
|
+
|
276
|
+
def resolve(**) # currently no keywords are passed, but `**` allows for future ones.
|
277
|
+
# Here we actually do the work: get the args, rotate strings and delegate to the block.
|
278
|
+
# `args` are the resolved nested args (so e.g. `!env MSG` is already resolved).
|
279
|
+
# `config` is the tag's config, and contains e.g. the block.
|
280
|
+
block = config.fetch(:block, :itself.to_proc)
|
281
|
+
# String#tr replaces any character from the first collection with the same position in the other:
|
282
|
+
args.join.tr(chars.join, chars.rotate(options[:n]).join).then(&block)
|
283
|
+
end
|
284
|
+
end
|
285
|
+
```
|
164
286
|
|
165
287
|
## Development
|
166
288
|
|
data/lib/nero/version.rb
CHANGED
data/lib/nero.rb
CHANGED
@@ -4,8 +4,9 @@ require "zeitwerk"
|
|
4
4
|
loader = Zeitwerk::Loader.for_gem
|
5
5
|
loader.setup
|
6
6
|
|
7
|
-
require "uri"
|
7
|
+
require "uri"
|
8
8
|
require "yaml"
|
9
|
+
require "pathname"
|
9
10
|
|
10
11
|
# TODO fail on unknown tag
|
11
12
|
# TODO show missing env's at once
|
@@ -16,7 +17,7 @@ module Nero
|
|
16
17
|
module Resolvable
|
17
18
|
def try_resolve(ctx, object)
|
18
19
|
if object.respond_to?(:resolve)
|
19
|
-
object.resolve(ctx)
|
20
|
+
object.resolve(**ctx)
|
20
21
|
else
|
21
22
|
object
|
22
23
|
end
|
@@ -29,6 +30,15 @@ module Nero
|
|
29
30
|
def deep_resolve(object, **ctx)
|
30
31
|
Util.deep_transform_values(object, &gen_resolve_tryer(ctx))
|
31
32
|
end
|
33
|
+
|
34
|
+
def resolve_nested!(coder, ctx = {})
|
35
|
+
case coder.type
|
36
|
+
when :seq
|
37
|
+
coder.seq.map!(&gen_resolve_tryer(ctx))
|
38
|
+
when :map
|
39
|
+
coder.map = deep_resolve(coder.map, **ctx)
|
40
|
+
end
|
41
|
+
end
|
32
42
|
end
|
33
43
|
extend Resolvable
|
34
44
|
private_class_method \
|
@@ -36,37 +46,23 @@ module Nero
|
|
36
46
|
:gen_resolve_tryer,
|
37
47
|
:try_resolve
|
38
48
|
|
39
|
-
class
|
40
|
-
|
49
|
+
class Configuration
|
50
|
+
attr_reader :tags, :config_dir
|
41
51
|
|
42
|
-
def
|
43
|
-
@
|
52
|
+
def config_dir=(dir)
|
53
|
+
@config_dir = Pathname(dir).expand_path
|
44
54
|
end
|
45
55
|
|
46
|
-
def
|
47
|
-
|
48
|
-
ctx[:tags][@coder.tag].call(@coder)
|
49
|
-
end
|
56
|
+
def add_tag(name, klass: BaseTag, &block)
|
57
|
+
klass, klass_options = klass
|
50
58
|
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
@coder.seq.map!(&gen_resolve_tryer(ctx))
|
55
|
-
when :map
|
56
|
-
@coder.map = deep_resolve(@coder.map, **ctx)
|
59
|
+
(@tags ||= {})[name] = {klass:}.tap do |h|
|
60
|
+
h[:block] = block if block
|
61
|
+
h[:options] = klass_options if klass_options
|
57
62
|
end
|
58
63
|
end
|
59
64
|
end
|
60
65
|
|
61
|
-
class Configuration
|
62
|
-
attr_reader :tags
|
63
|
-
attr_accessor :config_dir
|
64
|
-
|
65
|
-
def add_tag(name, &block)
|
66
|
-
(@tags ||= {})["!#{name}"] = block
|
67
|
-
end
|
68
|
-
end
|
69
|
-
|
70
66
|
def self.configuration
|
71
67
|
@configuration ||= Configuration.new
|
72
68
|
end
|
@@ -75,77 +71,221 @@ module Nero
|
|
75
71
|
yield configuration if block_given?
|
76
72
|
end
|
77
73
|
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
74
|
+
class BaseTag
|
75
|
+
include Resolvable
|
76
|
+
|
77
|
+
attr_reader :coder, :options, :ctx
|
78
|
+
|
79
|
+
def self.[](**options)
|
80
|
+
[self, options]
|
81
|
+
end
|
82
|
+
|
83
|
+
# used by YAML
|
84
|
+
def init_with(coder)
|
85
|
+
@coder = coder
|
86
|
+
end
|
83
87
|
|
84
|
-
|
85
|
-
|
88
|
+
def init(ctx:, options:)
|
89
|
+
init_ctx(ctx)
|
90
|
+
init_options(**options)
|
91
|
+
end
|
92
|
+
|
93
|
+
def init_ctx(ctx)
|
94
|
+
@ctx = ctx
|
95
|
+
end
|
96
|
+
|
97
|
+
def init_options(**options)
|
98
|
+
@options = options
|
99
|
+
end
|
100
|
+
|
101
|
+
def tag_name
|
102
|
+
coder.tag[1..]
|
103
|
+
end
|
104
|
+
|
105
|
+
def args
|
106
|
+
@args ||= begin
|
107
|
+
resolve_nested!(coder, {})
|
108
|
+
case coder.type
|
109
|
+
when :map then Util.deep_symbolize_keys(coder.map)
|
110
|
+
else
|
111
|
+
Array(coder.public_send(coder.type))
|
112
|
+
end
|
86
113
|
end
|
114
|
+
end
|
115
|
+
|
116
|
+
def config
|
117
|
+
ctx.dig(:tags, tag_name)
|
118
|
+
end
|
87
119
|
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
93
|
-
|
94
|
-
|
95
|
-
when re_true then true
|
96
|
-
when re_false then false
|
97
|
-
else
|
98
|
-
raise "bool value should be one of y(es)/n(o), on/off, true/false (got #{s.inspect})"
|
99
|
-
end
|
120
|
+
def resolve(**)
|
121
|
+
if (block = config[:block])
|
122
|
+
if block.parameters.map(&:last).include?(:coder)
|
123
|
+
# legacy
|
124
|
+
block.call(coder, ctx)
|
125
|
+
else
|
126
|
+
block.call(self)
|
100
127
|
end
|
128
|
+
else
|
129
|
+
args
|
130
|
+
end
|
131
|
+
end
|
132
|
+
end
|
101
133
|
|
102
|
-
|
134
|
+
# Requires an env-var to be available and coerces the value.
|
135
|
+
# When tag-name ends with "?", the env-var is optional.
|
136
|
+
#
|
137
|
+
# Given config:
|
138
|
+
# config.add_tag("env/upcase", klass: Nero::EnvTag[coerce: :upcase])
|
139
|
+
# config.add_tag("env/upcase?", klass: Nero::EnvTag[coerce: :upcase])
|
140
|
+
|
141
|
+
# Then YAML => result:
|
142
|
+
# "--- env/upcase [MSG, Hello World]" => "HELLO WORLD"
|
143
|
+
# "--- env/upcase MSG" => raises when not ENV.has_key? "MSG"
|
144
|
+
# "--- env/upcase? MSG" => nil
|
145
|
+
#
|
146
|
+
# Args supported:
|
147
|
+
# - scalar
|
148
|
+
# name of env-var, e.g. `!env HOME`
|
149
|
+
# - seq
|
150
|
+
# name of env-var and fallback, e.g. `!env [HOME, /root]`
|
151
|
+
|
152
|
+
# Options:
|
153
|
+
# - coerce - symbol or proc to be applied to value of env-var.
|
154
|
+
# when using coerce, the block is ignoerd.
|
155
|
+
#
|
156
|
+
class EnvTag < BaseTag
|
157
|
+
def resolve(**)
|
158
|
+
if coercer
|
159
|
+
coercer.call(env_value) unless env_value.nil?
|
160
|
+
elsif ctx.dig(:tags, tag_name, :block)
|
161
|
+
super
|
162
|
+
else
|
163
|
+
env_value
|
103
164
|
end
|
165
|
+
end
|
104
166
|
|
105
|
-
|
106
|
-
|
107
|
-
|
108
|
-
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
167
|
+
def coercer
|
168
|
+
return unless @coerce
|
169
|
+
|
170
|
+
@coercer ||= case @coerce
|
171
|
+
when Symbol then @coerce.to_proc
|
172
|
+
else
|
173
|
+
@coerce
|
174
|
+
end
|
175
|
+
end
|
176
|
+
|
177
|
+
def init_options(coerce: nil)
|
178
|
+
@coerce = coerce
|
179
|
+
end
|
180
|
+
|
181
|
+
def optional
|
182
|
+
tag_name.end_with?("?") || !!ENV["NERO_ENV_ALL_OPTIONAL"]
|
183
|
+
end
|
184
|
+
alias_method :optional?, :optional
|
118
185
|
|
119
|
-
|
186
|
+
def env_value
|
187
|
+
self.class.env_value(*args, optional:)
|
188
|
+
end
|
189
|
+
|
190
|
+
def self.env_value(k, fallback = nil, optional: false)
|
191
|
+
if fallback.nil? && !optional
|
192
|
+
ENV.fetch(k)
|
193
|
+
else
|
194
|
+
ENV.fetch(k, fallback)
|
120
195
|
end
|
196
|
+
end
|
121
197
|
|
122
|
-
|
123
|
-
|
198
|
+
def self.coerce_bool(v)
|
199
|
+
return false unless v
|
200
|
+
|
201
|
+
re_true = /y|Y|yes|Yes|YES|true|True|TRUE|on|On|ON/
|
202
|
+
re_false = /n|N|no|No|NO|false|False|FALSE|off|Off|OFF/
|
203
|
+
|
204
|
+
case v
|
205
|
+
when TrueClass, FalseClass then v
|
206
|
+
when re_true then true
|
207
|
+
when re_false then false
|
208
|
+
else
|
209
|
+
raise "bool value should be one of y(es)/n(o), on/off, true/false (got #{v.inspect})"
|
124
210
|
end
|
211
|
+
end
|
212
|
+
end
|
213
|
+
|
214
|
+
# Construct path relative to some root-path.
|
215
|
+
# Root-paths are expected to be ancestors of the yaml-file being parsed.
|
216
|
+
# They are found by traversing up and checking for specific files/folders, e.g. '.git' or 'Gemfile'.
|
217
|
+
# Any argument is appended to the root-path, constructing a path-instance that may exist.
|
218
|
+
class PathRootTag < BaseTag
|
219
|
+
# Config:
|
220
|
+
# config.add_tag("path/git_root", klass: PathRootTag[containing: ".git"])
|
221
|
+
# config.add_tag("path/rails_root", klass: PathRootTag[containing: "Gemfile"])
|
222
|
+
#
|
223
|
+
# YAML:
|
224
|
+
# project_root: !path/git_root
|
225
|
+
# config_path: !path/git_root [ config ]
|
226
|
+
def init_options(containing:)
|
227
|
+
super
|
228
|
+
end
|
125
229
|
|
126
|
-
|
127
|
-
|
128
|
-
|
230
|
+
def resolve(**)
|
231
|
+
# TODO validate upfront
|
232
|
+
raise <<~ERR unless root_path
|
233
|
+
#{tag_name}: failed to find root-path (ie an ancestor of #{ctx[:yaml_file]} containing #{options[:containing].inspect}).
|
234
|
+
ERR
|
235
|
+
root_path.join(*args).then(&config.fetch(:block, :itself.to_proc))
|
236
|
+
end
|
237
|
+
|
238
|
+
def root_path
|
239
|
+
find_up(ctx[:yaml_file], options[:containing])
|
240
|
+
end
|
241
|
+
|
242
|
+
def find_up(path, containing)
|
243
|
+
(path = path.parent) until path.root? || (path / containing).exist?
|
244
|
+
path unless path.root?
|
245
|
+
end
|
246
|
+
end
|
247
|
+
|
248
|
+
def self.add_default_tags!
|
249
|
+
configure do |config|
|
250
|
+
config.add_tag("ref") do |tag|
|
251
|
+
# validate: non-empty coder.seq, only strs, path must exists in ctx[:config]
|
252
|
+
|
253
|
+
path = tag.args.map(&:to_sym)
|
254
|
+
deep_resolve(tag.ctx[:yaml].dig(*path), **{})
|
129
255
|
end
|
130
256
|
|
131
|
-
config.add_tag("
|
132
|
-
|
257
|
+
config.add_tag("env", klass: EnvTag)
|
258
|
+
config.add_tag("env?", klass: EnvTag)
|
259
|
+
config.add_tag("env/float", klass: EnvTag[coerce: :to_f])
|
260
|
+
config.add_tag("env/float?", klass: EnvTag[coerce: :to_f])
|
261
|
+
|
262
|
+
config.add_tag("env/integer", klass: EnvTag[coerce: :to_i])
|
263
|
+
config.add_tag("env/integer?", klass: EnvTag[coerce: :to_i])
|
264
|
+
|
265
|
+
config.add_tag("env/bool", klass: EnvTag) do |tag|
|
266
|
+
EnvTag.coerce_bool(tag.env_value)
|
267
|
+
end
|
268
|
+
config.add_tag("env/bool?", klass: EnvTag) do |tag|
|
269
|
+
EnvTag.coerce_bool(tag.env_value)
|
133
270
|
end
|
134
271
|
|
135
|
-
config.add_tag("
|
136
|
-
|
272
|
+
config.add_tag("path") do |tag|
|
273
|
+
Pathname.new(tag.args.join("/"))
|
137
274
|
end
|
275
|
+
config.add_tag("path/git_root", klass: PathRootTag[containing: ".git"])
|
276
|
+
config.add_tag("path/rails_root", klass: PathRootTag[containing: "config.ru"])
|
138
277
|
|
139
|
-
config.add_tag("
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
278
|
+
config.add_tag("uri") do |tag|
|
279
|
+
URI.join(*tag.args.join)
|
280
|
+
end
|
281
|
+
|
282
|
+
config.add_tag("str/format") do |tag|
|
283
|
+
case tag.args
|
284
|
+
when Hash
|
285
|
+
fmt = tag.args.delete(:fmt)
|
286
|
+
sprintf(fmt, tag.args)
|
147
287
|
else
|
148
|
-
|
288
|
+
sprintf(*tag.args)
|
149
289
|
end
|
150
290
|
end
|
151
291
|
end
|
@@ -163,27 +303,24 @@ module Nero
|
|
163
303
|
end
|
164
304
|
reset_configuration!
|
165
305
|
|
166
|
-
def self.
|
167
|
-
|
168
|
-
|
169
|
-
|
306
|
+
def self.yaml_options
|
307
|
+
{
|
308
|
+
permitted_classes: [Symbol] + configuration.tags.values.map { _1[:klass] },
|
309
|
+
aliases: true
|
310
|
+
}
|
170
311
|
end
|
171
|
-
private_class_method :
|
172
|
-
|
173
|
-
@yaml_options = {
|
174
|
-
permitted_classes: [Symbol, TagResolver],
|
175
|
-
aliases: true
|
176
|
-
}
|
312
|
+
private_class_method :yaml_options
|
177
313
|
|
178
|
-
def self.load_config(file, root: nil)
|
314
|
+
def self.load_config(file, root: nil, env: nil, resolve: true)
|
315
|
+
root ||= env
|
179
316
|
add_tags!
|
180
317
|
|
181
|
-
|
318
|
+
config_file = resolve_file(file)
|
182
319
|
|
183
|
-
if
|
184
|
-
process_yaml(
|
320
|
+
if config_file.exist?
|
321
|
+
process_yaml(yaml_load_file(config_file, yaml_options), root:, config_file:, resolve:)
|
185
322
|
else
|
186
|
-
raise "Can't find file #{
|
323
|
+
raise "Can't find file #{config_file}"
|
187
324
|
end
|
188
325
|
end
|
189
326
|
|
@@ -197,27 +334,83 @@ module Nero
|
|
197
334
|
end
|
198
335
|
private_class_method :resolve_file
|
199
336
|
|
200
|
-
def self.load(raw, root: nil)
|
337
|
+
def self.load(raw, root: nil, env: nil, resolve: true)
|
338
|
+
root ||= env
|
201
339
|
add_tags!
|
202
340
|
|
203
|
-
process_yaml(
|
341
|
+
process_yaml(yaml_load(raw, yaml_options), root:, resolve:)
|
204
342
|
end
|
205
343
|
|
206
|
-
def self.process_yaml(yaml, root: nil)
|
344
|
+
def self.process_yaml(yaml, root: nil, resolve: true, config_file: nil)
|
345
|
+
config_file ||= (Pathname.pwd / __FILE__)
|
346
|
+
|
207
347
|
unresolved = Util.deep_symbolize_keys(yaml).then do
|
208
348
|
root ? _1[root.to_sym] : _1
|
209
349
|
end
|
350
|
+
ctx = {tags: configuration.tags, yaml: unresolved, yaml_file: config_file}
|
351
|
+
init_tags!(collect_tags(unresolved), ctx:)
|
210
352
|
|
211
|
-
|
353
|
+
return unresolved unless resolve
|
354
|
+
|
355
|
+
# NOTE originally ctx was passed at this point. Maybe delete this.
|
356
|
+
deep_resolve(unresolved, **{})
|
212
357
|
end
|
213
358
|
private_class_method :process_yaml
|
214
359
|
|
360
|
+
def self.init_tags!(tags, ctx:)
|
361
|
+
tags.each do |tag|
|
362
|
+
options = ctx.dig(:tags, tag.tag_name, :options) || {}
|
363
|
+
tag.init(ctx:, options:)
|
364
|
+
end
|
365
|
+
end
|
366
|
+
private_class_method :init_tags!
|
367
|
+
|
368
|
+
def self.yaml_load_file(file, opts = {})
|
369
|
+
if Psych::VERSION < "4"
|
370
|
+
YAML.load_file(file)
|
371
|
+
else
|
372
|
+
YAML.load_file(file, **opts)
|
373
|
+
end
|
374
|
+
end
|
375
|
+
private_class_method :yaml_load_file
|
376
|
+
|
377
|
+
def self.yaml_load(file, opts = {})
|
378
|
+
if Psych::VERSION < "4"
|
379
|
+
YAML.load(file)
|
380
|
+
else
|
381
|
+
YAML.load(file, **opts)
|
382
|
+
end
|
383
|
+
end
|
384
|
+
private_class_method :yaml_load
|
385
|
+
|
215
386
|
def self.add_tags!
|
216
|
-
configuration.tags.
|
217
|
-
YAML.add_tag(
|
387
|
+
configuration.tags.each do |tag_name, tag|
|
388
|
+
YAML.add_tag("!#{tag_name}", tag[:klass])
|
218
389
|
end
|
219
390
|
end
|
220
391
|
private_class_method :add_tags!
|
392
|
+
|
393
|
+
def self.collect_tags(obj)
|
394
|
+
case obj
|
395
|
+
when Hash
|
396
|
+
obj.each_value.flat_map { collect_tags(_1) }.compact
|
397
|
+
when Nero::BaseTag
|
398
|
+
[obj] +
|
399
|
+
case obj.coder.type
|
400
|
+
when :seq
|
401
|
+
collect_tags(obj.coder.seq)
|
402
|
+
when :map
|
403
|
+
collect_tags(obj.coder.map)
|
404
|
+
else
|
405
|
+
[]
|
406
|
+
end
|
407
|
+
when Array
|
408
|
+
obj.flat_map { collect_tags(_1) }.compact
|
409
|
+
else
|
410
|
+
[]
|
411
|
+
end
|
412
|
+
end
|
413
|
+
private_class_method :collect_tags
|
221
414
|
end
|
222
415
|
|
223
416
|
loader.eager_load if ENV.key?("CI")
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: nero
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Gert Goet
|
8
8
|
bindir: exe
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-03-20 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: zeitwerk
|
@@ -23,6 +23,20 @@ dependencies:
|
|
23
23
|
- - ">="
|
24
24
|
- !ruby/object:Gem::Version
|
25
25
|
version: '0'
|
26
|
+
- !ruby/object:Gem::Dependency
|
27
|
+
name: appraisal
|
28
|
+
requirement: !ruby/object:Gem::Requirement
|
29
|
+
requirements:
|
30
|
+
- - ">="
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '0'
|
33
|
+
type: :development
|
34
|
+
prerelease: false
|
35
|
+
version_requirements: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - ">="
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: '0'
|
26
40
|
description: |
|
27
41
|
Some convenient YAML-tags...
|
28
42
|
- to get environment values: env, env?, env/integer, env/integer?, env/bool, env/bool?.
|
@@ -39,10 +53,13 @@ files:
|
|
39
53
|
- ".envrc"
|
40
54
|
- ".rspec"
|
41
55
|
- ".standard.yml"
|
56
|
+
- Appraisals
|
42
57
|
- CHANGELOG.md
|
43
58
|
- LICENSE.txt
|
44
59
|
- README.md
|
45
60
|
- Rakefile
|
61
|
+
- gemfiles/psych_3.gemfile
|
62
|
+
- gemfiles/psych_4.gemfile
|
46
63
|
- lib/nero.rb
|
47
64
|
- lib/nero/util.rb
|
48
65
|
- lib/nero/version.rb
|