usable 3.7.1 → 3.9.3
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/lib/usable.rb +14 -2
- data/lib/usable/config.rb +20 -4
- data/lib/usable/mod_extender.rb +12 -3
- data/lib/usable/persistence.rb +82 -0
- data/lib/usable/struct.rb +26 -3
- data/lib/usable/version.rb +1 -1
- data/usable.gemspec +3 -3
- metadata +13 -13
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
|
-
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: bcc17af85e47b938b6f6f86196d7a1a244e062ac1e7c77c4b6260138c18d9bdb
|
4
|
+
data.tar.gz: d800dcd7e7d2cff57ba7f2d96e5168a388b503ae480b83980d67286b55d7d970
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 2c03ec68bf89ff9e4b9ac0d72e2a5a72b7977375169957ae4873462a3e77c68aa01bb91afd7d18415113c5f50c58e00648292ca6932c81849ddea5dab90858a0
|
7
|
+
data.tar.gz: e631aece8d224022d1d07c4e777d978111e987bb5a803ed9ea4309701ee553adf11ca516e77f02312a2523a53510ae7dfd699a84590de3027d9c3df470d3c404
|
data/lib/usable.rb
CHANGED
@@ -88,6 +88,15 @@ module Usable
|
|
88
88
|
super
|
89
89
|
end
|
90
90
|
|
91
|
+
def define_usable_accessors
|
92
|
+
usables.to_h.keys.each do |key|
|
93
|
+
define_singleton_method(key) { usables.send(key) }
|
94
|
+
define_singleton_method("#{key}=") { |new_val| usables.send("#{key}=", new_val) }
|
95
|
+
define_method(key) { usables.send(key) }
|
96
|
+
define_method("#{key}=") { |new_val| usables.send("#{key}=", new_val) }
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
91
100
|
def usables
|
92
101
|
@usables ||= Config.new
|
93
102
|
end
|
@@ -113,11 +122,14 @@ module Usable
|
|
113
122
|
# @return self
|
114
123
|
def usable(*args, &block)
|
115
124
|
options = args.last.is_a?(Hash) ? args.pop : {}
|
125
|
+
only = options.delete(:only)
|
126
|
+
extension_method = options.delete(:method)
|
116
127
|
args.each do |mod|
|
117
|
-
ModExtender.new(mod, only:
|
128
|
+
ModExtender.new(mod, only: only, method: extension_method).call self
|
118
129
|
# Define settings on @usables and on the scoped @usables
|
119
130
|
scope = Config.new
|
120
|
-
|
131
|
+
# Nest the new config under a namespace based on it's name, unless it's the default name we gave
|
132
|
+
if mod.name && !mod.name.include?("UsableMod")
|
121
133
|
scope_name = mod.name.split('::').last.gsub(/\B([A-Z])([a-z_0-9])/, '_\1\2').downcase
|
122
134
|
usables[scope_name] = scope
|
123
135
|
end
|
data/lib/usable/config.rb
CHANGED
@@ -30,8 +30,12 @@ module Usable
|
|
30
30
|
@spec.to_h.each(&block)
|
31
31
|
end
|
32
32
|
|
33
|
+
# @note The @spec[key] could already be set in cases where the config was merged with another
|
34
|
+
# For example, a usable module defines a default with a block, and when mounting the module
|
35
|
+
# the same usable option is overridden as an options hash (usable mod, key: 'val')
|
33
36
|
def to_h
|
34
|
-
@lazy_loads.each { |key| @spec[key]
|
37
|
+
@lazy_loads.each { |key| @spec[key] ||= @spec.public_send(key) }
|
38
|
+
@lazy_loads.clear
|
35
39
|
@spec.to_h
|
36
40
|
end
|
37
41
|
|
@@ -43,10 +47,22 @@ module Usable
|
|
43
47
|
to_h.merge(other)
|
44
48
|
end
|
45
49
|
|
50
|
+
# @param other [Hash] to update ourselves with
|
51
|
+
def merge!(other)
|
52
|
+
other.each do |key, val|
|
53
|
+
@spec[key] = val
|
54
|
+
end
|
55
|
+
self
|
56
|
+
end
|
57
|
+
|
58
|
+
def include?(key)
|
59
|
+
!!@spec[key] || @lazy_loads.include?(key.to_sym)
|
60
|
+
end
|
61
|
+
|
46
62
|
def method_missing(key, *args, &block)
|
47
63
|
if block
|
48
64
|
@lazy_loads << key
|
49
|
-
@spec.define_singleton_method(key)
|
65
|
+
@spec.define_singleton_method(key, &block)
|
50
66
|
else
|
51
67
|
# Needs to be a symbol so we can consistently access @lazy_loads
|
52
68
|
key = key.to_s.tr('=', '').to_sym
|
@@ -55,10 +71,10 @@ module Usable
|
|
55
71
|
# Cleanup, just in case we loaded it another way (e.g. combining with another usable config)
|
56
72
|
@lazy_loads.delete key
|
57
73
|
else
|
58
|
-
@spec[key] = call_spec_method(key)
|
74
|
+
@spec[key] = call_spec_method(key) unless frozen?
|
59
75
|
end
|
60
76
|
# Define method so we don't hit method missing again
|
61
|
-
define_singleton_method(key) { @spec[key] }
|
77
|
+
define_singleton_method(key) { @spec[key] } unless frozen?
|
62
78
|
@spec[key]
|
63
79
|
else
|
64
80
|
@spec[key] = args.first
|
data/lib/usable/mod_extender.rb
CHANGED
@@ -18,9 +18,7 @@ module Usable
|
|
18
18
|
# @description Directly include a module whose methods you want made available in +usables.available_methods+
|
19
19
|
# Gives the module a name when including so that it shows up properly in the list of ancestors
|
20
20
|
def call(target)
|
21
|
-
unwanted.each
|
22
|
-
copy.send :remove_method, method_name
|
23
|
-
end
|
21
|
+
unwanted.each(&method(:remove_from_module))
|
24
22
|
if copy.name.nil?
|
25
23
|
const_name = "#{mod_name}Used"
|
26
24
|
target.send :remove_const, const_name if target.const_defined? const_name, false
|
@@ -46,5 +44,16 @@ module Usable
|
|
46
44
|
@copy.instance_methods - Array(only)
|
47
45
|
end
|
48
46
|
end
|
47
|
+
|
48
|
+
def remove_from_module(method_name)
|
49
|
+
begin
|
50
|
+
copy.send :remove_method, method_name
|
51
|
+
rescue NameError => e
|
52
|
+
# Expected sometimes, like it could be raised trying to remove a superclass method
|
53
|
+
Usable.logger.debug("#{e.class}: #{e.message}")
|
54
|
+
end
|
55
|
+
# Block access to superclass method, and prevent it from being copied to the target
|
56
|
+
copy.send :undef_method, method_name if copy.instance_methods.include?(method_name)
|
57
|
+
end
|
49
58
|
end
|
50
59
|
end
|
@@ -0,0 +1,82 @@
|
|
1
|
+
require 'fileutils'
|
2
|
+
require 'yaml'
|
3
|
+
|
4
|
+
module Usable
|
5
|
+
module Persistence
|
6
|
+
# = Stores usables in a yaml file
|
7
|
+
|
8
|
+
extend Usable
|
9
|
+
|
10
|
+
config do
|
11
|
+
dir { defined?(Rails) ? Rails.root.join('tmp') : File.expand_path('..', __FILE__) }
|
12
|
+
end
|
13
|
+
|
14
|
+
def self.extended(base)
|
15
|
+
base.extend Usable
|
16
|
+
base.usable self
|
17
|
+
end
|
18
|
+
|
19
|
+
# Automatically copied over by +usable+
|
20
|
+
module InstanceMethods
|
21
|
+
# Accessor to read and write default ActiveRecord objects
|
22
|
+
# When assigning a config, a reference to it is saved to tmp/<mod_name>.yml
|
23
|
+
# and loaded in subsequent console sessions
|
24
|
+
def method_missing(name, *args, &block)
|
25
|
+
if block
|
26
|
+
usables.public_send(name, &block)
|
27
|
+
elsif name.to_s.end_with?('=') && args.length == 1
|
28
|
+
_save name.to_s.tr('=', '').to_sym, args.pop
|
29
|
+
elsif usables[name]
|
30
|
+
usables[name]
|
31
|
+
elsif _config[name]
|
32
|
+
usables[name] = if _config[name].is_a?(Hash)
|
33
|
+
_config[name].fetch(:class).constantize.find(_config[name].fetch(:id))
|
34
|
+
else
|
35
|
+
_config[name]
|
36
|
+
end
|
37
|
+
else
|
38
|
+
usables.public_send(name) || super
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
def has?(setting)
|
43
|
+
!!send(setting)
|
44
|
+
rescue NoMethodError => e
|
45
|
+
if e.message.include?("method `#{setting}'")
|
46
|
+
false
|
47
|
+
else
|
48
|
+
raise
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
#
|
53
|
+
# Private
|
54
|
+
#
|
55
|
+
|
56
|
+
def _save(key, val = nil)
|
57
|
+
if val.nil?
|
58
|
+
_config.delete(key)
|
59
|
+
elsif val.respond_to?(:persisted?) && val.persisted?
|
60
|
+
_config[key] = { class: val.class.name, id: val.id }
|
61
|
+
else
|
62
|
+
_config[key] = val
|
63
|
+
end
|
64
|
+
File.open(_config_file, 'wb') { |f| f.puts _config.to_yaml }
|
65
|
+
usables[key] = val
|
66
|
+
end
|
67
|
+
|
68
|
+
def _config_file
|
69
|
+
return @_config_file if @_config_file
|
70
|
+
FileUtils.mkdir_p(usables.dir) unless File.directory?(usables.dir)
|
71
|
+
filename = self.class.name ? self.class.name.downcase.gsub('::', '_') : 'usable'
|
72
|
+
@_config_file = File.join(usables.dir, "#{filename}.yml")
|
73
|
+
FileUtils.touch(@_config_file) unless File.exists?(@_config_file)
|
74
|
+
@_config_file
|
75
|
+
end
|
76
|
+
|
77
|
+
def _config
|
78
|
+
@_config ||= YAML.load_file(_config_file) || {}
|
79
|
+
end
|
80
|
+
end
|
81
|
+
end
|
82
|
+
end
|
data/lib/usable/struct.rb
CHANGED
@@ -2,10 +2,9 @@ module Usable
|
|
2
2
|
def self.Struct(attributes = {})
|
3
3
|
Class.new do
|
4
4
|
extend Usable
|
5
|
-
|
6
5
|
self.usables = Usable::Config.new(attributes)
|
7
|
-
|
8
|
-
attributes.keys.each do |key|
|
6
|
+
define_usable_accessors
|
7
|
+
attributes.keys.map(&:to_sym).each do |key|
|
9
8
|
define_method(key) { @attrs[key] }
|
10
9
|
define_method("#{key}=") { |new_val| @attrs[key] = new_val }
|
11
10
|
end
|
@@ -15,6 +14,30 @@ module Usable
|
|
15
14
|
def initialize(attrs = {})
|
16
15
|
@attrs = usables.merge(attrs)
|
17
16
|
end
|
17
|
+
|
18
|
+
def [](key)
|
19
|
+
@attrs[key]
|
20
|
+
end
|
21
|
+
|
22
|
+
def []=(key, val)
|
23
|
+
@attrs[key] = val
|
24
|
+
end
|
25
|
+
|
26
|
+
def each(&block)
|
27
|
+
@attrs.each(&block)
|
28
|
+
end
|
29
|
+
|
30
|
+
def to_h
|
31
|
+
@attrs.dup
|
32
|
+
end
|
33
|
+
|
34
|
+
alias to_hash to_h
|
35
|
+
|
36
|
+
def merge(other)
|
37
|
+
to_h.merge!(other)
|
38
|
+
end
|
39
|
+
|
40
|
+
alias + merge
|
18
41
|
end
|
19
42
|
end
|
20
43
|
end
|
data/lib/usable/version.rb
CHANGED
data/usable.gemspec
CHANGED
@@ -21,8 +21,8 @@ Gem::Specification.new do |spec|
|
|
21
21
|
|
22
22
|
spec.required_ruby_version = '>= 2.0.0'
|
23
23
|
|
24
|
-
spec.add_development_dependency
|
25
|
-
spec.add_development_dependency
|
26
|
-
spec.add_development_dependency
|
24
|
+
spec.add_development_dependency "bundler", "~> 2.2.11"
|
25
|
+
spec.add_development_dependency "rake", ">= 12.3.3"
|
26
|
+
spec.add_development_dependency "rspec", ">= 3.2", "< 4"
|
27
27
|
# spec.add_development_dependency 'rspec-scaffold', '>= 1.0', '< 2'
|
28
28
|
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: usable
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 3.
|
4
|
+
version: 3.9.3
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Ryan Buckley
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: exe
|
10
10
|
cert_chain: []
|
11
|
-
date:
|
11
|
+
date: 2021-04-01 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: bundler
|
@@ -16,28 +16,28 @@ dependencies:
|
|
16
16
|
requirements:
|
17
17
|
- - "~>"
|
18
18
|
- !ruby/object:Gem::Version
|
19
|
-
version:
|
19
|
+
version: 2.2.11
|
20
20
|
type: :development
|
21
21
|
prerelease: false
|
22
22
|
version_requirements: !ruby/object:Gem::Requirement
|
23
23
|
requirements:
|
24
24
|
- - "~>"
|
25
25
|
- !ruby/object:Gem::Version
|
26
|
-
version:
|
26
|
+
version: 2.2.11
|
27
27
|
- !ruby/object:Gem::Dependency
|
28
28
|
name: rake
|
29
29
|
requirement: !ruby/object:Gem::Requirement
|
30
30
|
requirements:
|
31
|
-
- - "
|
31
|
+
- - ">="
|
32
32
|
- !ruby/object:Gem::Version
|
33
|
-
version:
|
33
|
+
version: 12.3.3
|
34
34
|
type: :development
|
35
35
|
prerelease: false
|
36
36
|
version_requirements: !ruby/object:Gem::Requirement
|
37
37
|
requirements:
|
38
|
-
- - "
|
38
|
+
- - ">="
|
39
39
|
- !ruby/object:Gem::Version
|
40
|
-
version:
|
40
|
+
version: 12.3.3
|
41
41
|
- !ruby/object:Gem::Dependency
|
42
42
|
name: rspec
|
43
43
|
requirement: !ruby/object:Gem::Requirement
|
@@ -71,6 +71,7 @@ files:
|
|
71
71
|
- lib/usable/config_register.rb
|
72
72
|
- lib/usable/eager_load.rb
|
73
73
|
- lib/usable/mod_extender.rb
|
74
|
+
- lib/usable/persistence.rb
|
74
75
|
- lib/usable/railtie.rb
|
75
76
|
- lib/usable/struct.rb
|
76
77
|
- lib/usable/version.rb
|
@@ -79,7 +80,7 @@ homepage: https://github.com/ridiculous/usable
|
|
79
80
|
licenses:
|
80
81
|
- MIT
|
81
82
|
metadata: {}
|
82
|
-
post_install_message:
|
83
|
+
post_install_message:
|
83
84
|
rdoc_options: []
|
84
85
|
require_paths:
|
85
86
|
- lib
|
@@ -94,9 +95,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
94
95
|
- !ruby/object:Gem::Version
|
95
96
|
version: '0'
|
96
97
|
requirements: []
|
97
|
-
|
98
|
-
|
99
|
-
signing_key:
|
98
|
+
rubygems_version: 3.2.11
|
99
|
+
signing_key:
|
100
100
|
specification_version: 4
|
101
101
|
summary: Mounts and configures modules
|
102
102
|
test_files: []
|