1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
|
#!/usr/bin/perl -w
# p4genpatch - Generate a perl patch from the repository
# Usage: $0 -h
# andreas.koenig@anima.de
use strict;
use File::Temp qw(tempdir);
use File::Compare;
use Time::Local;
sub correctmtime ($$$);
sub Usage ();
my $VERSION = '0.04';
$0 =~ s|^.*/||;
our(%OPT, @P4opt);
%OPT = ( "d" => "u", b => "//depot/perl", "D" => "diff" );
use Getopt::Long;
Getopt::Long::Configure("no_ignore_case");
GetOptions(\%OPT, "b=s", "p=s", "d=s", "D=s", "h", "v", "V") or die Usage;
print Usage and exit if $OPT{h};
print "$VERSION\n" and exit if $OPT{V};
die Usage unless @ARGV == 1;
for my $p4opt (qw(p)) {
push @P4opt, "-$p4opt $OPT{$p4opt}" if $OPT{$p4opt};
}
my $system = "p4 @P4opt describe -s @ARGV |";
open my $p4, $system or die "Could not run $system";
my @action;
while (<$p4>) {
print;
next unless m|($OPT{b})|;
my($prefix) = $1;
$prefix =~ s|/[^/]+$||; # up to the last "/" in the match is to be stripped
if (my($file,$action) = m|^\.\.\. (//depot.*)\s(\w+)$|) {
next if $action eq "delete";
push @action, [$action, $file, $prefix];
}
}
close $p4;
my $tempdir;
print "Differences ...\n";
for my $a (@action) {
$tempdir ||= tempdir( "tmp-XXXX", CLEANUP => 1 );
my($action,$file,$prefix) = @$a;
my($path,$basename,$number) = $file =~ m|\Q$prefix\E/(.+/)?([^/]+)#(\d+)|;
die "Panic: Could not parse file[$file]" unless $number;
$path = "" unless defined $path;
my($d1,$d2,$prev);
$prev = $number-1;
if ($prev==0 or $action =~ /^(add|branch)$/) {
$d1 = "/dev/null";
} elsif ($action =~ /^(edit|integrate)$/) {
$d1 = "$path$basename#$prev";
warn "==> $d1 <==\n" if $OPT{v};
my $system = "p4 @P4opt print -o $tempdir/$d1 //depot/$path$basename#$prev";
my $status = `$system`;
if ($?) {
warn "$0: system[$system] failed, status[$?]\n";
next;
}
if (my($prevch) = $status =~ / \s change \s (\d+) \s /x) {
my $oldd1 = $d1;
$d1 .= "~$prevch~";
rename "$tempdir/$oldd1", "$tempdir/$d1";
}
} else {
die "Unknown action[$action]";
}
$d2 = "$path$basename";
warn "==> $d2#$number <==\n" if $OPT{v};
my $system = "p4 @P4opt print -o $tempdir/$d2 $file";
# warn "system[$system]";
my $type = `$system`;
if ($?) {
warn "$0: `$system` failed, status[$?]\n";
next;
}
$type =~ m|^//.*\((.+)\)$| or next;
$type = $1;
if (File::Compare::compare("$tempdir/$d1", "$tempdir/$d2")) {
print "\n==== $file ($type) ====\nIndex: $path$basename\n";
unless ($type =~ /text/) {
next;
}
my @filelog = `p4 @P4opt filelog $file`;
correctmtime(\@filelog,$prev,"$tempdir/$d1");
correctmtime(\@filelog,$number,"$tempdir/$d2");
$system = "cd $tempdir && $OPT{D} -$OPT{d} '$d1' '$d2'";
system($system); # no return check because diff doesn't always return 0
}
for ("$tempdir/$d1","$tempdir/$d2") {
unlink or warn "Could not unlink $_: $!" if -f;
}
}
print "End of Patch.\n";
sub correctmtime ($$$) {
my($filelog,$nr,$file) = @_;
for my $line (@$filelog) {
my($rev,$change,$action,$date) =
$line =~ m{ ^ \.\.\. \s
\#
(\d+) # rev
\s change \s
(\d+) # change
\s (\w+) \s # action
on \s (\S+) # date
}x or next;
# warn "rev[$rev]";
next unless $rev == $nr;
my(@date) = split m|/|, $date;
$date[0] -= 1900;
$date[1]--;
my $time = timelocal(0,0,0,reverse @date);
utime $time, $time, $file;
last;
}
}
sub Usage () {
qq{Usage: $0 [OPTIONS] patchnumber
-p host:port p4 port (e.g. myhost:1666)
-d diffopt option to pass to diff(1)
-D diff diff(1) to use
-b branch(es) which branches to include (regex); everything up
to the last slash of matched portion of path is
stripped on local copy (default: //depot/perl)
-v verbose
-h print this help and exit
-V print version number and exit
Fetches all required files from the repository, puts them into a
temporary directory with sensible names and sensible modification
times and composes a patch to STDOUT using external diff command.
Requires repository access.
Examples:
perl $0 12345 | gzip -c > 12345.gz
perl $0 -dc 12345 > change-12345.patch
perl $0 -b //depot/maint-5.6/perl -v 8571 > 8571
};
}
|