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