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 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
@@ -0,0 +1,5 @@
1
+ ## [Unreleased]
2
+
3
+ ## [0.1.0] - 2025-02-19
4
+
5
+ - Initial release
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,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'minitest/test_task'
5
+
6
+ Minitest::TestTask.create
7
+
8
+ require 'rubocop/rake_task'
9
+
10
+ RuboCop::RakeTask.new
11
+
12
+ task default: %i[test rubocop]
@@ -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
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Psyllium
4
+ VERSION = '0.1.0'
5
+ end
data/lib/psyllium.rb ADDED
@@ -0,0 +1,10 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'timeout'
4
+ require 'forwardable'
5
+ require_relative 'psyllium/version'
6
+ require_relative 'psyllium/fiber'
7
+
8
+ # Gem to make it easier to work with Fibers.
9
+ module Psyllium
10
+ end
data/sig/psyllium.rbs ADDED
@@ -0,0 +1,4 @@
1
+ module Psyllium
2
+ VERSION: String
3
+ # See the writing guide of rbs: https://github.com/ruby/rbs#guides
4
+ end
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: []