diff options
author | Wayne Davison <wayned@samba.org> | 2020-04-09 15:11:37 -0700 |
---|---|---|
committer | Wayne Davison <wayned@samba.org> | 2020-04-09 15:11:37 -0700 |
commit | c5fabfb068bba9381c0f65fea60a99a62b681008 (patch) | |
tree | f8062d8b1f2f5faca4e98c5b5238523130a11153 /support/git-set-file-times | |
parent | e2aee6c4afcae8f05d92cbf27741981b259f2b21 (diff) | |
download | rsync-c5fabfb068bba9381c0f65fea60a99a62b681008.tar.gz |
Set Copyright years and make them easier to update
I replaced git-set-file-times with an improved version that I wrote
recently (in python3). A new script uses it to figure out the
last-modified year for each *.[ch] file and updates its copyright.
It also puts the latest year into the latest-year.h file for the
output of --version.
Diffstat (limited to 'support/git-set-file-times')
-rwxr-xr-x | support/git-set-file-times | 119 |
1 files changed, 76 insertions, 43 deletions
diff --git a/support/git-set-file-times b/support/git-set-file-times index 53550b74..6fe641a3 100755 --- a/support/git-set-file-times +++ b/support/git-set-file-times @@ -1,43 +1,76 @@ -#!/usr/bin/perl -use strict; -use warnings; - -# Sets mtime and atime of files to the latest commit time in git. -# -# This is useful after the first clone of the rsync repository BEFORE you -# do any building. It is also safe if you have done a "make distclean". - -my %ls; -my $commit_time; -my $prefix = @ARGV && $ARGV[0] =~ s/^--prefix=// ? shift : ''; - -my $top_dir = `git rev-parse --show-toplevel`; -exit 1 unless $top_dir; -chomp($top_dir); - -chdir $top_dir or die "Failed to chdir to $top_dir\: $!\n"; - -$/ = "\0"; -open FH, '-|', qw( git ls-files -z ) or die "Failed to fork: $!"; -while (<FH>) { - chomp; - $ls{$_} = $_; -} -close FH; - -$/ = "\n"; -open FH, '-|', qw( git log -r --name-only --no-color --pretty=raw -z ), @ARGV or die "Failed to fork: $!"; -while (<FH>) { - chomp; - if (/^committer .*? (\d+) (?:[\-\+]\d+)$/) { - $commit_time = $1; - } elsif (s/\0\0commit [a-f0-9]{40}$// || s/\0$//) { - my @files = delete @ls{split(/\0/, $_)}; - @files = grep { defined $_ } @files; - next unless @files; - map { s/^/$prefix/ } @files; - utime $commit_time, $commit_time, @files; - } - last unless %ls; -} -close FH; +#!/usr/bin/python3 + +import os, re, argparse, subprocess +from datetime import datetime + +NULL_COMMIT_RE = re.compile(r'\0\0commit [a-f0-9]{40}$|\0$') + +def main(): + if not args.git_dir: + cmd = 'git rev-parse --show-toplevel 2>/dev/null || echo .' + top_dir = subprocess.check_output(cmd, shell=True).decode('utf-8').strip() + args.git_dir = os.path.join(top_dir, '.git') + if not args.prefix: + os.chdir(top_dir) + + git = [ 'git', '--git-dir=' + args.git_dir ] + + if args.tree: + cmd = git + 'ls-tree -z -r --name-only'.split() + [ args.tree ] + else: + cmd = git + 'ls-files -z'.split() + + proc = subprocess.Popen(cmd, stdout=subprocess.PIPE) + out = proc.communicate()[0].decode('utf-8') + ls = set(out.split('\0')) + ls.discard('') + + cmd = git + 'log -r --name-only --no-color --pretty=raw --no-renames -z'.split() + if args.tree: + cmd.append(args.tree) + cmd += ['--'] + args.files + + proc = subprocess.Popen(cmd, stdout=subprocess.PIPE) + for line in proc.stdout: + line = line.decode('utf-8').strip() + m = re.match(r'^committer .*? (\d+) [-+]\d+$', line) + if m: + commit_time = int(m[1]) + elif NULL_COMMIT_RE.search(line): + line = NULL_COMMIT_RE.sub('', line) + files = set(fn for fn in line.split('\0') if fn in ls) + if not files: + continue + for fn in files: + if args.prefix: + fn = args.prefix + fn + mtime = os.lstat(fn).st_mtime + if args.list: + if args.list > 1: + ts = str(commit_time).rjust(10) + else: + ts = datetime.utcfromtimestamp(commit_time).strftime("%Y-%m-%d %H:%M:%S") + chg = '.' if mtime == commit_time else '*' + print(chg, ts, fn) + elif mtime != commit_time: + if not args.quiet: + print(f"Setting {fn}") + os.utime(fn, (commit_time, commit_time), follow_symlinks = False) + ls -= files + if not ls: + break + proc.communicate() + + +if __name__ == '__main__': + parser = argparse.ArgumentParser(description="Set the times of the current git checkout to their last-changed time.") + parser.add_argument('--git-dir', metavar='GIT_DIR', help="The git dir to query (defaults to affecting the current git checkout).") + parser.add_argument('--tree', metavar='TREE-ISH', help="The tree-ish to query (defaults to the current branch).") + parser.add_argument('--prefix', metavar='PREFIX_STR', help="Prepend the PREFIX_STR to each filename we tweak.") + parser.add_argument('--quiet', '-q', action='store_true', help="Don't output the changed-file information.") + parser.add_argument('--list', '-l', action='count', help="List the files and their dates instead of changing them. Repeat for Unix Time instead of human reable.") + parser.add_argument('files', metavar='FILE', nargs='*', help="Specify a subset of checked-out files to tweak.") + args = parser.parse_args() + main() + +# vim: sw=4 et |