diff --git a/bridgetown-core/lib/bridgetown-core/concerns/site/ssr.rb b/bridgetown-core/lib/bridgetown-core/concerns/site/ssr.rb index 4567dc925..0aaf987ca 100644 --- a/bridgetown-core/lib/bridgetown-core/concerns/site/ssr.rb +++ b/bridgetown-core/lib/bridgetown-core/concerns/site/ssr.rb @@ -28,6 +28,7 @@ def ssr? def enable_ssr Bridgetown.logger.info "SSR:", "enabled." + config.fast_refresh = false # SSR mode and Fast Refresh mode are mututally exclusive @ssr_enabled = true end @@ -46,6 +47,7 @@ def ssr_setup(&block) end def ssr_first_read + # TODO: this shouldn't be running twice, right?! Bridgetown::Hooks.trigger :site, :pre_read, self defaults_reader.tap do |d| d.path_defaults.clear diff --git a/bridgetown-core/lib/bridgetown-core/rack/boot.rb b/bridgetown-core/lib/bridgetown-core/rack/boot.rb index f99ad186b..9e934c5a8 100644 --- a/bridgetown-core/lib/bridgetown-core/rack/boot.rb +++ b/bridgetown-core/lib/bridgetown-core/rack/boot.rb @@ -3,13 +3,11 @@ require "zeitwerk" require "roda" require "json" -require "roda/plugins/public" Bridgetown::Current.preloaded_configuration ||= Bridgetown.configuration require_relative "logger" require_relative "routes" -require_relative "static_indexes" module Bridgetown module Rack diff --git a/bridgetown-core/lib/bridgetown-core/rack/routes.rb b/bridgetown-core/lib/bridgetown-core/rack/routes.rb index 8002248ef..4f53346c5 100644 --- a/bridgetown-core/lib/bridgetown-core/rack/routes.rb +++ b/bridgetown-core/lib/bridgetown-core/rack/routes.rb @@ -112,36 +112,11 @@ def merge(roda_app) new(roda_app).handle_routes end - # Start the Roda app request cycle. There are two different code paths - # depending on if there's a site `base_path` configured + # Set up live reload if allowed, then run through all the Routes blocks. # # @param roda_app [Roda] # @return [void] - def start!(roda_app) - if Bridgetown::Current.preloaded_configuration.base_path == "/" - load_all_routes roda_app - return - end - - # Support custom base_path configurations - roda_app.request.on( - Bridgetown::Current.preloaded_configuration.base_path.delete_prefix("/") - ) do - load_all_routes roda_app - end - - nil - end - - # Run the Roda public plugin first, set up live reload if allowed, then - # run through all the Routes blocks. If the file-based router plugin - # is available, kick off that request process next. - # - # @param roda_app [Roda] - # @return [void] - def load_all_routes(roda_app) - roda_app.request.public - + def load_all(roda_app) if Bridgetown.env.development? && !Bridgetown::Current.preloaded_configuration.skip_live_reload setup_live_reload roda_app diff --git a/bridgetown-core/lib/bridgetown-core/rack/static_indexes.rb b/bridgetown-core/lib/bridgetown-core/rack/static_indexes.rb deleted file mode 100644 index 276e134f1..000000000 --- a/bridgetown-core/lib/bridgetown-core/rack/static_indexes.rb +++ /dev/null @@ -1,31 +0,0 @@ -# frozen_string_literal: true - -require "roda/plugins/public" - -# TODO: extract out to a standalone Roda plugin -Roda::RodaPlugins::Public::RequestMethods.module_eval do - SPLIT = Regexp.union(*[File::SEPARATOR, File::ALT_SEPARATOR].compact) # rubocop:disable Lint/ConstantDefinitionInBlock - def public_path_segments(path) # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity - segments = [] - - path.split(SPLIT).each do |seg| - next if seg.empty? || seg == "." - - seg == ".." ? segments.pop : segments << seg - end - - path = File.join(roda_class.opts[:public_root], *segments) - unless File.file?(path) - path = File.join(path, "index.html") - if File.file?(path) - segments << "index.html" - else - segments[segments.size - 1] = "#{segments.last}.html" - end - end - - segments - rescue IndexError - nil - end -end diff --git a/bridgetown-core/lib/roda/plugins/bridgetown_server.rb b/bridgetown-core/lib/roda/plugins/bridgetown_server.rb index 8abc2e7df..900f89a60 100644 --- a/bridgetown-core/lib/roda/plugins/bridgetown_server.rb +++ b/bridgetown-core/lib/roda/plugins/bridgetown_server.rb @@ -11,6 +11,7 @@ def self.load_dependencies(app) # rubocop:disable Metrics "plugin" end + app.extend ClassMethods # we need to do this here because Roda hasn't done it yet app.plugin :initializers app.plugin :method_override app.plugin :all_verbs @@ -20,7 +21,7 @@ def self.load_dependencies(app) # rubocop:disable Metrics app.plugin :json_parser app.plugin :indifferent_params app.plugin :cookies, path: "/" - app.plugin :public, root: Bridgetown::Current.preloaded_configuration.destination + app.plugin :ssg, root: Bridgetown::Current.preloaded_configuration.destination app.plugin :not_found do output_folder = Bridgetown::Current.preloaded_configuration.destination File.read(File.join(output_folder, "404.html")) @@ -51,6 +52,7 @@ def self.load_dependencies(app) # rubocop:disable Metrics result.transform!.output end + # TODO: there may be a better way to do this, see `exception_page_css` instance method ExceptionPage.class_eval do # rubocop:disable Metrics/BlockLength def self.css <<~CSS @@ -107,8 +109,16 @@ def self.css CSS end end + end + + module ClassMethods + def root_hook(&block) + opts[:root_hook] = block + end + end - app.before do + module InstanceMethods + def initialize_bridgetown_context if self.class.opts[:bridgetown_site] # The site had previously been initialized via the bridgetown_ssr plugin Bridgetown::Current.sites[self.class.opts[:bridgetown_site].label] = @@ -117,11 +127,22 @@ def self.css end Bridgetown::Current.preloaded_configuration ||= self.class.opts[:bridgetown_preloaded_config] + end + def initialize_bridgetown_root # rubocop:todo Metrics/AbcSize request.root do - output_folder = Bridgetown::Current.preloaded_configuration.destination - File.read(File.join(output_folder, "index.html")) - rescue StandardError + hook_result = instance_exec(&self.class.opts[:root_hook]) if self.class.opts[:root_hook] + next hook_result if hook_result + + status, headers, body = self.class.opts[:ssg_server].serving( + request, File.join(self.class.opts[:ssg_root], "index.html") + ) + response_headers = response.headers + response_headers.replace(headers) + + request.halt [status, response_headers, body] + rescue StandardError => e + Bridgetown.logger.debug("Root handler error: #{e.message}") response.status = 500 "

ERROR: cannot find index.html in the output folder.

" end @@ -138,9 +159,26 @@ def cookies _previous_roda_cookies.with_indifferent_access end - # Starts up the Bridgetown routing system + # Start up the Bridgetown routing system def bridgetown - Bridgetown::Rack::Routes.start!(scope) + scope.initialize_bridgetown_context + scope.initialize_bridgetown_root + + # Run the static file server + ssg + + # There are two different code paths depending on if there's a site `base_path` configured + if Bridgetown::Current.preloaded_configuration.base_path == "/" + Bridgetown::Rack::Routes.load_all scope + return + end + + # Support custom base_path configurations + on(Bridgetown::Current.preloaded_configuration.base_path.delete_prefix("/")) do + Bridgetown::Rack::Routes.load_all scope + end + + nil end end end diff --git a/bridgetown-core/lib/roda/plugins/ssg.rb b/bridgetown-core/lib/roda/plugins/ssg.rb new file mode 100644 index 000000000..51aead908 --- /dev/null +++ b/bridgetown-core/lib/roda/plugins/ssg.rb @@ -0,0 +1,72 @@ +# frozen_string_literal: true + +require "uri" +require "rack/files" + +class Roda + module RodaPlugins + # This is a simplifed and modified variant of Roda's Public core plugin. It adds additional + # functionality so that you can host an entire static site through Roda. What's necessary for + # this to work is handling "pretty" URLs, aka: + # + # /path/to/page -> /path/to/page.html or /path/to/page/index.html + # /path/to/page/ -> /path/to/page/index.html + # + # It does not support serving compressed files, as that should ideally be handled through a + # proxy or CDN layer in your architecture. + module SSG + PARSER = URI::DEFAULT_PARSER + + def self.configure(app, opts = {}) + app.opts[:ssg_root] = app.expand_path(opts.fetch(:root, "public")) + app.opts[:ssg_server] = Rack::Files.new(app.opts[:ssg_root]) + end + + module RequestMethods + def ssg + return unless is_get? + + path = PARSER.unescape(real_remaining_path) + return if path.include?("\0") + + server = roda_class.opts[:ssg_server] + path = File.join(server.root, *segments_for_path(path)) + + return unless File.file?(path) + + status, headers, body = server.serving(self, path) + response_headers = response.headers + response_headers.replace(headers) + halt [status, response_headers, body] + end + + # TODO: this could be refactored a bit + def segments_for_path(path) # rubocop:disable Metrics/CyclomaticComplexity, Metrics/PerceivedComplexity + segments = [] + + path.split("/").each do |seg| + next if seg.empty? || seg == "." + + seg == ".." ? segments.pop : segments << seg + end + + path = File.join(roda_class.opts[:ssg_root], *segments) + unless File.file?(path) + path = File.join(path, "index.html") + if File.file?(path) + segments << "index.html" + else + segments[segments.size - 1] = "#{segments.last}.html" + end + end + + segments + rescue IndexError + nil + end + end + end + + register_plugin :ssg, SSG + end +end diff --git a/bridgetown-core/lib/site_template/.gitignore b/bridgetown-core/lib/site_template/.gitignore index 39a713e44..3841a7cdd 100644 --- a/bridgetown-core/lib/site_template/.gitignore +++ b/bridgetown-core/lib/site_template/.gitignore @@ -6,7 +6,6 @@ output # Dependency folders node_modules -vendor # Caches .sass-cache diff --git a/bridgetown-routes/lib/bridgetown-routes/manifest.rb b/bridgetown-routes/lib/bridgetown-routes/manifest.rb index c0ffc3337..46f4db64e 100644 --- a/bridgetown-routes/lib/bridgetown-routes/manifest.rb +++ b/bridgetown-routes/lib/bridgetown-routes/manifest.rb @@ -2,110 +2,111 @@ module Bridgetown module Routes - module Manifest - class << self - def generate_manifest(site) # rubocop:todo Metrics/AbcSize, Metrics/CyclomaticComplexity - return @route_manifest[site.label] if @route_manifest && Bridgetown.env.production? + class Manifest + attr_reader :site, :config, :manifest + + def initialize(site, cache_routes: Bridgetown.env.production?) + @site = site + @manifest = [] + @config = site.config.routes + @cache_routes = cache_routes + @islands_dir = File.expand_path(site.config.islands_dir, site.config.source) + end - new_manifest = [] - routable_extensions = site.config.routes.extensions.join(",") + def routable_extensions = config.extensions.join(",") - expand_source_paths_with_islands(site.config).each do |routes_dir| - routes_dir = File.expand_path(routes_dir, site.config.source) + def routes + return @manifest if !@manifest.empty? && @cache_routes - # @type [Array] - routes = Dir.glob( - routes_dir + "/**/*.{#{routable_extensions}}" - ).filter_map do |file| - if File.basename(file).start_with?("_", ".") || - File.basename(file, ".*").end_with?(".test") - next - end + @manifest = [] - file_slug, segment_keys = file_slug_and_segments(site, routes_dir, file) + # Loop through all the directories (`src/_routes`, etc) looking for route files, then + # sort them and add them to the manifest: + expand_source_paths_with_islands.each do |routes_dir| + @manifest += glob_routes(routes_dir).map do |file| + file_slug, segment_keys = file_slug_and_segments(routes_dir, file) - # generate localized file slugs - localized_file_slugs = generate_localized_file_slugs(site, file_slug) + # generate localized file slugs + localized_file_slugs = generate_localized_file_slugs(file_slug) - [file, localized_file_slugs, segment_keys] - end + [file, localized_file_slugs, segment_keys] + end.then { sort_routes! _1 } + end - new_manifest += sort_routes!(routes) - end + @manifest + end - @route_manifest ||= {} - @route_manifest[site.label] = new_manifest - end + def expand_source_paths_with_islands + # clear out any past islands folders + config.source_paths.reject! { _1.start_with?(@islands_dir) } - def sort_routes!(routes) - routes.sort! do |route_a, route_b| - # @type [String] - slug_a = route_a[1][0] - # @type [String] - slug_b = route_b[1][0] - - # @type [Integer] - weight1 = slug_a.count("/") <=> slug_b.count("/") - if weight1.zero? - slug_b.count("/:") <=> slug_a.count("/:") - else - weight1 - end - end.reverse! + Dir.glob("#{@islands_dir}/**/routes").each do |route_folder| + config.source_paths << route_folder end - def locale_for(slug, site) - possible_locale_segment = slug.split("/").first.to_sym + config.source_paths.map { File.expand_path _1, site.config.source } + end - if site.config.available_locales.include? possible_locale_segment - possible_locale_segment - else - site.config.default_locale - end + def glob_routes(dir, pattern = "**/*") + files = Dir.glob("#{dir}/#{pattern}.{#{routable_extensions}}") + files.reject! do |file| + File.basename(file, ".*").then { _1.start_with?("_", ".") || _1.end_with?(".test") } end + files + end - private - - # Add any `routes` folders that may be living within islands - def expand_source_paths_with_islands(config) - @islands_dir ||= File.expand_path(config.islands_dir, config.source) + def file_slug_and_segments(routes_dir, file) + # @type [String] + file_slug = file.delete_prefix("#{routes_dir}/").then do |f| + if routes_dir.start_with?(@islands_dir) + # convert _islands/foldername/routes/someroute.rb to foldername/someroute.rb + f = routes_dir.delete_prefix("#{@islands_dir}/").sub(%r!/routes$!, "/") + f + end + [File.dirname(f), File.basename(f, ".*")].join("/").delete_prefix("./") + end.delete_suffix("/index") + segment_keys = [] + file_slug.gsub!(%r{\[([^/]+)\]}) do |_segment| + segment_keys << Regexp.last_match(1) + ":#{Regexp.last_match(1)}" + end - # clear out any past islands folders - config.routes.source_paths.reject! { _1.start_with?(@islands_dir) } + [file_slug, segment_keys] + end - Dir.glob("#{@islands_dir}/**/routes").each do |route_folder| - config.routes.source_paths << route_folder + def generate_localized_file_slugs(file_slug) + site.config.available_locales.map do |locale| + if locale == site.config.default_locale && !site.config.prefix_default_locale + file_slug + else + "#{locale}/#{file_slug}" end - - config.routes.source_paths end + end - def file_slug_and_segments(_site, routes_dir, file) + def sort_routes!(routes) + routes.sort! do |route_a, route_b| + # @type [String] + slug_a = route_a[1][0] # @type [String] - file_slug = file.delete_prefix("#{routes_dir}/").then do |f| - if routes_dir.start_with?(@islands_dir) - # convert _islands/foldername/routes/someroute.rb to foldername/someroute.rb - f = routes_dir.delete_prefix("#{@islands_dir}/").sub(%r!/routes$!, "/") + f - end - [File.dirname(f), File.basename(f, ".*")].join("/").delete_prefix("./") - end.delete_suffix("/index") - segment_keys = [] - file_slug.gsub!(%r{\[([^/]+)\]}) do |_segment| - segment_keys << Regexp.last_match(1) - ":#{Regexp.last_match(1)}" + slug_b = route_b[1][0] + + # @type [Integer] + weight1 = slug_a.count("/") <=> slug_b.count("/") + if weight1.zero? + slug_b.count("/:") <=> slug_a.count("/:") + else + weight1 end + end.reverse! + end - [file_slug, segment_keys] - end + def locale_for(slug) + possible_locale_segment = slug.split("/").first.to_sym - def generate_localized_file_slugs(site, file_slug) - site.config.available_locales.map do |locale| - if locale == site.config.default_locale && !site.config.prefix_default_locale - file_slug - else - "#{locale}/#{file_slug}" - end - end + if site.config.available_locales.include? possible_locale_segment + possible_locale_segment + else + site.config.default_locale end end end diff --git a/bridgetown-routes/lib/bridgetown-routes/manifest_router.rb b/bridgetown-routes/lib/bridgetown-routes/manifest_router.rb index aa823722a..880696216 100644 --- a/bridgetown-routes/lib/bridgetown-routes/manifest_router.rb +++ b/bridgetown-routes/lib/bridgetown-routes/manifest_router.rb @@ -1,55 +1,11 @@ # frozen_string_literal: true -require "bridgetown-core/rack/routes" - module Bridgetown module Routes class ManifestRouter < Bridgetown::Rack::Routes priority :lowest - route do |r| - unless bridgetown_site - Bridgetown.logger.warn( - "The `bridgetown_routes` plugin hasn't been configured in the Roda app." - ) - return - end - - Bridgetown::Routes::Manifest.generate_manifest(bridgetown_site).each do |route| - file, localized_file_slugs, segment_keys = route - - localized_file_slugs.each do |file_slug| - add_route(r, file, file_slug, segment_keys) - end - end - - nil - end - - private - - def add_route(route, file, file_slug, segment_keys) - route.on file_slug do |*segment_values| - response["X-Bridgetown-SSR"] = "1" - # eval_route_file caches when Bridgetown.env.production? - Bridgetown::Routes::CodeBlocks.eval_route_file file, file_slug, @_roda_app - - segment_values.each_with_index do |value, index| - route.params[segment_keys[index]] ||= value - end - - # set route locale - locale = Bridgetown::Routes::Manifest.locale_for(file_slug, bridgetown_site) - I18n.locale = locale - route.params[:locale] = locale - - route_block = Bridgetown::Routes::CodeBlocks.route_block(file_slug) - response.instance_variable_set( - :@_route_file_code, route_block.instance_variable_get(:@_route_file_code) - ) # could be nil - @_roda_app.instance_exec(route, &route_block) - end - end + route(&:file_routes) end end end diff --git a/bridgetown-routes/lib/roda/plugins/bridgetown_routes.rb b/bridgetown-routes/lib/roda/plugins/bridgetown_routes.rb index 72c13d537..ba771d27e 100644 --- a/bridgetown-routes/lib/roda/plugins/bridgetown_routes.rb +++ b/bridgetown-routes/lib/roda/plugins/bridgetown_routes.rb @@ -25,13 +25,49 @@ def self.load_dependencies(app) end def self.configure(app, _opts = {}) - return unless app.opts[:bridgetown_site].nil? + app.root_hook do + routes_dir = File.expand_path( + bridgetown_site.config.routes.source_paths.first, + bridgetown_site.config.source + ) + file = routes_manifest.glob_routes(routes_dir, "index").first + next unless file + + run_file_route(file, slug: "index") + end + + if app.opts[:bridgetown_site] + app.opts[:routes_manifest] ||= + Bridgetown::Routes::Manifest.new(app.opts[:bridgetown_site]) + return + end raise "Roda app failure: the bridgetown_ssr plugin must be registered before " \ "bridgetown_routes" end module InstanceMethods + def routes_manifest + self.class.opts[:routes_manifest] + end + + def run_file_route(file, slug:) + response["X-Bridgetown-Routes"] = "1" + # eval_route_file caches when Bridgetown.env.production? + Bridgetown::Routes::CodeBlocks.eval_route_file file, slug, self + + # set route locale + locale = routes_manifest.locale_for(slug) + I18n.locale = request.params[:locale] = locale + + # get the route block extracted from the file at slug + route_block = Bridgetown::Routes::CodeBlocks.route_block(slug) + response.instance_variable_set( + :@_route_file_code, route_block.instance_variable_get(:@_route_file_code) + ) # could be nil + instance_exec(request, &route_block) + end + def front_matter(&block) b = block.binding denylisted = %i(r app code ruby_content code_postmatch) @@ -85,6 +121,32 @@ def view(view_class: Bridgetown::ERBView) end end + module RequestMethods + # This runs through all of the routes in the manifest, setting up Roda blocks for execution + def file_routes + scope.routes_manifest.routes.each do |route| + file, localized_file_slugs, segment_keys = route + + localized_file_slugs.each do |slug| + # This sets up an initial Roda route block at the slug, and handles segments as params + # + # _routes/nested/[slug].erb -> "nested/:slug" + # "nested/123" -> r.params[:slug] == 123 + on slug do |*segment_values| + segment_values.each_with_index do |value, index| + params[segment_keys[index]] ||= value + end + + # This is provided as an instance method by our Roda plugin: + scope.run_file_route(file, slug:) + end + end + end + + nil # be sure not to return the above array loop + end + end + module ResponseMethods # template string provided, if available, by the saved code block def _route_file_code diff --git a/bridgetown-routes/test/ssr/src/_routes/test_index.erb b/bridgetown-routes/test/ssr/src/_routes/test_index.erb new file mode 100644 index 000000000..90f05122c --- /dev/null +++ b/bridgetown-routes/test/ssr/src/_routes/test_index.erb @@ -0,0 +1,8 @@ +---<% +# Note: this file gets renamed to `index.erb` temporarily during the test! +render_with do + title "Dynamic Index" +end +%>--- + +

<%= data.title %>

\ No newline at end of file diff --git a/bridgetown-routes/test/ssr/src/index.md b/bridgetown-routes/test/ssr/src/index.md deleted file mode 100644 index 0317634c8..000000000 --- a/bridgetown-routes/test/ssr/src/index.md +++ /dev/null @@ -1 +0,0 @@ -Hello **world**! \ No newline at end of file diff --git a/bridgetown-routes/test/test_routes.rb b/bridgetown-routes/test/test_routes.rb index b33fa9b6c..7752d1a4a 100644 --- a/bridgetown-routes/test/test_routes.rb +++ b/bridgetown-routes/test/test_routes.rb @@ -15,12 +15,21 @@ def site end context "Roda-powered Bridgetown server" do # rubocop:todo Metrics/BlockLength - should "return the index page" do + should "return the static index page" do get "/" assert last_response.ok? assert_equal "

Index

", last_response.body end + should "return the dynamic index page if present" do + index_file = File.expand_path("ssr/src/_routes/test_index.erb", __dir__) + FileUtils.cp(index_file, index_file.sub("test_index.erb", "index.erb")) + get "/" + assert last_response.ok? + assert_equal "

Dynamic Index

", last_response.body + FileUtils.remove_file(index_file.sub("test_index.erb", "index.erb")) + end + should "return JSON for the hello route" do get "/hello/world" assert last_response.ok?