blob: 1d4e99974f7acf91b068bada31c06a4d6e9b8635 [file] [log] [blame]
/*
* Copyright (C) 2012 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. AND ITS CONTRIBUTORS ``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 ITS 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 "config.h"
#include <limits>
#include <wtf/MathExtras.h>
#include <wtf/text/CString.h>
#include <wtf/text/WTFString.h>
namespace TestWebKitAPI {
TEST(WTF, StringCreationFromLiteral)
{
String stringFromLiteral(ASCIILiteral("Explicit construction syntax"));
ASSERT_EQ(strlen("Explicit construction syntax"), stringFromLiteral.length());
ASSERT_TRUE(stringFromLiteral == "Explicit construction syntax");
ASSERT_TRUE(stringFromLiteral.is8Bit());
ASSERT_TRUE(String("Explicit construction syntax") == stringFromLiteral);
String stringWithTemplate("Template Literal", String::ConstructFromLiteral);
ASSERT_EQ(strlen("Template Literal"), stringWithTemplate.length());
ASSERT_TRUE(stringWithTemplate == "Template Literal");
ASSERT_TRUE(stringWithTemplate.is8Bit());
ASSERT_TRUE(String("Template Literal") == stringWithTemplate);
}
TEST(WTF, StringASCII)
{
CString output;
// Null String.
output = String().ascii();
ASSERT_STREQ("", output.data());
// Empty String.
output = emptyString().ascii();
ASSERT_STREQ("", output.data());
// Regular String.
output = String(ASCIILiteral("foobar")).ascii();
ASSERT_STREQ("foobar", output.data());
}
static void testNumberToStringECMAScript(double number, const char* reference)
{
CString numberString = String::numberToStringECMAScript(number).latin1();
ASSERT_STREQ(reference, numberString.data());
}
TEST(WTF, StringNumberToStringECMAScriptBoundaries)
{
typedef std::numeric_limits<double> Limits;
// Infinity.
testNumberToStringECMAScript(Limits::infinity(), "Infinity");
testNumberToStringECMAScript(-Limits::infinity(), "-Infinity");
// NaN.
testNumberToStringECMAScript(-Limits::quiet_NaN(), "NaN");
// Zeros.
testNumberToStringECMAScript(0, "0");
testNumberToStringECMAScript(-0, "0");
// Min-Max.
testNumberToStringECMAScript(Limits::min(), "2.2250738585072014e-308");
testNumberToStringECMAScript(Limits::max(), "1.7976931348623157e+308");
}
TEST(WTF, StringNumberToStringECMAScriptRegularNumbers)
{
// Pi.
testNumberToStringECMAScript(piDouble, "3.141592653589793");
testNumberToStringECMAScript(piFloat, "3.1415927410125732");
testNumberToStringECMAScript(piOverTwoDouble, "1.5707963267948966");
testNumberToStringECMAScript(piOverTwoFloat, "1.5707963705062866");
testNumberToStringECMAScript(piOverFourDouble, "0.7853981633974483");
testNumberToStringECMAScript(piOverFourFloat, "0.7853981852531433");
// e.
const double e = 2.71828182845904523536028747135266249775724709369995;
testNumberToStringECMAScript(e, "2.718281828459045");
// c, speed of light in m/s.
const double c = 299792458;
testNumberToStringECMAScript(c, "299792458");
// Golen ratio.
const double phi = 1.6180339887498948482;
testNumberToStringECMAScript(phi, "1.618033988749895");
}
TEST(WTF, StringReplaceWithLiteral)
{
// Cases for 8Bit source.
String testString = "1224";
ASSERT_TRUE(testString.is8Bit());
testString.replaceWithLiteral('2', "");
ASSERT_STREQ("14", testString.utf8().data());
testString = "1224";
ASSERT_TRUE(testString.is8Bit());
testString.replaceWithLiteral('2', "3");
ASSERT_STREQ("1334", testString.utf8().data());
testString = "1224";
ASSERT_TRUE(testString.is8Bit());
testString.replaceWithLiteral('2', "555");
ASSERT_STREQ("15555554", testString.utf8().data());
testString = "1224";
ASSERT_TRUE(testString.is8Bit());
testString.replaceWithLiteral('3', "NotFound");
ASSERT_STREQ("1224", testString.utf8().data());
// Cases for 16Bit source.
testString = String::fromUTF8("résumé");
ASSERT_FALSE(testString.is8Bit());
testString.replaceWithLiteral(UChar(0x00E9 /*U+00E9 is 'é'*/), "e");
ASSERT_STREQ("resume", testString.utf8().data());
testString = String::fromUTF8("résumé");
ASSERT_FALSE(testString.is8Bit());
testString.replaceWithLiteral(UChar(0x00E9 /*U+00E9 is 'é'*/), "");
ASSERT_STREQ("rsum", testString.utf8().data());
testString = String::fromUTF8("résumé");
ASSERT_FALSE(testString.is8Bit());
testString.replaceWithLiteral('3', "NotFound");
ASSERT_STREQ("résumé", testString.utf8().data());
}
TEST(WTF, StringIsolatedCopy)
{
String original = "1234";
auto copy = WTFMove(original).isolatedCopy();
ASSERT_FALSE(original.impl() == copy.impl());
}
TEST(WTF, StringToInt)
{
bool ok = false;
EXPECT_EQ(0, String().toInt());
EXPECT_EQ(0, String().toInt(&ok));
EXPECT_FALSE(ok);
EXPECT_EQ(0, emptyString().toInt());
EXPECT_EQ(0, emptyString().toInt(&ok));
EXPECT_FALSE(ok);
EXPECT_EQ(0, String("0").toInt());
EXPECT_EQ(0, String("0").toInt(&ok));
EXPECT_TRUE(ok);
EXPECT_EQ(1, String("1").toInt());
EXPECT_EQ(1, String("1").toInt(&ok));
EXPECT_TRUE(ok);
EXPECT_EQ(2147483647, String("2147483647").toInt());
EXPECT_EQ(2147483647, String("2147483647").toInt(&ok));
EXPECT_TRUE(ok);
EXPECT_EQ(0, String("2147483648").toInt());
EXPECT_EQ(0, String("2147483648").toInt(&ok));
EXPECT_FALSE(ok);
EXPECT_EQ(-2147483648, String("-2147483648").toInt());
EXPECT_EQ(-2147483648, String("-2147483648").toInt(&ok));
EXPECT_TRUE(ok);
EXPECT_EQ(0, String("-2147483649").toInt());
EXPECT_EQ(0, String("-2147483649").toInt(&ok));
EXPECT_FALSE(ok);
// fail if we see leading junk
EXPECT_EQ(0, String("x1").toInt());
EXPECT_EQ(0, String("x1").toInt(&ok));
EXPECT_FALSE(ok);
// succeed if we see leading spaces
EXPECT_EQ(1, String(" 1").toInt());
EXPECT_EQ(1, String(" 1").toInt(&ok));
EXPECT_TRUE(ok);
// silently ignore trailing junk
EXPECT_EQ(1, String("1x").toInt());
EXPECT_EQ(1, String("1x").toInt(&ok));
EXPECT_TRUE(ok);
}
TEST(WTF, StringToDouble)
{
bool ok = false;
EXPECT_EQ(0.0, String().toDouble());
EXPECT_EQ(0.0, String().toDouble(&ok));
EXPECT_FALSE(ok);
EXPECT_EQ(0.0, emptyString().toDouble());
EXPECT_EQ(0.0, emptyString().toDouble(&ok));
EXPECT_FALSE(ok);
EXPECT_EQ(0.0, String("0").toDouble());
EXPECT_EQ(0.0, String("0").toDouble(&ok));
EXPECT_TRUE(ok);
EXPECT_EQ(1.0, String("1").toDouble());
EXPECT_EQ(1.0, String("1").toDouble(&ok));
EXPECT_TRUE(ok);
// fail if we see leading junk
EXPECT_EQ(0.0, String("x1").toDouble());
EXPECT_EQ(0.0, String("x1").toDouble(&ok));
EXPECT_FALSE(ok);
// succeed if we see leading spaces
EXPECT_EQ(1.0, String(" 1").toDouble());
EXPECT_EQ(1.0, String(" 1").toDouble(&ok));
EXPECT_TRUE(ok);
// ignore trailing junk, but return false for "ok"
// FIXME: This is an inconsistency with toInt, which always guarantees
// it will return 0 if it's also going to return false for ok.
EXPECT_EQ(1.0, String("1x").toDouble());
EXPECT_EQ(1.0, String("1x").toDouble(&ok));
EXPECT_FALSE(ok);
// parse only numbers, not special values such as "infinity"
EXPECT_EQ(0.0, String("infinity").toDouble());
EXPECT_EQ(0.0, String("infinity").toDouble(&ok));
EXPECT_FALSE(ok);
// parse only numbers, not special values such as "nan"
EXPECT_EQ(0.0, String("nan").toDouble());
EXPECT_EQ(0.0, String("nan").toDouble(&ok));
EXPECT_FALSE(ok);
}
TEST(WTF, StringhasInfixStartingAt)
{
EXPECT_TRUE(String("Test").is8Bit());
EXPECT_TRUE(String("Te").is8Bit());
EXPECT_TRUE(String("st").is8Bit());
EXPECT_TRUE(String("Test").hasInfixStartingAt(String("Te"), 0));
EXPECT_FALSE(String("Test").hasInfixStartingAt(String("Te"), 2));
EXPECT_TRUE(String("Test").hasInfixStartingAt(String("st"), 2));
EXPECT_FALSE(String("Test").hasInfixStartingAt(String("ST"), 2));
EXPECT_FALSE(String::fromUTF8("中国").is8Bit());
EXPECT_FALSE(String::fromUTF8("中").is8Bit());
EXPECT_FALSE(String::fromUTF8("国").is8Bit());
EXPECT_TRUE(String::fromUTF8("中国").hasInfixStartingAt(String::fromUTF8("中"), 0));
EXPECT_FALSE(String::fromUTF8("中国").hasInfixStartingAt(String::fromUTF8("中"), 1));
EXPECT_TRUE(String::fromUTF8("中国").hasInfixStartingAt(String::fromUTF8("国"), 1));
EXPECT_FALSE(String::fromUTF8("中国").hasInfixStartingAt(String("Te"), 0));
EXPECT_FALSE(String("Test").hasInfixStartingAt(String::fromUTF8("中"), 2));
}
TEST(WTF, StringExistingHash)
{
String string1("Template Literal");
ASSERT_FALSE(string1.isNull());
ASSERT_FALSE(string1.impl()->hasHash());
string1.impl()->hash();
ASSERT_EQ(string1.existingHash(), string1.impl()->existingHash());
String string2;
ASSERT_EQ(string2.existingHash(), 0u);
}
TEST(WTF, StringUnicodeEqualUCharArray)
{
String string1("abc");
ASSERT_FALSE(string1.isNull());
ASSERT_TRUE(string1.is8Bit());
UChar ab[] = { 'a', 'b' };
UChar abc[] = { 'a', 'b', 'c' };
UChar abcd[] = { 'a', 'b', 'c', 'd' };
UChar aBc[] = { 'a', 'B', 'c' };
ASSERT_FALSE(equal(string1, ab));
ASSERT_TRUE(equal(string1, abc));
ASSERT_FALSE(equal(string1, abcd));
ASSERT_FALSE(equal(string1, aBc));
String string2(abc, 3);
ASSERT_FALSE(equal(string2, ab));
ASSERT_TRUE(equal(string2, abc));
ASSERT_FALSE(equal(string2, abcd));
ASSERT_FALSE(equal(string2, aBc));
}
} // namespace TestWebKitAPI