mirror of
https://github.com/lloeki/ruby-tee.git
synced 2025-12-06 10:34:42 +01:00
refactored
This commit is contained in:
parent
d5895d2f68
commit
471e9d51ae
1 changed files with 51 additions and 32 deletions
83
tee.rb
83
tee.rb
|
|
@ -25,63 +25,82 @@ class IO
|
||||||
digest_with Digest::SHA2.new(256), chunk_size
|
digest_with Digest::SHA2.new(256), chunk_size
|
||||||
end
|
end
|
||||||
|
|
||||||
def fiber_chunks(chunk_size=1024)
|
|
||||||
Enumerator.new do |y|
|
|
||||||
until eof?
|
|
||||||
Fiber.yield
|
|
||||||
y << read(chunk_size)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
def tee(*procs)
|
def tee(*procs)
|
||||||
results = procs.map { nil }
|
|
||||||
ios = procs.map do |proc|
|
ios = procs.map do |proc|
|
||||||
#IO.new
|
FiberChunkedIO.new &proc
|
||||||
self
|
|
||||||
end
|
end
|
||||||
fibers = procs.map.with_index do |proc, i|
|
|
||||||
# set up communication
|
|
||||||
# start each proc, which will yield just before reading
|
|
||||||
Fiber.new do
|
|
||||||
results[i] = proc.call ios[i]
|
|
||||||
end
|
|
||||||
end.each { |fiber| fiber.resume }
|
|
||||||
prev_pos = tell
|
|
||||||
chunks.each do |chunk|
|
chunks.each do |chunk|
|
||||||
new_pos = tell
|
|
||||||
# for each proc
|
# for each proc
|
||||||
# - copy chunk into its dedicated IO stream
|
# - copy chunk into its dedicated IO stream
|
||||||
# - resume
|
# - resume
|
||||||
procs.each.with_index do |proc, i|
|
ios.each.with_index do |io, i|
|
||||||
ios[i].seek prev_pos
|
$stdout.puts "#{i}:#{tell}"
|
||||||
#ios[i].write chunk
|
io.write chunk
|
||||||
fibers[i].resume
|
io.resume
|
||||||
ios[i].seek new_pos
|
|
||||||
end
|
end
|
||||||
prev_pos = new_pos
|
|
||||||
end
|
end
|
||||||
results
|
ios.map { |io| io.resume }
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class FiberChunkedIO
|
||||||
|
def initialize(chunk_size=1024, &block)
|
||||||
|
@chunk_size = chunk_size
|
||||||
|
@chunks = []
|
||||||
|
@fiber = Fiber.new do
|
||||||
|
@result = block.call self
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
File.open("tee.rb") do |f|
|
def resume
|
||||||
|
@fiber.resume
|
||||||
|
@result
|
||||||
|
end
|
||||||
|
|
||||||
|
def write(chunk)
|
||||||
|
if chunk.size > @chunk_size
|
||||||
|
raise ArgumentError.new("chunk size mismatch: expected #{@chunk_size}, got #{chunk.size}")
|
||||||
|
end
|
||||||
|
|
||||||
|
@chunks << chunk
|
||||||
|
chunk.size
|
||||||
|
end
|
||||||
|
|
||||||
|
def read(chunk_size)
|
||||||
|
unless chunk_size == @chunk_size
|
||||||
|
raise ArgumentError.new("chunk size mismatch: expected #{@chunk_size}, got #{chunk_size}")
|
||||||
|
end
|
||||||
|
|
||||||
|
@chunks.shift
|
||||||
|
end
|
||||||
|
|
||||||
|
def chunks(chunk_size=1024)
|
||||||
|
Enumerator.new do |y|
|
||||||
|
while chunk = read(chunk_size)
|
||||||
|
y << chunk
|
||||||
|
Fiber.yield
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
File.open("test") do |f|
|
||||||
|
|
||||||
sha1_proc = lambda do |f|
|
sha1_proc = lambda do |f|
|
||||||
digest = Digest::SHA1.new
|
digest = Digest::SHA1.new
|
||||||
f.fiber_chunks.each { |chunk| digest << chunk }
|
f.chunks.each { |chunk| digest << chunk }
|
||||||
digest
|
digest
|
||||||
end
|
end
|
||||||
|
|
||||||
sha2_proc = lambda do |f|
|
sha2_proc = lambda do |f|
|
||||||
digest = Digest::SHA2.new(256)
|
digest = Digest::SHA2.new(256)
|
||||||
f.fiber_chunks.each { |chunk| digest << chunk }
|
f.chunks.each { |chunk| digest << chunk }
|
||||||
digest
|
digest
|
||||||
end
|
end
|
||||||
|
|
||||||
puts_proc = lambda do |f|
|
puts_proc = lambda do |f|
|
||||||
f.fiber_chunks.each { |chunk| puts chunk }
|
f.chunks.each { |chunk| puts chunk.length }
|
||||||
end
|
end
|
||||||
|
|
||||||
p f.tee(sha1_proc, sha2_proc, puts_proc)
|
p f.tee(sha1_proc, sha2_proc, puts_proc)
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue