require 'rapid_runty/router/route_parser' require 'rapid_runty/router/route' require 'rapid_runty/router/matcher' module RapidRunty module Router ## # This class registers all routes defined by the draw method. class Routes < Array def add(*args) self << RapidRunty::Router::Route.new(*args) end ## # Match incoming paths from env to existing routes # # @param verb [String] HTTP verb extracted from Rack env # @param path [String] the url path extracted from Rack env # # @return A RapidRunty::Router::Route Instance: # # Example: # RapidRunty::Router::Routes.find_route("GET", "/foo/4") => #"foo", "action"=>"bar"}, path="/foo/:id", placeholders=[], verb=:get def find_route(verb, path) return nil if empty? verb = verb.to_s.downcase.strip.to_sym routes = select { |route| route.verb == verb } urls = select_routes_by_verb(verb, routes) find_matching_route(path, urls, routes) end def select_routes_by_verb(_verb, routes) urls = routes.map { |route| { url: route.path }.merge route.options } urls end def find_matching_route(path, urls, routes) url, placeholders, controller_action = RapidRunty::Router::Matcher.new.match(path, urls) return nil if url.nil? route = routes.detect { |router| router.path == url }.clone route.placeholders = placeholders route.options = controller_action route end ## # Provides familiar DSL to defining routes # # Example: # DemoApplication.routes.draw do # root "demo#index" # get "/demo", to: "demo#index" # get "/demo/new", to: "demo#new" # get "/demo/:id", to: "demo#show" # get "/demo/:id/edit", to: "demo#edit" # post "/demo/", to: "demo#create" # put "/demo/:id", to: "demo#update" # patch "/demo/:id", to: "demo#update" # delete "/demo/:id", to: "demo#destroy" # end def draw(&block) instance_eval(&block) end def root(controller) get '/', to: controller end %w(get post put patch delete).each do |method_name| define_method(method_name) do |path, options| add(method_name.to_sym, path, options) end end end end end