git-cvsserver: add mechanism for managing working tree and current directory

There are various reasons git-cvsserver needs to manipulate the current
directory, and this patch attempts to clarify and validate such changes:

1. Temporary empty working directory (with index) for certain operations
   that require an index file to work.
2. Use a temporary directory with temporary file names for doing
   merges of user's dirty sandbox state with latest changes in
   repository.
3. Coming up soon: Set up an index and either a valid or empty
   working directory when calling git-check-attr to decide
   if a file should be marked binary (-kb).

Signed-off-by: Matthew Ogilvie <mmogilvi_git@miniinfo.net>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
Matthew Ogilvie 2008-05-14 22:35:46 -06:00 committed by Junio C Hamano
parent 4b172de81b
commit 044182ef82

View File

@ -21,6 +21,7 @@ use bytes;
use Fcntl; use Fcntl;
use File::Temp qw/tempdir tempfile/; use File::Temp qw/tempdir tempfile/;
use File::Path qw/rmtree/;
use File::Basename; use File::Basename;
use Getopt::Long qw(:config require_order no_ignore_case); use Getopt::Long qw(:config require_order no_ignore_case);
@ -86,6 +87,17 @@ my $methods = {
# $state holds all the bits of information the clients sends us that could # $state holds all the bits of information the clients sends us that could
# potentially be useful when it comes to actually _doing_ something. # potentially be useful when it comes to actually _doing_ something.
my $state = { prependdir => '' }; my $state = { prependdir => '' };
# Work is for managing temporary working directory
my $work =
{
state => undef, # undef, 1 (empty), 2 (with stuff)
workDir => undef,
index => undef,
emptyDir => undef,
tmpDir => undef
};
$log->info("--------------- STARTING -----------------"); $log->info("--------------- STARTING -----------------");
my $usage = my $usage =
@ -189,6 +201,9 @@ while (<STDIN>)
$log->debug("Processing time : user=" . (times)[0] . " system=" . (times)[1]); $log->debug("Processing time : user=" . (times)[0] . " system=" . (times)[1]);
$log->info("--------------- FINISH -----------------"); $log->info("--------------- FINISH -----------------");
chdir '/';
exit 0;
# Magic catchall method. # Magic catchall method.
# This is the method that will handle all commands we haven't yet # This is the method that will handle all commands we haven't yet
# implemented. It simply sends a warning to the log file indicating a # implemented. It simply sends a warning to the log file indicating a
@ -1101,10 +1116,10 @@ sub req_update
$log->info("Updating '$filename'"); $log->info("Updating '$filename'");
my ( $filepart, $dirpart ) = filenamesplit($meta->{name},1); my ( $filepart, $dirpart ) = filenamesplit($meta->{name},1);
my $dir = tempdir( DIR => $TEMP_DIR, CLEANUP => 1 ) . "/"; my $mergeDir = setupTmpDir();
chdir $dir;
my $file_local = $filepart . ".mine"; my $file_local = $filepart . ".mine";
my $mergedFile = "$mergeDir/$file_local";
system("ln","-s",$state->{entries}{$filename}{modified_filename}, $file_local); system("ln","-s",$state->{entries}{$filename}{modified_filename}, $file_local);
my $file_old = $filepart . "." . $oldmeta->{revision}; my $file_old = $filepart . "." . $oldmeta->{revision};
transmitfile($oldmeta->{filehash}, { targetfile => $file_old }); transmitfile($oldmeta->{filehash}, { targetfile => $file_old });
@ -1115,11 +1130,13 @@ sub req_update
$log->info("Merging $file_local, $file_old, $file_new"); $log->info("Merging $file_local, $file_old, $file_new");
print "M Merging differences between 1.$oldmeta->{revision} and 1.$meta->{revision} into $filename\n"; print "M Merging differences between 1.$oldmeta->{revision} and 1.$meta->{revision} into $filename\n";
$log->debug("Temporary directory for merge is $dir"); $log->debug("Temporary directory for merge is $mergeDir");
my $return = system("git", "merge-file", $file_local, $file_old, $file_new); my $return = system("git", "merge-file", $file_local, $file_old, $file_new);
$return >>= 8; $return >>= 8;
cleanupTmpDir();
if ( $return == 0 ) if ( $return == 0 )
{ {
$log->info("Merged successfully"); $log->info("Merged successfully");
@ -1168,13 +1185,11 @@ sub req_update
# transmit file, format is single integer on a line by itself (file # transmit file, format is single integer on a line by itself (file
# size) followed by the file contents # size) followed by the file contents
# TODO : we should copy files in blocks # TODO : we should copy files in blocks
my $data = `cat $file_local`; my $data = `cat $mergedFile`;
$log->debug("File size : " . length($data)); $log->debug("File size : " . length($data));
print length($data) . "\n"; print length($data) . "\n";
print $data; print $data;
} }
chdir "/";
} }
} }
@ -1195,6 +1210,7 @@ sub req_ci
if ( $state->{method} eq 'pserver') if ( $state->{method} eq 'pserver')
{ {
print "error 1 pserver access cannot commit\n"; print "error 1 pserver access cannot commit\n";
cleanupWorkTree();
exit; exit;
} }
@ -1202,6 +1218,7 @@ sub req_ci
{ {
$log->warn("file 'index' already exists in the git repository"); $log->warn("file 'index' already exists in the git repository");
print "error 1 Index already exists in git repo\n"; print "error 1 Index already exists in git repo\n";
cleanupWorkTree();
exit; exit;
} }
@ -1209,31 +1226,20 @@ sub req_ci
my $updater = GITCVS::updater->new($state->{CVSROOT}, $state->{module}, $log); my $updater = GITCVS::updater->new($state->{CVSROOT}, $state->{module}, $log);
$updater->update(); $updater->update();
my $tmpdir = tempdir ( DIR => $TEMP_DIR );
my ( undef, $file_index ) = tempfile ( DIR => $TEMP_DIR, OPEN => 0 );
$log->info("Lockless commit start, basing commit on '$tmpdir', index file is '$file_index'");
$ENV{GIT_DIR} = $state->{CVSROOT} . "/";
$ENV{GIT_WORK_TREE} = ".";
$ENV{GIT_INDEX_FILE} = $file_index;
# Remember where the head was at the beginning. # Remember where the head was at the beginning.
my $parenthash = `git show-ref -s refs/heads/$state->{module}`; my $parenthash = `git show-ref -s refs/heads/$state->{module}`;
chomp $parenthash; chomp $parenthash;
if ($parenthash !~ /^[0-9a-f]{40}$/) { if ($parenthash !~ /^[0-9a-f]{40}$/) {
print "error 1 pserver cannot find the current HEAD of module"; print "error 1 pserver cannot find the current HEAD of module";
cleanupWorkTree();
exit; exit;
} }
chdir $tmpdir; setupWorkTree($parenthash);
# populate the temporary index $log->info("Lockless commit start, basing commit on '$work->{workDir}', index file is '$work->{index}'");
system("git-read-tree", $parenthash);
unless ($? == 0) $log->info("Created index '$work->{index}' for head $state->{module} - exit status $?");
{
die "Error running git-read-tree $state->{module} $file_index $!";
}
$log->info("Created index '$file_index' for head $state->{module} - exit status $?");
my @committedfiles = (); my @committedfiles = ();
my %oldmeta; my %oldmeta;
@ -1271,7 +1277,7 @@ sub req_ci
{ {
# fail everything if an up to date check fails # fail everything if an up to date check fails
print "error 1 Up to date check failed for $filename\n"; print "error 1 Up to date check failed for $filename\n";
chdir "/"; cleanupWorkTree();
exit; exit;
} }
@ -1313,7 +1319,7 @@ sub req_ci
{ {
print "E No files to commit\n"; print "E No files to commit\n";
print "ok\n"; print "ok\n";
chdir "/"; cleanupWorkTree();
return; return;
} }
@ -1336,7 +1342,7 @@ sub req_ci
{ {
$log->warn("Commit failed (Invalid commit hash)"); $log->warn("Commit failed (Invalid commit hash)");
print "error 1 Commit failed (unknown reason)\n"; print "error 1 Commit failed (unknown reason)\n";
chdir "/"; cleanupWorkTree();
exit; exit;
} }
@ -1348,7 +1354,7 @@ sub req_ci
{ {
$log->warn("Commit failed (update hook declined to update ref)"); $log->warn("Commit failed (update hook declined to update ref)");
print "error 1 Commit failed (update hook declined)\n"; print "error 1 Commit failed (update hook declined)\n";
chdir "/"; cleanupWorkTree();
exit; exit;
} }
} }
@ -1358,6 +1364,7 @@ sub req_ci
"refs/heads/$state->{module}", $commithash, $parenthash)) { "refs/heads/$state->{module}", $commithash, $parenthash)) {
$log->warn("update-ref for $state->{module} failed."); $log->warn("update-ref for $state->{module} failed.");
print "error 1 Cannot commit -- update first\n"; print "error 1 Cannot commit -- update first\n";
cleanupWorkTree();
exit; exit;
} }
@ -1414,7 +1421,7 @@ sub req_ci
} }
} }
chdir "/"; cleanupWorkTree();
print "ok\n"; print "ok\n";
} }
@ -1757,15 +1764,9 @@ sub req_annotate
argsfromdir($updater); argsfromdir($updater);
# we'll need a temporary checkout dir # we'll need a temporary checkout dir
my $tmpdir = tempdir ( DIR => $TEMP_DIR ); setupWorkTree();
my ( undef, $file_index ) = tempfile ( DIR => $TEMP_DIR, OPEN => 0 );
$log->info("Temp checkoutdir creation successful, basing annotate session work on '$tmpdir', index file is '$file_index'");
$ENV{GIT_DIR} = $state->{CVSROOT} . "/"; $log->info("Temp checkoutdir creation successful, basing annotate session work on '$work->{workDir}', index file is '$ENV{GIT_INDEX_FILE}'");
$ENV{GIT_WORK_TREE} = ".";
$ENV{GIT_INDEX_FILE} = $file_index;
chdir $tmpdir;
# foreach file specified on the command line ... # foreach file specified on the command line ...
foreach my $filename ( @{$state->{args}} ) foreach my $filename ( @{$state->{args}} )
@ -1789,10 +1790,10 @@ sub req_annotate
system("git-read-tree", $lastseenin); system("git-read-tree", $lastseenin);
unless ($? == 0) unless ($? == 0)
{ {
print "E error running git-read-tree $lastseenin $file_index $!\n"; print "E error running git-read-tree $lastseenin $ENV{GIT_INDEX_FILE} $!\n";
return; return;
} }
$log->info("Created index '$file_index' with commit $lastseenin - exit status $?"); $log->info("Created index '$ENV{GIT_INDEX_FILE}' with commit $lastseenin - exit status $?");
# do a checkout of the file # do a checkout of the file
system('git-checkout-index', '-f', '-u', $filename); system('git-checkout-index', '-f', '-u', $filename);
@ -1808,7 +1809,7 @@ sub req_annotate
# git-jsannotate telling us about commits we are hiding # git-jsannotate telling us about commits we are hiding
# from the client. # from the client.
my $a_hints = "$tmpdir/.annotate_hints"; my $a_hints = "$work->{workDir}/.annotate_hints";
if (!open(ANNOTATEHINTS, '>', $a_hints)) { if (!open(ANNOTATEHINTS, '>', $a_hints)) {
print "E failed to open '$a_hints' for writing: $!\n"; print "E failed to open '$a_hints' for writing: $!\n";
return; return;
@ -1862,7 +1863,7 @@ sub req_annotate
} }
# done; get out of the tempdir # done; get out of the tempdir
chdir "/"; cleanupWorkDir();
print "ok\n"; print "ok\n";
@ -2115,6 +2116,179 @@ sub filecleanup
return $filename; return $filename;
} }
sub validateGitDir
{
if( !defined($state->{CVSROOT}) )
{
print "error 1 CVSROOT not specified\n";
cleanupWorkTree();
exit;
}
if( $ENV{GIT_DIR} ne ($state->{CVSROOT} . '/') )
{
print "error 1 Internally inconsistent CVSROOT\n";
cleanupWorkTree();
exit;
}
}
# Setup working directory in a work tree with the requested version
# loaded in the index.
sub setupWorkTree
{
my ($ver) = @_;
validateGitDir();
if( ( defined($work->{state}) && $work->{state} != 1 ) ||
defined($work->{tmpDir}) )
{
$log->warn("Bad work tree state management");
print "error 1 Internal setup multiple work trees without cleanup\n";
cleanupWorkTree();
exit;
}
$work->{workDir} = tempdir ( DIR => $TEMP_DIR );
if( !defined($work->{index}) )
{
(undef, $work->{index}) = tempfile ( DIR => $TEMP_DIR, OPEN => 0 );
}
chdir $work->{workDir} or
die "Unable to chdir to $work->{workDir}\n";
$log->info("Setting up GIT_WORK_TREE as '.' in '$work->{workDir}', index file is '$work->{index}'");
$ENV{GIT_WORK_TREE} = ".";
$ENV{GIT_INDEX_FILE} = $work->{index};
$work->{state} = 2;
if($ver)
{
system("git","read-tree",$ver);
unless ($? == 0)
{
$log->warn("Error running git-read-tree");
die "Error running git-read-tree $ver in $work->{workDir} $!\n";
}
}
# else # req_annotate reads tree for each file
}
# Ensure current directory is in some kind of working directory,
# with a recent version loaded in the index.
sub ensureWorkTree
{
if( defined($work->{tmpDir}) )
{
$log->warn("Bad work tree state management [ensureWorkTree()]");
print "error 1 Internal setup multiple dirs without cleanup\n";
cleanupWorkTree();
exit;
}
if( $work->{state} )
{
return;
}
validateGitDir();
if( !defined($work->{emptyDir}) )
{
$work->{emptyDir} = tempdir ( DIR => $TEMP_DIR, OPEN => 0);
}
chdir $work->{emptyDir} or
die "Unable to chdir to $work->{emptyDir}\n";
my $ver = `git show-ref -s refs/heads/$state->{module}`;
chomp $ver;
if ($ver !~ /^[0-9a-f]{40}$/)
{
$log->warn("Error from git show-ref -s refs/head$state->{module}");
print "error 1 cannot find the current HEAD of module";
cleanupWorkTree();
exit;
}
if( !defined($work->{index}) )
{
(undef, $work->{index}) = tempfile ( DIR => $TEMP_DIR, OPEN => 0 );
}
$ENV{GIT_WORK_TREE} = ".";
$ENV{GIT_INDEX_FILE} = $work->{index};
$work->{state} = 1;
system("git","read-tree",$ver);
unless ($? == 0)
{
die "Error running git-read-tree $ver $!\n";
}
}
# Cleanup working directory that is not needed any longer.
sub cleanupWorkTree
{
if( ! $work->{state} )
{
return;
}
chdir "/" or die "Unable to chdir '/'\n";
if( defined($work->{workDir}) )
{
rmtree( $work->{workDir} );
undef $work->{workDir};
}
undef $work->{state};
}
# Setup a temporary directory (not a working tree), typically for
# merging dirty state as in req_update.
sub setupTmpDir
{
$work->{tmpDir} = tempdir ( DIR => $TEMP_DIR );
chdir $work->{tmpDir} or die "Unable to chdir $work->{tmpDir}\n";
return $work->{tmpDir};
}
# Clean up a previously setupTmpDir. Restore previous work tree if
# appropriate.
sub cleanupTmpDir
{
if ( !defined($work->{tmpDir}) )
{
$log->warn("cleanup tmpdir that has not been setup");
die "Cleanup tmpDir that has not been setup\n";
}
if( defined($work->{state}) )
{
if( $work->{state} == 1 )
{
chdir $work->{emptyDir} or
die "Unable to chdir to $work->{emptyDir}\n";
}
elsif( $work->{state} == 2 )
{
chdir $work->{workDir} or
die "Unable to chdir to $work->{emptyDir}\n";
}
else
{
$log->warn("Inconsistent work dir state");
die "Inconsistent work dir state\n";
}
}
else
{
chdir "/" or die "Unable to chdir '/'\n";
}
}
# Given a path, this function returns a string containing the kopts # Given a path, this function returns a string containing the kopts
# that should go into that path's Entries line. For example, a binary # that should go into that path's Entries line. For example, a binary
# file should get -kb. # file should get -kb.