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.
- data/README.rdoc +12 -4
- data/lib/rack/mount/analysis/histogram.rb +55 -6
- data/lib/rack/mount/analysis/splitting.rb +103 -89
- data/lib/rack/mount/code_generation.rb +120 -0
- data/lib/rack/mount/generatable_regexp.rb +95 -48
- data/lib/rack/mount/multimap.rb +84 -41
- data/lib/rack/mount/prefix.rb +13 -8
- data/lib/rack/mount/regexp_with_named_groups.rb +27 -7
- data/lib/rack/mount/route.rb +75 -18
- data/lib/rack/mount/route_set.rb +308 -22
- data/lib/rack/mount/strexp/parser.rb +160 -0
- data/lib/rack/mount/strexp/tokenizer.rb +83 -0
- data/lib/rack/mount/strexp.rb +54 -79
- data/lib/rack/mount/utils.rb +65 -174
- data/lib/rack/mount/vendor/regin/regin/alternation.rb +40 -0
- data/lib/rack/mount/vendor/regin/regin/anchor.rb +4 -0
- data/lib/rack/mount/vendor/regin/regin/atom.rb +54 -0
- data/lib/rack/mount/vendor/regin/regin/character.rb +51 -0
- data/lib/rack/mount/vendor/regin/regin/character_class.rb +50 -0
- data/lib/rack/mount/vendor/regin/regin/collection.rb +77 -0
- data/lib/rack/mount/vendor/regin/regin/expression.rb +126 -0
- data/lib/rack/mount/vendor/regin/regin/group.rb +90 -0
- data/lib/rack/mount/vendor/regin/regin/options.rb +55 -0
- data/lib/rack/mount/vendor/regin/regin/parser.rb +546 -0
- data/lib/rack/mount/vendor/regin/regin/tokenizer.rb +255 -0
- data/lib/rack/mount/vendor/regin/regin/version.rb +3 -0
- data/lib/rack/mount/vendor/regin/regin.rb +75 -0
- data/lib/rack/mount/version.rb +3 -0
- data/lib/rack/mount.rb +13 -17
- metadata +88 -35
- data/lib/rack/mount/analysis/frequency.rb +0 -51
- data/lib/rack/mount/const.rb +0 -45
- data/lib/rack/mount/exceptions.rb +0 -3
- data/lib/rack/mount/generation/route.rb +0 -57
- data/lib/rack/mount/generation/route_set.rb +0 -163
- data/lib/rack/mount/meta_method.rb +0 -104
- data/lib/rack/mount/mixover.rb +0 -47
- data/lib/rack/mount/recognition/code_generation.rb +0 -99
- data/lib/rack/mount/recognition/route.rb +0 -59
- data/lib/rack/mount/recognition/route_set.rb +0 -88
- data/lib/rack/mount/vendor/multimap/multimap.rb +0 -466
- data/lib/rack/mount/vendor/multimap/multiset.rb +0 -153
- 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
|
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#
|
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
|
-
#
|
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, :
|
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
|
17
|
-
|
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
|
-
|
4
|
-
|
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] ==
|
12
|
-
keys <<
|
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[
|
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}[:#{
|
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
|
-
@
|
27
|
-
|
35
|
+
@raw_keys = []
|
36
|
+
@key_frequency = Analysis::Histogram.new
|
37
|
+
self
|
28
38
|
end
|
29
39
|
|
30
40
|
def <<(key)
|
31
|
-
|
32
|
-
|
33
|
-
|
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
|
38
|
-
@
|
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
|
-
|
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
|
54
|
-
|
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
|
-
|
83
|
-
|
84
|
-
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
92
|
-
|
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
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
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.
|
106
|
-
|
107
|
-
|
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
|
-
|
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
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
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
|
-
|
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
|