mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-10 05:47:04 +00:00
358 lines
10 KiB
C++
358 lines
10 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/* ***** BEGIN LICENSE BLOCK *****
|
|
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
* the License. You may obtain a copy of the License at
|
|
* http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
* for the specific language governing rights and limitations under the
|
|
* License.
|
|
*
|
|
* The Original Code is mozilla.org code.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2001
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either the GNU General Public License Version 2 or later (the "GPL"), or
|
|
* the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
|
|
* in which case the provisions of the GPL or the LGPL are applicable instead
|
|
* of those above. If you wish to allow use of your version of this file only
|
|
* under the terms of either the GPL or the LGPL, and not to allow others to
|
|
* use your version of this file under the terms of the MPL, indicate your
|
|
* decision by deleting the provisions above and replace them with the notice
|
|
* and other provisions required by the GPL or the LGPL. If you do not delete
|
|
* the provisions above, a recipient may use your version of this file under
|
|
* the terms of any one of the MPL, the GPL or the LGPL.
|
|
*
|
|
* ***** END LICENSE BLOCK ***** */
|
|
|
|
#include "nsEmbedChromeRegistry.h"
|
|
#include "nsString.h"
|
|
#include "nsReadableUtils.h"
|
|
#include "plstr.h"
|
|
|
|
#include "nsIDirectoryService.h"
|
|
#include "nsAppDirectoryServiceDefs.h"
|
|
#include "nsIProperties.h"
|
|
#include "nsILocalFile.h"
|
|
#include "nsIURI.h"
|
|
|
|
#define CHROME_TYPE_CONTENT 0
|
|
#define CHROME_TYPE_LOCALE 1
|
|
#define CHROME_TYPE_SKIN 2
|
|
|
|
const char kChromePrefix[] = "chrome://";
|
|
|
|
static nsresult
|
|
SplitURL(nsIURI *aChromeURI, nsCString& aPackage, nsCString& aProvider, nsCString& aFile,
|
|
PRBool *aModified = nsnull)
|
|
{
|
|
// Splits a "chrome:" URL into its package, provider, and file parts.
|
|
// Here are the current portions of a
|
|
// chrome: url that make up the chrome-
|
|
//
|
|
// chrome://global/skin/foo?bar
|
|
// \------/ \----/\---/ \-----/
|
|
// | | | |
|
|
// | | | `-- RemainingPortion
|
|
// | | |
|
|
// | | `-- Provider
|
|
// | |
|
|
// | `-- Package
|
|
// |
|
|
// `-- Always "chrome://"
|
|
//
|
|
//
|
|
|
|
nsresult rv;
|
|
|
|
nsCAutoString str;
|
|
rv = aChromeURI->GetSpec(str);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
// We only want to deal with "chrome:" URLs here. We could return
|
|
// an error code if the URL isn't properly prefixed here...
|
|
if (PL_strncmp(str.get(), kChromePrefix, sizeof(kChromePrefix) - 1) != 0)
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
// Cull out the "package" string; e.g., "navigator"
|
|
aPackage = str.get() + sizeof(kChromePrefix) - 1;
|
|
|
|
PRInt32 idx;
|
|
idx = aPackage.FindChar('/');
|
|
if (idx < 0)
|
|
return NS_OK;
|
|
|
|
// Cull out the "provider" string; e.g., "content"
|
|
aPackage.Right(aProvider, aPackage.Length() - (idx + 1));
|
|
aPackage.Truncate(idx);
|
|
|
|
idx = aProvider.FindChar('/');
|
|
if (idx < 0) {
|
|
// Force the provider to end with a '/'
|
|
idx = aProvider.Length();
|
|
aProvider.Append('/');
|
|
}
|
|
|
|
// Cull out the "file"; e.g., "navigator.xul"
|
|
aProvider.Right(aFile, aProvider.Length() - (idx + 1));
|
|
aProvider.Truncate(idx);
|
|
|
|
PRBool nofile = aFile.IsEmpty();
|
|
if (nofile) {
|
|
// If there is no file, then construct the default file
|
|
aFile = aPackage;
|
|
|
|
if (aProvider.Equals("content")) {
|
|
aFile += ".xul";
|
|
}
|
|
else if (aProvider.Equals("skin")) {
|
|
aFile += ".css";
|
|
}
|
|
else if (aProvider.Equals("locale")) {
|
|
aFile += ".dtd";
|
|
}
|
|
else {
|
|
NS_ERROR("unknown provider");
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
} else {
|
|
// Protect against URIs containing .. that reach up out of the
|
|
// chrome directory to grant chrome privileges to non-chrome files.
|
|
int depth = 0;
|
|
PRBool sawSlash = PR_TRUE; // .. at the beginning is suspect as well as /..
|
|
for (const char* p=aFile.get(); *p; p++) {
|
|
if (sawSlash) {
|
|
if (p[0] == '.' && p[1] == '.'){
|
|
depth--; // we have /.., decrement depth.
|
|
} else {
|
|
static const char escape[] = "%2E%2E";
|
|
if (PL_strncasecmp(p, escape, sizeof(escape)-1) == 0)
|
|
depth--; // we have the HTML-escaped form of /.., decrement depth.
|
|
}
|
|
} else if (p[0] != '/') {
|
|
depth++; // we have /x for some x that is not /
|
|
}
|
|
sawSlash = (p[0] == '/');
|
|
|
|
if (depth < 0) {
|
|
return NS_ERROR_FAILURE;
|
|
}
|
|
}
|
|
}
|
|
if (aModified)
|
|
*aModified = nofile;
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS1(nsEmbedChromeRegistry, nsIChromeRegistry)
|
|
|
|
nsEmbedChromeRegistry::nsEmbedChromeRegistry()
|
|
{
|
|
}
|
|
|
|
nsresult
|
|
nsEmbedChromeRegistry::Init()
|
|
{
|
|
NS_ASSERTION(0, "Creating embedding chrome registry\n");
|
|
nsresult rv;
|
|
|
|
rv = NS_NewISupportsArray(getter_AddRefs(mEmptyArray));
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
rv = ReadChromeRegistry();
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
nsEmbedChromeRegistry::ReadChromeRegistry()
|
|
{
|
|
nsresult rv;
|
|
nsCOMPtr<nsIProperties> directoryService =
|
|
do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID, &rv);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
nsCOMPtr<nsILocalFile> listFile;
|
|
rv = directoryService->Get(NS_APP_CHROME_DIR, NS_GET_IID(nsILocalFile),
|
|
getter_AddRefs(listFile));
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
rv = listFile->AppendRelativeNativePath(NS_LITERAL_CSTRING("installed-chrome.txt"));
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
PRFileDesc *file;
|
|
rv = listFile->OpenNSPRFileDesc(PR_RDONLY, 0, &file);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
PRFileInfo finfo;
|
|
|
|
if (PR_GetOpenFileInfo(file, &finfo) == PR_SUCCESS) {
|
|
char *dataBuffer = new char[finfo.size+1];
|
|
if (dataBuffer) {
|
|
PRInt32 bufferSize = PR_Read(file, dataBuffer, finfo.size);
|
|
if (bufferSize > 0) {
|
|
dataBuffer[bufferSize] = '\r';
|
|
rv = ProcessNewChromeBuffer(dataBuffer, bufferSize);
|
|
}
|
|
delete [] dataBuffer;
|
|
}
|
|
}
|
|
PR_Close(file);
|
|
|
|
return rv;
|
|
}
|
|
|
|
nsresult
|
|
nsEmbedChromeRegistry::ProcessNewChromeBuffer(char* aBuffer, PRInt32 aLength)
|
|
{
|
|
while (aLength > 0) {
|
|
PRInt32 processedBytes = ProcessChromeLine(aBuffer, aLength);
|
|
aBuffer += processedBytes;
|
|
aLength -= processedBytes;
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
#define MAX_TOKENS 5
|
|
struct chromeToken {
|
|
const char *tokenStart;
|
|
const char *tokenEnd;
|
|
};
|
|
|
|
PRInt32
|
|
nsEmbedChromeRegistry::ProcessChromeLine(const char* aBuffer, PRInt32 aLength)
|
|
{
|
|
PRInt32 bytesProcessed = 0;
|
|
chromeToken tokens[MAX_TOKENS];
|
|
PRInt32 tokenCount = 0;
|
|
PRBool expectingToken = PR_TRUE;
|
|
|
|
while (bytesProcessed <= aLength &&
|
|
*aBuffer != '\n' && *aBuffer != '\r' &&
|
|
tokenCount < MAX_TOKENS) {
|
|
|
|
if (*aBuffer == ',') {
|
|
tokenCount++;
|
|
expectingToken = PR_TRUE;
|
|
}
|
|
else if (expectingToken)
|
|
tokens[tokenCount].tokenStart = aBuffer;
|
|
else
|
|
tokens[tokenCount].tokenEnd = aBuffer;
|
|
|
|
|
|
aBuffer++;
|
|
bytesProcessed++;
|
|
}
|
|
NS_ASSERTION(tokenCount == 4, "Unexpected tokens in line");
|
|
|
|
nsDependentCSubstring
|
|
chromeType(tokens[0].tokenStart, tokens[0].tokenEnd);
|
|
nsDependentCSubstring
|
|
chromeProfile(tokens[1].tokenStart, tokens[1].tokenEnd);
|
|
nsDependentCSubstring
|
|
chromeLocType(tokens[2].tokenStart, tokens[2].tokenEnd);
|
|
nsDependentCSubstring
|
|
chromeLocation(tokens[3].tokenStart, tokens[3].tokenEnd);
|
|
|
|
RegisterChrome(chromeType, chromeProfile, chromeLocType, chromeLocation);
|
|
return bytesProcessed;
|
|
}
|
|
|
|
nsresult
|
|
nsEmbedChromeRegistry::RegisterChrome(const nsACString& aChromeType,
|
|
const nsACString& aChromeProfile,
|
|
const nsACString& aChromeLocType,
|
|
const nsACString& aChromeLocation)
|
|
{
|
|
PRInt32 chromeType;
|
|
if (aChromeType.EqualsLiteral("skin"))
|
|
chromeType = CHROME_TYPE_SKIN;
|
|
else if (aChromeType.EqualsLiteral("locale"))
|
|
chromeType = CHROME_TYPE_LOCALE;
|
|
else
|
|
chromeType = CHROME_TYPE_CONTENT;
|
|
|
|
PRBool chromeIsProfile =
|
|
aChromeProfile.EqualsLiteral("profile");
|
|
|
|
PRBool chromeIsURL =
|
|
aChromeProfile.EqualsLiteral("url");
|
|
|
|
return RegisterChrome(chromeType, chromeIsProfile, chromeIsURL,
|
|
aChromeLocation);
|
|
}
|
|
|
|
nsresult
|
|
nsEmbedChromeRegistry::RegisterChrome(PRInt32 aChromeType, // CHROME_TYPE_CONTENT, etc
|
|
PRBool aChromeIsProfile, // per-profile?
|
|
PRBool aChromeIsURL, // is it a url? (else path)
|
|
const nsACString& aChromeLocation)
|
|
{
|
|
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEmbedChromeRegistry::CheckForNewChrome()
|
|
{
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEmbedChromeRegistry::Canonify(nsIURI* aChromeURI)
|
|
{
|
|
#if 1
|
|
// Canonicalize 'chrome:' URLs. We'll take any 'chrome:' URL
|
|
// without a filename, and change it to a URL -with- a filename;
|
|
// e.g., "chrome://navigator/content" to
|
|
// "chrome://navigator/content/navigator.xul".
|
|
if (! aChromeURI)
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
PRBool modified = PR_TRUE; // default is we do canonification
|
|
nsCAutoString package, provider, file;
|
|
nsresult rv;
|
|
rv = SplitURL(aChromeURI, package, provider, file, &modified);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
if (!modified)
|
|
return NS_OK;
|
|
|
|
nsCAutoString canonical( kChromePrefix );
|
|
canonical += package;
|
|
canonical += "/";
|
|
canonical += provider;
|
|
canonical += "/";
|
|
canonical += file;
|
|
|
|
return aChromeURI->SetSpec(canonical);
|
|
#else
|
|
return NS_OK;
|
|
#endif
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsEmbedChromeRegistry::ConvertChromeURL(nsIURI* aChromeURL, nsACString& aResult)
|
|
{
|
|
nsresult rv;
|
|
|
|
rv = aChromeURL->GetSpec(aResult);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
return NS_OK;
|
|
}
|