From a4f80f73cc5274210d3592cc7d6f93d1ed8526b7 Mon Sep 17 00:00:00 2001 From: 8aviav <8aviav@gmail.com> Date: Wed, 6 Sep 2023 20:55:27 +0200 Subject: [PATCH] Use Rails fragment cache for scrolled We introduce a cache helper to decide if fragment caching should happen. Cache helper is included in seed HTML helper, because that one renders entries#show. The change is wrapped in a feature to be able to stagger a roll-out to production systems. REDMINE-20281 --- .../helpers/pageflow_scrolled/cache_helper.rb | 11 +++ .../editor/seed_html_helper.rb | 1 + .../pageflow_scrolled/entries/show.html.erb | 86 ++++++++++--------- .../scrolled/config/locales/new/cache.de.yml | 4 + .../scrolled/config/locales/new/cache.en.yml | 4 + .../scrolled/lib/pageflow_scrolled/plugin.rb | 1 + .../pageflow_scrolled/cache_helper_spec.rb | 63 ++++++++++++++ entry_types/scrolled/spec/spec_helper.rb | 14 +++ 8 files changed, 142 insertions(+), 42 deletions(-) create mode 100644 entry_types/scrolled/app/helpers/pageflow_scrolled/cache_helper.rb create mode 100644 entry_types/scrolled/config/locales/new/cache.de.yml create mode 100644 entry_types/scrolled/config/locales/new/cache.en.yml create mode 100644 entry_types/scrolled/spec/helpers/pageflow_scrolled/cache_helper_spec.rb diff --git a/entry_types/scrolled/app/helpers/pageflow_scrolled/cache_helper.rb b/entry_types/scrolled/app/helpers/pageflow_scrolled/cache_helper.rb new file mode 100644 index 0000000000..fe4dfb2cae --- /dev/null +++ b/entry_types/scrolled/app/helpers/pageflow_scrolled/cache_helper.rb @@ -0,0 +1,11 @@ +module PageflowScrolled + # @api private + module CacheHelper + def cache_scrolled_entry(entry:, widget_scope:, &block) + condition = + widget_scope == :published && + entry.feature_state('scrolled_entry_fragment_caching') + cache_if(condition, [entry, :head_and_body, widget_scope], &block) + end + end +end diff --git a/entry_types/scrolled/app/helpers/pageflow_scrolled/editor/seed_html_helper.rb b/entry_types/scrolled/app/helpers/pageflow_scrolled/editor/seed_html_helper.rb index cc69b5dc37..403f8e3f68 100644 --- a/entry_types/scrolled/app/helpers/pageflow_scrolled/editor/seed_html_helper.rb +++ b/entry_types/scrolled/app/helpers/pageflow_scrolled/editor/seed_html_helper.rb @@ -7,6 +7,7 @@ module SeedHtmlHelper include Pageflow::WidgetsHelper include Pageflow::StructuredDataHelper include Pageflow::TextDirectionHelper + include PageflowScrolled::CacheHelper include FaviconHelper include PacksHelper include WebpackPublicPathHelper diff --git a/entry_types/scrolled/app/views/pageflow_scrolled/entries/show.html.erb b/entry_types/scrolled/app/views/pageflow_scrolled/entries/show.html.erb index 236b38aa9d..6b19f34bcf 100644 --- a/entry_types/scrolled/app/views/pageflow_scrolled/entries/show.html.erb +++ b/entry_types/scrolled/app/views/pageflow_scrolled/entries/show.html.erb @@ -1,59 +1,61 @@ - -<%= content_tag(:html, lang: @entry.locale, dir: text_direction(@entry.locale)) do %> - - <%= pretty_entry_title(@entry) %> +<%= cache_scrolled_entry(entry: @entry, widget_scope: @widget_scope) do %> + + <%= content_tag(:html, lang: @entry.locale, dir: text_direction(@entry.locale)) do %> + + <%= pretty_entry_title(@entry) %> - - + + - <%= social_share_meta_tags_for(@entry) %> - <%= meta_tags_for_entry(@entry) %> - <%= feed_link_tags_for_entry(@entry) unless @skip_feed_link_tags%> + <%= social_share_meta_tags_for(@entry) %> + <%= meta_tags_for_entry(@entry) %> + <%= feed_link_tags_for_entry(@entry) unless @skip_feed_link_tags%> - <%= scrolled_favicons_for_entry(@entry, entry_mode: @widget_scope) %> + <%= scrolled_favicons_for_entry(@entry, entry_mode: @widget_scope) %> - <%= javascript_include_tag 'pageflow_scrolled/legacy' %> - <%= scrolled_frontend_stylesheet_packs_tag(@entry, widget_scope: @widget_scope) %> + <%= javascript_include_tag 'pageflow_scrolled/legacy' %> + <%= scrolled_frontend_stylesheet_packs_tag(@entry, widget_scope: @widget_scope) %> - <%= scrolled_theme_properties_style_tag(@entry.theme) %> - <%= scrolled_theme_stylesheet_pack_tags(@entry.theme) %> + <%= scrolled_theme_properties_style_tag(@entry.theme) %> + <%= scrolled_theme_stylesheet_pack_tags(@entry.theme) %> - <%= render_widget_head_fragments(@entry, scope: @widget_scope) %> + <%= render_widget_head_fragments(@entry, scope: @widget_scope) %> - <% if Rails.env.development? %> - - <% end %> + + <% end %> - <% ssr_html = @skip_ssr ? '' : render_scrolled_entry(@entry) %> + <% ssr_html = @skip_ssr ? '' : render_scrolled_entry(@entry) %> - <% if !@skip_ssr && (params[:frontend] == 'v2' || @entry.feature_state('frontend_v2')) %> - <%= generated_media_queries_tags_for(ssr_html) %> - <% end %> - - - <%= structured_data_for_entry(@entry) unless @skip_structured_data %> + <% if !@skip_ssr && (params[:frontend] == 'v2' || @entry.feature_state('frontend_v2')) %> + <%= generated_media_queries_tags_for(ssr_html) %> + <% end %> + + + <%= structured_data_for_entry(@entry) unless @skip_structured_data %> - <%= render 'pageflow_scrolled/entries/global_notices' %> + <%= render 'pageflow_scrolled/entries/global_notices' %> -
-
<%= ssr_html %>
+
+
<%= ssr_html %>
-
- <%= render_widgets(@entry, scope: @widget_scope, insert_point: :bottom_of_entry) %> -
+
+ <%= render_widgets(@entry, scope: @widget_scope, insert_point: :bottom_of_entry) %> +
- <%= scrolled_webpack_public_path_script_tag %> - <%= scrolled_frontend_javascript_packs_tag(@entry, widget_scope: @widget_scope) %> + <%= scrolled_webpack_public_path_script_tag %> + <%= scrolled_frontend_javascript_packs_tag(@entry, widget_scope: @widget_scope) %> - <%= scrolled_entry_json_seed_script_tag(@entry, @seed_options || {}) %> - + <%= scrolled_entry_json_seed_script_tag(@entry, @seed_options || {}) %> + + <% end %> <% end %> diff --git a/entry_types/scrolled/config/locales/new/cache.de.yml b/entry_types/scrolled/config/locales/new/cache.de.yml new file mode 100644 index 0000000000..d14793e2d1 --- /dev/null +++ b/entry_types/scrolled/config/locales/new/cache.de.yml @@ -0,0 +1,4 @@ +de: + pageflow: + scrolled_entry_fragment_caching: + feature_name: Pageflow-Next-Fragment-Caching diff --git a/entry_types/scrolled/config/locales/new/cache.en.yml b/entry_types/scrolled/config/locales/new/cache.en.yml new file mode 100644 index 0000000000..f9a27981b2 --- /dev/null +++ b/entry_types/scrolled/config/locales/new/cache.en.yml @@ -0,0 +1,4 @@ +en: + pageflow: + scrolled_entry_fragment_caching: + feature_name: Pageflow Next Fragment Caching diff --git a/entry_types/scrolled/lib/pageflow_scrolled/plugin.rb b/entry_types/scrolled/lib/pageflow_scrolled/plugin.rb index d0057e7138..08014b405b 100644 --- a/entry_types/scrolled/lib/pageflow_scrolled/plugin.rb +++ b/entry_types/scrolled/lib/pageflow_scrolled/plugin.rb @@ -26,6 +26,7 @@ def configure(config) c.features.register('iframe_embed_content_element') c.features.register('image_gallery_content_element') c.features.register('frontend_v2') + c.features.register('scrolled_entry_fragment_caching') c.additional_frontend_seed_data.register( 'frontendVersion', diff --git a/entry_types/scrolled/spec/helpers/pageflow_scrolled/cache_helper_spec.rb b/entry_types/scrolled/spec/helpers/pageflow_scrolled/cache_helper_spec.rb new file mode 100644 index 0000000000..8de29d0677 --- /dev/null +++ b/entry_types/scrolled/spec/helpers/pageflow_scrolled/cache_helper_spec.rb @@ -0,0 +1,63 @@ +require 'spec_helper' + +module PageflowScrolled + RSpec.describe CacheHelper, type: :helper do + describe '#cache_scrolled_entry', :use_clean_rails_memory_store_fragment_caching do + it 'caches if feature is enabled and entry is published' do + entry = create(:published_entry, + with_feature: 'scrolled_entry_fragment_caching') + + result = 'initial value' + + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) { result = 'old value' } + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) { result = 'new value' } + + expect(result).to eq('old value') + end + + it "doesn't cache if feature is disabled" do + entry = create(:published_entry) + + result = 'initial value' + + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) { result = 'old value' } + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) { result = 'new value' } + + expect(result).to eq('new value') + end + + it "doesn't cache if widget_scope isn't :published" do + # would typically imply widget scope :published + entry = create(:published_entry, + with_feature: 'scrolled_entry_fragment_caching') + + result = 'initial value' + + helper.cache_scrolled_entry(entry: entry, widget_scope: :editor) { result = 'old value' } + helper.cache_scrolled_entry(entry: entry, widget_scope: :editor) { result = 'new value' } + + expect(result).to eq('new value') + end + + it 'caches for different values of widget scope' do + entry = create(:published_entry, + with_feature: 'scrolled_entry_fragment_caching') + + result = 'initial value' + published_result = 'initial value' + + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) do + result = 'oldest value', published_result = 'oldest value' + end + helper.cache_scrolled_entry(entry: entry, widget_scope: :editor) { result = 'old value' } + helper.cache_scrolled_entry(entry: entry, widget_scope: :published) do + result = 'new value', published_result = 'new value' + end + helper.cache_scrolled_entry(entry: entry, widget_scope: :editor) { result = 'newest value' } + + expect(result).to eq('newest value') + expect(published_result).to eq('oldest value') + end + end + end +end diff --git a/entry_types/scrolled/spec/spec_helper.rb b/entry_types/scrolled/spec/spec_helper.rb index bfda4936ee..55592637e1 100644 --- a/entry_types/scrolled/spec/spec_helper.rb +++ b/entry_types/scrolled/spec/spec_helper.rb @@ -30,4 +30,18 @@ config.expect_with :rspec do |c| c.syntax = :expect end + + # Inspired by https://gitlab.com/gitlab-org/gitlab-foss + config.around(:each, :use_clean_rails_memory_store_fragment_caching) do |example| + begin + caching_store = ActionController::Base.cache_store + ActionController::Base.cache_store = ActiveSupport::Cache::MemoryStore.new + ActionController::Base.perform_caching = true + + example.run + ensure + ActionController::Base.perform_caching = false + ActionController::Base.cache_store = caching_store + end + end end