Module: Roda::RodaPlugins::Chunked::InstanceMethods

Defined in:
lib/roda/plugins/chunked.rb

Instance Method Summary collapse

Instance Method Details

#chunked(template, opts = OPTS, &block) ⇒ Object

Render a response to the user in chunks. See Chunked for an overview. If a block is given, it is passed to #delay.



227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
# File 'lib/roda/plugins/chunked.rb', line 227

def chunked(template, opts=OPTS, &block)
  unless defined?(@_chunked)
    @_chunked = !self.opts[:force_chunked_encoding] || @_request.http_version == "HTTP/1.1" 
  end

  if block
    delay(&block)
  end

  unless @_chunked
    # If chunking is disabled, do a normal rendering of the view.
    run_delayed_blocks
    return view(template, opts)
  end

  if template.is_a?(Hash)
    if opts.empty?
      opts = template
    else
      opts = Hash[opts].merge!(template)
    end
  end
  
  # Hack so that the arguments don't need to be passed
  # through the response and body objects.
  @_each_chunk_args = [template, opts]

  res = response
  headers = res.headers
  if chunk_headers = self.opts[:chunk_headers]
    headers.merge!(chunk_headers)
  end
  if self.opts[:force_chunked_encoding]
    res[RodaResponseHeaders::TRANSFER_ENCODING] = 'chunked'
    body = Body.new(self)
  else
    body = StreamBody.new(self)
  end

  throw :halt, res.finish_with_body(body)
end

#delay(&block) ⇒ Object

Delay the execution of the block until right before the content template is to be rendered.

Raises:



271
272
273
274
# File 'lib/roda/plugins/chunked.rb', line 271

def delay(&block)
  raise RodaError, "must pass a block to Roda#delay" unless block
  (@_delays ||= []) << block
end

#each_chunkObject

Yield each chunk of the template rendering separately.



277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
# File 'lib/roda/plugins/chunked.rb', line 277

def each_chunk
  response.body.each{|s| yield s}

  template, opts = @_each_chunk_args

  # Use a lambda for the flusher, so that a call to flush
  # by a template can result in this method yielding a chunk
  # of the response.
  @_flusher = lambda do
    yield @_out_buf
    @_out_buf = String.new
  end

  if layout_opts  = view_layout_opts(opts)
    @_out_buf = render_template(layout_opts) do
      flush
      run_delayed_blocks
      yield opts[:content] || render_template(template, opts)
      nil
    end
  else
    run_delayed_blocks
    yield view(template, opts)
  end

  flush
rescue => e
  handle_chunk_error(e)
end

#flushObject

Call the flusher if one is defined. If one is not defined, this is a no-op, so flush can be used inside views without breaking things if chunking is not used.



315
316
317
# File 'lib/roda/plugins/chunked.rb', line 315

def flush
  @_flusher.call if @_flusher
end

#handle_chunk_error(e) ⇒ Object

By default, raise the exception.



308
309
310
# File 'lib/roda/plugins/chunked.rb', line 308

def handle_chunk_error(e)
  raise e
end

#no_chunk!Object

Disable chunking for the current request. Mostly useful when chunking is turned on by default.



211
212
213
# File 'lib/roda/plugins/chunked.rb', line 211

def no_chunk!
  @_chunked = false
end

#view(*a) ⇒ Object

If chunking by default, call chunked if it hasn’t yet been called and chunking is not specifically disabled.



217
218
219
220
221
222
223
# File 'lib/roda/plugins/chunked.rb', line 217

def view(*a)
  if opts[:chunk_by_default] && !defined?(@_chunked) && !defined?(yield)
    chunked(*a)
  else
    super
  end
end