Commit 0a445981 authored by Tan Le's avatar Tan Le

Replace whitelist wording in Banzai module

We prefer a neutral tone when specifying a list of allowed
elements (aka. allowlist). We need to bump our version of
`html-pipeline` gem to `2.13.2` to bring in some changes in the method
that we override.

Related upstream change https://github.com/jch/html-pipeline/pull/339
parent 1f1f7509
...@@ -145,7 +145,7 @@ gem 'aws-sdk-s3', '~> 1' ...@@ -145,7 +145,7 @@ gem 'aws-sdk-s3', '~> 1'
gem 'faraday_middleware-aws-sigv4', '~>0.3.0' gem 'faraday_middleware-aws-sigv4', '~>0.3.0'
# Markdown and HTML processing # Markdown and HTML processing
gem 'html-pipeline', '~> 2.12' gem 'html-pipeline', '~> 2.13.2'
gem 'deckar01-task_list', '2.3.1' gem 'deckar01-task_list', '2.3.1'
gem 'gitlab-markup', '~> 1.7.1' gem 'gitlab-markup', '~> 1.7.1'
gem 'github-markup', '~> 1.7.0', require: 'github/markup' gem 'github-markup', '~> 1.7.0', require: 'github/markup'
......
...@@ -577,7 +577,7 @@ GEM ...@@ -577,7 +577,7 @@ GEM
hipchat (1.5.2) hipchat (1.5.2)
httparty httparty
mimemagic mimemagic
html-pipeline (2.12.2) html-pipeline (2.13.2)
activesupport (>= 2) activesupport (>= 2)
nokogiri (>= 1.4) nokogiri (>= 1.4)
html2text (0.2.0) html2text (0.2.0)
...@@ -1391,7 +1391,7 @@ DEPENDENCIES ...@@ -1391,7 +1391,7 @@ DEPENDENCIES
hashie-forbidden_attributes hashie-forbidden_attributes
health_check (~> 3.0) health_check (~> 3.0)
hipchat (~> 1.5.0) hipchat (~> 1.5.0)
html-pipeline (~> 2.12) html-pipeline (~> 2.13.2)
html2text html2text
httparty (~> 0.16.4) httparty (~> 0.16.4)
icalendar icalendar
......
...@@ -27,7 +27,7 @@ module Banzai ...@@ -27,7 +27,7 @@ module Banzai
TABLE_GRID_CLASSES = %w(grid-all grid-rows grid-cols grid-none).freeze TABLE_GRID_CLASSES = %w(grid-all grid-rows grid-cols grid-none).freeze
TABLE_STRIPES_CLASSES = %w(stripes-all stripes-odd stripes-even stripes-hover stripes-none).freeze TABLE_STRIPES_CLASSES = %w(stripes-all stripes-odd stripes-even stripes-hover stripes-none).freeze
ELEMENT_CLASSES_WHITELIST = { ELEMENT_CLASSES_ALLOWLIST = {
span: %w(big small underline overline line-through).freeze, span: %w(big small underline overline line-through).freeze,
div: ALIGNMENT_BUILTINS_CLASSES + ['admonitionblock'].freeze, div: ALIGNMENT_BUILTINS_CLASSES + ['admonitionblock'].freeze,
td: ['icon'].freeze, td: ['icon'].freeze,
...@@ -38,35 +38,35 @@ module Banzai ...@@ -38,35 +38,35 @@ module Banzai
table: TABLE_FRAME_CLASSES + TABLE_GRID_CLASSES + TABLE_STRIPES_CLASSES table: TABLE_FRAME_CLASSES + TABLE_GRID_CLASSES + TABLE_STRIPES_CLASSES
}.freeze }.freeze
def customize_whitelist(whitelist) def customize_allowlist(allowlist)
# Allow marks # Allow marks
whitelist[:elements].push('mark') allowlist[:elements].push('mark')
# Allow any classes in `span`, `i`, `div`, `td`, `ul`, `ol` and `a` elements # Allow any classes in `span`, `i`, `div`, `td`, `ul`, `ol` and `a` elements
# but then remove any unknown classes # but then remove any unknown classes
whitelist[:attributes]['span'] = %w(class) allowlist[:attributes]['span'] = %w(class)
whitelist[:attributes]['div'].push('class') allowlist[:attributes]['div'].push('class')
whitelist[:attributes]['td'] = %w(class) allowlist[:attributes]['td'] = %w(class)
whitelist[:attributes]['i'] = %w(class) allowlist[:attributes]['i'] = %w(class)
whitelist[:attributes]['ul'] = %w(class) allowlist[:attributes]['ul'] = %w(class)
whitelist[:attributes]['ol'] = %w(class) allowlist[:attributes]['ol'] = %w(class)
whitelist[:attributes]['a'].push('class') allowlist[:attributes]['a'].push('class')
whitelist[:attributes]['table'] = %w(class) allowlist[:attributes]['table'] = %w(class)
whitelist[:transformers].push(self.class.remove_element_classes) allowlist[:transformers].push(self.class.remove_element_classes)
# Allow `id` in anchor and footnote elements # Allow `id` in anchor and footnote elements
whitelist[:attributes]['a'].push('id') allowlist[:attributes]['a'].push('id')
whitelist[:attributes]['div'].push('id') allowlist[:attributes]['div'].push('id')
# Allow `id` in heading elements for section anchors # Allow `id` in heading elements for section anchors
SECTION_HEADINGS.each do |header| SECTION_HEADINGS.each do |header|
whitelist[:attributes][header] = %w(id) allowlist[:attributes][header] = %w(id)
end end
# Remove ids that are not explicitly allowed # Remove ids that are not explicitly allowed
whitelist[:transformers].push(self.class.remove_disallowed_ids) allowlist[:transformers].push(self.class.remove_disallowed_ids)
whitelist allowlist
end end
class << self class << self
...@@ -91,11 +91,11 @@ module Banzai ...@@ -91,11 +91,11 @@ module Banzai
lambda do |env| lambda do |env|
node = env[:node] node = env[:node]
return unless (classes_whitelist = ELEMENT_CLASSES_WHITELIST[node.name.to_sym]) return unless (classes_allowlist = ELEMENT_CLASSES_ALLOWLIST[node.name.to_sym])
return unless node.has_attribute?('class') return unless node.has_attribute?('class')
classes = node['class'].strip.split(' ') classes = node['class'].strip.split(' ')
allowed_classes = (classes & classes_whitelist) allowed_classes = (classes & classes_allowlist)
if allowed_classes.empty? if allowed_classes.empty?
node.remove_attribute('class') node.remove_attribute('class')
else else
......
...@@ -15,7 +15,7 @@ module Banzai ...@@ -15,7 +15,7 @@ module Banzai
needs(:asset_proxy, :asset_proxy_secret_key) if asset_proxy_enabled? needs(:asset_proxy, :asset_proxy_secret_key) if asset_proxy_enabled?
end end
def asset_host_whitelisted?(host) def asset_host_allowed?(host)
context[:asset_proxy_domain_regexp] ? context[:asset_proxy_domain_regexp].match?(host) : false context[:asset_proxy_domain_regexp] ? context[:asset_proxy_domain_regexp].match?(host) : false
end end
...@@ -44,21 +44,21 @@ module Banzai ...@@ -44,21 +44,21 @@ module Banzai
Gitlab.config.asset_proxy['enabled'] = application_settings.asset_proxy_enabled Gitlab.config.asset_proxy['enabled'] = application_settings.asset_proxy_enabled
Gitlab.config.asset_proxy['url'] = application_settings.asset_proxy_url Gitlab.config.asset_proxy['url'] = application_settings.asset_proxy_url
Gitlab.config.asset_proxy['secret_key'] = application_settings.asset_proxy_secret_key Gitlab.config.asset_proxy['secret_key'] = application_settings.asset_proxy_secret_key
Gitlab.config.asset_proxy['whitelist'] = determine_whitelist(application_settings) Gitlab.config.asset_proxy['allowlist'] = determine_allowlist(application_settings)
Gitlab.config.asset_proxy['domain_regexp'] = compile_whitelist(Gitlab.config.asset_proxy.whitelist) Gitlab.config.asset_proxy['domain_regexp'] = compile_allowlist(Gitlab.config.asset_proxy.allowlist)
else else
Gitlab.config.asset_proxy['enabled'] = ::ApplicationSetting.defaults[:asset_proxy_enabled] Gitlab.config.asset_proxy['enabled'] = ::ApplicationSetting.defaults[:asset_proxy_enabled]
end end
end end
def self.compile_whitelist(domain_list) def self.compile_allowlist(domain_list)
return if domain_list.empty? return if domain_list.empty?
escaped = domain_list.map { |domain| Regexp.escape(domain).gsub('\*', '.*?') } escaped = domain_list.map { |domain| Regexp.escape(domain).gsub('\*', '.*?') }
Regexp.new("^(#{escaped.join('|')})$", Regexp::IGNORECASE) Regexp.new("^(#{escaped.join('|')})$", Regexp::IGNORECASE)
end end
def self.determine_whitelist(application_settings) def self.determine_allowlist(application_settings)
application_settings.asset_proxy_whitelist.presence || [Gitlab.config.gitlab.host] application_settings.asset_proxy_whitelist.presence || [Gitlab.config.gitlab.host]
end end
end end
......
...@@ -16,42 +16,42 @@ module Banzai ...@@ -16,42 +16,42 @@ module Banzai
UNSAFE_PROTOCOLS = %w(data javascript vbscript).freeze UNSAFE_PROTOCOLS = %w(data javascript vbscript).freeze
def whitelist def allowlist
strong_memoize(:whitelist) do strong_memoize(:allowlist) do
whitelist = super.deep_dup allowlist = super.deep_dup
# Allow span elements # Allow span elements
whitelist[:elements].push('span') allowlist[:elements].push('span')
# Allow data-math-style attribute in order to support LaTeX formatting # Allow data-math-style attribute in order to support LaTeX formatting
whitelist[:attributes]['code'] = %w(data-math-style) allowlist[:attributes]['code'] = %w(data-math-style)
whitelist[:attributes]['pre'] = %w(data-math-style data-mermaid-style data-kroki-style) allowlist[:attributes]['pre'] = %w(data-math-style data-mermaid-style data-kroki-style)
# Allow html5 details/summary elements # Allow html5 details/summary elements
whitelist[:elements].push('details') allowlist[:elements].push('details')
whitelist[:elements].push('summary') allowlist[:elements].push('summary')
# Allow abbr elements with title attribute # Allow abbr elements with title attribute
whitelist[:elements].push('abbr') allowlist[:elements].push('abbr')
whitelist[:attributes]['abbr'] = %w(title) allowlist[:attributes]['abbr'] = %w(title)
# Disallow `name` attribute globally, allow on `a` # Disallow `name` attribute globally, allow on `a`
whitelist[:attributes][:all].delete('name') allowlist[:attributes][:all].delete('name')
whitelist[:attributes]['a'].push('name') allowlist[:attributes]['a'].push('name')
# Allow any protocol in `a` elements # Allow any protocol in `a` elements
# and then remove links with unsafe protocols # and then remove links with unsafe protocols
whitelist[:protocols].delete('a') allowlist[:protocols].delete('a')
whitelist[:transformers].push(self.class.method(:remove_unsafe_links)) allowlist[:transformers].push(self.class.method(:remove_unsafe_links))
# Remove `rel` attribute from `a` elements # Remove `rel` attribute from `a` elements
whitelist[:transformers].push(self.class.remove_rel) allowlist[:transformers].push(self.class.remove_rel)
customize_whitelist(whitelist) customize_allowlist(allowlist)
end end
end end
def customize_whitelist(whitelist) def customize_allowlist(allowlist)
raise NotImplementedError raise NotImplementedError
end end
......
...@@ -6,14 +6,14 @@ module Banzai ...@@ -6,14 +6,14 @@ module Banzai
# #
# Extends Banzai::Filter::BaseSanitizationFilter with specific rules. # Extends Banzai::Filter::BaseSanitizationFilter with specific rules.
class BroadcastMessageSanitizationFilter < Banzai::Filter::BaseSanitizationFilter class BroadcastMessageSanitizationFilter < Banzai::Filter::BaseSanitizationFilter
def customize_whitelist(whitelist) def customize_allowlist(allowlist)
whitelist[:elements].push('br') allowlist[:elements].push('br')
whitelist[:attributes]['a'].push('class', 'style') allowlist[:attributes]['a'].push('class', 'style')
whitelist[:css] = { properties: %w(color border background padding margin text-decoration) } allowlist[:css] = { properties: %w(color border background padding margin text-decoration) }
whitelist allowlist
end end
end end
end end
......
...@@ -9,26 +9,26 @@ module Banzai ...@@ -9,26 +9,26 @@ module Banzai
# Styles used by Markdown for table alignment # Styles used by Markdown for table alignment
TABLE_ALIGNMENT_PATTERN = /text-align: (?<alignment>center|left|right)/.freeze TABLE_ALIGNMENT_PATTERN = /text-align: (?<alignment>center|left|right)/.freeze
def customize_whitelist(whitelist) def customize_allowlist(allowlist)
# Allow table alignment; we whitelist specific text-align values in a # Allow table alignment; we allow specific text-align values in a
# transformer below # transformer below
whitelist[:attributes]['th'] = %w(style) allowlist[:attributes]['th'] = %w(style)
whitelist[:attributes]['td'] = %w(style) allowlist[:attributes]['td'] = %w(style)
whitelist[:css] = { properties: ['text-align'] } allowlist[:css] = { properties: ['text-align'] }
# Allow the 'data-sourcepos' from CommonMark on all elements # Allow the 'data-sourcepos' from CommonMark on all elements
whitelist[:attributes][:all].push('data-sourcepos') allowlist[:attributes][:all].push('data-sourcepos')
# Remove any `style` properties not required for table alignment # Remove any `style` properties not required for table alignment
whitelist[:transformers].push(self.class.remove_unsafe_table_style) allowlist[:transformers].push(self.class.remove_unsafe_table_style)
# Allow `id` in a and li elements for footnotes # Allow `id` in a and li elements for footnotes
# and remove any `id` properties not matching for footnotes # and remove any `id` properties not matching for footnotes
whitelist[:attributes]['a'].push('id') allowlist[:attributes]['a'].push('id')
whitelist[:attributes]['li'] = %w(id) allowlist[:attributes]['li'] = %w(id)
whitelist[:transformers].push(self.class.remove_non_footnote_ids) allowlist[:transformers].push(self.class.remove_non_footnote_ids)
whitelist allowlist
end end
class << self class << self
......
...@@ -3,14 +3,14 @@ ...@@ -3,14 +3,14 @@
module Banzai module Banzai
module Pipeline module Pipeline
class DescriptionPipeline < FullPipeline class DescriptionPipeline < FullPipeline
WHITELIST = Banzai::Filter::SanitizationFilter::LIMITED.deep_dup.merge( ALLOWLIST = Banzai::Filter::SanitizationFilter::LIMITED.deep_dup.merge(
elements: Banzai::Filter::SanitizationFilter::LIMITED[:elements] - %w(pre code img ol ul li) elements: Banzai::Filter::SanitizationFilter::LIMITED[:elements] - %w(pre code img ol ul li)
) )
def self.transform_context(context) def self.transform_context(context)
super(context).merge( super(context).merge(
# SanitizationFilter # SanitizationFilter
whitelist: WHITELIST allowlist: ALLOWLIST
) )
end end
end end
......
...@@ -35,8 +35,8 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do ...@@ -35,8 +35,8 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do
expect(Gitlab.config.asset_proxy.enabled).to be_truthy expect(Gitlab.config.asset_proxy.enabled).to be_truthy
expect(Gitlab.config.asset_proxy.secret_key).to eq 'shared-secret' expect(Gitlab.config.asset_proxy.secret_key).to eq 'shared-secret'
expect(Gitlab.config.asset_proxy.url).to eq 'https://assets.example.com' expect(Gitlab.config.asset_proxy.url).to eq 'https://assets.example.com'
expect(Gitlab.config.asset_proxy.whitelist).to eq %w(gitlab.com *.mydomain.com) expect(Gitlab.config.asset_proxy.allowlist).to eq %w(gitlab.com *.mydomain.com)
expect(Gitlab.config.asset_proxy.domain_regexp).to eq /^(gitlab\.com|.*?\.mydomain\.com)$/i expect(Gitlab.config.asset_proxy.domain_regexp).to eq(/^(gitlab\.com|.*?\.mydomain\.com)$/i)
end end
context 'when whitelist is empty' do context 'when whitelist is empty' do
...@@ -46,7 +46,7 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do ...@@ -46,7 +46,7 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do
described_class.initialize_settings described_class.initialize_settings
expect(Gitlab.config.asset_proxy.whitelist).to eq [Gitlab.config.gitlab.host] expect(Gitlab.config.asset_proxy.allowlist).to eq [Gitlab.config.gitlab.host]
end end
end end
end end
...@@ -56,8 +56,8 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do ...@@ -56,8 +56,8 @@ RSpec.describe Banzai::Filter::AssetProxyFilter do
stub_asset_proxy_setting(enabled: true) stub_asset_proxy_setting(enabled: true)
stub_asset_proxy_setting(secret_key: 'shared-secret') stub_asset_proxy_setting(secret_key: 'shared-secret')
stub_asset_proxy_setting(url: 'https://assets.example.com') stub_asset_proxy_setting(url: 'https://assets.example.com')
stub_asset_proxy_setting(whitelist: %W(gitlab.com *.mydomain.com #{Gitlab.config.gitlab.host})) stub_asset_proxy_setting(allowlist: %W(gitlab.com *.mydomain.com #{Gitlab.config.gitlab.host}))
stub_asset_proxy_setting(domain_regexp: described_class.compile_whitelist(Gitlab.config.asset_proxy.whitelist)) stub_asset_proxy_setting(domain_regexp: described_class.compile_allowlist(Gitlab.config.asset_proxy.allowlist))
@context = described_class.transform_context({}) @context = described_class.transform_context({})
end end
......
...@@ -5,9 +5,9 @@ require 'spec_helper' ...@@ -5,9 +5,9 @@ require 'spec_helper'
RSpec.describe Banzai::Filter::BroadcastMessageSanitizationFilter do RSpec.describe Banzai::Filter::BroadcastMessageSanitizationFilter do
include FilterSpecHelper include FilterSpecHelper
it_behaves_like 'default whitelist' it_behaves_like 'default allowlist'
describe 'custom whitelist' do describe 'custom allowlist' do
it_behaves_like 'XSS prevention' it_behaves_like 'XSS prevention'
it_behaves_like 'sanitize link' it_behaves_like 'sanitize link'
...@@ -26,19 +26,19 @@ RSpec.describe Banzai::Filter::BroadcastMessageSanitizationFilter do ...@@ -26,19 +26,19 @@ RSpec.describe Banzai::Filter::BroadcastMessageSanitizationFilter do
end end
context 'when `a` elements have `style` attribute' do context 'when `a` elements have `style` attribute' do
let(:whitelisted_style) { 'color: red; border: blue; background: green; padding: 10px; margin: 10px; text-decoration: underline;' } let(:allowed_style) { 'color: red; border: blue; background: green; padding: 10px; margin: 10px; text-decoration: underline;' }
context 'allows specific properties' do context 'allows specific properties' do
let(:exp) { %{<a href="#" style="#{whitelisted_style}">Stylish Link</a>} } let(:exp) { %{<a href="#" style="#{allowed_style}">Stylish Link</a>} }
it { is_expected.to eq(exp) } it { is_expected.to eq(exp) }
end end
it 'disallows other properties in `style` attribute on `a` elements' do it 'disallows other properties in `style` attribute on `a` elements' do
style = [whitelisted_style, 'position: fixed'].join(';') style = [allowed_style, 'position: fixed'].join(';')
doc = filter(%{<a href="#" style="#{style}">Stylish Link</a>}) doc = filter(%{<a href="#" style="#{style}">Stylish Link</a>})
expect(doc.at_css('a')['style']).to eq(whitelisted_style) expect(doc.at_css('a')['style']).to eq(allowed_style)
end end
end end
......
...@@ -5,31 +5,31 @@ require 'spec_helper' ...@@ -5,31 +5,31 @@ require 'spec_helper'
RSpec.describe Banzai::Filter::SanitizationFilter do RSpec.describe Banzai::Filter::SanitizationFilter do
include FilterSpecHelper include FilterSpecHelper
it_behaves_like 'default whitelist' it_behaves_like 'default allowlist'
describe 'custom whitelist' do describe 'custom allowlist' do
it_behaves_like 'XSS prevention' it_behaves_like 'XSS prevention'
it_behaves_like 'sanitize link' it_behaves_like 'sanitize link'
it 'customizes the whitelist only once' do it 'customizes the allowlist only once' do
instance = described_class.new('Foo') instance = described_class.new('Foo')
control_count = instance.whitelist[:transformers].size control_count = instance.allowlist[:transformers].size
3.times { instance.whitelist } 3.times { instance.allowlist }
expect(instance.whitelist[:transformers].size).to eq control_count expect(instance.allowlist[:transformers].size).to eq control_count
end end
it 'customizes the whitelist only once for different instances' do it 'customizes the allowlist only once for different instances' do
instance1 = described_class.new('Foo1') instance1 = described_class.new('Foo1')
instance2 = described_class.new('Foo2') instance2 = described_class.new('Foo2')
control_count = instance1.whitelist[:transformers].size control_count = instance1.allowlist[:transformers].size
instance1.whitelist instance1.allowlist
instance2.whitelist instance2.allowlist
expect(instance1.whitelist[:transformers].size).to eq control_count expect(instance1.allowlist[:transformers].size).to eq control_count
expect(instance2.whitelist[:transformers].size).to eq control_count expect(instance2.allowlist[:transformers].size).to eq control_count
end end
it 'sanitizes `class` attribute from all elements' do it 'sanitizes `class` attribute from all elements' do
......
...@@ -21,7 +21,7 @@ RSpec.describe Banzai::Pipeline::DescriptionPipeline do ...@@ -21,7 +21,7 @@ RSpec.describe Banzai::Pipeline::DescriptionPipeline do
stub_commonmark_sourcepos_disabled stub_commonmark_sourcepos_disabled
end end
it 'uses a limited whitelist' do it 'uses a limited allowlist' do
doc = parse('# Description') doc = parse('# Description')
expect(doc.strip).to eq 'Description' expect(doc.strip).to eq 'Description'
......
...@@ -176,8 +176,8 @@ RSpec.describe Banzai::Pipeline::GfmPipeline do ...@@ -176,8 +176,8 @@ RSpec.describe Banzai::Pipeline::GfmPipeline do
stub_asset_proxy_setting(enabled: true) stub_asset_proxy_setting(enabled: true)
stub_asset_proxy_setting(secret_key: 'shared-secret') stub_asset_proxy_setting(secret_key: 'shared-secret')
stub_asset_proxy_setting(url: 'https://assets.example.com') stub_asset_proxy_setting(url: 'https://assets.example.com')
stub_asset_proxy_setting(whitelist: %W(gitlab.com *.mydomain.com #{Gitlab.config.gitlab.host})) stub_asset_proxy_setting(allowlist: %W(gitlab.com *.mydomain.com #{Gitlab.config.gitlab.host}))
stub_asset_proxy_setting(domain_regexp: Banzai::Filter::AssetProxyFilter.compile_whitelist(Gitlab.config.asset_proxy.whitelist)) stub_asset_proxy_setting(domain_regexp: Banzai::Filter::AssetProxyFilter.compile_allowlist(Gitlab.config.asset_proxy.allowlist))
end end
it 'replaces a lazy loaded img src' do it 'replaces a lazy loaded img src' do
......
...@@ -17,12 +17,12 @@ RSpec.describe Gitlab::AssetProxy do ...@@ -17,12 +17,12 @@ RSpec.describe Gitlab::AssetProxy do
context 'when asset proxy is enabled' do context 'when asset proxy is enabled' do
before do before do
stub_asset_proxy_setting(whitelist: %w(gitlab.com *.mydomain.com)) stub_asset_proxy_setting(allowlist: %w(gitlab.com *.mydomain.com))
stub_asset_proxy_setting( stub_asset_proxy_setting(
enabled: true, enabled: true,
url: 'https://assets.example.com', url: 'https://assets.example.com',
secret_key: 'shared-secret', secret_key: 'shared-secret',
domain_regexp: Banzai::Filter::AssetProxyFilter.compile_whitelist(Gitlab.config.asset_proxy.whitelist) domain_regexp: Banzai::Filter::AssetProxyFilter.compile_allowlist(Gitlab.config.asset_proxy.allowlist)
) )
end end
......
# frozen_string_literal: true # frozen_string_literal: true
RSpec.shared_examples 'default whitelist' do RSpec.shared_examples 'default allowlist' do
it 'sanitizes tags that are not whitelisted' do it 'sanitizes tags that are not allowed' do
act = %q{<textarea>no inputs</textarea> and <blink>no blinks</blink>} act = %q{<textarea>no inputs</textarea> and <blink>no blinks</blink>}
exp = 'no inputs and no blinks' exp = 'no inputs and no blinks'
expect(filter(act).to_html).to eq exp expect(filter(act).to_html).to eq exp
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment