cache_box 0.0.1.pre.preview7 → 0.0.1.pre.preview8

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.
@@ -0,0 +1,139 @@
1
+ # frozen_string_literal: true
2
+
3
+ class CacheBox
4
+ module Helper
5
+ # Common validation methods.
6
+ module Validate
7
+ # Input:
8
+ #
9
+ # arg = String | Symbol
10
+ # name = String
11
+ #
12
+ # Output: arg
13
+ def validate_string_or_symbol!(arg, name)
14
+ return arg if arg.is_a?(Symbol) || arg.is_a?(String)
15
+
16
+ klass = arg.class
17
+ value = arg.inspect
18
+
19
+ raise(ArgumentError, "#{name} must be a Symbol or String, got #{klass}: #{value}")
20
+ end
21
+
22
+ # Input:
23
+ #
24
+ # args = Array[String | Symbol]
25
+ # name = String
26
+ #
27
+ # Output: args
28
+ def validate_array_of_string_or_symbol!(args, name)
29
+ unless args.is_a?(Array)
30
+ klass = args.class
31
+ value = args.inspect
32
+
33
+ raise(
34
+ ArgumentError,
35
+ "#{name} must be an Array, got #{klass}: #{value}"
36
+ )
37
+ end
38
+
39
+ args.each do |arg|
40
+ next if arg.is_a?(Symbol) || arg.is_a?(String)
41
+
42
+ klass = arg.class
43
+ value = arg.inspect
44
+
45
+ raise(
46
+ ArgumentError,
47
+ "#{name} must contain Symbol or String, got #{klass}: #{value} in #{args}"
48
+ )
49
+ end
50
+
51
+ args
52
+ end
53
+
54
+ # Input:
55
+ #
56
+ # block = &block
57
+ # name = String
58
+ #
59
+ # Output: N/A
60
+ def validate_block_presence!(block, name)
61
+ return block if block
62
+
63
+ raise(ArgumentError, "The `#{name}` method requires a block")
64
+ end
65
+
66
+ # Input:
67
+ #
68
+ # edge = String |
69
+ # Symbol |
70
+ # Hash{1 String | Symbol => String | Symbol | Array[...String | Symbol]}
71
+ #
72
+ # Output: edge
73
+ def validate_edge!(edge)
74
+ return edge if edge.is_a?(Symbol) || edge.is_a?(String)
75
+
76
+ if edge.is_a?(Hash)
77
+ validate_hash_edge!(edge)
78
+
79
+ edge
80
+ else
81
+ klass = edge.class
82
+ value = edge.inspect
83
+
84
+ raise(
85
+ ArgumentError,
86
+ "edge has an unknown structure; #{klass}: #{value}"
87
+ )
88
+ end
89
+ end
90
+
91
+ private
92
+
93
+ # Input:
94
+ #
95
+ # edge = Hash{1 String | Symbol => String | Symbol | Array[...String | Symbol]}
96
+ #
97
+ # Output: edge
98
+ def validate_hash_edge!(edge)
99
+ if (size = edge.size) != 1
100
+ raise(
101
+ ArgumentError,
102
+ "edge must have one mapping, got #{size}: #{edge.inspect}"
103
+ )
104
+ end
105
+
106
+ name = edge.keys.first
107
+ needs = edge[name]
108
+
109
+ validate_string_or_symbol!(name, 'edge name')
110
+ validate_needs!(needs)
111
+
112
+ edge
113
+ end
114
+
115
+ # Input:
116
+ #
117
+ # arg = String | Symbol | Array[...String | Symbol]
118
+ #
119
+ # Output: arg
120
+ def validate_needs!(arg)
121
+ return arg if arg.is_a?(Symbol) || arg.is_a?(String)
122
+
123
+ if arg.is_a?(Array)
124
+ validate_array_of_string_or_symbol!(arg, 'edge needs')
125
+
126
+ arg
127
+ else
128
+ klass = arg.class
129
+ value = arg.inspect
130
+
131
+ raise(
132
+ ArgumentError,
133
+ "edge needs has an unknown structure; #{klass}: #{value}"
134
+ )
135
+ end
136
+ end
137
+ end
138
+ end
139
+ end
@@ -0,0 +1,159 @@
1
+ # frozen_string_literal: true
2
+
3
+ class CacheBox
4
+ module Scheduler
5
+ class Serial
6
+ include CacheBox::Helper::Validate
7
+
8
+ # Input:
9
+ #
10
+ # logger = Logger
11
+ #
12
+ # Output: N/A
13
+ def initialize(logger: nil)
14
+ @logger = logger || Logger.new(STDOUT, level: Logger::INFO)
15
+
16
+ @edges = {}
17
+ @names = {}
18
+ @leaves = {}
19
+ @plan = []
20
+ end
21
+
22
+ # Input:
23
+ #
24
+ # edge = String |
25
+ # Symbol |
26
+ # Hash{String | Symbol => String | Symbol | Array[String | Symbol]}
27
+ #
28
+ # Output: self
29
+ def add(edge)
30
+ validate_edge!(edge)
31
+
32
+ needs_o = extract_needs(edge)
33
+ needs_s = needs_o.map(&:to_s)
34
+ needs_s.each_with_index do |arg, index|
35
+ next if @edges.key?(arg)
36
+
37
+ raise(
38
+ ArgumentError,
39
+ "Graph does not contain required node #{needs_o[index].inspect}"
40
+ )
41
+ end
42
+
43
+ name_o = extract_name(edge)
44
+ name_s = name_o.to_s
45
+ if @edges.key?(name_s)
46
+ raise(
47
+ ArgumentError,
48
+ "Graph already contains a node named #{name_o.inspect}"
49
+ )
50
+ else
51
+ @edges[name_s] = needs_s
52
+ end
53
+ @names[name_s] = name_o
54
+
55
+ add_to_plan(name_s, needs_s)
56
+
57
+ @leaves[name_s] = name_o
58
+ needs_s.each do |need_s|
59
+ @leaves.delete(need_s)
60
+ end
61
+
62
+ self
63
+ end
64
+
65
+ # Input:
66
+ #
67
+ # nodes = Hash{...String => Array[String | Symbol, Proc(Object)]}
68
+ # unit = CacheBox::Unit
69
+ #
70
+ # Output: Hash{...String | Symbol => Object}
71
+ def run!(nodes, unit)
72
+ sequence = @plan.flatten
73
+ sequence.each_with_index do |name, index|
74
+ next if unit.has?(name)
75
+
76
+ callable = nodes[name][1]
77
+ needs = @edges[name]
78
+ input = {}
79
+
80
+ needs.each do |need|
81
+ next unless unit.has?(need)
82
+
83
+ input[@names[need]] = unit.result(need)
84
+ end
85
+ callable.call(input)
86
+
87
+ needs.each do |need|
88
+ needed = false
89
+ sequence[(index + 1)..-1].each do |item|
90
+ needed = true if @edges[item].include?(need)
91
+ end
92
+
93
+ unit.clear(need) unless needed
94
+ end
95
+ end
96
+
97
+ result = {}
98
+ @leaves.each do |name_s, name_o|
99
+ next unless unit.has?(name_s)
100
+
101
+ result[name_o] = unit.result(name_s)
102
+ end
103
+ result
104
+ end
105
+
106
+ private
107
+
108
+ # Input: ?
109
+ #
110
+ # Output: ?
111
+ def add_to_plan(name, from)
112
+ result = 0
113
+
114
+ from.each do |node|
115
+ @plan.each_with_index do |row, index|
116
+ next unless row.include?(node)
117
+
118
+ result = index >= result ? index + 1 : result
119
+ break
120
+ end
121
+ end
122
+
123
+ @plan.each do |row|
124
+ next unless row.include?(from)
125
+
126
+ row.push(name)
127
+ break
128
+ end
129
+
130
+ if (row = @plan[result])
131
+ row.push(name)
132
+ else
133
+ @plan.push([name])
134
+ end
135
+ end
136
+
137
+ # Input: String | Symbol | Hash[1 String | Symbol => Object]
138
+ #
139
+ # Output: String | Symbol
140
+ def extract_name(edge)
141
+ return edge if edge.is_a?(Symbol) || edge.is_a?(String)
142
+
143
+ edge.keys.first
144
+ end
145
+
146
+ # Input: ?
147
+ #
148
+ # Output: Array[...String | Symbol]
149
+ def extract_needs(edge)
150
+ return [] unless edge.is_a?(Hash)
151
+
152
+ name = edge.keys.first
153
+ needs = edge[name]
154
+
155
+ needs.is_a?(Array) ? needs : [needs]
156
+ end
157
+ end
158
+ end
159
+ end
@@ -0,0 +1,118 @@
1
+ # frozen_string_literal: true
2
+
3
+ class CacheBox
4
+ module Storage
5
+ # A storage backed by files.
6
+ class File
7
+ # Input:
8
+ #
9
+ # namespace = Symbol | String # Default: 'namespace'
10
+ # path = String # Path; Default: nil
11
+ #
12
+ # Output: N/A
13
+ def initialize(namespace: nil, path: nil)
14
+ validate_symbol_or_string!(namespace, 'namespace')
15
+ validate_string!(path, 'path')
16
+
17
+ namespace = namespace.to_s || 'namespace'
18
+
19
+ root = path || ::File.join(Dir.pwd, '.cache')
20
+ @path = ::File.join(root, namespace)
21
+ end
22
+
23
+ # Output: self
24
+ def reset!
25
+ FileUtils.remove_entry_secure(@path, true)
26
+
27
+ self
28
+ end
29
+
30
+ # Reads the content.
31
+ #
32
+ # Input:
33
+ #
34
+ # name = String
35
+ #
36
+ # Output: Object # Anything
37
+ def read!(name)
38
+ validate_string!(name, 'name')
39
+
40
+ file = ::File.join(@path, name)
41
+ return unless ::File.exist?(file)
42
+
43
+ Marshal.load(::File.read(file))
44
+ end
45
+
46
+ # Input:
47
+ #
48
+ # name = String
49
+ # data = Object # Anything
50
+ #
51
+ # Output: self
52
+ def write!(name, data)
53
+ validate_string!(name, 'name')
54
+
55
+ FileUtils.mkdir_p(@path) unless Dir.exist?(@path)
56
+
57
+ content = Marshal.dump(data)
58
+ file = ::File.join(@path, name)
59
+
60
+ ::File.write(file, content)
61
+
62
+ self
63
+ end
64
+
65
+ # Input:
66
+ #
67
+ # name = String
68
+ #
69
+ # Output: self
70
+ def delete!(name)
71
+ validate_string!(name, 'name')
72
+
73
+ file = ::File.join(@path, name)
74
+ FileUtils.remove_entry_secure(file, true)
75
+
76
+ self
77
+ end
78
+
79
+ # Input:
80
+ #
81
+ # name = String
82
+ #
83
+ # Output: true | false
84
+ def has?(name)
85
+ validate_string!(name, 'name')
86
+
87
+ file = ::File.join(@path, name)
88
+ ::File.file?(file)
89
+ end
90
+
91
+ private
92
+
93
+ def validate_symbol_or_string!(arg, name)
94
+ return if arg.nil? || arg.is_a?(Symbol) || arg.is_a?(String)
95
+
96
+ klass = arg.class
97
+ value = arg.inspect
98
+
99
+ raise(
100
+ ArgumentError,
101
+ "#{name} must be a Symbol or String, got #{klass}: #{value}"
102
+ )
103
+ end
104
+
105
+ def validate_string!(arg, name)
106
+ return if arg.nil? || arg.is_a?(String)
107
+
108
+ klass = arg.class
109
+ value = arg.inspect
110
+
111
+ raise(
112
+ ArgumentError,
113
+ "#{name} must be a String, got #{klass}: #{value}"
114
+ )
115
+ end
116
+ end
117
+ end
118
+ end
@@ -0,0 +1,195 @@
1
+ # frozen_string_literal: true
2
+
3
+ class CacheBox
4
+ module Storage
5
+ # A storage backed by an in memory Hash.
6
+ class Memory
7
+ # Accepts a Hash or any Hash-like object as argument. Will use a plain Hash
8
+ # if none provided.
9
+ #
10
+ # Input:
11
+ #
12
+ # state = Hash{...Object => Object} # Default: nil
13
+ #
14
+ # Output: N/A
15
+ def initialize(state = nil)
16
+ validate!(state)
17
+
18
+ @state = state || {}
19
+ end
20
+
21
+ # Accepts a Hash or any Hash-like object as argument. Will use a plain Hash
22
+ # if none provided.
23
+ #
24
+ # Input:
25
+ #
26
+ # state = Hash{...Object => Object}
27
+ #
28
+ # Output: self
29
+ def reset!(state = nil)
30
+ initialize(state)
31
+
32
+ self
33
+ end
34
+
35
+ # Input:
36
+ #
37
+ # name = String
38
+ #
39
+ # Output: Object # Anything
40
+ def read!(name)
41
+ validate_string!(name, 'name')
42
+
43
+ @state[name]
44
+ end
45
+
46
+ # Input:
47
+ #
48
+ # name = String
49
+ # data = Object # Anything
50
+ #
51
+ # Output: self
52
+ def write!(name, data)
53
+ validate_string!(name, 'name')
54
+
55
+ @state[name] = data
56
+
57
+ self
58
+ end
59
+
60
+ # Input:
61
+ #
62
+ # name = String
63
+ #
64
+ # Output: self
65
+ def delete!(name)
66
+ validate_string!(name, 'name')
67
+
68
+ @state.delete(name)
69
+
70
+ self
71
+ end
72
+
73
+ # Input:
74
+ #
75
+ # name = String
76
+ #
77
+ # Output: true | false
78
+ def has?(name)
79
+ validate_string!(name, 'name')
80
+
81
+ @state.key?(name)
82
+ end
83
+
84
+ private
85
+
86
+ # Input:
87
+ #
88
+ # state = Object # Anything
89
+ #
90
+ # Output: N/A
91
+ def validate!(state)
92
+ return if state.nil? || state.is_a?(Hash)
93
+
94
+ validate_get!(state)
95
+ validate_set!(state)
96
+ validate_delete!(state)
97
+ validate_key!(state)
98
+ end
99
+
100
+ # Input:
101
+ #
102
+ # state = Object # Anything
103
+ #
104
+ # Output: N/A
105
+ def validate_get!(state)
106
+ unless state.respond_to?(:[])
107
+ raise(ArgumentError, 'Given state object does not respond to `:[]`')
108
+ end
109
+
110
+ arity = state.method(:[]).arity
111
+ unless arity == 1
112
+ raise(
113
+ ArgumentError,
114
+ "Given state object's `:[]` method arity must be 1, got: #{arity}"
115
+ )
116
+ end
117
+ end
118
+
119
+ # Input:
120
+ #
121
+ # state = Object # Anything
122
+ #
123
+ # Output: N/A
124
+ def validate_set!(state)
125
+ unless state.respond_to?(:[]=)
126
+ raise(ArgumentError, 'Given state object does not respond to `:[]=`')
127
+ end
128
+
129
+ arity = state.method(:[]=).arity
130
+ unless arity == 2
131
+ raise(
132
+ ArgumentError,
133
+ "Given state object's `:[]=` method arity must be 2, got: #{arity}"
134
+ )
135
+ end
136
+ end
137
+
138
+ # Input:
139
+ #
140
+ # state = Object # Anything
141
+ #
142
+ # Output: N/A
143
+ def validate_delete!(state)
144
+ unless state.respond_to?(:delete)
145
+ raise(ArgumentError, 'Given state object does not respond to `delete`')
146
+ end
147
+
148
+ arity = state.method(:delete).arity
149
+ unless arity == 1
150
+ raise(
151
+ ArgumentError,
152
+ "Given state object's `:delete` method arity must be 1, got: #{arity}"
153
+ )
154
+ end
155
+ end
156
+
157
+ # Input:
158
+ #
159
+ # state = Object # Anything
160
+ #
161
+ # Output: N/A
162
+ def validate_key!(state)
163
+ unless state.respond_to?(:key?)
164
+ raise(ArgumentError, 'Given state object does not respond to `:key?`')
165
+ end
166
+
167
+ arity = state.method(:key?).arity
168
+ unless arity == 1
169
+ raise(
170
+ ArgumentError,
171
+ "Given state object's `:key?` method arity must be 1, got: #{arity}"
172
+ )
173
+ end
174
+ end
175
+
176
+ # Input:
177
+ #
178
+ # arg = String
179
+ # name = String
180
+ #
181
+ # Output: N/A
182
+ def validate_string!(arg, name)
183
+ return if arg.nil? || arg.is_a?(String)
184
+
185
+ klass = arg.class
186
+ value = arg.inspect
187
+
188
+ raise(
189
+ ArgumentError,
190
+ "#{name} must be a String, got #{klass}: #{value}"
191
+ )
192
+ end
193
+ end
194
+ end
195
+ end