/* * Copyright (C) 2005-2010 Alfresco Software Limited. * * This file is part of Alfresco * * Alfresco is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Alfresco 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 Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Alfresco. If not, see . */ package org.alfresco.repo.module.tool; import java.io.BufferedInputStream; import java.io.IOException; import java.io.InputStream; import java.util.Date; import java.util.Map; import java.util.Map.Entry; import java.util.Properties; import org.alfresco.error.AlfrescoRuntimeException; import org.alfresco.service.cmr.module.ModuleDetails; import org.alfresco.service.cmr.module.ModuleInstallState; import org.alfresco.util.VersionNumber; import org.safehaus.uuid.UUIDGenerator; import de.schlichtherle.truezip.file.TArchiveDetector; import de.schlichtherle.truezip.file.TConfig; import de.schlichtherle.truezip.file.TFile; import de.schlichtherle.truezip.file.TFileInputStream; import de.schlichtherle.truezip.file.TVFS; import de.schlichtherle.truezip.fs.archive.zip.ZipDriver; import de.schlichtherle.truezip.socket.sl.IOPoolLocator; /** * Module management tool. *

* Manages the modules installed in a war file. Allows modules to be installed, updated, enabled, disabled and * uninstalled. Information about the module installed is also available. * * @since 2.0 * * @author Roy Wetherall * @author Derek Hulley */ public class ModuleManagementTool implements LogOutput { /** Location of the default mapping properties file */ private static final String DEFAULT_FILE_MAPPING_PROPERTIES = "org/alfresco/repo/module/tool/default-file-mapping.properties"; /** Location of the AMP-specific mappings file */ private static final String FILE_MAPPING_PROPERTIES = "file-mapping.properties"; /** * The property to add to a custom {@link #FILE_MAPPING_PROPERTIES file-mapping.properties} to inherit the default values. * The default is true. */ private static final String PROP_INHERIT_DEFAULT = "include.default"; /** Standard directories found in the alfresco war */ public static final String BACKUP_DIR = WarHelper.MODULE_NAMESPACE_DIR+ "/backup"; /** Operations and options supperted via the command line interface to this class */ private static final String OP_INSTALL = "install"; private static final String OP_UNINSTALL = "uninstall"; private static final String OP_LIST = "list"; private static final String OPTION_VERBOSE = "-verbose"; private static final String OPTION_FORCE = "-force"; private static final String OPTION_PREVIEW = "-preview"; private static final String OPTION_NOBACKUP = "-nobackup"; private static final String OPTION_DIRECTORY = "-directory"; private static final String OPTION_PURGE = "-purge"; private static final String OPTION_HELP = "-help"; private static final int ERROR_EXIT_CODE = 1; private static final int SUCCESS_EXIT_CODE = 0; /** File mapping properties */ private Properties defaultFileMappingProperties; /** Indicates the current verbose setting */ private boolean verbose = false; WarHelper warHelper = new WarHelperImpl(this); /** * Constructor */ public ModuleManagementTool() { TConfig config = TConfig.get(); config.setArchiveDetector(new TArchiveDetector("war|amp", new ZipDriver(IOPoolLocator.SINGLETON))); // Load the default file mapping properties this.defaultFileMappingProperties = new Properties(); InputStream is = this.getClass().getClassLoader().getResourceAsStream(DEFAULT_FILE_MAPPING_PROPERTIES); try { this.defaultFileMappingProperties.load(is); } catch (IOException exception) { throw new ModuleManagementToolException("Unable to load default extension file mapping properties.", exception); } } /** * Indicates whether the management tool is currently in verbose reporting mode. * * @return true if verbose, false otherwise */ public boolean isVerbose() { return verbose; } /** * Sets the verbose setting for the mangement tool * * @param verbose true if verbose, false otherwise */ public void setVerbose(boolean verbose) { this.verbose = verbose; } /** * Installs all modules within a folder into the given WAR file. * @throws IOException * * @see #installModule(String, String, boolean, boolean, boolean) */ public void installModules(String directory, String warFileLocation) throws IOException { installModules(directory, warFileLocation, false, false, true); } public void installModules(String directoryLocation, String warFileLocation, boolean preview, boolean forceInstall, boolean backupWAR) throws IOException { java.io.File dir = new java.io.File(directoryLocation); if (dir.exists() == true) { if (backupWAR) { backupWar(new TFile(warFileLocation),true); backupWAR = false; //Set it to false so a backup doesn't occur again. } installModules(dir, warFileLocation, preview, forceInstall,backupWAR); } else { throw new ModuleManagementToolException("Invalid directory '" + directoryLocation + "'"); } } private void installModules(java.io.File dir, String warFileLocation, boolean preview, boolean forceInstall, boolean backupWAR) { java.io.File[] children = dir.listFiles(); if (children != null) { for (java.io.File child : children) { if (child.isFile() == true && child.getName().toLowerCase().endsWith(".amp") == true) { installModule(child.getPath(), warFileLocation, preview, forceInstall, backupWAR); } else { installModules(child, warFileLocation, preview, forceInstall, backupWAR); } } } } /** * Installs a given AMP file into a given WAR file. * * @see ModuleManagementTool.installModule(String, String, boolean, boolean, boolean) * * @param ampFileLocation the location of the AMP file to be installed * @param warFileLocation the location of the WAR file into which the AMP file is to be installed */ public void installModule(String ampFileLocation, String warFileLocation) { installModule(ampFileLocation, warFileLocation, false, false, true); } /** * Installs a given AMP file into a given WAR file. * * @param ampFileLocation the location of the AMP file to be installed * @param warFileLocation the location of the WAR file into which the AMP file is to be installed. * @param preview indicates whether this should be a preview install. This means that the process of * installation will be followed and reported, but the WAR file will not be modified. * @param forceInstall indicates whether the installed files will be replaced regardless of the currently installed * version of the AMP. Generally used during development of the AMP. * @param backupWAR indicates whether we should backup the war we are modifying or not */ public void installModule(String ampFileLocation, String warFileLocation, boolean preview, boolean forceInstall, boolean backupWAR) { try { outputVerboseMessage("Installing AMP '" + ampFileLocation + "' into WAR '" + warFileLocation + "'"); TFile warFile = new TFile(warFileLocation); if (!warFile.exists()) { throw new ModuleManagementToolException("The war file '" + warFile + "' does not exist."); } if (preview == false) { // Make sure the module and backup directory exisits in the WAR file TFile moduleDir = new TFile(warFileLocation + WarHelper.MODULE_NAMESPACE_DIR); if (moduleDir.exists() == false) { moduleDir.mkdir(); } backupWar(warFile, backupWAR); } // Get the details of the installing module String propertiesLocation = ampFileLocation + "/module.properties"; ModuleDetails installingModuleDetails = ModuleDetailsHelper.createModuleDetailsFromPropertyLocation(propertiesLocation); if (installingModuleDetails == null) { throw new ModuleManagementToolException("No module.properties file has been found in the installing .amp file '" + ampFileLocation + "'"); } String installingId = installingModuleDetails.getId(); VersionNumber installingVersion = installingModuleDetails.getVersion(); //A series of checks warHelper.checkCompatibleVersion(warFile, installingModuleDetails); warHelper.checkCompatibleEdition(warFile, installingModuleDetails); warHelper.checkModuleDependencies(warFile, installingModuleDetails); // Try to find an installed module by the ID ModuleDetails installedModuleDetails = warHelper.getModuleDetailsOrAlias(warFile, installingModuleDetails); uninstallIfNecessary(warFileLocation, installedModuleDetails, preview, forceInstall, installingVersion); outputVerboseMessage("Adding files relating to version '" + installingVersion + "' of module '" + installingId + "'"); InstalledFiles installedFiles = new InstalledFiles(warFileLocation, installingId); Properties directoryChanges = calculateChanges(ampFileLocation, warFileLocation, preview, forceInstall, installedFiles); if (preview == false) { //Now actually do the changes if (directoryChanges != null && directoryChanges.size() > 0) { for (Entry entry : directoryChanges.entrySet()) { TFile source = new TFile((String) entry.getKey()); TFile destination = new TFile((String) entry.getValue()); source.cp_rp(destination); } } // Save the installed file list installedFiles.save(); // Update the installed module details installingModuleDetails.setInstallState(ModuleInstallState.INSTALLED); installingModuleDetails.setInstallDate(new Date()); ModuleDetailsHelper.saveModuleDetails(warFileLocation, installingModuleDetails); // Set the modified date if (warFile.exists()) { warFile.setLastModified(System.currentTimeMillis()); } // Update the zip filessync TVFS.umount(); } } catch (IOException exception) { throw new ModuleManagementToolException("An IO error was encountered during deployment of the AMP into the WAR", exception); } } private void uninstallIfNecessary(String warFileLocation, ModuleDetails installedModuleDetails, boolean preview, boolean forceInstall, VersionNumber installingVersion) throws IOException { // Now clean up the old instance if (installedModuleDetails != null) { String installedId = installedModuleDetails.getId(); VersionNumber installedVersion = installedModuleDetails.getVersion(); int compareValue = installedVersion.compareTo(installingVersion); if (compareValue > 0) { // Trying to install an earlier version of the extension outputVerboseMessage("WARNING: A later version of this module is already installed in the WAR. Installation skipped. "+ "You could force the installation by passing the -force option.",false); return; } if (forceInstall == true) { // Warn of forced install outputVerboseMessage("WARNING: The installation of this module is being forced. All files will be removed and replaced regardless of exiting versions present.",false); } if (compareValue == 0) { // Trying to install the same extension version again outputVerboseMessage("WARNING: This version of this module is already installed in the WAR..upgrading.",false); } if (forceInstall == true || compareValue <= 0) { // Trying to update the extension, old files need to cleaned before we proceed outputVerboseMessage("Clearing out files relating to version '" + installedVersion + "' of module '" + installedId + "'",false); uninstallModule(installedId, warFileLocation, preview, true); } } } /** */ private Properties calculateChanges(String ampFileLocation, String warFileLocation, boolean preview, boolean forceInstall, InstalledFiles installedFiles) throws IOException { Properties dirChanges = new Properties(); // Check if a custom mapping file has been defined Properties fileMappingProperties = null; Properties customFileMappingProperties = getCustomFileMappings(ampFileLocation); if (customFileMappingProperties == null) { fileMappingProperties = defaultFileMappingProperties; } else { fileMappingProperties = new Properties(); // A custom mapping file was present. Check if it must inherit the default mappings. String inheritDefaultStr = customFileMappingProperties.getProperty(PROP_INHERIT_DEFAULT, "true"); if (inheritDefaultStr.equalsIgnoreCase("true")) { fileMappingProperties.putAll(defaultFileMappingProperties); } fileMappingProperties.putAll(customFileMappingProperties); fileMappingProperties.remove(PROP_INHERIT_DEFAULT); } // Copy the files from the AMP file into the WAR file for (Map.Entry entry : fileMappingProperties.entrySet()) { // The file mappings are expected to start with "/" String mappingSource = (String) entry.getKey(); if (mappingSource.length() == 0 || !mappingSource.startsWith("/")) { throw new AlfrescoRuntimeException("File mapping sources must start with '/', but was: " + mappingSource); } String mappingTarget = (String) entry.getValue(); if (mappingTarget.length() == 0 || !mappingTarget.startsWith("/")) { throw new AlfrescoRuntimeException("File mapping targets must start with '/' but was '" + mappingTarget + "'"); } mappingSource = mappingSource.trim(); //trim whitespace mappingTarget = mappingTarget.trim(); //trim whitespace // Run through the files one by one figuring out what we are going to do during the copy calculateCopyToWar(ampFileLocation, warFileLocation, mappingSource, mappingTarget, installedFiles, preview, forceInstall); // Get a reference to the source folder (if it isn't present don't do anything) TFile source = new TFile(ampFileLocation + "/" + mappingSource); if (source != null && source.list() != null) { // Add to the list of directory changes so we can implement the changes later. String sourceDir = ampFileLocation + mappingSource; String destinationDir = warFileLocation + mappingTarget; dirChanges.put(sourceDir, destinationDir); } } return dirChanges; } private void backupWar(TFile warFile, boolean backupWAR) throws IOException { // Make a backup of the war we are going to modify if (backupWAR == true) { String backupLocation = warFile.getAbsolutePath()+"-" + System.currentTimeMillis() + ".bak"; if (warFile.isArchive()) { outputVerboseMessage("Backing up WAR file..."); TFile source = new TFile(warFile.getAbsolutePath(), TArchiveDetector.NULL); TFile backup = new TFile(backupLocation, TArchiveDetector.NULL); source.cp_rp(backup); //Just copy the file } else { outputVerboseMessage("Backing up war DIRECTORY..."); TFile backup = new TFile(backupLocation); warFile.cp_rp(backup); //Copy the directory } outputVerboseMessage("WAR has been backed up to '" + backupLocation + "'"); } } /** * @return Returns the custom file mapping properties or null if they weren't overwritten */ private Properties getCustomFileMappings(String ampFileLocation) { TFile file = new TFile(ampFileLocation + "/" + FILE_MAPPING_PROPERTIES); if (!file.exists()) { // Nothing there return null; } Properties mappingProperties = new Properties(); InputStream is = null; try { is = new BufferedInputStream(new TFileInputStream(file)); mappingProperties.load(is); } catch (IOException exception) { throw new ModuleManagementToolException("Unable to load default extension file mapping properties.", exception); } finally { if (is != null) { try { is.close(); } catch (Throwable e ) {} } } return mappingProperties; } /** * Cleans the WAR file of all files relating to the currently installed version of the the Module. * * @param warFileLocation the war file location * @param moduleId the module id * @param preview indicates whether this is a preview installation * @param purge Fully delete all files (including those marked "PRESERVED") * @throws IOException */ public void uninstallModule(String moduleId,String warFileLocation, boolean preview, boolean purge) throws IOException { InstalledFiles installedFiles = new InstalledFiles(warFileLocation, moduleId); installedFiles.load(); for (String add : installedFiles.getAdds()) { // Remove file removeFile(warFileLocation, add, preview); } for (String mkdir : installedFiles.getMkdirs()) { // Remove folder removeFile(warFileLocation, mkdir, preview); } for (Map.Entry update : installedFiles.getUpdates().entrySet()) { if (preview == false) { // Recover updated file and delete backups TFile modified = new TFile(warFileLocation + update.getKey()); TFile backup = new TFile(warFileLocation + update.getValue()); backup.cp_rp(modified); backup.deleteOnExit(); } outputVerboseMessage("Recovering file '" + update.getKey() + "' from backup '" + update.getValue() + "'", true); } // Now remove the installed files list String installedFilesPathInWar = installedFiles.getFilePathInWar(); removeFile(warFileLocation, installedFilesPathInWar, preview); // Remove the module properties String modulePropertiesFileLocationInWar = ModuleDetailsHelper.getModulePropertiesFilePathInWar(moduleId); removeFile(warFileLocation, modulePropertiesFileLocationInWar, preview); } /** * Removes a file from the given location in the war file. * * @param warLocation the war file location * @param filePath the path to the file that is to be deleted * @param preview indicates whether this is a preview install */ private void removeFile(String warLocation, String filePath, boolean preview) { TFile removeFile = new TFile(warLocation + filePath); if (removeFile.exists() == true) { outputVerboseMessage("Removing file '" + filePath + "' from war", true); if (preview == false) { removeFile.delete(); } } else { outputVerboseMessage("The file '" + filePath + "' was expected for removal but was not present in the war", true); } } /** * Copies a file from the AMP location to the correct location in the WAR, interating on directories where appropraite. * * @param ampFileLocation the AMP file location * @param warFileLocation the WAR file location * @param sourceDir the directory in the AMP to copy from. It must start with "/". * @param destinationDir the directory in the WAR to copy to. It must start with "/". * @param installedFiles a list of the currently installed files * @param preview indicates whether this is a preview install or not * @param forceInstall indicates whether the installed files will be replaces regardless of the currently installed * version of the AMP. * @throws IOException throws any IOExpceptions thar are raised */ private void calculateCopyToWar(String ampFileLocation, String warFileLocation, String sourceDir, String destinationDir, InstalledFiles installedFiles, boolean preview, boolean forceInstall) throws IOException { if (sourceDir.length() == 0 || !sourceDir.startsWith("/")) { throw new IllegalArgumentException("sourceDir must start with '/'"); } if (destinationDir.length() == 0 || !destinationDir.startsWith("/")) { throw new IllegalArgumentException("destinationDir must start with '/'"); } // Handle source and destination if they are just the root '/' if (sourceDir.equals("/")) { sourceDir = ""; } if (destinationDir.equals("/")) { destinationDir = ""; } String sourceLocation = ampFileLocation + sourceDir; TFile ampConfig = new TFile(sourceLocation); java.io.File[] files = ampConfig.listFiles(); if (files != null) { for (java.io.File sourceChild : files) { String destinationFileLocation = warFileLocation + destinationDir + "/" + sourceChild.getName(); TFile destinationChild = new TFile(destinationFileLocation); if (sourceChild.isFile() == true) { String backupLocation = null; boolean createFile = false; if (destinationChild.exists() == false) { createFile = true; } else { if (forceInstall) { // Backup file about to be updated backupLocation = BACKUP_DIR + "/" + generateGuid() + ".bin"; if (preview == false) { TFile backupFile = new TFile(warFileLocation + backupLocation); destinationChild.cp_rp(backupFile); } } else { //Not a forced install, there is an existing file in the war, lets rollback the transaction, //throw an error and explain the problem. // File. // ZipController zipController = ZipController.getInstance(warFile); // zipController.reset(); throw new ModuleManagementToolException("ERROR: The amp will overwrite an existing file in the war '" + destinationDir + "/" + sourceChild.getName() + "'. Execution halted. By specifying -force , you can force installation of AMP regardless of the current war state."); } } if (createFile == true) { installedFiles.addAdd(destinationDir + "/" + sourceChild.getName()); this.outputVerboseMessage("File '" + destinationDir + "/" + sourceChild.getName() + "' added to war from amp", true); } else { installedFiles.addUpdate(destinationDir + "/" + sourceChild.getName(), backupLocation); this.outputMessage("WARNING: The file '" + destinationDir + "/" + sourceChild.getName() + "' is being overwritten by this module. The original has been backed-up to '" + backupLocation + "'", true); } } else { boolean mkdir = false; if (destinationChild.exists() == false) { mkdir = true; } calculateCopyToWar(ampFileLocation, warFileLocation, sourceDir + "/" + sourceChild.getName(), destinationDir + "/" + sourceChild.getName(), installedFiles, preview, forceInstall); if (mkdir == true) { installedFiles.addMkdir(destinationDir + "/" + sourceChild.getName()); this.outputVerboseMessage("Directory '" + destinationDir + "/" + sourceChild.getName() + "' added to war", true); } } } } } /** * @throws UnsupportedOperationException */ public void disableModule(String moduleId, String warLocation) { throw new UnsupportedOperationException("Disable module is not currently supported"); } /** * @throws UnsupportedOperationException */ public void enableModule(String moduleId, String warLocation) { throw new UnsupportedOperationException("Enable module is not currently supported"); } /** * Lists all the currently installed modules in the WAR * * @param warLocation the war location */ public void listModules(String warLocation) { ModuleDetails moduleDetails = null; boolean previous = this.verbose; this.verbose = true; boolean moduleFound = false; try { TFile moduleDir = new TFile(warLocation + WarHelper.MODULE_NAMESPACE_DIR); if (moduleDir.exists() == false) { outputVerboseMessage("No modules are installed in this WAR file"); } java.io.File[] dirs = moduleDir.listFiles(); if (dirs != null && dirs.length != 0) { for (java.io.File dir : dirs) { if (dir.isDirectory() == true) { TFile moduleProperties = new TFile(dir.getPath() + WarHelper.MODULE_CONFIG_IN_WAR); if (moduleProperties.exists() == true) { InputStream is = null; try { moduleFound = true; is = new TFileInputStream(moduleProperties); moduleDetails = ModuleDetailsHelper.createModuleDetailsFromPropertiesStream(is); } catch (IOException exception) { throw new ModuleManagementToolException("Unable to open module properties file '" + moduleProperties.getPath() + "'", exception); } finally { if (is != null) { try { is.close(); } catch (Throwable e ) {} } } outputVerboseMessage("Module '" + moduleDetails.getId() + "' installed in '" + warLocation + "'"); outputVerboseMessage(" Title: " + moduleDetails.getTitle(), true); outputVerboseMessage(" Version: " + moduleDetails.getVersion(), true); outputVerboseMessage(" Install Date: " + moduleDetails.getInstallDate(), true); outputVerboseMessage(" Description: " + moduleDetails.getDescription(), true); } } } } else { outputVerboseMessage("No modules are installed in this WAR file"); } if (!moduleFound) { outputVerboseMessage("No modules were found in this WAR file"); } } finally { this.verbose = previous; } } /** * Outputs a message the console (in verbose mode). * * @param message the message to output */ private void outputVerboseMessage(String message) { outputMessage(message, false, false, false); } /** * Outputs a message the console (in verbose mode). * * @param message the message to output */ private void outputErrorMessage(String message) { outputMessage(message, false, true, false); } /** * Outputs a message the console (in verbose mode). * * @param message the message to output * @param indent indicates that the message should be formated with an indent */ private void outputVerboseMessage(String message, boolean indent) { outputMessage(message, indent, false, false); } /** * Outputs a message to the console regardless of the verbose setting. * * @param message the message to output * @param indent indicates that the message should be formated with an indent */ private void outputMessage(String message, boolean indent) { outputMessage(message, indent, false, true); } /** * Outputs a message the console. Errors are always output, but others are only output in verbose mode. * * @param message the message to output * @param indent indicates that the message should be formated with an indent * @param error indicates that the message is an error. * @param stdout indicates that the message should output to the console regardless of verbose setting */ private void outputMessage(String message, boolean indent, boolean error, boolean stdout) { if (indent == true) { message = " - " + message; } if (error) { System.err.println(message); } else if (this.verbose == true || stdout == true) { System.out.println(message); } } /** * Main * * @param args command line interface arguments */ public static void main(String[] args) { if (args.length <= 1) { outputUsage(); System.exit(ERROR_EXIT_CODE); } ModuleManagementTool manager = new ModuleManagementTool(); String operation = args[0]; try { if (operation.equals(OPTION_HELP) == true) { outputUsage(); System.exit(SUCCESS_EXIT_CODE); } else if (operation.equals(OP_INSTALL) == true) { if (args.length < 3) { throw new UsageException(OP_INSTALL + " requires at least 3 arguments."); } String aepFileLocation = args[1]; String warFileLocation = args[2]; boolean forceInstall = false; boolean previewInstall = false; boolean backup = true; boolean directory = false; if (args.length > 3) { for (int i = 3; i < args.length; i++) { String option = args[i]; if (OPTION_VERBOSE.equals(option) == true) { manager.setVerbose(true); } else if (OPTION_FORCE.equals(option) == true) { forceInstall = true; } else if (OPTION_PREVIEW.equals(option) == true) { previewInstall = true; manager.setVerbose(true); } else if (OPTION_NOBACKUP.equals(option) == true) { backup = false; } else if (OPTION_DIRECTORY.equals(option) == true) { directory = true; } } } try { if (directory == false) { // Install the module manager.installModule(aepFileLocation, warFileLocation, previewInstall, forceInstall, backup); } else { // Install the modules from the directory manager.installModules(aepFileLocation, warFileLocation, previewInstall, forceInstall, backup); } } catch (IOException error) { throw new ModuleManagementToolException(error.getMessage()); } System.exit(SUCCESS_EXIT_CODE); } else if (OP_LIST.equals(operation) == true) { if (args.length != 2) { throw new UsageException(OP_LIST + " requires 2 arguments."); } // List the installed modules String warFileLocation = args[1]; manager.listModules(warFileLocation); System.exit(SUCCESS_EXIT_CODE); } else if (OP_UNINSTALL.equals(operation) == true) { if (args.length < 3) { throw new UsageException(OP_UNINSTALL + " requires at least 3 arguments."); } String moduleId = args[1]; String warLocation = args[2]; boolean purge = false; boolean preview = false; if (args.length >= 4) { for (int i = 3; i < args.length; i++) { String option = args[i]; if (OPTION_PURGE.equals(option) == true) { purge = true; } if (OPTION_PREVIEW.equals(option) == true) { preview = true; manager.setVerbose(true); } } } manager.setVerbose(true); manager.uninstallModule(moduleId, warLocation,preview, purge); System.exit(SUCCESS_EXIT_CODE); } else { throw new UsageException("Unknown operation " + operation + "."); } } catch (UsageException e) { manager.outputErrorMessage("Usage error: " + e.getMessage()); outputUsage(); System.exit(ERROR_EXIT_CODE); } catch (ModuleManagementToolException e) { // These are user-friendly manager.outputErrorMessage(e.getMessage()); System.exit(ERROR_EXIT_CODE); } catch (IOException error) { manager.outputErrorMessage(error.getMessage()); System.exit(ERROR_EXIT_CODE); } } /** * Generates a GUID, avoiding undesired imports. */ private static String generateGuid() { return UUIDGenerator.getInstance().generateTimeBasedUUID().toString(); } /** * Outputs the module management tool usage */ private static void outputUsage() { System.out.println("Module managment tool available commands:"); System.out.println("-----------------------------------------------------------\n"); System.out.println("install: Installs a AMP file(s) into an Alfresco WAR file, updates if an older version is already installed."); System.out.println("usage: install [options]"); System.out.println("valid options: "); System.out.println(" -verbose : enable verbose output"); System.out.println(" -directory : indicates that the amp file location specified is a directory."); System.out.println(" All amp files found in the directory and its sub directories are installed."); System.out.println(" -force : forces installation of AMP regardless of currently installed module version"); System.out.println(" -preview : previews installation of AMP without modifying WAR file"); System.out.println(" -nobackup : indicates that no backup should be made of the WAR\n"); System.out.println("-----------------------------------------------------------\n"); System.out.println("list: Lists all the modules currently installed in an Alfresco WAR file."); System.out.println("usage: list \n"); System.out.println("-----------------------------------------------------------\n"); System.out.println("uninstall: Uninstalls a module from the Alfresco WAR file."); System.out.println("usage: uninstall \n"); System.out.println("-----------------------------------------------------------\n"); } @Override public void info(Object message) { outputVerboseMessage(String.valueOf(message)); } private static class UsageException extends Exception { private static final long serialVersionUID = 1L; public UsageException(String message) { super(message); } } }