non-blocking-spawn
non-blocking-spawn
A Crystal shard that spawns a fiber in any worker thread BUT the current one.
Purpose
When running Crystal with -Dpreview_mt sometimes fibers spawn in the current thread that might be blocked at that moment or for the duration of the run.
This happens by design. Top level spawn has an option to force a fiber to spawn in the current thread (same_thread), but not one for the opposite (since it's probably not in high demand).
That's what this shard adds.
It's a copy of the spawn method but when same_thread is false, it goes through all available worker threads, filters out the current one and picks one at random to spawn the fiber in.
While it might sound a bit useless, there are legit use cases:
- GTK: The main thread is blocked and remains blocked for the whole time. Using non-blocking-spawn you can spawn fibers and be sure that all of them will execute and finish.
- Wider range of examples (see: ./spec/non-blocking-spawn_spec.cr): In the spec, two of the tests are similar but one of them uses non-blocking-spawn while the other uses top level spawn. The first one finishes withibeing equal to4(/default amount of worker threads) while the second one finishes withibeing equal to3. This is due to one of the fibers being spawned in the current thread.
WARNING
- 
PLEASE don't use with newer or older versions of Crystal without checking that the spawn method is the same as the top level one (apart from the non-blocking-thread part). This shard uses private interfaces that may break at any point without notice. 
- 
Assumes that all crystal commands (including spec) are being ran with the-Dpreview_mtflag. TheNon::Blocking.threadsmethod is available to check whether there are worker threads - other than the current one - and act accordingly (probably run in sync).
Installation
- Add the dependency to your shard.yml:
dependencies:
  non-blocking-spawn:
    github: GeopJr/non-blocking-spawn
- Run shards install
Usage
You can find non-blocking-spawn's docs on the sidebar.
require "non-blocking-spawn"
Non::Blocking.spawn do
  # expensive blocking code
end
# other expensive blocking code
Development
- Everything runs with the -Dpreview_mtflag.
- When bumping Crystal version make sure that the method matches the top level one.
Contributing
- Read the Code of Conduct
- Fork it (https://github.com/GeopJr/non-blocking-spawn/fork)
- Create your feature branch (git checkout -b my-new-feature)
- Commit your changes (git commit -am 'Add some feature')
- Push to the branch (git push origin my-new-feature)
- Create a new Pull Request