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

org.math.R.RserverConf Maven / Gradle / Ivy

There is a newer version: 3.1.8
Show newest version
package org.math.R;

import java.io.IOException;
import java.net.BindException;
import java.net.ServerSocket;
import java.util.Properties;
import org.rosuda.REngine.Rserve.RConnection;
import org.rosuda.REngine.Rserve.RserveException;

public class RserverConf {

    public static String DEFAULT_RSERVE_HOST = "localhost";
    RConnection connection;
    public String host;
    public int port;
    public String login;
    public String password;
    //public String RLibPath;
    public Properties properties;
    //public String http_proxy;

    public RserverConf(String RserverHostName, int RserverPort, String login, String password, Properties props) {
        this.host = RserverHostName;
        this.port = RserverPort;
        this.login = login;
        this.password = password;
        properties = props;
    }
    public static long CONNECT_TIMEOUT = 2000;

    public abstract class TimeOut {

        /**
         * @return the result
         */
        public Object getResult() {
            return result;
        }

        public class TimeOutException extends Exception {

            public TimeOutException(String why) {
                super(why);
            }
        }

        private class TimeoutThread implements Runnable {

            public void run() {
                Object res = command();
                synchronized (TimeOut.this) {
                    if (timedOut && res != null) {
                    } else {
                        result = res;
                        TimeOut.this.notify();
                    }
                }
            }
        }
        private boolean timedOut = false;        
        private Object result = null;

        protected TimeOut() {
        }

        public synchronized void execute(long timeout) throws TimeOutException {
            new Thread(new TimeoutThread()).start();

            try {
                this.wait(timeout);
            } catch (InterruptedException e) {
                if (getResult() == null) {
                    timedOut = true;
                    result = defaultResult();
                    throw new TimeOutException("timed out");
                } else {
                    return;
                }
            }

            if (getResult() != null) {
                return;
            } else {
                timedOut = true;
                result = defaultResult();
                throw new TimeOutException("timed out");
            }
        }

        protected abstract Object defaultResult();

        protected abstract Object command();
    }

    /*private class ConnectionThread implements Runnable {
    
    public void run() {
    try {
    if (host == null) {
    if (port > 0) {
    connection = new RConnection(DEFAULT_RSERVE_HOST, port);
    } else {
    connection = new RConnection();
    }
    if (connection.needLogin()) {
    connection.login(login, password);
    }
    } else {
    if (port > 0) {
    connection = new RConnection(host, port);
    } else {
    connection = new RConnection(host);
    }
    if (connection.needLogin()) {
    connection.login(login, password);
    }
    }
    } catch (RserveException ex) {
    //ex.printStackTrace();
    //return null;
    }
    
    synchronized (this) {
    this.notify();
    }
    }
    }*/
    public synchronized RConnection connect() {
        //Logger.err.print("Connecting " + toString()+" ... ");

        TimeOut t = new TimeOut() {

            protected Object defaultResult() {
                return -2;
            }

            protected Object command() {
                try {
                    if (host == null) {
                        if (port > 0) {
                            connection = new RConnection(DEFAULT_RSERVE_HOST, port);
                        } else {
                            connection = new RConnection();
                        }
                        if (connection.needLogin()) {
                            connection.login(login, password);
                        }
                    } else {
                        if (port > 0) {
                            connection = new RConnection(host, port);
                        } else {
                            connection = new RConnection(host);
                        }
                        if (connection.needLogin()) {
                            connection.login(login, password);
                        }
                    }
                    return 0;
                } catch (RserveException ex) {
                    Log.Err.println("Failed to connect: " + ex.getMessage());
                    return -1;
                }
            }
        };

        try {
            t.execute(CONNECT_TIMEOUT);
        } catch (Exception e) {
            Log.Err.println("  failed: " + e.getMessage());
        }


        /*new Thread(new ConnectionThread()).start();
        
        try {
        this.wait(CONNECT_TIMEOUT);
        
        } catch (InterruptedException ie) {
        }*/

        if (connection != null && connection.isConnected()) {
            if (properties != null) {
                for (String p : properties.stringPropertyNames()) {
                    try {
                        connection.eval("Sys.setenv(" + p + "='" + properties.getProperty(p) + "')");
                    } catch (RserveException ex) {
                        ex.printStackTrace();
                    }
                }
            }

            return connection;
        } else {
            Log.Err.println("Connection " + toString() + " failed.");
            return null;
        }

    }
    public final static int RserverDefaultPort = 6311;
    private static int RserverPort = RserverDefaultPort; //used for windows multi-session emulation. Incremented at each new Rscript instance.

    public static boolean isPortAvailable(int p) {
        try {
            ServerSocket test = new ServerSocket(p);
            test.close();
        } catch (BindException e) {
            return false;
        } catch (IOException e) {
            return false;
        }
        return true;
    }

    public static final boolean UNIX_OPTIMIZE = false; // if we want to re-use older sessions. May wrongly fil if older session is already stucked...
    
    public static RserverConf newLocalInstance(Properties p) {
        RserverConf server = null;
        if (RserveDaemon.isWindows() || !UNIX_OPTIMIZE) {
            while (!isPortAvailable(RserverPort)) {
                RserverPort++;
            }
            server = new RserverConf(null, RserverPort, null, null, p);
        } else { // Unix supports multi-sessions natively, so no need to open a different Rserve on a new port
            server = new RserverConf(null, -1, null, null, p);
        }
        return server;
    }

    public boolean isLocal() {
        return host == null || host.equals(DEFAULT_RSERVE_HOST) || host.equals("127.0.0.1");
    }

    @Override
    public String toString() {
        return RURL_START + (login != null ? (login + ":" + password + "@") : "") + (host == null ? DEFAULT_RSERVE_HOST : host) + (port > 0 ? ":" + port : "") /*+ " http_proxy=" + http_proxy + " RLibPath=" + RLibPath*/;
    }
    public final static String RURL_START = "R://";

    public static RserverConf parse(String RURL) {
        String login = null;
        String passwd = null;
        String host = null;
        int port = -1;
        try {
            String hostport = null;
            if (RURL.contains("@")) {
                String loginpasswd = RURL.split("@")[0].substring((RURL_START).length());
                login = loginpasswd.split(":")[0];
                if (login.equals("user.name")) {
                    login = System.getProperty("user.name");
                }
                passwd = loginpasswd.split(":")[1];
                hostport = RURL.split("@")[1];
            } else {
                hostport = RURL.substring((RURL_START).length());
            }

            if (hostport.contains(":")) {
                host = hostport.split(":")[0];
                port = Integer.parseInt(hostport.split(":")[1]);
            } else {
                host = hostport;
            }

            return new RserverConf(host, port, login, passwd, null);
        } catch (Exception e) {
            throw new IllegalArgumentException("Impossible to parse " + RURL + ":\n  host=" + host + "\n  port=" + port + "\n  login=" + login + "\n  password=" + passwd);
        }

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy