5b80376543544d2e928927301414500976192f7f
[WebKit-https.git] / Source / WebCore / css / StyleRuleImport.cpp
1 /*
2  * (C) 1999-2003 Lars Knoll (knoll@kde.org)
3  * (C) 2002-2003 Dirk Mueller (mueller@kde.org)
4  * Copyright (C) 2002-2017 Apple Inc. All rights reserved.
5  *
6  * This library is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Library General Public
8  * License as published by the Free Software Foundation; either
9  * version 2 of the License, or (at your option) any later version.
10  *
11  * This library is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Library General Public License for more details.
15  *
16  * You should have received a copy of the GNU Library General Public License
17  * along with this library; see the file COPYING.LIB.  If not, write to
18  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19  * Boston, MA 02110-1301, USA.
20  */
21
22 #include "config.h"
23 #include "StyleRuleImport.h"
24
25 #include "CSSStyleSheet.h"
26 #include "CachedCSSStyleSheet.h"
27 #include "CachedResourceLoader.h"
28 #include "CachedResourceRequest.h"
29 #include "CachedResourceRequestInitiators.h"
30 #include "Document.h"
31 #include "MediaList.h"
32 #include "MediaQueryParser.h"
33 #include "SecurityOrigin.h"
34 #include "StyleSheetContents.h"
35 #include <wtf/StdLibExtras.h>
36
37 namespace WebCore {
38
39 Ref<StyleRuleImport> StyleRuleImport::create(const String& href, Ref<MediaQuerySet>&& media)
40 {
41     return adoptRef(*new StyleRuleImport(href, WTFMove(media)));
42 }
43
44 StyleRuleImport::StyleRuleImport(const String& href, Ref<MediaQuerySet>&& media)
45     : StyleRuleBase(Import)
46     , m_parentStyleSheet(0)
47     , m_styleSheetClient(this)
48     , m_strHref(href)
49     , m_mediaQueries(WTFMove(media))
50     , m_cachedSheet(0)
51     , m_loading(false)
52 {
53     if (!m_mediaQueries)
54         m_mediaQueries = MediaQuerySet::create(String(), MediaQueryParserContext());
55 }
56
57 StyleRuleImport::~StyleRuleImport()
58 {
59     if (m_styleSheet)
60         m_styleSheet->clearOwnerRule();
61     if (m_cachedSheet)
62         m_cachedSheet->removeClient(m_styleSheetClient);
63 }
64
65 void StyleRuleImport::setCSSStyleSheet(const String& href, const URL& baseURL, const String& charset, const CachedCSSStyleSheet* cachedStyleSheet)
66 {
67     if (m_styleSheet)
68         m_styleSheet->clearOwnerRule();
69
70     CSSParserContext context = m_parentStyleSheet ? m_parentStyleSheet->parserContext() : HTMLStandardMode;
71     context.charset = charset;
72     if (!baseURL.isNull())
73         context.baseURL = baseURL;
74
75     Document* document = m_parentStyleSheet ? m_parentStyleSheet->singleOwnerDocument() : nullptr;
76     m_styleSheet = StyleSheetContents::create(this, href, context);
77     m_styleSheet->parseAuthorStyleSheet(cachedStyleSheet, document ? &document->securityOrigin() : nullptr);
78
79     m_loading = false;
80
81     if (m_parentStyleSheet) {
82         m_parentStyleSheet->notifyLoadedSheet(cachedStyleSheet);
83         m_parentStyleSheet->checkLoaded();
84     }
85 }
86
87 bool StyleRuleImport::isLoading() const
88 {
89     return m_loading || (m_styleSheet && m_styleSheet->isLoading());
90 }
91
92 void StyleRuleImport::requestStyleSheet()
93 {
94     if (!m_parentStyleSheet)
95         return;
96     Document* document = m_parentStyleSheet->singleOwnerDocument();
97     if (!document)
98         return;
99
100     URL absURL;
101     if (!m_parentStyleSheet->baseURL().isNull())
102         // use parent styleheet's URL as the base URL
103         absURL = URL(m_parentStyleSheet->baseURL(), m_strHref);
104     else
105         absURL = document->completeURL(m_strHref);
106
107     // Check for a cycle in our import chain.  If we encounter a stylesheet
108     // in our parent chain with the same URL, then just bail.
109     StyleSheetContents* rootSheet = m_parentStyleSheet;
110     for (StyleSheetContents* sheet = m_parentStyleSheet; sheet; sheet = sheet->parentStyleSheet()) {
111         if (equalIgnoringFragmentIdentifier(absURL, sheet->baseURL())
112             || equalIgnoringFragmentIdentifier(absURL, document->completeURL(sheet->originalURL())))
113             return;
114         rootSheet = sheet;
115     }
116
117     // FIXME: Skip Content Security Policy check when stylesheet is in a user agent shadow tree.
118     // See <https://bugs.webkit.org/show_bug.cgi?id=146663>.
119     CachedResourceRequest request(absURL, CachedResourceLoader::defaultCachedResourceOptions(), std::nullopt, String(m_parentStyleSheet->charset()));
120     request.setInitiator(cachedResourceRequestInitiators().css);
121     if (m_cachedSheet)
122         m_cachedSheet->removeClient(m_styleSheetClient);
123     if (m_parentStyleSheet->isUserStyleSheet()) {
124         request.setOptions(ResourceLoaderOptions(DoNotSendCallbacks, SniffContent, BufferData, StoredCredentialsPolicy::Use, ClientCredentialPolicy::MayAskClientForCredentials, FetchOptions::Credentials::Include, SkipSecurityCheck, FetchOptions::Mode::NoCors, DoNotIncludeCertificateInfo, ContentSecurityPolicyImposition::SkipPolicyCheck, DefersLoadingPolicy::AllowDefersLoading, CachingPolicy::AllowCaching));
125         m_cachedSheet = document->cachedResourceLoader().requestUserCSSStyleSheet(WTFMove(request));
126     } else
127         m_cachedSheet = document->cachedResourceLoader().requestCSSStyleSheet(WTFMove(request)).value_or(nullptr);
128     if (m_cachedSheet) {
129         // if the import rule is issued dynamically, the sheet may be
130         // removed from the pending sheet count, so let the doc know
131         // the sheet being imported is pending.
132         if (m_parentStyleSheet && m_parentStyleSheet->loadCompleted() && rootSheet == m_parentStyleSheet)
133             m_parentStyleSheet->startLoadingDynamicSheet();
134         m_loading = true;
135         m_cachedSheet->addClient(m_styleSheetClient);
136     }
137 }
138
139 } // namespace WebCore