nashorn/src/jdk/nashorn/api/scripting/NashornScriptEngine.java
author jlaskey
Fri, 21 Dec 2012 16:36:24 -0400
changeset 16147 e63b63819133
child 16151 97c1e756ae1e
permissions -rw-r--r--
8005403: Open-source Nashorn Reviewed-by: attila, hannesw, lagergren, sundar Contributed-by: james.laskey@oracle.com, akhil.arora@oracle.com, andreas.woess@jku.at, attila.szegedi@oracle.com, hannes.wallnoefer@oracle.com, henry.jen@oracle.com, marcus.lagergren@oracle.com, pavel.semenov@oracle.com, pavel.stepanov@oracle.com, petr.hejl@oracle.com, petr.pisl@oracle.com, sundararajan.athijegannathan@oracle.com
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
16147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     1
/*
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     2
 * Copyright (c) 2010, 2012, Oracle and/or its affiliates. All rights reserved.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     4
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    10
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    15
 * accompanied this code).
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    16
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    20
 *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    23
 * questions.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    24
 */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    25
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    26
package jdk.nashorn.api.scripting;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    27
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    28
import static jdk.nashorn.internal.runtime.ECMAErrors.referenceError;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    29
import static jdk.nashorn.internal.runtime.ECMAErrors.typeError;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    30
import static jdk.nashorn.internal.runtime.ScriptRuntime.UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    31
import static jdk.nashorn.internal.runtime.linker.Lookup.MH;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    32
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    33
import java.io.IOException;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    34
import java.io.InputStream;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    35
import java.io.InputStreamReader;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    36
import java.io.Reader;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    37
import java.lang.invoke.MethodHandle;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    38
import java.lang.reflect.Method;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    39
import java.lang.reflect.Modifier;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    40
import java.net.URL;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    41
import java.security.AccessController;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    42
import java.security.PrivilegedAction;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    43
import java.security.PrivilegedActionException;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    44
import java.security.PrivilegedExceptionAction;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    45
import javax.script.AbstractScriptEngine;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    46
import javax.script.Bindings;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    47
import javax.script.Compilable;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    48
import javax.script.CompiledScript;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    49
import javax.script.Invocable;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    50
import javax.script.ScriptContext;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    51
import javax.script.ScriptEngine;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    52
import javax.script.ScriptEngineFactory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    53
import javax.script.ScriptException;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    54
import javax.script.SimpleBindings;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    55
import jdk.nashorn.internal.runtime.Context;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    56
import jdk.nashorn.internal.runtime.ErrorManager;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    57
import jdk.nashorn.internal.runtime.GlobalObject;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    58
import jdk.nashorn.internal.runtime.Property;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    59
import jdk.nashorn.internal.runtime.ScriptFunction;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    60
import jdk.nashorn.internal.runtime.ScriptObject;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    61
import jdk.nashorn.internal.runtime.ScriptRuntime;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    62
import jdk.nashorn.internal.runtime.Source;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    63
import jdk.nashorn.internal.runtime.linker.JavaAdapterFactory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    64
import jdk.nashorn.internal.runtime.options.Options;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    65
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    66
/**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    67
 * JSR-223 compliant script engine for Nashorn. Instances are not created directly, but rather returned through
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    68
 * {@link NashornScriptEngineFactory#getScriptEngine()}. Note that this engine implements the {@link Compilable} and
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    69
 * {@link Invocable} interfaces, allowing for efficient precompilation and repeated execution of scripts.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    70
 * @see NashornScriptEngineFactory
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    71
 */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    72
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    73
public final class NashornScriptEngine extends AbstractScriptEngine implements Compilable, Invocable {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    74
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    75
    private final ScriptEngineFactory factory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    76
    private final Context             nashornContext;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    77
    private final ScriptObject        global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    78
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    79
    // default options passed to Nashorn Options object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    80
    private static final String[] DEFAULT_OPTIONS = new String[] { "-scripting", "-af", "-doe" };
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    81
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    82
    NashornScriptEngine(final NashornScriptEngineFactory factory) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    83
        this(factory, DEFAULT_OPTIONS);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    84
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    85
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    86
    @SuppressWarnings("LeakingThisInConstructor")
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    87
    NashornScriptEngine(final NashornScriptEngineFactory factory, final String[] args) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    88
        this.factory = factory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    89
        final Options options = new Options("nashorn");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    90
        options.process(args);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    91
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    92
        // throw ParseException on first error from script
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    93
        final ErrorManager errors = new Context.ThrowErrorManager();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    94
        // create new Nashorn Context and get global object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    95
        this.nashornContext = AccessController.doPrivileged(new PrivilegedAction<Context>() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    96
            @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    97
            public Context run() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    98
                try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
    99
                    return new Context(options, errors);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   100
                } catch (final RuntimeException e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   101
                    if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   102
                        e.printStackTrace();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   103
                    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   104
                    throw e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   105
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   106
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   107
        });
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   108
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   109
        // create new global object
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   110
        this.global = nashornContext.createGlobal();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   111
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   112
        // current ScriptContext exposed as "context"
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   113
        global.addOwnProperty("context", Property.NOT_ENUMERABLE, context);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   114
        // current ScriptEngine instance exposed as "engine". We added @SuppressWarnings("LeakingThisInConstructor") as
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   115
        // NetBeans identifies this assignment as such a leak - this is a false positive as we're setting this property
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   116
        // in the Global of a Context we just created - both the Context and the Global were just created and can not be
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   117
        // seen from another thread outside of this constructor.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   118
        global.addOwnProperty("engine", Property.NOT_ENUMERABLE, this);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   119
        // global script arguments
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   120
        global.addOwnProperty("arguments", Property.NOT_ENUMERABLE, UNDEFINED);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   121
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   122
        // evaluate engine initial script
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   123
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   124
            AccessController.doPrivileged(new PrivilegedExceptionAction<Void>() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   125
                @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   126
                public Void run() throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   127
                    evalEngineScript();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   128
                    return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   129
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   130
            });
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   131
        } catch (final PrivilegedActionException e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   132
            if (Context.DEBUG) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   133
                e.printStackTrace();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   134
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   135
            throw new RuntimeException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   136
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   137
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   138
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   139
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   140
    public Object eval(final Reader reader, final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   141
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   142
            return evalImpl(Source.readFully(reader), ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   143
        } catch (final IOException e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   144
            throw new ScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   145
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   146
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   147
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   148
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   149
    public Object eval(final String script, final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   150
        return evalImpl(script.toCharArray(), ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   151
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   152
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   153
    /*
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   154
     * FIXME: This is not exactly right. Script execution should actually
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   155
     * put the variables in ENGINE_SCOPE bindings. But, it is difficult
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   156
     * (not possible?) with the way ScriptObject is implemented. So, for now
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   157
     * giving access to script variables by accessing it from "globals". This
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   158
     * way at least engine.get("foo") will work whenever "foo" is a global var
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   159
     * defined by eval'ed scripts.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   160
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   161
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   162
    public Object get(final String key) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   163
        Object value = super.get(key);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   164
        if (value == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   165
            value = ScriptObjectMirror.wrap(global.get(key), global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   166
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   167
        return (value == UNDEFINED) ? null : value;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   168
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   169
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   170
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   171
    public ScriptEngineFactory getFactory() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   172
        return factory;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   173
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   174
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   175
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   176
    public Bindings createBindings() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   177
        return new SimpleBindings();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   178
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   179
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   180
    // Compilable methods
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   181
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   182
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   183
    public CompiledScript compile(final Reader reader) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   184
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   185
            return asCompiledScript(compileImpl(Source.readFully(reader), context));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   186
        } catch (final IOException e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   187
            throw new ScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   188
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   189
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   190
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   191
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   192
    public CompiledScript compile(final String str) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   193
        return asCompiledScript(compileImpl(str.toCharArray(), context));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   194
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   195
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   196
    // Invocable methods
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   197
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   198
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   199
    public Object invokeFunction(final String name, final Object... args)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   200
            throws ScriptException, NoSuchMethodException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   201
        return invokeImpl(null, name, args);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   202
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   203
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   204
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   205
    public Object invokeMethod(final Object self, final String name, final Object... args)
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   206
            throws ScriptException, NoSuchMethodException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   207
        if (self == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   208
            throw new IllegalArgumentException("script object can not be null");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   209
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   210
        return invokeImpl(self, name, args);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   211
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   212
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   213
    private <T> T getInterfaceInner(final Object self, final Class<T> clazz) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   214
        final Object realSelf;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   215
        if(self == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   216
            realSelf = global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   217
        } else if (!(self instanceof ScriptObject)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   218
            realSelf = ScriptObjectMirror.unwrap(self, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   219
        } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   220
            realSelf = self;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   221
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   222
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   223
            final ScriptObject oldGlobal = Context.getGlobal();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   224
            try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   225
                if(oldGlobal != global) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   226
                    setNashornGlobal(global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   227
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   228
                return clazz.cast(JavaAdapterFactory.getConstructor(realSelf.getClass(), clazz).invoke(realSelf));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   229
            } finally {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   230
                if(oldGlobal != global) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   231
                    setNashornGlobal(oldGlobal);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   232
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   233
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   234
        } catch(final RuntimeException|Error e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   235
            throw e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   236
        } catch(final Throwable t) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   237
            throw new RuntimeException(t);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   238
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   239
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   240
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   241
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   242
    public <T> T getInterface(final Class<T> clazz) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   243
        return getInterfaceInner(null, clazz);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   244
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   245
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   246
    @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   247
    public <T> T getInterface(final Object self, final Class<T> clazz) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   248
        if (self == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   249
            throw new IllegalArgumentException("script object can not be null");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   250
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   251
        return getInterfaceInner(self, clazz);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   252
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   253
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   254
    // These are called from the "engine.js" script
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   255
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   256
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   257
     * This hook is used to search js global variables exposed from Java code.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   258
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   259
     * @param self 'this' passed from the script
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   260
     * @param ctxt current ScriptContext in which name is searched
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   261
     * @param name name of the variable searched
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   262
     * @return the value of the named variable
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   263
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   264
    public Object __noSuchProperty__(final Object self, final ScriptContext ctxt, final String name) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   265
        final int scope = ctxt.getAttributesScope(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   266
        if (scope != -1) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   267
            return ScriptObjectMirror.unwrap(ctxt.getAttribute(name, scope), global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   268
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   269
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   270
        if (self == UNDEFINED) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   271
            // scope access and so throw ReferenceError
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   272
            referenceError(global, "not.defined", name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   273
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   274
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   275
        return UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   276
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   277
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   278
    /**
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   279
     * This hook is used to call js global functions exposed from Java code.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   280
     *
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   281
     * @param self 'this' passed from the script
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   282
     * @param ctxt current ScriptContext in which method is searched
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   283
     * @param name name of the method
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   284
     * @param args arguments to be passed to the method
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   285
     * @return return value of the called method
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   286
     */
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   287
    public Object __noSuchMethod__(final Object self, final ScriptContext ctxt, final String name, final Object args) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   288
        final int scope = ctxt.getAttributesScope(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   289
        Object value;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   290
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   291
        if (scope != -1) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   292
            value = ctxt.getAttribute(name, scope);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   293
        } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   294
            if (self == UNDEFINED) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   295
                referenceError(global, "not.defined", name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   296
            } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   297
                typeError(global, "no.such.function", name, ScriptRuntime.safeToString(global));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   298
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   299
            return UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   300
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   301
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   302
        value = ScriptObjectMirror.unwrap(value, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   303
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   304
        if (value instanceof Method) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   305
            final Method method = (Method) value;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   306
            final int    mods   = method.getModifiers();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   307
            if (Modifier.isStatic(mods) && Modifier.isPublic(mods)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   308
               value = MH.find((Method)value);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   309
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   310
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   311
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   312
        if (value instanceof MethodHandle) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   313
            value = ((GlobalObject)global).newScriptFunction(name, (MethodHandle)value, null, false);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   314
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   315
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   316
        if (!(value instanceof ScriptFunction)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   317
            typeError(global, "not.a.function", name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   318
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   319
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   320
        if (value instanceof ScriptFunction) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   321
            return ScriptObjectMirror.unwrap(ScriptRuntime.apply((ScriptFunction)value, global, args), global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   322
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   323
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   324
        typeError(global, "not.a.function", ScriptRuntime.safeToString(name));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   325
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   326
        return UNDEFINED;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   327
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   328
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   329
    private void evalEngineScript() throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   330
        final URL url = NashornScriptEngine.class.getResource("resources/engine.js");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   331
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   332
            final InputStream is = url.openStream();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   333
            put(ScriptEngine.FILENAME, url);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   334
            try (final InputStreamReader isr = new InputStreamReader(is)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   335
                eval(isr);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   336
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   337
        } catch (final IOException e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   338
            throw new ScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   339
        } finally {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   340
            put(ScriptEngine.FILENAME, null);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   341
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   342
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   343
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   344
    // scripts should see "context" and "engine" as variables
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   345
    private void setContextVariables(final ScriptContext ctxt) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   346
        ctxt.setAttribute("context", ctxt, ScriptContext.ENGINE_SCOPE);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   347
        // current ScriptContext exposed as "context"
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   348
        global.set("context", ctxt, false);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   349
        Object args = ctxt.getAttribute("arguments");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   350
        // if no arguments passed, make it empty array
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   351
        if (args == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   352
            args = ScriptRuntime.EMPTY_ARRAY;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   353
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   354
        args = ((GlobalObject)global).wrapAsObject(args);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   355
        global.set("arguments", args, false);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   356
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   357
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   358
    private Object invokeImpl(final Object selfObject, final String name, final Object... args) throws ScriptException, NoSuchMethodException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   359
        final ScriptObject oldGlobal     = Context.getGlobal();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   360
        final boolean globalChanged = (oldGlobal != global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   361
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   362
        Object self = selfObject;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   363
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   364
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   365
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   366
                setNashornGlobal(global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   367
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   368
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   369
            ScriptObject sobj;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   370
            Object       value;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   371
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   372
            self = ScriptObjectMirror.unwrap(self, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   373
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   374
            // FIXME: should convert when self is not ScriptObject
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   375
            if (self instanceof ScriptObject) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   376
                sobj = (ScriptObject)self;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   377
                value = sobj.get(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   378
            } else if (self == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   379
                self  = global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   380
                sobj  = global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   381
                value = sobj.get(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   382
            } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   383
                // Find the java method and make a ScriptFunction of it.
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   384
                final Method[] methods = self.getClass().getMethods();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   385
                Method target = null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   386
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   387
                for (final Method mth : methods) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   388
                    // choose the right overload by number of arguments -- don't
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   389
                    // care overload resolution for now..
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   390
                    if (mth.getName().equals(name) &&
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   391
                        mth.getParameterTypes().length == args.length) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   392
                        target = mth;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   393
                        break;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   394
                    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   395
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   396
                if (target == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   397
                    throw new NoSuchMethodException(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   398
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   399
                final GlobalObject gobj = (GlobalObject) global;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   400
                value = gobj.newScriptFunction(name, MH.find(target), null, false);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   401
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   402
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   403
            if (value instanceof ScriptFunction) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   404
                final Object res;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   405
                try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   406
                    res = ScriptRuntime.apply((ScriptFunction)value, self, ScriptObjectMirror.unwrapArray(args, global));
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   407
                } catch (final Exception e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   408
                    throwAsScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   409
                    throw new AssertionError("should not reach here");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   410
                }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   411
                return ScriptObjectMirror.wrap(res, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   412
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   413
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   414
            throw new NoSuchMethodException(name);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   415
        } finally {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   416
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   417
                setNashornGlobal(oldGlobal);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   418
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   419
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   420
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   421
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   422
    private Object evalImpl(final char[] buf, final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   423
        return evalImpl(compileImpl(buf, ctxt), ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   424
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   425
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   426
    private Object evalImpl(final ScriptFunction script, final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   427
        if (script == null) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   428
            return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   429
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   430
        final ScriptObject oldGlobal = Context.getGlobal();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   431
        final boolean globalChanged = (oldGlobal != global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   432
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   433
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   434
                setNashornGlobal(global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   435
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   436
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   437
            setContextVariables(ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   438
            Object res = ScriptRuntime.apply(script, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   439
            res = ScriptObjectMirror.wrap(res, global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   440
            return (res == UNDEFINED) ? null : res;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   441
        } catch (final Exception e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   442
            throwAsScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   443
            throw new AssertionError("should not reach here");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   444
        } finally {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   445
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   446
                setNashornGlobal(oldGlobal);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   447
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   448
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   449
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   450
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   451
    private static void throwAsScriptException(final Exception e) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   452
        if (e instanceof ScriptException) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   453
            throw (ScriptException)e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   454
        } else if (e instanceof NashornException) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   455
            final NashornException ne = (NashornException)e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   456
            final ScriptException se = new ScriptException(
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   457
                ne.getMessage(), ne.getFileName(),
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   458
                ne.getLineNumber(), ne.getColumnNumber());
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   459
            se.initCause(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   460
            throw se;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   461
        } else if (e instanceof RuntimeException) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   462
            throw (RuntimeException)e;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   463
        } else {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   464
            // wrap any other exception as ScriptException
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   465
            throw new ScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   466
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   467
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   468
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   469
    private CompiledScript asCompiledScript(final ScriptFunction script) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   470
        return new CompiledScript() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   471
            @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   472
            public Object eval(final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   473
                return evalImpl(script, ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   474
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   475
            @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   476
            public ScriptEngine getEngine() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   477
                return NashornScriptEngine.this;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   478
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   479
        };
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   480
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   481
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   482
    private ScriptFunction compileImpl(final char[] buf, final ScriptContext ctxt) throws ScriptException {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   483
        final ScriptObject oldGlobal = Context.getGlobal();
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   484
        final boolean globalChanged = (oldGlobal != global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   485
        try {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   486
            final Object val = ctxt.getAttribute(ScriptEngine.FILENAME);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   487
            final String fileName = (val != null) ? val.toString() : "<eval>";
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   488
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   489
            // !!HACK!! do not evaluate "init.js" from jrunscript tool!!
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   490
            if ("<system-init>".equals(fileName)) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   491
                return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   492
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   493
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   494
            final Source source = new Source(fileName, buf);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   495
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   496
                setNashornGlobal(global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   497
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   498
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   499
            setContextVariables(ctxt);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   500
            return nashornContext.compileScript(source, global, nashornContext._strict);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   501
        } catch (final Exception e) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   502
            throwAsScriptException(e);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   503
            throw new AssertionError("should not reach here");
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   504
        } finally {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   505
            if (globalChanged) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   506
                setNashornGlobal(oldGlobal);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   507
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   508
        }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   509
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   510
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   511
    // don't make this public!!
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   512
    static void setNashornGlobal(final ScriptObject global) {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   513
        AccessController.doPrivileged(new PrivilegedAction<Void>() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   514
            @Override
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   515
            public Void run() {
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   516
               Context.setGlobal(global);
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   517
               return null;
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   518
            }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   519
        });
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   520
    }
e63b63819133 8005403: Open-source Nashorn
jlaskey
parents:
diff changeset
   521
}