Extend custom CSS cache time with digest paths (#33207)
This commit is contained in:
parent
b3243ef41c
commit
c0264c8013
11 changed files with 142 additions and 6 deletions
|
@ -2,7 +2,7 @@
|
||||||
|
|
||||||
class CustomCssController < ActionController::Base # rubocop:disable Rails/ApplicationController
|
class CustomCssController < ActionController::Base # rubocop:disable Rails/ApplicationController
|
||||||
def show
|
def show
|
||||||
expires_in 3.minutes, public: true
|
expires_in 1.month, public: true
|
||||||
render content_type: 'text/css'
|
render content_type: 'text/css'
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -23,8 +23,31 @@ module ThemeHelper
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
def custom_stylesheet
|
||||||
|
if active_custom_stylesheet.present?
|
||||||
|
stylesheet_link_tag(
|
||||||
|
custom_css_path(active_custom_stylesheet),
|
||||||
|
host: root_url,
|
||||||
|
media: :all,
|
||||||
|
skip_pipeline: true
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def active_custom_stylesheet
|
||||||
|
if cached_custom_css_digest.present?
|
||||||
|
[:custom, cached_custom_css_digest.to_s.first(8)]
|
||||||
|
.compact_blank
|
||||||
|
.join('-')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def cached_custom_css_digest
|
||||||
|
Rails.cache.read(:setting_digest_custom_css)
|
||||||
|
end
|
||||||
|
|
||||||
def theme_color_for(theme)
|
def theme_color_for(theme)
|
||||||
theme == 'mastodon-light' ? Themes::THEME_COLORS[:light] : Themes::THEME_COLORS[:dark]
|
theme == 'mastodon-light' ? Themes::THEME_COLORS[:light] : Themes::THEME_COLORS[:dark]
|
||||||
end
|
end
|
||||||
|
|
|
@ -69,6 +69,10 @@ class Form::AdminSettings
|
||||||
favicon
|
favicon
|
||||||
).freeze
|
).freeze
|
||||||
|
|
||||||
|
DIGEST_KEYS = %i(
|
||||||
|
custom_css
|
||||||
|
).freeze
|
||||||
|
|
||||||
OVERRIDEN_SETTINGS = {
|
OVERRIDEN_SETTINGS = {
|
||||||
authorized_fetch: :authorized_fetch_mode?,
|
authorized_fetch: :authorized_fetch_mode?,
|
||||||
}.freeze
|
}.freeze
|
||||||
|
@ -122,6 +126,8 @@ class Form::AdminSettings
|
||||||
KEYS.each do |key|
|
KEYS.each do |key|
|
||||||
next unless instance_variable_defined?(:"@#{key}")
|
next unless instance_variable_defined?(:"@#{key}")
|
||||||
|
|
||||||
|
cache_digest_value(key) if DIGEST_KEYS.include?(key)
|
||||||
|
|
||||||
if UPLOAD_KEYS.include?(key)
|
if UPLOAD_KEYS.include?(key)
|
||||||
public_send(key).save
|
public_send(key).save
|
||||||
else
|
else
|
||||||
|
@ -133,6 +139,18 @@ class Form::AdminSettings
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
|
def cache_digest_value(key)
|
||||||
|
Rails.cache.delete(:"setting_digest_#{key}")
|
||||||
|
|
||||||
|
key_value = instance_variable_get(:"@#{key}")
|
||||||
|
if key_value.present?
|
||||||
|
Rails.cache.write(
|
||||||
|
:"setting_digest_#{key}",
|
||||||
|
Digest::SHA256.hexdigest(key_value)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
def typecast_value(key, value)
|
def typecast_value(key, value)
|
||||||
if BOOLEAN_KEYS.include?(key)
|
if BOOLEAN_KEYS.include?(key)
|
||||||
value == '1'
|
value == '1'
|
||||||
|
|
|
@ -34,7 +34,7 @@
|
||||||
= csrf_meta_tags unless skip_csrf_meta_tags?
|
= csrf_meta_tags unless skip_csrf_meta_tags?
|
||||||
%meta{ name: 'style-nonce', content: request.content_security_policy_nonce }
|
%meta{ name: 'style-nonce', content: request.content_security_policy_nonce }
|
||||||
|
|
||||||
= stylesheet_link_tag custom_css_path, skip_pipeline: true, host: root_url, media: 'all'
|
= custom_stylesheet
|
||||||
|
|
||||||
= yield :header_tags
|
= yield :header_tags
|
||||||
|
|
||||||
|
|
18
config/initializers/settings_digests.rb
Normal file
18
config/initializers/settings_digests.rb
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
Rails.application.config.to_prepare do
|
||||||
|
custom_css = begin
|
||||||
|
Setting.custom_css
|
||||||
|
rescue ActiveRecord::AdapterError # Running without a database, not migrated, no connection, etc
|
||||||
|
nil
|
||||||
|
end
|
||||||
|
|
||||||
|
if custom_css.present?
|
||||||
|
Rails
|
||||||
|
.cache
|
||||||
|
.write(
|
||||||
|
:setting_digest_custom_css,
|
||||||
|
Digest::SHA256.hexdigest(custom_css)
|
||||||
|
)
|
||||||
|
end
|
||||||
|
end
|
|
@ -55,7 +55,8 @@ Rails.application.routes.draw do
|
||||||
|
|
||||||
get 'manifest', to: 'manifests#show', defaults: { format: 'json' }
|
get 'manifest', to: 'manifests#show', defaults: { format: 'json' }
|
||||||
get 'intent', to: 'intents#show'
|
get 'intent', to: 'intents#show'
|
||||||
get 'custom.css', to: 'custom_css#show', as: :custom_css
|
get 'custom.css', to: 'custom_css#show'
|
||||||
|
resources :custom_css, only: :show, path: :css
|
||||||
|
|
||||||
get 'remote_interaction_helper', to: 'remote_interaction_helper#index'
|
get 'remote_interaction_helper', to: 'remote_interaction_helper#index'
|
||||||
|
|
||||||
|
|
|
@ -79,6 +79,26 @@ RSpec.describe ThemeHelper do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe '#custom_stylesheet' do
|
||||||
|
context 'when custom css setting value digest is present' do
|
||||||
|
before { Rails.cache.write(:setting_digest_custom_css, '1a2s3d4f1a2s3d4f') }
|
||||||
|
|
||||||
|
it 'returns value from settings' do
|
||||||
|
expect(custom_stylesheet)
|
||||||
|
.to match('/css/custom-1a2s3d4f.css')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when custom css setting value digest is not present' do
|
||||||
|
before { Rails.cache.delete(:setting_digest_custom_css) }
|
||||||
|
|
||||||
|
it 'returns default value' do
|
||||||
|
expect(custom_stylesheet)
|
||||||
|
.to be_blank
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
private
|
private
|
||||||
|
|
||||||
def html_links
|
def html_links
|
||||||
|
|
|
@ -17,4 +17,40 @@ RSpec.describe Form::AdminSettings do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
describe '#save' do
|
||||||
|
describe 'updating digest values' do
|
||||||
|
context 'when updating custom css to real value' do
|
||||||
|
subject { described_class.new(custom_css: css) }
|
||||||
|
|
||||||
|
let(:css) { 'body { color: red; }' }
|
||||||
|
let(:digested) { Digest::SHA256.hexdigest(css) }
|
||||||
|
|
||||||
|
it 'changes relevant digest value' do
|
||||||
|
expect { subject.save }
|
||||||
|
.to(change { Rails.cache.read(:setting_digest_custom_css) }.to(digested))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when updating custom css to empty value' do
|
||||||
|
subject { described_class.new(custom_css: '') }
|
||||||
|
|
||||||
|
before { Rails.cache.write(:setting_digest_custom_css, 'previous-value') }
|
||||||
|
|
||||||
|
it 'changes relevant digest value' do
|
||||||
|
expect { subject.save }
|
||||||
|
.to(change { Rails.cache.read(:setting_digest_custom_css) }.to(be_blank))
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
context 'when updating other fields' do
|
||||||
|
subject { described_class.new(site_contact_email: 'test@example.host') }
|
||||||
|
|
||||||
|
it 'does not update digests' do
|
||||||
|
expect { subject.save }
|
||||||
|
.to(not_change { Rails.cache.read(:setting_digest_custom_css) })
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -10,6 +10,7 @@ module TestEndpoints
|
||||||
/.well-known/nodeinfo
|
/.well-known/nodeinfo
|
||||||
/nodeinfo/2.0
|
/nodeinfo/2.0
|
||||||
/manifest
|
/manifest
|
||||||
|
/css/custom-1a2s3d4f.css
|
||||||
/custom.css
|
/custom.css
|
||||||
/actor
|
/actor
|
||||||
/api/v1/instance/extended_description
|
/api/v1/instance/extended_description
|
||||||
|
|
|
@ -5,10 +5,10 @@ require 'rails_helper'
|
||||||
RSpec.describe 'Custom CSS' do
|
RSpec.describe 'Custom CSS' do
|
||||||
include RoutingHelper
|
include RoutingHelper
|
||||||
|
|
||||||
describe 'GET /custom.css' do
|
describe 'GET /css/:id.css' do
|
||||||
context 'without any CSS or User Roles' do
|
context 'without any CSS or User Roles' do
|
||||||
it 'returns empty stylesheet' do
|
it 'returns empty stylesheet' do
|
||||||
get '/custom.css'
|
get '/css/custom-123.css'
|
||||||
|
|
||||||
expect(response)
|
expect(response)
|
||||||
.to have_http_status(200)
|
.to have_http_status(200)
|
||||||
|
@ -27,7 +27,7 @@ RSpec.describe 'Custom CSS' do
|
||||||
end
|
end
|
||||||
|
|
||||||
it 'returns stylesheet from settings' do
|
it 'returns stylesheet from settings' do
|
||||||
get '/custom.css'
|
get '/css/custom-456.css'
|
||||||
|
|
||||||
expect(response)
|
expect(response)
|
||||||
.to have_http_status(200)
|
.to have_http_status(200)
|
||||||
|
|
19
spec/routing/custom_css_routing_spec.rb
Normal file
19
spec/routing/custom_css_routing_spec.rb
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
require 'rails_helper'
|
||||||
|
|
||||||
|
RSpec.describe 'Custom CSS routes' do
|
||||||
|
describe 'the legacy route' do
|
||||||
|
it 'routes to correct place' do
|
||||||
|
expect(get('/custom.css'))
|
||||||
|
.to route_to('custom_css#show')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
describe 'the custom digest route' do
|
||||||
|
it 'routes to correct place' do
|
||||||
|
expect(get('/css/custom-1a2s3d4f.css'))
|
||||||
|
.to route_to('custom_css#show', id: 'custom-1a2s3d4f', format: 'css')
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
Loading…
Reference in a new issue