rack-mount 0.0.1 → 0.8.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. data/README.rdoc +12 -4
  2. data/lib/rack/mount/analysis/histogram.rb +55 -6
  3. data/lib/rack/mount/analysis/splitting.rb +103 -89
  4. data/lib/rack/mount/code_generation.rb +120 -0
  5. data/lib/rack/mount/generatable_regexp.rb +95 -48
  6. data/lib/rack/mount/multimap.rb +84 -41
  7. data/lib/rack/mount/prefix.rb +13 -8
  8. data/lib/rack/mount/regexp_with_named_groups.rb +27 -7
  9. data/lib/rack/mount/route.rb +75 -18
  10. data/lib/rack/mount/route_set.rb +308 -22
  11. data/lib/rack/mount/strexp/parser.rb +160 -0
  12. data/lib/rack/mount/strexp/tokenizer.rb +83 -0
  13. data/lib/rack/mount/strexp.rb +54 -79
  14. data/lib/rack/mount/utils.rb +65 -174
  15. data/lib/rack/mount/vendor/regin/regin/alternation.rb +40 -0
  16. data/lib/rack/mount/vendor/regin/regin/anchor.rb +4 -0
  17. data/lib/rack/mount/vendor/regin/regin/atom.rb +54 -0
  18. data/lib/rack/mount/vendor/regin/regin/character.rb +51 -0
  19. data/lib/rack/mount/vendor/regin/regin/character_class.rb +50 -0
  20. data/lib/rack/mount/vendor/regin/regin/collection.rb +77 -0
  21. data/lib/rack/mount/vendor/regin/regin/expression.rb +126 -0
  22. data/lib/rack/mount/vendor/regin/regin/group.rb +90 -0
  23. data/lib/rack/mount/vendor/regin/regin/options.rb +55 -0
  24. data/lib/rack/mount/vendor/regin/regin/parser.rb +546 -0
  25. data/lib/rack/mount/vendor/regin/regin/tokenizer.rb +255 -0
  26. data/lib/rack/mount/vendor/regin/regin/version.rb +3 -0
  27. data/lib/rack/mount/vendor/regin/regin.rb +75 -0
  28. data/lib/rack/mount/version.rb +3 -0
  29. data/lib/rack/mount.rb +13 -17
  30. metadata +88 -35
  31. data/lib/rack/mount/analysis/frequency.rb +0 -51
  32. data/lib/rack/mount/const.rb +0 -45
  33. data/lib/rack/mount/exceptions.rb +0 -3
  34. data/lib/rack/mount/generation/route.rb +0 -57
  35. data/lib/rack/mount/generation/route_set.rb +0 -163
  36. data/lib/rack/mount/meta_method.rb +0 -104
  37. data/lib/rack/mount/mixover.rb +0 -47
  38. data/lib/rack/mount/recognition/code_generation.rb +0 -99
  39. data/lib/rack/mount/recognition/route.rb +0 -59
  40. data/lib/rack/mount/recognition/route_set.rb +0 -88
  41. data/lib/rack/mount/vendor/multimap/multimap.rb +0 -466
  42. data/lib/rack/mount/vendor/multimap/multiset.rb +0 -153
  43. data/lib/rack/mount/vendor/multimap/nested_multimap.rb +0 -156
data/README.rdoc CHANGED
@@ -2,7 +2,7 @@
2
2
 
3
3
  A stackable dynamic tree based Rack router.
4
4
 
5
- Rack::Mount supports Rack's Cascade style of trying several routes until it finds one that is not a 404. This allows multiple routes to be nested or stacked on top of each other. Since the application endpoint can trigger the router to continue matching, middleware can be used to add arbitrary conditions to any route. This allows you to route based on other request attributes, session information, or even data dynamically pulled from a database.
5
+ Rack::Mount supports Rack's +X-Cascade+ convention to continue trying routes if the response returns +pass+. This allows multiple routes to be nested or stacked on top of each other. Since the application endpoint can trigger the router to continue matching, middleware can be used to add arbitrary conditions to any route. This allows you to route based on other request attributes, session information, or even data dynamically pulled from a database.
6
6
 
7
7
  === Usage
8
8
 
@@ -12,17 +12,25 @@ The API is extremely minimal and only 3 methods are exposed as the public API.
12
12
 
13
13
  <tt>Rack::Mount::RouteSet#add_route</tt>:: builder method for adding routes to the set
14
14
  <tt>Rack::Mount::RouteSet#call</tt>:: Rack compatible recognition and dispatching method
15
- <tt>Rack::Mount::RouteSet#url</tt>:: generates path from identifiers or significant keys
15
+ <tt>Rack::Mount::RouteSet#generate</tt>:: generates a route condition from identifiers or significant keys
16
16
 
17
17
  === Example
18
18
 
19
19
  require 'rack/mount'
20
+
20
21
  Routes = Rack::Mount::RouteSet.new do |set|
21
22
  # add_route takes a rack application and conditions to match with
22
- # conditions may be strings or regexps
23
+ #
24
+ # valid conditions methods are any method on Rack::Request
25
+ # the values to match against may be strings or regexps
26
+ #
23
27
  # See Rack::Mount::RouteSet#add_route for more options.
24
- set.add_route FooApp, :method => 'get' :path => %{/foo}
28
+ set.add_route FooApp, { :request_method => 'GET', :path_info => %r{^/foo$} }, {}, :foo
25
29
  end
26
30
 
27
31
  # The route set itself is a simple rack app you mount
28
32
  run Routes
33
+
34
+
35
+ # generate path for route named "foo"
36
+ Routes.generate(:path_info, :foo) #=> "/foo"
@@ -6,20 +6,69 @@ module Rack::Mount
6
6
  def initialize
7
7
  @count = 0
8
8
  super(0)
9
+ expire_caches!
9
10
  end
10
11
 
11
12
  def <<(value)
12
13
  @count += 1
13
14
  self[value] += 1 if value
15
+ expire_caches!
16
+ self
14
17
  end
15
18
 
16
- def select_upper
17
- values = sort_by { |_, value| value }
18
- values.reverse!
19
- values = values.select { |_, value| value >= count / size }
20
- values.map! { |key, _| key }
21
- values
19
+ def sorted_by_frequency
20
+ sort_by { |_, value| value }.reverse!
22
21
  end
22
+
23
+ def max
24
+ @max ||= values.max || 0
25
+ end
26
+
27
+ def min
28
+ @min ||= values.min || 0
29
+ end
30
+
31
+ def mean
32
+ @mean ||= calculate_mean
33
+ end
34
+
35
+ def standard_deviation
36
+ @standard_deviation ||= calculate_standard_deviation
37
+ end
38
+
39
+ def upper_quartile_limit
40
+ @upper_quartile_limit ||= calculate_upper_quartile_limit
41
+ end
42
+
43
+ def keys_in_upper_quartile
44
+ @keys_in_upper_quartile ||= compute_keys_in_upper_quartile
45
+ end
46
+
47
+ private
48
+ def calculate_mean
49
+ count / size
50
+ end
51
+
52
+ def calculate_variance
53
+ values.inject(0) { |sum, e| sum + (e - mean) ** 2 } / count.to_f
54
+ end
55
+
56
+ def calculate_standard_deviation
57
+ Math.sqrt(calculate_variance)
58
+ end
59
+
60
+ def calculate_upper_quartile_limit
61
+ mean + standard_deviation
62
+ end
63
+
64
+ def compute_keys_in_upper_quartile
65
+ sorted_by_frequency.select { |_, value| value >= upper_quartile_limit }.map! { |key, _| key }
66
+ end
67
+
68
+ def expire_caches!
69
+ @max = @min = @mean = @standard_deviation = nil
70
+ @keys_in_upper_quartile = nil
71
+ end
23
72
  end
24
73
  end
25
74
  end
@@ -1,41 +1,69 @@
1
+ require 'rack/mount/utils'
2
+
1
3
  module Rack::Mount
2
4
  module Analysis
3
- module Splitting
4
- class Key < Array
5
- def initialize(method, index, separators)
6
- replace([method, index, separators])
7
- end
5
+ class Splitting
6
+ NULL = "\0"
8
7
 
8
+ class Key < Struct.new(:method, :index, :separators)
9
9
  def self.split(value, separator_pattern)
10
10
  keys = value.split(separator_pattern)
11
- keys.shift if keys[0] == Const::EMPTY_STRING
12
- keys << Const::NULL
11
+ keys.shift if keys[0] == ''
12
+ keys << NULL
13
13
  keys
14
14
  end
15
15
 
16
16
  def call(cache, obj)
17
- (cache[self[0]] ||= self.class.split(obj.send(self[0]), self[2]))[self[1]]
17
+ (cache[method] ||= self.class.split(obj.send(method), separators))[index]
18
18
  end
19
19
 
20
20
  def call_source(cache, obj)
21
- "(#{cache}[:#{self[0]}] ||= Analysis::Splitting::Key.split(#{obj}.#{self[0]}, #{self[2].inspect}))[#{self[1]}]"
21
+ "(#{cache}[:#{method}] ||= Analysis::Splitting::Key.split(#{obj}.#{method}, #{separators.inspect}))[#{index}]"
22
+ end
23
+
24
+ def inspect
25
+ "#{method}[#{index}].split(#{separators.inspect})"
22
26
  end
23
27
  end
24
28
 
29
+ def initialize(*keys)
30
+ clear
31
+ keys.each { |key| self << key }
32
+ end
33
+
25
34
  def clear
26
- @boundaries = {}
27
- super
35
+ @raw_keys = []
36
+ @key_frequency = Analysis::Histogram.new
37
+ self
28
38
  end
29
39
 
30
40
  def <<(key)
31
- super
32
- key.each_pair do |k, v|
33
- analyze_capture_boundaries(v, @boundaries[k] ||= Histogram.new)
41
+ raise ArgumentError unless key.is_a?(Hash)
42
+ @raw_keys << key
43
+ nil
44
+ end
45
+
46
+ def possible_keys
47
+ @possible_keys ||= begin
48
+ @raw_keys.map do |key|
49
+ key.inject({}) { |requirements, (method, requirement)|
50
+ process_key(requirements, method, requirement)
51
+ requirements
52
+ }
53
+ end
34
54
  end
35
55
  end
36
56
 
37
- def separators(key)
38
- @boundaries[key].select_upper
57
+ def report
58
+ @report ||= begin
59
+ possible_keys.each { |keys| keys.each_pair { |key, _| @key_frequency << key } }
60
+ return [] if @key_frequency.count <= 1
61
+ @key_frequency.keys_in_upper_quartile
62
+ end
63
+ end
64
+
65
+ def expire!
66
+ @possible_keys = @report = nil
39
67
  end
40
68
 
41
69
  def process_key(requirements, method, requirement)
@@ -45,77 +73,71 @@ module Rack::Mount
45
73
  requirements[Key.new(method, index, Regexp.union(*separators))] = value
46
74
  end
47
75
  else
48
- super
76
+ if requirement.is_a?(Regexp)
77
+ expression = Utils.parse_regexp(requirement)
78
+
79
+ if expression.is_a?(Regin::Expression) && expression.anchored_to_line?
80
+ expression = Regin::Expression.new(expression.reject { |e| e.is_a?(Regin::Anchor) })
81
+ return requirements[method] = expression.to_s if expression.literal?
82
+ end
83
+ end
84
+
85
+ requirements[method] = requirement
49
86
  end
50
87
  end
51
88
 
52
89
  private
53
- def analyze_capture_boundaries(regexp, boundaries) #:nodoc:
54
- return boundaries unless regexp.is_a?(Regexp)
55
-
56
- parts = Utils.extract_regexp_parts(regexp) rescue []
57
- parts.each_with_index do |part, index|
58
- break if part == Const::NULL
59
-
60
- if index > 0
61
- previous = parts[index-1]
62
- if previous.is_a?(Utils::Capture)
63
- previous = Utils.extract_static_regexp(previous.last_part) rescue nil
64
- end
65
- boundaries << previous[-1..-1] if previous.is_a?(String)
66
- end
67
-
68
- if index < parts.length
69
- following = parts[index+1]
70
- if following.is_a?(Utils::Capture)
71
- following = Utils.extract_static_regexp(following.first_part) rescue nil
72
- end
73
- if following.is_a?(String) && following != Const::NULL
74
- boundaries << following[0..0] == '\\' ? following[1..1] : following[0..0]
75
- end
76
- end
77
- end
78
- boundaries
90
+ def separators(key)
91
+ key == :path_info ? ["/", "."] : []
79
92
  end
80
93
 
81
94
  def generate_split_keys(regexp, separators) #:nodoc:
82
- escaped_separators = separators.map { |s| Regexp.escape(s) }
83
- separators_regexp = Regexp.union(*escaped_separators)
84
- segments, previous = [], nil
85
- regexp_options = regexp.options
86
-
87
- begin
88
- Utils.extract_regexp_parts(regexp).each do |part|
89
- if part.respond_to?(:optional?) && part.optional?
90
- if escaped_separators.include?(part.first)
91
- append_to_segments!(segments, previous, separators, regexp_options)
92
- end
93
-
94
- raise ArgumentError
95
+ segments = []
96
+ buf = nil
97
+ parts = Utils.parse_regexp(regexp)
98
+ parts.each_with_index do |part, index|
99
+ case part
100
+ when Regin::Anchor
101
+ if part.value == '$' || part.value == '\Z'
102
+ segments << join_buffer(buf, regexp) if buf
103
+ segments << NULL
104
+ buf = nil
105
+ break
95
106
  end
96
-
97
- append_to_segments!(segments, previous, separators, regexp_options)
98
- previous = nil
99
-
100
- if part == Const::NULL
101
- segments << Const::NULL
102
- raise ArgumentError
107
+ when Regin::CharacterClass
108
+ break if separators.any? { |s| part.include?(s) }
109
+ buf = nil
110
+ segments << part.to_regexp(true)
111
+ when Regin::Character
112
+ if separators.any? { |s| part.include?(s) }
113
+ segments << join_buffer(buf, regexp) if buf
114
+ peek = parts[index+1]
115
+ if peek.is_a?(Regin::Character) && separators.include?(peek.value)
116
+ segments << ''
117
+ end
118
+ buf = nil
119
+ else
120
+ buf ||= Regin::Expression.new([])
121
+ buf += [part]
103
122
  end
104
-
105
- if part.is_a?(Utils::Capture)
106
- source = part.map { |p| p.to_s }.join
107
- append_to_segments!(segments, source, separators, regexp_options)
123
+ when Regin::Group
124
+ if part.quantifier == '?'
125
+ value = part.expression.first
126
+ if separators.any? { |s| value.include?(s) }
127
+ segments << join_buffer(buf, regexp) if buf
128
+ buf = nil
129
+ end
130
+ break
131
+ elsif part.quantifier == nil
132
+ break if separators.any? { |s| part.include?(s) }
133
+ buf = nil
134
+ segments << part.to_regexp(true)
108
135
  else
109
- parts = part.split(separators_regexp)
110
- parts.shift if parts[0] == Const::EMPTY_STRING
111
- previous = parts.pop
112
- parts.each { |p| append_to_segments!(segments, p, separators, regexp_options) }
136
+ break
113
137
  end
138
+ else
139
+ break
114
140
  end
115
-
116
- append_to_segments!(segments, previous, separators, regexp_options)
117
- rescue ArgumentError
118
- # generation failed somewhere, but lets take what we can get
119
141
  end
120
142
 
121
143
  while segments.length > 0 && (segments.last.nil? || segments.last == '')
@@ -125,21 +147,13 @@ module Rack::Mount
125
147
  segments
126
148
  end
127
149
 
128
- def append_to_segments!(segments, s, separators, regexp_options) #:nodoc:
129
- return unless s
130
- separators.each do |separator|
131
- if s.gsub(/\[[^\]]+\]/, '').include?(separator)
132
- raise ArgumentError
133
- end
134
-
135
- if Regexp.compile("\\A#{s}\\Z") =~ separator
136
- raise ArgumentError
137
- end
150
+ def join_buffer(parts, regexp)
151
+ if parts.literal?
152
+ parts.to_s
153
+ else
154
+ parts.to_regexp(true)
138
155
  end
139
-
140
- static = Utils.extract_static_regexp(s, regexp_options)
141
- segments << (static.is_a?(String) ? static : static)
142
156
  end
143
- end
157
+ end
144
158
  end
145
159
  end
@@ -0,0 +1,120 @@
1
+ module Rack::Mount
2
+ module CodeGeneration #:nodoc:
3
+ def _expired_recognize(env) #:nodoc:
4
+ raise 'route set not finalized'
5
+ end
6
+
7
+ def rehash
8
+ super
9
+ optimize_recognize!
10
+ end
11
+
12
+ private
13
+ def expire!
14
+ if @optimized_recognize_defined
15
+ remove_metaclass_method :recognize
16
+
17
+ class << self
18
+ alias_method :recognize, :_expired_recognize
19
+ end
20
+
21
+ @optimized_recognize_defined = false
22
+ end
23
+
24
+ super
25
+ end
26
+
27
+ def optimize_container_iterator(container)
28
+ Utils.debug "optimizing container - size #{container.size}"
29
+
30
+ body = []
31
+
32
+ container.each_with_index { |route, i|
33
+ body << "route = self[#{i}]"
34
+ body << 'matches = {}'
35
+ body << 'params = route.defaults.dup'
36
+
37
+ conditions = []
38
+ route.conditions.each do |method, condition|
39
+ b = []
40
+ if condition.is_a?(Regexp)
41
+ b << "if m = obj.#{method}.match(#{condition.inspect})"
42
+ b << "matches[:#{method}] = m"
43
+ if (named_captures = route.named_captures[method]) && named_captures.any?
44
+ b << 'captures = m.captures'
45
+ b << 'p = nil'
46
+ b << named_captures.map { |k, j| "params[#{k.inspect}] = p if p = captures[#{j}]" }.join('; ')
47
+ end
48
+ else
49
+ b << "if m = obj.#{method} == route.conditions[:#{method}]"
50
+ end
51
+ b << 'true'
52
+ b << 'end'
53
+ conditions << "(#{b.join('; ')})"
54
+ end
55
+
56
+ body << <<-RUBY
57
+ if #{conditions.join(' && ')}
58
+ yield route, matches, params
59
+ end
60
+ RUBY
61
+ }
62
+
63
+ container.instance_eval(<<-RUBY, __FILE__, __LINE__)
64
+ def optimized_each(obj)
65
+ #{body.join("\n")}
66
+ nil
67
+ end
68
+ RUBY
69
+ end
70
+
71
+ def optimize_recognize!
72
+ Utils.debug "optimizing recognize"
73
+
74
+ uses_cache = false
75
+
76
+ keys = @recognition_keys.map { |key|
77
+ if key.respond_to?(:call_source)
78
+ uses_cache = true
79
+ key.call_source(:cache, :obj)
80
+ else
81
+ "obj.#{key}"
82
+ end
83
+ }.join(', ')
84
+
85
+ @optimized_recognize_defined = true
86
+
87
+ remove_metaclass_method :recognize
88
+
89
+ instance_eval(<<-RUBY, __FILE__, __LINE__)
90
+ def recognize(obj)
91
+ #{"cache = {}" if uses_cache}
92
+ container = @recognition_graph[#{keys}]
93
+ optimize_container_iterator(container) unless container.respond_to?(:optimized_each)
94
+
95
+ if block_given?
96
+ container.optimized_each(obj) do |route, matches, params|
97
+ yield route, matches, params
98
+ end
99
+ else
100
+ container.optimized_each(obj) do |route, matches, params|
101
+ return route, matches, params
102
+ end
103
+ end
104
+
105
+ nil
106
+ end
107
+ RUBY
108
+ end
109
+
110
+ # method_defined? can't distinguish between instance
111
+ # and meta methods. So we have to rescue if the method
112
+ # has not been defined in the metaclass yet.
113
+ def remove_metaclass_method(symbol)
114
+ metaclass = class << self; self; end
115
+ Utils.silence_debug { metaclass.send(:remove_method, symbol) }
116
+ rescue NameError
117
+ nil
118
+ end
119
+ end
120
+ end