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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 3da331b2512fc63cf1a9258991a33dda876e8b0aed2445bca34ed207f8cfc1fb
4
- data.tar.gz: 79c6afff69d21841b247de206f7b3f6482bf74496ad91c4986528fd0620e6551
3
+ metadata.gz: 4e28a4b8bcd989c20dd60dc592bfb31039314437b3d8ca5df33b79adefd5cb2e
4
+ data.tar.gz: f6eb0ec1e9ceb041088a7d2b850494e090f08d98f4e8369f82f0608a4449ab69
5
5
  SHA512:
6
- metadata.gz: e80cb4098d709df1231e7ee1cfc077be097517709ffabee489cdce8bc9da30aac7e6f89157823c147e691042945c9da806f337f1ca64ab90dc376d390f46d0b9
7
- data.tar.gz: 7637f0ae5983c3902a04092c2d1cf3ed9aacb0ae036f73d7c1067f7677a8c175ffe4c4826c972cb129c47cfe37517115e0bd51f81c1f6b4f132a8946b4311a3e
6
+ metadata.gz: 98dc2c475e0e164f47d1738dbe32d6dfeeba6b531f6db0b396a25d3e29219c312ac0750206fddea1d9ee450e9205425a5f0342c6b5b13d38867a364e1219bed0
7
+ data.tar.gz: e9daf1ee287679b10d4ea744c41d3f93bf50570394eb261f7c8b3587a1008efe57b58b7957141aeea48229b65efdaa614f23c181fbf08659a0a878a7a944e773
data/Appraisals ADDED
@@ -0,0 +1,7 @@
1
+ appraise "psych-3" do
2
+ gem "psych", "< 4"
3
+ end
4
+
5
+ appraise "psych-4" do
6
+ gem "psych", "< 5"
7
+ end
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
  [![Gem Version](https://badge.fury.io/rb/nero.svg)](https://badge.fury.io/rb/nero)
4
4
 
5
- Nero is a RubyGem that offers predefined tags and allows you to effortlessly create custom ones for YAML configuration files.
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
- E.g. instead of having the following settings file in your Rails project:
8
+ **Sample:**
8
9
 
9
10
  ```yaml
10
11
  development:
11
- secret: <%= ENV.fetch("SECRET", "dummy") %>
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 _only_ when loading production
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
- # using references
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
- Add one yourself:
139
- ```ruby
140
- Nero.configure do |nero|
141
- nero.add_tag("foo") do |coder|
142
- # coder.type is one of :scalar, :seq or :map
143
- # e.g. respective YAML:
144
- # ---
145
- # !foo bar
146
- # ---
147
- # !foo
148
- # - bar
149
- # ---
150
- # !foo
151
- # bar: baz
152
- #
153
- # Find the value in the respective attribute, e.g. `coder.scalar`:
154
- coder.scalar.upcase
155
- end
156
-
157
- # Other configuration options:
158
- #
159
- # `config_dir` (default: Pathname.pwd) - path used for expanding non-Pathnames passed to `load_config`, e.g.
160
- # `Nero.load_config(:app)` loads file `Pathname.pwd / "app.yml"`.
161
- nero.config_dir = Rails.root / "config"
162
- end
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
 
@@ -0,0 +1,10 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rake", "~> 13.0"
6
+ gem "rspec", "~> 3.0"
7
+ gem "standard", "~> 1.3"
8
+ gem "psych", "< 4"
9
+
10
+ gemspec path: "../"
@@ -0,0 +1,10 @@
1
+ # This file was generated by Appraisal
2
+
3
+ source "https://rubygems.org"
4
+
5
+ gem "rake", "~> 13.0"
6
+ gem "rspec", "~> 3.0"
7
+ gem "standard", "~> 1.3"
8
+ gem "psych", "< 5"
9
+
10
+ gemspec path: "../"
data/lib/nero/version.rb CHANGED
@@ -3,5 +3,5 @@
3
3
  module Nero
4
4
  # NOTE this is written upon release via:
5
5
  # $ rake gem:build[version=0.3.0]
6
- VERSION = "0.3.0"
6
+ VERSION = "0.5.0"
7
7
  end
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" # why needed?
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 TagResolver
40
- include Resolvable
49
+ class Configuration
50
+ attr_reader :tags, :config_dir
41
51
 
42
- def init_with(coder)
43
- @coder = coder
52
+ def config_dir=(dir)
53
+ @config_dir = Pathname(dir).expand_path
44
54
  end
45
55
 
46
- def resolve(ctx)
47
- resolve_nested!(ctx)
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
- def resolve_nested!(ctx)
52
- case @coder.type
53
- when :seq
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
- def self.add_default_tags!
79
- configure do |config|
80
- config.add_tag("env/integer") do |coder|
81
- Integer(env_fetch(*(coder.scalar || coder.seq), all_optional: "999"))
82
- end
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
- config.add_tag("env/integer?") do |coder|
85
- Integer(ENV[coder.scalar]) if ENV[coder.scalar]
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
- config.add_tag("env/bool") do |coder|
89
- re_true = /y|Y|yes|Yes|YES|true|True|TRUE|on|On|ON/
90
- re_false = /n|N|no|No|NO|false|False|FALSE|off|Off|OFF/
91
-
92
- coerce = ->(s) do
93
- case s
94
- when TrueClass, FalseClass then s
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
- coerce[env_fetch(*(coder.scalar || coder.seq), all_optional: "false")]
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
- config.add_tag("env/bool?") do |coder|
106
- re_true = /y|Y|yes|Yes|YES|true|True|TRUE|on|On|ON/
107
- re_false = /n|N|no|No|NO|false|False|FALSE|off|Off|OFF/
108
-
109
- coerce = ->(s) do
110
- case s
111
- when TrueClass, FalseClass then s
112
- when re_true then true
113
- when re_false then false
114
- else
115
- raise "bool value should be one of y(es)/n(o), on/off, true/false (got #{s.inspect})"
116
- end
117
- end
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
- ENV[coder.scalar] ? coerce[ENV[coder.scalar]] : false
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
- config.add_tag("env") do |coder|
123
- env_fetch(*(coder.scalar || coder.seq))
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
- config.add_tag("env?") do |coder|
127
- fetch_args = coder.scalar ? [coder.scalar, nil] : coder.seq
128
- ENV.fetch(*fetch_args)
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("path") do |coder|
132
- Pathname.new(coder.scalar || coder.seq.join("/"))
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("uri") do |coder|
136
- URI(coder.scalar || coder.seq.join)
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("str/format") do |coder|
140
- case coder.type
141
- when :seq
142
- sprintf(*coder.seq)
143
- when :map
144
- m = Util.deep_symbolize_keys(coder.map)
145
- fmt = m.delete(:fmt)
146
- sprintf(fmt, m)
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
- coder.scalar
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.env_fetch(k, fallback = nil, all_optional: "dummy")
167
- fallback ||= all_optional if ENV["NERO_ENV_ALL_OPTIONAL"]
168
-
169
- fallback.nil? ? ENV.fetch(k) : ENV.fetch(k, fallback)
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 :env_fetch
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
- file = resolve_file(file)
318
+ config_file = resolve_file(file)
182
319
 
183
- if file.exist?
184
- process_yaml(YAML.load_file(file, **@yaml_options), root:)
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 #{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(YAML.load(raw, **@yaml_options), root:)
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
- deep_resolve(unresolved, tags: configuration.tags)
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.keys.each do
217
- YAML.add_tag(_1, TagResolver)
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.3.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-02-02 00:00:00.000000000 Z
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