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.
- checksums.yaml +4 -4
- data/cache_box.gemspec +8 -3
- data/lib/cache_box.rb +32 -104
- data/lib/cache_box/box.rb +17 -0
- data/lib/cache_box/chain.rb +34 -48
- data/lib/cache_box/graph.rb +87 -0
- data/lib/cache_box/helper/validate.rb +139 -0
- data/lib/cache_box/scheduler/serial.rb +159 -0
- data/lib/cache_box/storage/file.rb +118 -0
- data/lib/cache_box/storage/memory.rb +195 -0
- data/lib/cache_box/unit.rb +168 -0
- metadata +14 -9
- data/lib/cache_box/file_storage.rb +0 -116
- data/lib/cache_box/memory_storage.rb +0 -193
@@ -0,0 +1,168 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class CacheBox
|
4
|
+
class Unit
|
5
|
+
include CacheBox::Helper::Validate
|
6
|
+
|
7
|
+
# Input:
|
8
|
+
#
|
9
|
+
# namespace = String | Symbol # Default: :namespace
|
10
|
+
#
|
11
|
+
# Output: N/A
|
12
|
+
def initialize(namespace = nil, logger: nil, storage: nil)
|
13
|
+
namespace ||= :namespace
|
14
|
+
validate_string_or_symbol!(namespace, 'namespace')
|
15
|
+
|
16
|
+
@namespace = namespace
|
17
|
+
@logger = logger || Logger.new(STDOUT, level: Logger::INFO)
|
18
|
+
@storage = storage || ::CacheBox::Storage::File.new(namespace: @namespace)
|
19
|
+
|
20
|
+
@state = { result: {}, stash: {} }
|
21
|
+
end
|
22
|
+
|
23
|
+
# Input:
|
24
|
+
#
|
25
|
+
# name = String | Symbol # Default: 'name'
|
26
|
+
# args = Object
|
27
|
+
# input = Object
|
28
|
+
# &block = Proc(CacheBox::Box)
|
29
|
+
#
|
30
|
+
# Output: Object # Anything the &block returns.
|
31
|
+
def with(name = nil, args = nil, input = nil, &block)
|
32
|
+
name ||= 'name'
|
33
|
+
validate_string_or_symbol!(name, 'name')
|
34
|
+
validate_block_presence!(block, '#with')
|
35
|
+
|
36
|
+
name_o = name
|
37
|
+
name_s = name.to_s
|
38
|
+
|
39
|
+
return @state[:result][name_s] if @state[:result].key?(name_s)
|
40
|
+
|
41
|
+
data = @storage.read!(name_s)
|
42
|
+
if data&.key?(:result)
|
43
|
+
@state[:result][name_s] = data[:result]
|
44
|
+
|
45
|
+
return data[:result]
|
46
|
+
elsif data&.key?(:stash)
|
47
|
+
@state[:stash][name_s] = data[:stash]
|
48
|
+
end
|
49
|
+
|
50
|
+
box = ::CacheBox::Box.new(
|
51
|
+
namespace: @namespace.dup,
|
52
|
+
name: name_o.dup,
|
53
|
+
args: args,
|
54
|
+
input: input,
|
55
|
+
stash: @state[:stash][name] || {},
|
56
|
+
logger: @logger
|
57
|
+
)
|
58
|
+
|
59
|
+
begin
|
60
|
+
@state[:result][name_s] = block.call(box)
|
61
|
+
ensure
|
62
|
+
if @state[:result].key?(name_s)
|
63
|
+
@state[:stash].delete(name_s)
|
64
|
+
@storage.write!(name_s, { result: @state[:result][name_s] })
|
65
|
+
else
|
66
|
+
stash = box.stash
|
67
|
+
dump = { stash: stash }
|
68
|
+
|
69
|
+
@state[:stash][name_s] = stash
|
70
|
+
@storage.write!(name_s, dump)
|
71
|
+
end
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
# Input:
|
76
|
+
#
|
77
|
+
# name = String | Symbol # Default: 'name'
|
78
|
+
#
|
79
|
+
# Output: true | false
|
80
|
+
def has?(name = nil)
|
81
|
+
name ||= 'name'
|
82
|
+
validate_string_or_symbol!(name, 'name')
|
83
|
+
name = name.to_s
|
84
|
+
|
85
|
+
return true if @state[:result].key?(name)
|
86
|
+
|
87
|
+
data = @storage.read!(name)
|
88
|
+
|
89
|
+
if data&.key?(:result)
|
90
|
+
@state[:result][name] = data[:result]
|
91
|
+
elsif data&.key?(:stash)
|
92
|
+
@state[:stash][name] = data[:stash]
|
93
|
+
end
|
94
|
+
|
95
|
+
@state[:result].key?(name)
|
96
|
+
end
|
97
|
+
|
98
|
+
# Input:
|
99
|
+
#
|
100
|
+
# name = String | Symbol # Default: 'name'
|
101
|
+
#
|
102
|
+
# Output: true | false
|
103
|
+
def result(name = nil)
|
104
|
+
name ||= 'name'
|
105
|
+
validate_string_or_symbol!(name, 'name')
|
106
|
+
name = name.to_s
|
107
|
+
|
108
|
+
return @state[:result][name] if @state[:result].key?(name)
|
109
|
+
|
110
|
+
data = @storage.read!(name)
|
111
|
+
|
112
|
+
if data&.key?(:result)
|
113
|
+
@state[:result][name] = data[:result]
|
114
|
+
elsif data&.key?(:stash)
|
115
|
+
@state[:stash][name] = data[:stash]
|
116
|
+
end
|
117
|
+
|
118
|
+
@state[:result][name]
|
119
|
+
end
|
120
|
+
|
121
|
+
# Input:
|
122
|
+
#
|
123
|
+
# name = Array[String | Symbol] # Default: []
|
124
|
+
#
|
125
|
+
# Output: self
|
126
|
+
def expire!(*names)
|
127
|
+
validate_array_of_string_or_symbol!(names, 'names')
|
128
|
+
|
129
|
+
if names.empty?
|
130
|
+
@state = { result: {}, stash: {} }
|
131
|
+
|
132
|
+
@storage.reset!
|
133
|
+
else
|
134
|
+
names.each do |name|
|
135
|
+
name = name.to_s
|
136
|
+
@state[:result].delete(name)
|
137
|
+
@state[:stash].delete(name)
|
138
|
+
|
139
|
+
@storage.delete!(name)
|
140
|
+
end
|
141
|
+
end
|
142
|
+
|
143
|
+
self
|
144
|
+
end
|
145
|
+
|
146
|
+
# Input:
|
147
|
+
#
|
148
|
+
# name = Array[String | Symbol] # Default: []
|
149
|
+
#
|
150
|
+
# Output: self
|
151
|
+
def clear(*names)
|
152
|
+
validate_array_of_string_or_symbol!(names, 'names')
|
153
|
+
|
154
|
+
if names.empty?
|
155
|
+
@state = { result: {}, stash: {} }
|
156
|
+
else
|
157
|
+
names.each do |name_o|
|
158
|
+
name_s = name_o.to_s
|
159
|
+
|
160
|
+
@state[:result].delete(name_s)
|
161
|
+
@state[:stash].delete(name_s)
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
self
|
166
|
+
end
|
167
|
+
end
|
168
|
+
end
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: cache_box
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0.1.pre.
|
4
|
+
version: 0.0.1.pre.preview8
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Codruț Constantin Gușoi
|
8
|
-
autorequire:
|
8
|
+
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2020-07-
|
11
|
+
date: 2020-07-26 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: minitest
|
@@ -66,7 +66,7 @@ dependencies:
|
|
66
66
|
- - "~>"
|
67
67
|
- !ruby/object:Gem::Version
|
68
68
|
version: '0.88'
|
69
|
-
description:
|
69
|
+
description:
|
70
70
|
email:
|
71
71
|
- codrut.gusoi+rubygems.org@gmail.com
|
72
72
|
executables: []
|
@@ -76,16 +76,21 @@ files:
|
|
76
76
|
- LICENSE
|
77
77
|
- cache_box.gemspec
|
78
78
|
- lib/cache_box.rb
|
79
|
+
- lib/cache_box/box.rb
|
79
80
|
- lib/cache_box/chain.rb
|
80
|
-
- lib/cache_box/
|
81
|
-
- lib/cache_box/
|
81
|
+
- lib/cache_box/graph.rb
|
82
|
+
- lib/cache_box/helper/validate.rb
|
83
|
+
- lib/cache_box/scheduler/serial.rb
|
84
|
+
- lib/cache_box/storage/file.rb
|
85
|
+
- lib/cache_box/storage/memory.rb
|
86
|
+
- lib/cache_box/unit.rb
|
82
87
|
homepage: https://gitlab.com/sdwolfz/cache_box_rb
|
83
88
|
licenses:
|
84
89
|
- BSD-3-Clause
|
85
90
|
metadata:
|
86
91
|
homepage_uri: https://gitlab.com/sdwolfz/cache_box_rb
|
87
92
|
source_code_uri: https://gitlab.com/sdwolfz/cache_box_rb
|
88
|
-
post_install_message:
|
93
|
+
post_install_message:
|
89
94
|
rdoc_options: []
|
90
95
|
require_paths:
|
91
96
|
- lib
|
@@ -100,8 +105,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
100
105
|
- !ruby/object:Gem::Version
|
101
106
|
version: 1.3.1
|
102
107
|
requirements: []
|
103
|
-
rubygems_version: 3.1.
|
104
|
-
signing_key:
|
108
|
+
rubygems_version: 3.1.3
|
109
|
+
signing_key:
|
105
110
|
specification_version: 4
|
106
111
|
summary: A simple, fast, and easy to use local cache.
|
107
112
|
test_files: []
|
@@ -1,116 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
class CacheBox
|
4
|
-
# A storage backed by files.
|
5
|
-
class FileStorage
|
6
|
-
# Input:
|
7
|
-
#
|
8
|
-
# namespace = Symbol | String # Default: 'namespace'
|
9
|
-
# path = String # Path; Default: nil
|
10
|
-
#
|
11
|
-
# Output: N/A
|
12
|
-
def initialize(namespace: nil, path: nil)
|
13
|
-
validate_symbol_or_string!(namespace, 'namespace')
|
14
|
-
validate_string!(path, 'path')
|
15
|
-
|
16
|
-
@namespace = namespace.to_s || 'namespace'
|
17
|
-
|
18
|
-
root = path || File.join(Dir.pwd, '.cache')
|
19
|
-
@path = File.join(root, @namespace)
|
20
|
-
end
|
21
|
-
|
22
|
-
# Output: self
|
23
|
-
def reset!
|
24
|
-
FileUtils.remove_entry_secure(@path, true)
|
25
|
-
|
26
|
-
self
|
27
|
-
end
|
28
|
-
|
29
|
-
# Reads the content.
|
30
|
-
#
|
31
|
-
# Input:
|
32
|
-
#
|
33
|
-
# name = String
|
34
|
-
#
|
35
|
-
# Output: Object # Anything
|
36
|
-
def read!(name)
|
37
|
-
validate_string!(name, 'name')
|
38
|
-
|
39
|
-
file = File.join(@path, name)
|
40
|
-
return unless File.exist?(file)
|
41
|
-
|
42
|
-
Marshal.load(File.read(file))
|
43
|
-
end
|
44
|
-
|
45
|
-
# Input:
|
46
|
-
#
|
47
|
-
# name = String
|
48
|
-
# data = Object # Anything
|
49
|
-
#
|
50
|
-
# Output: self
|
51
|
-
def write!(name, data)
|
52
|
-
validate_string!(name, 'name')
|
53
|
-
|
54
|
-
FileUtils.mkdir_p(@path) unless Dir.exist?(@path)
|
55
|
-
|
56
|
-
content = Marshal.dump(data)
|
57
|
-
file = File.join(@path, name)
|
58
|
-
|
59
|
-
File.write(file, content)
|
60
|
-
|
61
|
-
self
|
62
|
-
end
|
63
|
-
|
64
|
-
# Input:
|
65
|
-
#
|
66
|
-
# name = String
|
67
|
-
#
|
68
|
-
# Output: self
|
69
|
-
def delete!(name)
|
70
|
-
validate_string!(name, 'name')
|
71
|
-
|
72
|
-
file = File.join(@path, name)
|
73
|
-
FileUtils.remove_entry_secure(file, true)
|
74
|
-
|
75
|
-
self
|
76
|
-
end
|
77
|
-
|
78
|
-
# Input:
|
79
|
-
#
|
80
|
-
# name = String
|
81
|
-
#
|
82
|
-
# Output: true | false
|
83
|
-
def has?(name)
|
84
|
-
validate_string!(name, 'name')
|
85
|
-
|
86
|
-
file = File.join(@path, name)
|
87
|
-
File.file?(file)
|
88
|
-
end
|
89
|
-
|
90
|
-
private
|
91
|
-
|
92
|
-
def validate_symbol_or_string!(arg, name)
|
93
|
-
return if arg.nil? || arg.is_a?(Symbol) || arg.is_a?(String)
|
94
|
-
|
95
|
-
klass = arg.class
|
96
|
-
value = arg.inspect
|
97
|
-
|
98
|
-
raise(
|
99
|
-
ArgumentError,
|
100
|
-
"#{name} must be a Symbol or String, got #{klass}: #{value}"
|
101
|
-
)
|
102
|
-
end
|
103
|
-
|
104
|
-
def validate_string!(arg, name)
|
105
|
-
return if arg.nil? || arg.is_a?(String)
|
106
|
-
|
107
|
-
klass = arg.class
|
108
|
-
value = arg.inspect
|
109
|
-
|
110
|
-
raise(
|
111
|
-
ArgumentError,
|
112
|
-
"#{name} must be a String, got #{klass}: #{value}"
|
113
|
-
)
|
114
|
-
end
|
115
|
-
end
|
116
|
-
end
|
@@ -1,193 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
class CacheBox
|
4
|
-
# A storage backed by an in memory Hash.
|
5
|
-
class MemoryStorage
|
6
|
-
# Accepts a Hash or any Hash-like object as argument. Will use a plain Hash
|
7
|
-
# if none provided.
|
8
|
-
#
|
9
|
-
# Input:
|
10
|
-
#
|
11
|
-
# state = Hash{...Object => Object} # Default: nil
|
12
|
-
#
|
13
|
-
# Output: N/A
|
14
|
-
def initialize(state = nil)
|
15
|
-
validate!(state)
|
16
|
-
|
17
|
-
@state = state || {}
|
18
|
-
end
|
19
|
-
|
20
|
-
# Accepts a Hash or any Hash-like object as argument. Will use a plain Hash
|
21
|
-
# if none provided.
|
22
|
-
#
|
23
|
-
# Input:
|
24
|
-
#
|
25
|
-
# state = Hash{...Object => Object}
|
26
|
-
#
|
27
|
-
# Output: self
|
28
|
-
def reset!(state = nil)
|
29
|
-
initialize(state)
|
30
|
-
|
31
|
-
self
|
32
|
-
end
|
33
|
-
|
34
|
-
# Input:
|
35
|
-
#
|
36
|
-
# name = String
|
37
|
-
#
|
38
|
-
# Output: Object # Anything
|
39
|
-
def read!(name)
|
40
|
-
validate_string!(name, 'name')
|
41
|
-
|
42
|
-
@state[name]
|
43
|
-
end
|
44
|
-
|
45
|
-
# Input:
|
46
|
-
#
|
47
|
-
# name = String
|
48
|
-
# data = Object # Anything
|
49
|
-
#
|
50
|
-
# Output: self
|
51
|
-
def write!(name, data)
|
52
|
-
validate_string!(name, 'name')
|
53
|
-
|
54
|
-
@state[name] = data
|
55
|
-
|
56
|
-
self
|
57
|
-
end
|
58
|
-
|
59
|
-
# Input:
|
60
|
-
#
|
61
|
-
# name = String
|
62
|
-
#
|
63
|
-
# Output: self
|
64
|
-
def delete!(name)
|
65
|
-
validate_string!(name, 'name')
|
66
|
-
|
67
|
-
@state.delete(name)
|
68
|
-
|
69
|
-
self
|
70
|
-
end
|
71
|
-
|
72
|
-
# Input:
|
73
|
-
#
|
74
|
-
# name = String
|
75
|
-
#
|
76
|
-
# Output: true | false
|
77
|
-
def has?(name)
|
78
|
-
validate_string!(name, 'name')
|
79
|
-
|
80
|
-
@state.key?(name)
|
81
|
-
end
|
82
|
-
|
83
|
-
private
|
84
|
-
|
85
|
-
# Input:
|
86
|
-
#
|
87
|
-
# state = Object # Anything
|
88
|
-
#
|
89
|
-
# Output: N/A
|
90
|
-
def validate!(state)
|
91
|
-
return if state.nil? || state.is_a?(Hash)
|
92
|
-
|
93
|
-
validate_get!(state)
|
94
|
-
validate_set!(state)
|
95
|
-
validate_delete!(state)
|
96
|
-
validate_key!(state)
|
97
|
-
end
|
98
|
-
|
99
|
-
# Input:
|
100
|
-
#
|
101
|
-
# state = Object # Anything
|
102
|
-
#
|
103
|
-
# Output: N/A
|
104
|
-
def validate_get!(state)
|
105
|
-
unless state.respond_to?(:[])
|
106
|
-
raise(ArgumentError, 'Given state object does not respond to `:[]`')
|
107
|
-
end
|
108
|
-
|
109
|
-
arity = state.method(:[]).arity
|
110
|
-
unless arity == 1
|
111
|
-
raise(
|
112
|
-
ArgumentError,
|
113
|
-
"Given state object's `:[]` method arity must be 1, got: #{arity}"
|
114
|
-
)
|
115
|
-
end
|
116
|
-
end
|
117
|
-
|
118
|
-
# Input:
|
119
|
-
#
|
120
|
-
# state = Object # Anything
|
121
|
-
#
|
122
|
-
# Output: N/A
|
123
|
-
def validate_set!(state)
|
124
|
-
unless state.respond_to?(:[]=)
|
125
|
-
raise(ArgumentError, 'Given state object does not respond to `:[]=`')
|
126
|
-
end
|
127
|
-
|
128
|
-
arity = state.method(:[]=).arity
|
129
|
-
unless arity == 2
|
130
|
-
raise(
|
131
|
-
ArgumentError,
|
132
|
-
"Given state object's `:[]=` method arity must be 2, got: #{arity}"
|
133
|
-
)
|
134
|
-
end
|
135
|
-
end
|
136
|
-
|
137
|
-
# Input:
|
138
|
-
#
|
139
|
-
# state = Object # Anything
|
140
|
-
#
|
141
|
-
# Output: N/A
|
142
|
-
def validate_delete!(state)
|
143
|
-
unless state.respond_to?(:delete)
|
144
|
-
raise(ArgumentError, 'Given state object does not respond to `delete`')
|
145
|
-
end
|
146
|
-
|
147
|
-
arity = state.method(:delete).arity
|
148
|
-
unless arity == 1
|
149
|
-
raise(
|
150
|
-
ArgumentError,
|
151
|
-
"Given state object's `:delete` method arity must be 1, got: #{arity}"
|
152
|
-
)
|
153
|
-
end
|
154
|
-
end
|
155
|
-
|
156
|
-
# Input:
|
157
|
-
#
|
158
|
-
# state = Object # Anything
|
159
|
-
#
|
160
|
-
# Output: N/A
|
161
|
-
def validate_key!(state)
|
162
|
-
unless state.respond_to?(:key?)
|
163
|
-
raise(ArgumentError, 'Given state object does not respond to `:key?`')
|
164
|
-
end
|
165
|
-
|
166
|
-
arity = state.method(:key?).arity
|
167
|
-
unless arity == 1
|
168
|
-
raise(
|
169
|
-
ArgumentError,
|
170
|
-
"Given state object's `:key?` method arity must be 1, got: #{arity}"
|
171
|
-
)
|
172
|
-
end
|
173
|
-
end
|
174
|
-
|
175
|
-
# Input:
|
176
|
-
#
|
177
|
-
# arg = String
|
178
|
-
# name = String
|
179
|
-
#
|
180
|
-
# Output: N/A
|
181
|
-
def validate_string!(arg, name)
|
182
|
-
return if arg.nil? || arg.is_a?(String)
|
183
|
-
|
184
|
-
klass = arg.class
|
185
|
-
value = arg.inspect
|
186
|
-
|
187
|
-
raise(
|
188
|
-
ArgumentError,
|
189
|
-
"#{name} must be a String, got #{klass}: #{value}"
|
190
|
-
)
|
191
|
-
end
|
192
|
-
end
|
193
|
-
end
|