Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
weka.core.WekaPackageManager Maven / Gradle / Ivy
Go to download
The Waikato Environment for Knowledge Analysis (WEKA), a machine
learning workbench. This is the stable version. Apart from bugfixes, this version
does not receive any other updates.
/*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
/*
* WekaPackageManager.java
* Copyright (C) 2009-2013 University of Waikato, Hamilton, New Zealand
*/
package weka.core;
import weka.core.converters.ConverterUtils;
import weka.core.logging.Logger;
import weka.core.packageManagement.DefaultPackageManager;
import weka.core.packageManagement.Dependency;
import weka.core.packageManagement.Package;
import weka.core.packageManagement.PackageConstraint;
import weka.core.packageManagement.PackageManager;
import weka.core.packageManagement.VersionPackageConstraint;
import weka.gui.ConverterFileChooser;
import weka.gui.GenericObjectEditor;
import weka.gui.GenericPropertiesCreator;
import weka.gui.beans.BeansProperties;
import weka.gui.beans.KnowledgeFlowApp;
import weka.gui.explorer.ExplorerDefaults;
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.ByteArrayInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.PrintStream;
import java.io.PrintWriter;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URL;
import java.net.URLConnection;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.Properties;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
/**
* Class providing package management and manipulation routines. Also provides a
* command line interface for package management.
*
* @author Mark Hall (mhall{[at]}pentaho{[dot]}com)
* @version $Revision: 15541 $
*/
public class WekaPackageManager {
/** Package metadata key for dependency injection */
public static final String INJECT_DEPENDENCY_KEY = "InjectDependencies";
/** Package metadata key for version info */
public static final String VERSION_KEY = "Version";
/** Package metadata key for package disablement */
public static final String DISABLE_KEY = "Disable";
/** Package metadata key for package disablement */
public static final String DISABLED_KEY = "Disabled";
/** Package metadata key for package preclusion */
public static final String PRECLUDES_KEY = "Precludes";
/**
* Package metadata key for OS name. Entries in this list are checked against
* the java property os.name using a String.contains() operation - any match
* in the list is taken as passing the test.
*/
public static final String OS_NAME_KEY = "OSName";
/**
* Package metadata key for OS architecture. Entries in this list are checked
* against the java property os.arch using a String.equalsIgnoreCase()
* operation, with the exception of entries that are either "64" or "32" in
* which case the os.arch is checked to see if it contains these numbers. Any
* match in the list is considered as passing the test.
*/
public static final String OS_ARCH_KEY = "OSArch";
/**
* Package metadata key for JVM version. Values can be prefixed by a less-than
* or greater-than sign. Otherwise, equality is assumed.
*/
public static final String VM_VERSION_KEY = "JVMVersion";
/**
* Package metadata key for preventing load if an environment variable is not
* set
*/
public static final String DO_NOT_LOAD_IF_ENV_VAR_NOT_SET_KEY =
"DoNotLoadIfEnvVarNotSet";
/**
* Package metadata key for preventing load if an environment variable is not
* set
*/
public static final String DO_NOT_LOAD_IF_ENV_VAR_NOT_SET_MESSAGE_KEY =
"DoNotLoadIfEnvVarNotSetMessage";
/** Package metadata key for preventing load if a class is not available */
public static final String DO_NOT_LOAD_IF_CLASS_NOT_PRESENT_KEY =
"DoNotLoadIfClassNotPresent";
/** Package metadata key for preventing load if a class is not available */
public static final String DO_NOT_LOAD_IF_CLASS_NOT_PRESENT_MESSAGE_KEY =
"DoNotLoadIfClassNotPresentMessage";
/** Package metadata key for preventing load if a file is not present */
public static final String DO_NOT_LOAD_IF_FILE_NOT_PRESENT_KEY =
"DoNotLoadIfFileNotPresent";
/** Package metadata key for preventing load if a file is not present */
public static final String DO_NOT_LOAD_IF_FILE_NOT_PRESENT_MESSAGE_KEY =
"DoNotLoadIfFileNotPresentMessage";
/** Package metadata key for a message to display on installation */
public static final String MESSAGE_TO_DISPLAY_ON_INSTALLATION_KEY =
"MessageToDisplayOnInstallation";
/** Package metadata key for setting system properties */
public static final String SET_SYSTEM_PROPERTIES_KEY = "SetSystemProperties";
/** The default folder name for Weka bits and bobs */
private static String WEKAFILES_DIR_NAME = "wekafiles";
/** Default path to where Weka's configuration and packages are stored */
public static File WEKA_HOME = new File(System.getProperty("user.home")
+ File.separator + WEKAFILES_DIR_NAME);
/** The default packages directory */
public static File PACKAGES_DIR = new File(WEKA_HOME.toString()
+ File.separator + "packages");
/** The default props dir name */
private static String PROPERTIES_DIR_NAME = "props";
/** The default properties directory */
public static File PROPERTIES_DIR = new File(WEKA_HOME.toString()
+ File.separator + PROPERTIES_DIR_NAME);
public static String NATIVE_LIBS_DIR_NAME = "native";
/** The default native libraries directory */
public static File NATIVE_LIBS_DIR = new File(WEKA_HOME.toString()
+ File.separator + NATIVE_LIBS_DIR_NAME);
/** The underlying package manager */
private static PackageManager PACKAGE_MANAGER = PackageManager.create();
/** Current repository URL to use */
private static URL REP_URL;
/** Location of the repository cache */
private static URL CACHE_URL;
/** True if a cache build is required */
private static boolean INITIAL_CACHE_BUILD_NEEDED = false;
/**
* The name of the file that contains the list of packages in the repository
*/
private static String PACKAGE_LIST_FILENAME = "packageListWithVersion.txt";
/** Primary repository */
private static String PRIMARY_REPOSITORY =
"https://weka.sourceforge.io/packageMetaData";
/** Backup mirror of the repository */
private static String REP_MIRROR;
/**
* True if the user has specified a custom repository via a property in
* PackageManager.props
*/
private static boolean USER_SET_REPO = false;
/** The package manager's property file */
private static String PACKAGE_MANAGER_PROPS_FILE_NAME =
"PackageManager.props";
/** Operating offline? */
public static boolean m_offline;
/** Load packages? */
private static boolean m_loadPackages = true;
/** Established WEKA_HOME successfully? */
protected static boolean m_wekaHomeEstablished;
/** Packages loaded OK? */
protected static boolean m_packagesLoaded;
/** File to check against server for new/updated packages */
protected static final String PACKAGE_LIST_WITH_VERSION_FILE =
"packageListWithVersion.txt";
/** File to check against server equivalent for forced refresh */
protected static final String FORCED_REFRESH_COUNT_FILE =
"forcedRefreshCount.txt";
/** Package loading in progress? */
public static boolean m_initialPackageLoadingInProcess = false;
/* True if an initial cache build is needed and working offline */
public static boolean m_noPackageMetaDataAvailable;
/** The set of packages that the user has requested not to load */
public static Set m_doNotLoadList;
/**
* Holds how many times we've failed to establish the repository backup mirror
* URL
*/
protected static int m_mirrorFailureCount;
static {
establishWekaHome();
// make sure MTJ classes are injected to the root classloader
// so that MTJ native library loading can see them
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.injectMTJCoreClasses();
}
/**
* Establish WEKA_HOME if needed
*
* @return true if WEKA_HOME was successfully established
*/
protected static boolean establishWekaHome() {
if (m_wekaHomeEstablished) {
return true;
}
// process core PluginManager.props before any from packages.
// This is to have some control over the order of certain plugins
try {
PluginManager.addFromProperties(WekaPackageManager.class.getClassLoader()
.getResourceAsStream("weka/PluginManager.props"), true);
} catch (Exception ex) {
log(Logger.Level.WARNING,
"[WekaPackageManager] unable to read weka/PluginManager.props");
}
// look to see if WEKA_HOME has been defined as an environment
// variable
Environment env = Environment.getSystemWide();
String wh = env.getVariableValue("WEKA_HOME");
if (wh != null) {
WEKA_HOME = new File(wh);
PACKAGES_DIR = new File(wh + File.separator + "packages");
PROPERTIES_DIR = new File(wh + File.separator + PROPERTIES_DIR_NAME);
NATIVE_LIBS_DIR = new File(wh + File.separator + NATIVE_LIBS_DIR_NAME);
} else {
env.addVariableSystemWide("WEKA_HOME", WEKA_HOME.toString());
}
String nativePath = System.getProperty("java.library.path", "");
if (!nativePath.contains(NATIVE_LIBS_DIR.toString())) {
nativePath += File.pathSeparator + NATIVE_LIBS_DIR.toString();
System.setProperty("java.library.path", nativePath);
}
boolean ok = true;
if (!WEKA_HOME.exists()) {
// create it for the user
if (!WEKA_HOME.mkdir()) {
System.err.println("Unable to create WEKA_HOME ("
+ WEKA_HOME.getAbsolutePath() + ")");
ok = false;
}
}
if (!PACKAGES_DIR.exists()) {
// create the packages dir
if (!PACKAGES_DIR.mkdir()) {
System.err.println("Unable to create packages directory ("
+ PACKAGES_DIR.getAbsolutePath() + ")");
ok = false;
}
}
if (!NATIVE_LIBS_DIR.exists()) {
if (!NATIVE_LIBS_DIR.mkdir()) {
System.err.println("Unable to create native libs directory ("
+ NATIVE_LIBS_DIR.getAbsolutePath() + ")");
}
}
m_wekaHomeEstablished = ok;
PACKAGE_MANAGER.setPackageHome(PACKAGES_DIR);
m_doNotLoadList = getDoNotLoadList();
try {
// setup the backup mirror first
// establishMirror();
// user-supplied repository URL takes precedence over anything else
String repURL =
env.getVariableValue("weka.core.wekaPackageRepositoryURL");
if (repURL == null || repURL.length() == 0) {
// See if there is a URL named in
// $WEKA_HOME/props/PackageRepository.props
File repPropsFile =
new File(PROPERTIES_DIR.toString() + File.separator
+ "PackageRepository.props");
if (repPropsFile.exists()) {
Properties repProps = new Properties();
repProps.load(new FileInputStream(repPropsFile));
repURL = repProps.getProperty("weka.core.wekaPackageRepositoryURL");
}
}
if (repURL == null || repURL.length() == 0) {
repURL = PRIMARY_REPOSITORY;
} else {
log(weka.core.logging.Logger.Level.INFO,
"[WekaPackageManager] weka.core.WekaPackageRepositoryURL = " + repURL);
// System.err.println("[WekaPackageManager]
// weka.core.WekaPackageRepositoryURL = "
// + repURL);
USER_SET_REPO = true;
}
REP_URL = new URL(repURL);
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
} catch (MalformedURLException ex) {
ex.printStackTrace();
} catch (IOException ex) {
ex.printStackTrace();
}
PACKAGE_MANAGER.setBaseSystemName("weka");
PACKAGE_MANAGER.setBaseSystemVersion(weka.core.Version.VERSION);
// Now check the cache and establish it if necessary
File cacheDir =
new File(WEKA_HOME.toString() + File.separator + "repCache");
try {
String tempCacheString = "file://" + cacheDir.toString();
// sanitize URI and fix slashes (for Windows)
tempCacheString = tempCacheString.replace(" ", "%20");
tempCacheString = tempCacheString.replace('\\', '/');
if (tempCacheString.startsWith("file://")
&& !tempCacheString.startsWith("file:///")) {
tempCacheString = tempCacheString.substring(7);
tempCacheString = "file:///" + tempCacheString;
}
URI tempURI = new URI(tempCacheString);
// System.err.println(" -- " + tempURI.toString());
CACHE_URL = tempURI.toURL();
} catch (Exception e) {
e.printStackTrace();
}
File packagesList =
new File(cacheDir.getAbsolutePath() + File.separator
+ PACKAGE_LIST_FILENAME);
if (!cacheDir.exists()) {
if (!cacheDir.mkdir()) {
System.err.println("Unable to create repository cache directory ("
+ cacheDir.getAbsolutePath() + ")");
log(
weka.core.logging.Logger.Level.WARNING,
"Unable to create repository cache directory ("
+ cacheDir.getAbsolutePath() + ")");
CACHE_URL = null;
} else {
// refreshCache();
INITIAL_CACHE_BUILD_NEEDED = true;
}
}
if (!packagesList.exists()) {
INITIAL_CACHE_BUILD_NEEDED = true;
}
// Package manager general properties
// Set via system props first
String offline = env.getVariableValue("weka.packageManager.offline");
if (offline != null) {
m_offline = offline.equalsIgnoreCase("true");
}
String loadPackages =
env.getVariableValue("weka.packageManager.loadPackages");
if (loadPackages == null) {
// try legacy
loadPackages = env.getVariableValue("weka.core.loadPackages");
}
if (loadPackages != null) {
m_loadPackages = loadPackages.equalsIgnoreCase("true");
}
// load any general package manager properties from props file
File generalProps =
new File(PROPERTIES_DIR.toString() + File.separator
+ PACKAGE_MANAGER_PROPS_FILE_NAME);
if (generalProps.exists()) {
Properties gProps = new Properties();
try {
gProps.load(new FileInputStream(generalProps));
// this one takes precedence over the legacy one
String repURL =
gProps.getProperty("weka.core.wekaPackageRepositoryURL");
if (repURL != null && repURL.length() > 0) {
REP_URL = new URL(repURL);
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
}
offline = gProps.getProperty("weka.packageManager.offline");
if (offline != null && offline.length() > 0) {
m_offline = offline.equalsIgnoreCase("true");
}
loadPackages = gProps.getProperty("weka.packageManager.loadPackages");
if (loadPackages == null) {
// try legacy
loadPackages = env.getVariableValue("weka.core.loadPackages");
}
if (loadPackages != null) {
m_loadPackages = loadPackages.equalsIgnoreCase("true");
}
String pluginManagerDisableList =
gProps.getProperty("weka.pluginManager.disable");
if (pluginManagerDisableList != null
&& pluginManagerDisableList.length() > 0) {
List disable = new ArrayList();
String[] parts = pluginManagerDisableList.split(",");
for (String s : parts) {
disable.add(s.trim());
}
PluginManager.addToDisabledList(disable);
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
if (INITIAL_CACHE_BUILD_NEEDED && m_offline) {
m_noPackageMetaDataAvailable = true;
}
// Pass a valid http URL to the setProxyAuthentication()
// method to ensure that a proxy or direct connection
// is configured initially. This will prevent this method from
// trying to configure the ProxySelector and proxy
// via the file-based CACHE_URL somewhere down the track
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
PACKAGE_MANAGER.setProxyAuthentication(REP_URL);
return ok;
}
/**
* Establish the location of a mirror
*/
protected static void establishMirror() {
if (m_offline) {
return;
}
try {
String mirrorListURL =
"https://www.cs.waikato.ac.nz/ml/weka/packageMetaDataMirror.txt";
URLConnection conn = null;
URL connURL = new URL(mirrorListURL);
if (PACKAGE_MANAGER.setProxyAuthentication(connURL)) {
conn = connURL.openConnection(PACKAGE_MANAGER.getProxy());
} else {
conn = connURL.openConnection();
}
conn.setConnectTimeout(10000); // timeout after 10 seconds
conn.setReadTimeout(10000);
BufferedReader bi =
new BufferedReader(new InputStreamReader(conn.getInputStream()));
REP_MIRROR = bi.readLine();
bi.close();
if (REP_MIRROR != null && REP_MIRROR.length() > 0) {
// use the mirror if it is different from the primary repo
// and the user hasn't specified an explicit repo via the
// property
if (!REP_MIRROR.equals(PRIMARY_REPOSITORY) && !USER_SET_REPO) {
log(weka.core.logging.Logger.Level.INFO,
"[WekaPackageManager] Package manager using repository mirror: "
+ REP_MIRROR);
REP_URL = new URL(REP_MIRROR);
}
}
} catch (Exception ex) {
log(weka.core.logging.Logger.Level.WARNING,
"[WekaPackageManager] The repository meta data mirror file seems "
+ "to be unavailable (" + ex.getMessage() + ")");
m_mirrorFailureCount++;
}
}
/**
* Write to the weka log
*
* @param level logging level
* @param message message to write
*/
protected static void
log(weka.core.logging.Logger.Level level, String message) {
try {
File logFile =
new File(WEKA_HOME.toString() + File.separator + "weka.log");
BufferedWriter writer = new BufferedWriter(new FileWriter(logFile, true));
SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String linefeed = System.getProperty("line.separator");
String m =
format.format(new Date()) + " " + level + ": " + message + linefeed;
writer.write(m);
writer.flush();
writer.close();
} catch (Exception ex) {
}
}
/**
* Remove any ExplorerDefaults properties specified in the supplied package
*
* @param installedPackageName the package specifying properties that should
* be removed from ExplorerDefaults
*/
public static void removeExplorerProps(String installedPackageName) {
try {
Properties expProps = new Properties();
String explorerProps =
getPackageHome().getAbsolutePath() + File.separator
+ installedPackageName + File.separator + "Explorer.props";
BufferedInputStream bi =
new BufferedInputStream(new FileInputStream(explorerProps));
expProps.load(bi);
bi.close();
bi = null;
Set keys = expProps.keySet();
Iterator keysI = keys.iterator();
while (keysI.hasNext()) {
String key = (String) keysI.next();
if (!key.endsWith("Policy")) {
// See if this key is in the Explorer props
String existingVal = ExplorerDefaults.get(key, "");
String toRemove = expProps.getProperty(key);
if (existingVal.length() > 0) {
// cover the case when the value to remove is at the start
// or middle of a list
existingVal = existingVal.replace(toRemove + ",", "");
// the case when it's at the end
existingVal = existingVal.replace("," + toRemove, "");
ExplorerDefaults.set(key, existingVal);
}
}
}
} catch (Exception ex) {
}
}
/**
* Process a package's GenericPropertiesCreator.props file
*
* @param propsFile the props file to process
*/
protected static void processGenericPropertiesCreatorProps(File propsFile) {
try {
Properties expProps = new Properties();
BufferedInputStream bi =
new BufferedInputStream(new FileInputStream(propsFile));
expProps.load(bi);
bi.close();
bi = null;
Properties GPCInputProps =
GenericPropertiesCreator.getGlobalInputProperties();
Set keys = expProps.keySet();
Iterator keysI = keys.iterator();
while (keysI.hasNext()) {
String key = (String) keysI.next();
// see if this key is in the GPC input props
String existingVal = GPCInputProps.getProperty(key, "");
if (existingVal.length() > 0) {
// append
String newVal = expProps.getProperty(key);
// only append if this value is not already there!!
if (existingVal.indexOf(newVal) < 0) {
newVal = existingVal + "," + newVal;
GPCInputProps.put(key, newVal);
}
} else {
// simply add this new key/value combo
String newVal = expProps.getProperty(key);
GPCInputProps.put(key, newVal);
}
}
} catch (Exception ex) {
// ignore
}
}
/**
* Process a package's Explorer.props file
*
* @param propsFile the properties file to process
*/
protected static void processExplorerProps(File propsFile) {
try {
Properties expProps = new Properties();
BufferedInputStream bi =
new BufferedInputStream(new FileInputStream(propsFile));
expProps.load(bi);
bi.close();
bi = null;
Set keys = expProps.keySet();
Iterator keysI = keys.iterator();
while (keysI.hasNext()) {
String key = (String) keysI.next();
if (!key.endsWith("Policy")) {
// See if this key is in the Explorer props
String existingVal = ExplorerDefaults.get(key, "");
if (existingVal.length() > 0) {
// get the replacement policy (if any)
String replacePolicy = expProps.getProperty(key + "Policy");
if (replacePolicy != null && replacePolicy.length() > 0) {
if (replacePolicy.equalsIgnoreCase("replace")) {
String newVal = expProps.getProperty(key);
ExplorerDefaults.set(key, newVal);
} else {
// default to append
String newVal = expProps.getProperty(key);
// only append if this value is not already there!!
if (existingVal.indexOf(newVal) < 0) {
newVal = existingVal + "," + newVal;
ExplorerDefaults.set(key, newVal);
}
}
} else {
// default to append
String newVal = expProps.getProperty(key);
// only append if this value is not already there!!
if (existingVal.indexOf(newVal) < 0) {
newVal = existingVal + "," + newVal;
ExplorerDefaults.set(key, newVal);
}
}
} else {
// simply add this new key/value combo
String newVal = expProps.getProperty(key);
ExplorerDefaults.set(key, newVal);
}
}
}
} catch (Exception ex) {
// ignore
}
}
/**
* Process a package's GUIEditors.props file
*
* @param propsFile the properties file to process
* @param verbose true to output more info
*/
protected static void processGUIEditorsProps(File propsFile, boolean verbose) {
GenericObjectEditor.registerEditors();
try {
Properties editorProps = new Properties();
BufferedInputStream bi =
new BufferedInputStream(new FileInputStream(propsFile));
editorProps.load(bi);
bi.close();
bi = null;
Enumeration> enm = editorProps.propertyNames();
while (enm.hasMoreElements()) {
String name = enm.nextElement().toString();
String value = editorProps.getProperty(name, "");
if (verbose) {
System.err.println("Registering " + name + " " + value);
}
GenericObjectEditor.registerEditor(name, value);
}
} catch (Exception ex) {
// ignore
}
}
/**
* Process a package's PluginManager.props file
*
* @param packageName the name of the package that owns this PluginManager
* props file
* @param propsFile the properties file to process
*/
protected static void processPluginManagerProps(String packageName,
File propsFile) {
try {
PluginManager.addFromProperties(packageName, propsFile);
} catch (Exception ex) {
ex.printStackTrace();
}
}
/**
* Process properties in a package directory
*
* @param directory the package directory to process
* @param verbose true for verbose output
* @param goePropsFiles store any GenericObjectEditor.props files for
* post-processing after all jars have been loaded
* @throws Exception if a problem occurs
*/
protected static void processPackageDirectory(File directory,
boolean verbose, List goePropsFiles,
boolean avoidTriggeringFullClassDiscovery) throws Exception {
File[] contents = directory.listFiles();
if (contents != null) {
// make sure that jar files and lib directory get processed first
/*
* for (File content : contents) { if (content.isFile() &&
* content.getPath().endsWith(".jar")) { if (verbose) {
* System.out.println("[WekaPackageManager] loading " +
* content.getPath()); } ClassloaderUtil.addFile(content.getPath());
*
* } else if (content.isDirectory() &&
* content.getName().equalsIgnoreCase("lib")) { // add any jar files in
* the lib directory to the classpath processPackageDirectory(content,
* verbose, goePropsFiles, avoidTriggeringFullClassDiscovery); } }
*/
// now any auxilliary files
for (File content : contents) {
if (content.isFile() && content.getPath().endsWith("Beans.props")) {
// KnowledgeFlow plugin -- add the Beans.props file to the list of
// bean plugin props
BeansProperties.addToPluginBeanProps(content);
if (!avoidTriggeringFullClassDiscovery) {
KnowledgeFlowApp.disposeSingleton();
}
} else if (content.isFile()
&& content.getPath().endsWith("Explorer.props")
&& !avoidTriggeringFullClassDiscovery) {
// Explorer tabs plugin
// process the keys in the properties file and append/add values
processExplorerProps(content);
} else if (content.isFile()
&& content.getPath().endsWith("GUIEditors.props")
&& !avoidTriggeringFullClassDiscovery) {
// Editor for a particular component
processGUIEditorsProps(content, verbose);
} else if (content.isFile()
&& content.getPath().endsWith("GenericPropertiesCreator.props")
&& !avoidTriggeringFullClassDiscovery) {
if (goePropsFiles != null) {
goePropsFiles.add(content);
} else {
processGenericPropertiesCreatorProps(content);
}
} else if (content.isFile()
&& content.getPath().endsWith("PluginManager.props")) {
processPluginManagerProps(directory.getName(), content);
}
}
}
}
/**
* Check to see if the named package has been loaded successfully
*
* @param toCheck the name of the package to check for
* @return true if the named package has been loaded successfully
*/
public static boolean hasBeenLoaded(Package toCheck) {
// if it loaded successfully, passed all integrity checks etc., then there
// will be package classloader for it
return WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(toCheck.getName()) != null;
}
/**
* Check whether a package should be loaded or not. Checks for missing
* classes, unset environment variables, missing dependencies etc.
*
* @param toLoad the package to check
* @param packageRoot the root directory of the package
* @param progress for reporting loading progress
* @return true if the package is good to load
*/
public static boolean loadCheck(Package toLoad, File packageRoot,
PrintStream... progress) {
// first check against the base version of the system
boolean load;
try {
load = toLoad.isCompatibleBaseSystem();
} catch (Exception ex) {
ex.printStackTrace();
return false;
}
if (!load) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Skipping package " + toLoad.getName()
+ " because it is not compatible with Weka "
+ PACKAGE_MANAGER.getBaseSystemVersion().toString());
}
return false;
}
// check to see if this package has been disabled for all users
try {
Package repoP =
getRepositoryPackageInfo(toLoad.getName(), toLoad
.getPackageMetaDataElement(VERSION_KEY).toString());
if (repoP != null) {
Object disabled = repoP.getPackageMetaDataElement(DISABLED_KEY);
if (disabled == null) {
disabled = repoP.getPackageMetaDataElement(DISABLE_KEY);
}
if (disabled != null && disabled.toString().equalsIgnoreCase("true")) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Skipping package "
+ toLoad.getName()
+ " because it has been marked as disabled at the repository");
}
return false;
}
}
} catch (Exception ex) {
// Ignore - we will get an exception when checking for an unofficial
// package
// return true;
}
if (!osAndArchCheck(toLoad, progress)) {
return false;
}
if (!vmVersionCheck(toLoad, progress)) {
return false;
}
// check for precludes
Object precludes = toLoad.getPackageMetaDataElement(PRECLUDES_KEY);
if (precludes != null) {
try {
List installed = getInstalledPackages();
List preclusions = toLoad.getPrecludedPackages(installed);
if (preclusions.size() > 0) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Skipping package "
+ toLoad.getName()
+ " because it precludes one or more packages that are "
+ "already installed: ");
for (Package prec : preclusions) {
p.println("\t" + prec);
}
}
return false;
}
} catch (Exception ex) {
ex.printStackTrace();
}
}
load = !m_doNotLoadList.contains(toLoad.getName());
if (!load) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Skipping package " + toLoad.getName()
+ " because it is has been marked as do not load");
}
return false;
}
if (m_offline) {
return true;
}
// now check for missing dependencies
try {
List missing = toLoad.getMissingDependencies();
if (missing.size() > 0) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] " + toLoad.getName()
+ " can't be loaded because the following"
+ " packages are missing:");
for (Dependency d : missing) {
p.println(d.getTarget());
}
}
return false;
}
} catch (Exception ex) {
ex.printStackTrace();
return false;
}
// now check for buggered dependencies
try {
List depends = toLoad.getDependencies();
for (Dependency d : depends) {
if (d.getTarget().getPackage().isInstalled()) {
if (!loadCheck(d.getTarget().getPackage(), packageRoot, progress)) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Can't load " + toLoad.getName()
+ " because " + d.getTarget() + " can't be loaded.");
}
return false;
}
// check that the version of installed dependency is OK
Package installedD =
getInstalledPackageInfo(d.getTarget().getPackage().getName());
if (!d.getTarget().checkConstraint(installedD)) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Can't load " + toLoad.getName()
+ " because the installed "
+ d.getTarget().getPackage().getName()
+ " is not compatible (requires: " + d.getTarget() + ")");
}
return false;
}
}
}
} catch (Exception ex) {
ex.printStackTrace();
return false;
}
return true;
}
/**
* Checks the supplied package against the JVM version running Weka. Packages
* that don't specify a JVM version are assumed to be OK. The entry in the
* JVMVersion key are expected to be a floating point number, optionally
* prefixed by either a greater-than or less-than symbol. Absence of either
* of these symbols imply equality as the test.
*
* @param toLoad the package to check
* @param progress PrintStream for progress info
* @return true if the supplied package passes the JVM version test.
*/
public static boolean vmVersionCheck(Package toLoad, PrintStream... progress) {
String thisVMVersion = System.getProperty("java.specification.version");
if (thisVMVersion != null && thisVMVersion.length() > 0) {
double actualVM = -1;
try {
actualVM = Double.parseDouble(thisVMVersion);
} catch (NumberFormatException e) {
e.printStackTrace();
return true;
}
try {
Object reqVMS = toLoad.getPackageMetaDataElement(VM_VERSION_KEY);
if (reqVMS != null) {
String vm = reqVMS.toString();
String op = "=";
if (vm.charAt(0) == '>') {
op = ">";
vm = vm.substring(1);
} else if (vm.charAt(0) == '<') {
op = "<";
vm = vm.substring(1);
}
double requestedVM = -1;
try {
requestedVM = Double.parseDouble(vm);
} catch (NumberFormatException e) {
e.printStackTrace();
return true;
}
boolean result = true;
String failureMessage = "";
if (op.equals("=")) {
result = requestedVM == actualVM;
if (!result) {
failureMessage = "[WekaPackageManager] Unable to load '"
+ toLoad.getName() + "' because "
+ "JVM " + actualVM + " does not match requested JVM " + requestedVM;
}
} else if (op.equals("<")) {
result = actualVM < requestedVM;
if (!result) {
failureMessage = "[WekaPackageManager] Unable to load '"
+ toLoad.getName() +"' because "
+ "JVM " + actualVM + " is not < requested JVM " + requestedVM;
}
} else {
result = actualVM > requestedVM;
if (!result) {
failureMessage = "[WekaPackageManager] Unable to laod '"
+ toLoad.getName() + "' because "
+ "JVM " + actualVM + " is not > requested JVM " + requestedVM;
}
}
if (!result) {
for (PrintStream p : progress) {
p.println(failureMessage);
}
}
return result;
}
} catch (Exception ex) {
ex.printStackTrace();
}
}
return true;
}
/**
* Checks the supplied package against the current OS and architecture.
* Packages that don't specify OS and (optionally) architecture constraints
* are assumed to be OK. OS names in the OSName entry of the package's
* Description.props are checked against System.getProperty("os.name") via a
* String.contains() comparison. Any single match results in a pass. If
* supplied, the package's OSArch entries are checked against
* System.getProperty("os.arch") using a String.equalsIgnoreCase() comparison,
* with the exception of the values "64" and "32" which are checked for with
* String.contains().
*
* @param toLoad the package to check
* @param progress PrintStream for progress info
* @return true if the supplied package passes OS/arch constraints.
*/
public static boolean osAndArchCheck(Package toLoad, PrintStream... progress) {
// check for OS restrictions
try {
Object osNames = toLoad.getPackageMetaDataElement(OS_NAME_KEY);
Object archNames = toLoad.getPackageMetaDataElement(OS_ARCH_KEY);
if (osNames != null) {
boolean osCheck = false;
String[] osElements = osNames.toString().split(",");
String thisOs = System.getProperty("os.name");
if (thisOs != null) {
for (String osEntry : osElements) {
if (thisOs.toLowerCase().contains(osEntry.trim().toLowerCase())) {
osCheck = true;
break;
}
}
String thisArch = System.getProperty("os.arch");
if (osCheck && archNames != null) {
String[] archElements = archNames.toString().split(",");
if (thisArch != null) {
boolean archCheck = false;
for (String archEntry : archElements) {
if (archEntry.trim().equalsIgnoreCase("32")
|| archEntry.trim().equalsIgnoreCase("64")) {
if (thisArch.toLowerCase().contains(archEntry.trim())) {
archCheck = true;
break;
}
} else {
if (thisArch.trim().equalsIgnoreCase(archEntry.trim())) {
archCheck = true;
break;
}
}
}
osCheck = archCheck;
}
}
if (!osCheck) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] Skipping package "
+ toLoad.getName() + " because the OS/arch (" + thisOs + " "
+ (thisArch != null ? thisArch : "")
+ ") does not meet package OS/arch constraints: "
+ osNames.toString() + " "
+ (archNames != null ? archNames.toString() : ""));
}
return false;
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
return true;
}
/**
* Checks to see if there are any missing files/directories for a given
* package. If there are missing files, then the package can't be loaded. An
* example would be a connector package that, for whatever reason, can't
* include a necessary third-party jar file in its lib folder, and requires
* the user to download and install this jar file manually.
*
* @param toLoad the package to check
* @param packageRoot the root directory of the package
* @return true if good to go
*/
public static boolean checkForMissingFiles(Package toLoad, File packageRoot,
PrintStream... progress) {
boolean result = true;
Object doNotLoadIfFileMissing =
toLoad.getPackageMetaDataElement(DO_NOT_LOAD_IF_FILE_NOT_PRESENT_KEY);
String packageRootPath = packageRoot.getPath() + File.separator;
if (doNotLoadIfFileMissing != null
&& doNotLoadIfFileMissing.toString().length() > 0) {
StringTokenizer tok =
new StringTokenizer(doNotLoadIfFileMissing.toString(), ",");
while (tok.hasMoreTokens()) {
String nextT = tok.nextToken().trim();
File toCheck = new File(packageRootPath + nextT);
if (!toCheck.exists()) {
for (PrintStream p : progress) {
p.println("[WekaPackageManager] " + toLoad.getName()
+ " can't be loaded because " + toCheck.getPath()
+ " appears to be missing.");
}
result = false;
break;
}
}
}
if (!result) {
// grab the message to print to the log (if any)
Object doNotLoadMessage =
toLoad
.getPackageMetaDataElement(DO_NOT_LOAD_IF_FILE_NOT_PRESENT_MESSAGE_KEY);
if (doNotLoadMessage != null && doNotLoadMessage.toString().length() > 0) {
String dnlM = doNotLoadMessage.toString();
try {
dnlM = Environment.getSystemWide().substitute(dnlM);
} catch (Exception ex) {
// quietly ignore
}
for (PrintStream p : progress) {
p.println("[WekaPackageManager] " + dnlM);
}
}
}
return result;
}
/**
* Reads the doNotLoad list (if it exists) from the packages directory
*
* @return a set of package names that should not be loaded. This will be
* empty if the doNotLoadList does not exist on disk.
*/
@SuppressWarnings("unchecked")
protected static Set getDoNotLoadList() {
Set doNotLoad = new HashSet();
File doNotLoadList =
new File(PACKAGES_DIR.toString() + File.separator + "doNotLoad.ser");
if (doNotLoadList.exists() && doNotLoadList.isFile()) {
ObjectInputStream ois = null;
try {
ois =
new ObjectInputStream(new BufferedInputStream(new FileInputStream(
doNotLoadList)));
doNotLoad = (Set) ois.readObject();
} catch (FileNotFoundException ex) {
} catch (IOException e) {
System.err
.println("An error occurred while reading the doNotLoad list: "
+ e.getMessage());
} catch (ClassNotFoundException e) {
System.err
.println("An error occurred while reading the doNotLoad list: "
+ e.getMessage());
} finally {
if (ois != null) {
try {
ois.close();
} catch (IOException ex) {
ex.printStackTrace();
}
}
}
}
return doNotLoad;
}
/**
* Toggle the load status of the supplied list of package names
*
* @param packageNames the packages to toggle the load status for
* @return a list of unknown packages (i.e. any supplied package names that
* don't appear to be installed)
* @throws Exception if a problem occurs
*/
public static List toggleLoadStatus(List packageNames)
throws Exception {
List unknownPackages = new ArrayList();
boolean changeMade = false;
for (String s : packageNames) {
Package p = PACKAGE_MANAGER.getInstalledPackageInfo(s);
if (p == null) {
unknownPackages.add(s);
} else {
if (m_doNotLoadList.contains(s)) {
m_doNotLoadList.remove(s);
} else {
// only mark as do not load if a package is loadable
if (loadCheck(p, new File(WekaPackageManager.getPackageHome()
.toString() + File.separator + s))) {
m_doNotLoadList.add(s);
}
}
changeMade = true;
}
}
if (changeMade) {
// write the list back to disk
File doNotLoadList =
new File(PACKAGES_DIR.toString() + File.separator + "doNotLoad.ser");
ObjectOutputStream oos = null;
try {
oos =
new ObjectOutputStream(new BufferedOutputStream(new FileOutputStream(
doNotLoadList)));
oos.writeObject(m_doNotLoadList);
} finally {
if (oos != null) {
oos.flush();
oos.close();
}
}
}
return unknownPackages;
}
/**
* Load all packages
*
* @param verbose true if loading progress should be output
*/
public static synchronized void loadPackages(boolean verbose) {
loadPackages(verbose, false, true);
}
/**
* Load all packages
*
* @param verbose true if loading progress should be output
* @param avoidTriggeringFullClassDiscovery true if we should avoid processing
* any properties files that might cause a full class discovery run,
* and may involve instantiating GUI classes.
* @param refreshGOEProperties true if the GOE properties should be refreshed
* after loading (i.e. a re-run of the class discovery mechanism,
* re-initialization of the Knowledge Flow etc.)
*/
public static synchronized void loadPackages(boolean verbose,
boolean avoidTriggeringFullClassDiscovery, boolean refreshGOEProperties) {
if (!verbose) {
// debug property overrides
String debug = System.getProperty("weka.core.classloader.debug", "false");
verbose = debug.equalsIgnoreCase("true");
} else {
System.setProperty("weka.core.classloader.debug", "true");
}
List goePropsFiles = new ArrayList();
if (!m_loadPackages) {
return;
}
if (m_packagesLoaded) {
return;
}
m_packagesLoaded = true;
m_initialPackageLoadingInProcess = true;
if (establishWekaHome()) {
// try and load any jar files and add to the classpath
File[] contents = PACKAGES_DIR.listFiles();
// if we have a non-empty packages dir then check
// the integrity of our cache first
if (contents.length > 0) {
// establishCacheIfNeeded(System.out);
startupCheck(false, System.out);
}
// dynamic injection of dependencies between packages
Map> injectDependencies = new HashMap<>();
for (File content : contents) {
if (content.isDirectory()) {
try {
Package toLoad = getInstalledPackageInfo(content.getName());
boolean load;
// Only perform the check against the current version of Weka if
// there exists
// a Description.props file
if (toLoad != null) {
load = loadCheck(toLoad, content, System.err);
if (load) {
if (verbose) {
System.out.println("[WekaPackageManager] loading package "
+ content.getName());
}
checkForInjectDependencies(toLoad, injectDependencies);
WekaPackageClassLoaderManager
.getWekaPackageClassLoaderManager().addPackageToClassLoader(
content);
}
}
} catch (Exception ex) {
ex.printStackTrace();
System.err.println("[WekaPackageManager] Problem loading package "
+ content.getName() + " skipping...");
}
}
}
// now inject any package dependencies
injectPackageDependencies(injectDependencies);
// now check overall integrity
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.performIntegrityCheck();
// now process the various properties files in the packages
for (File content : contents) {
try {
if (content.isDirectory()
&& WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(content.getName()) != null) {
processPackageDirectory(content, verbose, goePropsFiles,
avoidTriggeringFullClassDiscovery);
}
} catch (Exception ex) {
ex.printStackTrace();
System.err.println("[WekaPackageManager] Problem loading package "
+ content.getName() + " skipping...");
}
}
}
m_initialPackageLoadingInProcess = false;
// it is best to process all of these after all jars have been
// inserted into the classpath since the dynamic class discovery
// mechanism will load classes during the process of determining
// all implementations of base types, and this can cause problems
// if processed at the time of package loading and there are
// dependencies between packages
if (!avoidTriggeringFullClassDiscovery) {
for (File f : goePropsFiles) {
processGenericPropertiesCreatorProps(f);
}
}
// do we need to regenerate the list of available schemes for
// the GUIs (this is not necessary when executing stuff from
// the command line)
if (refreshGOEProperties) {
if (verbose) {
System.err.println("Refreshing GOE props...");
}
refreshGOEProperties();
}
}
/**
* Checks a package for the presence of any specified package dependencies to
* inject at load time
*
* @param toLoad the package to check
* @param injectDeps a map (keyed by source package name) containing lists of
* target packages. Each source package will have the entries in its
* target package list injected (via the WekaLibIsolatingClassLoader)
* as dependencies.
*/
protected static void checkForInjectDependencies(Package toLoad,
Map> injectDeps) {
Object injectList = toLoad.getPackageMetaDataElement(INJECT_DEPENDENCY_KEY);
if (injectList != null) {
String[] deps = injectList.toString().split(",");
for (String dep : deps) {
String[] sourceAndTarget = dep.split(":");
if (sourceAndTarget.length == 2) {
String depender = sourceAndTarget[0].trim();
String dependee = sourceAndTarget[1].trim();
List depList = injectDeps.get(depender);
if (depList == null) {
depList = new ArrayList<>();
injectDeps.put(depender, depList);
}
depList.add(dependee);
}
}
}
}
/**
* Processes a map of dependencies to inject. Each source package (key) will
* have the packages named in the associated list (value) injected as
* dependencies at load time.
*
* @param injectDependencies a map of source to target dependencies.
*/
protected static void injectPackageDependencies(
Map> injectDependencies) {
for (Map.Entry> e : injectDependencies.entrySet()) {
String source = e.getKey();
List targets = e.getValue();
WekaPackageLibIsolatingClassLoader sourceLoader =
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(source);
if (sourceLoader != null) {
String debugS =
System.getProperty("weka.core.classloader.debug", "false");
boolean debug = debugS.equalsIgnoreCase("true");
for (String targetPackage : targets) {
if (WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(targetPackage) != null) {
if (debug) {
System.out
.println("[WekaPackageManager] Added a dependency between "
+ source + " and " + targetPackage);
}
sourceLoader.addPackageDependency(targetPackage);
}
}
}
}
}
/**
* Refresh the generic object editor properties via re-running of the dynamic
* class discovery process.
*/
public static void refreshGOEProperties() {
ClassDiscovery.clearClassCache();
GenericPropertiesCreator.regenerateGlobalOutputProperties();
GenericObjectEditor.determineClasses();
ConverterUtils.initialize();
ConverterFileChooser.initDefaultFilters();
KnowledgeFlowApp.disposeSingleton();
KnowledgeFlowApp.reInitialize();
}
/**
* Get the underlying package manager implementation
*
* @return the underlying concrete package management implementation.
*/
public static PackageManager getUnderlyingPackageManager() {
return PACKAGE_MANAGER;
}
/**
* Retrieves the size (in KB) of the repository zip archive stored on the
* server.
*
* @return the size of the repository zip archive in KB.
*/
public static int repoZipArchiveSize() {
int size = -1;
try {
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
String numPackagesS =
PACKAGE_MANAGER.getPackageRepositoryURL().toString() + "/repoSize.txt";
URLConnection conn = null;
URL connURL = new URL(numPackagesS);
if (PACKAGE_MANAGER.setProxyAuthentication(connURL)) {
conn = connURL.openConnection(PACKAGE_MANAGER.getProxy());
} else {
conn = connURL.openConnection();
}
conn.setConnectTimeout(30000); // timeout after 30 seconds
BufferedReader bi =
new BufferedReader(new InputStreamReader(conn.getInputStream()));
String n = bi.readLine();
try {
size = Integer.parseInt(n);
} catch (NumberFormatException ne) {
System.err.println("[WekaPackageManager] problem parsing the size "
+ "of repository zip archive from the server.");
}
bi.close();
} catch (Exception ex) {
ex.printStackTrace();
}
return size;
}
/**
* Get the number of packages that are available at the repository.
*
* @return the number of packages that are available (or -1 if this can't be
* determined for some reason.
*/
public static int numRepositoryPackages() {
if (m_offline) {
return -1;
}
int numPackages = -1;
try {
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
String numPackagesS =
PACKAGE_MANAGER.getPackageRepositoryURL().toString()
+ "/numPackages.txt";
URLConnection conn = null;
URL connURL = new URL(numPackagesS);
if (PACKAGE_MANAGER.setProxyAuthentication(connURL)) {
conn = connURL.openConnection(PACKAGE_MANAGER.getProxy());
} else {
conn = connURL.openConnection();
}
conn.setConnectTimeout(30000); // timeout after 30 seconds
BufferedReader bi =
new BufferedReader(new InputStreamReader(conn.getInputStream()));
String n = bi.readLine();
try {
numPackages = Integer.parseInt(n);
} catch (NumberFormatException ne) {
System.err.println("[WekaPackageManager] problem parsing number "
+ "of packages from server.");
}
bi.close();
} catch (Exception ex) {
ex.printStackTrace();
}
return numPackages;
}
/**
* Just get a list of the package names. This is faster than calling
* getAllPackages(), especially if fetching from the online repository, since
* the full meta data for each package doesn't have to be read.
*
* @param local true if the local package list in the cache should be read
* rather than the online repository
*
* @return a Map of all the package names available either
* locally or at the repository
*/
public static Map getPackageList(boolean local) {
Map result = new HashMap();
try {
useCacheOrOnlineRepository();
if (!local) {
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
}
String packageListS =
PACKAGE_MANAGER.getPackageRepositoryURL().toString() + "/"
+ PACKAGE_LIST_WITH_VERSION_FILE;
URLConnection conn = null;
URL connURL = new URL(packageListS);
if (PACKAGE_MANAGER.setProxyAuthentication(connURL)) {
conn = connURL.openConnection(PACKAGE_MANAGER.getProxy());
} else {
conn = connURL.openConnection();
}
conn.setConnectTimeout(30000); // timeout after 30 seconds
BufferedReader bi =
new BufferedReader(new InputStreamReader(conn.getInputStream()));
String l = null;
while ((l = bi.readLine()) != null) {
String[] parts = l.split(":");
if (parts.length == 2) {
result.put(parts[0], parts[1]);
}
}
bi.close();
} catch (Exception ex) {
// ex.printStackTrace();
result = null;
}
return result;
}
/**
* Establish the local copy of the package meta data if needed
*
* @param progress for reporting progress
* @return any Exception raised or null if all is good
*/
public static Exception establishCacheIfNeeded(PrintStream... progress) {
if (m_offline) {
return null;
}
Exception problem = null;
if (INITIAL_CACHE_BUILD_NEEDED) {
for (PrintStream p : progress) {
p.println("Caching repository metadata, please wait...");
}
problem = refreshCache(progress);
INITIAL_CACHE_BUILD_NEEDED = false;
}
return problem;
}
protected static boolean checkForForcedCacheRefresh()
throws MalformedURLException {
int refreshCountServer = getForcedRefreshCount(false);
if (refreshCountServer > 0) {
// now check local version of this file...
int refreshCountLocal = getForcedRefreshCount(true);
return refreshCountServer > refreshCountLocal;
}
return false;
}
protected static int getForcedRefreshCount(boolean local)
throws MalformedURLException {
useCacheOrOnlineRepository();
if (!local) {
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
}
String refreshCountS =
PACKAGE_MANAGER.getPackageRepositoryURL().toString() + "/"
+ FORCED_REFRESH_COUNT_FILE;
int refreshCount = -1;
URLConnection conn = null;
URL connURL = new URL(refreshCountS);
try {
if (PACKAGE_MANAGER.setProxyAuthentication(connURL)) {
conn = connURL.openConnection(PACKAGE_MANAGER.getProxy());
} else {
conn = connURL.openConnection();
}
conn.setConnectTimeout(30000); // timeout after 30 seconds
BufferedReader bi =
new BufferedReader(new InputStreamReader(conn.getInputStream()));
String countS = bi.readLine();
if (countS != null && countS.length() > 0) {
try {
refreshCount = Integer.parseInt(countS);
} catch (NumberFormatException ne) {
// ignore
}
}
} catch (IOException ex) {
// ignore
}
return refreshCount;
}
/**
* Check for new packages on the server and refresh the local cache if needed
*
* @param progress to report progress to
* @return any Exception raised or null if all is good
*/
public static Exception checkForNewPackages(PrintStream... progress) {
if (m_offline) {
return null;
}
Exception problem = null;
Map localPackageNameList = getPackageList(true);
if (localPackageNameList == null) {
System.err.println("Local package list is missing, trying a "
+ "cache refresh to restore...");
refreshCache(progress);
localPackageNameList = getPackageList(true);
if (localPackageNameList == null) {
// quietly return and see if we can continue anyway
return null;
}
}
Map repositoryPackageNameList = getPackageList(false);
if (repositoryPackageNameList == null) {
// quietly return and see if we can continue anyway
return null;
}
if (repositoryPackageNameList.keySet().size() != localPackageNameList
.keySet().size()) {
// package(s) have disappeared from the repository.
// Force a cache refresh...
if (repositoryPackageNameList.keySet().size() < localPackageNameList
.keySet().size()) {
for (PrintStream p : progress) {
p.println("Some packages no longer exist at the repository. "
+ "Refreshing cache...");
}
} else {
for (PrintStream p : progress) {
p.println("There are new packages at the repository. "
+ "Refreshing cache...");
}
}
problem = refreshCache(progress);
} else {
// check for new versions of packages
boolean refresh = false;
for (String localPackage : localPackageNameList.keySet()) {
String localVersion = localPackageNameList.get(localPackage);
String repoVersion = repositoryPackageNameList.get(localPackage);
if (repoVersion == null) {
continue;
}
// a difference here indicates a newer version on the server
if (!localVersion.equals(repoVersion)) {
refresh = true;
break;
}
}
if (refresh) {
for (PrintStream p : progress) {
p.println("There are newer versions of existing packages "
+ "at the repository. Refreshing cache...");
}
problem = refreshCache(progress);
}
}
return problem;
}
/**
* Deletes the contents of $WEKA_HOME/repCache
*/
protected static void cleanRepCacheDir() {
String cacheDir = WEKA_HOME.toString() + File.separator + "repCache";
File cacheDirF = new File(cacheDir);
if (cacheDirF.exists()) {
File[] contents = cacheDirF.listFiles();
if (contents != null) {
for (File f : contents) {
if (f.isDirectory()) {
File[] packageMetaDirContents = f.listFiles();
if (packageMetaDirContents != null) {
for (File pf : packageMetaDirContents) {
pf.delete();
}
}
}
f.delete();
}
}
}
}
/**
* Refresh the local copy of the package meta data
*
* @param progress to report progress to
* @return any Exception raised or null if all is successful
*/
public static Exception refreshCache(PrintStream... progress) {
Exception problem = null;
if (CACHE_URL == null) {
return null;
}
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
String cacheDir = WEKA_HOME.toString() + File.separator + "repCache";
try {
for (PrintStream p : progress) {
p.println("Refresh in progress. Please wait...");
}
byte[] zip =
PACKAGE_MANAGER.getRepositoryPackageMetaDataOnlyAsZip(progress);
// only blow away the repCache if we successfully get a new zip!
cleanRepCacheDir();
ZipInputStream zis = new ZipInputStream(new ByteArrayInputStream(zip));
ZipEntry ze;
final byte[] buff = new byte[1024];
while ((ze = zis.getNextEntry()) != null) {
// System.out.println("Cache: inflating " + ze.getName());
if (ze.isDirectory()) {
new File(cacheDir, ze.getName()).mkdir();
continue;
}
BufferedOutputStream bo =
new BufferedOutputStream(new FileOutputStream(new File(cacheDir,
ze.getName())));
while (true) {
int amountRead = zis.read(buff);
if (amountRead == -1) {
break;
}
// write the data here
bo.write(buff, 0, amountRead);
}
bo.close();
}
} catch (Exception e) {
e.printStackTrace();
// OK, we have a problem with the repository cache - use
// the repository itself instead and delete repCache
CACHE_URL = null;
try {
DefaultPackageManager.deleteDir(new File(cacheDir), System.out);
} catch (Exception e1) {
e1.printStackTrace();
}
return e;
}
return problem;
}
/**
* Check if a named resource exists in an installed package
*
* @param packageName the name of the package in question
* @param resourceName the name of the resource to check for
* @return true if the resource exists in the package
*/
public static boolean installedPackageResourceExists(String packageName,
String resourceName) {
String fullResourcePath =
getPackageHome().toString() + File.separator + packageName
+ File.separator + resourceName;
return new File(fullResourcePath).exists();
}
/**
* Determines whether to use the local cache or online repository for meta
* data
*/
private static void useCacheOrOnlineRepository() {
if (REP_MIRROR == null && m_mirrorFailureCount < 3) {
establishMirror();
if (REP_MIRROR == null) {
m_mirrorFailureCount++;
}
}
if (CACHE_URL != null) {
PACKAGE_MANAGER.setPackageRepositoryURL(CACHE_URL);
} else if (REP_URL != null) {
PACKAGE_MANAGER.setPackageRepositoryURL(REP_URL);
}
}
public static File getPackageHome() {
return PACKAGE_MANAGER.getPackageHome();
}
/**
* Find the most recent version of the package encapsulated in the supplied
* PackageConstraint argument that satisfies the constraint
*
* @param toCheck the PackageConstraint containing the package in question
* @return the most recent version of the package satisfying the constraint
* @throws Exception if a version can't be found that satisfies the constraint
* or an error occurs while communicating with the respository
*/
public static Package mostRecentVersionWithRespectToConstraint(
PackageConstraint toCheck) throws Exception {
Package target = toCheck.getPackage();
Package result = null;
List availableVersions =
PACKAGE_MANAGER.getRepositoryPackageVersions(target.getName());
// version numbers will be in descending sorted order from the repository
// we want the most recent version that meets the target constraint
for (Object version : availableVersions) {
Package candidate =
PACKAGE_MANAGER.getRepositoryPackageInfo(target.getName(), version);
if (toCheck.checkConstraint(candidate)
&& candidate.isCompatibleBaseSystem()) {
result = candidate;
break;
}
}
if (result == null) {
throw new Exception("[WekaPackageManager] unable to find a version of "
+ "package " + target.getName() + " that meets constraint "
+ toCheck.toString());
}
return result;
}
/**
* Install the supplied list of packages
*
* @param toInstall packages to install
* @param progress to report progress to
* @return true if successful
* @throws Exception if a problem occurs
*/
public static boolean installPackages(List toInstall,
PrintStream... progress) throws Exception {
useCacheOrOnlineRepository();
List upgrades = new ArrayList();
for (Package p : toInstall) {
if (p.isInstalled()) {
upgrades.add(new Boolean(true));
} else {
upgrades.add(new Boolean(false));
}
}
PACKAGE_MANAGER.installPackages(toInstall, progress);
boolean atLeastOneUpgrade = false;
List gpcFiles = new ArrayList();
int i = 0;
Map> injectDependencies = new HashMap<>();
for (Package p : toInstall) {
boolean isAnUpgrade = upgrades.get(i++);
if (isAnUpgrade) {
atLeastOneUpgrade = true;
}
String packageName = p.getName();
File packageDir =
new File(PACKAGE_MANAGER.getPackageHome().toString() + File.separator
+ packageName);
checkForInjectDependencies(p, injectDependencies);
boolean loadIt = loadCheck(p, packageDir, progress);
if (loadIt & !isAnUpgrade) {
processPackageDirectory(packageDir, false, gpcFiles, false);
}
}
// inject dependencies before triggering any class discovery
injectPackageDependencies(injectDependencies);
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.performIntegrityCheck();
for (File f : gpcFiles) {
processGenericPropertiesCreatorProps(f);
}
return atLeastOneUpgrade;
}
/**
* Get the versions of the supplied package available on the server
*
* @param packageName the package name to get available versions for
* @return a list of available versions
* @throws Exception if a problem occurs
*/
public static List getRepositoryPackageVersions(String packageName)
throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getRepositoryPackageVersions(packageName);
}
/**
* Get the package repository URL
*
* @return the package repository URL
*/
public static URL getPackageRepositoryURL() {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getPackageRepositoryURL();
}
/**
* Get a list of all packages
*
* @return a list of all packages
* @throws Exception if a problem occurs
*/
public static List getAllPackages() throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getAllPackages();
}
/**
* Get a list of all available packages (i.e. those not yet installed(.
*
* @return a list of all available packages
* @throws Exception if a problem occurs
*/
public static List getAvailablePackages() throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getAvailablePackages();
}
/**
* Get a list of the most recent version of all available packages (i.e. those
* not yet installed or there is a higher version in the repository) that are
* compatible with the version of Weka that is installed.
*
* @return a list of packages that are compatible with the installed version
* of Weka
* @throws Exception if a problem occurs
*/
public static List getAvailableCompatiblePackages() throws Exception {
// List allAvail = getAvailablePackages();
List allAvail = getAllPackages();
List compatible = new ArrayList();
for (Package p : allAvail) {
List availableVersions =
PACKAGE_MANAGER.getRepositoryPackageVersions(p.getName());
// version numbers will be in descending sorted order from the repository
// we want the most recent version that is compatible with the base weka
// version
for (Object version : availableVersions) {
Package versionedPackage =
getRepositoryPackageInfo(p.getName(), version.toString());
if (versionedPackage.isCompatibleBaseSystem()) {
if (p.isInstalled()) {
// see if the latest compatible version is newer than the installed
// version
Package installed = getInstalledPackageInfo(p.getName());
String installedV =
installed.getPackageMetaDataElement(
VersionPackageConstraint.VERSION_KEY).toString();
String versionedV =
versionedPackage.getPackageMetaDataElement(
VersionPackageConstraint.VERSION_KEY).toString();
VersionPackageConstraint.VersionComparison v =
VersionPackageConstraint.compare(versionedV, installedV);
if (v == VersionPackageConstraint.VersionComparison.GREATERTHAN) {
compatible.add(versionedPackage);
}
} else {
compatible.add(versionedPackage);
}
break;
}
}
}
return compatible;
}
/**
* Get the latest version of the named package that is compatible with the
* base version of Weka being used.
*
* @param packageName the name of the package to get the latest compatible
* version of
* @return the latest compatible version or null if there is no compatible
* package
* @throws Exception if a problem occurs
*/
public static Package getLatestCompatibleVersion(String packageName)
throws Exception {
Package latest = null;
List availableVersions =
PACKAGE_MANAGER.getRepositoryPackageVersions(packageName);
for (Object version : availableVersions) {
Package versionedPackage =
getRepositoryPackageInfo(packageName, version.toString());
if (versionedPackage.isCompatibleBaseSystem()) {
latest = versionedPackage;
break;
}
}
return latest;
}
/**
* Get a list of installed packages
*
* @return a list of installed packages
* @throws Exception if a problem occurs
*/
public static List getInstalledPackages() throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getInstalledPackages();
}
/**
* Get a list of dependencies for a given package
*
* @param target the package to get the dependencies for
* @param conflicts will hold any conflicts
* @return a list of dependencies for the target package
* @throws Exception if a problem occurs
*/
public static List getAllDependenciesForPackage(Package target,
Map> conflicts) throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getAllDependenciesForPackage(target, conflicts);
}
/**
* Extract meta data from a package archive
*
* @param packageArchivePath the path to the package archive
* @return the meta data for the package
* @throws Exception if a problem occurs
*/
public static Package getPackageArchiveInfo(String packageArchivePath)
throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getPackageArchiveInfo(packageArchivePath);
}
/**
* Get meta data for an installed package
*
* @param packageName the name of the package
* @return the meta data for the package
* @throws Exception if a problem occurs
*/
public static Package getInstalledPackageInfo(String packageName)
throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getInstalledPackageInfo(packageName);
}
/**
* Get meta data for the latest version of a package from the repository
*
* @param packageName the name of the package
* @return the meta data for the package
* @throws Exception if a problem occurs
*/
public static Package getRepositoryPackageInfo(String packageName)
throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getRepositoryPackageInfo(packageName);
}
/**
* Get meta data for a specific version of a package from the repository
*
* @param packageName the name of the package
* @param version the version to get meta data for
* @return the meta data for the package
* @throws Exception if a problem occurs
*/
public static Package getRepositoryPackageInfo(String packageName,
String version) throws Exception {
useCacheOrOnlineRepository();
return PACKAGE_MANAGER.getRepositoryPackageInfo(packageName, version);
}
/**
* Install a named package by retrieving the location of the archive from the
* meta data stored in the repository
*
* @param packageName the name of the package to install
* @param version the version of the package to install
* @param progress for reporting progress
* @return true if the package was installed successfully
* @throws Exception if a problem occurs
*/
public static boolean installPackageFromRepository(String packageName,
String version, PrintStream... progress) throws Exception {
useCacheOrOnlineRepository();
Package toLoad = getRepositoryPackageInfo(packageName);
// check to see if a version is already installed. If so, we
// wont load the updated version into the classpath immediately in
// order to avoid conflicts, class not found exceptions etc. The
// user is told to restart Weka for the changes to come into affect
// anyway, so there is no point in loading the updated package now.
boolean isAnUpgrade = toLoad.isInstalled();
Object specialInstallMessage =
toLoad.getPackageMetaDataElement(MESSAGE_TO_DISPLAY_ON_INSTALLATION_KEY);
if (specialInstallMessage != null
&& specialInstallMessage.toString().length() > 0) {
String siM = specialInstallMessage.toString();
try {
siM = Environment.getSystemWide().substitute(siM);
} catch (Exception ex) {
// quietly ignore
}
String message =
"**** Special installation message ****\n" + siM
+ "\n**** Special installation message ****";
for (PrintStream p : progress) {
p.println(message);
}
}
PACKAGE_MANAGER
.installPackageFromRepository(packageName, version, progress);
File packageDir =
new File(PACKAGE_MANAGER.getPackageHome().toString() + File.separator
+ packageName);
if (loadCheck(getInstalledPackageInfo(packageName), packageDir, progress)) {
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.addPackageToClassLoader(packageDir);
}
Map> injectDependencies = new HashMap<>();
checkForInjectDependencies(toLoad, injectDependencies);
// inject dependencies before triggering any class discovery
injectPackageDependencies(injectDependencies);
// check that all dependencies are available, there are no missing classes
// and files etc.
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.performIntegrityCheck();
// If the classloader for the package is still in play then
// process all props files
if (WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(packageName) != null) {
processPackageDirectory(packageDir, false, null, false);
}
return isAnUpgrade;
}
/**
* Install a package from an archive (unofficial package install route)
*
* @param packageArchivePath the path to the package archive file to install
* @param progress for reporting progress
* @return true if the package was installed successfully
* @throws Exception if a problem occurs
*/
public static String installPackageFromArchive(String packageArchivePath,
PrintStream... progress) throws Exception {
useCacheOrOnlineRepository();
Package toInstall =
PACKAGE_MANAGER.getPackageArchiveInfo(packageArchivePath);
Object specialInstallMessage =
toInstall
.getPackageMetaDataElement(MESSAGE_TO_DISPLAY_ON_INSTALLATION_KEY);
if (specialInstallMessage != null
&& specialInstallMessage.toString().length() > 0) {
String siM = specialInstallMessage.toString();
try {
siM = Environment.getSystemWide().substitute(siM);
} catch (Exception ex) {
// quietly ignore
}
String message =
"**** Special installation message ****\n" + siM
+ "\n**** Special installation message ****";
for (PrintStream p : progress) {
p.println(message);
}
}
PACKAGE_MANAGER.installPackageFromArchive(packageArchivePath, progress);
return initializeAndLoadUnofficialPackage(toInstall, progress);
}
private static String initializeAndLoadUnofficialPackage(Package toInstall,
PrintStream... progress) throws Exception {
File packageDir =
new File(PACKAGE_MANAGER.getPackageHome() + File.separator
+ toInstall.getName());
Package toLoad = getInstalledPackageInfo(toInstall.getName());
// no load check here as those checks involve the central repository (and
// this is an unofficial package)
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.addPackageToClassLoader(packageDir);
Map> injectDependencies = new HashMap<>();
checkForInjectDependencies(toLoad, injectDependencies);
// inject dependencies before triggering any class discovery
injectPackageDependencies(injectDependencies);
// check that all dependencies are available, there are no missing classes
// and files etc.
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.performIntegrityCheck();
// If the classloader for the package is still in play then
// process all props files
if (WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.getPackageClassLoader(toInstall.getName()) != null) {
processPackageDirectory(packageDir, false, null, false);
}
return toInstall.getName();
}
/**
* Install a package from the supplied URL
*
* @param packageURL the URL to the package archive to install
* @param progress for reporting progress
* @return true if the package was installed successfully
* @throws Exception if a problem occurs
*/
public static String installPackageFromURL(URL packageURL,
PrintStream... progress) throws Exception {
useCacheOrOnlineRepository();
String packageName =
PACKAGE_MANAGER.installPackageFromURL(packageURL, progress);
Package installed = PACKAGE_MANAGER.getInstalledPackageInfo(packageName);
Object specialInstallMessage =
installed
.getPackageMetaDataElement(MESSAGE_TO_DISPLAY_ON_INSTALLATION_KEY);
if (specialInstallMessage != null
&& specialInstallMessage.toString().length() > 0) {
String message =
"**** Special installation message ****\n"
+ specialInstallMessage.toString()
+ "\n**** Special installation message ****";
for (PrintStream p : progress) {
p.println(message);
}
}
return initializeAndLoadUnofficialPackage(installed, progress);
}
/**
* Uninstall a named package
*
* @param packageName the name of the package to remove
* @param updateKnowledgeFlow true if any Knowledge Flow beans provided by the
* package should be deregistered from the KnoweledgeFlow
* @param progress for reporting progress
* @throws Exception if a problem occurs
*/
public static void uninstallPackage(String packageName,
boolean updateKnowledgeFlow, PrintStream... progress) throws Exception {
// check to see if this is a KnowledgeFlow package (presence of Beans.props
// file)
if (updateKnowledgeFlow) {
File packageToDel =
new File(PACKAGE_MANAGER.getPackageHome().toString() + File.separator
+ packageName);
if (packageToDel.exists() && packageToDel.isDirectory()) {
File[] contents = packageToDel.listFiles();
if (contents != null) {
for (File content : contents) {
if (content.isFile() && content.getPath().endsWith("Beans.props")) {
// KnowledgeFlow plugin -- remove this properties file from the
// list
// of
// bean plugin props
KnowledgeFlowApp.removeFromPluginBeanProps(content);
KnowledgeFlowApp.disposeSingleton();
break;
}
}
}
WekaPackageClassLoaderManager.getWekaPackageClassLoaderManager()
.removeClassLoaderForPackage(packageName);
}
}
PACKAGE_MANAGER.uninstallPackage(packageName, progress);
}
private static void printPackageInfo(Map, ?> packageProps) {
Set> keys = packageProps.keySet();
Iterator> i = keys.iterator();
while (i.hasNext()) {
Object key = i.next();
Object value = packageProps.get(key);
System.out.println(Utils.padLeft(key.toString(), 11) + ":\t"
+ value.toString());
}
}
/**
* Print meta data on a package
*
* @param packagePath the path to the package to print meta data for
* @throws Exception if a problem occurs
*/
protected static void printPackageArchiveInfo(String packagePath)
throws Exception {
Map, ?> packageProps =
getPackageArchiveInfo(packagePath).getPackageMetaData();
printPackageInfo(packageProps);
}
/**
* Print meta data for an installed package
*
* @param packageName the name of the package to print meta data for
* @throws Exception if a problem occurs
*/
protected static void printInstalledPackageInfo(String packageName)
throws Exception {
Map, ?> packageProps =
getInstalledPackageInfo(packageName).getPackageMetaData();
printPackageInfo(packageProps);
}
/**
* Print meta data for a package listed in the repository
*
* @param packageName the name of the package to print meta data for
* @param version the version of the package
* @throws Exception if a problem occurs
*/
protected static void printRepositoryPackageInfo(String packageName,
String version) throws Exception {
Map, ?> packageProps =
getRepositoryPackageInfo(packageName, version).getPackageMetaData();
printPackageInfo(packageProps);
}
private static String queryUser() {
java.io.BufferedReader br =
new java.io.BufferedReader(new java.io.InputStreamReader(System.in));
String result = null;
try {
result = br.readLine();
} catch (java.io.IOException ex) {
// ignore
}
return result;
}
private static void removeInstalledPackage(String packageName, boolean force,
PrintStream... progress) throws Exception {
List compromised = new ArrayList();
// Now check to see which other installed packages depend on this one
List installedPackages = null;
if (!force) {
installedPackages = getInstalledPackages();
for (Package p : installedPackages) {
List tempDeps = p.getDependencies();
for (Dependency d : tempDeps) {
if (d.getTarget().getPackage().getName().equals(packageName)) {
// add this installed package to the list
compromised.add(p);
break;
}
}
}
if (compromised.size() > 0) {
System.out.println("The following installed packages depend on "
+ packageName + " :\n");
for (Package p : compromised) {
System.out.println("\t" + p.getName());
}
System.out.println("\nDo you wish to proceed [y/n]?");
String response = queryUser();
if (response != null
&& (response.equalsIgnoreCase("n") || response.equalsIgnoreCase("no"))) {
return; // bail out here
}
}
}
if (force) {
System.out.println("Forced uninstall.");
}
compromised = null;
installedPackages = null;
uninstallPackage(packageName, false, progress);
}
private static void installPackageFromRepository(String packageName,
String version, boolean force) throws Exception {
if (version.equals("Latest")) {
// if no version/latest has been specified by the user then
// look for the latest version of the package that is compatible
// with the installed version of Weka.
version = "none";
List availableVersions =
PACKAGE_MANAGER.getRepositoryPackageVersions(packageName);
// version numbers will be in descending sorted order from the
// repository. We want the most recent version that is compatible
// with the base weka install
for (Object v : availableVersions) {
Package versionedPackage =
getRepositoryPackageInfo(packageName, v.toString());
if (versionedPackage.isCompatibleBaseSystem()) {
version =
versionedPackage.getPackageMetaDataElement(
VersionPackageConstraint.VERSION_KEY).toString();
break;
}
}
if (version.equals("none")) {
throw new Exception("Was unable to find a version of '" + packageName
+ "' that is compatible with Weka " + Version.VERSION);
}
}
Package toInstall = null;
try {
toInstall = getRepositoryPackageInfo(packageName, version);
} catch (Exception ex) {
System.err.println("[WekaPackageManager] Package " + packageName
+ " at version " + version + " doesn't seem to exist!");
// System.exit(1);
return;
}
// First check to see if this package is compatible with the base system
if (!force) {
// check to see if it's disabled
Object disabled = toInstall.getPackageMetaDataElement(DISABLE_KEY);
if (disabled == null) {
disabled = toInstall.getPackageMetaDataElement(DISABLED_KEY);
}
if (disabled != null && disabled.toString().equalsIgnoreCase("true")) {
System.err.println("Can't install package " + packageName
+ " because it " + "has been disabled at the repository.");
return;
}
boolean ok = toInstall.isCompatibleBaseSystem();
if (!ok) {
List baseSysDep = toInstall.getBaseSystemDependency();
StringBuffer depList = new StringBuffer();
for (Dependency bd : baseSysDep) {
depList.append(bd.getTarget().toString() + " ");
}
System.err.println("Can't install package " + packageName
+ " because it requires " + depList.toString());
return;
}
if (!osAndArchCheck(toInstall, System.out)) {
return; // bail out here
}
if (!vmVersionCheck(toInstall, System.out)) {
return; // bail out
}
if (toInstall.isInstalled()) {
Package installedVersion = getInstalledPackageInfo(packageName);
if (!toInstall.equals(installedVersion)) {
System.out.println("Package " + packageName + "[" + installedVersion
+ "] is already installed. Replace with " + toInstall + " [y/n]?");
String response = queryUser();
if (response != null
&& (response.equalsIgnoreCase("n") || response
.equalsIgnoreCase("no"))) {
return; // bail out here
}
} else {
System.out.println("Package " + packageName + "[" + installedVersion
+ "] is already installed. Install again [y/n]?");
String response = queryUser();
if (response != null
&& (response.equalsIgnoreCase("n") || response
.equalsIgnoreCase("no"))) {
return; // bail out here
}
}
}
// Now get a full list of dependencies for this package and
// check for any conflicts
Map> conflicts =
new HashMap>();
List dependencies =
getAllDependenciesForPackage(toInstall, conflicts);
if (conflicts.size() > 0) {
System.err.println("Package " + packageName
+ " requires the following packages:\n");
Iterator depI = dependencies.iterator();
while (depI.hasNext()) {
Dependency d = depI.next();
System.err.println("\t" + d);
}
System.err.println("\nThere are conflicting dependencies:\n");
Set pNames = conflicts.keySet();
Iterator pNameI = pNames.iterator();
while (pNameI.hasNext()) {
String pName = pNameI.next();
System.err.println("Conflicts for " + pName);
List confsForPackage = conflicts.get(pName);
Iterator confs = confsForPackage.iterator();
while (confs.hasNext()) {
Dependency problem = confs.next();
System.err.println("\t" + problem);
}
}
System.err.println("Unable to continue with installation.");
return; // bail out here.
}
// Next check all dependencies against what is installed and
// inform the user about which installed packages will be altered. Also
// build the list of only those packages that need to be installed or
// upgraded (excluding those that are already installed and are OK).
List needsUpgrade = new ArrayList();
List finalListToInstall = new ArrayList();
Iterator depI = dependencies.iterator();
while (depI.hasNext()) {
Dependency toCheck = depI.next();
if (toCheck.getTarget().getPackage().isInstalled()) {
String toCheckName =
toCheck.getTarget().getPackage()
.getPackageMetaDataElement("PackageName").toString();
Package installedVersion =
PACKAGE_MANAGER.getInstalledPackageInfo(toCheckName);
if (!toCheck.getTarget().checkConstraint(installedVersion)) {
needsUpgrade.add(toCheck.getTarget());
Package mostRecent = toCheck.getTarget().getPackage();
if (toCheck.getTarget() instanceof weka.core.packageManagement.VersionPackageConstraint) {
mostRecent =
WekaPackageManager
.mostRecentVersionWithRespectToConstraint(toCheck.getTarget());
}
finalListToInstall.add(mostRecent);
}
} else {
Package mostRecent = toCheck.getTarget().getPackage();
if (toCheck.getTarget() instanceof weka.core.packageManagement.VersionPackageConstraint) {
mostRecent =
WekaPackageManager
.mostRecentVersionWithRespectToConstraint(toCheck.getTarget());
}
finalListToInstall.add(mostRecent);
}
}
// now check for precludes - first add compile all installed packages and
// then potentially overwrite with what's in the finalListToInstall
if (toInstall.getPackageMetaDataElement(PRECLUDES_KEY) != null) {
List installed = getInstalledPackages();
Map packageMap = new HashMap<>();
for (Package p : installed) {
packageMap.put(p.getName(), p);
}
for (Package p : finalListToInstall) {
packageMap.put(p.getName(), p);
}
List precluded =
toInstall.getPrecludedPackages(new ArrayList(packageMap
.values()));
if (precluded.size() > 0) {
List finalPrecluded = new ArrayList<>();
Set doNotLoadList = getDoNotLoadList();
for (Package p : precluded) {
if (!doNotLoadList.contains(p.getName())) {
finalPrecluded.add(p);
}
}
if (finalPrecluded.size() > 0) {
System.out.println("\nPackage " + toInstall.getName()
+ " cannot be "
+ "installed because it precludes the following packages:\n");
for (Package p : finalPrecluded) {
System.out.println("\n\t" + p.toString());
}
System.out
.println("Either uninstall or disable these packages before "
+ "continuing.");
return; // bail out here
}
}
}
if (needsUpgrade.size() > 0) {
System.out
.println("The following packages will be upgraded in order to install "
+ packageName);
Iterator upI = needsUpgrade.iterator();
while (upI.hasNext()) {
PackageConstraint tempC = upI.next();
System.out.println("\t" + tempC);
}
System.out.print("\nOK to continue [y/n]? > ");
String response = queryUser();
if (response != null
&& (response.equalsIgnoreCase("n") || response.equalsIgnoreCase("no"))) {
return; // bail out here
}
// now take a look at the other installed packages and see if
// any would have a problem when these ones are upgraded
boolean conflictsAfterUpgrade = false;
List installed = getInstalledPackages();
List toUpgrade = new ArrayList();
upI = needsUpgrade.iterator();
while (upI.hasNext()) {
toUpgrade.add(upI.next().getPackage());
}
// add the actual package the user is wanting to install if it
// is going to be an up/downgrade rather than a first install since
// other installed packages may depend on the currently installed
// version
// and thus could be affected after the up/downgrade
toUpgrade.add(toInstall);
for (int i = 0; i < installed.size(); i++) {
Package tempP = installed.get(i);
String tempPName = tempP.getName();
boolean checkIt = true;
for (int j = 0; j < needsUpgrade.size(); j++) {
if (tempPName.equals(needsUpgrade.get(j).getPackage().getName())) {
checkIt = false;
break;
}
}
if (checkIt) {
List problem =
tempP.getIncompatibleDependencies(toUpgrade);
if (problem.size() > 0) {
conflictsAfterUpgrade = true;
System.err.println("Package " + tempP.getName()
+ " will have a compatibility"
+ "problem with the following packages after upgrading them:");
Iterator dI = problem.iterator();
while (dI.hasNext()) {
System.err.println("\t" + dI.next().getTarget().getPackage());
}
}
}
}
if (conflictsAfterUpgrade) {
System.err.println("Unable to continue with installation.");
return; // bail out here
}
}
if (finalListToInstall.size() > 0) {
System.out.println("To install " + packageName
+ " the following packages will" + " be installed/upgraded:\n\n");
Iterator i = finalListToInstall.iterator();
while (i.hasNext()) {
System.out.println("\t" + i.next());
}
System.out.print("\nOK to proceed [y/n]? > ");
String response = queryUser();
if (response != null
&& (response.equalsIgnoreCase("n") || response.equalsIgnoreCase("no"))) {
return; // bail out here
}
}
// OK, now we can download and install everything
// First install the final list of dependencies
installPackages(finalListToInstall, System.out);
// Now install the package itself
installPackageFromRepository(packageName, version, System.out);
} else {
// just install this package without checking/downloading dependencies
// etc.
installPackageFromRepository(packageName, version, System.out);
}
}
private static void listPackages(String arg) throws Exception {
if (m_offline
&& (arg.equalsIgnoreCase("all") || arg.equalsIgnoreCase("available"))) {
System.out.println("Running offline - unable to display "
+ "available or all package information");
return;
}
List packageList = null;
useCacheOrOnlineRepository();
if (arg.equalsIgnoreCase("all")) {
packageList = PACKAGE_MANAGER.getAllPackages();
} else if (arg.equals("installed")) {
packageList = PACKAGE_MANAGER.getInstalledPackages();
} else if (arg.equals("available")) {
// packageList = PACKAGE_MANAGER.getAvailablePackages();
packageList = getAvailableCompatiblePackages();
} else {
System.err.println("[WekaPackageManager] Unknown argument " + arg);
printUsage();
// System.exit(1);
return;
}
StringBuffer result = new StringBuffer();
result.append("Installed\tRepository\tLoaded\tPackage\n");
result.append("=========\t==========\t======\t=======\n");
boolean userOptedNoLoad = false;
Iterator i = packageList.iterator();
while (i.hasNext()) {
Package p = i.next();
String installedV = "----- ";
String repositoryV = "----- ";
String loaded = "No";
if (p.isInstalled()) {
Package installedP = getInstalledPackageInfo(p.getName());
if (loadCheck(installedP, new File(WekaPackageManager.getPackageHome()
.toString() + File.separator + p.getName()))) {
loaded = "Yes";
} else {
if (m_doNotLoadList.contains(installedP.getName())) {
loaded = "No*";
userOptedNoLoad = true;
}
}
installedV =
installedP.getPackageMetaDataElement(VERSION_KEY).toString() + " ";
try {
if (!m_offline) {
Package repP = getRepositoryPackageInfo(p.getName());
repositoryV =
repP.getPackageMetaDataElement(VERSION_KEY).toString() + " ";
}
} catch (Exception ex) {
// not at the repository
}
} else {
repositoryV =
p.getPackageMetaDataElement(VERSION_KEY).toString() + " ";
}
String title = p.getPackageMetaDataElement("Title").toString();
result.append(installedV + "\t" + repositoryV + "\t" + loaded + "\t"
+ p.getName() + ": " + title + "\n");
}
if (userOptedNoLoad) {
result.append("* User flagged as no load\n");
}
System.out.println(result.toString());
}
public static Exception startupCheck(boolean force, PrintStream... progress) {
if (m_offline) {
return null;
}
Exception problem = null;
BufferedReader br = null;
PrintWriter pw = null;
File newPackageLastTimeCheckFile =
new File(WEKA_HOME.toString() + File.separator + "new_package_check.txt");
try {
// first check against last time that new packages were checked for
boolean doChecks = false;
long currentTime = System.currentTimeMillis();
if (!newPackageLastTimeCheckFile.exists()) {
doChecks = true;
} else if (!force) {
br = new BufferedReader(new FileReader(newPackageLastTimeCheckFile));
String t = br.readLine();
long lastTime = Long.parseLong(t);
doChecks = (currentTime - lastTime > 720L * 60L * 1000L);
}
if (doChecks || force) {
if (REP_MIRROR == null) {
establishMirror();
}
establishCacheIfNeeded(progress);
boolean forcedCacheRefresh = false;
try {
if (forcedCacheRefresh = checkForForcedCacheRefresh()) {
for (PrintStream p : progress) {
p.println("Forced repository metadata refresh, please wait...");
}
problem = refreshCache(progress);
}
} catch (MalformedURLException ex) {
problem = ex;
}
if (!forcedCacheRefresh) {
checkForNewPackages(progress);
}
pw = new PrintWriter(new FileWriter(newPackageLastTimeCheckFile));
pw.println(currentTime);
pw.flush();
}
} catch (IOException e) {
e.printStackTrace();
} finally {
if (br != null) {
try {
br.close();
} catch (IOException e) {
e.printStackTrace();
}
}
if (pw != null) {
pw.close();
}
}
return problem;
}
private static void printUsage() {
System.out
.println("Usage: weka.core.WekaPackageManager [-offline] [option]");
System.out
.println("Options:\n"
+ "\t-list-packages \n"
+ "\t-package-info "
+ "\n\t-install-package [version]\n"
+ "\t-uninstall-package packageName\n"
+ "\t-toggle-load-status packageName [packageName packageName ...]\n"
+ "\t-refresh-cache");
}
/**
* Main method for using the package manager from the command line
*
* @param args command line arguments
*/
public static void main(String[] args) {
weka.core.logging.Logger.log(weka.core.logging.Logger.Level.INFO,
"Logging started");
try {
// scan for -offline
for (int i = 0; i < args.length; i++) {
if (args[i].equals("-offline")) {
m_offline = true;
String[] temp = new String[args.length - 1];
if (i > 0) {
System.arraycopy(args, 0, temp, 0, i);
}
System.arraycopy(args, i + 1, temp, i, args.length - (i + 1));
args = temp;
}
}
// establishCacheIfNeeded(System.out);
// checkForNewPackages(System.out);
startupCheck(true, System.out);
if (args.length == 0 || args[0].equalsIgnoreCase("-h")
|| args[0].equalsIgnoreCase("-help")) {
printUsage();
// System.exit(1);
return;
}
if (args[0].equals("-package-info")) {
if (args.length < 3) {
printUsage();
return;
// System.exit(1);
}
if (args[1].equals("archive")) {
printPackageArchiveInfo(args[2]);
} else if (args[1].equals("installed")) {
printInstalledPackageInfo(args[2]);
} else if (args[1].equals("repository")) {
String version = "Latest";
if (args.length == 4) {
version = args[3];
}
try {
printRepositoryPackageInfo(args[2], version);
} catch (Exception ex) {
// problem with getting info on package from repository?
// Must not be an "official" repository package
System.out
.println("[WekaPackageManager] Nothing known about package "
+ args[2] + " at the repository!");
}
} else {
System.err
.println("[WekaPackageManager] Unknown argument " + args[2]);
printUsage();
return;
// System.exit(1);
}
} else if (args[0].equals("-install-package")) {
String targetLowerCase = args[1].toLowerCase();
if (targetLowerCase.startsWith("http://")
|| targetLowerCase.startsWith("https://")) {
URL packageURL = new URL(args[1]);
installPackageFromURL(packageURL, System.out);
} else if (targetLowerCase.endsWith(".zip")) {
installPackageFromArchive(args[1], System.out);
} else {
// assume a named package at the central repository
String version = "Latest";
if (args.length == 3) {
version = args[2];
}
installPackageFromRepository(args[1], version, false);
}
System.exit(0);
} else if (args[0].equals("-uninstall-package")) {
if (args.length < 2) {
printUsage();
return;
// System.exit(1);
}
boolean force = false;
if (args.length == 3) {
if (args[2].equals("-force")) {
force = true;
}
}
removeInstalledPackage(args[1], force, System.out);
// System.exit(0);
return;
} else if (args[0].equals("-list-packages")) {
if (args.length < 2) {
printUsage();
// System.exit(1);
return;
}
listPackages(args[1]);
} else if (args[0].equals("-toggle-load-status")) {
if (args.length == 1) {
printUsage();
return;
}
List toToggle = new ArrayList();
for (int i = 1; i < args.length; i++) {
toToggle.add(args[i].trim());
}
if (toToggle.size() >= 1) {
toggleLoadStatus(toToggle);
}
} else if (args[0].equals("-refresh-cache")) {
refreshCache(System.out);
} else {
System.err.println("Unknown option: " + args[0]);
printUsage();
}
// System.exit(0);
} catch (Exception ex) {
ex.printStackTrace();
}
}
}