FIX: Issues with custom icons in themes (#13732)

Fixes two issues:
- ignores invalid XML in custom icon sprite SVG file (and outputs an error if sprite was uploaded via admin UI)
- clears SVG sprite cache when deleting an `icons-sprite` upload in a theme
This commit is contained in:
Penar Musaraj 2021-07-14 15:18:29 -04:00 committed by GitHub
parent 7d43e51821
commit f7ab852e12
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 68 additions and 3 deletions

View File

@ -170,6 +170,29 @@ class ThemeField < ActiveRecord::Base
[js_compiler.content, errors&.join("\n")] [js_compiler.content, errors&.join("\n")]
end end
def validate_svg_sprite_xml
upload = Upload.find(self.upload_id) rescue nil
if Discourse.store.external?
external_copy = Discourse.store.download(upload) rescue nil
path = external_copy.try(:path)
else
path = Discourse.store.path_for(upload)
end
content = File.read(path)
error = nil
begin
svg_file = Nokogiri::XML(content) do |config|
config.options = Nokogiri::XML::ParseOptions::NOBLANKS
end
rescue => e
error = "Error with #{self.name}: #{e.inspect}"
end
error
end
def raw_translation_data(internal: false) def raw_translation_data(internal: false)
# Might raise ThemeTranslationParser::InvalidYaml # Might raise ThemeTranslationParser::InvalidYaml
ThemeTranslationParser.new(self, internal: internal).load ThemeTranslationParser.new(self, internal: internal).load
@ -358,7 +381,7 @@ class ThemeField < ActiveRecord::Base
self.compiler_version = Theme.compiler_version self.compiler_version = Theme.compiler_version
elsif svg_sprite_field? elsif svg_sprite_field?
DB.after_commit { SvgSprite.expire_cache } DB.after_commit { SvgSprite.expire_cache }
self.error = nil self.error = validate_svg_sprite_xml
self.value_baked = "baked" self.value_baked = "baked"
self.compiler_version = Theme.compiler_version self.compiler_version = Theme.compiler_version
end end
@ -554,6 +577,12 @@ class ThemeField < ActiveRecord::Base
MessageBus.publish "/footer-change/#{theme.id}", self.value if theme && self.name == "footer" MessageBus.publish "/footer-change/#{theme.id}", self.value if theme && self.name == "footer"
end end
after_destroy do
if svg_sprite_field?
DB.after_commit { SvgSprite.expire_cache }
end
end
private private
JAVASCRIPT_TYPES = %w( JAVASCRIPT_TYPES = %w(

View File

@ -351,9 +351,15 @@ License - https://fontawesome.com/license/free (Icons: CC BY 4.0, Fonts: SIL OFL
end end
custom_svg_sprites(theme_id).each do |item| custom_svg_sprites(theme_id).each do |item|
begin
svg_file = Nokogiri::XML(item[:sprite]) do |config| svg_file = Nokogiri::XML(item[:sprite]) do |config|
config.options = Nokogiri::XML::ParseOptions::NOBLANKS config.options = Nokogiri::XML::ParseOptions::NOBLANKS
end end
rescue => e
Rails.logger.warn("Bad XML in custom sprite in theme with ID=#{theme_id}. Error info: #{e.inspect}")
end
next if !svg_file
svg_file.css("symbol").each do |sym| svg_file.css("symbol").each do |sym|
icon_id = prepare_symbol(sym, item[:filename]) icon_id = prepare_symbol(sym, item[:filename])

View File

@ -264,6 +264,19 @@ describe SvgSprite do
expect(SvgSprite.bundle(theme.id)).to match(/my-custom-theme-icon/) expect(SvgSprite.bundle(theme.id)).to match(/my-custom-theme-icon/)
end end
it 'does not fail on bad XML in custom icon sprite' do
theme = Fabricate(:theme)
fname = "bad-xml-icon-sprite.svg"
upload = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
theme.set_field(target: :common, name: SvgSprite.theme_sprite_variable_name, upload_id: upload.id, type: :theme_upload_var)
theme.save!
expect(Upload.where(id: upload.id)).to be_exist
expect(SvgSprite.bundle(theme.id)).to match(/arrow-down/)
end
it 'includes custom icons in a child theme' do it 'includes custom icons in a child theme' do
theme = Fabricate(:theme) theme = Fabricate(:theme)
fname = "custom-theme-icon-sprite.svg" fname = "custom-theme-icon-sprite.svg"

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<svg width="0" height="0" class="hidden">
<symbol id="mytheme-icon-menu xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24">
<path d="M0 0h24v24H0V0z" fill="#9c9c9c"></path>
</symbol>
</svg>

After

Width:  |  Height:  |  Size: 240 B

View File

@ -419,6 +419,17 @@ HTML
theme_field.update(upload: Fabricate(:upload)) theme_field.update(upload: Fabricate(:upload))
expect(theme_field.value_baked).to eq(nil) expect(theme_field.value_baked).to eq(nil)
end end
it "clears SVG sprite cache when upload is deleted" do
fname = "custom-theme-icon-sprite.svg"
sprite = UploadCreator.new(file_from_fixtures(fname), fname, for_theme: true).create_for(-1)
theme_field.update(upload: sprite)
expect(SvgSprite.custom_svg_sprites(theme.id).size).to eq(1)
theme_field.destroy!
expect(SvgSprite.custom_svg_sprites(theme.id).size).to eq(0)
end
end end
end end