#!/bin/sh # # imerge - help merge one file to another interactively # # Copyright (c) 2007 Akinori MUSHA # # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF # SUCH DAMAGE. # # # The latest version of this script can be downloaded from a read-only # subversion repository available at the following URL: # # http://svn.idaemons.org/repos/imerge/ # # Typically you just want the trunk/ subdirectory out of the whole # source tree including all the past release snapshots and so on. # # $Id$ MYNAME="$(basename "$0")" VERSION="0.2.2" # workaround for ksh not having "local" __f__ () { local __x__; }; eval __f__ 2>/dev/null || local () { for __x__; do case "$__x__" in *=*) $__x__ ;; esac; done }; unset __f__ __x__ usage () { { echo "imerge version $VERSION - help merge one file to another interactively" echo "usage: $MYNAME [ -x diff_flags ] [ -X sdiff_flags ] source destination" } >&2 exit 1 } call_diff () { diff ${DIFF_FLAGS} "$@" } call_sdiff () { sdiff ${SDIFF_FLAGS} "$@" } do_imerge () { local src dest merged ans endloop contloop line1 line2 src="$1"; shift dest="$1"; shift if [ -L "$src" ]; then echo "$MYNAME: $src => $dest: Source file is a symlink." return fi if [ -L "$dest" ]; then echo "$MYNAME: $src => $dest: Destination file is a symlink." return fi if [ ! -f "$src" ]; then echo "$MYNAME: $src => $dest: Source file is not found." while printf "%s" "$MYNAME: Delete destination file, or skip? (d/S): "; do read ans $dest: Destination file is not found." while printf "%s" "$MYNAME: Install source file? (i/S): "; do read ans $dest: Destination file differs from source file." endloop= while printf "%s" "$MYNAME: Install source file, edit files, merge them, or skip? (i/e/m/S): "; do read ans "$dest" break ;; [Ee]*) "$EDITOR" "$src" "$dest" "$merged" contloop=t break ;; [Rr]*) echo "--- left (source): $(ls -l "$src")" echo "+++ right (destination): $(ls -l "$dest")" call_sdiff -o "$merged" "$src" "$dest" contloop=t break ;; [Ss]*) break ;; esac done [ -n "$contloop" ] && continue finalize return done } DIFF_FLAGS="-u" SDIFF_FLAGS="-a -s" while getopts x:X: o; do case "$o" in x) DIFF_FLAGS="$OPTARG" ;; X) SDIFF_FLAGS="$OPTARG" ;; \?) usage ;; esac done shift $(($OPTIND - 1)) if [ $# -ne 2 ]; then usage fi do_imerge "$@"