blob: 4f118544505d69f8441593a92feb1bb05ddaeb7c [file] [log] [blame]
/*
* Copyright (C) 2012 Igalia S.L.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "WebKitURISchemeRequest.h"
#include "APIData.h"
#include "WebKitPrivate.h"
#include "WebKitURISchemeRequestPrivate.h"
#include "WebKitWebContextPrivate.h"
#include "WebKitWebView.h"
#include "WebPageProxy.h"
#include <WebCore/GUniquePtrSoup.h>
#include <WebCore/HTTPParsers.h>
#include <WebCore/MIMETypeRegistry.h>
#include <WebCore/ResourceError.h>
#include <WebCore/URLSoup.h>
#include <libsoup/soup.h>
#include <wtf/glib/GRefPtr.h>
#include <wtf/glib/RunLoopSourcePriority.h>
#include <wtf/glib/WTFGType.h>
#include <wtf/text/CString.h>
using namespace WebKit;
using namespace WebCore;
/**
* SECTION: WebKitURISchemeRequest
* @Short_description: Represents a URI scheme request
* @Title: WebKitURISchemeRequest
*
* If you register a particular URI scheme in a #WebKitWebContext,
* using webkit_web_context_register_uri_scheme(), you have to provide
* a #WebKitURISchemeRequestCallback. After that, when a URI request
* is made with that particular scheme, your callback will be
* called. There you will be able to access properties such as the
* scheme, the URI and path, and the #WebKitWebView that initiated the
* request, and also finish the request with
* webkit_uri_scheme_request_finish().
*
*/
static const unsigned int gReadBufferSize = 8192;
struct _WebKitURISchemeRequestPrivate {
WebKitWebContext* webContext;
RefPtr<WebURLSchemeTask> task;
RefPtr<WebPageProxy> initiatingPage;
CString uri;
GUniquePtr<SoupURI> soupURI;
GRefPtr<GInputStream> stream;
uint64_t streamLength;
GRefPtr<GCancellable> cancellable;
char readBuffer[gReadBufferSize];
uint64_t bytesRead;
CString contentType;
};
WEBKIT_DEFINE_TYPE(WebKitURISchemeRequest, webkit_uri_scheme_request, G_TYPE_OBJECT)
static void webkit_uri_scheme_request_class_init(WebKitURISchemeRequestClass*)
{
}
WebKitURISchemeRequest* webkitURISchemeRequestCreate(WebKitWebContext* webContext, WebPageProxy& page, WebURLSchemeTask& task)
{
WebKitURISchemeRequest* request = WEBKIT_URI_SCHEME_REQUEST(g_object_new(WEBKIT_TYPE_URI_SCHEME_REQUEST, nullptr));
request->priv->webContext = webContext;
request->priv->task = &task;
request->priv->initiatingPage = &page;
return request;
}
void webkitURISchemeRequestCancel(WebKitURISchemeRequest* request)
{
g_cancellable_cancel(request->priv->cancellable.get());
}
/**
* webkit_uri_scheme_request_get_scheme:
* @request: a #WebKitURISchemeRequest
*
* Get the URI scheme of @request
*
* Returns: the URI scheme of @request
*/
const char* webkit_uri_scheme_request_get_scheme(WebKitURISchemeRequest* request)
{
g_return_val_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request), nullptr);
if (!request->priv->soupURI)
request->priv->soupURI = urlToSoupURI(request->priv->task->request().url());
return request->priv->soupURI->scheme;
}
/**
* webkit_uri_scheme_request_get_uri:
* @request: a #WebKitURISchemeRequest
*
* Get the URI of @request
*
* Returns: the full URI of @request
*/
const char* webkit_uri_scheme_request_get_uri(WebKitURISchemeRequest* request)
{
g_return_val_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request), nullptr);
if (request->priv->uri.isNull())
request->priv->uri = request->priv->task->request().url().string().utf8();
return request->priv->uri.data();
}
/**
* webkit_uri_scheme_request_get_path:
* @request: a #WebKitURISchemeRequest
*
* Get the URI path of @request
*
* Returns: the URI path of @request
*/
const char* webkit_uri_scheme_request_get_path(WebKitURISchemeRequest* request)
{
g_return_val_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request), nullptr);
if (!request->priv->soupURI)
request->priv->soupURI = urlToSoupURI(request->priv->task->request().url());
return request->priv->soupURI->path;
}
/**
* webkit_uri_scheme_request_get_web_view:
* @request: a #WebKitURISchemeRequest
*
* Get the #WebKitWebView that initiated the request.
*
* Returns: (transfer none): the #WebKitWebView that initiated @request.
*/
WebKitWebView* webkit_uri_scheme_request_get_web_view(WebKitURISchemeRequest* request)
{
g_return_val_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request), nullptr);
return webkitWebContextGetWebViewForPage(request->priv->webContext, request->priv->initiatingPage.get());
}
static void webkitURISchemeRequestReadCallback(GInputStream* inputStream, GAsyncResult* result, WebKitURISchemeRequest* schemeRequest)
{
GRefPtr<WebKitURISchemeRequest> request = adoptGRef(schemeRequest);
GUniqueOutPtr<GError> error;
gssize bytesRead = g_input_stream_read_finish(inputStream, result, &error.outPtr());
if (bytesRead == -1) {
webkit_uri_scheme_request_finish_error(request.get(), error.get());
return;
}
WebKitURISchemeRequestPrivate* priv = request->priv;
// Need to check the stream before proceeding as it can be cancelled if finish_error
// was previously call, which won't be detected by g_input_stream_read_finish().
if (!priv->stream)
return;
if (!priv->bytesRead) {
ResourceResponse response(priv->task->request().url(), extractMIMETypeFromMediaType(priv->contentType.data()), priv->streamLength, emptyString());
response.setTextEncodingName(extractCharsetFromMediaType(priv->contentType.data()));
if (response.mimeType().isEmpty())
response.setMimeType(MIMETypeRegistry::getMIMETypeForPath(response.url().path()));
priv->task->didReceiveResponse(response);
}
if (!bytesRead) {
priv->task->didComplete({ });
return;
}
priv->task->didReceiveData(SharedBuffer::create(priv->readBuffer, bytesRead));
priv->bytesRead += bytesRead;
g_input_stream_read_async(inputStream, priv->readBuffer, gReadBufferSize, RunLoopSourcePriority::AsyncIONetwork, priv->cancellable.get(),
reinterpret_cast<GAsyncReadyCallback>(webkitURISchemeRequestReadCallback), g_object_ref(request.get()));
}
/**
* webkit_uri_scheme_request_finish:
* @request: a #WebKitURISchemeRequest
* @stream: a #GInputStream to read the contents of the request
* @stream_length: the length of the stream or -1 if not known
* @content_type: (allow-none): the content type of the stream or %NULL if not known
*
* Finish a #WebKitURISchemeRequest by setting the contents of the request and its mime type.
*/
void webkit_uri_scheme_request_finish(WebKitURISchemeRequest* request, GInputStream* inputStream, gint64 streamLength, const gchar* contentType)
{
g_return_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request));
g_return_if_fail(G_IS_INPUT_STREAM(inputStream));
g_return_if_fail(streamLength == -1 || streamLength >= 0);
request->priv->stream = inputStream;
// We use -1 in the API for consistency with soup when the content length is not known, but 0 internally.
request->priv->streamLength = streamLength == -1 ? 0 : streamLength;
request->priv->cancellable = adoptGRef(g_cancellable_new());
request->priv->bytesRead = 0;
request->priv->contentType = contentType;
g_input_stream_read_async(inputStream, request->priv->readBuffer, gReadBufferSize, RunLoopSourcePriority::AsyncIONetwork, request->priv->cancellable.get(),
reinterpret_cast<GAsyncReadyCallback>(webkitURISchemeRequestReadCallback), g_object_ref(request));
}
/**
* webkit_uri_scheme_request_finish_error:
* @request: a #WebKitURISchemeRequest
* @error: a #GError that will be passed to the #WebKitWebView
*
* Finish a #WebKitURISchemeRequest with a #GError.
*
* Since: 2.2
*/
void webkit_uri_scheme_request_finish_error(WebKitURISchemeRequest* request, GError* error)
{
g_return_if_fail(WEBKIT_IS_URI_SCHEME_REQUEST(request));
g_return_if_fail(error);
WebKitURISchemeRequestPrivate* priv = request->priv;
priv->stream = nullptr;
ResourceError resourceError(g_quark_to_string(error->domain), toWebCoreError(error->code), priv->task->request().url(), String::fromUTF8(error->message));
priv->task->didComplete(resourceError);
}