blob: 569b98264b45b8878b5872c3312fe13842c520c7 [file] [log] [blame]
/*
* Copyright (C) 2006-2018 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
* OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "WebURLResponse.h"
#include "WebKitDLL.h"
#include "WebKit.h"
#include "COMPropertyBag.h"
#include "MarshallingHelpers.h"
#if USE(CG)
#include <CoreGraphics/CoreGraphics.h>
#endif
#if USE(CFURLCONNECTION)
#include <pal/spi/cf/CFNetworkSPI.h>
#endif
#include <WebCore/BString.h>
#include <WebCore/LocalizedStrings.h>
#include <WebCore/ResourceError.h>
#include <WebCore/ResourceHandle.h>
#include <shlobj.h>
#include <shlwapi.h>
#include <wchar.h>
#include <wtf/URL.h>
using namespace WebCore;
static String localizedShortDescriptionForStatusCode(int statusCode)
{
String result;
if (statusCode < 100 || statusCode >= 600)
result = WEB_UI_STRING("server error", "HTTP result code string");
else if (statusCode >= 100 && statusCode <= 199) {
switch (statusCode) {
case 100:
result = WEB_UI_STRING("continue", "HTTP result code string");
break;
case 101:
result = WEB_UI_STRING("switching protocols", "HTTP result code string");
break;
default:
result = WEB_UI_STRING("informational", "HTTP result code string");
break;
}
} else if (statusCode >= 200 && statusCode <= 299) {
switch (statusCode) {
case 200:
result = WEB_UI_STRING("no error", "HTTP result code string");
break;
case 201:
result = WEB_UI_STRING("created", "HTTP result code string");
break;
case 202:
result = WEB_UI_STRING("accepted", "HTTP result code string");
break;
case 203:
result = WEB_UI_STRING("non-authoritative information", "HTTP result code string");
break;
case 204:
result = WEB_UI_STRING("no content", "HTTP result code string");
break;
case 205:
result = WEB_UI_STRING("reset content", "HTTP result code string");
break;
case 206:
result = WEB_UI_STRING("partial content", "HTTP result code string");
break;
default:
result = WEB_UI_STRING("success", "HTTP result code string");
break;
}
} else if (statusCode >= 300 && statusCode <= 399) {
switch (statusCode) {
case 300:
result = WEB_UI_STRING("multiple choices", "HTTP result code string");
break;
case 301:
result = WEB_UI_STRING("moved permanently", "HTTP result code string");
break;
case 302:
result = WEB_UI_STRING("found", "HTTP result code string");
break;
case 303:
result = WEB_UI_STRING("see other", "HTTP result code string");
break;
case 304:
result = WEB_UI_STRING("not modified", "HTTP result code string");
break;
case 305:
result = WEB_UI_STRING("needs proxy", "HTTP result code string");
break;
case 307:
result = WEB_UI_STRING("temporarily redirected", "HTTP result code string");
break;
case 306: // 306 status code unused in HTTP
default:
result = WEB_UI_STRING("redirected", "HTTP result code string");
break;
}
} else if (statusCode >= 400 && statusCode <= 499) {
switch (statusCode) {
case 400:
result = WEB_UI_STRING("bad request", "HTTP result code string");
break;
case 401:
result = WEB_UI_STRING("unauthorized", "HTTP result code string");
break;
case 402:
result = WEB_UI_STRING("payment required", "HTTP result code string");
break;
case 403:
result = WEB_UI_STRING("forbidden", "HTTP result code string");
break;
case 404:
result = WEB_UI_STRING("not found", "HTTP result code string");
break;
case 405:
result = WEB_UI_STRING("method not allowed", "HTTP result code string");
break;
case 406:
result = WEB_UI_STRING("unacceptable", "HTTP result code string");
break;
case 407:
result = WEB_UI_STRING("proxy authentication required", "HTTP result code string");
break;
case 408:
result = WEB_UI_STRING("request timed out", "HTTP result code string");
break;
case 409:
result = WEB_UI_STRING("conflict", "HTTP result code string");
break;
case 410:
result = WEB_UI_STRING("no longer exists", "HTTP result code string");
break;
case 411:
result = WEB_UI_STRING("length required", "HTTP result code string");
break;
case 412:
result = WEB_UI_STRING("precondition failed", "HTTP result code string");
break;
case 413:
result = WEB_UI_STRING("request too large", "HTTP result code string");
break;
case 414:
result = WEB_UI_STRING("requested URL too long", "HTTP result code string");
break;
case 415:
result = WEB_UI_STRING("unsupported media type", "HTTP result code string");
break;
case 416:
result = WEB_UI_STRING("requested range not satisfiable", "HTTP result code string");
break;
case 417:
result = WEB_UI_STRING("expectation failed", "HTTP result code string");
break;
default:
result = WEB_UI_STRING("client error", "HTTP result code string");
break;
}
} else if (statusCode >= 500 && statusCode <= 599) {
switch (statusCode) {
case 500:
result = WEB_UI_STRING("internal server error", "HTTP result code string");
break;
case 501:
result = WEB_UI_STRING("unimplemented", "HTTP result code string");
break;
case 502:
result = WEB_UI_STRING("bad gateway", "HTTP result code string");
break;
case 503:
result = WEB_UI_STRING("service unavailable", "HTTP result code string");
break;
case 504:
result = WEB_UI_STRING("gateway timed out", "HTTP result code string");
break;
case 505:
result = WEB_UI_STRING("unsupported version", "HTTP result code string");
break;
default:
result = WEB_UI_STRING("server error", "HTTP result code string");
break;
}
}
return result;
}
// IWebURLResponse ----------------------------------------------------------------
WebURLResponse::WebURLResponse()
{
gClassCount++;
gClassNameCount().add("WebURLResponse");
}
WebURLResponse::~WebURLResponse()
{
gClassCount--;
gClassNameCount().remove("WebURLResponse");
}
WebURLResponse* WebURLResponse::createInstance()
{
WebURLResponse* instance = new WebURLResponse();
// fake an http response - so it has the IWebHTTPURLResponse interface
instance->m_response = ResourceResponse(WTF::URL({ }, "http://"), String(), 0, String());
instance->AddRef();
return instance;
}
WebURLResponse* WebURLResponse::createInstance(const ResourceResponse& response)
{
if (response.isNull())
return 0;
WebURLResponse* instance = new WebURLResponse();
instance->AddRef();
instance->m_response = response;
return instance;
}
// IUnknown -------------------------------------------------------------------
HRESULT WebURLResponse::QueryInterface(_In_ REFIID riid, _COM_Outptr_ void** ppvObject)
{
if (!ppvObject)
return E_POINTER;
*ppvObject = nullptr;
if (IsEqualGUID(riid, IID_IUnknown))
*ppvObject = static_cast<IWebURLResponse*>(this);
else if (IsEqualGUID(riid, __uuidof(this)))
*ppvObject = this;
else if (IsEqualGUID(riid, IID_IWebURLResponse))
*ppvObject = static_cast<IWebURLResponse*>(this);
else if (IsEqualGUID(riid, IID_IWebURLResponsePrivate))
*ppvObject = static_cast<IWebURLResponsePrivate*>(this);
else if (m_response.isHTTP() && IsEqualGUID(riid, IID_IWebHTTPURLResponse))
*ppvObject = static_cast<IWebHTTPURLResponse*>(this);
else
return E_NOINTERFACE;
AddRef();
return S_OK;
}
ULONG WebURLResponse::AddRef()
{
return ++m_refCount;
}
ULONG WebURLResponse::Release()
{
ULONG newRef = --m_refCount;
if (!newRef)
delete(this);
return newRef;
}
// IWebURLResponse --------------------------------------------------------------------
HRESULT WebURLResponse::expectedContentLength(_Out_ long long* result)
{
if (!result)
return E_POINTER;
*result = m_response.expectedContentLength();
return S_OK;
}
HRESULT WebURLResponse::initWithURL(_In_ BSTR url, _In_ BSTR mimeType, int expectedContentLength, _In_ BSTR textEncodingName)
{
m_response = ResourceResponse(MarshallingHelpers::BSTRToKURL(url), String(mimeType), expectedContentLength, String(textEncodingName));
return S_OK;
}
HRESULT WebURLResponse::MIMEType(__deref_opt_out BSTR* result)
{
if (!result)
return E_POINTER;
BString mimeType(m_response.mimeType());
*result = mimeType.release();
if (!m_response.mimeType().isNull() && !*result)
return E_OUTOFMEMORY;
return S_OK;
}
HRESULT WebURLResponse::suggestedFilename(__deref_opt_out BSTR* result)
{
if (!result) {
ASSERT_NOT_REACHED();
return E_POINTER;
}
*result = nullptr;
if (m_response.url().isEmpty())
return E_FAIL;
*result = BString(m_response.suggestedFilename()).release();
return S_OK;
}
HRESULT WebURLResponse::textEncodingName(__deref_opt_out BSTR* result)
{
if (!result)
return E_INVALIDARG;
BString textEncodingName(m_response.textEncodingName());
*result = textEncodingName.release();
if (!m_response.textEncodingName().isNull() && !*result)
return E_OUTOFMEMORY;
return S_OK;
}
HRESULT WebURLResponse::URL(__deref_opt_out BSTR* result)
{
if (!result)
return E_INVALIDARG;
BString url(m_response.url().string());
*result = url.release();
if (!m_response.url().isEmpty() && !*result)
return E_OUTOFMEMORY;
return S_OK;
}
// IWebHTTPURLResponse --------------------------------------------------------
HRESULT WebURLResponse::allHeaderFields(_COM_Outptr_opt_ IPropertyBag** headerFields)
{
ASSERT(m_response.isHTTP());
if (!headerFields)
return E_POINTER;
HashMap<String, String, ASCIICaseInsensitiveHash> fields;
for (const auto& keyValuePair : m_response.httpHeaderFields())
fields.add(keyValuePair.key, keyValuePair.value);
*headerFields = COMPropertyBag<String, String, ASCIICaseInsensitiveHash>::adopt(fields);
return S_OK;
}
HRESULT WebURLResponse::localizedStringForStatusCode(int statusCode, __deref_opt_out BSTR* statusString)
{
ASSERT(m_response.isHTTP());
if (!statusString)
return E_POINTER;
*statusString = nullptr;
const String& statusText = localizedShortDescriptionForStatusCode(statusCode);
if (!statusText)
return E_FAIL;
*statusString = BString(statusText).release();
return S_OK;
}
HRESULT WebURLResponse::statusCode(_Out_ int* statusCode)
{
ASSERT(m_response.isHTTP());
if (statusCode)
*statusCode = m_response.httpStatusCode();
return S_OK;
}
HRESULT WebURLResponse::isAttachment(_Out_ BOOL* attachment)
{
if (!attachment)
return E_POINTER;
*attachment = m_response.isAttachment();
return S_OK;
}
HRESULT WebURLResponse::sslPeerCertificate(_Out_ ULONG_PTR* result)
{
if (!result)
return E_POINTER;
*result = 0;
#if USE(CFURLCONNECTION)
CFDictionaryRef dict = certificateDictionary();
if (!dict)
return E_FAIL;
const void* data = ResourceError::getSSLPeerCertificateDataBytePtr(dict);
if (!data)
return E_FAIL;
*result = reinterpret_cast<ULONG_PTR>(const_cast<void*>(data));
#endif
return *result ? S_OK : E_FAIL;
}
// WebURLResponse -------------------------------------------------------------
HRESULT WebURLResponse::suggestedFileExtension(BSTR* result)
{
if (!result)
return E_POINTER;
*result = nullptr;
if (m_response.mimeType().isEmpty())
return E_FAIL;
BString mimeType(m_response.mimeType());
HKEY key;
LONG err = RegOpenKeyEx(HKEY_CLASSES_ROOT, TEXT("MIME\\Database\\Content Type"), 0, KEY_QUERY_VALUE, &key);
if (!err) {
HKEY subKey;
err = RegOpenKeyEx(key, mimeType, 0, KEY_QUERY_VALUE, &subKey);
if (!err) {
DWORD keyType = REG_SZ;
WCHAR extension[MAX_PATH];
DWORD keySize = sizeof(extension)/sizeof(extension[0]);
err = RegQueryValueEx(subKey, TEXT("Extension"), 0, &keyType, (LPBYTE)extension, &keySize);
if (!err && keyType != REG_SZ)
err = ERROR_INVALID_DATA;
if (err) {
// fallback handlers
if (!wcscmp(mimeType, L"text/html")) {
wcscpy(extension, L".html");
err = 0;
} else if (!wcscmp(mimeType, L"application/xhtml+xml")) {
wcscpy(extension, L".xhtml");
err = 0;
} else if (!wcscmp(mimeType, L"image/svg+xml")) {
wcscpy(extension, L".svg");
err = 0;
}
}
if (!err) {
*result = SysAllocString(extension);
if (!*result)
err = ERROR_OUTOFMEMORY;
}
RegCloseKey(subKey);
}
RegCloseKey(key);
}
return HRESULT_FROM_WIN32(err);
}
const ResourceResponse& WebURLResponse::resourceResponse() const
{
return m_response;
}
#if USE(CFURLCONNECTION)
CFDictionaryRef WebURLResponse::certificateDictionary() const
{
if (m_SSLCertificateInfo)
return m_SSLCertificateInfo.get();
CFURLResponseRef cfResponse = m_response.cfURLResponse();
if (!cfResponse)
return nullptr;
m_SSLCertificateInfo = _CFURLResponseGetSSLCertificateContext(cfResponse);
return m_SSLCertificateInfo.get();
}
#endif