diverter 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: 2b1f6a4d524f4b2f602291f5b6ffa9907cb6ea9bd1d6df9f1029b83d0029002c
4
+ data.tar.gz: 976cb8b2bd3d951c82588a3756933d4c9e1d5cdc3034188244891d053defc1f2
5
+ SHA512:
6
+ metadata.gz: ffe5489de472fe31f1a582fe9dd6827f3e6941676ee88754c2743f4d0b27fc78a04e999df36aae1650a5042f4be5ce5356192e39d3db7e547ad519f9a0f6b7f4
7
+ data.tar.gz: 4f612a8ab935cad717d62e15ce8a94e85dfcd18fc0c98c2067d3dc8ce94e461a3f2c18e4c92b0bb042433eb9b0593c5f5acfa3714cbcb29b90b061094ac1a2d8
data/.gitignore ADDED
@@ -0,0 +1,22 @@
1
+ *.gem
2
+ *.rbc
3
+ .bundle
4
+ .config
5
+ .yardoc
6
+ Gemfile.lock
7
+ InstalledFiles
8
+ _yardoc
9
+ coverage
10
+ doc/
11
+ lib/bundler/man
12
+ pkg
13
+ rdoc
14
+ spec/reports
15
+ test/tmp
16
+ test/version_tmp
17
+ tmp
18
+ *.bundle
19
+ *.so
20
+ *.o
21
+ *.a
22
+ mkmf.log
data/.rspec ADDED
@@ -0,0 +1,2 @@
1
+ --format documentation
2
+ --color
data/Gemfile ADDED
@@ -0,0 +1,4 @@
1
+ source 'https://rubygems.org'
2
+
3
+ # Specify your gem's dependencies in levenshtein_rb.gemspec
4
+ gemspec
data/LICENSE.txt ADDED
@@ -0,0 +1,23 @@
1
+ Copyright (c) 2024 Robin Stammer
2
+
3
+
4
+ MIT License
5
+
6
+ Permission is hereby granted, free of charge, to any person obtaining
7
+ a copy of this software and associated documentation files (the
8
+ "Software"), to deal in the Software without restriction, including
9
+ without limitation the rights to use, copy, modify, merge, publish,
10
+ distribute, sublicense, and/or sell copies of the Software, and to
11
+ permit persons to whom the Software is furnished to do so, subject to
12
+ the following conditions:
13
+
14
+ The above copyright notice and this permission notice shall be
15
+ included in all copies or substantial portions of the Software.
16
+
17
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
18
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
19
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
20
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
21
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
22
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
23
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,44 @@
1
+ # diverter
2
+
3
+ Lightweight Ruby library for executing tasks in a seperate Ruby subprocesses
4
+
5
+ ### Motivation
6
+
7
+ Ruby very rarely releases memory back to the operating system once allocated.
8
+ This gem realizes a neat trick as suggested by the Book "Ruby Performance Optimization"
9
+ by [Alexander Dymo](https://github.com/adymo): https://pragprog.com/book/adrpo/ruby-performance-optimization
10
+
11
+ The crucial point is that exiting a subprocess will force Ruby to release
12
+ all the memory allocated in the subprocess back to the OS.
13
+
14
+ ### Synopsis
15
+
16
+ ```Ruby
17
+ class DecadentMemoryConsumer
18
+
19
+
20
+ # Needs a lot of memory
21
+ # to execute some magical
22
+ # task
23
+
24
+
25
+ end
26
+
27
+ # If the predicate :memory_almost_exceeded? evaluates to true,
28
+ # the block will be executed in a forked subprocess
29
+ # that is forced to terminate afterwards
30
+ Diverter.new(condition: memory_almost_exceeded?) do
31
+ DecadentMemoryConsumer.perform_memory_expensive_operation
32
+ end
33
+
34
+ # condition defaults to true, so this is equivalent of condition set to true
35
+ Diverter.new do
36
+ DecadentMemoryConsumer.perform_memory_expensive_operation
37
+ end
38
+ ```
39
+
40
+ ### Changelog
41
+
42
+ #### 0.0.1
43
+
44
+ Initial version
data/Rakefile ADDED
@@ -0,0 +1,5 @@
1
+ require "bundler/gem_tasks"
2
+ require "rspec/core/rake_task"
3
+
4
+ RSpec::Core::RakeTask.new(:spec)
5
+ task :default => :spec
data/diverter.gemspec ADDED
@@ -0,0 +1,24 @@
1
+ # coding: utf-8
2
+ lib = File.expand_path('../lib', __FILE__)
3
+ $LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
4
+ require 'diverter/version'
5
+
6
+ Gem::Specification.new do |spec|
7
+ spec.name = "diverter"
8
+ spec.version = Diverter::VERSION
9
+ spec.authors = ["Robin Stammer\n"]
10
+ spec.email = ["robin.stammer@posteo.de"]
11
+ spec.summary = %q{Lightweight tool for execution of Ruby code in a forked Ruby process}
12
+ spec.description = %q{Lightweight tool for execution of Ruby code in a forked Ruby process}
13
+ spec.homepage = ""
14
+ spec.license = "MIT"
15
+
16
+ spec.files = `git ls-files -z`.split("\x0")
17
+ spec.executables = spec.files.grep(%r{^bin/}) { |f| File.basename(f) }
18
+ spec.test_files = spec.files.grep(%r{^(test|spec|features)/})
19
+ spec.require_paths = ["lib"]
20
+
21
+ spec.add_development_dependency "bundler"
22
+ spec.add_development_dependency "rake"
23
+ spec.add_development_dependency "rspec"
24
+ end
@@ -0,0 +1,3 @@
1
+ class Diverter
2
+ VERSION = "0.0.1"
3
+ end
data/lib/diverter.rb ADDED
@@ -0,0 +1,21 @@
1
+ require "diverter/version"
2
+
3
+ class Diverter
4
+
5
+ def initialize(condition: true)
6
+ return unless block_given?
7
+
8
+ if !!condition
9
+ pid = fork do
10
+ yield
11
+ exit!(0)
12
+ end
13
+
14
+ # ensure the child process has terminated
15
+ Process::waitpid(pid)
16
+ else
17
+ yield
18
+ end
19
+ end
20
+
21
+ end
@@ -0,0 +1,57 @@
1
+ require 'spec_helper'
2
+
3
+ describe Diverter do
4
+
5
+ class Monkey
6
+ class << self
7
+ def cheek
8
+ end
9
+ end
10
+ end
11
+
12
+ it 'has a version number' do
13
+ expect(Diverter::VERSION).not_to be nil
14
+ end
15
+
16
+ describe 'its constructor' do
17
+ context 'without condition delivered' do
18
+ subject { described_class.new { Monkey.cheek } }
19
+
20
+ it 'executes block in a forked process' do
21
+ expect_any_instance_of(Object).to receive(:fork).and_call_original
22
+ subject
23
+ end
24
+ end
25
+
26
+ context 'with condition false' do
27
+ subject { described_class.new(condition: false) { Monkey.cheek } }
28
+
29
+ it 'executes block in unforked' do
30
+ expect(Monkey).to receive(:cheek)
31
+ expect_any_instance_of(Object).not_to receive(:fork)
32
+ subject
33
+ end
34
+ end
35
+
36
+ context 'with condition nil' do
37
+ subject { described_class.new(condition: nil) { Monkey.cheek } }
38
+
39
+ it 'executes block in unforked' do
40
+ expect(Monkey).to receive(:cheek)
41
+ expect_any_instance_of(Object).not_to receive(:fork)
42
+ subject
43
+ end
44
+ end
45
+
46
+
47
+ context 'with condition true' do
48
+ subject { described_class.new(condition: true) { Monkey.cheek } }
49
+
50
+ it 'executes block in a forked process' do
51
+ expect_any_instance_of(Object).to receive(:fork).and_call_original
52
+ subject
53
+ end
54
+ end
55
+
56
+ end
57
+ end
@@ -0,0 +1,2 @@
1
+ $LOAD_PATH.unshift File.expand_path('../../lib', __FILE__)
2
+ require 'diverter'
metadata ADDED
@@ -0,0 +1,100 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: diverter
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - 'Robin Stammer
8
+
9
+ '
10
+ autorequire:
11
+ bindir: bin
12
+ cert_chain: []
13
+ date: 2024-09-20 00:00:00.000000000 Z
14
+ dependencies:
15
+ - !ruby/object:Gem::Dependency
16
+ name: bundler
17
+ requirement: !ruby/object:Gem::Requirement
18
+ requirements:
19
+ - - ">="
20
+ - !ruby/object:Gem::Version
21
+ version: '0'
22
+ type: :development
23
+ prerelease: false
24
+ version_requirements: !ruby/object:Gem::Requirement
25
+ requirements:
26
+ - - ">="
27
+ - !ruby/object:Gem::Version
28
+ version: '0'
29
+ - !ruby/object:Gem::Dependency
30
+ name: rake
31
+ requirement: !ruby/object:Gem::Requirement
32
+ requirements:
33
+ - - ">="
34
+ - !ruby/object:Gem::Version
35
+ version: '0'
36
+ type: :development
37
+ prerelease: false
38
+ version_requirements: !ruby/object:Gem::Requirement
39
+ requirements:
40
+ - - ">="
41
+ - !ruby/object:Gem::Version
42
+ version: '0'
43
+ - !ruby/object:Gem::Dependency
44
+ name: rspec
45
+ requirement: !ruby/object:Gem::Requirement
46
+ requirements:
47
+ - - ">="
48
+ - !ruby/object:Gem::Version
49
+ version: '0'
50
+ type: :development
51
+ prerelease: false
52
+ version_requirements: !ruby/object:Gem::Requirement
53
+ requirements:
54
+ - - ">="
55
+ - !ruby/object:Gem::Version
56
+ version: '0'
57
+ description: Lightweight tool for execution of Ruby code in a forked Ruby process
58
+ email:
59
+ - robin.stammer@posteo.de
60
+ executables: []
61
+ extensions: []
62
+ extra_rdoc_files: []
63
+ files:
64
+ - ".gitignore"
65
+ - ".rspec"
66
+ - Gemfile
67
+ - LICENSE.txt
68
+ - README.md
69
+ - Rakefile
70
+ - diverter.gemspec
71
+ - lib/diverter.rb
72
+ - lib/diverter/version.rb
73
+ - spec/diverter_spec.rb
74
+ - spec/spec_helper.rb
75
+ homepage: ''
76
+ licenses:
77
+ - MIT
78
+ metadata: {}
79
+ post_install_message:
80
+ rdoc_options: []
81
+ require_paths:
82
+ - lib
83
+ required_ruby_version: !ruby/object:Gem::Requirement
84
+ requirements:
85
+ - - ">="
86
+ - !ruby/object:Gem::Version
87
+ version: '0'
88
+ required_rubygems_version: !ruby/object:Gem::Requirement
89
+ requirements:
90
+ - - ">="
91
+ - !ruby/object:Gem::Version
92
+ version: '0'
93
+ requirements: []
94
+ rubygems_version: 3.5.17
95
+ signing_key:
96
+ specification_version: 4
97
+ summary: Lightweight tool for execution of Ruby code in a forked Ruby process
98
+ test_files:
99
+ - spec/diverter_spec.rb
100
+ - spec/spec_helper.rb