org.math.R.Rsession Maven / Gradle / Ivy
package org.math.R;
import java.io.File;
import java.io.FileFilter;
import java.io.PrintStream;
import java.net.InetAddress;
import java.net.UnknownHostException;
import java.util.Arrays;
import java.util.Calendar;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
/**
* @author richet
*/
public abstract class Rsession implements RLog {
public static final String HEAD_TRY = "[?] ";//-try- ";
public boolean TRY_MODE_DEFAULT = true;
public boolean TRY_MODE = false;
public static final String CAST_ERROR = "Cannot cast ";
private static final String __ = " ";
public static final String _PACKAGE_ = " package ";
RLog console;
public static final String PACKAGEINSTALLED = "Package installed.";
public static final String PACKAGELOADED = "Package loaded.";
static String separator = ",";
String install_packages_moreargs = "";
public class RException extends Exception {
public RException(String cause) {
this(cause, false);
}
public RException(String cause, boolean details) {
super(cause + "\nR: " + getLastLogEntry() + "\nR! " + getLastError() + (details ? "\nR: " + Arrays.asList(ls()) : ""));
}
}
//
List loggers;
public boolean debug;
//** GLG HACK: Logging fix **//
// No sink file (Passed to false) a lot faster not to sink the output
boolean SINK_OUTPUT = true, SINK_MESSAGE = true;
public void sinkOutput(boolean s) {
SINK_OUTPUT = s;
}
public void sinkMessage(boolean s) {
SINK_MESSAGE = s;
}
// GLG HACK: fixed sink file in case of multiple instances
// (Appending the port number of the instance to file name)
String SINK_FILE_BASE = System.getProperty("java.io.tmpdir") + File.separator + ".Rout";
String SINK_FILE = null;
String lastOuput = "";
String lastMessage = "";
public static String fixPathSeparator(String p) {
return p.replace(File.separatorChar, '/');
}
void cleanupListeners() {
if (loggers != null) {
loggers.clear();
/*while (!loggers.isEmpty()) {
removeLogger(loggers.get(0));
}*/
}
if (busy != null) {
busy.clear();
/*while (!busy.isEmpty()) {
removeBusyListener(busy.get(0));
}*/
}
if (updateObjects != null) {
updateObjects.clear();
/*while (!updateObjects.isEmpty()) {
removeUpdateObjectsListener(updateObjects.get(0));
}*/
}
if (eval != null) {
eval.clear();
/*while (!rawEval.isEmpty()) {
removeEvalListener(rawEval.get(0));
}*/
}
}
public abstract boolean isAvailable();
protected void finalize() throws Throwable {
closeLog();
super.finalize();
}
public void addLogger(RLog l) {
if (!loggers.contains(l)) {
loggers.add(l);
}
}
public void removeLogger(RLog l) {
if (loggers.contains(l)) {
l.closeLog();
loggers.remove(l);
}
}
@Override
public void closeLog() {
if (loggers != null) {
for (RLog l : loggers) {
if (l != null) {
l.closeLog();
}
}
}
}
public void end() {
closeLog();
}
List busy = new LinkedList();
public void addBusyListener(BusyListener b) {
if (!busy.contains(b)) {
busy.add(b);
}
}
public void removeBusyListener(BusyListener b) {
if (busy.contains(b)) {
busy.remove(b);
}
}
public void setBusy(boolean bb) {
for (BusyListener b : busy) {
b.setBusy(bb);
}
}
List updateObjects = new LinkedList();
public void addUpdateObjectsListener(UpdateObjectsListener b) {
if (!updateObjects.contains(b)) {
updateObjects.add(b);
}
}
public void removeUpdateObjectsListener(UpdateObjectsListener b) {
if (updateObjects.contains(b)) {
b.setTarget(null);
updateObjects.remove(b);
}
}
List eval = new LinkedList();
public void addEvalListener(EvalListener b) {
if (!eval.contains(b)) {
eval.add(b);
}
}
public void removeEvalListener(EvalListener b) {
if (eval.contains(b)) {
eval.remove(b);
}
}
//
//
public static String cat(double[] array) {
if (array == null || array.length == 0) {
return "NA";
}
String o = array[0] + "";
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += (separator + (array[i] + ""));
}
}
return o;
}
public static String cat(int[] array) {
if (array == null || array.length == 0) {
return "NA";
}
String o = array[0] + "";
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += (separator + (array[i] + ""));
}
}
return o;
}
public static String cat(double[][] array) {
if (array == null || array.length == 0 || array[0].length == 0) {
return "NA";
}
String o = cat(array[0]);
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += "\n" + cat(array[i]);
}
}
return o;
}
public static String cat(int[][] array) {
if (array == null || array.length == 0 || array[0].length == 0) {
return "NA";
}
String o = cat(array[0]);
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += "\n" + cat(array[i]);
}
}
return o;
}
public static String cat(Object[] array) {
if (array == null || array.length == 0 || array[0] == null) {
return "";
}
String o = array[0].toString();
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += (separator + (array[i] == null ? "" : array[i].toString()));
}
}
return o;
}
public static String cat(String sep, String[] array) {
if (array == null || array.length == 0 || array[0] == null) {
return "";
}
String o = array[0].toString();
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += (sep + (array[i] == null ? "" : array[i].toString()));
}
}
return o;
}
public static String cat(Object[][] array) {
if (array == null || array.length == 0 || array[0].length == 0) {
return "NA";
}
String o = cat(array[0]);
if (array.length > 1) {
for (int i = 1; i < array.length; i++) {
o += "\n" + cat(array[i]);
}
}
return o;
}
//
/**
* Map java File object to R path (as string)
*
* @param path java File object
* @return R path with suitable level delimiter "/"
*/
public static String toRpath(File path) {
return toRpath(path.getAbsolutePath());
}
/**
* Map java path to R path (as string)
*
* @param path java string path
* @return R path with suitable level delimiter "/"
*/
public static String toRpath(String path) {
return path.replaceAll("\\\\", "/");
}
/**
* create a new Rsession.
*
* @param console RLog for R output
*/
public Rsession(final RLog console) {
this.console = console;
loggers = new LinkedList();
loggers.add(console);
// Make sink file specific to current Rserve instance
SINK_FILE = SINK_FILE_BASE;
}
/**
* create rsession using System as a logger
*
* @param p PrintStream for R output
*/
public Rsession(final PrintStream p) {
this(new RLog() {
public void log(String string, Level level) {
if (level == Level.WARNING) {
p.print("(?) ");
} else if (level == Level.ERROR) {
p.print("(!!) ");
} else if (level == Level.OUTPUT) {
p.print("> ");
}
p.println(string);
}
public void closeLog() {
p.close();
}
});
}
void setenv(Properties properties) {
if (properties != null) {
for (String p : properties.stringPropertyNames()) {
if (p != null) {
try {
log("Setting environment " + p + ": '" + properties.getProperty(p).replaceAll("\\:([^/])(.*)\\@", ":???@") + "'", Level.INFO);
boolean done = asLogical(silentlyRawEval("Sys.setenv(" + p + "='" + properties.getProperty(p) + "')", false));
if (!done) {
log("Failed setting environment " + p, Level.WARNING);
}
} catch (Exception ex) {
log(ex.getMessage(), Level.WARNING);
ex.printStackTrace();
}
}
}
}
}
String lastLog = "";
int repeated = 0;
public String getLastLogEntry() {
return lastLog;
}
private void err_println(String message, RLog.Level level) {
if (level == Level.ERROR) {
try {
message = message + "\n R> " + getLastLogEntry();
} catch (Exception e) {
message = message + "\n ! " + e.getMessage();
}
}
}
String hostname() {
String hostname = "?";
try {
hostname = InetAddress.getLocalHost().getHostName();
} catch (UnknownHostException e) {
hostname = "?";
}
return hostname;
}
String nb_header = "---\n"
+ "title: '" + this.getClass().getName() + " notebook'\n"
+ "date: '" + Calendar.getInstance().getTime().toString() + "'\n"
+ "author: '" + System.getProperty("user.name") + "'\n"
+ "host: '" + hostname() + "'\n"
+ "output: html_notebook\n"
+ "---\n\n";
public void note_header(String title, String author) {
nb_header = "---\n"
+ "title: '" + title + "'\n"
+ "date: '" + Calendar.getInstance().getTime().toString() + "'\n"
+ "author: '" + author + "'\n"
+ "host: '" + hostname() + "'\n"
+ "output: html_notebook\n"
+ "---\n\n";
}
StringBuilder nb = new StringBuilder();
public void note_text(String txt) {
nb.append(txt).append("\n");
}
public void note_code(String... code) {
StringBuilder codes = new StringBuilder();
for (String c : code) {
codes.append(c + "\n");
}
note_code(codes.toString());
}
public void note_code(String code) {
nb.append("```{r}\n").append(code).append("\n```\n");
}
protected static String toRcode(Object o) {
if (o instanceof double[]) {
return toRcode((double[]) o);
} else if (o instanceof double[][]) {
return toRcode((double[][]) o);
} else if (o instanceof Double[]) {
return toRcode((Double[]) o);
} else if (o instanceof Double[][]) {
return toRcode((Double[][]) o);
} else if (o instanceof String[]) {
return toRcode((String[]) o);
} else if (o instanceof String) {
return "'" + o + "'";
} else if (o instanceof Integer) {
return o.toString();
} else if (o instanceof Double) {
return o.toString();
} else if (o instanceof File) {
return toRcode(((File) o).getName());
}
return "stop('Cannot code in R object:" + o + "')";
}
protected static String toRcode(double[] d) {
String ds = "c( ";
for (double e : d) {
ds = ds + e + ",";
}
return ds.substring(0, ds.length() - 1) + ")";
}
protected static String toRcode(Double[] d) {
String ds = "c( ";
for (double e : d) {
ds = ds + e + ",";
}
return ds.substring(0, ds.length() - 1) + ")";
}
protected static String toRcode(double[][] d) {
String ds = "rbind( ";
for (double[] e : d) {
ds = ds + toRcode(e) + ",";
}
return ds.substring(0, ds.length() - 1) + ")";
}
protected static String toRcode(Double[][] d) {
String ds = "rbind( ";
for (Double[] e : d) {
ds = ds + toRcode(e) + ",";
}
return ds.substring(0, ds.length() - 1) + ")";
}
protected static String toRcode(String[] d) {
String ds = "c( ";
for (String e : d) {
ds = ds + "'" + e + "',";
}
return ds.substring(0, ds.length() - 1) + ")";
}
protected static String toRcode(Map l) {
String sl = "list(";
for (Object k : l.keySet()) {
if (l.get(k) instanceof double[]) {
double[] v = (double[]) l.get(k);
sl = sl + k + "=" + toRcode(v) + ",";
} else {
sl = sl + k + "=" + l.get(k) + ",";
}
}
return sl.substring(0, sl.length() - 1) + ")";
}
public String notebook() {
return nb_header.toString() + "\n" + nb.toString().replace("```\n```{r}\n", "");
}
@Override
public void log(String message, Level level) {
if (message != null && message.trim().length() > 0 && !message.trim().equals("\n") && level == Level.OUTPUT) {
// nothing to do
} else {
if (message == null) {
return;
} else {
message = message.trim();
while (message.endsWith("\n")) {
message = message.substring(0, message.length() - 2); // to delete when many return chars
}
}
if (message.equals(lastLog) && repeated < 100) {
repeated++;
} else {
if (repeated > 0) {
err_println(" Repeated " + repeated + " times.", level);
repeated = 0;
lastLog = message;
err_println(message, level);
} else {
lastLog = message;
err_println(message, level);
}
}
}
if (loggers != null) {
for (RLog l : loggers) {
if (l != null) {
l.log(message, level);
}
}
}
}
/**
* @return available R commands
*/
public String[] listCommands() {
silentlyRawEval(".keyWords <- function() {n <- length(search());result <- c();for (i in 1:n) {result <- c(result,ls(pos=i,all.names=TRUE))}; result}");
Object rexp = silentlyRawEval(".keyWords()");
String as[] = null;
try {
if (rexp != null && (as = asStrings(rexp)) != null) {
return as;
} else {
return null;
}
} catch (Exception ex) {
log(HEAD_ERROR + ex.getMessage() + "\n listCommands()", Level.ERROR);
return null;
}
}
//
public static String DEFAULT_REPOS = "http://cloud.r-project.org";
public String repos = DEFAULT_REPOS;
/**
* @param url CRAN repository to use for packages installation (eg http://cran.r-project.org)
*/
public void setCRANRepository(String url) {
repos = url;
}
/**
* @return CRAN repository used for packages installation
*/
public String getCRANRepository() {
return repos;
}
private static String loadedpacks = "loadedpacks";
/**
* Check for package loaded in R environment.
*
* @param pack R package name
* @return package loading status
*/
public boolean isPackageLoaded(String pack) {
silentlyVoidEval(loadedpacks + "<-.packages()", false);
boolean isloaded = false;
try {
Object i = silentlyRawEval("is.element(set=" + loadedpacks + ",el='" + pack + "')");
if (i != null) {
isloaded = asLogical(i);
}
} catch (Exception ex) {
log(HEAD_ERROR + ex.getMessage() + "\n isPackageLoaded(String pack=" + pack + ")", Level.ERROR);
}
if (isloaded) {
log(_PACKAGE_ + pack + " is loaded.", Level.INFO);
} else {
log(_PACKAGE_ + pack + " is not loaded.", Level.INFO);
}
//silentlyEval("rm(" + loadedpacks + ")");
return isloaded;
}
private static String packs = "packs";
/**
* Check for package installed in R environment.
*
* @param pack R package name
* @param version R package version
* @return package loading status
*/
public boolean isPackageInstalled(String pack, String version) {
silentlyVoidEval(packs + "<-installed.packages(noCache=TRUE" + install_packages_moreargs + ")", false);
boolean isinstalled = false;
Object r = silentlyRawEval("is.element(set=row.names(" + packs + "),el='" + pack + "')");
try {
if (r != null) {
isinstalled = (asInteger(r) == 1);
} else {
log(HEAD_ERROR + "Could not list installed packages" + "\n isPackageInstalled(String pack=" + pack + ", String version=" + version + ")", Level.ERROR);
}
} catch (Exception ex) {
log(HEAD_ERROR + ex.getMessage() + "\n isPackageInstalled(String pack=" + pack + ", String version=" + version + ")", Level.ERROR);
}
if (isinstalled) {
log(_PACKAGE_ + pack + " is installed.", Level.INFO);
} else {
log(_PACKAGE_ + pack + " is not installed.", Level.INFO);
}
if (isinstalled && version != null && version.length() > 0) {
try {
isinstalled = asLogical(silentlyRawEval(packs + "['" + pack + "','Version'] == \"" + version + "\""));
} catch (Exception ex) {
log(HEAD_ERROR + ex.getMessage() + "\n isPackageInstalled(String pack=" + pack + ", String version=" + version + ")", Level.ERROR);
}
try {
log(" version of package " + pack + " is " + asString(silentlyRawEval(packs + "['" + pack + "','Version']")), Level.INFO);
} catch (Exception ex) {
log(HEAD_ERROR + ex.getMessage() + "\n isPackageInstalled(String pack=" + pack + ", String version=" + version + ")", Level.ERROR);
}
if (isinstalled) {
log(_PACKAGE_ + pack + " (" + version + ") " + " is installed.", Level.INFO);
} else {
log(_PACKAGE_ + pack + " (" + version + ") " + " is not installed.", Level.INFO);
}
}
//silentlyEval("rm(" + packs + ")");
return isinstalled;
}
/**
* Start installation procedure of R packages
*
* @param pack packages to install
* @param load automatically load packages after successfull installation
* @return installation status
*/
public String installPackages(String[] pack, boolean load) {
String resall = "";
for (String pv : pack) {
String res = installPackage(pv, load);
if (load) {
if (!res.equals(PACKAGELOADED)) {
resall += "\n" + res;
}
} else {
if (!res.equals(PACKAGEINSTALLED)) {
resall += "\n" + res;
}
}
}
if (resall.length() > 0) {
return resall;
} else {
return load ? PACKAGELOADED : PACKAGEINSTALLED;
}
}
/**
* Start installation procedure of local R package
*
* @param pack package file to install
* @param load automatically load package after successfull installation
* @return installation status
*/
public String installPackage(File pack, boolean load) {
try {
rawEval("install.packages('" + pack.getAbsolutePath().replace("\\", "/") + "',repos=NULL,quiet=T" + install_packages_moreargs + ")");
} catch (Exception ex) {
log(ex.getMessage(), Level.ERROR);
}
log(" request package " + pack + " install...", Level.INFO);
String name = pack.getName();
String version = null;
if (name.contains("_")) {
name = name.substring(0, name.indexOf("_"));
version = name.substring(name.indexOf("_") + 1);
version = version.substring(0, version.indexOf("."));
} else if (name.contains(".")) {
name = name.substring(0, name.indexOf("."));
}
if (isPackageInstalled(name, version)) {
log(_PACKAGE_ + pack + " version " + version + " installation sucessfull.", Level.INFO);
if (load) {
return loadPackage(name);
} else {
return PACKAGEINSTALLED;
}
} else {
log(_PACKAGE_ + pack + " installation failed.", Level.ERROR);
return "Impossible to install package " + pack + " !";
}
}
abstract boolean isWindows();
abstract boolean isLinux();
abstract boolean isMacOSX();
/**
* Start installation procedure of local R package
*
* @param pack package to install
* @param dir directory where package file (.zip, .tar.gz or .tgz) is located.
* @param load automatically load package after successfull installation
* @return installation status
*/
public String installPackage(final String pack, File dir, boolean load) {
log(" trying to load package " + pack, Level.INFO);
if (isPackageInstalled(pack, null)) {
log(_PACKAGE_ + pack + " already installed.", Level.INFO);
if (load) {
return loadPackage(pack);
} else {
return PACKAGEINSTALLED;
}
} else {
log(_PACKAGE_ + pack + " not yet installed.", Level.INFO);
}
File[] pack_files = (dir == null ? null : dir.listFiles(new FileFilter() {
public boolean accept(File pathname) {
if (!pathname.getName().contains(pack)) {
return false;
}
if (isWindows()) {
return pathname.getName().endsWith(".zip");
}
if (isLinux()) {
return pathname.getName().endsWith(".tar.gz");
}
if (isMacOSX()) {
return pathname.getName().endsWith(".tgz");
}
return false;
}
}));
if (pack_files == null || pack_files.length == 0) {
log(" impossible to find package " + pack + " in directory " + dir.getAbsolutePath() + " !", Level.WARNING);
return "Impossible to find package " + pack + " in directory " + dir.getAbsolutePath() + " !";
} else {
log(" found package " + pack + " : " + pack_files[0].getAbsolutePath(), Level.INFO);
}
return installPackage(pack_files[0], load);
}
/**
* Start installation procedure of CRAN R package
*
* @param pack package to install
* @param load automatically load package after successfull installation
* @return installation status
*/
public String installPackage(String pack, boolean load) {
log(" trying to intall " + (load ? "& load " : "") + "package " + pack, Level.INFO);
if (isPackageInstalled(pack, null)) {
log(_PACKAGE_ + pack + " already installed.", Level.INFO);
if (load) {
return loadPackage(pack);
} else {
return PACKAGEINSTALLED;
}
} else {
log(_PACKAGE_ + pack + " not yet installed.", Level.INFO);
}
/*if (!Configuration.isWWWConnected()) {
log(" package " + pack + " not accessible on " + repos + ": CRAN unreachable.");
return "Impossible to get package " + pack + " from " + repos;
}*/
rawEval("install.packages('" + pack + "',repos='" + repos + "',quiet=T" + install_packages_moreargs + ")", TRY_MODE);
log(" request if package " + pack + " is installed...", Level.INFO);
if (isPackageInstalled(pack, null)) {
log(_PACKAGE_ + pack + " installation sucessfull.", Level.INFO);
if (load) {
return loadPackage(pack);
} else {
return PACKAGEINSTALLED;
}
} else {
log(_PACKAGE_ + pack + " installation failed.", Level.ERROR);
if (load) {
return loadPackage(pack);
} else {
return "Impossible to install package " + pack + " !";
}
}
}
/**
* load R backage using library() command
*
* @param pack R package name
* @return loading status
*/
public String loadPackage(String pack) {
log(" request package " + pack + " loading...", Level.INFO);
try {
boolean ok = asLogical(rawEval("library(" + pack + ",logical.return=T,quietly=T,verbose=F" + install_packages_moreargs + ")", TRY_MODE));
if (ok) {
log(_PACKAGE_ + pack + " loading sucessfull.", Level.INFO);
return PACKAGELOADED;
} else {
log(_PACKAGE_ + pack + " loading failed.", Level.ERROR);
return "Impossible to load package " + pack + ": " + getLastLogEntry();
}
} catch (Exception ex) {
log(_PACKAGE_ + pack + " loading failed.", Level.ERROR);
return "Impossible to load package " + pack + ": " + ex.getLocalizedMessage();
}
}
//
final static String HEAD_EVAL = "[eval] ";
final static String HEAD_EXCEPTION = "[exception] ";
final static String HEAD_ERROR = "[error] ";
final static String HEAD_CACHE = "[cache] ";
public String getLastOutput() {
if (!SINK_OUTPUT) {
return null;
} else {
return lastOuput;
}
}
public String getLastError() {
if (!SINK_MESSAGE) {
Object err = silentlyRawEval("geterrmessage()");
return (err == null ? "" : asString(err));
} else {
return lastMessage;
}
}
/**
* Silently (ie no log) launch R command without return value. Encapsulate command in try() to cacth errors
*
* @param expression R expresison to evaluate
* @return well evaluated ?
*/
protected boolean silentlyVoidEval(String expression) {
return silentlyVoidEval(expression, TRY_MODE_DEFAULT);
}
/**
* Silently (ie no log) launch R command without return value.
*
* @param expression R expresison to evaluate
* @param tryEval encapsulate command in try() to cacth errors
* @return well evaluated ?
*/
protected abstract boolean silentlyVoidEval(String expression, boolean tryEval);
/**
* Silently (ie no log) launch R command and return value. Encapsulate command in try() to cacth errors.
*
* @param expression R expresison to evaluate
* @return REXP R expression
*/
protected Object silentlyRawEval(String expression) {
return silentlyRawEval(expression, TRY_MODE_DEFAULT);
}
/**
* Silently (ie no log) launch R command and return value.
*
* @param expression R expression to evaluate
* @param tryEval encapsulate command in try() to cacth errors
* @return REXP R expression
*/
protected abstract Object silentlyRawEval(String expression, boolean tryEval);
/**
* Launch R command and return value.
*
* @param expression R expresison to evaluate
* @param tryEval encapsulate command in try() to cacth errors
* @return REXP R expression
*/
protected Object rawEval(String expression, boolean tryEval) {
log(HEAD_EVAL + (tryEval ? HEAD_TRY : "") + expression, Level.INFO);
note_code(expression);
Object e = silentlyRawEval(expression, tryEval);
for (UpdateObjectsListener b : updateObjects) {
b.update();
}
if (e != null) {
log(__ + toString(e), Level.INFO);
}
return e;
}
/**
* Launch R command and return value. Encapsulate command in try() to cacth errors.
*
* @param expression R expresison to evaluate
* @return REXP R expression
*/
protected Object rawEval(String expression) {
return rawEval(expression, TRY_MODE_DEFAULT);
}
/**
* Launch R command without return value.
*
* @param expression R expresison to evaluate
* @param tryEval encapsulate command in try() to cacth errors
* @return well evaluated ?
* @throws org.math.R.Rsession.RException Could not eval
*/
public boolean voidEval(String expression, boolean tryEval) throws RException {
log(HEAD_EVAL + (tryEval ? HEAD_TRY : " ") + expression, Level.INFO);
note_code(expression);
boolean done = silentlyVoidEval(expression, tryEval);
if (!done) {
note_text("Failed to evaluate " + expression);
throw new RException("Failed to evaluate " + expression);
}
if (done) {
for (UpdateObjectsListener b : updateObjects) {
b.update();
}
}
return done;
}
/**
* Launch R command without return value. Encapsulate command in try() to catch R errors.
*
* @param expression R expresison to evaluate
* @return well evaluated ?
* @throws org.math.R.Rsession.RException Could not eval
*/
public boolean voidEval(String expression) throws RException {
boolean done = voidEval(expression, TRY_MODE_DEFAULT);
if (!done) {
throw new RException("Failed to evaluate " + expression);
}
return done;
}
public Object eval(String expression, boolean tryEval) throws RException {
Object o = rawEval(expression, tryEval);
if (o != null && o instanceof RException) throw (RException)o;
return cast(o);
}
public Object eval(String expression) throws RException {
Object o = rawEval(expression);
if (o != null && o instanceof RException) throw (RException)o;
return cast(o);
}
public class Function {
String name;
public Function(String name) {
this.name = name;
}
public Object evaluate() throws RException {
return eval(name + "()");
}
public Object evaluate(Object... args) throws RException {
String[] x = new String[args.length];
String arg = "";
for (int i = 0; i < x.length; i++) {
set(".x" + i, args[i]);
arg = arg + ",.x" + i;
}
return eval(name + "(" + arg.substring(1) + ")");
}
}
/**
* delete all variables in R environment
*
* @param all also shadow variables?
* @return well removed ?
*/
public boolean rmAll(boolean all) {
try {
return voidEval("rm(list=ls(all=" + (all ? "TRUE" : "FALSE") + "))", TRY_MODE);
} catch (Exception ex) {
return false;
}
}
public boolean rmAll() {
return rmAll(false);
}
/**
* create a R list with given R objects
*
* @param vars R object names
* @return list expression
*/
public static String buildList(String... vars) {
if (vars.length > 1) {
StringBuffer b = new StringBuffer("c(");
for (String v : vars) {
b.append(v + ",");
}
return b.substring(0, b.length() - 1) + ")";
} else {
return vars[0];
}
}
/**
* create a R list with given R strings
*
* @param vars R strings
* @return String list expression
*/
public static String buildListString(String... vars) {
if (vars == null) {
return null;
}
if (vars.length > 1) {
StringBuffer b = new StringBuffer("c(");
for (String v : vars) {
b.append("'" + v + "',");
}
return b.substring(0, b.length() - 1) + ")";
} else {
if (vars.length == 1 && vars[0] != null) {
return "'" + vars[0] + "'";
} else {
return "''";
}
}
}
/**
* create a R list with given R string patterns
*
* @param vars R string patterns
* @return ls pattern expression
*/
public static String buildListPattern(String... vars) {
if (vars.length > 1) {
StringBuffer b = new StringBuffer("c(");
for (String v : vars) {
b.append("ls(pattern='" + v + "'),");
}
return b.substring(0, b.length() - 1) + ")";
} else {
return "ls(pattern='" + vars[0] + "')";
}
}
/**
* loads R source file (eg ".R" file)
*
* @param f ".R" file to source
*/
public void source(File f) {
try {
assert asLogical(rawEval("file.exists('" + f.getAbsolutePath().replace("\\", "/") + "')", TRY_MODE));
} catch (Exception r) {
log(r.getMessage(), Level.ERROR);
}
try {
voidEval("source('" + f.getAbsolutePath().replace("\\", "/") + "')", TRY_MODE);
} catch (Exception ex) {
log(ex.getMessage(), Level.ERROR);
}
}
/**
* loads R data file (eg ".Rdata" file)
*
* @param f ".Rdata" file to load
*/
public void load(File f) {
try {
assert asLogical(rawEval("file.exists('" + f.getAbsolutePath().replace("\\", "/") + "')", TRY_MODE));
} catch (Exception r) {
log(r.getMessage(), Level.ERROR);
}
try {
voidEval("load('" + f.getAbsolutePath().replace("\\", "/") + "')", TRY_MODE);
} catch (Exception ex) {
log(ex.getMessage(), Level.ERROR);
}
}
public String[] ls() {
return ls(false);
}
/**
* list R variables in R env.
*
* @param all - If TRUE, all object names are returned. If FALSE, names which begin with a . are omitted.
* @return list of R objects names
*/
public String[] ls(boolean all) {
try {
String[] ls = asStrings(rawEval("ls(all.names=" + (all ? "TRUE" : "FALSE") + ")", false));
if (ls == null) {
return new String[]{};
}
return ls;
} catch (Exception re) {
log(re.getMessage(), Level.ERROR);
return new String[0];
}
}
/**
* list R variables in R env. matching patterns
*
* @param vars R object name patterns
* @return list of R objects names
*/
public String[] ls(String... vars) {
if (vars == null || vars.length == 0) {
try {
return asStrings(rawEval("ls()", false));
} catch (Exception re) {
log(re.getMessage(), Level.ERROR);
return new String[0];
}
} else if (vars.length == 1) {
try {
return asStrings(rawEval(buildListPattern(vars[0]), TRY_MODE));
} catch (Exception re) {
log(re.getMessage(), Level.ERROR);
return new String[0];
}
} else {
try {
return asStrings(rawEval(buildListPattern(vars), TRY_MODE));
} catch (Exception re) {
log(re.getMessage(), Level.ERROR);
return new String[0];
}
}
}
/**
* delete R variables in R env.
*
* @param vars R objects names
* @return well removed ?
* @throws org.math.R.Rsession.RException Could not do rm
*/
public boolean rm(String... vars) throws RException {
if (vars.length == 1) {
return voidEval("rm(" + vars[0] + ")", TRY_MODE);
} else {
return voidEval("rm(list=" + buildListString(vars) + ")", TRY_MODE);
}
}
/**
* delete R variables in R env. matching patterns
*
* @param vars R object name patterns
* @return well removed ?
* @throws org.math.R.Rsession.RException Could not do rm
*/
public boolean rmls(String... vars) throws RException {
if (vars.length == 1) {
return voidEval("rm(list=" + buildListPattern(vars[0]) + ")", TRY_MODE);
} else {
return voidEval("rm(list=" + buildListPattern(vars) + ")", TRY_MODE);
}
}
public boolean SAVE_ASCII = false;
/**
* Save R variables in data file
*
* @param f file to store data (eg ".Rdata")
* @param vars R variables to save
* @throws org.math.R.Rsession.RException Could not do save
*/
public void save(File f, String... vars) throws RException {
if (vars == null || vars.length == 0) {
log("Nothing to save.", Level.WARNING);
return;
}
if (vars.length == 1) {
if (vars[0] == null) {
log("Nothing to save.", Level.WARNING);
return;
}
voidEval("save(file='" + f.getAbsolutePath().replace("\\", "/") + "','" + vars[0] + "',ascii=" + (SAVE_ASCII ? "TRUE" : "FALSE") + ")", TRY_MODE);
} else {
voidEval("save(file='" + f.getAbsolutePath().replace("\\", "/") + "',list=" + buildListString(vars) + ",ascii=" + (SAVE_ASCII ? "TRUE" : "FALSE") + ")", TRY_MODE);
}
}
/**
* Save R variables in data file
*
* @param f file to store data (eg ".Rdata")
* @param vars R variables names patterns to save
* @throws org.math.R.Rsession.RException Could not do save
*/
public void savels(File f, String... vars) throws RException {
if (vars == null) {
log("Nothing to save.", Level.WARNING);
return;
}
if (vars.length == 1) {
if (vars[0] == null) {
log("Nothing to save.", Level.WARNING);
return;
}
voidEval("save(file='" + f.getAbsolutePath().replace("\\", "/") + "',list=" + buildListPattern(vars[0]) + ",ascii=" + (SAVE_ASCII ? "TRUE" : "FALSE") + ")", TRY_MODE);
} else {
voidEval("save(file='" + f.getAbsolutePath().replace("\\", "/") + "',list=" + buildListPattern(vars) + ",ascii=" + (SAVE_ASCII ? "TRUE" : "FALSE") + ")", TRY_MODE);
}
try {
Thread.sleep(1000);
} catch (InterruptedException ex) {
}
}
final static String[] types = {"data.frame", "null", "function", "array", "integer", "character", "double"};
/**
*
* @param robject R object name
* @return R type of object
*/
public String typeOf(String robject) {
if (robject == null) {
return "NULL";
}
for (String t : types) {
try {
boolean is = asLogical(silentlyRawEval("is." + t + "(" + robject + ")"));
if (is) {
return t;
}
} catch (Exception ex) {
log(HEAD_ERROR + "[typeOf] " + robject + " type unknown.", Level.ERROR);
return null;
}
}
return "unknown";
}
public final static String HEAD_SET = "[set] ";
/**
* delete R object in R env.
*
* @param varname R objects to delete
* @return succeeded ?
* @throws org.math.R.Rsession.RException Could not unset
*/
public boolean unset(String... varname) throws RException {
return rm(varname);
}
/**
* delete R object in R env.
*
* @param varname R objects to delete
* @return succeeded ?
* @throws org.math.R.Rsession.RException Could not unset
*/
public boolean unset(Collection varname) throws RException {
boolean done = true;
for (Object v : varname) {
done = done & rm(v.toString());
}
return done;
}
/**
* Set R object in R env.
*
* @param _vars R objects to set as key/values
* @return succeeded ?
* @throws org.math.R.Rsession.RException Could not set one var
*/
public boolean set(Map _vars) throws RException {
boolean done = true;
for (String varname : _vars.keySet()) {
done = done & set(varname, _vars.get(varname));
}
return done;
}
/**
* Set R data.frame in R env.
*
* @param varname R list name
* @param data numeric data in list
* @param names names of columns
* @return succeeded ?
* @throws org.math.R.Rsession.RException Could not set
*/
public abstract boolean set(String varname, double[][] data, String... names) throws RException;
/**
* Set R object in R env.
*
* @param varname R object name
* @param var R object value
* @return succeeded ?
* @throws org.math.R.Rsession.RException Could not set
*/
public abstract boolean set(String varname, Object var) throws RException;
protected static double[] reshapeAsRow(double[][] a) {
double[] reshaped = new double[a.length * a[0].length];
int ir = 0;
for (int j = 0; j < a[0].length; j++) {
for (int i = 0; i < a.length; i++) {
reshaped[ir] = a[i][j];
ir++;
}
}
return reshaped;
}
protected static double[] reshapeAsRow(Double[][] a) {
double[] reshaped = new double[a.length * a[0].length];
int ir = 0;
for (int j = 0; j < a[0].length; j++) {
for (int i = 0; i < a.length; i++) {
reshaped[ir] = a[i][j];
ir++;
}
}
return reshaped;
}
protected double[][] t(double[][] m) {
double[][] tm = new double[m[0].length][m.length];
for (int i = 0; i < tm.length; i++) {
for (int j = 0; j < tm[i].length; j++) {
tm[i][j] = m[j][i];
}
}
return tm;
}
public abstract double asDouble(Object o) throws ClassCastException;
public abstract double[] asArray(Object o) throws ClassCastException;
public abstract double[][] asMatrix(Object o) throws ClassCastException;
public abstract String asString(Object o) throws ClassCastException;
public abstract String[] asStrings(Object o) throws ClassCastException;
public abstract int asInteger(Object o) throws ClassCastException;
public abstract int[] asIntegers(Object o) throws ClassCastException;
public abstract boolean asLogical(Object o) throws ClassCastException;
public abstract boolean[] asLogicals(Object o) throws ClassCastException;
public abstract Map asList(Object o) throws ClassCastException;
public abstract boolean isNull(Object o);
public abstract String toString(Object o);
public abstract Object cast(Object o) throws ClassCastException;
/*public Object cast(Object o) {
Object oo = o;
try {
oo = castStrict(o);
} catch (ClassCastException e) {
}
return oo;
}*/
/**
* Create a file for R graphical command output
*
* @param f File to store data (eg .jpg file)
* @param width width of image
* @param height height of image
* @param fileformat format of image: png,tiff,jpeg,bmp
* @param commands R command to create image (eg plot())
*/
public void toGraphic(File f, int width, int height, String fileformat, String... commands) {
int h = Math.abs(f.hashCode());
try {
set("plotfile_" + h, f.getAbsolutePath().replace("\\", "/"));
} catch (Exception ex) {
log(ex.getMessage(), Level.ERROR);
}
silentlyRawEval(fileformat + "(plotfile_" + h + ", width=" + width + ", height=" + height + ")");
for (String command : commands) {
note_code(command);
silentlyVoidEval(command);
}
silentlyRawEval("dev.off()");
try {
rm("plotfile_" + h);
} catch (Exception ex) {
log(ex.getMessage(), Level.ERROR);
}
}
public final static String GRAPHIC_PNG = "png";
public final static String GRAPHIC_JPEG = "jpeg";
public final static String GRAPHIC_BMP = "bmp";
public final static String GRAPHIC_TIFF = "tiff";
public void toGraphic(File f, int width, int height, String... commands) {
if (f.getName().endsWith(GRAPHIC_BMP)) {
toBMP(f, width, height, commands);
} else if (f.getName().endsWith(GRAPHIC_JPEG)) {
toJPEG(f, width, height, commands);
} else if (f.getName().endsWith(GRAPHIC_PNG)) {
toPNG(f, width, height, commands);
} else if (f.getName().endsWith(GRAPHIC_TIFF)) {
toTIFF(f, width, height, commands);
} else {
toPNG(f, width, height, commands);
}
}
public void toJPEG(File f, int width, int height, String... commands) {
toGraphic(f, width, height, GRAPHIC_JPEG, commands);
}
public void toPNG(File f, int width, int height, String... commands) {
toGraphic(f, width, height, GRAPHIC_PNG, commands);
}
public void toBMP(File f, int width, int height, String... commands) {
toGraphic(f, width, height, GRAPHIC_BMP, commands);
}
public void toTIFF(File f, int width, int height, String... commands) {
toGraphic(f, width, height, GRAPHIC_TIFF, commands);
}
/**
* Get R command text output in HTML format
*
* @param command R command returning text
* @return HTML string
*/
public String asR2HTML(String command) {
String ret = installPackage("R2HTML", true);
if (!ret.equals(PACKAGELOADED)) {
return ret;
}
int h = Math.abs(command.hashCode());
note_code("HTML(" + command + ")");
silentlyRawEval("HTML(file=\"htmlfile_" + h + "\", " + command + ")");
String[] lines = null;
try {
lines = asStrings(silentlyRawEval("readLines(\"htmlfile_" + h + "\")"));
} catch (Exception e) {
return e.getMessage();
}
if (lines == null) {
return "";
}
StringBuffer sb = new StringBuffer();
for (String l : lines) {
sb.append(l);
sb.append("\n");
}
String str = sb.toString();
str = str.replace("align= center ", "align='center'");
str = str.replace("cellspacing=0", "cellspacing='0'");
str = str.replace("border=1", "border='1'");
str = str.replace("align=bottom", "align='bottom'");
str = str.replace("class=dataframe", "class='dataframe'");
str = str.replace("class= firstline ", "class='firstline'");
str = str.replace("class=firstcolumn", "class='firstcolumn'");
str = str.replace("class=cellinside", "class='cellinside'");
str = str.replace("border=0", "border='0'");
str = str.replace("class=captiondataframe", "class='captiondataframe'");
str = str.replace("", "");
return str;
}
/**
* Get R command text output in HTML format
*
* @param command R command returning text
* @return HTML string
*/
public String asHTML(String command) {
return toHTML(print(command));
}
public static String toHTML(String src) {
if (src == null) {
return src;
}
src = src.replace("&", "&");
src = src.replace("\"", """);
src = src.replace("'", "'");
src = src.replace("<", "<");
src = src.replace(">", ">");
return "" + src.replace("\n", "
") + "";
}
/**
* Get R command text output
*
* @param command R command returning text
* @return String
*/
public String print(String command) {
try {
note_code("print(" + command + ")");
String s = asString(silentlyRawEval("paste(capture.output(print(" + command + ")),collapse='\\n')"));
return s;
} catch (Exception ex) {
return ex.getMessage();
}
}
final static String IO_HEAD = "[IO] ";
final static String testExpression = "1+pi";
final static double testResult = 1 + Math.PI;
Map noVarsEvals = new HashMap();
/**
* Method to rawEval expression. Holds many optimizations (@see noVarsEvals) and turn around for reliable usage (like engine auto restart). 1D Numeric "vars" are replaced using
* Java replace engine instead of R one. Intended to not interfer with current R env vars. Yes, it's hard-code :)
*
* @param expression String to evaluate
* @param vars HashMap<String, Object> vars inside expression. Passively overload current R env variables.
* @return java castStrict Object Warning, UNSTABLE and high CPU cost.
* @throws org.math.R.Rsession.RException Could not proxyEval
*/
public synchronized Object proxyEval(String expression, Map vars) throws RException {
if (expression.length() == 0) {
return null;
}
try {
double d = Double.parseDouble(expression);
log(HEAD_CACHE + "No evaluation needed for " + expression, Level.INFO);
return d;
} catch (NumberFormatException ne) {
if (!uses(expression, vars) && noVarsEvals.containsKey(expression)) {
log(HEAD_CACHE + "Cached evaluation of " + expression + " in " + noVarsEvals, Level.INFO);
return noVarsEvals.get(expression);
}
if (vars != null && vars.containsKey(expression)) {
log(HEAD_CACHE + "Get evaluation of " + expression + " in " + vars, Level.INFO);
return vars.get(expression);
}
Map clean_vars = new HashMap();
String clean_expression = expression;
if (vars != null) {
for (String v : vars.keySet()) {
if (vars.get(v) instanceof Number) {
while (containsVar(clean_expression, v)) {
clean_expression = replaceVar(clean_expression, v, "(" + vars.get(v) + ")");
}
log(HEAD_CACHE + "Replacing " + v + " in " + clean_expression, Level.INFO);
} else {
if (containsVar(clean_expression, v)/*clean_expression.contains(v)*/) {
String newvarname = v;
while (ls(newvarname).length > 0) {
newvarname = "_" + newvarname;
}
log(HEAD_CACHE + "Renaming " + v + " by " + newvarname + " in " + clean_expression, Level.INFO);
while (containsVar(clean_expression, v)) {
clean_expression = replaceVar(clean_expression, v, newvarname);
}
clean_vars.put(newvarname, vars.get(v));
}
}
}
}
if (!uses(clean_expression, clean_vars) && noVarsEvals.containsKey(clean_expression)) {
log(HEAD_CACHE + "Cached evaluation of " + expression + " in " + noVarsEvals, Level.INFO);
return noVarsEvals.get(clean_expression);
}
Object out = null;
try {
if (uses(clean_expression, clean_vars)) {
set(clean_vars);
}
log(HEAD_CACHE + "True evaluation of " + clean_expression + " with " + clean_vars, Level.INFO);
out = eval(clean_expression);
if (clean_vars.isEmpty() && out != null) {
log(HEAD_CACHE + "Saving result of " + clean_expression, Level.INFO);
noVarsEvals.put(clean_expression, out);
}
if (!uses(expression, vars) && out != null) {
log(HEAD_CACHE + "Saving result of " + expression, Level.INFO);
noVarsEvals.put(expression, out);
}
} catch (Exception e) {
log(HEAD_CACHE + "Failed cast of " + expression, Level.INFO);
throw new RException(CAST_ERROR + expression + ": " + e.getMessage());
} finally {
if (uses(clean_expression, clean_vars)) {
unset(clean_vars.keySet());
}
}
return out;
}
}
final static String AW = "((\\A)|(\\W))(";
final static String Az = ")((\\W)|(\\z))";
static String replaceVar(final String expr, final String var, final String val) {
String regexp = AW + var + Az;
Matcher m = Pattern.compile(regexp).matcher(expr);
if (m.find()) {
return expr.replace(m.group(), m.group().replace(var, val));
} else {
return expr;
}
}
static boolean containsVar(final String expr, final String var) {
String regexp = AW + var + Az;
Matcher m = Pattern.compile(regexp).matcher(expr);
return m.find();
}
static boolean areUsed(String expression, Set vars) {
for (String v : vars) {
if (containsVar(expression, v)) {
return true;
}
}
return false;
}
static boolean uses(String expression, Map vars) {
return vars != null && !vars.isEmpty() && areUsed(expression, vars.keySet());
}
public abstract void setGlobalEnv(String envName);
protected String envName = "UNDEFINED_ENV";
public String getGlobalEnv() {return envName.substring(2,envName.length()-2);};
public abstract void copyGlobalEnv(String newEnvName);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy