blob: 74a14b357520a6e24386ac8e3ad50f0c0c6edaf1 [file] [log] [blame]
/*
* Copyright (C) 2013 Google 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:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * 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.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
* OWNER OR 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 "third_party/blink/renderer/platform/bindings/exception_messages.h"
#include "base/notreached.h"
#include "third_party/blink/renderer/platform/bindings/exception_context.h"
#include "third_party/blink/renderer/platform/bindings/v8_binding.h"
#include "third_party/blink/renderer/platform/wtf/decimal.h"
#include "third_party/blink/renderer/platform/wtf/math_extras.h"
#include "third_party/blink/renderer/platform/wtf/text/strcat.h"
namespace blink {
namespace {
String OptionalNameProperty(const String& property) {
if (property.empty()) {
return String();
}
return StrCat({" '", property, "'"});
}
String OptionalIndexProperty(const String& property) {
if (!property) {
return String();
}
return StrCat({" [", property, "]"});
}
} // namespace
String ExceptionMessages::AddContextToMessage(v8::ExceptionContext type,
const char* class_name,
const String& property_name,
const String& message) {
switch (type) {
case v8::ExceptionContext::kConstructor:
return ExceptionMessages::FailedToConstruct(class_name, message);
case v8::ExceptionContext::kOperation:
return ExceptionMessages::FailedToExecute(property_name, class_name,
message);
case v8::ExceptionContext::kAttributeGet:
return ExceptionMessages::FailedToGet(property_name, class_name, message);
case v8::ExceptionContext::kAttributeSet:
return ExceptionMessages::FailedToSet(property_name, class_name, message);
case v8::ExceptionContext::kNamedEnumerator:
return ExceptionMessages::FailedToEnumerate(class_name, message);
case v8::ExceptionContext::kIndexedGetter:
case v8::ExceptionContext::kIndexedDescriptor:
case v8::ExceptionContext::kIndexedQuery:
return ExceptionMessages::FailedToGetIndexed(property_name, class_name,
message);
case v8::ExceptionContext::kIndexedSetter:
case v8::ExceptionContext::kIndexedDefiner:
return ExceptionMessages::FailedToSetIndexed(property_name, class_name,
message);
case v8::ExceptionContext::kIndexedDeleter:
return ExceptionMessages::FailedToDeleteIndexed(property_name, class_name,
message);
case v8::ExceptionContext::kNamedGetter:
case v8::ExceptionContext::kNamedDescriptor:
case v8::ExceptionContext::kNamedQuery:
return ExceptionMessages::FailedToGetNamed(property_name, class_name,
message);
case v8::ExceptionContext::kNamedSetter:
case v8::ExceptionContext::kNamedDefiner:
return ExceptionMessages::FailedToSetNamed(property_name, class_name,
message);
case v8::ExceptionContext::kNamedDeleter:
return ExceptionMessages::FailedToDeleteNamed(property_name, class_name,
message);
case v8::ExceptionContext::kUnknown:
return message;
}
NOTREACHED();
}
String ExceptionMessages::FailedToConvertJSValue(const char* type) {
return StrCat({"Failed to convert value to '", type, "'."});
}
String ExceptionMessages::FailedToConstruct(const char* type,
const String& detail) {
String type_string = String(type);
if (type_string.empty()) {
return detail;
}
if (detail.empty()) {
return StrCat({"Failed to construct '", type_string, "'"});
}
return StrCat({"Failed to construct '", type_string, "': ", detail});
}
String ExceptionMessages::FailedToEnumerate(const char* type,
const String& detail) {
if (detail.empty()) {
return StrCat({"Failed to enumerate the properties of '", type, "'"});
}
return StrCat(
{"Failed to enumerate the properties of '", type, "': ", detail});
}
String ExceptionMessages::FailedToExecute(const String& method,
const char* type,
const String& detail) {
if (detail.empty()) {
return StrCat({"Failed to execute '", method, "' on '", type, "'"});
}
return StrCat({"Failed to execute '", method, "' on '", type, "': ", detail});
}
String ExceptionMessages::FailedToGet(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to read the '", property, "' property from '", type,
"': ", detail});
}
String ExceptionMessages::FailedToSet(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to set the '", property, "' property on '", type,
"': ", detail});
}
String ExceptionMessages::FailedToDelete(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to delete the '", property, "' property from '", type,
"': ", detail});
}
String ExceptionMessages::FailedToGetIndexed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to read an indexed property",
OptionalIndexProperty(property), " from '", type,
"': ", detail});
}
String ExceptionMessages::FailedToSetIndexed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to set an indexed property",
OptionalIndexProperty(property), " on '", type,
"': ", detail});
}
String ExceptionMessages::FailedToDeleteIndexed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to delete an indexed property",
OptionalIndexProperty(property), " from '", type,
"': ", detail});
}
String ExceptionMessages::FailedToGetNamed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to read a named property",
OptionalNameProperty(property), " from '", type,
"': ", detail});
}
String ExceptionMessages::FailedToSetNamed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to set a named property",
OptionalNameProperty(property), " on '", type, "': ", detail});
}
String ExceptionMessages::FailedToDeleteNamed(const String& property,
const char* type,
const String& detail) {
return StrCat({"Failed to delete a named property",
OptionalNameProperty(property), " from '", type,
"': ", detail});
}
String ExceptionMessages::ConstructorNotCallableAsFunction(const char* type) {
return FailedToConstruct(type,
"Please use the 'new' operator, this DOM object "
"constructor cannot be called as a function.");
}
String ExceptionMessages::ConstructorCalledAsFunction() {
return (
"Please use the 'new' operator, this DOM object constructor cannot "
"be called as a function.");
}
String ExceptionMessages::IncorrectPropertyType(const String& property,
const String& detail) {
return StrCat({"The '", property, "' property ", detail});
}
String ExceptionMessages::InvalidArity(const char* expected,
unsigned provided) {
return StrCat({"Valid arities are: ", expected, ", but ",
String::Number(provided), " arguments provided."});
}
String ExceptionMessages::ArgumentNullOrIncorrectType(
int argument_index,
const String& expected_type) {
return StrCat({"The ", OrdinalNumber(argument_index),
" argument provided is either null, or an invalid ",
expected_type, " object."});
}
String ExceptionMessages::ArgumentNotOfType(int argument_index,
const char* expected_type) {
return String::Format("parameter %d is not of type '%s'.", argument_index + 1,
expected_type);
}
String ExceptionMessages::NotASequenceTypeProperty(
const String& property_name) {
return StrCat(
{"'", property_name,
"' property is neither an array, nor does it have indexed properties."});
}
String ExceptionMessages::NotEnoughArguments(unsigned expected,
unsigned provided) {
return StrCat({String::Number(expected), " argument", expected > 1 ? "s" : "",
" required, but only ", String::Number(provided),
" present."});
}
String ExceptionMessages::NotAFiniteNumber(double value, const char* name) {
DCHECK(!std::isfinite(value));
return StrCat({"The ", name, " is ",
std::isinf(value) ? "infinite." : "not a number."});
}
String ExceptionMessages::NotAFiniteNumber(const Decimal& value,
const char* name) {
DCHECK(!value.IsFinite());
return StrCat({"The ", name, " is ",
value.IsInfinity() ? "infinite." : "not a number."});
}
String ExceptionMessages::OrdinalNumber(int number) {
String suffix("th");
switch (number % 10) {
case 1:
if (number % 100 != 11)
suffix = "st";
break;
case 2:
if (number % 100 != 12)
suffix = "nd";
break;
case 3:
if (number % 100 != 13)
suffix = "rd";
break;
}
return StrCat({String::Number(number), suffix});
}
String ExceptionMessages::IndexExceedsMaximumBound(const char* name,
bool eq,
const String& given,
const String& bound) {
return StrCat({"The ", name, " provided (", given, ") is greater than ",
eq ? "or equal to " : "", "the maximum bound (", bound, ")."});
}
String ExceptionMessages::IndexExceedsMinimumBound(const char* name,
bool eq,
const String& given,
const String& bound) {
return StrCat({"The ", name, " provided (", given, ") is less than ",
eq ? "or equal to " : "", "the minimum bound (", bound, ")."});
}
String ExceptionMessages::IndexOutsideRange(const char* name,
const String& given,
const String& lower_bound,
BoundType lower_type,
const String& upper_bound,
BoundType upper_type) {
return StrCat({"The ", name, " provided (", given, ") is outside the range ",
lower_type == kExclusiveBound ? "(" : "[", lower_bound, ", ",
upper_bound, upper_type == kExclusiveBound ? ")." : "]."});
}
String ExceptionMessages::ReadOnly(const char* detail) {
DEFINE_STATIC_LOCAL(String, read_only, ("This object is read-only."));
return detail ? StrCat({"This object is read-only, because ", detail, "."})
: read_only;
}
String ExceptionMessages::SharedArrayBufferNotAllowed(
const char* expected_type) {
return StrCat({"The provided ", expected_type, " value must not be shared."});
}
String ExceptionMessages::ResizableArrayBufferNotAllowed(
const char* expected_type) {
return StrCat(
{"The provided ", expected_type, " value must not be resizable."});
}
String ExceptionMessages::ValueNotOfType(const char* expected_type) {
return StrCat({"The provided value is not of type '", expected_type, "'."});
}
template <>
String ExceptionMessages::FormatNumber<float>(float number) {
return FormatPotentiallyNonFiniteNumber(number);
}
template <>
String ExceptionMessages::FormatNumber<double>(double number) {
return FormatPotentiallyNonFiniteNumber(number);
}
} // namespace blink