All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.cometd.javascript.JavaScript Maven / Gradle / Ivy

There is a newer version: 4.0.9
Show newest version
/*
 * Copyright (c) 2008-2019 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package org.cometd.javascript;

import java.net.URL;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.FutureTask;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.RejectedExecutionException;

import javax.script.Bindings;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.script.ScriptException;

import jdk.nashorn.api.scripting.ScriptObjectMirror;
import jdk.nashorn.api.scripting.URLReader;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

public class JavaScript implements Runnable {
    private static final ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn");

    private final Logger logger = LoggerFactory.getLogger(getClass());
    private final BlockingQueue> queue = new LinkedBlockingQueue<>();
    private final Thread thread = new Thread(this, "javascript");
    private final Bindings bindings;
    private volatile boolean running;

    public JavaScript() {
        bindings = engine.createBindings();
        bindings.put("javaScript", this);
    }

    public void init() throws Exception {
        queue.clear();
        running = true;
        thread.start();
    }

    public void destroy() throws Exception {
        running = false;
        for (FutureTask task : queue) {
            task.cancel(false);
        }
        thread.interrupt();
        thread.join();
        bindings.clear();
    }

    @Override
    public void run() {
        try {
            while (running) {
                FutureTask task = queue.take();
                task.run();
            }
        } catch (InterruptedException x) {
            // We've been destroyed, just exit
        }
    }

    public  T invoke(boolean sync, ScriptObjectMirror thiz, Object function, Object... arguments) {
        FutureTask task = new FutureTask<>(() -> {
            try {
                if (function instanceof ScriptObjectMirror) {
                    return (T)((ScriptObjectMirror)function).call(thiz, arguments);
                } else {
                    String name = function.toString();
                    if (thiz.hasMember(name)) {
                        Object member = thiz.getMember(name);
                        if (member instanceof ScriptObjectMirror) {
                            if (logger.isDebugEnabled()) {
                                logger.debug("Invoking function {}", name);
                            }
                            return (T)((ScriptObjectMirror)member).call(thiz, arguments);
                        } else {
                            logger.info("Function {} not defined in {}", name, thiz);
                            return null;
                        }
                    } else {
                        logger.info("Function {} not a member of {}", name, thiz);
                        return null;
                    }
                }
            } catch (Throwable x) {
                logger.info("Exception while trying to invoke " + function, x);
                throw x;
            }
        });
        submit(task);

        if (!sync) {
            return null;
        }

        return result(task);
    }

    public  T evaluate(URL url) {
        FutureTask task = new FutureTask<>(() -> {
            try(URLReader r = new URLReader(url)) {
                return (T)engine.eval(r, bindings);
            } catch (ScriptException x) {
                throw new JavaScriptException(x);
            }
        });
        return submitTask(task);
    }

    public  T evaluate(String name, String code) {
        String script = name == null ? code : String.format("//# sourceURL=%s%n%s", name, code);
        FutureTask task = new FutureTask<>(() -> {
            try {
                bindings.put(ScriptEngine.FILENAME, name);
                return (T)engine.eval(script, bindings);
            } catch (ScriptException x) {
                throw new JavaScriptException(x);
            }
        });
        return submitTask(task);
    }

    private  T submitTask(FutureTask task) {
        submit(task);
        return result(task);
    }

    public  T get(String key) {
        FutureTask task = new FutureTask<>(() -> getAsync(key));
        submit(task);
        return result(task);
    }

    public  T getAsync(String key) {
        return (T)bindings.get(key);
    }

    public void putAsync(String key, Object value) {
        bindings.put(key, value);
    }

    private void submit(FutureTask task) {
        if (Thread.currentThread() == thread) {
            task.run();
        } else {
            if (running) {
                queue.offer(task);
            } else {
                throw new RejectedExecutionException();
            }
        }
    }

    private  T result(FutureTask task) {
        try {
            return task.get();
        } catch (InterruptedException x) {
            return null;
        } catch (ExecutionException x) {
            Throwable xx = x.getCause();
            if (xx instanceof Error) {
                throw (Error)xx;
            }
            throw (RuntimeException)xx;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy