scripts/unify-sources.sh
author Alpar Juttner <alpar@cs.elte.hu>
Wed, 22 Oct 2008 14:39:04 +0100
changeset 332 3f9f3550dbf5
parent 325 0fbbb4bc42dd
parent 329 d900fd1e760f
child 341 f8832dc16d45
permissions -rwxr-xr-x
Merge
alpar@38
     1
#!/bin/bash
alpar@38
     2
alpar@38
     3
YEAR=`date +2003-%Y`
alpar@38
     4
HGROOT=`hg root`
alpar@38
     5
deba@324
     6
# file enumaration modes
deba@324
     7
deba@324
     8
function all_files() {
deba@324
     9
    hg status -a -m -c |
deba@324
    10
    cut -d ' ' -f 2 | grep -E '(\.(cc|h|dox)$|Makefile\.am$)' |
deba@324
    11
    while read file; do echo $HGROOT/$file; done
deba@324
    12
}
deba@324
    13
deba@324
    14
function modified_files() {
deba@324
    15
    hg status -a -m |
deba@324
    16
    cut -d ' ' -f 2 | grep -E  '(\.(cc|h|dox)$|Makefile\.am$)' |
deba@324
    17
    while read file; do echo $HGROOT/$file; done
deba@324
    18
}
deba@324
    19
deba@324
    20
function changed_files() {
deba@324
    21
    {
deba@324
    22
        if [ -n "$HG_PARENT1" ]
deba@324
    23
        then
deba@324
    24
            hg status --rev $HG_PARENT1:$HG_NODE -a -m
deba@324
    25
        fi
deba@324
    26
        if [ -n "$HG_PARENT2" ]
deba@324
    27
        then
deba@324
    28
            hg status --rev $HG_PARENT2:$HG_NODE -a -m
deba@324
    29
        fi
deba@324
    30
    } | cut -d ' ' -f 2 | grep -E '(\.(cc|h|dox)$|Makefile\.am$)' | 
deba@324
    31
    sort | uniq |
deba@324
    32
    while read file; do echo $HGROOT/$file; done
deba@324
    33
}
deba@324
    34
deba@324
    35
function given_files() {
deba@324
    36
    for file in $GIVEN_FILES
deba@324
    37
    do
deba@324
    38
	echo $file
deba@324
    39
    done
deba@324
    40
}
deba@324
    41
deba@324
    42
# actions
deba@324
    43
deba@324
    44
function update_action() {
deba@324
    45
    if ! diff -q $1 $2 >/dev/null
deba@324
    46
    then
deba@324
    47
	echo -n " [$3 updated]"
deba@324
    48
	rm $2
deba@324
    49
	mv $1 $2
deba@324
    50
	CHANGED=YES
deba@324
    51
    fi
deba@324
    52
}
deba@324
    53
deba@324
    54
function update_warning() {
deba@324
    55
    echo -n " [$2 warning]"
deba@324
    56
    WARNED=YES
deba@324
    57
}
deba@324
    58
deba@324
    59
function update_init() {
deba@324
    60
    echo Update source files...
deba@324
    61
    TOTAL_FILES=0
deba@324
    62
    CHANGED_FILES=0
deba@324
    63
    WARNED_FILES=0
deba@324
    64
}
deba@324
    65
deba@324
    66
function update_done() {
deba@324
    67
    echo $CHANGED_FILES out of $TOTAL_FILES files has been changed.
alpar@325
    68
    echo $WARNED_FILES out of $TOTAL_FILES files triggered warnings.
deba@324
    69
}
deba@324
    70
deba@324
    71
function update_begin() {
deba@324
    72
    ((TOTAL_FILES++))
deba@324
    73
    CHANGED=NO
deba@324
    74
    WARNED=NO
deba@324
    75
}
deba@324
    76
deba@324
    77
function update_end() {
deba@324
    78
    if [ $CHANGED == YES ]
deba@324
    79
    then
deba@324
    80
	((++CHANGED_FILES))
deba@324
    81
    fi
deba@324
    82
    if [ $WARNED == YES ]
deba@324
    83
    then
deba@324
    84
	((++WARNED_FILES))
deba@324
    85
    fi
deba@324
    86
}
deba@324
    87
deba@324
    88
function check_action() {
deba@324
    89
    if ! diff -q $1 $2 >/dev/null
deba@324
    90
    then
kpeter@329
    91
	echo
kpeter@329
    92
	echo -n "      $3 failed at line(s): "
kpeter@329
    93
	echo -n $(diff $1 $2 | grep '^[0-9]' | sed "s/^\(.*\)c.*$/ \1/g" | 
kpeter@329
    94
	          sed "s/,/-/g" | paste -s -d',')
deba@324
    95
	FAILED=YES
deba@324
    96
    fi
deba@324
    97
}
deba@324
    98
deba@324
    99
function check_warning() {
kpeter@329
   100
    echo
kpeter@329
   101
    if [ "$2" == 'long lines' ]
kpeter@329
   102
    then
kpeter@329
   103
        echo -n "      $2 warning at line(s): "
kpeter@329
   104
        echo -n $(grep -n -E '.{81,}' $1 | sed "s/^\([0-9]*\)/ \1\t/g" | 
kpeter@329
   105
                  cut -f 1 | paste -s -d',')
kpeter@329
   106
    else
kpeter@329
   107
        echo -n "      $2 warning"
kpeter@329
   108
    fi
deba@324
   109
    WARNED=YES
deba@324
   110
}
deba@324
   111
deba@324
   112
function check_init() {
deba@324
   113
    echo Check source files...
deba@324
   114
    FAILED_FILES=0
deba@324
   115
    WARNED_FILES=0
deba@324
   116
    TOTAL_FILES=0
deba@324
   117
}
deba@324
   118
deba@324
   119
function check_done() {
deba@324
   120
    echo $FAILED_FILES out of $TOTAL_FILES files has been failed.
alpar@325
   121
    echo $WARNED_FILES out of $TOTAL_FILES files triggered warnings.
deba@324
   122
deba@324
   123
    if [ $FAILED_FILES -gt 0 ]
deba@324
   124
    then
deba@324
   125
	return 1
deba@324
   126
    elif [ $WARNED_FILES -gt 0 ]
deba@324
   127
    then
deba@324
   128
	if [ "$WARNING" == 'INTERACTIVE' ]
deba@324
   129
	then
alpar@325
   130
	    echo -n "Are the files with warnings acceptable? (yes/no) "
deba@324
   131
	    while read answer
deba@324
   132
	    do
deba@324
   133
		if [ "$answer" == 'yes' ]
deba@324
   134
		then
deba@324
   135
		    return 0
deba@324
   136
		elif [ "$answer" == 'no' ]
deba@324
   137
		then
deba@324
   138
		    return 1
deba@324
   139
		fi
alpar@325
   140
		echo -n "Are the files with warnings acceptable? (yes/no) "
deba@324
   141
	    done
deba@324
   142
	elif [ "$WARNING" == 'WERROR' ]
deba@324
   143
	then
deba@324
   144
	    return 1
deba@324
   145
	fi
deba@324
   146
    fi
deba@324
   147
}
deba@324
   148
deba@324
   149
function check_begin() {
deba@324
   150
    ((TOTAL_FILES++))
deba@324
   151
    FAILED=NO
deba@324
   152
    WARNED=NO
deba@324
   153
}
deba@324
   154
deba@324
   155
function check_end() {
deba@324
   156
    if [ $FAILED == YES ]
deba@324
   157
    then
deba@324
   158
	((++FAILED_FILES))
deba@324
   159
    fi
deba@324
   160
    if [ $WARNED == YES ]
deba@324
   161
    then
deba@324
   162
	((++WARNED_FILES))
deba@324
   163
    fi
deba@324
   164
}
deba@324
   165
deba@324
   166
deba@324
   167
deba@324
   168
# checks
deba@324
   169
deba@324
   170
function header_check() {
deba@324
   171
    if echo $1 | grep -q -E 'Makefile\.am$'
deba@324
   172
    then
deba@324
   173
	return
deba@324
   174
    fi
deba@324
   175
alpar@38
   176
    TMP_FILE=`mktemp`
alpar@38
   177
alpar@208
   178
    (echo "/* -*- mode: C++; indent-tabs-mode: nil; -*-
alpar@38
   179
 *
alpar@208
   180
 * This file is a part of LEMON, a generic C++ optimization library.
alpar@38
   181
 *
alpar@38
   182
 * Copyright (C) "$YEAR"
alpar@38
   183
 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
alpar@38
   184
 * (Egervary Research Group on Combinatorial Optimization, EGRES).
alpar@38
   185
 *
alpar@38
   186
 * Permission to use, modify and distribute this software is granted
alpar@38
   187
 * provided that this copyright notice appears in all copies. For
alpar@38
   188
 * precise terms see the accompanying LICENSE file.
alpar@38
   189
 *
alpar@38
   190
 * This software is provided \"AS IS\" with no warranty of any kind,
alpar@38
   191
 * express or implied, and with no claim as to its suitability for any
alpar@38
   192
 * purpose.
alpar@38
   193
 *
alpar@38
   194
 */
alpar@38
   195
"
deba@324
   196
    awk 'BEGIN { pm=0; }
alpar@38
   197
     pm==3 { print }
alpar@38
   198
     /\/\* / && pm==0 { pm=1;}
alpar@38
   199
     /[^:blank:]/ && (pm==0 || pm==2) { pm=3; print;}
alpar@38
   200
     /\*\// && pm==1 { pm=2;}
alpar@38
   201
    ' $1
deba@324
   202
    ) >$TMP_FILE
alpar@208
   203
deba@324
   204
    "$ACTION"_action "$TMP_FILE" "$1" header
alpar@38
   205
}
alpar@38
   206
deba@324
   207
function tabs_check() {
deba@324
   208
    if echo $1 | grep -q -v -E 'Makefile\.am$'
deba@324
   209
    then
deba@324
   210
        OLD_PATTERN=$(echo -e '\t')
deba@324
   211
        NEW_PATTERN='        '
deba@324
   212
    else
deba@324
   213
        OLD_PATTERN='        '
deba@324
   214
        NEW_PATTERN=$(echo -e '\t')
deba@324
   215
    fi
alpar@208
   216
    TMP_FILE=`mktemp`
deba@324
   217
    cat $1 | sed -e "s/$OLD_PATTERN/$NEW_PATTERN/g" >$TMP_FILE
alpar@38
   218
deba@324
   219
    "$ACTION"_action "$TMP_FILE" "$1" 'tabs'
alpar@208
   220
}
alpar@208
   221
deba@324
   222
function spaces_check() {
alpar@208
   223
    TMP_FILE=`mktemp`
deba@324
   224
    cat $1 | sed -e 's/ \+$//g' >$TMP_FILE
alpar@208
   225
kpeter@328
   226
    "$ACTION"_action "$TMP_FILE" "$1" 'trailing spaces'
alpar@208
   227
}
alpar@208
   228
deba@324
   229
function long_lines_check() {
deba@324
   230
    if cat $1 | grep -q -E '.{81,}'
alpar@208
   231
    then
deba@324
   232
	"$ACTION"_warning $1 'long lines'
alpar@208
   233
    fi
alpar@208
   234
}
alpar@208
   235
deba@324
   236
# process the file
deba@324
   237
deba@324
   238
function process_file() {
kpeter@328
   239
    echo -n "    $ACTION $1..."
deba@324
   240
deba@324
   241
    CHECKING="header tabs spaces long_lines"
deba@324
   242
deba@324
   243
    "$ACTION"_begin $1
deba@324
   244
    for check in $CHECKING
alpar@38
   245
    do
deba@324
   246
	"$check"_check $1
alpar@38
   247
    done
deba@324
   248
    "$ACTION"_end $1
deba@324
   249
    echo
deba@324
   250
}
deba@324
   251
deba@324
   252
function process_all {
deba@324
   253
    "$ACTION"_init
deba@324
   254
    while read file
alpar@38
   255
    do
deba@324
   256
	process_file $file
deba@324
   257
    done < <($FILES)
deba@324
   258
    "$ACTION"_done
deba@324
   259
}
deba@324
   260
deba@324
   261
while [ $# -gt 0 ]
deba@324
   262
do
deba@324
   263
    
deba@324
   264
    if [ "$1" == '--help' ] || [ "$1" == '-h' ]
deba@324
   265
    then
deba@324
   266
	echo -n \
deba@324
   267
"Usage:
deba@324
   268
  $0 [OPTIONS] [files]
deba@324
   269
Options:
deba@324
   270
  --dry-run|-n
alpar@325
   271
     Check the files, but do not modify them.
deba@324
   272
  --interactive|-i
alpar@325
   273
     If --dry-run is specified and the checker emits warnings,
alpar@325
   274
     then the user is asked if the warnings should be considered
alpar@325
   275
     errors.
deba@324
   276
  --werror|-w
alpar@325
   277
     Make all warnings into errors.
deba@324
   278
  --all|-a
kpeter@328
   279
     Check all source files in the repository.
deba@324
   280
  --modified|-m
alpar@325
   281
     Check only the modified (and new) source files. This option is
alpar@325
   282
     useful to check the modification before making a commit.
deba@324
   283
  --changed|-c
deba@324
   284
     Check only the changed source files compared to the parent(s) of
alpar@325
   285
     the current hg node.  This option is useful as hg hook script.
alpar@325
   286
     To automatically check all your changes before making a commit,
alpar@325
   287
     add the following section to the appropriate .hg/hgrc file.
deba@324
   288
deba@324
   289
       [hooks]
deba@324
   290
       pretxncommit.checksources = scripts/unify-sources.sh -c -n -i
deba@324
   291
deba@324
   292
  --help|-h
deba@324
   293
     Print this help message.
deba@324
   294
  files
kpeter@328
   295
     The files to check/unify. If no file names are given, the modified
kpeter@328
   296
     source files will be checked/unified (just like using the
kpeter@328
   297
     --modified|-m option).
deba@324
   298
"
deba@324
   299
        exit 0
deba@324
   300
    elif [ "$1" == '--dry-run' ] || [ "$1" == '-n' ]
deba@324
   301
    then
kpeter@328
   302
	[ -n "$ACTION" ] && echo "Conflicting action options" >&2 && exit 1
deba@324
   303
	ACTION=check
deba@324
   304
    elif [ "$1" == "--all" ] || [ "$1" == '-a' ]
deba@324
   305
    then
kpeter@328
   306
	[ -n "$FILES" ] && echo "Conflicting target options" >&2 && exit 1
deba@324
   307
	FILES=all_files
deba@324
   308
    elif [ "$1" == "--changed" ] || [ "$1" == '-c' ]
deba@324
   309
    then
kpeter@328
   310
	[ -n "$FILES" ] && echo "Conflicting target options" >&2 && exit 1
deba@324
   311
	FILES=changed_files
deba@324
   312
    elif [ "$1" == "--modified" ] || [ "$1" == '-m' ]
deba@324
   313
    then
kpeter@328
   314
	[ -n "$FILES" ] && echo "Conflicting target options" >&2 && exit 1
deba@324
   315
	FILES=modified_files
deba@324
   316
    elif [ "$1" == "--interactive" ] || [ "$1" == "-i" ]
deba@324
   317
    then
kpeter@328
   318
	[ -n "$WARNING" ] && echo "Conflicting warning options" >&2 && exit 1
deba@324
   319
	WARNING='INTERACTIVE'
deba@324
   320
    elif [ "$1" == "--werror" ] || [ "$1" == "-w" ]
deba@324
   321
    then
kpeter@328
   322
	[ -n "$WARNING" ] && echo "Conflicting warning options" >&2 && exit 1
deba@324
   323
	WARNING='WERROR'
kpeter@328
   324
    elif [ $(echo x$1 | cut -c 2) == '-' ]
deba@324
   325
    then
deba@324
   326
	echo "Invalid option $1" >&2 && exit 1
deba@324
   327
    else
deba@324
   328
	[ -n "$FILES" ] && echo "Invalid option $1" >&2 && exit 1
deba@324
   329
	GIVEN_FILES=$@
deba@324
   330
	FILES=given_files
deba@324
   331
	break
deba@324
   332
    fi
deba@324
   333
    
deba@324
   334
    shift
deba@324
   335
done
deba@324
   336
deba@324
   337
if [ -z $FILES ]
deba@324
   338
then
deba@324
   339
    FILES=modified_files
alpar@38
   340
fi
deba@324
   341
deba@324
   342
if [ -z $ACTION ]
deba@324
   343
then
deba@324
   344
    ACTION=update
alpar@208
   345
fi
deba@324
   346
deba@324
   347
process_all