/* * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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. * */ package org.apache.tools.ant.taskdefs; import java.io.File; import java.io.IOException; import org.apache.tools.ant.BuildException; import org.apache.tools.ant.Project; import org.apache.tools.ant.Task; import org.apache.tools.ant.types.Commandline; /** * Patches a file by applying a 'diff' file to it; requires "patch" to be * on the execution path. * * @since Ant 1.1 * * @ant.task category="utility" */ public class Patch extends Task { private File originalFile; private File directory; private boolean havePatchfile = false; private Commandline cmd = new Commandline(); /** * Halt on error return value from patch invocation. */ private boolean failOnError = false; /** * The file to patch; optional if it can be inferred from * the diff file * @param file the file to patch */ public void setOriginalfile(File file) { originalFile = file; } /** * The name of a file to send the output to, instead of patching * the file(s) in place; optional. * @param file the file to send the output to * @since Ant 1.6 */ public void setDestfile(File file) { if (file != null) { cmd.createArgument().setValue("-o"); cmd.createArgument().setFile(file); } } /** * The file containing the diff output; required. * @param file the file containing the diff output */ public void setPatchfile(File file) { if (!file.exists()) { throw new BuildException("patchfile " + file + " doesn\'t exist", getLocation()); } cmd.createArgument().setValue("-i"); cmd.createArgument().setFile(file); havePatchfile = true; } /** * flag to create backups; optional, default=false * @param backups if true create backups */ public void setBackups(boolean backups) { if (backups) { cmd.createArgument().setValue("-b"); } } /** * flag to ignore whitespace differences; default=false * @param ignore if true ignore whitespace differences */ public void setIgnorewhitespace(boolean ignore) { if (ignore) { cmd.createArgument().setValue("-l"); } } /** * Strip the smallest prefix containing <i>num</i> leading slashes * from filenames. * * <p>patch's <i>-p</i> option. * @param num number of lines to strip * @exception BuildException if num is < 0, or other errors */ public void setStrip(int num) throws BuildException { if (num < 0) { throw new BuildException("strip has to be >= 0", getLocation()); } cmd.createArgument().setValue("-p" + num); } /** * Work silently unless an error occurs; optional, default=false * @param q if true suppress set the -s option on the patch command */ public void setQuiet(boolean q) { if (q) { cmd.createArgument().setValue("-s"); } } /** * Assume patch was created with old and new files swapped; optional, * default=false * @param r if true set the -R option on the patch command */ public void setReverse(boolean r) { if (r) { cmd.createArgument().setValue("-R"); } } /** * The directory to run the patch command in, defaults to the * project's base directory. * @param directory the directory to run the patch command in * @since Ant 1.5 */ public void setDir(File directory) { this.directory = directory; } /** * If <code>true</code>, stop the build process if the patch command * exits with an error status. * @param value <code>true</code> if it should halt, otherwise * <code>false</code>. The default is <code>false</code>. * @since Ant 1.8.0 */ public void setFailOnError(boolean value) { failOnError = value; } private static final String PATCH = "patch"; /** * execute patch * @throws BuildException when it all goes a bit pear shaped */ @Override public void execute() throws BuildException { if (!havePatchfile) { throw new BuildException("patchfile argument is required", getLocation()); } Commandline toExecute = cmd.clone(); toExecute.setExecutable(PATCH); if (originalFile != null) { toExecute.createArgument().setFile(originalFile); } Execute exe = new Execute(new LogStreamHandler(this, Project.MSG_INFO, Project.MSG_WARN), null); exe.setCommandline(toExecute.getCommandline()); if (directory == null) { exe.setWorkingDirectory(getProject().getBaseDir()); } else { if (!directory.isDirectory()) { throw new BuildException(directory + " is not a directory.", getLocation()); } exe.setWorkingDirectory(directory); } log(toExecute.describeCommand(), Project.MSG_VERBOSE); try { int returncode = exe.execute(); if (Execute.isFailure(returncode)) { String msg = "'" + PATCH + "' failed with exit code " + returncode; if (failOnError) { throw new BuildException(msg); } log(msg, Project.MSG_ERR); } } catch (IOException e) { throw new BuildException(e, getLocation()); } } }