From 0bb26e6c31c7c3ab2800ab64be7ec3055533de22 Mon Sep 17 00:00:00 2001 From: Jeremy Woertink Date: Sun, 24 Mar 2024 14:02:57 -0700 Subject: [PATCH] optimize route building when there's query params. Fixes #1831 --- src/lucky/routable.cr | 94 ++++++++++++++++++++++++++++--------------- 1 file changed, 61 insertions(+), 33 deletions(-) diff --git a/src/lucky/routable.cr b/src/lucky/routable.cr index 728a0d142..626939559 100644 --- a/src/lucky/routable.cr +++ b/src/lucky/routable.cr @@ -282,29 +282,37 @@ module Lucky::Routable {% end %} anchor : String? = nil ) : Lucky::RouteHelper - path = path_from_parts( - {% for param in path_params %} - {{ param.gsub(/:/, "").id }}, - {% end %} - {% for param in optional_path_params %} - {{ param.gsub(/^\?:/, "").id }}, + path = String.build do |io| + path_from_parts( + io, + {% for param in path_params %} + {{ param.gsub(/:/, "").id }}, + {% end %} + {% for param in optional_path_params %} + {{ param.gsub(/^\?:/, "").id }}, + {% end %} + ) + query_params = {} of String => String + {% for param in PARAM_DECLARATIONS %} + # add query param if given and not nil + query_params["{{ param.var }}"] = {{ param.var }}.to_s unless {{ param.var }}.nil? {% end %} - ) - query_params = {} of String => String - {% for param in PARAM_DECLARATIONS %} - # add query param if given and not nil - query_params["{{ param.var }}"] = {{ param.var }}.to_s unless {{ param.var }}.nil? - {% end %} - unless query_params.empty? - path += "?#{HTTP::Params.encode(query_params)}" - end + unless query_params.empty? + io << '?' + {% if compare_versions(Crystal::VERSION, "1.10.0") < 0 %} + io << HTTP::Params.encode(query_params) + {% else %} + HTTP::Params.encode(io, query_params) + {% end %} + end - anchor.try do |value| - path += "#" - path += URI.encode_www_form(value) + anchor.try do |value| + io << '#' + URI.encode_www_form(value, io) + end end - Lucky::RouteHelper.new {{ method }}, path + Lucky::RouteHelper.new({{ method }}, path.presence || "/") end def self.with( @@ -338,6 +346,31 @@ module Lucky::Routable \{% end %} end + private def self.path_from_parts( + io : IO, + {% for param in path_params %} + {{ param.gsub(/:/, "").id }}, + {% end %} + {% for param in optional_path_params %} + {{ param.gsub(/^\?:/, "").id }}, + {% end %} + ) : Nil + {% for part in path_parts %} + {% if part.starts_with?("?:") %} + if {{ part.gsub(/^\?:/, "").id }} + io << '/' + URI.encode_www_form({{ part.gsub(/^\?:/, "").id }}.to_param, io) + end + {% elsif part.starts_with?(':') %} + io << '/' + URI.encode_www_form({{ part.gsub(/:/, "").id }}.to_param, io) + {% else %} + io << '/' + URI.encode_www_form({{ part }}, io) + {% end %} + {% end %} + end + private def self.path_from_parts( {% for param in path_params %} {{ param.gsub(/:/, "").id }}, @@ -346,21 +379,16 @@ module Lucky::Routable {{ param.gsub(/^\?:/, "").id }}, {% end %} ) : String - path = String.build do |path| - {% for part in path_parts %} - {% if part.starts_with?("?:") %} - if {{ part.gsub(/^\?:/, "").id }} - path << '/' - URI.encode_www_form({{ part.gsub(/^\?:/, "").id }}.to_param, path) - end - {% elsif part.starts_with?(':') %} - path << '/' - URI.encode_www_form({{ part.gsub(/:/, "").id }}.to_param, path) - {% else %} - path << '/' - URI.encode_www_form({{ part }}, path) + path = String.build do |io| + path_from_parts( + io, + {% for param in path_params %} + {{ param.gsub(/:/, "").id }}, {% end %} - {% end %} + {% for param in optional_path_params %} + {{ param.gsub(/^\?:/, "").id }}, + {% end %} + ) end path.presence || "/"