dry-types 1.0.1 → 1.1.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +33 -3
- data/Gemfile +4 -3
- data/Rakefile +2 -2
- data/benchmarks/hash_schemas.rb +8 -6
- data/benchmarks/lax_schema.rb +0 -1
- data/benchmarks/profile_invalid_input.rb +1 -1
- data/benchmarks/profile_lax_schema_valid.rb +1 -1
- data/benchmarks/profile_valid_input.rb +1 -1
- data/dry-types.gemspec +19 -19
- data/lib/dry/types.rb +2 -2
- data/lib/dry/types/array.rb +6 -0
- data/lib/dry/types/array/constructor.rb +32 -0
- data/lib/dry/types/array/member.rb +7 -0
- data/lib/dry/types/builder.rb +1 -1
- data/lib/dry/types/builder_methods.rb +15 -15
- data/lib/dry/types/coercions.rb +19 -6
- data/lib/dry/types/coercions/params.rb +4 -4
- data/lib/dry/types/constrained.rb +5 -0
- data/lib/dry/types/constructor.rb +3 -8
- data/lib/dry/types/constructor/function.rb +4 -5
- data/lib/dry/types/core.rb +27 -8
- data/lib/dry/types/enum.rb +1 -0
- data/lib/dry/types/extensions/maybe.rb +9 -1
- data/lib/dry/types/hash.rb +8 -10
- data/lib/dry/types/hash/constructor.rb +5 -5
- data/lib/dry/types/json.rb +4 -0
- data/lib/dry/types/lax.rb +3 -3
- data/lib/dry/types/map.rb +8 -4
- data/lib/dry/types/module.rb +3 -3
- data/lib/dry/types/nominal.rb +3 -4
- data/lib/dry/types/params.rb +4 -0
- data/lib/dry/types/printer.rb +11 -10
- data/lib/dry/types/schema.rb +14 -20
- data/lib/dry/types/schema/key.rb +19 -1
- data/lib/dry/types/spec/types.rb +3 -6
- data/lib/dry/types/version.rb +1 -1
- metadata +40 -33
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 689634e9152c629af622ef69379633b674b6c60b7f5cd87093894de7ee8ce571
|
4
|
+
data.tar.gz: a23593e6c4d904343dadd46f919cfead42b45fbe963995380712f4c6a8e719fd
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 88fd3e79ea403515fe242baa57c6824757bdc0b04f99054a9da2394c745cce54415fc3c260b35e49eed8a5240f78614efe92e052f6e6bde0e1f9b961a04f41ac
|
7
|
+
data.tar.gz: 3445b5726dd5325261965409883b6830450f3d710ae2b0a7be411b0d45cb25d3550a25827d9f0661dfbfd0fd55f15954d77335312237c3bf754ca7af5eae6079
|
data/CHANGELOG.md
CHANGED
@@ -1,8 +1,30 @@
|
|
1
|
+
# 1.1.0 2019-07-02
|
2
|
+
|
3
|
+
## Added
|
4
|
+
|
5
|
+
- New builder method `Interface` constructs a type which accepts objects that respond to the given methods (waiting-for-dev)
|
6
|
+
```ruby
|
7
|
+
Types = Dry.Types()
|
8
|
+
Types::Callable = Types.Interface(:call)
|
9
|
+
Types::Callable.valid?(Object.new) # => false
|
10
|
+
Types::Callable.valid?(proc {}) # => true
|
11
|
+
```
|
12
|
+
- New types: `coercible.symbol`, `params.symbol`, and `json.symbol`, all use `.to_sym` for coercion (waiting-for-dev)
|
13
|
+
|
14
|
+
## Fixed
|
15
|
+
|
16
|
+
- Converting schema keys to maybe types (flash-gordon)
|
17
|
+
- Using `Schema#key` and `Array#member` on constuctors (flash-gordon)
|
18
|
+
- Using `meta(omittable: true)` within `transform_types` works again but produces a warning, please migrate to `.omittable` or `.required(false)` (flash-gordon)
|
19
|
+
- Bug with a constructror defined on top of enum (flash-gordon)
|
20
|
+
|
21
|
+
[Compare v1.0.1...v1.1.0](https://github.com/dry-rb/dry-types/compare/v1.0.1...v1.1.0)
|
22
|
+
|
1
23
|
# 1.0.1 2019-06-04
|
2
24
|
|
3
25
|
## Added
|
4
26
|
|
5
|
-
- In
|
27
|
+
- In a case of failure the constructor block can now pass a different value (flash-gordon)
|
6
28
|
```ruby
|
7
29
|
not_empty_string = Types::String.constructor do |value, &failure|
|
8
30
|
value.strip.empty? ? failure.(nil) : value.strip
|
@@ -21,6 +43,7 @@
|
|
21
43
|
|
22
44
|
- [BREAKING] Behavior of built-in constructor types was changed to be more strict. They will always raise an error on failed coercion (flash-gordon)
|
23
45
|
Compare:
|
46
|
+
|
24
47
|
```ruby
|
25
48
|
# 0.15.0
|
26
49
|
Types::Params::Integer.('foo')
|
@@ -36,7 +59,9 @@
|
|
36
59
|
```ruby
|
37
60
|
Types::Params::Integer.('foo') { :invalid } # => :invalid
|
38
61
|
```
|
62
|
+
|
39
63
|
This makes work with coercions more straightforward and way faster.
|
64
|
+
|
40
65
|
- [BREAKING] Safe types were renamed to Lax, this name better serves their purpose. The previous name is available but prints a warning (flash-gordon)
|
41
66
|
- [BREAKING] Metadata is now pushed down to the decorated type. It is not likely you will notice a difference but this a breaking change that enables some use cases in rom related to the usage of default types in relations (flash-gordon)
|
42
67
|
- Nominal types are now completely unconstrained. This fixes some inconsistencies when using them with constraints. `Nominal#try` will always return a successful result, for the previous behavior use `Nominal#try_coerce` or switch to strict types with passing a block to `#call` (flash-gordon)
|
@@ -159,6 +184,7 @@
|
|
159
184
|
)
|
160
185
|
```
|
161
186
|
- Key types have API for making keys omittable and back (flash-gordon)
|
187
|
+
|
162
188
|
```ruby
|
163
189
|
# defining a base schema with optional keys
|
164
190
|
lax_hash = Dry::Types['hash'].with_type_transform { |key| key.required(false) }
|
@@ -168,8 +194,10 @@
|
|
168
194
|
# keys in user_schema are not required
|
169
195
|
user_schema = lax_hash.schema(name: 'string', age: 'integer')
|
170
196
|
```
|
197
|
+
|
171
198
|
- `Type#optional?` now recognizes more cases where `nil` is an allowed value (flash-gordon)
|
172
199
|
- `Constructor#{prepend,append}` with `<<` and `>>` as aliases. `Constructor#append` works the same way `Constructor#constrcutor` does. `Constuctor#prepend` chains functions in the reverse order, see examples (flash-gordon)
|
200
|
+
|
173
201
|
```ruby
|
174
202
|
to_int = Types::Coercible::Integer
|
175
203
|
inc = to_int.append { |x| x + 2 }
|
@@ -178,6 +206,7 @@
|
|
178
206
|
inc = to_int.prepend { |x| x + "2" }
|
179
207
|
inc.("1") # => "1" -> "12" -> 12
|
180
208
|
```
|
209
|
+
|
181
210
|
- Partial schema application for cases when you want to validate only a subset of keys (flash-gordon)
|
182
211
|
This is useful when you want to update a key or two in an already-validated hash. A perfect example is `Dry::Struct#new` where this feature is now used.
|
183
212
|
```ruby
|
@@ -189,14 +218,15 @@
|
|
189
218
|
|
190
219
|
## Fixed
|
191
220
|
|
192
|
-
|
193
|
-
|
221
|
+
- `Hash::Map` now behaves as a constrained type if its values are constrained (flash-gordon)
|
222
|
+
- `coercible.integer` now doesn't blow up on invalid strings (exterm)
|
194
223
|
|
195
224
|
[Compare v0.14.0...v0.15.0](https://github.com/dry-rb/dry-types/compare/v0.14.0...v0.15.0)
|
196
225
|
|
197
226
|
# v0.14.1 2019-03-25
|
198
227
|
|
199
228
|
## Fixed
|
229
|
+
|
200
230
|
- `coercible.integer` now doesn't blow up on invalid strings (exterm)
|
201
231
|
|
202
232
|
[Compare v0.14.0...v0.14.1](https://github.com/dry-rb/dry-types/compare/v0.14.0...v0.14.1)
|
data/Gemfile
CHANGED
@@ -14,13 +14,14 @@ end
|
|
14
14
|
|
15
15
|
group :tools do
|
16
16
|
gem 'pry-byebug', platform: :mri
|
17
|
+
gem 'rubocop'
|
17
18
|
end
|
18
19
|
|
19
20
|
group :benchmarks do
|
21
|
+
gem 'attrio'
|
20
22
|
gem 'benchmark-ips'
|
23
|
+
gem 'dry-struct'
|
24
|
+
gem 'fast_attributes'
|
21
25
|
gem 'hotch'
|
22
26
|
gem 'virtus'
|
23
|
-
gem 'fast_attributes'
|
24
|
-
gem 'attrio'
|
25
|
-
gem 'dry-struct'
|
26
27
|
end
|
data/Rakefile
CHANGED
data/benchmarks/hash_schemas.rb
CHANGED
@@ -7,11 +7,13 @@ require 'dry-types'
|
|
7
7
|
|
8
8
|
module SchemaBench
|
9
9
|
def self.hash_schema(type)
|
10
|
-
Dry::Types['nominal.hash'].public_send(
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
10
|
+
Dry::Types['nominal.hash'].public_send(
|
11
|
+
type,
|
12
|
+
email: Dry::Types['nominal.string'],
|
13
|
+
age: Dry::Types['params.integer'],
|
14
|
+
admin: Dry::Types['params.bool'],
|
15
|
+
address: Dry::Types['nominal.hash'].public_send(
|
16
|
+
type,
|
15
17
|
city: Dry::Types['nominal.string'],
|
16
18
|
street: Dry::Types['nominal.string']
|
17
19
|
)
|
@@ -31,7 +33,7 @@ module SchemaBench
|
|
31
33
|
age: '20',
|
32
34
|
admin: '1',
|
33
35
|
address: { city: 'NYC', street: 'Street 1/2' }
|
34
|
-
}
|
36
|
+
}.freeze
|
35
37
|
end
|
36
38
|
|
37
39
|
require 'benchmark/ips'
|
data/benchmarks/lax_schema.rb
CHANGED
data/dry-types.gemspec
CHANGED
@@ -1,47 +1,47 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
lib = File.expand_path('
|
3
|
+
lib = File.expand_path('lib', __dir__)
|
4
4
|
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
5
5
|
require 'dry/types/version'
|
6
6
|
|
7
7
|
Gem::Specification.new do |spec|
|
8
|
-
spec.name =
|
8
|
+
spec.name = 'dry-types'
|
9
9
|
spec.version = Dry::Types::VERSION.dup
|
10
|
-
spec.authors = [
|
11
|
-
spec.email = [
|
10
|
+
spec.authors = ['Piotr Solnica']
|
11
|
+
spec.email = ['piotr.solnica@gmail.com']
|
12
12
|
spec.license = 'MIT'
|
13
13
|
|
14
14
|
spec.summary = 'Type system for Ruby supporting coercions, constraints and complex types like structs, value objects, enums etc.'
|
15
15
|
spec.description = spec.summary
|
16
|
-
spec.homepage =
|
16
|
+
spec.homepage = 'https://github.com/dry-rb/dry-types'
|
17
17
|
|
18
18
|
# Prevent pushing this gem to RubyGems.org by setting 'allowed_push_host', or
|
19
19
|
# delete this section to allow pushing this gem to any host.
|
20
20
|
if spec.respond_to?(:metadata)
|
21
|
-
spec.metadata['allowed_push_host'] =
|
22
|
-
spec.metadata['changelog_uri'] =
|
23
|
-
spec.metadata['source_code_uri'] =
|
24
|
-
spec.metadata['bug_tracker_uri'] =
|
21
|
+
spec.metadata['allowed_push_host'] = 'https://rubygems.org'
|
22
|
+
spec.metadata['changelog_uri'] = 'https://github.com/dry-rb/dry-types/blob/master/CHANGELOG.md'
|
23
|
+
spec.metadata['source_code_uri'] = 'https://github.com/dry-rb/dry-types'
|
24
|
+
spec.metadata['bug_tracker_uri'] = 'https://github.com/dry-rb/dry-types/issues'
|
25
25
|
else
|
26
|
-
raise
|
26
|
+
raise 'RubyGems 2.0 or newer is required to protect against public gem pushes.'
|
27
27
|
end
|
28
28
|
|
29
29
|
spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) } - ['bin/console', 'bin/setup']
|
30
|
-
spec.bindir =
|
30
|
+
spec.bindir = 'exe'
|
31
31
|
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
32
|
-
spec.require_paths = [
|
33
|
-
spec.required_ruby_version =
|
32
|
+
spec.require_paths = ['lib']
|
33
|
+
spec.required_ruby_version = '>= 2.4.0'
|
34
34
|
|
35
35
|
spec.add_runtime_dependency 'concurrent-ruby', '~> 1.0'
|
36
|
-
spec.add_runtime_dependency 'dry-core', '~> 0.4', '>= 0.4.4'
|
37
|
-
spec.add_runtime_dependency 'dry-inflector', '~> 0.1', '>= 0.1.2'
|
38
36
|
spec.add_runtime_dependency 'dry-container', '~> 0.3'
|
37
|
+
spec.add_runtime_dependency 'dry-core', '~> 0.4', '>= 0.4.4'
|
39
38
|
spec.add_runtime_dependency 'dry-equalizer', '~> 0.2', '>= 0.2.2'
|
40
|
-
spec.add_runtime_dependency 'dry-
|
39
|
+
spec.add_runtime_dependency 'dry-inflector', '~> 0.1', '>= 0.1.2'
|
40
|
+
spec.add_runtime_dependency 'dry-logic', '~> 1.0', '>= 1.0.2'
|
41
41
|
|
42
|
-
spec.add_development_dependency
|
43
|
-
spec.add_development_dependency "rake", "~> 11.0"
|
44
|
-
spec.add_development_dependency "rspec", "~> 3.3"
|
42
|
+
spec.add_development_dependency 'bundler'
|
45
43
|
spec.add_development_dependency 'dry-monads', '~> 0.2'
|
44
|
+
spec.add_development_dependency 'rake', '~> 11.0'
|
45
|
+
spec.add_development_dependency 'rspec', '~> 3.3'
|
46
46
|
spec.add_development_dependency 'yard', '~> 0.9.5'
|
47
47
|
end
|
data/lib/dry/types.rb
CHANGED
@@ -33,7 +33,7 @@ module Dry
|
|
33
33
|
extend Dry::Core::Deprecations[:'dry-types']
|
34
34
|
include Dry::Core::Constants
|
35
35
|
|
36
|
-
TYPE_SPEC_REGEX =
|
36
|
+
TYPE_SPEC_REGEX = /(.+)<(.+)>/.freeze
|
37
37
|
|
38
38
|
# @see Dry.Types
|
39
39
|
def self.module(*namespaces, default: :nominal, **aliases)
|
@@ -45,7 +45,7 @@ module Dry
|
|
45
45
|
|
46
46
|
# @api private
|
47
47
|
def self.included(*)
|
48
|
-
raise
|
48
|
+
raise 'Import Dry.Types, not Dry::Types'
|
49
49
|
end
|
50
50
|
|
51
51
|
# Return container with registered built-in type objects
|
data/lib/dry/types/array.rb
CHANGED
@@ -1,6 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
3
|
require 'dry/types/array/member'
|
4
|
+
require 'dry/types/array/constructor'
|
4
5
|
|
5
6
|
module Dry
|
6
7
|
module Types
|
@@ -24,6 +25,11 @@ module Dry
|
|
24
25
|
|
25
26
|
Array::Member.new(primitive, **options, member: member)
|
26
27
|
end
|
28
|
+
|
29
|
+
# @api private
|
30
|
+
def constructor_type
|
31
|
+
::Dry::Types::Array::Constructor
|
32
|
+
end
|
27
33
|
end
|
28
34
|
end
|
29
35
|
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'dry/types/constructor'
|
4
|
+
|
5
|
+
module Dry
|
6
|
+
module Types
|
7
|
+
# @api public
|
8
|
+
class Array < Nominal
|
9
|
+
# @api private
|
10
|
+
class Constructor < ::Dry::Types::Constructor
|
11
|
+
# @api private
|
12
|
+
def constructor_type
|
13
|
+
::Dry::Types::Array::Constructor
|
14
|
+
end
|
15
|
+
|
16
|
+
# @return [Lax]
|
17
|
+
#
|
18
|
+
# @api public
|
19
|
+
def lax
|
20
|
+
Lax.new(type.lax.constructor(fn, meta: meta))
|
21
|
+
end
|
22
|
+
|
23
|
+
# @see Dry::Types::Array#of
|
24
|
+
#
|
25
|
+
# @api public
|
26
|
+
def of(member)
|
27
|
+
type.of(member).constructor(fn, meta: meta)
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
@@ -1,5 +1,7 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
+
require 'dry/types/array/constructor'
|
4
|
+
|
3
5
|
module Dry
|
4
6
|
module Types
|
5
7
|
class Array < Nominal
|
@@ -111,6 +113,11 @@ module Dry
|
|
111
113
|
[:array, [member, meta ? self.meta : EMPTY_HASH]]
|
112
114
|
end
|
113
115
|
end
|
116
|
+
|
117
|
+
# @api private
|
118
|
+
def constructor_type
|
119
|
+
::Dry::Types::Array::Constructor
|
120
|
+
end
|
114
121
|
end
|
115
122
|
end
|
116
123
|
end
|
data/lib/dry/types/builder.rb
CHANGED
@@ -75,7 +75,7 @@ module Dry
|
|
75
75
|
' Be careful: types will return the same instance of the default'\
|
76
76
|
' value every time. Call `.freeze` when setting the default'\
|
77
77
|
' or pass `shared: true` to discard this warning.'\
|
78
|
-
"\n#{
|
78
|
+
"\n#{where}",
|
79
79
|
tag: :'dry-types'
|
80
80
|
)
|
81
81
|
end
|
@@ -108,21 +108,21 @@ module Dry
|
|
108
108
|
Types['nominal.hash'].map(key_type, value_type)
|
109
109
|
end
|
110
110
|
|
111
|
-
#
|
112
|
-
#
|
113
|
-
#
|
114
|
-
#
|
115
|
-
#
|
116
|
-
#
|
117
|
-
#
|
118
|
-
#
|
119
|
-
#
|
120
|
-
#
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
111
|
+
# Builds a constrained nominal type accepting any value that
|
112
|
+
# responds to given methods
|
113
|
+
#
|
114
|
+
# @example
|
115
|
+
# Types::Callable = Types.Interface(:call)
|
116
|
+
# Types::Contact = Types.Interface(:name, :address)
|
117
|
+
#
|
118
|
+
# @param methods [Array<String, Symbol>] Method names
|
119
|
+
#
|
120
|
+
# @return [Dry::Types::Contrained]
|
121
|
+
def Interface(*methods)
|
122
|
+
methods.reduce(Types['nominal.any']) do |type, method|
|
123
|
+
type.constrained(respond_to: method)
|
124
|
+
end
|
125
|
+
end
|
126
126
|
end
|
127
127
|
end
|
128
128
|
end
|
data/lib/dry/types/coercions.rb
CHANGED
@@ -36,8 +36,8 @@ module Dry
|
|
36
36
|
if input.respond_to?(:to_str)
|
37
37
|
begin
|
38
38
|
::Date.parse(input)
|
39
|
-
rescue ArgumentError, RangeError =>
|
40
|
-
CoercionError.handle(
|
39
|
+
rescue ArgumentError, RangeError => e
|
40
|
+
CoercionError.handle(e, &block)
|
41
41
|
end
|
42
42
|
elsif block_given?
|
43
43
|
yield
|
@@ -57,8 +57,8 @@ module Dry
|
|
57
57
|
if input.respond_to?(:to_str)
|
58
58
|
begin
|
59
59
|
::DateTime.parse(input)
|
60
|
-
rescue ArgumentError =>
|
61
|
-
CoercionError.handle(
|
60
|
+
rescue ArgumentError => e
|
61
|
+
CoercionError.handle(e, &block)
|
62
62
|
end
|
63
63
|
elsif block_given?
|
64
64
|
yield
|
@@ -78,8 +78,8 @@ module Dry
|
|
78
78
|
if input.respond_to?(:to_str)
|
79
79
|
begin
|
80
80
|
::Time.parse(input)
|
81
|
-
rescue ArgumentError =>
|
82
|
-
CoercionError.handle(
|
81
|
+
rescue ArgumentError => e
|
82
|
+
CoercionError.handle(e, &block)
|
83
83
|
end
|
84
84
|
elsif block_given?
|
85
85
|
yield
|
@@ -88,6 +88,19 @@ module Dry
|
|
88
88
|
end
|
89
89
|
end
|
90
90
|
|
91
|
+
# @param [#to_sym, Object] input
|
92
|
+
#
|
93
|
+
# @return [Symbol, Object]
|
94
|
+
#
|
95
|
+
# @raise CoercionError
|
96
|
+
#
|
97
|
+
# @api public
|
98
|
+
def to_symbol(input, &block)
|
99
|
+
input.to_sym
|
100
|
+
rescue NoMethodError => e
|
101
|
+
CoercionError.handle(e, &block)
|
102
|
+
end
|
103
|
+
|
91
104
|
private
|
92
105
|
|
93
106
|
# Checks whether String is empty
|