2007-03-22 10:30:00 -07:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* ***** 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) 1998
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
* Pierre Phaneuf <pp@ludusdesign.com>
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
|
|
* either of 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 "nsString.h"
|
|
|
|
#include "nsIUnicodeEncoder.h"
|
|
|
|
#include "nsICharsetConverterManager.h"
|
|
|
|
#include "nsReadableUtils.h"
|
|
|
|
#include "nsITextToSubURI.h"
|
|
|
|
#include "nsIServiceManager.h"
|
|
|
|
#include "nsEscape.h"
|
|
|
|
#include "prmem.h"
|
|
|
|
#include "nsTextToSubURI.h"
|
|
|
|
#include "nsCRT.h"
|
|
|
|
|
|
|
|
static NS_DEFINE_CID(kCharsetConverterManagerCID, NS_ICHARSETCONVERTERMANAGER_CID);
|
|
|
|
|
|
|
|
nsTextToSubURI::nsTextToSubURI()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
nsTextToSubURI::~nsTextToSubURI()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMPL_ISUPPORTS1(nsTextToSubURI, nsITextToSubURI)
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsTextToSubURI::ConvertAndEscape(
|
|
|
|
const char *charset, const PRUnichar *text, char **_retval)
|
|
|
|
{
|
|
|
|
if(nsnull == _retval)
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
|
|
|
*_retval = nsnull;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// Get Charset, get the encoder.
|
|
|
|
nsICharsetConverterManager *ccm;
|
|
|
|
rv = CallGetService(kCharsetConverterManagerCID, &ccm);
|
|
|
|
if(NS_SUCCEEDED(rv)) {
|
|
|
|
nsIUnicodeEncoder *encoder;
|
|
|
|
rv = ccm->GetUnicodeEncoder(charset, &encoder);
|
|
|
|
NS_RELEASE(ccm);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
rv = encoder->SetOutputErrorBehavior(nsIUnicodeEncoder::kOnError_Replace, nsnull, (PRUnichar)'?');
|
|
|
|
if(NS_SUCCEEDED(rv))
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
char *pBuf = buf;
|
2012-04-03 17:22:42 -07:00
|
|
|
PRInt32 ulen = nsCRT::strlen(text);
|
2007-03-22 10:30:00 -07:00
|
|
|
PRInt32 outlen = 0;
|
|
|
|
if(NS_SUCCEEDED(rv = encoder->GetMaxLength(text, ulen, &outlen)))
|
|
|
|
{
|
|
|
|
if(outlen >= 256) {
|
2009-05-15 03:37:59 -07:00
|
|
|
pBuf = (char*)NS_Alloc(outlen+1);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
if(nsnull == pBuf) {
|
|
|
|
outlen = 255;
|
|
|
|
pBuf = buf;
|
|
|
|
}
|
|
|
|
PRInt32 bufLen = outlen;
|
|
|
|
if(NS_SUCCEEDED(rv = encoder->Convert(text,&ulen, pBuf, &outlen))) {
|
|
|
|
// put termination characters (e.g. ESC(B of ISO-2022-JP) if necessary
|
|
|
|
PRInt32 finLen = bufLen - outlen;
|
|
|
|
if (finLen > 0) {
|
|
|
|
if (NS_SUCCEEDED(encoder->Finish((char *)(pBuf+outlen), &finLen)))
|
|
|
|
outlen += finLen;
|
|
|
|
}
|
|
|
|
pBuf[outlen] = '\0';
|
|
|
|
*_retval = nsEscape(pBuf, url_XPAlphas);
|
|
|
|
if(nsnull == *_retval)
|
|
|
|
rv = NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(pBuf != buf)
|
2009-05-15 03:37:59 -07:00
|
|
|
NS_Free(pBuf);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
NS_RELEASE(encoder);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsTextToSubURI::UnEscapeAndConvert(
|
|
|
|
const char *charset, const char *text, PRUnichar **_retval)
|
|
|
|
{
|
|
|
|
if(nsnull == _retval)
|
|
|
|
return NS_ERROR_NULL_POINTER;
|
2009-10-30 02:17:19 -07:00
|
|
|
if(nsnull == text) {
|
|
|
|
// set empty string instead of returning error
|
|
|
|
// due to compatibility for old version
|
|
|
|
text = "";
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
*_retval = nsnull;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// unescape the string, unescape changes the input
|
2009-05-15 03:37:59 -07:00
|
|
|
char *unescaped = NS_strdup(text);
|
2007-03-22 10:30:00 -07:00
|
|
|
if (nsnull == unescaped)
|
|
|
|
return NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
unescaped = nsUnescape(unescaped);
|
|
|
|
NS_ASSERTION(unescaped, "nsUnescape returned null");
|
|
|
|
|
|
|
|
// Convert from the charset to unicode
|
|
|
|
nsCOMPtr<nsICharsetConverterManager> ccm =
|
|
|
|
do_GetService(kCharsetConverterManagerCID, &rv);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
nsIUnicodeDecoder *decoder;
|
|
|
|
rv = ccm->GetUnicodeDecoder(charset, &decoder);
|
|
|
|
if (NS_SUCCEEDED(rv)) {
|
|
|
|
PRUnichar *pBuf = nsnull;
|
|
|
|
PRInt32 len = strlen(unescaped);
|
|
|
|
PRInt32 outlen = 0;
|
|
|
|
if (NS_SUCCEEDED(rv = decoder->GetMaxLength(unescaped, len, &outlen))) {
|
2010-04-18 23:33:00 -07:00
|
|
|
pBuf = (PRUnichar *) NS_Alloc((outlen+1)*sizeof(PRUnichar));
|
2007-03-22 10:30:00 -07:00
|
|
|
if (nsnull == pBuf)
|
|
|
|
rv = NS_ERROR_OUT_OF_MEMORY;
|
|
|
|
else {
|
|
|
|
if (NS_SUCCEEDED(rv = decoder->Convert(unescaped, &len, pBuf, &outlen))) {
|
|
|
|
pBuf[outlen] = 0;
|
|
|
|
*_retval = pBuf;
|
|
|
|
}
|
|
|
|
else
|
2009-05-15 03:37:59 -07:00
|
|
|
NS_Free(pBuf);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
NS_RELEASE(decoder);
|
|
|
|
}
|
|
|
|
}
|
2009-05-15 03:37:59 -07:00
|
|
|
NS_Free(unescaped);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2011-09-28 23:19:26 -07:00
|
|
|
static bool statefulCharset(const char *charset)
|
2007-03-22 10:30:00 -07:00
|
|
|
{
|
|
|
|
if (!nsCRT::strncasecmp(charset, "ISO-2022-", sizeof("ISO-2022-")-1) ||
|
|
|
|
!nsCRT::strcasecmp(charset, "UTF-7") ||
|
|
|
|
!nsCRT::strcasecmp(charset, "HZ-GB-2312"))
|
2011-10-17 07:59:28 -07:00
|
|
|
return true;
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-10-17 07:59:28 -07:00
|
|
|
return false;
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
nsresult nsTextToSubURI::convertURItoUnicode(const nsAFlatCString &aCharset,
|
|
|
|
const nsAFlatCString &aURI,
|
2011-09-28 23:19:26 -07:00
|
|
|
bool aIRI,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsAString &_retval)
|
|
|
|
{
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
// check for 7bit encoding the data may not be ASCII after we decode
|
2011-09-28 23:19:26 -07:00
|
|
|
bool isStatefulCharset = statefulCharset(aCharset.get());
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
if (!isStatefulCharset && IsASCII(aURI)) {
|
|
|
|
CopyASCIItoUTF16(aURI, _retval);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isStatefulCharset && aIRI) {
|
|
|
|
if (IsUTF8(aURI)) {
|
|
|
|
CopyUTF8toUTF16(aURI, _retval);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// empty charset could indicate UTF-8, but aURI turns out not to be UTF-8.
|
|
|
|
NS_ENSURE_FALSE(aCharset.IsEmpty(), NS_ERROR_INVALID_ARG);
|
|
|
|
|
|
|
|
nsCOMPtr<nsICharsetConverterManager> charsetConverterManager;
|
|
|
|
|
|
|
|
charsetConverterManager = do_GetService(NS_CHARSETCONVERTERMANAGER_CONTRACTID, &rv);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
nsCOMPtr<nsIUnicodeDecoder> unicodeDecoder;
|
|
|
|
rv = charsetConverterManager->GetUnicodeDecoder(aCharset.get(),
|
|
|
|
getter_AddRefs(unicodeDecoder));
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
|
|
|
PRInt32 srcLen = aURI.Length();
|
|
|
|
PRInt32 dstLen;
|
|
|
|
rv = unicodeDecoder->GetMaxLength(aURI.get(), srcLen, &dstLen);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
|
2009-05-15 03:37:59 -07:00
|
|
|
PRUnichar *ustr = (PRUnichar *) NS_Alloc(dstLen * sizeof(PRUnichar));
|
2007-03-22 10:30:00 -07:00
|
|
|
NS_ENSURE_TRUE(ustr, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
|
|
|
|
rv = unicodeDecoder->Convert(aURI.get(), &srcLen, ustr, &dstLen);
|
|
|
|
|
|
|
|
if (NS_SUCCEEDED(rv))
|
|
|
|
_retval.Assign(ustr, dstLen);
|
|
|
|
|
2009-05-15 03:37:59 -07:00
|
|
|
NS_Free(ustr);
|
2007-03-22 10:30:00 -07:00
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsTextToSubURI::UnEscapeURIForUI(const nsACString & aCharset,
|
|
|
|
const nsACString &aURIFragment,
|
|
|
|
nsAString &_retval)
|
|
|
|
{
|
|
|
|
nsCAutoString unescapedSpec;
|
|
|
|
// skip control octets (0x00 - 0x1f and 0x7f) when unescaping
|
|
|
|
NS_UnescapeURL(PromiseFlatCString(aURIFragment),
|
|
|
|
esc_SkipControl | esc_AlwaysCopy, unescapedSpec);
|
|
|
|
|
|
|
|
// in case of failure, return escaped URI
|
|
|
|
// Test for != NS_OK rather than NS_FAILED, because incomplete multi-byte
|
|
|
|
// sequences are also considered failure in this context
|
|
|
|
if (convertURItoUnicode(
|
2011-10-17 07:59:28 -07:00
|
|
|
PromiseFlatCString(aCharset), unescapedSpec, true, _retval)
|
2007-03-22 10:30:00 -07:00
|
|
|
!= NS_OK)
|
|
|
|
// assume UTF-8 instead of ASCII because hostname (IDN) may be in UTF-8
|
|
|
|
CopyUTF8toUTF16(aURIFragment, _retval);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP nsTextToSubURI::UnEscapeNonAsciiURI(const nsACString & aCharset,
|
2011-11-06 07:14:27 -08:00
|
|
|
const nsACString & aURIFragment,
|
2007-03-22 10:30:00 -07:00
|
|
|
nsAString &_retval)
|
|
|
|
{
|
|
|
|
nsCAutoString unescapedSpec;
|
|
|
|
NS_UnescapeURL(PromiseFlatCString(aURIFragment),
|
|
|
|
esc_AlwaysCopy | esc_OnlyNonASCII, unescapedSpec);
|
2011-11-06 07:14:27 -08:00
|
|
|
// leave the URI as it is if it's not UTF-8 and aCharset is not a ASCII
|
|
|
|
// superset since converting "http:" with such an encoding is always a bad
|
|
|
|
// idea.
|
|
|
|
if (!IsUTF8(unescapedSpec) &&
|
|
|
|
(aCharset.LowerCaseEqualsLiteral("utf-16") ||
|
|
|
|
aCharset.LowerCaseEqualsLiteral("utf-16be") ||
|
|
|
|
aCharset.LowerCaseEqualsLiteral("utf-16le") ||
|
|
|
|
aCharset.LowerCaseEqualsLiteral("utf-7") ||
|
|
|
|
aCharset.LowerCaseEqualsLiteral("x-imap4-modified-utf7"))){
|
|
|
|
CopyASCIItoUTF16(aURIFragment, _retval);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2007-03-22 10:30:00 -07:00
|
|
|
|
2011-10-17 07:59:28 -07:00
|
|
|
return convertURItoUnicode(PromiseFlatCString(aCharset), unescapedSpec, true, _retval);
|
2007-03-22 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
//----------------------------------------------------------------------
|