invokable 0.4.2 → 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/Gemfile.lock +1 -1
- data/README.md +49 -18
- data/lib/invokable.rb +1 -0
- data/lib/invokable/command.rb +65 -0
- data/lib/invokable/core.rb +9 -0
- data/lib/invokable/data.rb +0 -1
- data/lib/invokable/version.rb +1 -1
- metadata +6 -5
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 6dcc085ff857ef0a95578db5f816338c0ade3da6ef4ecb263f86924e4b4e2cfb
|
4
|
+
data.tar.gz: 235f058b7b39b8ff91b8d2580281fad56a388fcea46d19ae3a4bc3657c913d1c
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: bc8294c008fedd2a69337d886515b8fb9c3a2dd6fea8330b15371ffa9b44db0a5ce3c679badede4e2455230db25845ac723b11aa6736f9b62d690b462f7623e9
|
7
|
+
data.tar.gz: f6cab459416371d902933ee1ef3fd24f935bcc0ad7b6c9e0a9bd261bf95d8736c763ec416b94b063acf52fbc4a08df04586ac4b6bde8c02aedcb47b8f8074ef9
|
data/Gemfile.lock
CHANGED
data/README.md
CHANGED
@@ -8,46 +8,76 @@ Objects are functions! Treat any Object, Hashes, Arrays, and Sets as Procs (like
|
|
8
8
|
## Synopsis
|
9
9
|
|
10
10
|
```ruby
|
11
|
-
|
11
|
+
require 'invokable'
|
12
|
+
require 'invokable/hash'
|
12
13
|
|
13
|
-
|
14
|
-
|
14
|
+
number_names = { 1 => "One", 2 => "Two", 3 => "Three" }
|
15
|
+
[1, 2, 3, 4].map(&number_names) # => ["One", "Two", "Three", nil]
|
15
16
|
```
|
16
17
|
|
17
18
|
```ruby
|
18
|
-
|
19
|
+
require 'invokable'
|
20
|
+
require 'invokable/array'
|
19
21
|
|
20
|
-
|
21
|
-
|
22
|
+
alpha = ('a'..'z').to_a
|
23
|
+
[1, 2, 3, 4].map(&alpha) # => ["b", "c", "d", "e"]
|
22
24
|
```
|
23
25
|
|
24
26
|
```ruby
|
25
|
-
|
27
|
+
require 'invokable'
|
28
|
+
require 'invokable/set'
|
26
29
|
|
27
|
-
|
28
|
-
|
30
|
+
favorite_numbers = Set[3, Math::PI]
|
31
|
+
[1, 2, 3, 4].select(&favorite_numbers) # => [3]
|
29
32
|
```
|
30
33
|
|
31
34
|
```ruby
|
32
|
-
|
35
|
+
# service objects
|
36
|
+
require 'invokable'
|
33
37
|
|
34
|
-
|
35
|
-
|
36
|
-
include Invokable
|
38
|
+
class GetDataFromSomeService
|
39
|
+
include Invokable
|
37
40
|
|
38
|
-
|
39
|
-
|
40
|
-
end
|
41
|
+
def call(user)
|
42
|
+
# do the dirt
|
41
43
|
end
|
44
|
+
end
|
42
45
|
|
43
|
-
|
44
|
-
|
46
|
+
data_for_user = GetDataFromSomeService.new.memoize # 'memoize' makes a proc that caches results
|
47
|
+
User.all.map(&data_for_user)
|
48
|
+
```
|
49
|
+
```ruby
|
50
|
+
# command objects that enclose state, can be treated as automatically curried functions.
|
51
|
+
require 'invokable'
|
52
|
+
require 'invokable/command'
|
53
|
+
|
54
|
+
class TwitterPoster
|
55
|
+
include Invokable::Command
|
56
|
+
|
57
|
+
enclose do |model|
|
58
|
+
@model = model
|
59
|
+
end
|
60
|
+
|
61
|
+
def call(user)
|
62
|
+
# do the dirt
|
63
|
+
...
|
64
|
+
TwitterStatus.new(user, data)
|
65
|
+
end
|
66
|
+
end
|
67
|
+
|
68
|
+
TwitterPoster.call(Model.find(1)) # => #<TwitterPoster ...>
|
69
|
+
TwitterPoster.call(Model.find(1), current_user) # => #<TwitterStatus ...>
|
70
|
+
|
71
|
+
# both the class and it's instances can be used any where Procs are.
|
72
|
+
|
73
|
+
Model.where(created_at: Date.today).map(&:TwitterPoster) # => [#<TwitterPoster ...>, ...]
|
45
74
|
```
|
46
75
|
|
47
76
|
Use as much or a little as you need:
|
48
77
|
|
49
78
|
```ruby
|
50
79
|
require 'invokable' # loads Invokable module
|
80
|
+
require 'invokable/command' # loads Invokable::Command module
|
51
81
|
require 'invokable/hash' # loads hash patch
|
52
82
|
require 'invokable/array' # loads array patch
|
53
83
|
require 'invokable/set' # loads set patch
|
@@ -111,6 +141,7 @@ Returns a proc that is a composition of this invokable and the given invokable.
|
|
111
141
|
|
112
142
|
## See Also
|
113
143
|
|
144
|
+
- [Closures and Objects are Equivalent](http://wiki.c2.com/?ClosuresAndObjectsAreEquivalent)
|
114
145
|
- [Clojure](https://clojure.org)
|
115
146
|
- [Arc](http://www.arclanguage.org)
|
116
147
|
|
data/lib/invokable.rb
CHANGED
@@ -0,0 +1,65 @@
|
|
1
|
+
module Invokable
|
2
|
+
# Treat "Command Objects" as curried functions
|
3
|
+
#
|
4
|
+
# @see https://ruby-doc.org/core-2.7.0/Proc.html#method-i-curry Proc#curry
|
5
|
+
#
|
6
|
+
# @version 0.5.0
|
7
|
+
module Command
|
8
|
+
def self.included(klass)
|
9
|
+
klass.include(Invokable)
|
10
|
+
klass.extend(Invokable::Core)
|
11
|
+
klass.extend(Invokable::Compose)
|
12
|
+
klass.extend(ClassMethods)
|
13
|
+
end
|
14
|
+
|
15
|
+
module ClassMethods
|
16
|
+
# Return the "total" arity of the class (i.e. the arity of the initializer and the arity of the call method)
|
17
|
+
#
|
18
|
+
# @version 0.5.0
|
19
|
+
# @see https://ruby-doc.org/core-2.7.1/Proc.html#method-i-arity Proc#arity
|
20
|
+
# @see initializer_arity
|
21
|
+
#
|
22
|
+
# @return [Integer]
|
23
|
+
def arity
|
24
|
+
initializer_arity + instance_method(:call).arity
|
25
|
+
end
|
26
|
+
|
27
|
+
# Return the arity of the initializer
|
28
|
+
#
|
29
|
+
# @version 0.5.0
|
30
|
+
# @see arity
|
31
|
+
#
|
32
|
+
# @return [Integer]
|
33
|
+
def initializer_arity
|
34
|
+
@initializer ? @initializer.arity : 0
|
35
|
+
end
|
36
|
+
|
37
|
+
# To specify any enclosed state
|
38
|
+
def enclose(&block)
|
39
|
+
raise 'A block is required' if block.nil?
|
40
|
+
|
41
|
+
@initializer = block
|
42
|
+
define_method :initialize, &block
|
43
|
+
end
|
44
|
+
|
45
|
+
# Handle automatic currying--will accept either the initializer arity or the total arity of the class. If
|
46
|
+
# the initializer arity is used return a class instance. If the total arity is used instantiate the class
|
47
|
+
# and return the results of the `call` method.
|
48
|
+
#
|
49
|
+
# @version 0.5.0
|
50
|
+
# @see arity
|
51
|
+
# @see initializer_arity
|
52
|
+
def call(*args)
|
53
|
+
if args.length == initializer_arity
|
54
|
+
new(*args)
|
55
|
+
elsif args.length == arity
|
56
|
+
init_args = args.slice(0, initializer_arity)
|
57
|
+
call_args = args.slice(initializer_arity, args.length)
|
58
|
+
new(*init_args).call(*call_args)
|
59
|
+
else
|
60
|
+
raise "Wrong number of arguments expected #{initializer_arity} or #{arity}, got: #{args.length}"
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
data/lib/invokable/core.rb
CHANGED
@@ -36,5 +36,14 @@ module Invokable
|
|
36
36
|
@memo[args.hash] ||= call(*args)
|
37
37
|
end
|
38
38
|
end
|
39
|
+
|
40
|
+
# Return the arity (i.e. the number of arguments) of the `call` method.
|
41
|
+
#
|
42
|
+
# @version 0.5.0
|
43
|
+
# @see https://ruby-doc.org/core-2.7.1/Proc.html#method-i-arity Proc#arity
|
44
|
+
# @return [Integer]
|
45
|
+
def arity
|
46
|
+
method(:call).arity
|
47
|
+
end
|
39
48
|
end
|
40
49
|
end
|
data/lib/invokable/data.rb
CHANGED
data/lib/invokable/version.rb
CHANGED
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: invokable
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.5.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Delon Newman
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2020-
|
11
|
+
date: 2020-07-21 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: bundler
|
@@ -72,6 +72,7 @@ files:
|
|
72
72
|
- invokable.gemspec
|
73
73
|
- lib/invokable.rb
|
74
74
|
- lib/invokable/array.rb
|
75
|
+
- lib/invokable/command.rb
|
75
76
|
- lib/invokable/compose.rb
|
76
77
|
- lib/invokable/core.rb
|
77
78
|
- lib/invokable/data.rb
|
@@ -89,7 +90,7 @@ metadata:
|
|
89
90
|
source_code_uri: https://github.com/delonnewman/invokable
|
90
91
|
changelog_uri: https://github.com/delonnewman/invokable#changelog
|
91
92
|
documentation_uri: https://www.rubydoc.info/gems/invokable
|
92
|
-
post_install_message:
|
93
|
+
post_install_message:
|
93
94
|
rdoc_options: []
|
94
95
|
require_paths:
|
95
96
|
- lib
|
@@ -105,7 +106,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
105
106
|
version: '0'
|
106
107
|
requirements: []
|
107
108
|
rubygems_version: 3.0.6
|
108
|
-
signing_key:
|
109
|
+
signing_key:
|
109
110
|
specification_version: 4
|
110
111
|
summary: Objects are functions! Treat any Object, Hashes, Arrays and Sets as Procs
|
111
112
|
(like Enumerable but for Proc-like objects)
|