Class: Upr::InputWrapper

Inherits:
Struct
  • Object
show all
Includes:
Params
Defined in:
lib/upr/input_wrapper.rb

Overview

the underlying middlware for for wrapping env, this should typically be installed before other middlewares that may wrap env in the middleware chain.

Instance Attribute Summary collapse

Instance Method Summary collapse

Methods included from Params

#extract_upload_id

Constructor Details

#initialize(app, options = {}) ⇒ InputWrapper

Returns a new instance of InputWrapper.



14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
# File 'lib/upr/input_wrapper.rb', line 14

def initialize(app, options = {})
  super(app,
        Array(options[:path_info] || nil),
        options[:frequency] || 1,
        options[:backend])

  # support :drb for compatibility with mongrel_upload_progress
  if options[:drb]
    backend and raise ArgumentError, ":backend and :drb are incompatible"
    require 'drb'
    DRb.start_service
    self.backend = DRbObject.new(nil, options[:drb])
  elsif String === backend
    # allow people to use strings in case their backend gets
    # lazy-loaded (like an ActiveRecord model)
    self.backend = eval(backend)
  else
    self.backend ||= Upr::Monitor.new
  end
end

Instance Attribute Details

#appObject

Returns the value of attribute app

Returns:

  • (Object)

    the current value of app



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def app
  @app
end

#backendObject

Returns the value of attribute backend

Returns:

  • (Object)

    the current value of backend



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def backend
  @backend
end

#content_lengthObject

Returns the value of attribute content_length

Returns:

  • (Object)

    the current value of content_length



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def content_length
  @content_length
end

#frequencyObject

Returns the value of attribute frequency

Returns:

  • (Object)

    the current value of frequency



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def frequency
  @frequency
end

#inputObject

Returns the value of attribute input

Returns:

  • (Object)

    the current value of input



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def input
  @input
end

#mtimeObject

Returns the value of attribute mtime

Returns:

  • (Object)

    the current value of mtime



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def mtime
  @mtime
end

#path_infoObject

Returns the value of attribute path_info

Returns:

  • (Object)

    the current value of path_info



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def path_info
  @path_info
end

#posObject

Returns the value of attribute pos

Returns:

  • (Object)

    the current value of pos



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def pos
  @pos
end

#seenObject

Returns the value of attribute seen

Returns:

  • (Object)

    the current value of seen



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def seen
  @seen
end

#upload_idObject

Returns the value of attribute upload_id

Returns:

  • (Object)

    the current value of upload_id



8
9
10
# File 'lib/upr/input_wrapper.rb', line 8

def upload_id
  @upload_id
end

Instance Method Details

#_call(env, uid, length) ⇒ Object



51
52
53
54
55
56
57
58
59
60
# File 'lib/upr/input_wrapper.rb', line 51

def _call(env, uid, length)
  self.upload_id = uid
  self.mtime = self.pos = self.seen = 0
  self.input = env["rack.input"]
  env["rack.input"] = self
  self.content_length = length
  backend.start(upload_id, length)

  app.call(env)
end

#_finishObject



72
73
74
75
# File 'lib/upr/input_wrapper.rb', line 72

def _finish
  self.seen = backend.finish(upload_id).seen
  self.content_length ||= self.seen
end

#_incr(nr) ⇒ Object



62
63
64
65
66
67
68
69
70
# File 'lib/upr/input_wrapper.rb', line 62

def _incr(nr)
  self.pos += nr
  _finish if content_length && pos >= content_length
  if (nr = pos - seen) > 0 && mtime <= (Time.now.to_i - frequency)
    backend.incr(upload_id, nr)
    self.seen = pos
    self.mtime = Time.now.to_i
  end
end

#call(env) ⇒ Object



35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
# File 'lib/upr/input_wrapper.rb', line 35

def call(env)
  if path_info.empty? || path_info.include?(env["PATH_INFO"])
    # benefit curl users...
    /\A100-continue\z/i =~ env['HTTP_EXPECT'] and return [ 100, {}, [] ]

    length = env["CONTENT_LENGTH"] and length = length.to_i
    chunked = env["TRANSFER_ENCODING"] =~ %r{\Achunked\z}i and length = nil
    if chunked || (length && length > 0)
      if uid = extract_upload_id(env)
        return dup._call(env, uid, length)
      end
    end
  end
  app.call(env)
end

#each(&block) ⇒ Object



105
106
107
108
109
110
111
# File 'lib/upr/input_wrapper.rb', line 105

def each(&block)
  input.each do |chunk| # usually just a line
    _incr(chunk.size)
    yield chunk
  end
  _finish
end

#getsObject



93
94
95
96
97
# File 'lib/upr/input_wrapper.rb', line 93

def gets
  rv = input.gets
  rv.nil? ? _finish : _incr(rv.size)
  rv
end

#read(*args) ⇒ Object



99
100
101
102
103
# File 'lib/upr/input_wrapper.rb', line 99

def read(*args)
  rv = input.read(*args)
  rv.nil? || rv.size == 0 ? _finish : _incr(rv.size)
  rv
end

#rewindObject



88
89
90
91
# File 'lib/upr/input_wrapper.rb', line 88

def rewind
  self.pos = 0
  input.rewind
end

#sizeObject



77
78
79
80
81
82
83
84
85
86
# File 'lib/upr/input_wrapper.rb', line 77

def size
  rv = input.size

  # we had an unknown length and just had to read in everything to get it
  if content_length.nil?
    _incr(rv - seen)
    _finish
  end
  rv
end