mirror of
https://github.com/glitch-soc/mastodon.git
synced 2024-11-25 09:34:16 -05:00
178e151019
Conflicts: - `app/models/user_settings.rb`: Upstream added a constraint on a setting textually close to glitch-soc-only settings. Applied upstream's change. - `lib/sanitize_ext/sanitize_config.rb`: Upstream added support for the `translate` attribute on a few elements, where glitch-soc had a different set of allowed elements and attributes. Extended glitch-soc's allowed attributes with `translate` as upstream did. - `spec/validators/status_length_validator_spec.rb`: Upstream refactored to use RSpec's `instance_double` instead of `double`, but glitch-soc had changes to tests due to configurable max toot chars. Applied upstream's changes while keeping tests against configurable max toot chars.
176 lines
4.5 KiB
Ruby
176 lines
4.5 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
class Sanitize
|
|
module Config
|
|
HTTP_PROTOCOLS = %w(
|
|
http
|
|
https
|
|
).freeze
|
|
|
|
LINK_PROTOCOLS = %w(
|
|
http
|
|
https
|
|
dat
|
|
dweb
|
|
ipfs
|
|
ipns
|
|
ssb
|
|
gopher
|
|
xmpp
|
|
magnet
|
|
gemini
|
|
).freeze
|
|
|
|
CLASS_WHITELIST_TRANSFORMER = lambda do |env|
|
|
node = env[:node]
|
|
class_list = node['class']&.split(/[\t\n\f\r ]/)
|
|
|
|
return unless class_list
|
|
|
|
class_list.keep_if do |e|
|
|
next true if /^(h|p|u|dt|e)-/.match?(e) # microformats classes
|
|
next true if /^(mention|hashtag)$/.match?(e) # semantic classes
|
|
next true if /^(ellipsis|invisible)$/.match?(e) # link formatting classes
|
|
end
|
|
|
|
node['class'] = class_list.join(' ')
|
|
end
|
|
|
|
IMG_TAG_TRANSFORMER = lambda do |env|
|
|
node = env[:node]
|
|
|
|
return unless env[:node_name] == 'img'
|
|
|
|
node.name = 'a'
|
|
|
|
node['href'] = node['src']
|
|
if node['alt'].present?
|
|
node.content = "[🖼 #{node['alt']}]"
|
|
else
|
|
url = node['href']
|
|
prefix = url.match(%r{\Ahttps?://(www\.)?}).to_s
|
|
text = url[prefix.length, 30]
|
|
text += '…' if url.length - prefix.length > 30
|
|
node.content = "[🖼 #{text}]"
|
|
end
|
|
end
|
|
|
|
TRANSLATE_TRANSFORMER = lambda do |env|
|
|
node = env[:node]
|
|
node.remove_attribute('translate') unless node['translate'] == 'no'
|
|
end
|
|
|
|
UNSUPPORTED_HREF_TRANSFORMER = lambda do |env|
|
|
return unless env[:node_name] == 'a'
|
|
|
|
current_node = env[:node]
|
|
|
|
scheme = if current_node['href'] =~ Sanitize::REGEX_PROTOCOL
|
|
Regexp.last_match(1).downcase
|
|
else
|
|
:relative
|
|
end
|
|
|
|
current_node.replace(Nokogiri::XML::Text.new(current_node.text, current_node.document)) unless LINK_PROTOCOLS.include?(scheme)
|
|
end
|
|
|
|
MASTODON_STRICT ||= freeze_config(
|
|
elements: %w(p br span a abbr del pre blockquote code b strong u sub sup i em h1 h2 h3 h4 h5 ul ol li),
|
|
|
|
attributes: {
|
|
'a' => %w(href rel class title translate),
|
|
'abbr' => %w(title),
|
|
'span' => %w(class translate),
|
|
'blockquote' => %w(cite),
|
|
'ol' => %w(start reversed),
|
|
'li' => %w(value),
|
|
},
|
|
|
|
add_attributes: {
|
|
'a' => {
|
|
'rel' => 'nofollow noopener noreferrer',
|
|
'target' => '_blank',
|
|
},
|
|
},
|
|
|
|
protocols: {
|
|
'a' => { 'href' => LINK_PROTOCOLS },
|
|
'blockquote' => { 'cite' => LINK_PROTOCOLS },
|
|
},
|
|
|
|
transformers: [
|
|
CLASS_WHITELIST_TRANSFORMER,
|
|
IMG_TAG_TRANSFORMER,
|
|
TRANSLATE_TRANSFORMER,
|
|
UNSUPPORTED_HREF_TRANSFORMER,
|
|
]
|
|
)
|
|
|
|
MASTODON_OEMBED ||= freeze_config merge(
|
|
RELAXED,
|
|
elements: RELAXED[:elements] + %w(audio embed iframe source video),
|
|
|
|
attributes: merge(
|
|
RELAXED[:attributes],
|
|
'audio' => %w(controls),
|
|
'embed' => %w(height src type width),
|
|
'iframe' => %w(allowfullscreen frameborder height scrolling src width),
|
|
'source' => %w(src type),
|
|
'video' => %w(controls height loop width),
|
|
'div' => [:data]
|
|
),
|
|
|
|
protocols: merge(
|
|
RELAXED[:protocols],
|
|
'embed' => { 'src' => HTTP_PROTOCOLS },
|
|
'iframe' => { 'src' => HTTP_PROTOCOLS },
|
|
'source' => { 'src' => HTTP_PROTOCOLS }
|
|
)
|
|
)
|
|
|
|
LINK_REL_TRANSFORMER = lambda do |env|
|
|
return unless env[:node_name] == 'a' && env[:node]['href']
|
|
|
|
node = env[:node]
|
|
|
|
rel = (node['rel'] || '').split & ['tag']
|
|
rel += %w(nofollow noopener noreferrer) unless TagManager.instance.local_url?(node['href'])
|
|
|
|
if rel.empty?
|
|
node.remove_attribute('rel')
|
|
else
|
|
node['rel'] = rel.join(' ')
|
|
end
|
|
end
|
|
|
|
LINK_TARGET_TRANSFORMER = lambda do |env|
|
|
return unless env[:node_name] == 'a' && env[:node]['href']
|
|
|
|
node = env[:node]
|
|
if node['target'] != '_blank' && TagManager.instance.local_url?(node['href'])
|
|
node.remove_attribute('target')
|
|
else
|
|
node['target'] = '_blank'
|
|
end
|
|
end
|
|
|
|
MASTODON_OUTGOING ||= freeze_config MASTODON_STRICT.merge(
|
|
attributes: merge(
|
|
MASTODON_STRICT[:attributes],
|
|
'a' => %w(href rel class title target translate)
|
|
),
|
|
|
|
add_attributes: {},
|
|
|
|
transformers: [
|
|
CLASS_WHITELIST_TRANSFORMER,
|
|
IMG_TAG_TRANSFORMER,
|
|
TRANSLATE_TRANSFORMER,
|
|
UNSUPPORTED_HREF_TRANSFORMER,
|
|
LINK_REL_TRANSFORMER,
|
|
LINK_TARGET_TRANSFORMER,
|
|
]
|
|
)
|
|
end
|
|
end
|