X-Git-Url: https://iankelling.org/git/?a=blobdiff_plain;f=appendu-function;h=6c97f6cb5f57914469c38d54a6445757a34506c0;hb=27a02f00b4498bf7b1519f36d73a634c81710957;hp=7a11f28e909ac3846435f6d121b45cddb7f909bf;hpb=91ceba1b2739bb4d21d3c7887c0aff55f6da129d;p=tee-unique diff --git a/appendu-function b/appendu-function index 7a11f28..6c97f6c 100644 --- a/appendu-function +++ b/appendu-function @@ -3,14 +3,19 @@ # This program is under GPL v. 3 or later, see appendu() { - local help="Usage: appendu [OPTION]... FILE LINE... -Append unique. Append each LINE to FILE if it does not exist in FILE. -Appended lines are output to the terminal. + local help="Usage: appendu [OPTION]... FILE [LINE_SET]... + +Append unique. + +A LINE_SET is one or more lines. Append LINE_SET to FILE if it does not exist in +FILE. If no LINE_SET argument is given, read lines from stdin, and treat each +as a single LINE_SET. Appended text is output to the terminal. -s don't try to use sudo when it would help us read or write the file -- stop processing arguments --help display this message" + local readsudo writesudo x strings string content local dosudo=true while true; do @@ -28,33 +33,69 @@ Appended lines are output to the terminal. fi done - if (( ${#@} < 2 )); then - echo "error: need 2 or more arguments" + if [[ ${#@} == 0 ]]; then + echo "error: need 1 or more arguments" echo "$help" return 1 fi - local readsudo writesudo local file="$1" shift - + + local file_exists=false if [[ -e $file ]]; then + file_exists=true [[ -r $file ]] || readsudo=sudo [[ -w $file ]] || writesudo=sudo - else + else local dir="$(dirname "$file")" if [[ -d $dir ]]; then [[ ! -w $dir ]] && writesudo=sudo else echo "appendu error: $dir does not exist" - exit 1 + return 1 fi fi if ! $dosudo; then readsudo= writesudo= fi - for x in "$@"; do - [[ -e "$file" ]] && $readsudo grep -q "^$x$" "$file" || $writesudo tee -a "$file"<<<"$x" - done + + if (( $# == 0 )); then + unset IFS + while read -r x; do + strings+=( "$x" ) + done + else + strings=( "$@" ) + fi + + if $file_exists; then + # fix files with no newline at the end. + # the following command won't work right on them. + # e = run script, $a\ means append following text, but there is none, + # so sed only does what it always does when it was supposed to modify a file, + # which is append a newline if there was none. + sed -ie '$a\' "$file" + # command substitution removes any trailing newlines, so we have to add + # a non-newline ending, we randomly chose "b", then remove it. + content=$($readsudo cat "$file"; echo b) content=${content%b} + + # we aren't using regex because we want to match strings, + # but we also want our match to start at the beginning of a line, + # or the beginning of the file, and to end at a line ending. + # So we do some slick bash to match this. + local start="?(* +)" + local end=" +*" + for string in "${strings[@]}"; do + [[ $content != $start"$string"$end ]] && $writesudo tee -a "$file"<<<"$string" + done + else + for string in "${strings[@]}"; do + $writesudo tee -a "$file"<<<"${strings[@]}" + done + fi + return 0 }