Class: ZipKit::OutputEnumerator
- Inherits:
-
Object
- Object
- ZipKit::OutputEnumerator
- Defined in:
- lib/zip_kit/output_enumerator.rb
Overview
The output enumerator makes it possible to "pull" from a ZipKit streamer
object instead of having it "push" writes to you. It will "stash" the block which
writes the ZIP archive through the streamer, and when you call each
on the Enumerator
it will yield you the bytes the block writes. Since it is an enumerator you can
use next
to take chunks written by the ZipKit streamer one by one. It can be very
convenient when you need to segment your ZIP output into bigger chunks for, say,
uploading them to a cloud storage provider such as S3.
Another use of the OutputEnumerator
is as a Rack response body - since a Rack
response body object must support #each
yielding successive binary strings.
Which is exactly what OutputEnumerator
does.
The enumerator can provide you some more conveinences for HTTP output - correct streaming headers and a body with chunked transfer encoding.
iterable_zip_body = ZipKit::OutputEnumerator.new do | streamer |
streamer.write_file('big.csv') do |sink|
CSV(sink) do |csv_writer|
csv_writer << Person.column_names
Person.all.find_each do |person|
csv_writer << person.attributes.values
end
end
end
end
You can grab the headers one usually needs for streaming from #streaming_http_headers
:
[200, iterable_zip_body.streaming_http_headers, iterable_zip_body]
to bypass things like Rack::ETag
and the nginx buffering.
Constant Summary collapse
- DEFAULT_WRITE_BUFFER_SIZE =
64 * 1024
Instance Method Summary collapse
-
#each {|String| ... } ⇒ Object
Executes the block given to the constructor with a Streamer and passes each written chunk to the block given to the method.
-
#initialize(write_buffer_size: DEFAULT_WRITE_BUFFER_SIZE, **streamer_options, &blk) ⇒ ZipKit::OutputEnumerator
constructor
Creates a new OutputEnumerator enumerator.
-
#streaming_http_headers ⇒ Hash
Returns a Hash of HTTP response headers you are likely to need to have your response stream correctly.
-
#to_headers_and_rack_response_body ⇒ Array
Returns a tuple of
headers, body
- headers are aHash
and the body is an object that can be used as a Rack response body.
Constructor Details
#initialize(write_buffer_size: DEFAULT_WRITE_BUFFER_SIZE, **streamer_options, &blk) ⇒ ZipKit::OutputEnumerator
Creates a new OutputEnumerator enumerator. The enumerator can be read from using each
,
and the creation of the ZIP is in lockstep with the caller calling each
on the returned
output enumerator object. This can be used when the calling program wants to stream the
output of the ZIP archive and throttle that output, or split it into chunks, or use it
as a generator.
For example:
# The block given to {output_enum} won't be executed immediately - rather it
# will only start to execute when the caller starts to read from the output
# by calling `each`
body = ::ZipKit::OutputEnumerator.new(writer: CustomWriter) do |streamer|
streamer.add_stored_entry(filename: 'large.tif', size: 1289894, crc32: 198210)
streamer << large_file.read(1024*1024) until large_file.eof?
...
end
body.each do |bin_string|
# Send the output somewhere, buffer it in a file etc.
# The block passed into `initialize` will only start executing once `#each`
# is called
...
end
73 74 75 76 77 |
# File 'lib/zip_kit/output_enumerator.rb', line 73 def initialize(write_buffer_size: DEFAULT_WRITE_BUFFER_SIZE, **, &blk) @streamer_options = .to_h @bufsize = write_buffer_size.to_i @archiving_block = blk end |
Instance Method Details
#each {|String| ... } ⇒ Object
Executes the block given to the constructor with a Streamer
and passes each written chunk to the block given to the method. This allows one
to "take" output of the ZIP piecewise. If called without a block will return an Enumerator
that you can pull data from using next
.
NOTE Because the WriteBuffer
inside this object can reuse the buffer, it is important
that the String
that is yielded either gets consumed eagerly (written byte-by-byte somewhere, or #dup
-ed)
since the write buffer will clear it after your block returns. If you expand this Enumerator
eagerly into an Array you might notice that a lot of the segments of your ZIP output are
empty - this means that you need to duplicate them.
91 92 93 94 95 96 97 98 99 100 |
# File 'lib/zip_kit/output_enumerator.rb', line 91 def each if block_given? block_write = ZipKit::BlockWrite.new { |chunk| yield(chunk) } buffer = ZipKit::WriteBuffer.new(block_write, @bufsize) ZipKit::Streamer.open(buffer, **@streamer_options, &@archiving_block) buffer.flush else enum_for(:each) end end |
#streaming_http_headers ⇒ Hash
Returns a Hash of HTTP response headers you are likely to need to have your response stream correctly.
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 |
# File 'lib/zip_kit/output_enumerator.rb', line 105 def streaming_http_headers _headers = { # We need to ensure Rack::ETag does not suddenly start buffering us, see # https://github.com/rack/rack/issues/1619#issuecomment-606315714 # Set this even when not streaming for consistency. The fact that there would be # a weak ETag generated would mean that the middleware buffers, so we have tests for that. "Last-Modified" => Time.now.httpdate, # Make sure Rack::Deflater does not touch our response body either, see # https://github.com/felixbuenemann/xlsxtream/issues/14#issuecomment-529569548 "Content-Encoding" => "identity", # Disable buffering for both nginx and Google Load Balancer, see # https://cloud.google.com/appengine/docs/flexible/how-requests-are-handled?tab=python#x-accel-buffering "X-Accel-Buffering" => "no", # Set the correct content type. This should be overridden if you need to # serve things such as EPubs and other derived ZIP formats. "Content-Type" => "application/zip" } end |
#to_headers_and_rack_response_body ⇒ Array
Returns a tuple of headers, body
- headers are a Hash
and the body is
an object that can be used as a Rack response body. This method used to accept arguments
but will now just ignore them.
129 130 131 |
# File 'lib/zip_kit/output_enumerator.rb', line 129 def to_headers_and_rack_response_body(*, **) [streaming_http_headers, self] end |