Home | History | Annotate | Download | only in v8
      1 /*
      2  * Copyright (C) 2013 Google Inc. All rights reserved.
      3  *
      4  * Redistribution and use in source and binary forms, with or without
      5  * modification, are permitted provided that the following conditions are
      6  * met:
      7  *
      8  *     * Redistributions of source code must retain the above copyright
      9  * notice, this list of conditions and the following disclaimer.
     10  *     * Redistributions in binary form must reproduce the above
     11  * copyright notice, this list of conditions and the following disclaimer
     12  * in the documentation and/or other materials provided with the
     13  * distribution.
     14  *     * Neither the name of Google Inc. nor the names of its
     15  * contributors may be used to endorse or promote products derived from
     16  * this software without specific prior written permission.
     17  *
     18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
     19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
     20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
     21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
     22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
     23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
     24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
     25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
     26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
     27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
     28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
     29  */
     30 
     31 #include "config.h"
     32 #include "bindings/core/v8/ExceptionState.h"
     33 
     34 #include "bindings/core/v8/ExceptionMessages.h"
     35 #include "bindings/core/v8/V8ThrowException.h"
     36 #include "core/dom/ExceptionCode.h"
     37 
     38 namespace blink {
     39 
     40 void ExceptionState::clearException()
     41 {
     42     m_code = 0;
     43     m_exception.clear();
     44 }
     45 
     46 ScriptPromise ExceptionState::reject(ScriptState* scriptState)
     47 {
     48     ScriptPromise promise = ScriptPromise::reject(scriptState, m_exception.newLocal(scriptState->isolate()));
     49     clearException();
     50     return promise;
     51 }
     52 
     53 void ExceptionState::throwDOMException(const ExceptionCode& ec, const String& message)
     54 {
     55     ASSERT(ec);
     56     ASSERT(m_isolate);
     57     ASSERT(!m_creationContext.IsEmpty());
     58 
     59     // SecurityError is thrown via ::throwSecurityError, and _careful_ consideration must be given to the data exposed to JavaScript via the 'sanitizedMessage'.
     60     ASSERT(ec != SecurityError);
     61 
     62     m_code = ec;
     63     String processedMessage = addExceptionContext(message);
     64     m_message = processedMessage;
     65     setException(V8ThrowException::createDOMException(ec, processedMessage, m_creationContext, m_isolate));
     66 }
     67 
     68 void ExceptionState::throwSecurityError(const String& sanitizedMessage, const String& unsanitizedMessage)
     69 {
     70     ASSERT(m_isolate);
     71     ASSERT(!m_creationContext.IsEmpty());
     72     m_code = SecurityError;
     73     String finalSanitized = addExceptionContext(sanitizedMessage);
     74     m_message = finalSanitized;
     75     String finalUnsanitized = addExceptionContext(unsanitizedMessage);
     76 
     77     setException(V8ThrowException::createDOMException(SecurityError, finalSanitized, finalUnsanitized, m_creationContext, m_isolate));
     78 }
     79 
     80 void ExceptionState::setException(v8::Handle<v8::Value> exception)
     81 {
     82     // FIXME: Assert that exception is not empty?
     83     if (exception.IsEmpty()) {
     84         clearException();
     85         return;
     86     }
     87 
     88     m_exception.set(m_isolate, exception);
     89 }
     90 
     91 void ExceptionState::throwException()
     92 {
     93     ASSERT(!m_exception.isEmpty());
     94     V8ThrowException::throwException(m_exception.newLocal(m_isolate), m_isolate);
     95 }
     96 
     97 void ExceptionState::throwTypeError(const String& message)
     98 {
     99     ASSERT(m_isolate);
    100     m_code = V8TypeError;
    101     m_message = message;
    102     setException(V8ThrowException::createTypeError(addExceptionContext(message), m_isolate));
    103 }
    104 
    105 void ExceptionState::throwRangeError(const String& message)
    106 {
    107     ASSERT(m_isolate);
    108     m_code = V8RangeError;
    109     m_message = message;
    110     setException(V8ThrowException::createRangeError(addExceptionContext(message), m_isolate));
    111 }
    112 
    113 void NonThrowableExceptionState::throwDOMException(const ExceptionCode& ec, const String& message)
    114 {
    115     ASSERT_NOT_REACHED();
    116     m_code = ec;
    117     m_message = message;
    118 }
    119 
    120 void NonThrowableExceptionState::throwTypeError(const String& message)
    121 {
    122     ASSERT_NOT_REACHED();
    123     m_code = V8TypeError;
    124     m_message = message;
    125 }
    126 
    127 void NonThrowableExceptionState::throwSecurityError(const String& sanitizedMessage, const String&)
    128 {
    129     ASSERT_NOT_REACHED();
    130     m_code = SecurityError;
    131     m_message = sanitizedMessage;
    132 }
    133 
    134 void NonThrowableExceptionState::throwRangeError(const String& message)
    135 {
    136     ASSERT_NOT_REACHED();
    137     m_code = V8RangeError;
    138     m_message = message;
    139 }
    140 
    141 void TrackExceptionState::throwDOMException(const ExceptionCode& ec, const String& message)
    142 {
    143     m_code = ec;
    144     m_message = message;
    145 }
    146 
    147 void TrackExceptionState::throwTypeError(const String& message)
    148 {
    149     m_code = V8TypeError;
    150     m_message = message;
    151 }
    152 
    153 void TrackExceptionState::throwSecurityError(const String& sanitizedMessage, const String&)
    154 {
    155     m_code = SecurityError;
    156     m_message = sanitizedMessage;
    157 }
    158 
    159 void TrackExceptionState::throwRangeError(const String& message)
    160 {
    161     m_code = V8RangeError;
    162     m_message = message;
    163 }
    164 
    165 String ExceptionState::addExceptionContext(const String& message) const
    166 {
    167     if (message.isEmpty())
    168         return message;
    169 
    170     String processedMessage = message;
    171     if (propertyName() && interfaceName() && m_context != UnknownContext) {
    172         if (m_context == DeletionContext)
    173             processedMessage = ExceptionMessages::failedToDelete(propertyName(), interfaceName(), message);
    174         else if (m_context == ExecutionContext)
    175             processedMessage = ExceptionMessages::failedToExecute(propertyName(), interfaceName(), message);
    176         else if (m_context == GetterContext)
    177             processedMessage = ExceptionMessages::failedToGet(propertyName(), interfaceName(), message);
    178         else if (m_context == SetterContext)
    179             processedMessage = ExceptionMessages::failedToSet(propertyName(), interfaceName(), message);
    180     } else if (!propertyName() && interfaceName()) {
    181         if (m_context == ConstructionContext)
    182             processedMessage = ExceptionMessages::failedToConstruct(interfaceName(), message);
    183         else if (m_context == EnumerationContext)
    184             processedMessage = ExceptionMessages::failedToEnumerate(interfaceName(), message);
    185         else if (m_context == IndexedDeletionContext)
    186             processedMessage = ExceptionMessages::failedToDeleteIndexed(interfaceName(), message);
    187         else if (m_context == IndexedGetterContext)
    188             processedMessage = ExceptionMessages::failedToGetIndexed(interfaceName(), message);
    189         else if (m_context == IndexedSetterContext)
    190             processedMessage = ExceptionMessages::failedToSetIndexed(interfaceName(), message);
    191     }
    192     return processedMessage;
    193 }
    194 
    195 } // namespace blink
    196