require 'fuzzy_hash' class Usher class Node class Response < Struct.new(:path, :params, :remaining_path, :matched_path) def partial_match? !remaining_path.nil? end def params_as_hash params.inject({}){|hash, pair| hash[pair.first] = pair.last; hash} end def destination path && path.route.destination end end attr_reader :normal, :greedy, :request attr_accessor :terminates, :request_method_type, :parent, :value, :request_methods def initialize(parent, value) @parent = parent @value = value @request = nil @normal = nil @greedy = nil @request_method_type = nil end def activate_normal! @normal ||= Hash.new end def activate_greedy! @greedy ||= Hash.new end def activate_request! @request ||= Hash.new end def upgrade_normal! @normal = FuzzyHash.new(@normal) end def upgrade_greedy! @greedy = FuzzyHash.new(@greedy) end def upgrade_request! @request = FuzzyHash.new(@request) end def depth @depth ||= @parent.is_a?(Node) ? @parent.depth + 1 : 0 end def greedy? @greedy end def self.root(route_set, request_methods) root = self.new(route_set, nil) root.request_methods = request_methods root end def terminates? @terminates && @terminates.route.recognizable? end def pp $stdout << " " * depth $stdout << "#{terminates? ? '* ' : ''}#{depth}: #{value.inspect}\n" normal.each do |k,v| $stdout << " " * (depth + 1) $stdout << ". #{k.inspect} ==> \n" v.pp end if normal greedy.each do |k,v| $stdout << " " * (depth + 1) $stdout << "g #{k.inspect} ==> \n" v.pp end if greedy request.each do |k,v| $stdout << " " * (depth + 1) $stdout << "r #{k.inspect} ==> \n" v.pp end if request end def add(route) route.paths.each do |path| set_path_with_destination(path) end end def delete(route) route.paths.each do |path| set_path_with_destination(path, nil) end end def unique_routes(node = self, routes = []) routes << node.terminates.route if node.terminates node.normal.values.each do |v| unique_routes(v, routes) end if node.normal node.greedy.values.each do |v| unique_routes(v, routes) end if node.greedy node.request.values.each do |v| unique_routes(v, routes) end if node.request routes.uniq! routes end def find(usher, request_object, original_path, path, params = [], position = 0) if terminates? && (path.empty? || terminates.route.partial_match? || (usher.ignore_trailing_delimiters? && path.all?{|p| usher.delimiters.include?(p)})) terminates.route.partial_match? ? Response.new(terminates, terminates.convert_params_array(params), original_path[position, original_path.size], original_path[0, position]) : Response.new(terminates, terminates.convert_params_array(params), nil, original_path) elsif !path.empty? and greedy and match_with_result_output = greedy.match_with_result(whole_path = original_path[position, original_path.size]) next_path, matched_part = match_with_result_output position += matched_part.size whole_path.slice!(0, matched_part.size) params << matched_part if next_path.value.is_a?(Route::Variable) next_path.find(usher, request_object, original_path, whole_path.empty? ? whole_path : usher.splitter.split(whole_path), params, position) elsif !path.empty? and normal and next_part = normal[path.first] || normal[nil] part = path.shift position += part.size case next_part.value when String # do nothing when Route::Variable::Single # get the variable var = next_part.value # do a validity check var.valid!(part) # because its a variable, we need to add it to the params array params << part until path.empty? || (var.look_ahead === path.first) # variables have a look ahead notion, next_path_part = path.shift # and until they are satified, position += next_path_part.size # keep appending to the value in params params.last << next_path_part end if var.look_ahead && usher.delimiters.size > 1 when Route::Variable::Glob params << [] while true if (next_part.value.look_ahead === part || (!usher.delimiters.unescaped.include?(part) && next_part.value.regex_matcher && !next_part.value.regex_matcher.match(part))) path.unshift(part) position -= part.size if usher.delimiters.unescaped.include?(next_part.parent.value) path.unshift(next_part.parent.value) position -= next_part.parent.value.size end break elsif !usher.delimiters.unescaped.include?(part) next_part.value.valid!(part) params.last << part end if path.empty? break else part = path.shift end end end next_part.find(usher, request_object, original_path, path, params, position) elsif request_method_type return_value = if (specific_node = request[request_object.send(request_method_type)] and ret = specific_node.find(usher, request_object, original_path, path.dup, params.dup, position)) usher.priority_lookups? ? [ret] : ret end if usher.priority_lookups? || return_value.nil? and general_node = request[nil] and ret = general_node.find(usher, request_object, original_path, path.dup, params.dup, position) return_value = usher.priority_lookups? && return_value ? [return_value, ret] : ret end unless usher.priority_lookups? return_value else return_value = Array(return_value).flatten.compact return_value.sort!{|r1, r2| r1.path.route.priority <=> r2.path.route.priority} return_value.last end else nil end end private def set_path_with_destination(path, destination = path) nodes = [path.parts.inject(self){ |node, key| process_path_part(node, key) }] nodes = process_request_parts(nodes, request_methods_for_path(path)) if request_methods nodes.each do |node| while node.request_method_type node = (node.request[nil] ||= Node.new(node, Route::RequestMethod.new(node.request_method_type, nil))) end node.terminates = destination end end def request_method_index(type) request_methods.index(type) end def process_request_parts(nodes, parts) while parts.any?{ |p| !p.trivial? } key = parts.shift next if key.trivial? nodes.map! do |node| node.activate_request! if node.request_method_type.nil? node.request_method_type = key.type node.upgrade_request! if key.value.is_a?(Regexp) Array(key.value).map{|k| node.request[k] ||= Node.new(node, key) } else case request_method_index(node.request_method_type) <=> request_method_index(key.type) when -1 parts.unshift(key) Array(key.value).map{|k| node.request[nil] ||= Node.new(node, Route::RequestMethod.new(node.request_method_type, nil)) } when 0 node.upgrade_request! if key.value.is_a?(Regexp) Array(key.value).map{|k| node.request[k] ||= Node.new(node, key) } when 1 previous_node = node.parent current_node_entry_key = nil current_node_entry_lookup = nil [previous_node.normal, previous_node.greedy, previous_node.request].compact.each do |l| current_node_entry_key = l.each{|k,v| break k if node == v} current_node_entry_lookup = l and break if current_node_entry_key end current_node_entry_lookup.respond_to?(:delete_value) ? current_node_entry_lookup.delete_value(node) : current_node_entry_lookup.delete_if{|k,v| v == node} new_node = Node.new(previous_node, Route::RequestMethod.new(key.type, nil)) new_node.activate_request! new_node.request_method_type = key.type current_node_entry_lookup[current_node_entry_key] = new_node node.parent = new_node new_node.request[nil] = node parts.unshift(key) new_node end end end nodes.flatten! end nodes end def process_path_part(node, key) case key when Route::Variable::Greedy node.activate_greedy! if key.regex_matcher node.upgrade_greedy! node.greedy[key.regex_matcher] ||= Node.new(node, key) else node.greedy[nil] ||= Node.new(node, key) end when Route::Variable node.activate_normal! if key.regex_matcher node.upgrade_normal! node.normal[key.regex_matcher] ||= Node.new(node, key) else node.normal[nil] ||= Node.new(node, key) end when Route::Static::Greedy node.activate_greedy! node.upgrade_greedy! node.greedy[key] ||= Node.new(node, key) else node.activate_normal! node.upgrade_normal! if key.is_a?(Regexp) node.normal[key] ||= Node.new(node, key) end end def request_methods_for_path(path) request_methods.collect do |type| Route::RequestMethod.new(type, path.route.conditions && path.route.conditions[type]) end end end end