psyllium 0.1.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 +7 -0
- data/.rubocop.yml +28 -0
- data/CHANGELOG.md +5 -0
- data/LICENSE.txt +12 -0
- data/README.md +134 -0
- data/Rakefile +12 -0
- data/lib/psyllium/fiber.rb +189 -0
- data/lib/psyllium/version.rb +5 -0
- data/lib/psyllium.rb +10 -0
- data/sig/psyllium.rbs +4 -0
- metadata +56 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: fec6f648a7f186a7ec0c885aadbf36b7770a5338453c0b22d903041901b08ea3
|
4
|
+
data.tar.gz: bfdbbc18b3def92c75f9dbc654ddc5878eef182e2b762172519cdda88a45e3dc
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 1b1fd0486f6ca2028f1630fc0ac34e16d6a99f6be152030a14ad225c2c916dbc048f69fe8e7a83d9fc021d9c25edd4a224d2caf1e71e7124be832f83532c544e
|
7
|
+
data.tar.gz: 619214a39896eb66ac91d853ff622112200783d2934464b61edc57c0f045d6b128068f2dbc91b4b49fdd3a912da222bdf0eaacad311a530e8c26cfb1d5822a6c
|
data/.rubocop.yml
ADDED
@@ -0,0 +1,28 @@
|
|
1
|
+
plugins:
|
2
|
+
- rubocop-minitest
|
3
|
+
- rubocop-rake
|
4
|
+
|
5
|
+
AllCops:
|
6
|
+
TargetRubyVersion: 3.1
|
7
|
+
NewCops: enable
|
8
|
+
|
9
|
+
Style/SymbolArray:
|
10
|
+
Enabled: false
|
11
|
+
|
12
|
+
Style/TrailingCommaInArguments:
|
13
|
+
Enabled: false
|
14
|
+
|
15
|
+
Style/TrailingCommaInHashLiteral:
|
16
|
+
Enabled: false
|
17
|
+
|
18
|
+
Style/TrailingCommaInArrayLiteral:
|
19
|
+
Enabled: false
|
20
|
+
|
21
|
+
Style/RaiseArgs:
|
22
|
+
Enabled: false
|
23
|
+
|
24
|
+
Metrics/BlockLength:
|
25
|
+
Enabled: false
|
26
|
+
|
27
|
+
Metrics/MethodLength:
|
28
|
+
Enabled: false
|
data/CHANGELOG.md
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,12 @@
|
|
1
|
+
Copyright (C) 2025 Ethan D. Estrada <ethan@misterfidget.com>
|
2
|
+
|
3
|
+
Permission to use, copy, modify, and/or distribute this software for any purpose
|
4
|
+
with or without fee is hereby granted.
|
5
|
+
|
6
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
7
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
8
|
+
FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
9
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
|
10
|
+
OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
11
|
+
TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
|
12
|
+
THIS SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,134 @@
|
|
1
|
+
# Psyllium: Makes using Ruby Fibers easier
|
2
|
+
|
3
|
+
> Psyllium \| SIL-ee-um \|
|
4
|
+
>
|
5
|
+
> 1. _Dietary_ the seed of a fleawort (especially Plantago psyllium). Mainly
|
6
|
+
> used as a supplement to improve dietary fiber consumption.
|
7
|
+
> 2. _Programming_ a Ruby gem to improve the experience of using Ruby Fiber
|
8
|
+
> primitives.
|
9
|
+
|
10
|
+
## What is Psyllium?
|
11
|
+
|
12
|
+
Psyllium is a library to make it easier to use Ruby Fibers for everyday
|
13
|
+
programming.
|
14
|
+
|
15
|
+
Ruby version 3 introduced the Fiber Scheduler interface, which makes it easier
|
16
|
+
to use Fibers for concurrent programming. However, native Thread objects still
|
17
|
+
have several useful methods that Fibers do not have.
|
18
|
+
|
19
|
+
The Psyllium library adds many of these methods to the builtin Fiber class such
|
20
|
+
as `start`, `join`, `value`, and others to make it easier to replace Thread
|
21
|
+
usage with Fiber usage, or to mix and match Thread and Fiber usage without
|
22
|
+
concern for which concurrency primitive is being used.
|
23
|
+
|
24
|
+
Assuming that a Fiber Scheduler is set, Psyllium Fibers can be used in ways
|
25
|
+
similar to Threads, with a similar interface, and with the added benefit of
|
26
|
+
much lower memory usage compared to native Threads.
|
27
|
+
|
28
|
+
## Why Psyllium?
|
29
|
+
|
30
|
+
Psyllium makes it easier to use auto-scheduled fibers and to block on their
|
31
|
+
execution.
|
32
|
+
|
33
|
+
Before Psyllium, the Fiber interface seemed to be centered around two types of
|
34
|
+
usage: it was assumed that Fibers would be used in one of two ways:
|
35
|
+
|
36
|
+
1. (Before Ruby 3) Explicitly and manually manipulated using `Fiber.resume`,
|
37
|
+
`Fiber.yield`, and `Fiber.alive?`.
|
38
|
+
2. (After Ruby 3) Fired off and forgotten about. In essence, left to the
|
39
|
+
scheduler to deal with. If you want a final value back you must use some
|
40
|
+
separate mechanism to track and retrieve it.
|
41
|
+
|
42
|
+
With Psyllium, it is possible to call `join` on a Fiber, just like a Thread.
|
43
|
+
Assuming other Fibers are simultaneously scheduled, they will continue
|
44
|
+
executing concurrently until the Fiber in question finishes joining.
|
45
|
+
|
46
|
+
It is also possible to call `value` to retrieve the final value (or exception)
|
47
|
+
returned from the block given to `Fiber.start`, in the exact same way as a
|
48
|
+
Thread. And just like with a Thread, calling `value` will first implicitly
|
49
|
+
`join` the Fiber. It is also possible to give a timeout limit when calling
|
50
|
+
`join` on a Fiber, just like with a Thread.
|
51
|
+
|
52
|
+
By using Fibers in this way, instead of Threads, memory usage can be
|
53
|
+
significantly reduced. Potentially thousands of Fibers can be spawned and
|
54
|
+
joined at a fraction of the memory cost of native Threads.
|
55
|
+
|
56
|
+
If your Ruby application directly manipulates Threads or Thread pools, and
|
57
|
+
those Threads spend most (or all) of their time just waiting on IO, then
|
58
|
+
consider trying Psyllium enhanced Fibers instead of Threads.
|
59
|
+
|
60
|
+
## Why _not_ Psyllium?
|
61
|
+
|
62
|
+
Circumstances where you shouldn't use Psyllium (or Fibers generally):
|
63
|
+
|
64
|
+
1. Your application is compute heavy. It uses Threads that call out to FFI code
|
65
|
+
and release the GVL when doing so (i.e. truly parallel code execution).
|
66
|
+
2. If you don't have concurrent code at all (i.e. the code must run serially).
|
67
|
+
|
68
|
+
## Installation
|
69
|
+
|
70
|
+
Install the gem and add to the application's Gemfile by executing:
|
71
|
+
|
72
|
+
```bash
|
73
|
+
bundle add psyllium
|
74
|
+
```
|
75
|
+
|
76
|
+
If bundler is not being used to manage dependencies, install the gem by executing:
|
77
|
+
|
78
|
+
```bash
|
79
|
+
gem install psyllium
|
80
|
+
```
|
81
|
+
|
82
|
+
## Usage
|
83
|
+
|
84
|
+
Instead of doing the following in your code:
|
85
|
+
|
86
|
+
```ruby
|
87
|
+
thread1 = Thread.start { long_running_io_operation_with_result1() }
|
88
|
+
thread2 = Thread.start { long_running_io_operation_with_result2() }
|
89
|
+
|
90
|
+
thread1.join
|
91
|
+
thread2.join
|
92
|
+
|
93
|
+
# `value` implicitly calls `join`, so the explicit `join` calls above are
|
94
|
+
# not strictly necessary.
|
95
|
+
value1 = thread1.value
|
96
|
+
value2 = thread2.value
|
97
|
+
```
|
98
|
+
|
99
|
+
You can now do this:
|
100
|
+
|
101
|
+
```ruby
|
102
|
+
# Adds new methods to Fiber
|
103
|
+
require 'psyllium'
|
104
|
+
|
105
|
+
# Calls to `Fiber.start` will fail if no scheduler is set beforehand.
|
106
|
+
Fiber.set_scheduler(SomeSchedulerImplementation.new)
|
107
|
+
|
108
|
+
fiber1 = Fiber.start { long_running_io_operation_with_result1() }
|
109
|
+
fiber2 = Fiber.start { long_running_io_operation_with_result2() }
|
110
|
+
|
111
|
+
fiber1.join
|
112
|
+
fiber2.join
|
113
|
+
|
114
|
+
# `value` implicitly calls `join`, so the explicit `join` calls above are
|
115
|
+
# not strictly necessary.
|
116
|
+
value1 = fiber1.value
|
117
|
+
value2 = fiber2.value
|
118
|
+
```
|
119
|
+
|
120
|
+
## Development
|
121
|
+
|
122
|
+
After checking out the repo, run `bin/setup` to install dependencies.
|
123
|
+
Then, run `rake test` to run the tests.
|
124
|
+
You can also run `bin/console` for an interactive prompt
|
125
|
+
that will allow you to experiment.
|
126
|
+
|
127
|
+
To install this gem onto your local machine, run `bundle exec rake install`.
|
128
|
+
To release a new version, update the version number in `version.rb`,
|
129
|
+
and then run `bundle exec rake release`, which will create a git tag for the version,
|
130
|
+
push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
131
|
+
|
132
|
+
## Contributing
|
133
|
+
|
134
|
+
Bug reports and pull requests are welcome on GitHub at: <https://github.com/eestrada/psyllium>
|
data/Rakefile
ADDED
@@ -0,0 +1,189 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'timeout'
|
4
|
+
|
5
|
+
module Psyllium
|
6
|
+
# Wrap Exception instances for propagation
|
7
|
+
class ExceptionalCompletionError < FiberError
|
8
|
+
def initialize(expt)
|
9
|
+
@internal_exception = expt
|
10
|
+
super(@internal_exception)
|
11
|
+
end
|
12
|
+
|
13
|
+
def cause
|
14
|
+
@internal_exception
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
# Holds per-Fiber state for Psyllium operations.
|
19
|
+
class State
|
20
|
+
attr_reader :mutex
|
21
|
+
attr_accessor :started, :joined, :value, :exception
|
22
|
+
|
23
|
+
def initialize
|
24
|
+
@mutex = Thread::Mutex.new
|
25
|
+
@started = false
|
26
|
+
@joined = false
|
27
|
+
@value = nil
|
28
|
+
@exception = nil
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
# Meant to be used with `extend` on Fiber class.
|
33
|
+
module FiberClassMethods
|
34
|
+
# A new method is used to create Psyllium Fibers for several reasons:
|
35
|
+
#
|
36
|
+
# 1. This ensures that existing behavior for Fibers is not changed.
|
37
|
+
#
|
38
|
+
# 2. Modifying the actual instances variables of Fibers does not work well
|
39
|
+
# with certain schedulers like Async which expect to also wrap the given
|
40
|
+
# block in another block.
|
41
|
+
#
|
42
|
+
# 3. The `start` method is also available on the `Thread` class, so this
|
43
|
+
# makes it easy to change out one for the other.
|
44
|
+
def start(*args, &block)
|
45
|
+
raise ArgumentError.new('No block given') unless block
|
46
|
+
|
47
|
+
Fiber.schedule do
|
48
|
+
state = state_get(create_missing: true)
|
49
|
+
state.mutex.synchronize do
|
50
|
+
state.started = true
|
51
|
+
state.value = block.call(*args)
|
52
|
+
rescue StandardError => e
|
53
|
+
state.exception = e
|
54
|
+
ensure
|
55
|
+
state.joined = true
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
|
60
|
+
def state_get(fiber: Fiber.current, create_missing: false)
|
61
|
+
# Psyllium state is a thread local variable because Fibers cannot (yet)
|
62
|
+
# migrates across threads anyway.
|
63
|
+
#
|
64
|
+
# A `WeakKeyMap` is used so that when a Fiber is garbage collected, the
|
65
|
+
# associated Psyllium::State will be garbage collected as well.
|
66
|
+
state = Thread.current.thread_variable_get(:psyllium_state) || Thread.current.thread_variable_set(
|
67
|
+
:psyllium_state, ObjectSpace::WeakKeyMap.new
|
68
|
+
)
|
69
|
+
create_missing ? (state[fiber] ||= State.new) : state[fiber]
|
70
|
+
end
|
71
|
+
end
|
72
|
+
|
73
|
+
# A module meant to extend the builtin Fiber class to make it easier to use
|
74
|
+
# in a more Thread-like manner.
|
75
|
+
module FiberInstanceMethods
|
76
|
+
# Waits for Fiber to complete, using join, and returns its value. If Fiber
|
77
|
+
# completed with an exception, raises `ExceptionalCompletionError`, with
|
78
|
+
# the original exception as its `cause`.
|
79
|
+
def value
|
80
|
+
join
|
81
|
+
raise ExceptionalCompletionError.new(state.exception) if state.exception
|
82
|
+
|
83
|
+
state.value
|
84
|
+
end
|
85
|
+
|
86
|
+
# Mimic Thread `status` method.
|
87
|
+
#
|
88
|
+
# `"run"` will only be returned if this method is called on
|
89
|
+
# `Fiber.current`.
|
90
|
+
#
|
91
|
+
# `"sleep"` is returned for any Fiber that is `alive?`, but not
|
92
|
+
# `Fiber.current`.
|
93
|
+
#
|
94
|
+
# `nil` is returned if the Fiber completed with an exception.
|
95
|
+
#
|
96
|
+
# `false` is returned if the Fiber completed without exception.
|
97
|
+
#
|
98
|
+
# `"abort"` status is never returned because a Fiber does not have a state
|
99
|
+
# like this that is detectable or observable. If `kill` is called on a
|
100
|
+
# Fiber, the operation will happen immediately; there is not a point in
|
101
|
+
# time where `status` can be called between the call to `kill` and the
|
102
|
+
# point at which the Fiber is killed.
|
103
|
+
def status
|
104
|
+
if self == ::Fiber.current
|
105
|
+
'run'
|
106
|
+
elsif alive?
|
107
|
+
'sleep'
|
108
|
+
elsif state(suppress_error: true)&.exception
|
109
|
+
nil
|
110
|
+
else
|
111
|
+
false
|
112
|
+
end
|
113
|
+
end
|
114
|
+
|
115
|
+
# Mimic Thread `stop?` method.
|
116
|
+
#
|
117
|
+
# Return `true` if sleeping or completed.
|
118
|
+
def stop?
|
119
|
+
status == 'sleep' || !alive?
|
120
|
+
end
|
121
|
+
|
122
|
+
# Wait until execution completes. Return the Fiber instance. If `limit` is
|
123
|
+
# reached, returns `nil` instead.
|
124
|
+
#
|
125
|
+
# `join` may be called more than once.
|
126
|
+
def join(limit = nil) # rubocop:disable Metrics/CyclomaticComplexity,Metrics/AbcSize
|
127
|
+
return self if state.joined
|
128
|
+
raise FiberError.new('Cannot join self') if eql?(::Fiber.current)
|
129
|
+
raise FiberError.new('Cannot join when calling Fiber is blocking') if ::Fiber.current.blocking?
|
130
|
+
raise FiberError.new('Cannot join when called Fiber is blocking') if blocking?
|
131
|
+
raise FiberError.new('Cannot join without Fiber scheduler set') unless ::Fiber.scheduler
|
132
|
+
raise FiberError.new('Cannot join unstarted Fiber') unless state.started
|
133
|
+
|
134
|
+
# Once this mutex finishes synchronizing, that means the initial
|
135
|
+
# calculation is done and we can return `self`, which is the Fiber
|
136
|
+
# instance.
|
137
|
+
Timeout.timeout(limit) { state.mutex.synchronize { self } }
|
138
|
+
rescue Timeout::Error
|
139
|
+
# mimic Thread behavior by returning `nil` on timeout.
|
140
|
+
nil
|
141
|
+
end
|
142
|
+
|
143
|
+
private
|
144
|
+
|
145
|
+
def state(suppress_error: false)
|
146
|
+
fiber_state = self.class.state_get(fiber: self)
|
147
|
+
raise Error.new('No Psyllium state for this fiber') unless fiber_state || suppress_error
|
148
|
+
|
149
|
+
fiber_state
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
# Inherits from the builtin Fiber class, and adds additional functionality to
|
154
|
+
# make it behave more like a Thread.
|
155
|
+
class Fiber < ::Fiber
|
156
|
+
extend ::Psyllium::FiberClassMethods
|
157
|
+
# This must be prepended so that its implementation of `initialize` is called
|
158
|
+
# first.
|
159
|
+
include ::Psyllium::FiberInstanceMethods
|
160
|
+
|
161
|
+
# The `Fiber.kill` method only exists in later versions of Ruby.
|
162
|
+
if instance_methods.include?(:kill)
|
163
|
+
# Thread has the same aliases
|
164
|
+
alias terminate kill
|
165
|
+
alias exit kill
|
166
|
+
end
|
167
|
+
end
|
168
|
+
|
169
|
+
# TODO: figure out how to do this properly
|
170
|
+
# def self.patch_builtin_fiber!
|
171
|
+
# return if ::Fiber.is_a?(FiberMethods)
|
172
|
+
|
173
|
+
# ::Fiber.singleton_class.prepend(FiberMethods)
|
174
|
+
# end
|
175
|
+
end
|
176
|
+
|
177
|
+
class ::Fiber # rubocop:disable Style/Documentation
|
178
|
+
extend ::Psyllium::FiberClassMethods
|
179
|
+
# This must be prepended so that its implementation of `initialize` is called
|
180
|
+
# first.
|
181
|
+
include ::Psyllium::FiberInstanceMethods
|
182
|
+
|
183
|
+
# The `Fiber.kill` method only exists in later versions of Ruby.
|
184
|
+
if instance_methods.include?(:kill)
|
185
|
+
# Thread has the same aliases
|
186
|
+
alias terminate kill
|
187
|
+
alias exit kill
|
188
|
+
end
|
189
|
+
end
|
data/lib/psyllium.rb
ADDED
data/sig/psyllium.rbs
ADDED
metadata
ADDED
@@ -0,0 +1,56 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: psyllium
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Ethan Estrada
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2025-03-12 00:00:00.000000000 Z
|
12
|
+
dependencies: []
|
13
|
+
description:
|
14
|
+
email:
|
15
|
+
- ethan@misterfidget.com
|
16
|
+
executables: []
|
17
|
+
extensions: []
|
18
|
+
extra_rdoc_files: []
|
19
|
+
files:
|
20
|
+
- ".rubocop.yml"
|
21
|
+
- CHANGELOG.md
|
22
|
+
- LICENSE.txt
|
23
|
+
- README.md
|
24
|
+
- Rakefile
|
25
|
+
- lib/psyllium.rb
|
26
|
+
- lib/psyllium/fiber.rb
|
27
|
+
- lib/psyllium/version.rb
|
28
|
+
- sig/psyllium.rbs
|
29
|
+
homepage: https://github.com/eestrada/psyllium
|
30
|
+
licenses:
|
31
|
+
- 0BSD
|
32
|
+
metadata:
|
33
|
+
allowed_push_host: https://rubygems.org/
|
34
|
+
homepage_uri: https://github.com/eestrada/psyllium
|
35
|
+
source_code_uri: https://github.com/eestrada/psyllium
|
36
|
+
rubygems_mfa_required: 'true'
|
37
|
+
post_install_message:
|
38
|
+
rdoc_options: []
|
39
|
+
require_paths:
|
40
|
+
- lib
|
41
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
42
|
+
requirements:
|
43
|
+
- - ">="
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: 3.1.0
|
46
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
47
|
+
requirements:
|
48
|
+
- - ">="
|
49
|
+
- !ruby/object:Gem::Version
|
50
|
+
version: '0'
|
51
|
+
requirements: []
|
52
|
+
rubygems_version: 3.5.22
|
53
|
+
signing_key:
|
54
|
+
specification_version: 4
|
55
|
+
summary: A Ruby gem making it easier to use Fibers.
|
56
|
+
test_files: []
|