/* * (C) Copyright 2006-2012 Nuxeo SA (http://nuxeo.com/) and others. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * * Contributors: * bstefanescu, jcarsique */ package org.nuxeo.connect.update.task.update; import java.io.File; import java.util.Map; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.nuxeo.connect.update.PackageException; import org.nuxeo.connect.update.ValidationStatus; import org.nuxeo.connect.update.task.Command; import org.nuxeo.connect.update.task.Task; import org.nuxeo.connect.update.task.standalone.AbstractTask; import org.nuxeo.connect.update.task.standalone.commands.AbstractCommand; import org.nuxeo.connect.update.task.standalone.commands.CompositeCommand; import org.nuxeo.connect.update.task.standalone.commands.DeployPlaceholder; import org.nuxeo.connect.update.task.update.JarUtils.Match; import org.nuxeo.connect.update.xml.XmlWriter; import org.w3c.dom.Element; /** * @since 5.5 * @author <a href="mailto:bs@nuxeo.com">Bogdan Stefanescu</a> */ public class Update extends AbstractCommand { protected static final Log log = LogFactory.getLog(Update.class); public static final String ID = "update"; /** * The source file. It can be a file or a directory. */ protected File file; /** * The target file. It can be a directory since 5.5 */ protected File todir; protected boolean removeOnExit; protected boolean allowDowngrade = false; protected boolean upgradeOnly = false; protected Update(String id) { super(id); } public Update() { this(ID); } @Override public void initialize(Element element) throws PackageException { super.initialize(element); } @Override public void readFrom(Element element) throws PackageException { File dir = null; String v = element.getAttribute("dir"); if (v.length() > 0) { dir = new File(v); } v = element.getAttribute("file"); if (v.length() > 0) { if (dir != null) { file = new File(dir, v); } else { file = new File(v); } guardVars.put("file", file); } else { file = dir; guardVars.put("dir", dir); } v = element.getAttribute("todir"); if (v.length() > 0) { todir = new File(v); guardVars.put("todir", todir); } v = element.getAttribute("removeOnExit"); if (v.length() > 0) { removeOnExit = Boolean.parseBoolean(v); } v = element.getAttribute("allowDowngrade"); if (v.length() > 0) { allowDowngrade = Boolean.parseBoolean(v); } v = element.getAttribute("upgradeOnly"); if (v.length() > 0) { upgradeOnly = Boolean.parseBoolean(v); } } @Override public void writeTo(XmlWriter writer) { writer.start(ID); if (file != null) { writer.attr("file", file.getAbsolutePath()); } if (todir != null) { writer.attr("todir", todir.getAbsolutePath()); } if (removeOnExit) { writer.attr("removeOnExit", "true"); } if (allowDowngrade) { writer.attr("allowDowngrade", "true"); } if (upgradeOnly) { writer.attr("upgradeOnly", "true"); } writer.end(); } @Override protected void doValidate(Task task, ValidationStatus status) throws PackageException { if (file == null || todir == null) { status.addError("Cannot execute command in installer." + " Invalid update syntax: file or todir was not specified."); } if (todir.isFile()) { status.addError("Cannot execute command in installer." + " Invalid update command: todir should be a directory!"); } if (file.isFile()) { Match<String> match = JarUtils.findJarVersion(file.getName()); if (match == null) { status.addError("Cannot execute command in installer." + " Cannot use 'update' command for non versioned files!. File name must contain a version: " + file.getName()); } } else if (!file.isDirectory()) { status.addWarning("Ignored command in installer." + " Source file not found! " + file.getName()); } } @Override protected Command doRun(Task task, Map<String, String> prefs) throws PackageException { if (!file.exists()) { log.warn("Can't update using " + file + ". File is missing."); return null; } UpdateManager mgr = ((AbstractTask) task).getUpdateManager(); Command rollback; if (file.isDirectory()) { rollback = updateDirectory(task, file, mgr); } else { rollback = updateFile(task, file, mgr); } Command deploy = getDeployCommand(mgr, rollback); if (deploy != null) { deploy.run(task, prefs); } return rollback; } protected CompositeCommand updateDirectory(Task task, File dir, UpdateManager mgr) throws PackageException { CompositeCommand cmd = new CompositeCommand(); File[] files = dir.listFiles(); if (files != null) { for (File fileInDir : files) { cmd.addCommand(updateFile(task, fileInDir, mgr)); } } return cmd; } protected Rollback updateFile(Task task, File fileToUpdate, UpdateManager mgr) throws PackageException { UpdateOptions opt = UpdateOptions.newInstance(task.getPackage().getId(), fileToUpdate, todir); if (opt == null) { return null; } opt.setAllowDowngrade(allowDowngrade); opt.setUpgradeOnly(upgradeOnly); opt.deleteOnExit = removeOnExit; try { RollbackOptions r = mgr.update(opt); return new Rollback(r); } catch (VersionAlreadyExistException e) { // should never happen log.error(e, e); return null; } } /** * Method to be overridden by subclasses to provide a deploy command for hot reload * * @since 5.6 */ protected Command getDeployCommand(UpdateManager updateManager, Command rollbackCommand) { return new DeployPlaceholder(file); } }