From 3c4b86edd6745275da22d92290872da202d73e64 Mon Sep 17 00:00:00 2001 From: Neil Matatall Date: Tue, 21 Jan 2020 08:44:58 -1000 Subject: [PATCH] escape semicolons by replacing them with spaces See https://github.com/twitter/secure_headers/issues/418 --- lib/secure_headers/headers/content_security_policy.rb | 11 ++++++++--- .../headers/content_security_policy_spec.rb | 5 +++++ 2 files changed, 13 insertions(+), 3 deletions(-) diff --git a/lib/secure_headers/headers/content_security_policy.rb b/lib/secure_headers/headers/content_security_policy.rb index abac5c5b..9f50623b 100644 --- a/lib/secure_headers/headers/content_security_policy.rb +++ b/lib/secure_headers/headers/content_security_policy.rb @@ -103,10 +103,15 @@ def build_media_type_list_directive(directive) # Returns a string representing a directive. def build_source_list_directive(directive) source_list = @config.directive_value(directive) - if source_list != OPT_OUT && source_list && source_list.any? - normalized_source_list = minify_source_list(directive, source_list) - [symbol_to_hyphen_case(directive), normalized_source_list].join(" ") + minified_source_list = minify_source_list(directive, source_list).join(" ") + + if minified_source_list.include?(";") + Kernel.warn("#{directive} contains a ; in '#{minified_source_list}' which will raise an error in future versions. It has been replaced with a blank space.") + end + + escaped_source_list = minified_source_list.gsub(";", " ") + [symbol_to_hyphen_case(directive), escaped_source_list].join(" ").strip end end diff --git a/spec/lib/secure_headers/headers/content_security_policy_spec.rb b/spec/lib/secure_headers/headers/content_security_policy_spec.rb index fface8ca..32b9b44d 100644 --- a/spec/lib/secure_headers/headers/content_security_policy_spec.rb +++ b/spec/lib/secure_headers/headers/content_security_policy_spec.rb @@ -28,6 +28,11 @@ module SecureHeaders expect(ContentSecurityPolicy.new.value).to eq("default-src https:; form-action 'self'; img-src https: data: 'self'; object-src 'none'; script-src https:; style-src 'self' 'unsafe-inline' https:") end + it "deprecates and escapes semicolons in directive source lists" do + expect(Kernel).to receive(:warn).with("frame_ancestors contains a ; in 'google.com;script-src *;.;' which will raise an error in future versions. It has been replaced with a blank space.") + expect(ContentSecurityPolicy.new(frame_ancestors: %w(https://google.com;script-src https://*;.;)).value).to eq("frame-ancestors google.com script-src * .") + end + it "discards 'none' values if any other source expressions are present" do csp = ContentSecurityPolicy.new(default_opts.merge(child_src: %w('self' 'none'))) expect(csp.value).not_to include("'none'")