gecko-dev/netwerk/protocol/viewsource/nsViewSourceHandler.cpp

200 строки
5.2 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/* vim:set ts=4 sw=4 sts=4 et: */
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "nsViewSourceHandler.h"
#include "nsViewSourceChannel.h"
#include "nsNetUtil.h"
#include "nsSimpleNestedURI.h"
#define VIEW_SOURCE "view-source"
#define DEFAULT_FLAGS (URI_NORELATIVE | URI_NOAUTH | URI_DANGEROUS_TO_LOAD | URI_NON_PERSISTABLE)
namespace mozilla {
namespace net {
////////////////////////////////////////////////////////////////////////////////
NS_IMPL_ISUPPORTS(nsViewSourceHandler, nsIProtocolHandler,
nsIProtocolHandlerWithDynamicFlags)
////////////////////////////////////////////////////////////////////////////////
// nsIProtocolHandler methods:
NS_IMETHODIMP
nsViewSourceHandler::GetScheme(nsACString &result)
{
result.AssignLiteral(VIEW_SOURCE);
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::GetDefaultPort(int32_t *result)
{
*result = -1;
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::GetProtocolFlags(uint32_t *result)
{
*result = DEFAULT_FLAGS;
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::GetFlagsForURI(nsIURI* aURI, uint32_t* result)
{
*result = DEFAULT_FLAGS;
nsCOMPtr<nsINestedURI> nestedURI(do_QueryInterface(aURI));
if (!nestedURI) {
return NS_OK;
}
nsCOMPtr<nsIURI> innerURI;
nestedURI->GetInnerURI(getter_AddRefs(innerURI));
nsCOMPtr<nsINetUtil> netUtil = do_GetNetUtil();
bool isLoadable = false;
nsresult rv = netUtil->ProtocolHasFlags(innerURI, URI_LOADABLE_BY_ANYONE, &isLoadable);
NS_ENSURE_SUCCESS(rv, rv);
if (isLoadable) {
*result |= URI_LOADABLE_BY_EXTENSIONS;
}
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::NewURI(const nsACString &aSpec,
const char *aCharset,
nsIURI *aBaseURI,
nsIURI **aResult)
{
*aResult = nullptr;
// Extract inner URL and normalize to ASCII. This is done to properly
// support IDN in cases like "view-source:http://www.szalagavató.hu/"
int32_t colon = aSpec.FindChar(':');
if (colon == kNotFound)
return NS_ERROR_MALFORMED_URI;
nsCOMPtr<nsIURI> innerURI;
nsresult rv = NS_NewURI(getter_AddRefs(innerURI),
Substring(aSpec, colon + 1), aCharset, aBaseURI);
if (NS_FAILED(rv))
return rv;
nsAutoCString asciiSpec;
rv = innerURI->GetAsciiSpec(asciiSpec);
if (NS_FAILED(rv))
return rv;
// put back our scheme and construct a simple-uri wrapper
asciiSpec.Insert(VIEW_SOURCE ":", 0);
// We can't swap() from an RefPtr<nsSimpleNestedURI> to an nsIURI**,
// sadly.
RefPtr<nsSimpleNestedURI> ourURI = new nsSimpleNestedURI(innerURI);
nsCOMPtr<nsIURI> uri;
rv = NS_MutateURI(ourURI)
.SetSpec(asciiSpec)
.Finalize(uri);
if (NS_FAILED(rv)) {
return rv;
}
// Make the URI immutable so it's impossible to get it out of sync
// with its inner URI.
NS_TryToSetImmutable(uri);
uri.swap(*aResult);
return rv;
}
NS_IMETHODIMP
nsViewSourceHandler::NewChannel2(nsIURI* uri,
nsILoadInfo* aLoadInfo,
nsIChannel** result)
{
NS_ENSURE_ARG_POINTER(uri);
nsViewSourceChannel *channel = new nsViewSourceChannel();
if (!channel)
return NS_ERROR_OUT_OF_MEMORY;
NS_ADDREF(channel);
nsresult rv = channel->Init(uri);
if (NS_FAILED(rv)) {
NS_RELEASE(channel);
return rv;
}
// set the loadInfo on the new channel
rv = channel->SetLoadInfo(aLoadInfo);
if (NS_FAILED(rv)) {
NS_RELEASE(channel);
return rv;
}
*result = static_cast<nsIViewSourceChannel*>(channel);
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::NewChannel(nsIURI* uri, nsIChannel* *result)
{
return NewChannel2(uri, nullptr, result);
}
nsresult
nsViewSourceHandler::NewSrcdocChannel(nsIURI *aURI,
nsIURI *aBaseURI,
const nsAString &aSrcdoc,
nsILoadInfo* aLoadInfo,
nsIChannel** outChannel)
{
NS_ENSURE_ARG_POINTER(aURI);
RefPtr<nsViewSourceChannel> channel = new nsViewSourceChannel();
nsresult rv = channel->InitSrcdoc(aURI, aBaseURI, aSrcdoc, aLoadInfo);
if (NS_FAILED(rv)) {
return rv;
}
*outChannel = static_cast<nsIViewSourceChannel*>(channel.forget().take());
return NS_OK;
}
NS_IMETHODIMP
nsViewSourceHandler::AllowPort(int32_t port, const char *scheme, bool *_retval)
{
// don't override anything.
*_retval = false;
return NS_OK;
}
nsViewSourceHandler::nsViewSourceHandler()
{
gInstance = this;
}
nsViewSourceHandler::~nsViewSourceHandler()
{
gInstance = nullptr;
}
nsViewSourceHandler* nsViewSourceHandler::gInstance = nullptr;
nsViewSourceHandler*
nsViewSourceHandler::GetInstance()
{
return gInstance;
}
} // namespace net
} // namespace mozilla