blob: 2fcde837e88b691ee4073c580e1ce9c8423735d5 [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "core/frame/csp/CSPDirectiveList.h"
#include "core/frame/csp/ContentSecurityPolicy.h"
#include "core/frame/csp/SourceListDirective.h"
#include "platform/network/ContentSecurityPolicyParsers.h"
#include "platform/network/ResourceRequest.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "wtf/text/StringOperators.h"
#include "wtf/text/WTFString.h"
namespace blink {
class CSPDirectiveListTest : public ::testing::Test {
public:
CSPDirectiveListTest() : csp(ContentSecurityPolicy::create()) {}
CSPDirectiveList* createList(const String& list,
ContentSecurityPolicyHeaderType type) {
Vector<UChar> characters;
list.appendTo(characters);
const UChar* begin = characters.data();
const UChar* end = begin + characters.size();
return CSPDirectiveList::create(csp, begin, end, type,
ContentSecurityPolicyHeaderSourceHTTP);
}
protected:
Persistent<ContentSecurityPolicy> csp;
};
TEST_F(CSPDirectiveListTest, Header) {
struct TestCase {
const char* list;
const char* expected;
} cases[] = {{"script-src 'self'", "script-src 'self'"},
{" script-src 'self' ", "script-src 'self'"},
{"\t\tscript-src 'self'", "script-src 'self'"},
{"script-src 'self' \t", "script-src 'self'"}};
for (const auto& test : cases) {
Member<CSPDirectiveList> directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(test.expected, directiveList->header());
directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->header());
}
}
TEST_F(CSPDirectiveListTest, IsMatchingNoncePresent) {
struct TestCase {
const char* list;
const char* nonce;
bool expected;
} cases[] = {
{"script-src 'self'", "yay", false},
{"script-src 'self'", "boo", false},
{"script-src 'nonce-yay'", "yay", true},
{"script-src 'nonce-yay'", "boo", false},
{"script-src 'nonce-yay' 'nonce-boo'", "yay", true},
{"script-src 'nonce-yay' 'nonce-boo'", "boo", true},
// Falls back to 'default-src'
{"default-src 'nonce-yay'", "yay", true},
{"default-src 'nonce-yay'", "boo", false},
{"default-src 'nonce-boo'; script-src 'nonce-yay'", "yay", true},
{"default-src 'nonce-boo'; script-src 'nonce-yay'", "boo", false},
// Unrelated directives do not affect result
{"style-src 'nonce-yay'", "yay", false},
{"style-src 'nonce-yay'", "boo", false},
};
for (const auto& test : cases) {
// Report-only
Member<CSPDirectiveList> directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeReport);
Member<SourceListDirective> scriptSrc =
directiveList->operativeDirective(directiveList->m_scriptSrc.get());
EXPECT_EQ(test.expected,
directiveList->isMatchingNoncePresent(scriptSrc, test.nonce));
// Empty/null strings are always not present, regardless of the policy.
EXPECT_FALSE(directiveList->isMatchingNoncePresent(scriptSrc, ""));
EXPECT_FALSE(directiveList->isMatchingNoncePresent(scriptSrc, String()));
// Enforce
directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeEnforce);
scriptSrc =
directiveList->operativeDirective(directiveList->m_scriptSrc.get());
EXPECT_EQ(test.expected,
directiveList->isMatchingNoncePresent(scriptSrc, test.nonce));
// Empty/null strings are always not present, regardless of the policy.
EXPECT_FALSE(directiveList->isMatchingNoncePresent(scriptSrc, ""));
EXPECT_FALSE(directiveList->isMatchingNoncePresent(scriptSrc, String()));
}
}
TEST_F(CSPDirectiveListTest, AllowScriptFromSourceNoNonce) {
struct TestCase {
const char* list;
const char* url;
bool expected;
} cases[] = {
{"script-src https://example.com", "https://example.com/script.js", true},
{"script-src https://example.com/", "https://example.com/script.js",
true},
{"script-src https://example.com/",
"https://example.com/script/script.js", true},
{"script-src https://example.com/script", "https://example.com/script.js",
false},
{"script-src https://example.com/script",
"https://example.com/script/script.js", false},
{"script-src https://example.com/script/",
"https://example.com/script.js", false},
{"script-src https://example.com/script/",
"https://example.com/script/script.js", true},
{"script-src https://example.com", "https://not.example.com/script.js",
false},
{"script-src https://*.example.com", "https://not.example.com/script.js",
true},
{"script-src https://*.example.com", "https://example.com/script.js",
false},
// Falls back to default-src:
{"default-src https://example.com", "https://example.com/script.js",
true},
{"default-src https://example.com/", "https://example.com/script.js",
true},
{"default-src https://example.com/",
"https://example.com/script/script.js", true},
{"default-src https://example.com/script",
"https://example.com/script.js", false},
{"default-src https://example.com/script",
"https://example.com/script/script.js", false},
{"default-src https://example.com/script/",
"https://example.com/script.js", false},
{"default-src https://example.com/script/",
"https://example.com/script/script.js", true},
{"default-src https://example.com", "https://not.example.com/script.js",
false},
{"default-src https://*.example.com", "https://not.example.com/script.js",
true},
{"default-src https://*.example.com", "https://example.com/script.js",
false},
};
for (const auto& test : cases) {
SCOPED_TRACE(testing::Message() << "List: `" << test.list << "`, URL: `"
<< test.url << "`");
KURL scriptSrc = KURL(KURL(), test.url);
// Report-only
Member<CSPDirectiveList> directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
scriptSrc, String(), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Enforce
directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
scriptSrc, String(), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
}
}
TEST_F(CSPDirectiveListTest, AllowFromSourceWithNonce) {
struct TestCase {
const char* list;
const char* url;
const char* nonce;
bool expected;
} cases[] = {
// Doesn't affect lists without nonces:
{"https://example.com", "https://example.com/file", "yay", true},
{"https://example.com", "https://example.com/file", "boo", true},
{"https://example.com", "https://example.com/file", "", true},
{"https://example.com", "https://not.example.com/file", "yay", false},
{"https://example.com", "https://not.example.com/file", "boo", false},
{"https://example.com", "https://not.example.com/file", "", false},
// Doesn't affect URLs that match the whitelist.
{"https://example.com 'nonce-yay'", "https://example.com/file", "yay",
true},
{"https://example.com 'nonce-yay'", "https://example.com/file", "boo",
true},
{"https://example.com 'nonce-yay'", "https://example.com/file", "", true},
// Does affect URLs that don't.
{"https://example.com 'nonce-yay'", "https://not.example.com/file", "yay",
true},
{"https://example.com 'nonce-yay'", "https://not.example.com/file", "boo",
false},
{"https://example.com 'nonce-yay'", "https://not.example.com/file", "",
false},
};
for (const auto& test : cases) {
SCOPED_TRACE(testing::Message() << "List: `" << test.list << "`, URL: `"
<< test.url << "`");
KURL resource = KURL(KURL(), test.url);
// Report-only 'script-src'
Member<CSPDirectiveList> directiveList =
createList(String("script-src ") + test.list,
ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
resource, String(test.nonce), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Enforce 'script-src'
directiveList = createList(String("script-src ") + test.list,
ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
resource, String(test.nonce), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Report-only 'style-src'
directiveList = createList(String("style-src ") + test.list,
ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(test.expected, directiveList->allowStyleFromSource(
resource, String(test.nonce),
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Enforce 'style-src'
directiveList = createList(String("style-src ") + test.list,
ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->allowStyleFromSource(
resource, String(test.nonce),
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Report-only 'style-src'
directiveList = createList(String("default-src ") + test.list,
ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
resource, String(test.nonce), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
EXPECT_EQ(test.expected, directiveList->allowStyleFromSource(
resource, String(test.nonce),
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Enforce 'style-src'
directiveList = createList(String("default-src ") + test.list,
ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->allowScriptFromSource(
resource, String(test.nonce), ParserInserted,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
EXPECT_EQ(test.expected, directiveList->allowStyleFromSource(
resource, String(test.nonce),
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
}
}
TEST_F(CSPDirectiveListTest, allowRequestWithoutIntegrity) {
struct TestCase {
const char* list;
const char* url;
const WebURLRequest::RequestContext context;
bool expected;
} cases[] = {
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
// Extra WSP
{"require-sri-for script script ", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for style script", "https://example.com/file",
WebURLRequest::RequestContextStyle, false},
{"require-sri-for style script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for style script", "https://example.com/file",
WebURLRequest::RequestContextImport, false},
{"require-sri-for style script", "https://example.com/file",
WebURLRequest::RequestContextImage, true},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextAudio, true},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextImport, false},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextServiceWorker, false},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextSharedWorker, false},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextWorker, false},
{"require-sri-for script", "https://example.com/file",
WebURLRequest::RequestContextStyle, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextAudio, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextScript, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextImport, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextServiceWorker, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextSharedWorker, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextWorker, true},
{"require-sri-for style", "https://example.com/file",
WebURLRequest::RequestContextStyle, false},
// Multiple tokens
{"require-sri-for script style", "https://example.com/file",
WebURLRequest::RequestContextStyle, false},
{"require-sri-for script style", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for script style", "https://example.com/file",
WebURLRequest::RequestContextImport, false},
{"require-sri-for script style", "https://example.com/file",
WebURLRequest::RequestContextImage, true},
// Matching is case-insensitive
{"require-sri-for Script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
// Unknown tokens do not affect result
{"require-sri-for blabla12 as", "https://example.com/file",
WebURLRequest::RequestContextScript, true},
{"require-sri-for blabla12 as script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for script style img", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for script style img", "https://example.com/file",
WebURLRequest::RequestContextImport, false},
{"require-sri-for script style img", "https://example.com/file",
WebURLRequest::RequestContextStyle, false},
{"require-sri-for script style img", "https://example.com/file",
WebURLRequest::RequestContextImage, true},
// Empty token list has no effect
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextScript, true},
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextImport, true},
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextStyle, true},
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextServiceWorker, true},
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextSharedWorker, true},
{"require-sri-for ", "https://example.com/file",
WebURLRequest::RequestContextWorker, true},
// Order does not matter
{"require-sri-for a b script", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
{"require-sri-for a script b", "https://example.com/file",
WebURLRequest::RequestContextScript, false},
};
for (const auto& test : cases) {
KURL resource = KURL(KURL(), test.url);
// Report-only
Member<CSPDirectiveList> directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeReport);
EXPECT_EQ(true, directiveList->allowRequestWithoutIntegrity(
test.context, resource,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
// Enforce
directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.expected, directiveList->allowRequestWithoutIntegrity(
test.context, resource,
ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
}
}
TEST_F(CSPDirectiveListTest, workerSrc) {
struct TestCase {
const char* list;
bool allowed;
} cases[] = {
{"worker-src 'none'", false},
{"worker-src http://not.example.test", false},
{"worker-src https://example.test", true},
{"default-src *; worker-src 'none'", false},
{"default-src *; worker-src http://not.example.test", false},
{"default-src *; worker-src https://example.test", true},
{"child-src *; worker-src 'none'", false},
{"child-src *; worker-src http://not.example.test", false},
{"child-src *; worker-src https://example.test", true},
{"default-src *; child-src *; worker-src 'none'", false},
{"default-src *; child-src *; worker-src http://not.example.test", false},
{"default-src *; child-src *; worker-src https://example.test", true},
// Fallback to child-src.
{"child-src 'none'", false},
{"child-src http://not.example.test", false},
{"child-src https://example.test", true},
{"default-src *; child-src 'none'", false},
{"default-src *; child-src http://not.example.test", false},
{"default-src *; child-src https://example.test", true},
// Fallback to default-src.
{"default-src 'none'", false},
{"default-src http://not.example.test", false},
{"default-src https://example.test", true},
};
for (const auto& test : cases) {
SCOPED_TRACE(test.list);
KURL resource = KURL(KURL(), "https://example.test/worker.js");
Member<CSPDirectiveList> directiveList =
createList(test.list, ContentSecurityPolicyHeaderTypeEnforce);
EXPECT_EQ(test.allowed,
directiveList->allowWorkerFromSource(
resource, ResourceRequest::RedirectStatus::NoRedirect,
ContentSecurityPolicy::SuppressReport));
}
}
} // namespace blink