Git.pm: Add support for subdirectories inside of working copies
This patch adds support for subdirectories inside of working copies; you can specify them in the constructor either as the Directory option (it will just get autodetected using rev-parse) or explicitly using the WorkingSubdir option. This makes Git->repository() do the exact same path setup and repository lookup as the Git porcelain does. This patch also introduces repo_path(), wc_path() and wc_subdir() accessor methods and wc_chdir() mutator. Signed-off-by: Petr Baudis <pasky@suse.cz> Signed-off-by: Junio C Hamano <junkio@cox.net>maint
							parent
							
								
									d43ba46807
								
							
						
					
					
						commit
						d5c7721d58
					
				
							
								
								
									
										157
									
								
								perl/Git.pm
								
								
								
								
							
							
						
						
									
										157
									
								
								perl/Git.pm
								
								
								
								
							|  | @ -69,20 +69,18 @@ means getting an instance of the Git object using the repository() constructor. | ||||||
| called as methods of the object are then executed in the context of the | called as methods of the object are then executed in the context of the | ||||||
| repository. | repository. | ||||||
|  |  | ||||||
| TODO: In the future, we might also do | Part of the "repository state" is also information about path to the attached | ||||||
|  | working copy (unless you work with a bare repository). You can also navigate | ||||||
|  | inside of the working copy using the C<wc_chdir()> method. (Note that | ||||||
|  | the repository object is self-contained and will not change working directory | ||||||
|  | of your process.) | ||||||
|  |  | ||||||
| 	my $subdir = $repo->subdir('Documentation'); | TODO: In the future, we might also do | ||||||
| 	# Gets called in the subdirectory context: |  | ||||||
| 	$subdir->command('status'); |  | ||||||
|  |  | ||||||
| 	my $remoterepo = $repo->remote_repository (Name => 'cogito', Branch => 'master'); | 	my $remoterepo = $repo->remote_repository (Name => 'cogito', Branch => 'master'); | ||||||
| 	$remoterepo ||= Git->remote_repository ('http://git.or.cz/cogito.git/'); | 	$remoterepo ||= Git->remote_repository ('http://git.or.cz/cogito.git/'); | ||||||
| 	my @refs = $remoterepo->refs(); | 	my @refs = $remoterepo->refs(); | ||||||
|  |  | ||||||
| So far, all functions just die if anything goes wrong. If you don't want that, |  | ||||||
| make appropriate provisions to catch the possible deaths. Better error recovery |  | ||||||
| mechanisms will be provided in the future. |  | ||||||
|  |  | ||||||
| Currently, the module merely wraps calls to external Git tools. In the future, | Currently, the module merely wraps calls to external Git tools. In the future, | ||||||
| it will provide a much faster way to interact with Git by linking directly | it will provide a much faster way to interact with Git by linking directly | ||||||
| to libgit. This should be completely opaque to the user, though (performance | to libgit. This should be completely opaque to the user, though (performance | ||||||
|  | @ -93,6 +91,7 @@ increate nonwithstanding). | ||||||
|  |  | ||||||
| use Carp qw(carp croak); # but croak is bad - throw instead | use Carp qw(carp croak); # but croak is bad - throw instead | ||||||
| use Error qw(:try); | use Error qw(:try); | ||||||
|  | use Cwd qw(abs_path); | ||||||
|  |  | ||||||
| require XSLoader; | require XSLoader; | ||||||
| XSLoader::load('Git', $VERSION); | XSLoader::load('Git', $VERSION); | ||||||
|  | @ -119,12 +118,17 @@ B<Repository> - Path to the Git repository. | ||||||
| B<WorkingCopy> - Path to the associated working copy; not strictly required | B<WorkingCopy> - Path to the associated working copy; not strictly required | ||||||
| as many commands will happily crunch on a bare repository. | as many commands will happily crunch on a bare repository. | ||||||
|  |  | ||||||
| B<Directory> - Path to the Git working directory in its usual setup. This | B<WorkingSubdir> - Subdirectory in the working copy to work inside. | ||||||
| is just for convenient setting of both C<Repository> and C<WorkingCopy> | Just left undefined if you do not want to limit the scope of operations. | ||||||
| at once: If the directory as a C<.git> subdirectory, C<Repository> is pointed |  | ||||||
| to the subdirectory and the directory is assumed to be the working copy. | B<Directory> - Path to the Git working directory in its usual setup. | ||||||
| If the directory does not have the subdirectory, C<WorkingCopy> is left | The C<.git> directory is searched in the directory and all the parent | ||||||
| undefined and C<Repository> is pointed to the directory itself. | directories; if found, C<WorkingCopy> is set to the directory containing | ||||||
|  | it and C<Repository> to the C<.git> directory itself. If no C<.git> | ||||||
|  | directory was found, the C<Directory> is assumed to be a bare repository, | ||||||
|  | C<Repository> is set to point at it and C<WorkingCopy> is left undefined. | ||||||
|  | If the C<$GIT_DIR> environment variable is set, things behave as expected | ||||||
|  | as well. | ||||||
|  |  | ||||||
| You should not use both C<Directory> and either of C<Repository> and | You should not use both C<Directory> and either of C<Repository> and | ||||||
| C<WorkingCopy> - the results of that are undefined. | C<WorkingCopy> - the results of that are undefined. | ||||||
|  | @ -134,7 +138,10 @@ to the constructor; it is equivalent to setting only the C<Directory> option | ||||||
| field. | field. | ||||||
|  |  | ||||||
| Calling the constructor with no options whatsoever is equivalent to | Calling the constructor with no options whatsoever is equivalent to | ||||||
| calling it with C<< Directory => '.' >>. | calling it with C<< Directory => '.' >>. In general, if you are building | ||||||
|  | a standard porcelain command, simply doing C<< Git->repository() >> should | ||||||
|  | do the right thing and setup the object to reflect exactly where the user | ||||||
|  | is right now. | ||||||
|  |  | ||||||
| =cut | =cut | ||||||
|  |  | ||||||
|  | @ -152,18 +159,59 @@ sub repository { | ||||||
| 		} else { | 		} else { | ||||||
| 			%opts = @args; | 			%opts = @args; | ||||||
| 		} | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
| 		if ($opts{Directory}) { | 	if (not defined $opts{Repository} and not defined $opts{WorkingCopy}) { | ||||||
| 			-d $opts{Directory} or throw Error::Simple("Directory not found: $!"); | 		$opts{Directory} ||= '.'; | ||||||
| 			if (-d $opts{Directory}."/.git") { | 	} | ||||||
| 				# TODO: Might make this more clever |  | ||||||
| 				$opts{WorkingCopy} = $opts{Directory}; | 	if ($opts{Directory}) { | ||||||
| 				$opts{Repository} = $opts{Directory}."/.git"; | 		-d $opts{Directory} or throw Error::Simple("Directory not found: $!"); | ||||||
| 			} else { |  | ||||||
| 				$opts{Repository} = $opts{Directory}; | 		my $search = Git->repository(WorkingCopy => $opts{Directory}); | ||||||
|  | 		my $dir; | ||||||
|  | 		try { | ||||||
|  | 			$dir = $search->command_oneline(['rev-parse', '--git-dir'], | ||||||
|  | 			                                STDERR => 0); | ||||||
|  | 		} catch Git::Error::Command with { | ||||||
|  | 			$dir = undef; | ||||||
|  | 		}; | ||||||
|  |  | ||||||
|  | 		if ($dir) { | ||||||
|  | 			$opts{Repository} = abs_path($dir); | ||||||
|  |  | ||||||
|  | 			# If --git-dir went ok, this shouldn't die either. | ||||||
|  | 			my $prefix = $search->command_oneline('rev-parse', '--show-prefix'); | ||||||
|  | 			$dir = abs_path($opts{Directory}) . '/'; | ||||||
|  | 			if ($prefix) { | ||||||
|  | 				if (substr($dir, -length($prefix)) ne $prefix) { | ||||||
|  | 					throw Error::Simple("rev-parse confused me - $dir does not have trailing $prefix"); | ||||||
|  | 				} | ||||||
|  | 				substr($dir, -length($prefix)) = ''; | ||||||
| 			} | 			} | ||||||
| 			delete $opts{Directory}; | 			$opts{WorkingCopy} = $dir; | ||||||
|  | 			$opts{WorkingSubdir} = $prefix; | ||||||
|  |  | ||||||
|  | 		} else { | ||||||
|  | 			# A bare repository? Let's see... | ||||||
|  | 			$dir = $opts{Directory}; | ||||||
|  |  | ||||||
|  | 			unless (-d "$dir/refs" and -d "$dir/objects" and -e "$dir/HEAD") { | ||||||
|  | 				# Mimick git-rev-parse --git-dir error message: | ||||||
|  | 				throw Error::Simple('fatal: Not a git repository'); | ||||||
|  | 			} | ||||||
|  | 			my $search = Git->repository(Repository => $dir); | ||||||
|  | 			try { | ||||||
|  | 				$search->command('symbolic-ref', 'HEAD'); | ||||||
|  | 			} catch Git::Error::Command with { | ||||||
|  | 				# Mimick git-rev-parse --git-dir error message: | ||||||
|  | 				throw Error::Simple('fatal: Not a git repository'); | ||||||
|  | 			} | ||||||
|  |  | ||||||
|  | 			$opts{Repository} = abs_path($dir); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
|  | 		delete $opts{Directory}; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	$self = { opts => \%opts }; | 	$self = { opts => \%opts }; | ||||||
|  | @ -256,7 +304,7 @@ sub command_oneline { | ||||||
| 	my ($fh, $ctx) = command_output_pipe(@_); | 	my ($fh, $ctx) = command_output_pipe(@_); | ||||||
|  |  | ||||||
| 	my $line = <$fh>; | 	my $line = <$fh>; | ||||||
| 	chomp $line; | 	defined $line and chomp $line; | ||||||
| 	try { | 	try { | ||||||
| 		_cmd_close($fh, $ctx); | 		_cmd_close($fh, $ctx); | ||||||
| 	} catch Git::Error::Command with { | 	} catch Git::Error::Command with { | ||||||
|  | @ -374,7 +422,7 @@ are involved. | ||||||
|  |  | ||||||
| =item exec_path () | =item exec_path () | ||||||
|  |  | ||||||
| Return path to the git sub-command executables (the same as | Return path to the Git sub-command executables (the same as | ||||||
| C<git --exec-path>). Useful mostly only internally. | C<git --exec-path>). Useful mostly only internally. | ||||||
|  |  | ||||||
| Implementation of this function is very fast; no external command calls | Implementation of this function is very fast; no external command calls | ||||||
|  | @ -385,6 +433,58 @@ are involved. | ||||||
| # Implemented in Git.xs. | # Implemented in Git.xs. | ||||||
|  |  | ||||||
|  |  | ||||||
|  | =item repo_path () | ||||||
|  |  | ||||||
|  | Return path to the git repository. Must be called on a repository instance. | ||||||
|  |  | ||||||
|  | =cut | ||||||
|  |  | ||||||
|  | sub repo_path { $_[0]->{opts}->{Repository} } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | =item wc_path () | ||||||
|  |  | ||||||
|  | Return path to the working copy. Must be called on a repository instance. | ||||||
|  |  | ||||||
|  | =cut | ||||||
|  |  | ||||||
|  | sub wc_path { $_[0]->{opts}->{WorkingCopy} } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | =item wc_subdir () | ||||||
|  |  | ||||||
|  | Return path to the subdirectory inside of a working copy. Must be called | ||||||
|  | on a repository instance. | ||||||
|  |  | ||||||
|  | =cut | ||||||
|  |  | ||||||
|  | sub wc_subdir { $_[0]->{opts}->{WorkingSubdir} ||= '' } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | =item wc_chdir ( SUBDIR ) | ||||||
|  |  | ||||||
|  | Change the working copy subdirectory to work within. The C<SUBDIR> is | ||||||
|  | relative to the working copy root directory (not the current subdirectory). | ||||||
|  | Must be called on a repository instance attached to a working copy | ||||||
|  | and the directory must exist. | ||||||
|  |  | ||||||
|  | =cut | ||||||
|  |  | ||||||
|  | sub wc_chdir { | ||||||
|  | 	my ($self, $subdir) = @_; | ||||||
|  |  | ||||||
|  | 	$self->wc_path() | ||||||
|  | 		or throw Error::Simple("bare repository"); | ||||||
|  |  | ||||||
|  | 	-d $self->wc_path().'/'.$subdir | ||||||
|  | 		or throw Error::Simple("subdir not found: $!"); | ||||||
|  | 	# Of course we will not "hold" the subdirectory so anyone | ||||||
|  | 	# can delete it now and we will never know. But at least we tried. | ||||||
|  |  | ||||||
|  | 	$self->{opts}->{WorkingSubdir} = $subdir; | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
| =item hash_object ( FILENAME [, TYPE ] ) | =item hash_object ( FILENAME [, TYPE ] ) | ||||||
|  |  | ||||||
| =item hash_object ( FILEHANDLE [, TYPE ] ) | =item hash_object ( FILEHANDLE [, TYPE ] ) | ||||||
|  | @ -584,8 +684,9 @@ sub _command_common_pipe { | ||||||
| sub _cmd_exec { | sub _cmd_exec { | ||||||
| 	my ($self, @args) = @_; | 	my ($self, @args) = @_; | ||||||
| 	if ($self) { | 	if ($self) { | ||||||
| 		$self->{opts}->{Repository} and $ENV{'GIT_DIR'} = $self->{opts}->{Repository}; | 		$self->repo_path() and $ENV{'GIT_DIR'} = $self->repo_path(); | ||||||
| 		$self->{opts}->{WorkingCopy} and chdir($self->{opts}->{WorkingCopy}); | 		$self->wc_path() and chdir($self->wc_path()); | ||||||
|  | 		$self->wc_subdir() and chdir($self->wc_subdir()); | ||||||
| 	} | 	} | ||||||
| 	_execv_git_cmd(@args); | 	_execv_git_cmd(@args); | ||||||
| 	die "exec failed: $!"; | 	die "exec failed: $!"; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	 Petr Baudis
						Petr Baudis