diff options
109 files changed, 6722 insertions, 2908 deletions
| diff --git a/.gitignore b/.gitignore index f15155d1b7..eb8a1f8606 100644 --- a/.gitignore +++ b/.gitignore @@ -139,6 +139,7 @@ git-whatchanged  git-write-tree  git-core-*/?*  gitweb/gitweb.cgi +test-chmtime  test-date  test-delta  test-dump-cache-tree @@ -27,6 +27,7 @@ Nguyễn Thái Ngọc Duy <pclouds@gmail.com>  Ramsay Allan Jones <ramsay@ramsay1.demon.co.uk>  René Scharfe <rene.scharfe@lsrfire.ath.cx>  Robert Fitzsimons <robfitz@273k.net> +Sam Vilain <sam@vilain.net>  Santi Béjar <sbejar@gmail.com>  Sean Estabrooks <seanlkml@sympatico.ca>  Shawn O. Pearce <spearce@spearce.org> diff --git a/Documentation/RelNotes-1.5.0.2.txt b/Documentation/RelNotes-1.5.0.2.txt new file mode 100644 index 0000000000..b061e50ff0 --- /dev/null +++ b/Documentation/RelNotes-1.5.0.2.txt @@ -0,0 +1,65 @@ +GIT v1.5.0.2 Release Notes +========================== + +Fixes since v1.5.0.1 +-------------------- + +* Bugfixes + +  - Automated merge conflict handling when changes to symbolic +    links conflicted were completely broken.  The merge-resolve +    strategy created a regular file with conflict markers in it +    in place of the symbolic link.  The default strategy, +    merge-recursive was even more broken.  It removed the path +    that was pointed at by the symbolic link.  Both of these +    problems have been fixed. + +  - 'git diff maint master next' did not correctly give combined +    diff across three trees. + +  - 'git fast-import' portability fix for Solaris. + +  - 'git show-ref --verify' without arguments did not error out +    but segfaulted. + +  - 'git diff :tracked-file `pwd`/an-untracked-file' gave an extra +    slashes after a/ and b/. + +  - 'git format-patch' produced too long filenames if the commit +    message had too long line at the beginning. + +  - Running 'make all' and then without changing anything +    running 'make install' still rebuilt some files.  This +    was inconvenient when building as yourself and then +    installing as root (especially problematic when the source +    directory is on NFS and root is mapped to nobody). + +  - 'git-rerere' failed to deal with two unconflicted paths that +    sorted next to each other. + +  - 'git-rerere' attempted to open(2) a symlink and failed if +    there was a conflict.  Since a conflicting change to a +    symlink would not benefit from rerere anyway, the command +    now ignores conflicting changes to symlinks. + +  - 'git-repack' did not like to pass more than 64 arguments +    internally to underlying 'rev-list' logic, which made it +    impossible to repack after accumulating many (small) packs +    in the repository. + +  - 'git-diff' to review the combined diff during a conflicted +    merge were not reading the working tree version correctly +    when changes to a symbolic link conflicted.  It should have +    read the data using readlink(2) but read from the regular +    file the symbolic link pointed at. + +  - 'git-remote' did not like period in a remote's name. + +* Documentation updates + +  - added and clarified core.bare, core.legacyheaders configurations. + +  - updated "git-clone --depth" documentation. + + +* Assorted git-gui fixes. diff --git a/Documentation/RelNotes-1.5.0.txt b/Documentation/RelNotes-1.5.0.txt index 599efb8c90..daf4bdb0d7 100644 --- a/Documentation/RelNotes-1.5.0.txt +++ b/Documentation/RelNotes-1.5.0.txt @@ -448,7 +448,7 @@ Updates in v1.5.0 since v1.4.4 series   - There is a partial support for 'shallow' repositories that     keeps only recent history.  A 'shallow clone' is created by     specifying how deep that truncated history should be -   (e.g. "git clone --depth=5 git://some.where/repo.git"). +   (e.g. "git clone --depth 5 git://some.where/repo.git").     Currently a shallow repository has number of limitations: diff --git a/Documentation/config.txt b/Documentation/config.txt index 38655350f2..d2b4a05ca5 100644 --- a/Documentation/config.txt +++ b/Documentation/config.txt @@ -5,7 +5,8 @@ The git configuration file contains a number of variables that affect  the git command's behavior. `.git/config` file for each repository  is used to store the information for that repository, and  `$HOME/.gitconfig` is used to store per user information to give -fallback values for `.git/config` file. +fallback values for `.git/config` file. The file `/etc/gitconfig` +can be used to store system-wide defaults.  They can be used by both the git plumbing  and the porcelains. The variables are divided into sections, where @@ -142,6 +143,18 @@ core.preferSymlinkRefs::  	This is sometimes needed to work with old scripts that  	expect HEAD to be a symbolic link. +core.bare:: +	If true this repository is assumed to be 'bare' and has no +	working directory associated with it.  If this is the case a +	number of commands that require a working directory will be +	disabled, such as gitlink:git-add[1] or gitlink:git-merge[1]. ++ +This setting is automatically guessed by gitlink:git-clone[1] or +gitlink:git-init[1] when the repository was created.  By default a +repository that ends in "/.git" is assumed to be not bare (bare = +false), while all other repositories are assumed to be bare (bare += true). +  core.logAllRefUpdates::  	Updates to a ref <ref> is logged to the file  	"$GIT_DIR/logs/<ref>", by appending the new and old @@ -180,10 +193,17 @@ core.compression::  	slowest.  core.legacyheaders:: -	A boolean which enables the legacy object header format in case -	you want to interoperate with old clients accessing the object -	database directly (where the "http://" and "rsync://" protocols -	count as direct access). +	A boolean which +	changes the format of loose objects so that they are more +	efficient to pack and to send out of the repository over git +	native protocol, since v1.4.2.  However, loose objects +	written in the new format cannot be read by git older than +	that version; people fetching from your repository using +	older versions of git over dumb transports (e.g. http) +	will also be affected. ++ +To let git use the new loose object format, you have to +set core.legacyheaders to false.  core.packedGitWindowSize::  	Number of bytes of a pack file to map into memory in a @@ -451,6 +471,10 @@ remote.<name>.push::  	The default set of "refspec" for gitlink:git-push[1]. See  	gitlink:git-push[1]. +remote.<name>.skipDefaultUpdate:: +	If true, this remote will be skipped by default when updating +	using the remote subcommand of gitlink:git-remote[1]. +  remote.<name>.receivepack::  	The default program to execute on the remote side when pushing.  See  	option \--exec of gitlink:git-push[1]. @@ -459,6 +483,14 @@ remote.<name>.uploadpack::  	The default program to execute on the remote side when fetching.  See  	option \--exec of gitlink:git-fetch-pack[1]. +remote.<name>.tagopt:: +	Setting this value to --no-tags disables automatic tag following when fetching +	from remote <name> + +remotes.<group>:: +	The list of remotes which are fetched by "git remote update +	<group>".  See gitlink:git-remote[1]. +  repack.usedeltabaseoffset::  	Allow gitlink:git-repack[1] to create packs that uses  	delta-base offset.  Defaults to false. diff --git a/Documentation/git-clone.txt b/Documentation/git-clone.txt index 707376f22c..6d32c491a5 100644 --- a/Documentation/git-clone.txt +++ b/Documentation/git-clone.txt @@ -11,7 +11,7 @@ SYNOPSIS  [verse]  'git-clone' [--template=<template_directory>] [-l [-s]] [-q] [-n] [--bare]  	  [-o <name>] [-u <upload-pack>] [--reference <repository>] -	  [--depth=<depth>] <repository> [<directory>] +	  [--depth <depth>] <repository> [<directory>]  DESCRIPTION  ----------- @@ -96,7 +96,7 @@ OPTIONS  	if unset the templates are taken from the installation  	defined default, typically `/usr/share/git-core/templates`. ---depth=<depth>:: +--depth <depth>::  	Create a 'shallow' clone with a history truncated to the  	specified number of revs.  A shallow repository has  	number of limitations (you cannot clone or fetch from diff --git a/Documentation/git-cvsexportcommit.txt b/Documentation/git-cvsexportcommit.txt index 27d531b888..555b8234f0 100644 --- a/Documentation/git-cvsexportcommit.txt +++ b/Documentation/git-cvsexportcommit.txt @@ -8,7 +8,7 @@ git-cvsexportcommit - Export a single commit to a CVS checkout  SYNOPSIS  -------- -'git-cvsexportcommit' [-h] [-v] [-c] [-P] [-p] [-a] [-f] [-m msgprefix] [PARENTCOMMIT] COMMITID +'git-cvsexportcommit' [-h] [-v] [-c] [-P] [-p] [-a] [-d cvsroot] [-f] [-m msgprefix] [PARENTCOMMIT] COMMITID  DESCRIPTION @@ -43,6 +43,11 @@ OPTIONS  	Add authorship information. Adds Author line, and Committer (if  	different from Author) to the message. +-d:: +	Set an alternative CVSROOT to use.  This corresponds to the CVS +	-d parameter.  Usually users will not want to set this, except +	if using CVS in an asymmetric fashion. +  -f::  	Force the merge even if the files are not up to date. diff --git a/Documentation/git-remote.txt b/Documentation/git-remote.txt index a60c31a315..266faade31 100644 --- a/Documentation/git-remote.txt +++ b/Documentation/git-remote.txt @@ -13,6 +13,7 @@ SYNOPSIS  'git-remote' add <name> <url>  'git-remote' show <name>  'git-remote' prune <name> +'git-remote' update [group]  DESCRIPTION  ----------- @@ -31,6 +32,19 @@ subcommands are available to perform operations on the remotes.  Adds a remote named <name> for the repository at  <url>.  The command `git fetch <name>` can then be used to create and  update remote-tracking branches <name>/<branch>. ++ +With `-f` option, `git fetch <name>` is run immediately after +the remote information is set up. ++ +With `-t <branch>` option, instead of the default glob +refspec for the remote to track all branches under +`$GIT_DIR/remotes/<name>/`, a refspec to track only `<branch>` +is created.  You can give more than one `-t <branch>` to track +multiple branche without grabbing all branches. ++ +With `-m <master>` option, `$GIT_DIR/remotes/<name>/HEAD` is set +up to point at remote's `<master>` branch instead of whatever +branch the `HEAD` at the remote repository actually points at.  'show':: @@ -40,7 +54,17 @@ Gives some information about the remote <name>.  Deletes all stale tracking branches under <name>.  These stale branches have already been removed from the remote repository -referenced by <name>, but are still locally available in "remotes/<name>". +referenced by <name>, but are still locally available in +"remotes/<name>". + +'update':: + +Fetch updates for a named set of remotes in the repository as defined by +remotes.<group>.  If a named group is not specified on the command line, +the configuration parameter remotes.default will get used; if +remotes.default is not defined, all remotes which do not the +configuration parameter remote.<name>.skipDefaultUpdate set to true will +be updated.  (See gitlink:git-config[1]).  DISCUSSION diff --git a/Documentation/git-svn.txt b/Documentation/git-svn.txt index 6ce6a3944d..cf094ca357 100644 --- a/Documentation/git-svn.txt +++ b/Documentation/git-svn.txt @@ -13,14 +13,13 @@ DESCRIPTION  -----------  git-svn is a simple conduit for changesets between Subversion and git.  It is not to be confused with gitlink:git-svnimport[1], which is -read-only and geared towards tracking multiple branches. +read-only.  git-svn was originally designed for an individual developer who wants a  bidirectional flow of changesets between a single branch in Subversion  and an arbitrary number of branches in git.  Since its inception,  git-svn has gained the ability to track multiple branches in a manner -similar to git-svnimport; but it cannot (yet) automatically detect new -branches and tags like git-svnimport does. +similar to git-svnimport.  git-svn is especially useful when it comes to tracking repositories  not organized in the way Subversion developers recommend (trunk, @@ -31,26 +30,80 @@ COMMANDS  --  'init':: -	Creates an empty git repository with additional metadata -	directories for git-svn.  The Subversion URL must be specified -	as a command-line argument.  Optionally, the target directory -	to operate on can be specified as a second argument.  Normally -	this command initializes the current directory. +	Initializes an empty git repository with additional +	metadata directories for git-svn.  The Subversion URL +	may be specified as a command-line argument, or as full +	URL arguments to -T/-t/-b.  Optionally, the target +	directory to operate on can be specified as a second +	argument.  Normally this command initializes the current +	directory. -'fetch':: +-T<trunk_subdir>:: +--trunk=<trunk_subdir>:: +-t<tags_subdir>:: +--tags=<tags_subdir>:: +-b<branches_subdir>:: +--branches=<branches_subdir>:: +	These are optional command-line options for init.  Each of +	these flags can point to a relative repository path +	(--tags=project/tags') or a full url +	(--tags=https://foo.org/project/tags) -Fetch unfetched revisions from the Subversion URL we are -tracking.  refs/remotes/git-svn will be updated to the -latest revision. +--no-metadata:: +	Set the 'noMetadata' option in the [svn-remote] config. +--use-svm-props:: +	Set the 'useSvmProps' option in the [svn-remote] config. +--use-svnsync-props:: +	Set the 'useSvnsyncProps' option in the [svn-remote] config. +--rewrite-root=<URL>:: +	Set the 'rewriteRoot' option in the [svn-remote] config. +--username=<USER>:: +	For transports that SVN handles authentication for (http, +	https, and plain svn), specify the username.  For other +	transports (eg svn+ssh://), you must include the username in +	the URL, eg svn+ssh://foo@svn.bar.com/project -Note: You should never attempt to modify the remotes/git-svn -branch outside of git-svn.  Instead, create a branch from -remotes/git-svn and work on that branch.  Use the 'dcommit' -command (see below) to write git commits back to -remotes/git-svn. +--prefix=<prefix> +	This allows one to specify a prefix which is prepended +	to the names of remotes if trunk/branches/tags are +	specified.  The prefix does not automatically include a +	trailing slash, so be sure you include one in the +	argument if that is what you want.  This is useful if +	you wish to track multiple projects that share a common +	repository. + +'fetch':: -See '<<fetch-args,Additional Fetch Arguments>>' if you are interested in -manually joining branches on commit. +	Fetch unfetched revisions from the Subversion remote we are +	tracking.  The name of the [svn-remote "..."] section in the +	.git/config file may be specified as an optional command-line +	argument. + +'clone':: +	Runs 'init' and 'fetch'.  It will automatically create a +	directory based on the basename of the URL passed to it; +	or if a second argument is passed; it will create a directory +	and work within that.  It accepts all arguments that the +	'init' and 'fetch' commands accept; with the exception of +	'--fetch-all'.   After a repository is cloned, the 'fetch' +	command will be able to update revisions without affecting +	the working tree; and the 'rebase' command will be able +	to update the working tree with the latest changes. + +'rebase':: +	This fetches revisions from the SVN parent of the current HEAD +	and rebases the current (uncommitted to SVN) work against it. + +	This works similarly to 'svn update' or 'git-pull' except that +	it preserves linear history with 'git-rebase' instead of +	'git-merge' for ease of dcommit-ing with git-svn. + +	This accepts all options that 'git-svn fetch' and 'git-rebase' +	accepts.  However '--fetch-all' only fetches from the current +	[svn-remote], and not all [svn-remote] definitions. + +	Like 'git-rebase'; this requires that the working tree be clean +	and have no uncommitted changes.  'dcommit'::  	Commit each diff from a specified head directly to the SVN @@ -96,16 +149,6 @@ manually joining branches on commit.  	commit.  All merging is assumed to have taken place  	independently of git-svn functions. -'rebuild':: -	Not a part of daily usage, but this is a useful command if -	you've just cloned a repository (using gitlink:git-clone[1]) that was -	tracked with git-svn.  Unfortunately, git-clone does not clone -	git-svn metadata and the svn working tree that git-svn uses for -	its operations.  This rebuilds the metadata so git-svn can -	resume fetch operations.  A Subversion URL may be optionally -	specified at the command-line if the directory/repository you're -	tracking has moved or changed protocols. -  'show-ignore'::  	Recursively finds and lists the svn:ignore property on  	directories.  The output is suitable for appending to @@ -122,53 +165,13 @@ manually joining branches on commit.  	repository (that has been init-ed with git-svn).  	The -r<revision> option is required for this. -'graft-branches':: -	This command attempts to detect merges/branches from already -	imported history.  Techniques used currently include regexes, -	file copies, and tree-matches).  This command generates (or -	modifies) the $GIT_DIR/info/grafts file.  This command is -	considered experimental, and inherently flawed because -	merge-tracking in SVN is inherently flawed and inconsistent -	across different repositories. - -'multi-init':: -	This command supports git-svnimport-like command-line syntax for -	importing repositories that are laid out as recommended by the -	SVN folks.  This is a bit more tolerant than the git-svnimport -	command-line syntax and doesn't require the user to figure out -	where the repository URL ends and where the repository path -	begins. - --T<trunk_subdir>:: ---trunk=<trunk_subdir>:: --t<tags_subdir>:: ---tags=<tags_subdir>:: --b<branches_subdir>:: ---branches=<branches_subdir>:: -	These are the command-line options for multi-init.  Each of -	these flags can point to a relative repository path -	(--tags=project/tags') or a full url -	(--tags=https://foo.org/project/tags) - ---prefix=<prefix> -	This allows one to specify a prefix which is prepended to the -	names of remotes.  The prefix does not automatically include a -	trailing slash, so be sure you include one in the argument if -	that is what you want.  This is useful if you wish to track -	multiple projects that share a common repository. - -'multi-fetch':: -	This runs fetch on all known SVN branches we're tracking.  This -	will NOT discover new branches (unlike git-svnimport), so -	multi-init will need to be re-run (it's idempotent). -  --  OPTIONS  -------  -- ---shared:: +--shared[={false|true|umask|group|all|world|everybody}]::  --template=<template_directory>::  	Only used with the 'init' command.  	These are passed directly to gitlink:git-init[1]. @@ -176,14 +179,15 @@ OPTIONS  -r <ARG>::  --revision <ARG>:: -Only used with the 'fetch' command. +Used with the 'fetch' command. -Takes any valid -r<argument> svn would accept and passes it -directly to svn. -r<ARG1>:<ARG2> ranges and "{" DATE "}" syntax -is also supported.  This is passed directly to svn, see svn -documentation for more details. +This allows revision ranges for partial/cauterized history +to be supported.  $NUMBER, $NUMBER1:$NUMBER2 (numeric ranges), +$NUMBER:HEAD, and BASE:$NUMBER are all supported. -This can allow you to make partial mirrors when running fetch. +This can allow you to make partial mirrors when running fetch; +but is generally not recommended because history will be skipped +and lost.  -::  --stdin:: @@ -270,7 +274,7 @@ config key: svn.repackflags  -s<strategy>::  --strategy=<strategy>:: -These are only used with the 'dcommit' command. +These are only used with the 'dcommit' and 'rebase' commands.  Passed directly to git-rebase when using 'dcommit' if a  'git-reset' cannot be used (see dcommit). @@ -289,75 +293,79 @@ ADVANCED OPTIONS  ----------------  -- --b<refname>:: ---branch <refname>:: -Used with 'fetch', 'dcommit' or 'set-tree'. - -This can be used to join arbitrary git branches to remotes/git-svn -on new commits where the tree object is equivalent. - -When used with different GIT_SVN_ID values, tags and branches in -SVN can be tracked this way, as can some merges where the heads -end up having completely equivalent content.  This can even be -used to track branches across multiple SVN _repositories_. - -This option may be specified multiple times, once for each -branch. - -config key: svn.branch -  -i<GIT_SVN_ID>::  --id <GIT_SVN_ID>:: -This sets GIT_SVN_ID (instead of using the environment).  See the -section on -'<<tracking-multiple-repos,Tracking Multiple Repositories or Branches>>' -for more information on using GIT_SVN_ID. +This sets GIT_SVN_ID (instead of using the environment).  This +allows the user to override the default refname to fetch from +when tracking a single URL.  The 'log' and 'dcommit' commands +no longer require this switch as an argument. + +-R<remote name>:: +--svn-remote <remote name>:: +	Specify the [svn-remote "<remote name>"] section to use, +	this allows SVN multiple repositories to be tracked. +	Default: "svn"  --follow-parent::  	This is especially helpful when we're tracking a directory  	that has been moved around within the repository, or if we  	started tracking a branch and never tracked the trunk it was -	descended from. +	descended from. This feature is enabled by default, use +	--no-follow-parent to disable it.  config key: svn.followparent ---no-metadata:: -	This gets rid of the git-svn-id: lines at the end of every commit. - -	With this, you lose the ability to use the rebuild command.  If -	you ever lose your .git/svn/git-svn/.rev_db file, you won't be -	able to fetch again, either.  This is fine for one-shot imports. - -	The 'git-svn log' command will not work on repositories using this, -	either. - -config key: svn.nometadata -  -- - -COMPATIBILITY OPTIONS ---------------------- +CONFIG FILE-ONLY OPTIONS +------------------------  -- ---upgrade:: -Only used with the 'rebuild' command. - -Run this if you used an old version of git-svn that used -"git-svn-HEAD" instead of "remotes/git-svn" as the branch -for tracking the remote. - ---ignore-nodate:: -Only used with the 'fetch' command. - -By default git-svn will crash if it tries to import a revision -from SVN which has '(no date)' listed as the date of the revision. -This is repository corruption on SVN's part, plain and simple. -But sometimes you really need those revisions anyway. +svn.noMetadata:: +svn-remote.<name>.noMetadata:: +	This gets rid of the git-svn-id: lines at the end of every commit. -If supplied git-svn will convert '(no date)' entries to the UNIX -epoch (midnight on Jan. 1, 1970).  Yes, that's probably very wrong. -SVN was very wrong. +	If you lose your .git/svn/git-svn/.rev_db file, git-svn will not +	be able to rebuild it and you won't be able to fetch again, +	either.  This is fine for one-shot imports. + +	The 'git-svn log' command will not work on repositories using +	this, either.  Using this conflicts with the 'useSvmProps' +	option for (hopefully) obvious reasons. + +svn.useSvmProps:: +svn-remote.<name>.useSvmProps:: +	This allows git-svn to re-map repository URLs and UUIDs from +	mirrors created using SVN::Mirror (or svk) for metadata. + +	If an SVN revision has a property, "svm:headrev", it is likely +	that the revision was created by SVN::Mirror (also used by SVK). +	The property contains a repository UUID and a revision.  We want +	to make it look like we are mirroring the original URL, so +	introduce a helper function that returns the original identity +	URL and UUID, and use it when generating metadata in commit +	messages. + +svn.useSvnsyncProps:: +svn-remote.<name>.useSvnsyncprops:: +	Similar to the useSvmProps option; this is for users +	of the svnsync(1) command distributed with SVN 1.4.x and +	later. + +svn-remote.<name>.rewriteRoot:: +	This allows users to create repositories from alternate +	URLs.  For example, an administrator could run git-svn on the +	server locally (accessing via file://) but wish to distribute +	the repository with a public http:// or svn:// URL in the +	metadata so users of it will see the public URL. + +Since the noMetadata, rewriteRoot, useSvnsyncProps and useSvmProps +options all affect the metadata generated and used by git-svn; they +*must* be set in the configuration file before any history is imported +and these settings should never be changed once they are set. + +Additionally, only one of these four options can be used per-svn-remote +section because they affect the 'git-svn-id:' metadata line.  -- @@ -367,43 +375,37 @@ Basic Examples  Tracking and contributing to a the trunk of a Subversion-managed project:  ------------------------------------------------------------------------ -# Initialize a repo (like git init): -	git-svn init http://svn.foo.org/project/trunk -# Fetch remote revisions: -	git-svn fetch -# Create your own branch to hack on: -	git checkout -b my-branch remotes/git-svn -# Do some work, and then commit your new changes to SVN, as well as -# automatically updating your working HEAD: +# Clone a repo (like git clone): +	git-svn clone http://svn.foo.org/project/trunk +# Enter the newly cloned directory: +	cd trunk +# You should be on master branch, double-check with git-branch +	git branch +# Do some work and commit locally to git: +	git commit ... +# Something is committed to SVN, rebase your local changes against the +# latest changes in SVN: +	git-svn rebase +# Now commit your changes (that were committed previously using git) to SVN, +# as well as automatically updating your working HEAD:  	git-svn dcommit -# Something is committed to SVN, rebase the latest into your branch: -	git-svn fetch && git rebase remotes/git-svn  # Append svn:ignore settings to the default git exclude file:  	git-svn show-ignore >> .git/info/exclude  ------------------------------------------------------------------------  Tracking and contributing to an entire Subversion-managed project  (complete with a trunk, tags and branches): -See also: -'<<tracking-multiple-repos,Tracking Multiple Repositories or Branches>>'  ------------------------------------------------------------------------ -# Initialize a repo (like git init): -	git-svn multi-init http://svn.foo.org/project \ -		-T trunk -b branches -t tags -# Fetch remote revisions: -	git-svn multi-fetch -# Create your own branch of trunk to hack on: -	git checkout -b my-trunk remotes/trunk -# Do some work, and then commit your new changes to SVN, as well as -# automatically updating your working HEAD: -	git-svn dcommit -i trunk -# Something has been committed to trunk, rebase the latest into your branch: -	git-svn multi-fetch && git rebase remotes/trunk -# Append svn:ignore settings of trunk to the default git exclude file: -	git-svn show-ignore -i trunk >> .git/info/exclude -# Check for new branches and tags (no arguments are needed): -	git-svn multi-init +# Clone a repo (like git clone): +	git-svn clone http://svn.foo.org/project -T trunk -b branches -t tags +# View all branches and tags you have cloned: +	git branch -r +# Reset your master to trunk (or any other branch, replacing 'trunk' +# with the appropriate name): +	git reset --hard remotes/trunk +# You may only dcommit to one branch/tag/trunk at a time.  The usage +# of dcommit/rebase/show-ignore should be teh same as above.  ------------------------------------------------------------------------  REBASE VS. PULL/MERGE @@ -416,7 +418,7 @@ pulled or merged from.  This is because the author favored  If you use 'git-svn set-tree A..B' to commit several diffs and you do  not have the latest remotes/git-svn merged into my-branch, you should -use 'git rebase' to update your work branch instead of 'git pull' or +use 'git-svn rebase' to update your work branch instead of 'git pull' or  'git merge'.  'pull/merge' can cause non-linear history to be flattened  when committing into SVN, which can lead to merge commits reversing  previous commits in SVN. @@ -426,67 +428,49 @@ DESIGN PHILOSOPHY  Merge tracking in Subversion is lacking and doing branched development  with Subversion is cumbersome as a result.  git-svn does not do  automated merge/branch tracking by default and leaves it entirely up to -the user on the git side. - -[[tracking-multiple-repos]] -TRACKING MULTIPLE REPOSITORIES OR BRANCHES ------------------------------------------- -Because git-svn does not care about relationships between different -branches or directories in a Subversion repository, git-svn has a simple -hack to allow it to track an arbitrary number of related _or_ unrelated -SVN repositories via one git repository.  Simply use the --id/-i flag or -set the GIT_SVN_ID environment variable to a name other other than -"git-svn" (the default) and git-svn will ignore the contents of the -$GIT_DIR/svn/git-svn directory and instead do all of its work in -$GIT_DIR/svn/$GIT_SVN_ID for that invocation.  The interface branch will -be remotes/$GIT_SVN_ID, instead of remotes/git-svn.  Any -remotes/$GIT_SVN_ID branch should never be modified by the user outside -of git-svn commands. - -[[fetch-args]] -ADDITIONAL FETCH ARGUMENTS --------------------------- -This is for advanced users, most users should ignore this section. - -Unfetched SVN revisions may be imported as children of existing commits -by specifying additional arguments to 'fetch'.  Additional parents may -optionally be specified in the form of sha1 hex sums at the -command-line.  Unfetched SVN revisions may also be tied to particular -git commits with the following syntax: - ------------------------------------------------- -	svn_revision_number=git_commit_sha1 ------------------------------------------------- - -This allows you to tie unfetched SVN revision 375 to your current HEAD: - ------------------------------------------------- -	git-svn fetch 375=$(git-rev-parse HEAD) ------------------------------------------------- - -If you're tracking a directory that has moved, or otherwise been -branched or tagged off of another directory in the repository and you -care about the full history of the project, then you can use -the --follow-parent option. - ------------------------------------------------- -	git-svn fetch --follow-parent ------------------------------------------------- +the user on the git side.  git-svn does however follow copy +history of the directory that it is tracking, however (much like +how 'svn log' works).  BUGS  ---- -We ignore all SVN properties except svn:executable.  Too difficult to -map them since we rely heavily on git write-tree being _exactly_ the -same on both the SVN and git working trees and I prefer not to clutter -working trees with metadata files. +We ignore all SVN properties except svn:executable.  Any unhandled +properties are logged to $GIT_DIR/svn/<refname>/unhandled.log  Renamed and copied directories are not detected by git and hence not  tracked when committing to SVN.  I do not plan on adding support for  this as it's quite difficult and time-consuming to get working for all -the possible corner cases (git doesn't do it, either).  Renamed and -copied files are fully supported if they're similar enough for git to -detect them. +the possible corner cases (git doesn't do it, either).  Committing +renamed and copied files are fully supported if they're similar enough +for git to detect them. + +CONFIGURATION +------------- + +git-svn stores [svn-remote] configuration information in the +repository .git/config file.  It is similar the core git +[remote] sections except 'fetch' keys do not accept glob +arguments; but they are instead handled by the 'branches' +and 'tags' keys.  Since some SVN repositories are oddly +configured with multiple projects glob expansions such those +listed below are allowed: + +------------------------------------------------------------------------ +[svn-remote "project-a"] +	url = http://server.org/svn +	branches = branches/*/project-a:refs/remotes/project-a/branches/* +	tags = tags/*/project-a:refs/remotes/project-a/tags/* +	trunk = trunk/project-a:refs/remotes/project-a/trunk +------------------------------------------------------------------------ + +Keep in mind that the '*' (asterisk) wildcard of the local ref +(left of the ':') *must* be the farthest right path component; +however the remote wildcard may be anywhere as long as it's own +independent path componet (surrounded by '/' or EOL).   This +type of configuration is not automatically created by 'init' and +should be manually entered with a text-editor or using +gitlink:git-config[1]  SEE ALSO  -------- diff --git a/Documentation/git.txt b/Documentation/git.txt index 3d8be5931c..9a74747989 100644 --- a/Documentation/git.txt +++ b/Documentation/git.txt @@ -35,6 +35,12 @@ ifdef::stalenotes[]  You are reading the documentation for the latest version of git.  Documentation for older releases are available here: +* link:v1.5.0.2/git.html[documentation for release 1.5.0.2] + +* link:v1.5.0.2/RelNotes-1.5.0.2.txt[release notes for 1.5.0.2] + +* link:v1.5.0.1/RelNotes-1.5.0.1.txt[release notes for 1.5.0.1] +  * link:v1.5.0/git.html[documentation for release 1.5.0]  * link:v1.5.0/RelNotes-1.5.0.txt[release notes for 1.5.0] @@ -28,6 +28,10 @@ all::  #  # Define NO_STRLCPY if you don't have strlcpy.  # +# Define NO_STRTOUMAX if you don't have strtoumax in the C library. +# If your compiler also does not support long long or does not have +# strtoull, define NO_STRTOULL. +#  # Define NO_SETENV if you don't have setenv in the C library.  #  # Define NO_SYMLINK_HEAD if you never want .git/HEAD to be a symbolic link. @@ -124,6 +128,7 @@ prefix = $(HOME)  bindir = $(prefix)/bin  gitexecdir = $(bindir)  template_dir = $(prefix)/share/git-core/templates/ +ETC_GITCONFIG = $(prefix)/etc/gitconfig  # DESTDIR=  # default configuration for gitweb @@ -262,7 +267,8 @@ LIB_OBJS = \  	revision.o pager.o tree-walk.o xdiff-interface.o \  	write_or_die.o trace.o list-objects.o grep.o \  	alloc.o merge-file.o path-list.o help.o unpack-trees.o $(DIFF_OBJS) \ -	color.o wt-status.o archive-zip.o archive-tar.o shallow.o utf8.o +	color.o wt-status.o archive-zip.o archive-tar.o shallow.o utf8.o \ +	convert.o  BUILTIN_OBJS = \  	builtin-add.o \ @@ -353,11 +359,13 @@ ifeq ($(uname_S),SunOS)  		NO_UNSETENV = YesPlease  		NO_SETENV = YesPlease  		NO_C99_FORMAT = YesPlease +		NO_STRTOUMAX = YesPlease  	endif  	ifeq ($(uname_R),5.9)  		NO_UNSETENV = YesPlease  		NO_SETENV = YesPlease  		NO_C99_FORMAT = YesPlease +		NO_STRTOUMAX = YesPlease  	endif  	INSTALL = ginstall  	TAR = gtar @@ -517,6 +525,13 @@ ifdef NO_STRLCPY  	COMPAT_CFLAGS += -DNO_STRLCPY  	COMPAT_OBJS += compat/strlcpy.o  endif +ifdef NO_STRTOUMAX +	COMPAT_CFLAGS += -DNO_STRTOUMAX +	COMPAT_OBJS += compat/strtoumax.o +endif +ifdef NO_STRTOULL +	COMPAT_CFLAGS += -DNO_STRTOULL +endif  ifdef NO_SETENV  	COMPAT_CFLAGS += -DNO_SETENV  	COMPAT_OBJS += compat/setenv.o @@ -584,6 +599,7 @@ endif  # Shell quote (do not use $(call) to accommodate ancient setups);  SHA1_HEADER_SQ = $(subst ','\'',$(SHA1_HEADER)) +ETC_GITCONFIG_SQ = $(subst ','\'',$(ETC_GITCONFIG))  DESTDIR_SQ = $(subst ','\'',$(DESTDIR))  bindir_SQ = $(subst ','\'',$(bindir)) @@ -596,7 +612,8 @@ PERL_PATH_SQ = $(subst ','\'',$(PERL_PATH))  LIBS = $(GITLIBS) $(EXTLIBS) -BASIC_CFLAGS += -DSHA1_HEADER='$(SHA1_HEADER_SQ)' $(COMPAT_CFLAGS) +BASIC_CFLAGS += -DSHA1_HEADER='$(SHA1_HEADER_SQ)' \ +	-DETC_GITCONFIG='"$(ETC_GITCONFIG_SQ)"' $(COMPAT_CFLAGS)  LIB_OBJS += $(COMPAT_OBJS)  ALL_CFLAGS += $(BASIC_CFLAGS) @@ -812,7 +829,7 @@ GIT-CFLAGS: .FORCE-GIT-CFLAGS  export NO_SVN_TESTS -test: all +test: all test-chmtime$X  	$(MAKE) -C t/ all  test-date$X: test-date.c date.o ctype.o @@ -827,6 +844,9 @@ test-dump-cache-tree$X: dump-cache-tree.o $(GITLIBS)  test-sha1$X: test-sha1.o $(GITLIBS)  	$(CC) $(ALL_CFLAGS) -o $@ $(ALL_LDFLAGS) $(filter %.o,$^) $(LIBS) +test-chmtime$X: test-chmtime.c +	$(CC) $(ALL_CFLAGS) -o $@ $(ALL_LDFLAGS) $< +  check-sha1:: test-sha1$X  	./test-sha1.sh @@ -882,7 +902,8 @@ dist: git.spec git-archive  	$(TAR) rf $(GIT_TARNAME).tar \  		$(GIT_TARNAME)/git.spec \  		$(GIT_TARNAME)/version \ -		$(GIT_TARNAME)/git-gui/version +		$(GIT_TARNAME)/git-gui/version \ +		$(GIT_TARNAME)/git-gui/credits  	@rm -rf $(GIT_TARNAME)  	gzip -f -9 $(GIT_TARNAME).tar diff --git a/builtin-apply.c b/builtin-apply.c index abe3538715..2dde34186a 100644 --- a/builtin-apply.c +++ b/builtin-apply.c @@ -28,6 +28,7 @@ static int newfd = -1;  static int unidiff_zero;  static int p_value = 1; +static int p_value_known;  static int check_index;  static int write_index;  static int cached; @@ -144,6 +145,7 @@ struct patch {  	unsigned long deflate_origlen;  	int lines_added, lines_deleted;  	int score; +	unsigned int is_toplevel_relative:1;  	unsigned int inaccurate_eof:1;  	unsigned int is_binary:1;  	unsigned int is_copy:1; @@ -238,7 +240,7 @@ static int name_terminate(const char *name, int namelen, int c, int terminate)  	return 1;  } -static char * find_name(const char *line, char *def, int p_value, int terminate) +static char *find_name(const char *line, char *def, int p_value, int terminate)  {  	int len;  	const char *start = line; @@ -311,11 +313,54 @@ static char * find_name(const char *line, char *def, int p_value, int terminate)  	return name;  } +static int count_slashes(const char *cp) +{ +	int cnt = 0; +	char ch; + +	while ((ch = *cp++)) +		if (ch == '/') +			cnt++; +	return cnt; +} + +/* + * Given the string after "--- " or "+++ ", guess the appropriate + * p_value for the given patch. + */ +static int guess_p_value(const char *nameline) +{ +	char *name, *cp; +	int val = -1; + +	if (is_dev_null(nameline)) +		return -1; +	name = find_name(nameline, NULL, 0, TERM_SPACE | TERM_TAB); +	if (!name) +		return -1; +	cp = strchr(name, '/'); +	if (!cp) +		val = 0; +	else if (prefix) { +		/* +		 * Does it begin with "a/$our-prefix" and such?  Then this is +		 * very likely to apply to our directory. +		 */ +		if (!strncmp(name, prefix, prefix_length)) +			val = count_slashes(prefix); +		else { +			cp++; +			if (!strncmp(cp, prefix, prefix_length)) +				val = count_slashes(prefix) + 1; +		} +	} +	free(name); +	return val; +} +  /*   * Get the name etc info from the --/+++ lines of a traditional patch header   * - * NOTE! This hardcodes "-p1" behaviour in filename detection. - *   * FIXME! The end-of-filename heuristics are kind of screwy. For existing   * files, we can happily check the index for a match, but for creating a   * new file we should try to match whatever "patch" does. I have no idea. @@ -326,6 +371,16 @@ static void parse_traditional_patch(const char *first, const char *second, struc  	first += 4;	/* skip "--- " */  	second += 4;	/* skip "+++ " */ +	if (!p_value_known) { +		int p, q; +		p = guess_p_value(first); +		q = guess_p_value(second); +		if (p < 0) p = q; +		if (0 <= p && p == q) { +			p_value = p; +			p_value_known = 1; +		} +	}  	if (is_dev_null(first)) {  		patch->is_new = 1;  		patch->is_delete = 0; @@ -787,6 +842,7 @@ static int find_header(char *line, unsigned long size, int *hdrsize, struct patc  {  	unsigned long offset, len; +	patch->is_toplevel_relative = 0;  	patch->is_rename = patch->is_copy = 0;  	patch->is_new = patch->is_delete = -1;  	patch->old_mode = patch->new_mode = 0; @@ -831,6 +887,7 @@ static int find_header(char *line, unsigned long size, int *hdrsize, struct patc  					die("git diff header lacks filename information (line %d)", linenr);  				patch->old_name = patch->new_name = patch->def_name;  			} +			patch->is_toplevel_relative = 1;  			*hdrsize = git_hdr_len;  			return offset;  		} @@ -1129,11 +1186,11 @@ static struct fragment *parse_binary_hunk(char **buf_p,  	*status_p = 0; -	if (!strncmp(buffer, "delta ", 6)) { +	if (!prefixcmp(buffer, "delta ")) {  		patch_method = BINARY_DELTA_DEFLATED;  		origlen = strtoul(buffer + 6, NULL, 10);  	} -	else if (!strncmp(buffer, "literal ", 8)) { +	else if (!prefixcmp(buffer, "literal ")) {  		patch_method = BINARY_LITERAL_DEFLATED;  		origlen = strtoul(buffer + 8, NULL, 10);  	} @@ -1393,28 +1450,39 @@ static void show_stats(struct patch *patch)  	free(qname);  } -static int read_old_data(struct stat *st, const char *path, void *buf, unsigned long size) +static int read_old_data(struct stat *st, const char *path, char **buf_p, unsigned long *alloc_p, unsigned long *size_p)  {  	int fd;  	unsigned long got; +	unsigned long nsize; +	char *nbuf; +	unsigned long size = *size_p; +	char *buf = *buf_p;  	switch (st->st_mode & S_IFMT) {  	case S_IFLNK: -		return readlink(path, buf, size); +		return readlink(path, buf, size) != size;  	case S_IFREG:  		fd = open(path, O_RDONLY);  		if (fd < 0)  			return error("unable to open %s", path);  		got = 0;  		for (;;) { -			int ret = xread(fd, (char *) buf + got, size - got); +			int ret = xread(fd, buf + got, size - got);  			if (ret <= 0)  				break;  			got += ret;  		}  		close(fd); -		return got; - +		nsize = got; +		nbuf = buf; +		if (convert_to_git(path, &nbuf, &nsize)) { +			free(buf); +			*buf_p = nbuf; +			*alloc_p = nsize; +			*size_p = nsize; +		} +		return got != size;  	default:  		return -1;  	} @@ -1655,6 +1723,8 @@ static int apply_one_fragment(struct buffer_desc *desc, struct fragment *frag, i  			/* Ignore it, we already handled it */  			break;  		default: +			if (apply_verbosely) +				error("invalid start of line: '%c'", first);  			return -1;  		}  		patch += len; @@ -1752,6 +1822,9 @@ static int apply_one_fragment(struct buffer_desc *desc, struct fragment *frag, i  		}  	} +	if (offset && apply_verbosely) +		error("while searching for:\n%.*s", oldsize, oldlines); +  	free(old);  	free(new);  	return offset; @@ -1910,7 +1983,7 @@ static int apply_data(struct patch *patch, struct stat *st, struct cache_entry *  		size = st->st_size;  		alloc = size + 8192;  		buf = xmalloc(alloc); -		if (read_old_data(st, patch->old_name, buf, alloc) != size) +		if (read_old_data(st, patch->old_name, &buf, &alloc, &size))  			return error("read of %s failed", patch->old_name);  	} @@ -2232,7 +2305,7 @@ static void patch_stats(struct patch *patch)  	}  } -static void remove_file(struct patch *patch) +static void remove_file(struct patch *patch, int rmdir_empty)  {  	if (write_index) {  		if (remove_file_from_cache(patch->old_name) < 0) @@ -2240,7 +2313,7 @@ static void remove_file(struct patch *patch)  		cache_tree_invalidate_path(active_cache_tree, patch->old_name);  	}  	if (!cached) { -		if (!unlink(patch->old_name)) { +		if (!unlink(patch->old_name) && rmdir_empty) {  			char *name = xstrdup(patch->old_name);  			char *end = strrchr(name, '/');  			while (end) { @@ -2282,12 +2355,22 @@ static void add_index_file(const char *path, unsigned mode, void *buf, unsigned  static int try_create_file(const char *path, unsigned int mode, const char *buf, unsigned long size)  {  	int fd; +	char *nbuf; +	unsigned long nsize;  	if (S_ISLNK(mode))  		/* Although buf:size is counted string, it also is NUL  		 * terminated.  		 */  		return symlink(buf, path); +	nsize = size; +	nbuf = (char *) buf; +	if (convert_to_working_tree(path, &nbuf, &nsize)) { +		free((char *) buf); +		buf = nbuf; +		size = nsize; +	} +  	fd = open(path, O_CREAT | O_EXCL | O_WRONLY, (mode & 0100) ? 0777 : 0666);  	if (fd < 0)  		return -1; @@ -2373,7 +2456,7 @@ static void write_out_one_result(struct patch *patch, int phase)  {  	if (patch->is_delete > 0) {  		if (phase == 0) -			remove_file(patch); +			remove_file(patch, 1);  		return;  	}  	if (patch->is_new > 0 || patch->is_copy) { @@ -2386,7 +2469,7 @@ static void write_out_one_result(struct patch *patch, int phase)  	 * thing: remove the old, write the new  	 */  	if (phase == 0) -		remove_file(patch); +		remove_file(patch, 0);  	if (phase == 1)  		create_file(patch);  } @@ -2508,6 +2591,32 @@ static int use_patch(struct patch *p)  	return 1;  } +static void prefix_one(char **name) +{ +	char *old_name = *name; +	if (!old_name) +		return; +	*name = xstrdup(prefix_filename(prefix, prefix_length, *name)); +	free(old_name); +} + +static void prefix_patches(struct patch *p) +{ +	if (!prefix || p->is_toplevel_relative) +		return; +	for ( ; p; p = p->next) { +		if (p->new_name == p->old_name) { +			char *prefixed = p->new_name; +			prefix_one(&prefixed); +			p->new_name = p->old_name = prefixed; +		} +		else { +			prefix_one(&p->new_name); +			prefix_one(&p->old_name); +		} +	} +} +  static int apply_patch(int fd, const char *filename, int inaccurate_eof)  {  	unsigned long offset, size; @@ -2530,11 +2639,14 @@ static int apply_patch(int fd, const char *filename, int inaccurate_eof)  			break;  		if (apply_in_reverse)  			reverse_patches(patch); +		if (prefix) +			prefix_patches(patch);  		if (use_patch(patch)) {  			patch_stats(patch);  			*listp = patch;  			listp = &patch->next; -		} else { +		} +		else {  			/* perhaps free it a bit better? */  			free(patch);  			skipped_patch++; @@ -2595,9 +2707,16 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  	int read_stdin = 1;  	int inaccurate_eof = 0;  	int errs = 0; +	int is_not_gitdir = 0;  	const char *whitespace_option = NULL; +	prefix = setup_git_directory_gently(&is_not_gitdir); +	prefix_length = prefix ? strlen(prefix) : 0; +	git_config(git_apply_config); +	if (apply_default_whitespace) +		parse_whitespace_option(apply_default_whitespace); +  	for (i = 1; i < argc; i++) {  		const char *arg = argv[i];  		char *end; @@ -2608,15 +2727,16 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  			read_stdin = 0;  			continue;  		} -		if (!strncmp(arg, "--exclude=", 10)) { +		if (!prefixcmp(arg, "--exclude=")) {  			struct excludes *x = xmalloc(sizeof(*x));  			x->path = arg + 10;  			x->next = excludes;  			excludes = x;  			continue;  		} -		if (!strncmp(arg, "-p", 2)) { +		if (!prefixcmp(arg, "-p")) {  			p_value = atoi(arg + 2); +			p_value_known = 1;  			continue;  		}  		if (!strcmp(arg, "--no-add")) { @@ -2648,10 +2768,14 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  			continue;  		}  		if (!strcmp(arg, "--index")) { +			if (is_not_gitdir) +				die("--index outside a repository");  			check_index = 1;  			continue;  		}  		if (!strcmp(arg, "--cached")) { +			if (is_not_gitdir) +				die("--cached outside a repository");  			check_index = 1;  			cached = 1;  			continue; @@ -2669,13 +2793,13 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  			line_termination = 0;  			continue;  		} -		if (!strncmp(arg, "-C", 2)) { +		if (!prefixcmp(arg, "-C")) {  			p_context = strtoul(arg + 2, &end, 0);  			if (*end != '\0')  				die("unrecognized context count '%s'", arg + 2);  			continue;  		} -		if (!strncmp(arg, "--whitespace=", 13)) { +		if (!prefixcmp(arg, "--whitespace=")) {  			whitespace_option = arg + 13;  			parse_whitespace_option(arg + 13);  			continue; @@ -2692,7 +2816,7 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  			apply = apply_with_reject = apply_verbosely = 1;  			continue;  		} -		if (!strcmp(arg, "--verbose")) { +		if (!strcmp(arg, "-v") || !strcmp(arg, "--verbose")) {  			apply_verbosely = 1;  			continue;  		} @@ -2700,14 +2824,6 @@ int cmd_apply(int argc, const char **argv, const char *unused_prefix)  			inaccurate_eof = 1;  			continue;  		} - -		if (check_index && prefix_length < 0) { -			prefix = setup_git_directory(); -			prefix_length = prefix ? strlen(prefix) : 0; -			git_config(git_apply_config); -			if (!whitespace_option && apply_default_whitespace) -				parse_whitespace_option(apply_default_whitespace); -		}  		if (0 < prefix_length)  			arg = prefix_filename(prefix, prefix_length, arg); diff --git a/builtin-archive.c b/builtin-archive.c index f613ac2516..8ea6cb1efc 100644 --- a/builtin-archive.c +++ b/builtin-archive.c @@ -35,7 +35,7 @@ static int run_remote_archiver(const char *remote, int argc,  	for (i = 1; i < argc; i++) {  		const char *arg = argv[i]; -		if (!strncmp("--exec=", arg, 7)) { +		if (!prefixcmp(arg, "--exec=")) {  			if (exec_at)  				die("multiple --exec specified");  			exec = arg + 7; @@ -62,7 +62,7 @@ static int run_remote_archiver(const char *remote, int argc,  	if (buf[len-1] == '\n')  		buf[--len] = 0;  	if (strcmp(buf, "ACK")) { -		if (len > 5 && !strncmp(buf, "NACK ", 5)) +		if (len > 5 && !prefixcmp(buf, "NACK "))  			die("git-archive: NACK %s", buf + 5);  		die("git-archive: protocol error");  	} @@ -166,11 +166,11 @@ int parse_archive_args(int argc, const char **argv, struct archiver *ar)  			verbose = 1;  			continue;  		} -		if (!strncmp(arg, "--format=", 9)) { +		if (!prefixcmp(arg, "--format=")) {  			format = arg + 9;  			continue;  		} -		if (!strncmp(arg, "--prefix=", 9)) { +		if (!prefixcmp(arg, "--prefix=")) {  			base = arg + 9;  			continue;  		} @@ -218,7 +218,7 @@ static const char *extract_remote_arg(int *ac, const char **av)  		if (!strcmp(arg, "--"))  			no_more_options = 1;  		if (!no_more_options) { -			if (!strncmp(arg, "--remote=", 9)) { +			if (!prefixcmp(arg, "--remote=")) {  				if (remote)  					die("Multiple --remote specified");  				remote = arg + 9; diff --git a/builtin-blame.c b/builtin-blame.c index 5669a169ff..530b97f97d 100644 --- a/builtin-blame.c +++ b/builtin-blame.c @@ -2097,17 +2097,17 @@ int cmd_blame(int argc, const char **argv, const char *prefix)  			output_option |= OUTPUT_LONG_OBJECT_NAME;  		else if (!strcmp("-S", arg) && ++i < argc)  			revs_file = argv[i]; -		else if (!strncmp("-M", arg, 2)) { +		else if (!prefixcmp(arg, "-M")) {  			opt |= PICKAXE_BLAME_MOVE;  			blame_move_score = parse_score(arg+2);  		} -		else if (!strncmp("-C", arg, 2)) { +		else if (!prefixcmp(arg, "-C")) {  			if (opt & PICKAXE_BLAME_COPY)  				opt |= PICKAXE_BLAME_COPY_HARDER;  			opt |= PICKAXE_BLAME_COPY | PICKAXE_BLAME_MOVE;  			blame_copy_score = parse_score(arg+2);  		} -		else if (!strncmp("-L", arg, 2)) { +		else if (!prefixcmp(arg, "-L")) {  			if (!arg[2]) {  				if (++i >= argc)  					usage(blame_usage); diff --git a/builtin-branch.c b/builtin-branch.c index d0e7209368..d0179b00a2 100644 --- a/builtin-branch.c +++ b/builtin-branch.c @@ -59,7 +59,7 @@ int git_branch_config(const char *var, const char *value)  		branch_use_color = git_config_colorbool(var, value);  		return 0;  	} -	if (!strncmp(var, "color.branch.", 13)) { +	if (!prefixcmp(var, "color.branch.")) {  		int slot = parse_branch_color_slot(var, 13);  		color_parse(value, var, branch_colors[slot]);  		return 0; @@ -178,13 +178,13 @@ static int append_ref(const char *refname, const unsigned char *sha1, int flags,  	int len;  	/* Detect kind */ -	if (!strncmp(refname, "refs/heads/", 11)) { +	if (!prefixcmp(refname, "refs/heads/")) {  		kind = REF_LOCAL_BRANCH;  		refname += 11; -	} else if (!strncmp(refname, "refs/remotes/", 13)) { +	} else if (!prefixcmp(refname, "refs/remotes/")) {  		kind = REF_REMOTE_BRANCH;  		refname += 13; -	} else if (!strncmp(refname, "refs/tags/", 10)) { +	} else if (!prefixcmp(refname, "refs/tags/")) {  		kind = REF_TAG;  		refname += 10;  	} @@ -446,7 +446,7 @@ int cmd_branch(int argc, const char **argv, const char *prefix)  			reflog = 1;  			continue;  		} -		if (!strncmp(arg, "--abbrev=", 9)) { +		if (!prefixcmp(arg, "--abbrev=")) {  			abbrev = atoi(arg+9);  			continue;  		} @@ -476,7 +476,7 @@ int cmd_branch(int argc, const char **argv, const char *prefix)  		detached = 1;  	}  	else { -		if (strncmp(head, "refs/heads/", 11)) +		if (prefixcmp(head, "refs/heads/"))  			die("HEAD not found below refs/heads!");  		head += 11;  	} diff --git a/builtin-checkout-index.c b/builtin-checkout-index.c index b097c888a0..afe4b0e452 100644 --- a/builtin-checkout-index.c +++ b/builtin-checkout-index.c @@ -223,12 +223,12 @@ int cmd_checkout_index(int argc, const char **argv, const char *prefix)  			to_tempfile = 1;  			continue;  		} -		if (!strncmp(arg, "--prefix=", 9)) { +		if (!prefixcmp(arg, "--prefix=")) {  			state.base_dir = arg+9;  			state.base_dir_len = strlen(state.base_dir);  			continue;  		} -		if (!strncmp(arg, "--stage=", 8)) { +		if (!prefixcmp(arg, "--stage=")) {  			if (!strcmp(arg + 8, "all")) {  				to_tempfile = 1;  				checkout_stage = CHECKOUT_ALL; diff --git a/builtin-config.c b/builtin-config.c index 0f9051da17..f1433a4ab6 100644 --- a/builtin-config.c +++ b/builtin-config.c @@ -64,7 +64,7 @@ static int get_value(const char* key_, const char* regex_)  	int ret = -1;  	char *tl;  	char *global = NULL, *repo_config = NULL; -	const char *local; +	const char *system_wide = NULL, *local;  	local = getenv(CONFIG_ENVIRONMENT);  	if (!local) { @@ -74,6 +74,7 @@ static int get_value(const char* key_, const char* regex_)  			local = repo_config = xstrdup(git_path("config"));  		if (home)  			global = xstrdup(mkpath("%s/.gitconfig", home)); +		system_wide = ETC_GITCONFIG;  	}  	key = xstrdup(key_); @@ -103,11 +104,15 @@ static int get_value(const char* key_, const char* regex_)  		}  	} +	if (do_all && system_wide) +		git_config_from_file(show_config, system_wide);  	if (do_all && global)  		git_config_from_file(show_config, global);  	git_config_from_file(show_config, local);  	if (!do_all && !seen && global)  		git_config_from_file(show_config, global); +	if (!do_all && !seen && system_wide) +		git_config_from_file(show_config, system_wide);  	free(key);  	if (regexp) { @@ -147,7 +152,10 @@ int cmd_config(int argc, const char **argv, const char *prefix)  			} else {  				die("$HOME not set");  			} -		} else if (!strcmp(argv[1], "--rename-section")) { +		} +		else if (!strcmp(argv[1], "--system")) +			setenv("GIT_CONFIG", ETC_GITCONFIG, 1); +		else if (!strcmp(argv[1], "--rename-section")) {  			int ret;  			if (argc != 4)  				usage(git_config_set_usage); @@ -159,7 +167,8 @@ int cmd_config(int argc, const char **argv, const char *prefix)  				return 1;  			}  			return 0; -		} else +		} +		else  			break;  		argc--;  		argv++; diff --git a/builtin-describe.c b/builtin-describe.c index bcc645622a..165917e40d 100644 --- a/builtin-describe.c +++ b/builtin-describe.c @@ -52,7 +52,7 @@ static int get_name(const char *path, const unsigned char *sha1, int flag, void  	 * If --tags, then any tags are used.  	 * Otherwise only annotated tags are used.  	 */ -	if (!strncmp(path, "refs/tags/", 10)) { +	if (!prefixcmp(path, "refs/tags/")) {  		if (object->type == OBJ_TAG)  			prio = 2;  		else @@ -254,12 +254,12 @@ int cmd_describe(int argc, const char **argv, const char *prefix)  			all = 1;  		else if (!strcmp(arg, "--tags"))  			tags = 1; -		else if (!strncmp(arg, "--abbrev=", 9)) { +		else if (!prefixcmp(arg, "--abbrev=")) {  			abbrev = strtoul(arg + 9, NULL, 10);  			if (abbrev != 0 && (abbrev < MINIMUM_ABBREV || 40 < abbrev))  				abbrev = DEFAULT_ABBREV;  		} -		else if (!strncmp(arg, "--candidates=", 13)) { +		else if (!prefixcmp(arg, "--candidates=")) {  			max_candidates = strtoul(arg + 13, NULL, 10);  			if (max_candidates < 1)  				max_candidates = 1; diff --git a/builtin-diff.c b/builtin-diff.c index 45faa2067a..28b660a780 100644 --- a/builtin-diff.c +++ b/builtin-diff.c @@ -162,7 +162,8 @@ static int builtin_diff_combined(struct rev_info *revs,  	parent = xmalloc(ents * sizeof(*parent));  	/* Again, the revs are all reverse */  	for (i = 0; i < ents; i++) -		hashcpy((unsigned char*)parent + i, ent[ents - 1 - i].item->sha1); +		hashcpy((unsigned char *)(parent + i), +			ent[ents - 1 - i].item->sha1);  	diff_tree_combined(parent[0], parent + 1, ents - 1,  			   revs->dense_combined_merges, revs);  	return 0; @@ -232,6 +233,8 @@ int cmd_diff(int argc, const char **argv, const char *prefix)  				break;  			else if (!strcmp(arg, "--cached")) {  				add_head(&rev); +				if (!rev.pending.nr) +					die("No HEAD commit to compare with (yet)");  				break;  			}  		} diff --git a/builtin-fmt-merge-msg.c b/builtin-fmt-merge-msg.c index 87d3d63ec7..1489883564 100644 --- a/builtin-fmt-merge-msg.c +++ b/builtin-fmt-merge-msg.c @@ -81,7 +81,7 @@ static int handle_line(char *line)  	if (len < 43 || line[40] != '\t')  		return 1; -	if (!strncmp(line + 41, "not-for-merge", 13)) +	if (!prefixcmp(line + 41, "not-for-merge"))  		return 0;  	if (line[41] != '\t') @@ -119,15 +119,15 @@ static int handle_line(char *line)  	if (pulling_head) {  		origin = xstrdup(src);  		src_data->head_status |= 1; -	} else if (!strncmp(line, "branch ", 7)) { +	} else if (!prefixcmp(line, "branch ")) {  		origin = xstrdup(line + 7);  		append_to_list(&src_data->branch, origin, NULL);  		src_data->head_status |= 2; -	} else if (!strncmp(line, "tag ", 4)) { +	} else if (!prefixcmp(line, "tag ")) {  		origin = line;  		append_to_list(&src_data->tag, xstrdup(origin + 4), NULL);  		src_data->head_status |= 2; -	} else if (!strncmp(line, "remote branch ", 14)) { +	} else if (!prefixcmp(line, "remote branch ")) {  		origin = xstrdup(line + 14);  		append_to_list(&src_data->r_branch, origin, NULL);  		src_data->head_status |= 2; @@ -280,7 +280,7 @@ int cmd_fmt_merge_msg(int argc, const char **argv, const char *prefix)  	current_branch = resolve_ref("HEAD", head_sha1, 1, NULL);  	if (!current_branch)  		die("No current branch"); -	if (!strncmp(current_branch, "refs/heads/", 11)) +	if (!prefixcmp(current_branch, "refs/heads/"))  		current_branch += 11;  	while (fgets(line, sizeof(line), in)) { diff --git a/builtin-for-each-ref.c b/builtin-for-each-ref.c index 16c785f047..ac0b9f6088 100644 --- a/builtin-for-each-ref.c +++ b/builtin-for-each-ref.c @@ -814,7 +814,7 @@ int cmd_for_each_ref(int ac, const char **av, char *prefix)  			i++;  			break;  		} -		if (!strncmp(arg, "--format=", 9)) { +		if (!prefixcmp(arg, "--format=")) {  			if (format)  				die("more than one --format?");  			format = arg + 9; @@ -844,7 +844,7 @@ int cmd_for_each_ref(int ac, const char **av, char *prefix)  			quote_style = QUOTE_TCL;  			continue;  		} -		if (!strncmp(arg, "--count=", 8)) { +		if (!prefixcmp(arg, "--count=")) {  			if (maxcount)  				die("more than one --count?");  			maxcount = atoi(arg + 8); @@ -852,7 +852,7 @@ int cmd_for_each_ref(int ac, const char **av, char *prefix)  				die("The number %s did not parse", arg);  			continue;  		} -		if (!strncmp(arg, "--sort=", 7)) { +		if (!prefixcmp(arg, "--sort=")) {  			struct ref_sort *s = xcalloc(1, sizeof(*s));  			int len; diff --git a/builtin-fsck.c b/builtin-fsck.c index 6da3814d59..6abf498d2b 100644 --- a/builtin-fsck.c +++ b/builtin-fsck.c @@ -546,7 +546,7 @@ static int fsck_head_link(void)  	if (!head_points_at || !(flag & REF_ISSYMREF))  		return error("HEAD is not a symbolic ref"); -	if (strncmp(head_points_at, "refs/heads/", 11)) +	if (prefixcmp(head_points_at, "refs/heads/"))  		return error("HEAD points to something strange (%s)",  			     head_points_at);  	if (is_null_sha1(sha1)) diff --git a/builtin-grep.c b/builtin-grep.c index 2bfbdb7140..f35f2d023c 100644 --- a/builtin-grep.c +++ b/builtin-grep.c @@ -527,9 +527,9 @@ int cmd_grep(int argc, const char **argv, const char *prefix)  			opt.word_regexp = 1;  			continue;  		} -		if (!strncmp("-A", arg, 2) || -		    !strncmp("-B", arg, 2) || -		    !strncmp("-C", arg, 2) || +		if (!prefixcmp(arg, "-A") || +		    !prefixcmp(arg, "-B") || +		    !prefixcmp(arg, "-C") ||  		    (arg[0] == '-' && '1' <= arg[1] && arg[1] <= '9')) {  			unsigned num;  			const char *scan; diff --git a/builtin-init-db.c b/builtin-init-db.c index 12e43d0db4..4df9fd0fad 100644 --- a/builtin-init-db.c +++ b/builtin-init-db.c @@ -283,11 +283,11 @@ int cmd_init_db(int argc, const char **argv, const char *prefix)  	for (i = 1; i < argc; i++, argv++) {  		const char *arg = argv[1]; -		if (!strncmp(arg, "--template=", 11)) +		if (!prefixcmp(arg, "--template="))  			template_dir = arg+11;  		else if (!strcmp(arg, "--shared"))  			shared_repository = PERM_GROUP; -		else if (!strncmp(arg, "--shared=", 9)) +		else if (!prefixcmp(arg, "--shared="))  			shared_repository = git_config_perm("arg", arg+9);  		else  			usage(init_db_usage); diff --git a/builtin-log.c b/builtin-log.c index af2de54371..f43790cbce 100644 --- a/builtin-log.c +++ b/builtin-log.c @@ -32,7 +32,7 @@ static void cmd_log_init(int argc, const char **argv, const char *prefix,  		rev->always_show_header = 0;  	for (i = 1; i < argc; i++) {  		const char *arg = argv[i]; -		if (!strncmp(arg, "--encoding=", 11)) { +		if (!prefixcmp(arg, "--encoding=")) {  			arg += 11;  			if (strcmp(arg, "none"))  				git_log_output_encoding = strdup(arg); @@ -224,6 +224,9 @@ int cmd_log(int argc, const char **argv, const char *prefix)  	return cmd_log_walk(&rev);  } +/* format-patch */ +#define FORMAT_PATCH_NAME_MAX 64 +  static int istitlechar(char c)  {  	return (c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || @@ -264,15 +267,18 @@ static int git_format_config(const char *var, const char *value)  static FILE *realstdout = NULL;  static const char *output_directory = NULL; -static void reopen_stdout(struct commit *commit, int nr, int keep_subject) +static int reopen_stdout(struct commit *commit, int nr, int keep_subject)  { -	char filename[1024]; +	char filename[PATH_MAX];  	char *sol;  	int len = 0; -	int suffix_len = strlen(fmt_patch_suffix) + 10; /* ., NUL and slop */ +	int suffix_len = strlen(fmt_patch_suffix) + 1;  	if (output_directory) { -		strlcpy(filename, output_directory, 1000); +		if (strlen(output_directory) >= +		    sizeof(filename) - FORMAT_PATCH_NAME_MAX - suffix_len) +			return error("name of output directory is too long"); +		strlcpy(filename, output_directory, sizeof(filename) - suffix_len);  		len = strlen(filename);  		if (filename[len - 1] != '/')  			filename[len++] = '/'; @@ -287,7 +293,7 @@ static void reopen_stdout(struct commit *commit, int nr, int keep_subject)  		sol += 2;  		/* strip [PATCH] or [PATCH blabla] */ -		if (!keep_subject && !strncmp(sol, "[PATCH", 6)) { +		if (!keep_subject && !prefixcmp(sol, "[PATCH")) {  			char *eos = strchr(sol + 6, ']');  			if (eos) {  				while (isspace(*eos)) @@ -297,7 +303,8 @@ static void reopen_stdout(struct commit *commit, int nr, int keep_subject)  		}  		for (j = 0; -		     len < sizeof(filename) - suffix_len && +		     j < FORMAT_PATCH_NAME_MAX - suffix_len - 5 && +			     len < sizeof(filename) - suffix_len &&  			     sol[j] && sol[j] != '\n';  		     j++) {  			if (istitlechar(sol[j])) { @@ -314,10 +321,16 @@ static void reopen_stdout(struct commit *commit, int nr, int keep_subject)  		}  		while (filename[len - 1] == '.' || filename[len - 1] == '-')  			len--; +		filename[len] = 0;  	} +	if (len + suffix_len >= sizeof(filename)) +		return error("Patch pathname too long");  	strcpy(filename + len, fmt_patch_suffix);  	fprintf(realstdout, "%s\n", filename); -	freopen(filename, "w", stdout); +	if (freopen(filename, "w", stdout) == NULL) +		return error("Cannot open patch file %s",filename); +	return 0; +  }  static int get_patch_id(struct commit *commit, struct diff_options *options, @@ -435,7 +448,7 @@ int cmd_format_patch(int argc, const char **argv, const char *prefix)  		else if (!strcmp(argv[i], "-n") ||  				!strcmp(argv[i], "--numbered"))  			numbered = 1; -		else if (!strncmp(argv[i], "--start-number=", 15)) +		else if (!prefixcmp(argv[i], "--start-number="))  			start_number = strtol(argv[i] + 15, NULL, 10);  		else if (!strcmp(argv[i], "--start-number")) {  			i++; @@ -471,13 +484,13 @@ int cmd_format_patch(int argc, const char **argv, const char *prefix)  		}  		else if (!strcmp(argv[i], "--attach"))  			rev.mime_boundary = git_version_string; -		else if (!strncmp(argv[i], "--attach=", 9)) +		else if (!prefixcmp(argv[i], "--attach="))  			rev.mime_boundary = argv[i] + 9;  		else if (!strcmp(argv[i], "--ignore-if-in-upstream"))  			ignore_if_in_upstream = 1;  		else if (!strcmp(argv[i], "--thread"))  			thread = 1; -		else if (!strncmp(argv[i], "--in-reply-to=", 14)) +		else if (!prefixcmp(argv[i], "--in-reply-to="))  			in_reply_to = argv[i] + 14;  		else if (!strcmp(argv[i], "--in-reply-to")) {  			i++; @@ -485,7 +498,7 @@ int cmd_format_patch(int argc, const char **argv, const char *prefix)  				die("Need a Message-Id for --in-reply-to");  			in_reply_to = argv[i];  		} -		else if (!strncmp(argv[i], "--suffix=", 9)) +		else if (!prefixcmp(argv[i], "--suffix="))  			fmt_patch_suffix = argv[i] + 9;  		else  			argv[j++] = argv[i]; @@ -573,7 +586,8 @@ int cmd_format_patch(int argc, const char **argv, const char *prefix)  			rev.message_id = message_id;  		}  		if (!use_stdout) -			reopen_stdout(commit, rev.nr, keep_subject); +			if (reopen_stdout(commit, rev.nr, keep_subject)) +				die("Failed to create output files");  		shown = log_tree_commit(&rev, commit);  		free(commit->buffer);  		commit->buffer = NULL; diff --git a/builtin-ls-files.c b/builtin-ls-files.c index ac89eb2f77..4e1d5af634 100644 --- a/builtin-ls-files.c +++ b/builtin-ls-files.c @@ -406,7 +406,7 @@ int cmd_ls_files(int argc, const char **argv, const char *prefix)  			add_exclude(argv[++i], "", 0, &dir.exclude_list[EXC_CMDL]);  			continue;  		} -		if (!strncmp(arg, "--exclude=", 10)) { +		if (!prefixcmp(arg, "--exclude=")) {  			exc_given = 1;  			add_exclude(arg+10, "", 0, &dir.exclude_list[EXC_CMDL]);  			continue; @@ -416,12 +416,12 @@ int cmd_ls_files(int argc, const char **argv, const char *prefix)  			add_excludes_from_file(&dir, argv[++i]);  			continue;  		} -		if (!strncmp(arg, "--exclude-from=", 15)) { +		if (!prefixcmp(arg, "--exclude-from=")) {  			exc_given = 1;  			add_excludes_from_file(&dir, arg+15);  			continue;  		} -		if (!strncmp(arg, "--exclude-per-directory=", 24)) { +		if (!prefixcmp(arg, "--exclude-per-directory=")) {  			exc_given = 1;  			dir.exclude_per_dir = arg + 24;  			continue; @@ -434,7 +434,7 @@ int cmd_ls_files(int argc, const char **argv, const char *prefix)  			error_unmatch = 1;  			continue;  		} -		if (!strncmp(arg, "--abbrev=", 9)) { +		if (!prefixcmp(arg, "--abbrev=")) {  			abbrev = strtoul(arg+9, NULL, 10);  			if (abbrev && abbrev < MINIMUM_ABBREV)  				abbrev = MINIMUM_ABBREV; diff --git a/builtin-ls-tree.c b/builtin-ls-tree.c index 201defd934..6472610ac2 100644 --- a/builtin-ls-tree.c +++ b/builtin-ls-tree.c @@ -118,7 +118,7 @@ int cmd_ls_tree(int argc, const char **argv, const char *prefix)  				chomp_prefix = 0;  				break;  			} -			if (!strncmp(argv[1]+2, "abbrev=",7)) { +			if (!prefixcmp(argv[1]+2, "abbrev=")) {  				abbrev = strtoul(argv[1]+9, NULL, 10);  				if (abbrev && abbrev < MINIMUM_ABBREV)  					abbrev = MINIMUM_ABBREV; diff --git a/builtin-mailinfo.c b/builtin-mailinfo.c index 583da38b67..6ee6b0b26c 100644 --- a/builtin-mailinfo.c +++ b/builtin-mailinfo.c @@ -811,7 +811,7 @@ int cmd_mailinfo(int argc, const char **argv, const char *prefix)  			metainfo_charset = def_charset;  		else if (!strcmp(argv[1], "-n"))  			metainfo_charset = NULL; -		else if (!strncmp(argv[1], "--encoding=", 11)) +		else if (!prefixcmp(argv[1], "--encoding="))  			metainfo_charset = argv[1] + 11;  		else  			usage(mailinfo_usage); diff --git a/builtin-name-rev.c b/builtin-name-rev.c index 36f1ba6d38..c022224361 100644 --- a/builtin-name-rev.c +++ b/builtin-name-rev.c @@ -57,13 +57,17 @@ copy_data:  			parents;  			parents = parents->next, parent_number++) {  		if (parent_number > 1) { -			char *new_name = xmalloc(strlen(tip_name)+8); +			int len = strlen(tip_name); +			char *new_name = xmalloc(len + 8); +			if (len > 2 && !strcmp(tip_name + len - 2, "^0")) +				len -= 2;  			if (generation > 0) -				sprintf(new_name, "%s~%d^%d", tip_name, +				sprintf(new_name, "%.*s~%d^%d", len, tip_name,  						generation, parent_number);  			else -				sprintf(new_name, "%s^%d", tip_name, parent_number); +				sprintf(new_name, "%.*s^%d", len, tip_name, +						parent_number);  			name_rev(parents->item, new_name,  				merge_traversals + 1 , 0, 0); @@ -85,7 +89,7 @@ static int name_ref(const char *path, const unsigned char *sha1, int flags, void  	struct name_ref_data *data = cb_data;  	int deref = 0; -	if (data->tags_only && strncmp(path, "refs/tags/", 10)) +	if (data->tags_only && prefixcmp(path, "refs/tags/"))  		return 0;  	if (data->ref_filter && fnmatch(data->ref_filter, path, 0)) @@ -101,9 +105,9 @@ static int name_ref(const char *path, const unsigned char *sha1, int flags, void  	if (o && o->type == OBJ_COMMIT) {  		struct commit *commit = (struct commit *)o; -		if (!strncmp(path, "refs/heads/", 11)) +		if (!prefixcmp(path, "refs/heads/"))  			path = path + 11; -		else if (!strncmp(path, "refs/", 5)) +		else if (!prefixcmp(path, "refs/"))  			path = path + 5;  		name_rev(commit, xstrdup(path), 0, 0, deref); @@ -127,10 +131,15 @@ static const char* get_rev_name(struct object *o)  	if (!n->generation)  		return n->tip_name; - -	snprintf(buffer, sizeof(buffer), "%s~%d", n->tip_name, n->generation); - -	return buffer; +	else { +		int len = strlen(n->tip_name); +		if (len > 2 && !strcmp(n->tip_name + len - 2, "^0")) +			len -= 2; +		snprintf(buffer, sizeof(buffer), "%.*s~%d", len, n->tip_name, +				n->generation); + +		return buffer; +	}  }  int cmd_name_rev(int argc, const char **argv, const char *prefix) @@ -156,7 +165,7 @@ int cmd_name_rev(int argc, const char **argv, const char *prefix)  			} else if (!strcmp(*argv, "--tags")) {  				data.tags_only = 1;  				continue; -			} else  if (!strncmp(*argv, "--refs=", 7)) { +			} else  if (!prefixcmp(*argv, "--refs=")) {  				data.ref_filter = *argv + 7;  				continue;  			} else if (!strcmp(*argv, "--all")) { diff --git a/builtin-pack-objects.c b/builtin-pack-objects.c index 3824ee33ac..426ffddfff 100644 --- a/builtin-pack-objects.c +++ b/builtin-pack-objects.c @@ -1551,9 +1551,12 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)  	int use_internal_rev_list = 0;  	int thin = 0;  	int i; -	const char *rp_av[64]; +	const char **rp_av; +	int rp_ac_alloc = 64;  	int rp_ac; +	rp_av = xcalloc(rp_ac_alloc, sizeof(*rp_av)); +  	rp_av[0] = "pack-objects";  	rp_av[1] = "--objects"; /* --thin will make it --objects-edge */  	rp_ac = 2; @@ -1579,14 +1582,14 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)  			incremental = 1;  			continue;  		} -		if (!strncmp("--window=", arg, 9)) { +		if (!prefixcmp(arg, "--window=")) {  			char *end;  			window = strtoul(arg+9, &end, 0);  			if (!arg[9] || *end)  				usage(pack_usage);  			continue;  		} -		if (!strncmp("--depth=", arg, 8)) { +		if (!prefixcmp(arg, "--depth=")) {  			char *end;  			depth = strtoul(arg+8, &end, 0);  			if (!arg[8] || *end) @@ -1622,12 +1625,15 @@ int cmd_pack_objects(int argc, const char **argv, const char *prefix)  			continue;  		}  		if (!strcmp("--unpacked", arg) || -		    !strncmp("--unpacked=", arg, 11) || +		    !prefixcmp(arg, "--unpacked=") ||  		    !strcmp("--reflog", arg) ||  		    !strcmp("--all", arg)) {  			use_internal_rev_list = 1; -			if (ARRAY_SIZE(rp_av) - 1 <= rp_ac) -				die("too many internal rev-list options"); +			if (rp_ac >= rp_ac_alloc - 1) { +				rp_ac_alloc = alloc_nr(rp_ac_alloc); +				rp_av = xrealloc(rp_av, +						 rp_ac_alloc * sizeof(*rp_av)); +			}  			rp_av[rp_ac++] = arg;  			continue;  		} diff --git a/builtin-pack-refs.c b/builtin-pack-refs.c index 3de9b3eefd..d080e30d67 100644 --- a/builtin-pack-refs.c +++ b/builtin-pack-refs.c @@ -36,7 +36,7 @@ static int handle_one_ref(const char *path, const unsigned char *sha1,  	/* Do not pack the symbolic refs */  	if ((flags & REF_ISSYMREF))  		return 0; -	is_tag_ref = !strncmp(path, "refs/tags/", 10); +	is_tag_ref = !prefixcmp(path, "refs/tags/");  	/* ALWAYS pack refs that were already packed or are tags */  	if (!cb->all && !is_tag_ref && !(flags & REF_ISPACKED)) diff --git a/builtin-push.c b/builtin-push.c index c45649e26c..979efcc45f 100644 --- a/builtin-push.c +++ b/builtin-push.c @@ -32,7 +32,7 @@ static int expand_one_ref(const char *ref, const unsigned char *sha1, int flag,  	/* Ignore the "refs/" at the beginning of the refname */  	ref += 5; -	if (!strncmp(ref, "tags/", 5)) +	if (!prefixcmp(ref, "tags/"))  		add_refspec(xstrdup(ref));  	return 0;  } @@ -149,10 +149,10 @@ static int get_remotes_uri(const char *repo, const char *uri[MAX_URI])  		int is_refspec;  		char *s, *p; -		if (!strncmp("URL:", buffer, 4)) { +		if (!prefixcmp(buffer, "URL:")) {  			is_refspec = 0;  			s = buffer + 4; -		} else if (!strncmp("Push:", buffer, 5)) { +		} else if (!prefixcmp(buffer, "Push:")) {  			is_refspec = 1;  			s = buffer + 5;  		} else @@ -195,7 +195,7 @@ static int config_get_receivepack;  static int get_remote_config(const char* key, const char* value)  { -	if (!strncmp(key, "remote.", 7) && +	if (!prefixcmp(key, "remote.") &&  	    !strncmp(key + 7, config_repo, config_repo_len)) {  		if (!strcmp(key + 7 + config_repo_len, ".url")) {  			if (config_current_uri < MAX_URI) @@ -324,8 +324,8 @@ static int do_push(const char *repo)  		const char **dest_refspec = refspec;  		const char *dest = uri[i];  		const char *sender = "git-send-pack"; -		if (!strncmp(dest, "http://", 7) || -		    !strncmp(dest, "https://", 8)) +		if (!prefixcmp(dest, "http://") || +		    !prefixcmp(dest, "https://"))  			sender = "git-http-push";  		else if (thin)  			argv[dest_argc++] = "--thin"; @@ -373,7 +373,7 @@ int cmd_push(int argc, const char **argv, const char *prefix)  			verbose=1;  			continue;  		} -		if (!strncmp(arg, "--repo=", 7)) { +		if (!prefixcmp(arg, "--repo=")) {  			repo = arg+7;  			continue;  		} @@ -397,11 +397,11 @@ int cmd_push(int argc, const char **argv, const char *prefix)  			thin = 0;  			continue;  		} -		if (!strncmp(arg, "--receive-pack=", 15)) { +		if (!prefixcmp(arg, "--receive-pack=")) {  			receivepack = arg;  			continue;  		} -		if (!strncmp(arg, "--exec=", 7)) { +		if (!prefixcmp(arg, "--exec=")) {  			receivepack = arg;  			continue;  		} diff --git a/builtin-read-tree.c b/builtin-read-tree.c index 8ba436dbac..e47715538b 100644 --- a/builtin-read-tree.c +++ b/builtin-read-tree.c @@ -133,7 +133,7 @@ int cmd_read_tree(int argc, const char **argv, const char *unused_prefix)  		 *  entries and put the entries from the tree under the  		 * given subdirectory.  		 */ -		if (!strncmp(arg, "--prefix=", 9)) { +		if (!prefixcmp(arg, "--prefix=")) {  			if (stage || opts.merge || opts.prefix)  				usage(read_tree_usage);  			opts.prefix = arg + 9; @@ -179,7 +179,7 @@ int cmd_read_tree(int argc, const char **argv, const char *unused_prefix)  			continue;  		} -		if (!strncmp(arg, "--exclude-per-directory=", 24)) { +		if (!prefixcmp(arg, "--exclude-per-directory=")) {  			struct dir_struct *dir;  			if (opts.dir) diff --git a/builtin-reflog.c b/builtin-reflog.c index 341555139e..cefb40da81 100644 --- a/builtin-reflog.c +++ b/builtin-reflog.c @@ -321,9 +321,9 @@ static int cmd_reflog_expire(int argc, const char **argv, const char *prefix)  		const char *arg = argv[i];  		if (!strcmp(arg, "--dry-run") || !strcmp(arg, "-n"))  			cb.dry_run = 1; -		else if (!strncmp(arg, "--expire=", 9)) +		else if (!prefixcmp(arg, "--expire="))  			cb.expire_total = approxidate(arg + 9); -		else if (!strncmp(arg, "--expire-unreachable=", 21)) +		else if (!prefixcmp(arg, "--expire-unreachable="))  			cb.expire_unreachable = approxidate(arg + 21);  		else if (!strcmp(arg, "--stale-fix"))  			cb.stalefix = 1; diff --git a/builtin-rerere.c b/builtin-rerere.c index 318d959d89..b8867ab4ad 100644 --- a/builtin-rerere.c +++ b/builtin-rerere.c @@ -105,11 +105,11 @@ static int handle_file(const char *path,  		SHA1_Init(&ctx);  	while (fgets(buf, sizeof(buf), f)) { -		if (!strncmp("<<<<<<< ", buf, 8)) +		if (!prefixcmp(buf, "<<<<<<< "))  			hunk = 1; -		else if (!strncmp("=======", buf, 7)) +		else if (!prefixcmp(buf, "======="))  			hunk = 2; -		else if (!strncmp(">>>>>>> ", buf, 8)) { +		else if (!prefixcmp(buf, ">>>>>>> ")) {  			hunk_no++;  			hunk = 0;  			if (memcmp(one->ptr, two->ptr, one->nr < two->nr ? @@ -154,13 +154,17 @@ static int find_conflict(struct path_list *conflict)  		return error("Could not read index");  	for (i = 0; i + 2 < active_nr; i++) {  		struct cache_entry *e1 = active_cache[i]; -		struct cache_entry *e2 = active_cache[i + 1]; -		struct cache_entry *e3 = active_cache[i + 2]; -		if (ce_stage(e1) == 1 && ce_stage(e2) == 2 && -				ce_stage(e3) == 3 && ce_same_name(e1, e2) && -				ce_same_name(e1, e3)) { +		struct cache_entry *e2 = active_cache[i+1]; +		struct cache_entry *e3 = active_cache[i+2]; +		if (ce_stage(e1) == 1 && +		    ce_stage(e2) == 2 && +		    ce_stage(e3) == 3 && +		    ce_same_name(e1, e2) && ce_same_name(e1, e3) && +		    S_ISREG(ntohl(e1->ce_mode)) && +		    S_ISREG(ntohl(e2->ce_mode)) && +		    S_ISREG(ntohl(e3->ce_mode))) {  			path_list_insert((const char *)e1->name, conflict); -			i += 3; +			i += 2;  		}  	}  	return 0; diff --git a/builtin-rev-parse.c b/builtin-rev-parse.c index d53deaa369..37addb25fa 100644 --- a/builtin-rev-parse.c +++ b/builtin-rev-parse.c @@ -233,7 +233,7 @@ int cmd_rev_parse(int argc, const char **argv, const char *prefix)  			}  			continue;  		} -		if (!strncmp(arg,"-n",2)) { +		if (!prefixcmp(arg, "-n")) {  			if ((filter & DO_FLAGS) && (filter & DO_REVS))  				show(arg);  			continue; @@ -274,7 +274,7 @@ int cmd_rev_parse(int argc, const char **argv, const char *prefix)  				continue;  			}  			if (!strcmp(arg, "--short") || -			    !strncmp(arg, "--short=", 8)) { +			    !prefixcmp(arg, "--short=")) {  				filter &= ~(DO_FLAGS|DO_NOREV);  				verify = 1;  				abbrev = DEFAULT_ABBREV; @@ -352,19 +352,19 @@ int cmd_rev_parse(int argc, const char **argv, const char *prefix)  						: "false");  				continue;  			} -			if (!strncmp(arg, "--since=", 8)) { +			if (!prefixcmp(arg, "--since=")) {  				show_datestring("--max-age=", arg+8);  				continue;  			} -			if (!strncmp(arg, "--after=", 8)) { +			if (!prefixcmp(arg, "--after=")) {  				show_datestring("--max-age=", arg+8);  				continue;  			} -			if (!strncmp(arg, "--before=", 9)) { +			if (!prefixcmp(arg, "--before=")) {  				show_datestring("--min-age=", arg+9);  				continue;  			} -			if (!strncmp(arg, "--until=", 8)) { +			if (!prefixcmp(arg, "--until=")) {  				show_datestring("--min-age=", arg+8);  				continue;  			} diff --git a/builtin-shortlog.c b/builtin-shortlog.c index edb40429ec..2f71a2a6e2 100644 --- a/builtin-shortlog.c +++ b/builtin-shortlog.c @@ -124,7 +124,7 @@ static void insert_author_oneline(struct path_list *list,  	else  		free(buffer); -	if (!strncmp(oneline, "[PATCH", 6)) { +	if (!prefixcmp(oneline, "[PATCH")) {  		char *eob = strchr(oneline, ']');  		if (eob) { @@ -179,7 +179,7 @@ static void read_from_stdin(struct path_list *list)  	while (fgets(buffer, sizeof(buffer), stdin) != NULL) {  		char *bob;  		if ((buffer[0] == 'A' || buffer[0] == 'a') && -				!strncmp(buffer + 1, "uthor: ", 7) && +				!prefixcmp(buffer + 1, "uthor: ") &&  				(bob = strchr(buffer + 7, '<')) != NULL) {  			char buffer2[1024], offset = 0; @@ -230,7 +230,7 @@ static void get_from_rev(struct rev_info *rev, struct path_list *list)  			else  				eol++; -			if (!strncmp(buffer, "author ", 7)) { +			if (!prefixcmp(buffer, "author ")) {  				char *bracket = strchr(buffer, '<');  				if (bracket == NULL || bracket > eol) diff --git a/builtin-show-branch.c b/builtin-show-branch.c index 0d94e40df8..67ae6bacda 100644 --- a/builtin-show-branch.c +++ b/builtin-show-branch.c @@ -266,7 +266,7 @@ static void show_one_commit(struct commit *commit, int no_name)  				    pretty, sizeof(pretty), 0, NULL, NULL, 0);  	else  		strcpy(pretty, "(unavailable)"); -	if (!strncmp(pretty, "[PATCH] ", 8)) +	if (!prefixcmp(pretty, "[PATCH] "))  		cp = pretty + 8;  	else  		cp = pretty; @@ -378,7 +378,7 @@ static int append_head_ref(const char *refname, const unsigned char *sha1, int f  {  	unsigned char tmp[20];  	int ofs = 11; -	if (strncmp(refname, "refs/heads/", ofs)) +	if (prefixcmp(refname, "refs/heads/"))  		return 0;  	/* If both heads/foo and tags/foo exists, get_sha1 would  	 * get confused. @@ -392,7 +392,7 @@ static int append_remote_ref(const char *refname, const unsigned char *sha1, int  {  	unsigned char tmp[20];  	int ofs = 13; -	if (strncmp(refname, "refs/remotes/", ofs)) +	if (prefixcmp(refname, "refs/remotes/"))  		return 0;  	/* If both heads/foo and tags/foo exists, get_sha1 would  	 * get confused. @@ -404,7 +404,7 @@ static int append_remote_ref(const char *refname, const unsigned char *sha1, int  static int append_tag_ref(const char *refname, const unsigned char *sha1, int flag, void *cb_data)  { -	if (strncmp(refname, "refs/tags/", 10)) +	if (prefixcmp(refname, "refs/tags/"))  		return 0;  	return append_ref(refname + 5, sha1, 0);  } @@ -435,9 +435,9 @@ static int append_matching_ref(const char *refname, const unsigned char *sha1, i  		return 0;  	if (fnmatch(match_ref_pattern, tail, 0))  		return 0; -	if (!strncmp("refs/heads/", refname, 11)) +	if (!prefixcmp(refname, "refs/heads/"))  		return append_head_ref(refname, sha1, flag, cb_data); -	if (!strncmp("refs/tags/", refname, 10)) +	if (!prefixcmp(refname, "refs/tags/"))  		return append_tag_ref(refname, sha1, flag, cb_data);  	return append_ref(refname, sha1, 0);  } @@ -462,11 +462,11 @@ static int rev_is_head(char *head, int headlen, char *name,  	if ((!head[0]) ||  	    (head_sha1 && sha1 && hashcmp(head_sha1, sha1)))  		return 0; -	if (!strncmp(head, "refs/heads/", 11)) +	if (!prefixcmp(head, "refs/heads/"))  		head += 11; -	if (!strncmp(name, "refs/heads/", 11)) +	if (!prefixcmp(name, "refs/heads/"))  		name += 11; -	else if (!strncmp(name, "heads/", 6)) +	else if (!prefixcmp(name, "heads/"))  		name += 6;  	return !strcmp(head, name);  } @@ -635,7 +635,7 @@ int cmd_show_branch(int ac, const char **av, const char *prefix)  			with_current_branch = 1;  		else if (!strcmp(arg, "--sha1-name"))  			sha1_name = 1; -		else if (!strncmp(arg, "--more=", 7)) +		else if (!prefixcmp(arg, "--more="))  			extra = atoi(arg + 7);  		else if (!strcmp(arg, "--merge-base"))  			merge_base = 1; @@ -652,9 +652,9 @@ int cmd_show_branch(int ac, const char **av, const char *prefix)  		else if (!strcmp(arg, "--reflog") || !strcmp(arg, "-g")) {  			reflog = DEFAULT_REFLOG;  		} -		else if (!strncmp(arg, "--reflog=", 9)) +		else if (!prefixcmp(arg, "--reflog="))  			parse_reflog_param(arg + 9, &reflog, &reflog_base); -		else if (!strncmp(arg, "-g=", 3)) +		else if (!prefixcmp(arg, "-g="))  			parse_reflog_param(arg + 3, &reflog, &reflog_base);  		else  			usage(show_branch_usage); diff --git a/builtin-show-ref.c b/builtin-show-ref.c index 853f13f6ae..9463ff0e69 100644 --- a/builtin-show-ref.c +++ b/builtin-show-ref.c @@ -28,8 +28,8 @@ static int show_ref(const char *refname, const unsigned char *sha1, int flag, vo  	if (tags_only || heads_only) {  		int match; -		match = heads_only && !strncmp(refname, "refs/heads/", 11); -		match |= tags_only && !strncmp(refname, "refs/tags/", 10); +		match = heads_only && !prefixcmp(refname, "refs/heads/"); +		match |= tags_only && !prefixcmp(refname, "refs/tags/");  		if (!match)  			return 0;  	} @@ -178,8 +178,8 @@ int cmd_show_ref(int argc, const char **argv, const char *prefix)  			hash_only = 1;  			continue;  		} -		if (!strncmp(arg, "--hash=", 7) || -		    (!strncmp(arg, "--abbrev", 8) && +		if (!prefixcmp(arg, "--hash=") || +		    (!prefixcmp(arg, "--abbrev") &&  		     (arg[8] == '=' || arg[8] == '\0'))) {  			if (arg[2] != 'h' && !arg[8])  				/* --abbrev only */ @@ -215,16 +215,18 @@ int cmd_show_ref(int argc, const char **argv, const char *prefix)  		}  		if (!strcmp(arg, "--exclude-existing"))  			return exclude_existing(NULL); -		if (!strncmp(arg, "--exclude-existing=", 19)) +		if (!prefixcmp(arg, "--exclude-existing="))  			return exclude_existing(arg + 19);  		usage(show_ref_usage);  	}  	if (verify) { -		unsigned char sha1[20]; - +		if (!pattern) +			die("--verify requires a reference");  		while (*pattern) { -			if (!strncmp(*pattern, "refs/", 5) && +			unsigned char sha1[20]; + +			if (!prefixcmp(*pattern, "refs/") &&  			    resolve_ref(*pattern, sha1, 1, NULL)) {  				if (!quiet)  					show_one(*pattern, sha1); diff --git a/builtin-tar-tree.c b/builtin-tar-tree.c index 8055ddab9b..b04719ef20 100644 --- a/builtin-tar-tree.c +++ b/builtin-tar-tree.c @@ -31,7 +31,7 @@ int cmd_tar_tree(int argc, const char **argv, const char *prefix)  	nargv[nargc++] = "git-archive";  	nargv[nargc++] = "--format=tar"; -	if (2 <= argc && !strncmp("--remote=", argv[1], 9)) { +	if (2 <= argc && !prefixcmp(argv[1], "--remote=")) {  		nargv[nargc++] = argv[1];  		argv++;  		argc--; diff --git a/builtin-unpack-objects.c b/builtin-unpack-objects.c index d351e02649..8f8e898516 100644 --- a/builtin-unpack-objects.c +++ b/builtin-unpack-objects.c @@ -369,7 +369,7 @@ int cmd_unpack_objects(int argc, const char **argv, const char *prefix)  				recover = 1;  				continue;  			} -			if (!strncmp(arg, "--pack_header=", 14)) { +			if (!prefixcmp(arg, "--pack_header=")) {  				struct pack_header *hdr;  				char *c; diff --git a/builtin-write-tree.c b/builtin-write-tree.c index 50670dc7bf..90fc1cfcf4 100644 --- a/builtin-write-tree.c +++ b/builtin-write-tree.c @@ -70,7 +70,7 @@ int cmd_write_tree(int argc, const char **argv, const char *unused_prefix)  		const char *arg = argv[1];  		if (!strcmp(arg, "--missing-ok"))  			missing_ok = 1; -		else if (!strncmp(arg, "--prefix=", 9)) +		else if (!prefixcmp(arg, "--prefix="))  			prefix = arg + 9;  		else  			usage(write_tree_usage); @@ -211,6 +211,7 @@ extern const char *apply_default_whitespace;  extern int zlib_compression_level;  extern size_t packed_git_window_size;  extern size_t packed_git_limit; +extern int auto_crlf;  #define GIT_REPO_VERSION 0  extern int repository_format_version; @@ -478,4 +479,8 @@ extern int nfvasprintf(char **str, const char *fmt, va_list va);  extern void trace_printf(const char *format, ...);  extern void trace_argv_printf(const char **argv, int count, const char *format, ...); +/* convert.c */ +extern int convert_to_git(const char *path, char **bufp, unsigned long *sizep); +extern int convert_to_working_tree(const char *path, char **bufp, unsigned long *sizep); +  #endif /* CACHE_H */ diff --git a/combine-diff.c b/combine-diff.c index a5f2c8dd4a..6b7c6be959 100644 --- a/combine-diff.c +++ b/combine-diff.c @@ -678,9 +678,25 @@ static void show_patch_diff(struct combine_diff_path *elem, int num_parent,  	else {  		/* Used by diff-tree to read from the working tree */  		struct stat st; -		int fd; -		if (0 <= (fd = open(elem->path, O_RDONLY)) && -		    !fstat(fd, &st)) { +		int fd = -1; + +		if (lstat(elem->path, &st) < 0) +			goto deleted_file; + +		if (S_ISLNK(st.st_mode)) { +			int len = st.st_size; +			result_size = len; +			result = xmalloc(len + 1); +			if (result_size != readlink(elem->path, result, len)) { +				error("readlink(%s): %s", elem->path, +				      strerror(errno)); +				return; +			} +			result[len] = 0; +			elem->mode = canon_mode(st.st_mode); +		} +		else if (0 <= (fd = open(elem->path, O_RDONLY)) && +			 !fstat(fd, &st)) {  			int len = st.st_size;  			int sz = 0; @@ -698,11 +714,12 @@ static void show_patch_diff(struct combine_diff_path *elem, int num_parent,  			result[len] = 0;  		}  		else { -			/* deleted file */ +		deleted_file:  			result_size = 0;  			elem->mode = 0;  			result = xcalloc(1, 1);  		} +  		if (0 <= fd)  			close(fd);  	} diff --git a/compat/strtoumax.c b/compat/strtoumax.c new file mode 100644 index 0000000000..5541353a77 --- /dev/null +++ b/compat/strtoumax.c @@ -0,0 +1,10 @@ +#include "../git-compat-util.h" + +uintmax_t gitstrtoumax (const char *nptr, char **endptr, int base) +{ +#if defined(NO_STRTOULL) +	return strtoul(nptr, endptr, base); +#else +	return strtoull(nptr, endptr, base); +#endif +} @@ -326,6 +326,15 @@ int git_default_config(const char *var, const char *value)  		return 0;  	} +	if (!strcmp(var, "core.autocrlf")) { +		if (value && !strcasecmp(value, "input")) { +			auto_crlf = -1; +			return 0; +		} +		auto_crlf = git_config_bool(var, value); +		return 0; +	} +  	if (!strcmp(var, "user.name")) {  		strlcpy(git_default_name, value, sizeof(git_default_name));  		return 0; @@ -385,6 +394,8 @@ int git_config(config_fn_t fn)  	 * config file otherwise. */  	filename = getenv(CONFIG_ENVIRONMENT);  	if (!filename) { +		if (!access(ETC_GITCONFIG, R_OK)) +			ret += git_config_from_file(fn, ETC_GITCONFIG);  		home = getenv("HOME");  		filename = getenv(CONFIG_LOCAL_ENVIRONMENT);  		if (!filename) diff --git a/configure.ac b/configure.ac index 7cfb3a0666..3a8e778def 100644 --- a/configure.ac +++ b/configure.ac @@ -114,13 +114,32 @@ AC_CHECK_LIB([expat], [XML_ParserCreate],  [NO_EXPAT=YesPlease])  AC_SUBST(NO_EXPAT)  # -# Define NEEDS_LIBICONV if linking with libc is not enough (Darwin). +# Define NEEDS_LIBICONV if linking with libc is not enough (Darwin and +# some Solaris installations).  # Define NO_ICONV if neither libc nor libiconv support iconv. -AC_CHECK_LIB([c], [iconv], -	[NEEDS_LIBICONV=], -	AC_CHECK_LIB([iconv], [iconv], -		[NEEDS_LIBICONV=YesPlease], -		[NO_ICONV=YesPlease])) +AC_DEFUN([ICONVTEST_SRC], [ +#include <iconv.h> + +int main(void) +{ +	iconv_open("", ""); +	return 0; +} +]) +AC_MSG_CHECKING([for iconv in -lc]) +AC_LINK_IFELSE(ICONVTEST_SRC, +	[AC_MSG_RESULT([yes]) +	NEEDS_LIBICONV=], +	[AC_MSG_RESULT([no]) +	old_LIBS="$LIBS" +	LIBS="$LIBS -liconv" +	AC_MSG_CHECKING([for iconv in -liconv]) +	AC_LINK_IFELSE(ICONVTEST_SRC, +		[AC_MSG_RESULT([yes]) +		NEEDS_LIBICONV=YesPlease], +		[AC_MSG_RESULT([no]) +		NO_ICONV=YesPlease]) +	LIBS="$old_LIBS"])  AC_SUBST(NEEDS_LIBICONV)  AC_SUBST(NO_ICONV)  test -n "$NEEDS_LIBICONV" && LIBS="$LIBS -liconv" @@ -96,7 +96,7 @@ int get_ack(int fd, unsigned char *result_sha1)  		line[--len] = 0;  	if (!strcmp(line, "NAK"))  		return 0; -	if (!strncmp(line, "ACK ", 4)) { +	if (!prefixcmp(line, "ACK ")) {  		if (!get_sha1_hex(line+4, result_sha1)) {  			if (strstr(line+45, "continue"))  				return 2; @@ -196,8 +196,8 @@ static int count_refspec_match(const char *pattern,  		 */  		if (namelen != patlen &&  		    patlen != namelen - 5 && -		    strncmp(name, "refs/heads/", 11) && -		    strncmp(name, "refs/tags/", 10)) { +		    prefixcmp(name, "refs/heads/") && +		    prefixcmp(name, "refs/tags/")) {  			/* We want to catch the case where only weak  			 * matches are found and there are multiple  			 * matches, and where more than one strong diff --git a/contrib/fast-import/import-tars.perl b/contrib/fast-import/import-tars.perl index 990c9e70b2..5585a8b2c5 100755 --- a/contrib/fast-import/import-tars.perl +++ b/contrib/fast-import/import-tars.perl @@ -25,11 +25,14 @@ foreach my $tar_file (@ARGV)  	my $tar_name = $1;  	if ($tar_name =~ s/\.(tar\.gz|tgz)$//) { -		open(I, '-|', 'gzcat', $tar_file) or die "Unable to gzcat $tar_file: $!\n"; +		open(I, '-|', 'gunzip', '-c', $tar_file) +			or die "Unable to gunzip -c $tar_file: $!\n";  	} elsif ($tar_name =~ s/\.(tar\.bz2|tbz2)$//) { -		open(I, '-|', 'bzcat', $tar_file) or die "Unable to bzcat $tar_file: $!\n"; +		open(I, '-|', 'bunzip2', '-c', $tar_file) +			or die "Unable to bunzip2 -c $tar_file: $!\n";  	} elsif ($tar_name =~ s/\.tar\.Z$//) { -		open(I, '-|', 'zcat', $tar_file) or die "Unable to zcat $tar_file: $!\n"; +		open(I, '-|', 'uncompress', '-c', $tar_file) +			or die "Unable to uncompress -c $tar_file: $!\n";  	} elsif ($tar_name =~ s/\.tar$//) {  		open(I, $tar_file) or die "Unable to open $tar_file: $!\n";  	} else { diff --git a/convert.c b/convert.c new file mode 100644 index 0000000000..898bfe3eb2 --- /dev/null +++ b/convert.c @@ -0,0 +1,186 @@ +#include "cache.h" +/* + * convert.c - convert a file when checking it out and checking it in. + * + * This should use the pathname to decide on whether it wants to do some + * more interesting conversions (automatic gzip/unzip, general format + * conversions etc etc), but by default it just does automatic CRLF<->LF + * translation when the "auto_crlf" option is set. + */ + +struct text_stat { +	/* CR, LF and CRLF counts */ +	unsigned cr, lf, crlf; + +	/* These are just approximations! */ +	unsigned printable, nonprintable; +}; + +static void gather_stats(const char *buf, unsigned long size, struct text_stat *stats) +{ +	unsigned long i; + +	memset(stats, 0, sizeof(*stats)); + +	for (i = 0; i < size; i++) { +		unsigned char c = buf[i]; +		if (c == '\r') { +			stats->cr++; +			if (i+1 < size && buf[i+1] == '\n') +				stats->crlf++; +			continue; +		} +		if (c == '\n') { +			stats->lf++; +			continue; +		} +		if (c == 127) +			/* DEL */ +			stats->nonprintable++; +		else if (c < 32) { +			switch (c) { +				/* BS, HT, ESC and FF */ +			case '\b': case '\t': case '\033': case '\014': +				stats->printable++; +				break; +			default: +				stats->nonprintable++; +			} +		} +		else +			stats->printable++; +	} +} + +/* + * The same heuristics as diff.c::mmfile_is_binary() + */ +static int is_binary(unsigned long size, struct text_stat *stats) +{ + +	if ((stats->printable >> 7) < stats->nonprintable) +		return 1; +	/* +	 * Other heuristics? Average line length might be relevant, +	 * as might LF vs CR vs CRLF counts.. +	 * +	 * NOTE! It might be normal to have a low ratio of CRLF to LF +	 * (somebody starts with a LF-only file and edits it with an editor +	 * that adds CRLF only to lines that are added..). But do  we +	 * want to support CR-only? Probably not. +	 */ +	return 0; +} + +int convert_to_git(const char *path, char **bufp, unsigned long *sizep) +{ +	char *buffer, *nbuf; +	unsigned long size, nsize; +	struct text_stat stats; + +	/* +	 * FIXME! Other pluggable conversions should go here, +	 * based on filename patterns. Right now we just do the +	 * stupid auto-CRLF one. +	 */ +	if (!auto_crlf) +		return 0; + +	size = *sizep; +	if (!size) +		return 0; +	buffer = *bufp; + +	gather_stats(buffer, size, &stats); + +	/* No CR? Nothing to convert, regardless. */ +	if (!stats.cr) +		return 0; + +	/* +	 * We're currently not going to even try to convert stuff +	 * that has bare CR characters. Does anybody do that crazy +	 * stuff? +	 */ +	if (stats.cr != stats.crlf) +		return 0; + +	/* +	 * And add some heuristics for binary vs text, of course... +	 */ +	if (is_binary(size, &stats)) +		return 0; + +	/* +	 * Ok, allocate a new buffer, fill it in, and return true +	 * to let the caller know that we switched buffers on it. +	 */ +	nsize = size - stats.crlf; +	nbuf = xmalloc(nsize); +	*bufp = nbuf; +	*sizep = nsize; +	do { +		unsigned char c = *buffer++; +		if (c != '\r') +			*nbuf++ = c; +	} while (--size); + +	return 1; +} + +int convert_to_working_tree(const char *path, char **bufp, unsigned long *sizep) +{ +	char *buffer, *nbuf; +	unsigned long size, nsize; +	struct text_stat stats; +	unsigned char last; + +	/* +	 * FIXME! Other pluggable conversions should go here, +	 * based on filename patterns. Right now we just do the +	 * stupid auto-CRLF one. +	 */ +	if (auto_crlf <= 0) +		return 0; + +	size = *sizep; +	if (!size) +		return 0; +	buffer = *bufp; + +	gather_stats(buffer, size, &stats); + +	/* No LF? Nothing to convert, regardless. */ +	if (!stats.lf) +		return 0; + +	/* Was it already in CRLF format? */ +	if (stats.lf == stats.crlf) +		return 0; + +	/* If we have any bare CR characters, we're not going to touch it */ +	if (stats.cr != stats.crlf) +		return 0; + +	if (is_binary(size, &stats)) +		return 0; + +	/* +	 * Ok, allocate a new buffer, fill it in, and return true +	 * to let the caller know that we switched buffers on it. +	 */ +	nsize = size + stats.lf - stats.crlf; +	nbuf = xmalloc(nsize); +	*bufp = nbuf; +	*sizep = nsize; +	last = 0; +	do { +		unsigned char c = *buffer++; +		if (c == '\n' && last != '\r') +			*nbuf++ = '\r'; +		*nbuf++ = c; +		last = c; +	} while (--size); + +	return 1; +} @@ -286,7 +286,7 @@ static int service_enabled;  static int git_daemon_config(const char *var, const char *value)  { -	if (!strncmp(var, "daemon.", 7) && +	if (!prefixcmp(var, "daemon.") &&  	    !strcmp(var + 7, service_looking_at->config_name)) {  		service_enabled = git_config_bool(var, value);  		return 0; @@ -562,7 +562,7 @@ static int execute(struct sockaddr *addr)  	for (i = 0; i < ARRAY_SIZE(daemon_service); i++) {  		struct daemon_service *s = &(daemon_service[i]);  		int namelen = strlen(s->name); -		if (!strncmp("git-", line, 4) && +		if (!prefixcmp(line, "git-") &&  		    !strncmp(s->name, line + 4, namelen) &&  		    line[namelen + 4] == ' ') {  			/* @@ -1011,7 +1011,7 @@ int main(int argc, char **argv)  	for (i = 1; i < argc; i++) {  		char *arg = argv[i]; -		if (!strncmp(arg, "--listen=", 9)) { +		if (!prefixcmp(arg, "--listen=")) {  		    char *p = arg + 9;  		    char *ph = listen_addr = xmalloc(strlen(arg + 9) + 1);  		    while (*p) @@ -1019,7 +1019,7 @@ int main(int argc, char **argv)  		    *ph = 0;  		    continue;  		} -		if (!strncmp(arg, "--port=", 7)) { +		if (!prefixcmp(arg, "--port=")) {  			char *end;  			unsigned long n;  			n = strtoul(arg+7, &end, 0); @@ -1045,11 +1045,11 @@ int main(int argc, char **argv)  			export_all_trees = 1;  			continue;  		} -		if (!strncmp(arg, "--timeout=", 10)) { +		if (!prefixcmp(arg, "--timeout=")) {  			timeout = atoi(arg+10);  			continue;  		} -		if (!strncmp(arg, "--init-timeout=", 15)) { +		if (!prefixcmp(arg, "--init-timeout=")) {  			init_timeout = atoi(arg+15);  			continue;  		} @@ -1057,11 +1057,11 @@ int main(int argc, char **argv)  			strict_paths = 1;  			continue;  		} -		if (!strncmp(arg, "--base-path=", 12)) { +		if (!prefixcmp(arg, "--base-path=")) {  			base_path = arg+12;  			continue;  		} -		if (!strncmp(arg, "--interpolated-path=", 20)) { +		if (!prefixcmp(arg, "--interpolated-path=")) {  			interpolated_path = arg+20;  			continue;  		} @@ -1073,11 +1073,11 @@ int main(int argc, char **argv)  			user_path = "";  			continue;  		} -		if (!strncmp(arg, "--user-path=", 12)) { +		if (!prefixcmp(arg, "--user-path=")) {  			user_path = arg + 12;  			continue;  		} -		if (!strncmp(arg, "--pid-file=", 11)) { +		if (!prefixcmp(arg, "--pid-file=")) {  			pid_file = arg + 11;  			continue;  		} @@ -1086,27 +1086,27 @@ int main(int argc, char **argv)  			log_syslog = 1;  			continue;  		} -		if (!strncmp(arg, "--user=", 7)) { +		if (!prefixcmp(arg, "--user=")) {  			user_name = arg + 7;  			continue;  		} -		if (!strncmp(arg, "--group=", 8)) { +		if (!prefixcmp(arg, "--group=")) {  			group_name = arg + 8;  			continue;  		} -		if (!strncmp(arg, "--enable=", 9)) { +		if (!prefixcmp(arg, "--enable=")) {  			enable_service(arg + 9, 1);  			continue;  		} -		if (!strncmp(arg, "--disable=", 10)) { +		if (!prefixcmp(arg, "--disable=")) {  			enable_service(arg + 10, 0);  			continue;  		} -		if (!strncmp(arg, "--allow-override=", 17)) { +		if (!prefixcmp(arg, "--allow-override=")) {  			make_service_overridable(arg + 17, 1);  			continue;  		} -		if (!strncmp(arg, "--forbid-override=", 18)) { +		if (!prefixcmp(arg, "--forbid-override=")) {  			make_service_overridable(arg + 18, 0);  			continue;  		} diff --git a/diff-lib.c b/diff-lib.c index 17b9a56fa2..ae8364b42a 100644 --- a/diff-lib.c +++ b/diff-lib.c @@ -248,17 +248,27 @@ int run_diff_files(struct rev_info *revs, int silent_on_removed)  			path_len = ce_namelen(ce); -			dpath = xmalloc (combine_diff_path_size (5, path_len)); +			dpath = xmalloc(combine_diff_path_size(5, path_len));  			dpath->path = (char *) &(dpath->parent[5]);  			dpath->next = NULL;  			dpath->len = path_len;  			memcpy(dpath->path, ce->name, path_len);  			dpath->path[path_len] = '\0'; -			dpath->mode = 0;  			hashclr(dpath->sha1);  			memset(&(dpath->parent[0]), 0, -					sizeof(struct combine_diff_parent)*5); +			       sizeof(struct combine_diff_parent)*5); + +			if (lstat(ce->name, &st) < 0) { +				if (errno != ENOENT && errno != ENOTDIR) { +					perror(ce->name); +					continue; +				} +				if (silent_on_removed) +					continue; +			} +			else +				dpath->mode = canon_mode(st.st_mode);  			while (i < entries) {  				struct cache_entry *nce = active_cache[i]; @@ -77,7 +77,7 @@ int git_diff_ui_config(const char *var, const char *value)  			diff_detect_rename_default = DIFF_DETECT_RENAME;  		return 0;  	} -	if (!strncmp(var, "diff.color.", 11) || !strncmp(var, "color.diff.", 11)) { +	if (!prefixcmp(var, "diff.color.") || !prefixcmp(var, "color.diff.")) {  		int slot = parse_diff_color_slot(var, 11);  		color_parse(value, var, diff_colors[slot]);  		return 0; @@ -184,31 +184,43 @@ static void print_line_count(int count)  	}  } -static void copy_file(int prefix, const char *data, int size) +static void copy_file(int prefix, const char *data, int size, +		const char *set, const char *reset)  {  	int ch, nl_just_seen = 1;  	while (0 < size--) {  		ch = *data++; -		if (nl_just_seen) +		if (nl_just_seen) { +			fputs(set, stdout);  			putchar(prefix); -		putchar(ch); -		if (ch == '\n') +		} +		if (ch == '\n') {  			nl_just_seen = 1; -		else +			fputs(reset, stdout); +		} else  			nl_just_seen = 0; +		putchar(ch);  	}  	if (!nl_just_seen) -		printf("\n\\ No newline at end of file\n"); +		printf("%s\n\\ No newline at end of file\n", reset);  }  static void emit_rewrite_diff(const char *name_a,  			      const char *name_b,  			      struct diff_filespec *one, -			      struct diff_filespec *two) +			      struct diff_filespec *two, +			      int color_diff)  {  	int lc_a, lc_b;  	const char *name_a_tab, *name_b_tab; - +	const char *metainfo = diff_get_color(color_diff, DIFF_METAINFO); +	const char *fraginfo = diff_get_color(color_diff, DIFF_FRAGINFO); +	const char *old = diff_get_color(color_diff, DIFF_FILE_OLD); +	const char *new = diff_get_color(color_diff, DIFF_FILE_NEW); +	const char *reset = diff_get_color(color_diff, DIFF_RESET); + +	name_a += (*name_a == '/'); +	name_b += (*name_b == '/');  	name_a_tab = strchr(name_a, ' ') ? "\t" : "";  	name_b_tab = strchr(name_b, ' ') ? "\t" : ""; @@ -216,17 +228,17 @@ static void emit_rewrite_diff(const char *name_a,  	diff_populate_filespec(two, 0);  	lc_a = count_lines(one->data, one->size);  	lc_b = count_lines(two->data, two->size); -	printf("--- a/%s%s\n+++ b/%s%s\n@@ -", -	       name_a, name_a_tab, -	       name_b, name_b_tab); +	printf("%s--- a/%s%s%s\n%s+++ b/%s%s%s\n%s@@ -", +	       metainfo, name_a, name_a_tab, reset, +	       metainfo, name_b, name_b_tab, reset, fraginfo);  	print_line_count(lc_a);  	printf(" +");  	print_line_count(lc_b); -	printf(" @@\n"); +	printf(" @@%s\n", reset);  	if (lc_a) -		copy_file('-', one->data, one->size); +		copy_file('-', one->data, one->size, old, reset);  	if (lc_b) -		copy_file('+', two->data, two->size); +		copy_file('+', two->data, two->size, new, reset);  }  static int fill_mmfile(mmfile_t *mf, struct diff_filespec *one) @@ -405,22 +417,16 @@ static void emit_line(const char *set, const char *reset, const char *line, int  	puts(reset);  } -static void emit_add_line(const char *reset, struct emit_callback *ecbdata, const char *line, int len) +static void emit_line_with_ws(int nparents, +		const char *set, const char *reset, const char *ws, +		const char *line, int len)  { -	int col0 = ecbdata->nparents; +	int col0 = nparents;  	int last_tab_in_indent = -1;  	int last_space_in_indent = -1;  	int i;  	int tail = len;  	int need_highlight_leading_space = 0; -	const char *ws = diff_get_color(ecbdata->color_diff, DIFF_WHITESPACE); -	const char *set = diff_get_color(ecbdata->color_diff, DIFF_FILE_NEW); - -	if (!*ws) { -		emit_line(set, reset, line, len); -		return; -	} -  	/* The line is a newly added line.  Does it have funny leading  	 * whitespaces?  In indent, SP should never precede a TAB.  	 */ @@ -475,6 +481,18 @@ static void emit_add_line(const char *reset, struct emit_callback *ecbdata, cons  		emit_line(set, reset, line + i, len - i);  } +static void emit_add_line(const char *reset, struct emit_callback *ecbdata, const char *line, int len) +{ +	const char *ws = diff_get_color(ecbdata->color_diff, DIFF_WHITESPACE); +	const char *set = diff_get_color(ecbdata->color_diff, DIFF_FILE_NEW); + +	if (!*ws) +		emit_line(set, reset, line, len); +	else +		emit_line_with_ws(ecbdata->nparents, set, reset, ws, +				line, len); +} +  static void fn_out_consume(void *priv, char *line, unsigned long len)  {  	int i; @@ -884,30 +902,44 @@ static void show_numstat(struct diffstat_t* data, struct diff_options *options)  struct checkdiff_t {  	struct xdiff_emit_state xm;  	const char *filename; -	int lineno; +	int lineno, color_diff;  };  static void checkdiff_consume(void *priv, char *line, unsigned long len)  {  	struct checkdiff_t *data = priv; +	const char *ws = diff_get_color(data->color_diff, DIFF_WHITESPACE); +	const char *reset = diff_get_color(data->color_diff, DIFF_RESET); +	const char *set = diff_get_color(data->color_diff, DIFF_FILE_NEW);  	if (line[0] == '+') { -		int i, spaces = 0; +		int i, spaces = 0, space_before_tab = 0, white_space_at_end = 0;  		/* check space before tab */  		for (i = 1; i < len && (line[i] == ' ' || line[i] == '\t'); i++)  			if (line[i] == ' ')  				spaces++;  		if (line[i - 1] == '\t' && spaces) -			printf("%s:%d: space before tab:%.*s\n", -				data->filename, data->lineno, (int)len, line); +			space_before_tab = 1;  		/* check white space at line end */  		if (line[len - 1] == '\n')  			len--;  		if (isspace(line[len - 1])) -			printf("%s:%d: white space at end: %.*s\n", -				data->filename, data->lineno, (int)len, line); +			white_space_at_end = 1; + +		if (space_before_tab || white_space_at_end) { +			printf("%s:%d: %s", data->filename, data->lineno, ws); +			if (space_before_tab) { +				printf("space before tab"); +				if (white_space_at_end) +					putchar(','); +			} +			if (white_space_at_end) +				printf("white space at end"); +			printf(":%s ", reset); +			emit_line_with_ws(1, set, reset, ws, line, len); +		}  		data->lineno++;  	} else if (line[0] == ' ') @@ -1034,8 +1066,8 @@ static void builtin_diff(const char *name_a,  	const char *set = diff_get_color(o->color_diff, DIFF_METAINFO);  	const char *reset = diff_get_color(o->color_diff, DIFF_RESET); -	a_one = quote_two("a/", name_a); -	b_two = quote_two("b/", name_b); +	a_one = quote_two("a/", name_a + (*name_a == '/')); +	b_two = quote_two("b/", name_b + (*name_b == '/'));  	lbl[0] = DIFF_FILE_VALID(one) ? a_one : "/dev/null";  	lbl[1] = DIFF_FILE_VALID(two) ? b_two : "/dev/null";  	printf("%sdiff --git %s %s%s\n", set, a_one, b_two, reset); @@ -1064,7 +1096,8 @@ static void builtin_diff(const char *name_a,  		if ((one->mode ^ two->mode) & S_IFMT)  			goto free_ab_and_return;  		if (complete_rewrite) { -			emit_rewrite_diff(name_a, name_b, one, two); +			emit_rewrite_diff(name_a, name_b, one, two, +					o->color_diff);  			goto free_ab_and_return;  		}  	} @@ -1099,9 +1132,9 @@ static void builtin_diff(const char *name_a,  		xecfg.flags = XDL_EMIT_FUNCNAMES;  		if (!diffopts)  			; -		else if (!strncmp(diffopts, "--unified=", 10)) +		else if (!prefixcmp(diffopts, "--unified="))  			xecfg.ctxlen = strtoul(diffopts + 10, NULL, 10); -		else if (!strncmp(diffopts, "-u", 2)) +		else if (!prefixcmp(diffopts, "-u"))  			xecfg.ctxlen = strtoul(diffopts + 2, NULL, 10);  		ecb.outf = xdiff_outf;  		ecb.priv = &ecbdata; @@ -1165,7 +1198,7 @@ static void builtin_diffstat(const char *name_a, const char *name_b,  static void builtin_checkdiff(const char *name_a, const char *name_b,  			     struct diff_filespec *one, -			     struct diff_filespec *two) +			     struct diff_filespec *two, struct diff_options *o)  {  	mmfile_t mf1, mf2;  	struct checkdiff_t data; @@ -1177,6 +1210,7 @@ static void builtin_checkdiff(const char *name_a, const char *name_b,  	data.xm.consume = checkdiff_consume;  	data.filename = name_b ? name_b : name_a;  	data.lineno = 0; +	data.color_diff = o->color_diff;  	if (fill_mmfile(&mf1, one) < 0 || fill_mmfile(&mf2, two) < 0)  		die("unable to read files to diff"); @@ -1346,6 +1380,9 @@ int diff_populate_filespec(struct diff_filespec *s, int size_only)  	    reuse_worktree_file(s->path, s->sha1, 0)) {  		struct stat st;  		int fd; +		char *buf; +		unsigned long size; +  		if (lstat(s->path, &st) < 0) {  			if (errno == ENOENT) {  			err_empty: @@ -1378,7 +1415,19 @@ int diff_populate_filespec(struct diff_filespec *s, int size_only)  		s->data = xmmap(NULL, s->size, PROT_READ, MAP_PRIVATE, fd, 0);  		close(fd);  		s->should_munmap = 1; -		/* FIXME! CRLF -> LF conversion goes here, based on "s->path" */ + +		/* +		 * Convert from working tree format to canonical git format +		 */ +		buf = s->data; +		size = s->size; +		if (convert_to_git(s->path, &buf, &size)) { +			munmap(s->data, s->size); +			s->should_munmap = 0; +			s->data = buf; +			s->size = size; +			s->should_free = 1; +		}  	}  	else {  		char type[20]; @@ -1787,7 +1836,7 @@ static void run_checkdiff(struct diff_filepair *p, struct diff_options *o)  	diff_fill_sha1_info(p->one);  	diff_fill_sha1_info(p->two); -	builtin_checkdiff(name, other, p->one, p->two); +	builtin_checkdiff(name, other, p->one, p->two, o);  }  void diff_setup(struct diff_options *options) @@ -1936,7 +1985,7 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)  	else if (!strcmp(arg, "--shortstat")) {  		options->output_format |= DIFF_FORMAT_SHORTSTAT;  	} -	else if (!strncmp(arg, "--stat", 6)) { +	else if (!prefixcmp(arg, "--stat")) {  		char *end;  		int width = options->stat_width;  		int name_width = options->stat_name_width; @@ -1945,9 +1994,9 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)  		switch (*arg) {  		case '-': -			if (!strncmp(arg, "-width=", 7)) +			if (!prefixcmp(arg, "-width="))  				width = strtoul(arg + 7, &end, 10); -			else if (!strncmp(arg, "-name-width=", 12)) +			else if (!prefixcmp(arg, "-name-width="))  				name_width = strtoul(arg + 12, &end, 10);  			break;  		case '=': @@ -1972,7 +2021,7 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)  	}  	else if (!strcmp(arg, "-z"))  		options->line_termination = 0; -	else if (!strncmp(arg, "-l", 2)) +	else if (!prefixcmp(arg, "-l"))  		options->rename_limit = strtoul(arg+2, NULL, 10);  	else if (!strcmp(arg, "--full-index"))  		options->full_index = 1; @@ -1989,31 +2038,31 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)  		options->output_format |= DIFF_FORMAT_NAME_STATUS;  	else if (!strcmp(arg, "-R"))  		options->reverse_diff = 1; -	else if (!strncmp(arg, "-S", 2)) +	else if (!prefixcmp(arg, "-S"))  		options->pickaxe = arg + 2;  	else if (!strcmp(arg, "-s")) {  		options->output_format |= DIFF_FORMAT_NO_OUTPUT;  	} -	else if (!strncmp(arg, "-O", 2)) +	else if (!prefixcmp(arg, "-O"))  		options->orderfile = arg + 2; -	else if (!strncmp(arg, "--diff-filter=", 14)) +	else if (!prefixcmp(arg, "--diff-filter="))  		options->filter = arg + 14;  	else if (!strcmp(arg, "--pickaxe-all"))  		options->pickaxe_opts = DIFF_PICKAXE_ALL;  	else if (!strcmp(arg, "--pickaxe-regex"))  		options->pickaxe_opts = DIFF_PICKAXE_REGEX; -	else if (!strncmp(arg, "-B", 2)) { +	else if (!prefixcmp(arg, "-B")) {  		if ((options->break_opt =  		     diff_scoreopt_parse(arg)) == -1)  			return -1;  	} -	else if (!strncmp(arg, "-M", 2)) { +	else if (!prefixcmp(arg, "-M")) {  		if ((options->rename_score =  		     diff_scoreopt_parse(arg)) == -1)  			return -1;  		options->detect_rename = DIFF_DETECT_RENAME;  	} -	else if (!strncmp(arg, "-C", 2)) { +	else if (!prefixcmp(arg, "-C")) {  		if ((options->rename_score =  		     diff_scoreopt_parse(arg)) == -1)  			return -1; @@ -2023,7 +2072,7 @@ int diff_opt_parse(struct diff_options *options, const char **av, int ac)  		options->find_copies_harder = 1;  	else if (!strcmp(arg, "--abbrev"))  		options->abbrev = DEFAULT_ABBREV; -	else if (!strncmp(arg, "--abbrev=", 9)) { +	else if (!prefixcmp(arg, "--abbrev=")) {  		options->abbrev = strtoul(arg + 9, NULL, 10);  		if (options->abbrev < MINIMUM_ABBREV)  			options->abbrev = MINIMUM_ABBREV; @@ -2533,7 +2582,7 @@ static void patch_id_consume(void *priv, char *line, unsigned long len)  	int new_len;  	/* Ignore line numbers when computing the SHA1 of the patch */ -	if (!strncmp(line, "@@ -", 4)) +	if (!prefixcmp(line, "@@ -"))  		return;  	new_len = remove_space(line, len); @@ -78,6 +78,9 @@ static int write_entry(struct cache_entry *ce, char *path, struct checkout *stat  			path, sha1_to_hex(ce->sha1));  	}  	switch (ntohl(ce->ce_mode) & S_IFMT) { +		char *buf; +		unsigned long nsize; +  	case S_IFREG:  		if (to_tempfile) {  			strcpy(path, ".merge_file_XXXXXX"); @@ -89,7 +92,18 @@ static int write_entry(struct cache_entry *ce, char *path, struct checkout *stat  			return error("git-checkout-index: unable to create file %s (%s)",  				path, strerror(errno));  		} -		/* FIXME: LF -> CRLF conversion goes here, based on "ce->name" */ + +		/* +		 * Convert from git internal format to working tree format +		 */ +		buf = new; +		nsize = size; +		if (convert_to_working_tree(ce->name, &buf, &nsize)) { +			free(new); +			new = buf; +			size = nsize; +		} +  		wrote = write_in_full(fd, new, size);  		close(fd);  		free(new); diff --git a/environment.c b/environment.c index 54c22f8248..570e32ac3c 100644 --- a/environment.c +++ b/environment.c @@ -28,6 +28,7 @@ size_t packed_git_window_size = DEFAULT_PACKED_GIT_WINDOW_SIZE;  size_t packed_git_limit = DEFAULT_PACKED_GIT_LIMIT;  int pager_in_use;  int pager_use_color = 1; +int auto_crlf = 0;	/* 1: both ways, -1: only when adding git objects */  static const char *git_dir;  static char *git_object_dir, *git_index_file, *git_refs_dir, *git_graft_file; diff --git a/exec_cmd.c b/exec_cmd.c index 3996bce33f..9b74ed2f42 100644 --- a/exec_cmd.c +++ b/exec_cmd.c @@ -56,7 +56,7 @@ int execv_git_cmd(const char **argv)  			len = strlen(git_command);  			/* Trivial cleanup */ -			while (!strncmp(exec_dir, "./", 2)) { +			while (!prefixcmp(exec_dir, "./")) {  				exec_dir += 2;  				while (*exec_dir == '/')  					exec_dir++; diff --git a/fast-import.c b/fast-import.c index 404d911390..5d040fdb00 100644 --- a/fast-import.c +++ b/fast-import.c @@ -133,6 +133,10 @@ Format of STDIN stream:  #define PACK_ID_BITS 16  #define MAX_PACK_ID ((1<<PACK_ID_BITS)-1) +#ifndef PRIuMAX +#define PRIuMAX "llu" +#endif +  struct object_entry  {  	struct object_entry *next; @@ -475,7 +479,7 @@ static struct object_entry *find_mark(uintmax_t idnum)  			oe = s->data.marked[idnum];  	}  	if (!oe) -		die("mark :%ju not declared", orig_idnum); +		die("mark :%" PRIuMAX " not declared", orig_idnum);  	return oe;  } @@ -1361,7 +1365,7 @@ static void dump_marks_helper(FILE *f,  	} else {  		for (k = 0; k < 1024; k++) {  			if (m->data.marked[k]) -				fprintf(f, ":%ju %s\n", base + k, +				fprintf(f, ":%" PRIuMAX " %s\n", base + k,  					sha1_to_hex(m->data.marked[k]->sha1));  		}  	} @@ -1388,7 +1392,7 @@ static void read_next_command(void)  static void cmd_mark(void)  { -	if (!strncmp("mark :", command_buf.buf, 6)) { +	if (!prefixcmp(command_buf.buf, "mark :")) {  		next_mark = strtoumax(command_buf.buf + 6, NULL, 10);  		read_next_command();  	} @@ -1401,10 +1405,10 @@ static void *cmd_data (size_t *size)  	size_t length;  	char *buffer; -	if (strncmp("data ", command_buf.buf, 5)) +	if (prefixcmp(command_buf.buf, "data "))  		die("Expected 'data n' command, found: %s", command_buf.buf); -	if (!strncmp("<<", command_buf.buf + 5, 2)) { +	if (!prefixcmp(command_buf.buf + 5, "<<")) {  		char *term = xstrdup(command_buf.buf + 5 + 2);  		size_t sz = 8192, term_len = command_buf.len - 5 - 2;  		length = 0; @@ -1591,7 +1595,7 @@ static void file_change_m(struct branch *b)  		oe = find_mark(strtoumax(p + 1, &x, 10));  		hashcpy(sha1, oe->sha1);  		p = x; -	} else if (!strncmp("inline", p, 6)) { +	} else if (!prefixcmp(p, "inline")) {  		inline_data = 1;  		p += 6;  	} else { @@ -1664,7 +1668,7 @@ static void cmd_from(struct branch *b)  	const char *from;  	struct branch *s; -	if (strncmp("from ", command_buf.buf, 5)) +	if (prefixcmp(command_buf.buf, "from "))  		return;  	if (b->branch_tree.tree) { @@ -1687,7 +1691,7 @@ static void cmd_from(struct branch *b)  		unsigned long size;  		char *buf;  		if (oe->type != OBJ_COMMIT) -			die("Mark :%ju not a commit", idnum); +			die("Mark :%" PRIuMAX " not a commit", idnum);  		hashcpy(b->sha1, oe->sha1);  		buf = gfi_unpack_entry(oe, &size);  		if (!buf || size < 46) @@ -1730,7 +1734,7 @@ static struct hash_list *cmd_merge(unsigned int *count)  	struct branch *s;  	*count = 0; -	while (!strncmp("merge ", command_buf.buf, 6)) { +	while (!prefixcmp(command_buf.buf, "merge ")) {  		from = strchr(command_buf.buf, ' ') + 1;  		n = xmalloc(sizeof(*n));  		s = lookup_branch(from); @@ -1740,7 +1744,7 @@ static struct hash_list *cmd_merge(unsigned int *count)  			uintmax_t idnum = strtoumax(from + 1, NULL, 10);  			struct object_entry *oe = find_mark(idnum);  			if (oe->type != OBJ_COMMIT) -				die("Mark :%ju not a commit", idnum); +				die("Mark :%" PRIuMAX " not a commit", idnum);  			hashcpy(n->sha1, oe->sha1);  		} else if (get_sha1(from, n->sha1))  			die("Invalid ref name or SHA1 expression: %s", from); @@ -1776,11 +1780,11 @@ static void cmd_new_commit(void)  	read_next_command();  	cmd_mark(); -	if (!strncmp("author ", command_buf.buf, 7)) { +	if (!prefixcmp(command_buf.buf, "author ")) {  		author = parse_ident(command_buf.buf + 7);  		read_next_command();  	} -	if (!strncmp("committer ", command_buf.buf, 10)) { +	if (!prefixcmp(command_buf.buf, "committer ")) {  		committer = parse_ident(command_buf.buf + 10);  		read_next_command();  	} @@ -1801,9 +1805,9 @@ static void cmd_new_commit(void)  	for (;;) {  		if (1 == command_buf.len)  			break; -		else if (!strncmp("M ", command_buf.buf, 2)) +		else if (!prefixcmp(command_buf.buf, "M "))  			file_change_m(b); -		else if (!strncmp("D ", command_buf.buf, 2)) +		else if (!prefixcmp(command_buf.buf, "D "))  			file_change_d(b);  		else if (!strcmp("deleteall", command_buf.buf))  			file_change_deleteall(b); @@ -1873,7 +1877,7 @@ static void cmd_new_tag(void)  	read_next_command();  	/* from ... */ -	if (strncmp("from ", command_buf.buf, 5)) +	if (prefixcmp(command_buf.buf, "from "))  		die("Expected from command, got %s", command_buf.buf);  	from = strchr(command_buf.buf, ' ') + 1;  	s = lookup_branch(from); @@ -1884,7 +1888,7 @@ static void cmd_new_tag(void)  		from_mark = strtoumax(from + 1, NULL, 10);  		oe = find_mark(from_mark);  		if (oe->type != OBJ_COMMIT) -			die("Mark :%ju not a commit", from_mark); +			die("Mark :%" PRIuMAX " not a commit", from_mark);  		hashcpy(sha1, oe->sha1);  	} else if (!get_sha1(from, sha1)) {  		unsigned long size; @@ -1900,7 +1904,7 @@ static void cmd_new_tag(void)  	read_next_command();  	/* tagger ... */ -	if (strncmp("tagger ", command_buf.buf, 7)) +	if (prefixcmp(command_buf.buf, "tagger "))  		die("Expected tagger command, got %s", command_buf.buf);  	tagger = parse_ident(command_buf.buf + 7); @@ -1977,7 +1981,7 @@ int main(int argc, const char **argv)  		if (*a != '-' || !strcmp(a, "--"))  			break; -		else if (!strncmp(a, "--date-format=", 14)) { +		else if (!prefixcmp(a, "--date-format=")) {  			const char *fmt = a + 14;  			if (!strcmp(fmt, "raw"))  				whenspec = WHENSPEC_RAW; @@ -1988,15 +1992,15 @@ int main(int argc, const char **argv)  			else  				die("unknown --date-format argument %s", fmt);  		} -		else if (!strncmp(a, "--max-pack-size=", 16)) +		else if (!prefixcmp(a, "--max-pack-size="))  			max_packsize = strtoumax(a + 16, NULL, 0) * 1024 * 1024; -		else if (!strncmp(a, "--depth=", 8)) +		else if (!prefixcmp(a, "--depth="))  			max_depth = strtoul(a + 8, NULL, 0); -		else if (!strncmp(a, "--active-branches=", 18)) +		else if (!prefixcmp(a, "--active-branches="))  			max_active_branches = strtoul(a + 18, NULL, 0); -		else if (!strncmp(a, "--export-marks=", 15)) +		else if (!prefixcmp(a, "--export-marks="))  			mark_file = a + 15; -		else if (!strncmp(a, "--export-pack-edges=", 20)) { +		else if (!prefixcmp(a, "--export-pack-edges=")) {  			if (pack_edges)  				fclose(pack_edges);  			pack_edges = fopen(a + 20, "a"); @@ -2029,11 +2033,11 @@ int main(int argc, const char **argv)  			break;  		else if (!strcmp("blob", command_buf.buf))  			cmd_new_blob(); -		else if (!strncmp("commit ", command_buf.buf, 7)) +		else if (!prefixcmp(command_buf.buf, "commit "))  			cmd_new_commit(); -		else if (!strncmp("tag ", command_buf.buf, 4)) +		else if (!prefixcmp(command_buf.buf, "tag "))  			cmd_new_tag(); -		else if (!strncmp("reset ", command_buf.buf, 6)) +		else if (!prefixcmp(command_buf.buf, "reset "))  			cmd_reset_branch();  		else if (!strcmp("checkpoint", command_buf.buf))  			cmd_checkpoint(); @@ -2059,18 +2063,18 @@ int main(int argc, const char **argv)  		fprintf(stderr, "%s statistics:\n", argv[0]);  		fprintf(stderr, "---------------------------------------------------------------------\n"); -		fprintf(stderr, "Alloc'd objects: %10ju\n", alloc_count); -		fprintf(stderr, "Total objects:   %10ju (%10ju duplicates                  )\n", total_count, duplicate_count); -		fprintf(stderr, "      blobs  :   %10ju (%10ju duplicates %10ju deltas)\n", object_count_by_type[OBJ_BLOB], duplicate_count_by_type[OBJ_BLOB], delta_count_by_type[OBJ_BLOB]); -		fprintf(stderr, "      trees  :   %10ju (%10ju duplicates %10ju deltas)\n", object_count_by_type[OBJ_TREE], duplicate_count_by_type[OBJ_TREE], delta_count_by_type[OBJ_TREE]); -		fprintf(stderr, "      commits:   %10ju (%10ju duplicates %10ju deltas)\n", object_count_by_type[OBJ_COMMIT], duplicate_count_by_type[OBJ_COMMIT], delta_count_by_type[OBJ_COMMIT]); -		fprintf(stderr, "      tags   :   %10ju (%10ju duplicates %10ju deltas)\n", object_count_by_type[OBJ_TAG], duplicate_count_by_type[OBJ_TAG], delta_count_by_type[OBJ_TAG]); +		fprintf(stderr, "Alloc'd objects: %10" PRIuMAX "\n", alloc_count); +		fprintf(stderr, "Total objects:   %10" PRIuMAX " (%10" PRIuMAX " duplicates                  )\n", total_count, duplicate_count); +		fprintf(stderr, "      blobs  :   %10" PRIuMAX " (%10" PRIuMAX " duplicates %10" PRIuMAX " deltas)\n", object_count_by_type[OBJ_BLOB], duplicate_count_by_type[OBJ_BLOB], delta_count_by_type[OBJ_BLOB]); +		fprintf(stderr, "      trees  :   %10" PRIuMAX " (%10" PRIuMAX " duplicates %10" PRIuMAX " deltas)\n", object_count_by_type[OBJ_TREE], duplicate_count_by_type[OBJ_TREE], delta_count_by_type[OBJ_TREE]); +		fprintf(stderr, "      commits:   %10" PRIuMAX " (%10" PRIuMAX " duplicates %10" PRIuMAX " deltas)\n", object_count_by_type[OBJ_COMMIT], duplicate_count_by_type[OBJ_COMMIT], delta_count_by_type[OBJ_COMMIT]); +		fprintf(stderr, "      tags   :   %10" PRIuMAX " (%10" PRIuMAX " duplicates %10" PRIuMAX " deltas)\n", object_count_by_type[OBJ_TAG], duplicate_count_by_type[OBJ_TAG], delta_count_by_type[OBJ_TAG]);  		fprintf(stderr, "Total branches:  %10lu (%10lu loads     )\n", branch_count, branch_load_count); -		fprintf(stderr, "      marks:     %10ju (%10ju unique    )\n", (((uintmax_t)1) << marks->shift) * 1024, marks_set_count); +		fprintf(stderr, "      marks:     %10" PRIuMAX " (%10" PRIuMAX " unique    )\n", (((uintmax_t)1) << marks->shift) * 1024, marks_set_count);  		fprintf(stderr, "      atoms:     %10u\n", atom_cnt); -		fprintf(stderr, "Memory total:    %10ju KiB\n", (total_allocd + alloc_count*sizeof(struct object_entry))/1024); +		fprintf(stderr, "Memory total:    %10" PRIuMAX " KiB\n", (total_allocd + alloc_count*sizeof(struct object_entry))/1024);  		fprintf(stderr, "       pools:    %10lu KiB\n", (unsigned long)(total_allocd/1024)); -		fprintf(stderr, "     objects:    %10ju KiB\n", (alloc_count*sizeof(struct object_entry))/1024); +		fprintf(stderr, "     objects:    %10" PRIuMAX " KiB\n", (alloc_count*sizeof(struct object_entry))/1024);  		fprintf(stderr, "---------------------------------------------------------------------\n");  		pack_report();  		fprintf(stderr, "---------------------------------------------------------------------\n"); diff --git a/fetch-pack.c b/fetch-pack.c index c787106764..41bdd27b8f 100644 --- a/fetch-pack.c +++ b/fetch-pack.c @@ -198,13 +198,13 @@ static int find_common(int fd[2], unsigned char *result_sha1,  		int len;  		while ((len = packet_read_line(fd[0], line, sizeof(line)))) { -			if (!strncmp("shallow ", line, 8)) { +			if (!prefixcmp(line, "shallow ")) {  				if (get_sha1_hex(line + 8, sha1))  					die("invalid shallow line: %s", line);  				register_shallow(sha1);  				continue;  			} -			if (!strncmp("unshallow ", line, 10)) { +			if (!prefixcmp(line, "unshallow ")) {  				if (get_sha1_hex(line + 10, sha1))  					die("invalid unshallow line: %s", line);  				if (!lookup_object(sha1)) @@ -346,7 +346,7 @@ static void filter_refs(struct ref **refs, int nr_match, char **match)  		    check_ref_format(ref->name + 5))  			; /* trash */  		else if (fetch_all && -			 (!depth || strncmp(ref->name, "refs/tags/", 10) )) { +			 (!depth || prefixcmp(ref->name, "refs/tags/") )) {  			*newtail = ref;  			ref->next = NULL;  			newtail = &ref->next; @@ -683,11 +683,11 @@ int main(int argc, char **argv)  		char *arg = argv[i];  		if (*arg == '-') { -			if (!strncmp("--upload-pack=", arg, 14)) { +			if (!prefixcmp(arg, "--upload-pack=")) {  				uploadpack = arg + 14;  				continue;  			} -			if (!strncmp("--exec=", arg, 7)) { +			if (!prefixcmp(arg, "--exec=")) {  				uploadpack = arg + 7;  				continue;  			} @@ -712,7 +712,7 @@ int main(int argc, char **argv)  				verbose = 1;  				continue;  			} -			if (!strncmp("--depth=", arg, 8)) { +			if (!prefixcmp(arg, "--depth=")) {  				depth = strtol(arg + 8, NULL, 0);  				if (stat(git_path("shallow"), &st))  					st.st_mtime = 0; @@ -66,7 +66,7 @@ fall_back_3way () {      git-update-index -z --index-info <"$dotest/patch-merge-index-info" &&      GIT_INDEX_FILE="$dotest/patch-merge-tmp-index" \      git-write-tree >"$dotest/patch-merge-base+" || -    cannot_fallback "Patch does not record usable index information." +    cannot_fallback "Repository lacks necessary blobs to fall back on 3-way merge."      echo Using index info to reconstruct a base tree...      if GIT_INDEX_FILE="$dotest/patch-merge-tmp-index" \ diff --git a/git-commit.sh b/git-commit.sh index cfa15110f6..be3677c204 100755 --- a/git-commit.sh +++ b/git-commit.sh @@ -318,6 +318,10 @@ esac  case "$all,$also" in  t,) +	if test ! -f "$THIS_INDEX" +	then +		die 'nothing to commit (use "git add file1 file2" to include for commit)' +	fi  	save_index &&  	(  		cd_to_toplevel && diff --git a/git-compat-util.h b/git-compat-util.h index 105ac28f97..5d154faef6 100644 --- a/git-compat-util.h +++ b/git-compat-util.h @@ -1,6 +1,8 @@  #ifndef GIT_COMPAT_UTIL_H  #define GIT_COMPAT_UTIL_H +#define _FILE_OFFSET_BITS 64 +  #ifndef FLEX_ARRAY  #if defined(__GNUC__) && (__GNUC__ < 3)  #define FLEX_ARRAY 0 @@ -139,6 +141,11 @@ extern char *gitstrcasestr(const char *haystack, const char *needle);  extern size_t gitstrlcpy(char *, const char *, size_t);  #endif +#ifdef NO_STRTOUMAX +#define strtoumax gitstrtoumax +extern uintmax_t gitstrtoumax(const char *, char **, int); +#endif +  extern void release_pack_memory(size_t);  static inline char* xstrdup(const char *str) @@ -274,4 +281,9 @@ static inline int sane_case(int x, int high)  	return x;  } +static inline int prefixcmp(const char *str, const char *prefix) +{ +	return strncmp(str, prefix, strlen(prefix)); +} +  #endif diff --git a/git-cvsexportcommit.perl b/git-cvsexportcommit.perl index 870554eade..d08216cfd7 100755 --- a/git-cvsexportcommit.perl +++ b/git-cvsexportcommit.perl @@ -15,14 +15,21 @@ unless ($ENV{GIT_DIR} && -r $ENV{GIT_DIR}){      die "GIT_DIR is not defined or is unreadable";  } -our ($opt_h, $opt_P, $opt_p, $opt_v, $opt_c, $opt_f, $opt_a, $opt_m ); +our ($opt_h, $opt_P, $opt_p, $opt_v, $opt_c, $opt_f, $opt_a, $opt_m, $opt_d); -getopts('hPpvcfam:'); +getopts('hPpvcfam:d:');  $opt_h && usage();  die "Need at least one commit identifier!" unless @ARGV; +my @cvs; +if ($opt_d) { +	@cvs = ('cvs', '-d', $opt_d); +} else { +	@cvs = ('cvs'); +} +  # setup a tempdir  our ($tmpdir, $tmpdirname) = tempdir('git-cvsapplycommit-XXXXXX',  				     TMPDIR => 1, @@ -160,7 +167,7 @@ foreach my $f (@afiles) {  	my $p = $1;  	next if (grep { $_ eq $p } @dirs);      } -    my @status = grep(m/^File/,  safe_pipe_capture('cvs', '-q', 'status' ,$f)); +    my @status = grep(m/^File/,  safe_pipe_capture(@cvs, '-q', 'status' ,$f));      if (@status > 1) { warn 'Strange! cvs status returned more than one line?'};      if (-d dirname $f and $status[0] !~ m/Status: Unknown$/  	and $status[0] !~ m/^File: no file /) { @@ -173,7 +180,7 @@ foreach my $f (@afiles) {  foreach my $f (@files) {      next if grep { $_ eq $f } @afiles;      # TODO:we need to handle removed in cvs -    my @status = grep(m/^File/,  safe_pipe_capture('cvs', '-q', 'status' ,$f)); +    my @status = grep(m/^File/,  safe_pipe_capture(@cvs, '-q', 'status' ,$f));      if (@status > 1) { warn 'Strange! cvs status returned more than one line?'};      unless ($status[0] =~ m/Status: Up-to-date$/) {  	$dirty = 1; @@ -194,7 +201,7 @@ print "Applying\n";  print "Patch applied successfully. Adding new files and directories to CVS\n";  my $dirtypatch = 0;  foreach my $d (@dirs) { -    if (system('cvs','add',$d)) { +    if (system(@cvs,'add',$d)) {  	$dirtypatch = 1;  	warn "Failed to cvs add directory $d -- you may need to do it manually";      } @@ -202,9 +209,9 @@ foreach my $d (@dirs) {  foreach my $f (@afiles) {      if (grep { $_ eq $f } @bfiles) { -      system('cvs', 'add','-kb',$f); +      system(@cvs, 'add','-kb',$f);      } else { -      system('cvs', 'add', $f); +      system(@cvs, 'add', $f);      }      if ($?) {  	$dirtypatch = 1; @@ -213,7 +220,7 @@ foreach my $f (@afiles) {  }  foreach my $f (@dfiles) { -    system('cvs', 'rm', '-f', $f); +    system(@cvs, 'rm', '-f', $f);      if ($?) {  	$dirtypatch = 1;  	warn "Failed to cvs rm -f $f -- you may need to do it manually"; @@ -223,7 +230,7 @@ foreach my $f (@dfiles) {  print "Commit to CVS\n";  print "Patch title (first comment line): $title\n";  my @commitfiles = map { unless (m/\s/) { '\''.$_.'\''; } else { $_; }; } (@files); -my $cmd = "cvs commit -F .msg @commitfiles"; +my $cmd = join(' ', @cvs)." commit -F .msg @commitfiles";  if ($dirtypatch) {      print "NOTE: One or more hunks failed to apply cleanly.\n"; @@ -236,7 +243,7 @@ if ($dirtypatch) {  if ($opt_c) {      print "Autocommit\n  $cmd\n"; -    print safe_pipe_capture('cvs', 'commit', '-F', '.msg', @files); +    print safe_pipe_capture(@cvs, 'commit', '-F', '.msg', @files);      if ($?) {  	die "Exiting: The commit did not succeed";      } diff --git a/git-cvsserver.perl b/git-cvsserver.perl index 9371788fab..84520e7ad5 100755 --- a/git-cvsserver.perl +++ b/git-cvsserver.perl @@ -1171,6 +1171,21 @@ sub req_ci          exit;      } +	# Check that this is allowed, just as we would with a receive-pack +	my @cmd = ( $ENV{GIT_DIR}.'hooks/update', "refs/heads/$state->{module}", +			$parenthash, $commithash ); +	if( -x $cmd[0] ) { +		unless( system( @cmd ) == 0 ) +		{ +			$log->warn("Commit failed (update hook declined to update ref)"); +			print "error 1 Commit failed (update hook declined)\n"; +			close LOCKFILE; +			unlink($lockfile); +			chdir "/"; +			exit; +		} +	} +      print LOCKFILE $commithash;      $updater->update(); diff --git a/git-fetch.sh b/git-fetch.sh index ca984e739a..d230995f6e 100755 --- a/git-fetch.sh +++ b/git-fetch.sh @@ -243,6 +243,15 @@ then  	orig_head=$(git-rev-parse --verify HEAD 2>/dev/null)  fi +# Allow --notags from remote.$1.tagopt +case "$tags$no_tags" in +'') +	case "$(git-config --get "remote.$1.tagopt")" in +	--no-tags) +		no_tags=t ;; +	esac +esac +  # If --tags (and later --heads or --all) is specified, then we are  # not talking about defaults stored in Pull: line of remotes or  # branches file, and just fetch those and refspecs explicitly given. diff --git a/git-gui/.gitignore b/git-gui/.gitignore index c714d382e8..805ca2e1c7 100644 --- a/git-gui/.gitignore +++ b/git-gui/.gitignore @@ -1,3 +1,4 @@ +CREDITS-FILE  GIT-VERSION-FILE  git-citool  git-gui diff --git a/git-gui/CREDITS-GEN b/git-gui/CREDITS-GEN new file mode 100755 index 0000000000..d1b0f86355 --- /dev/null +++ b/git-gui/CREDITS-GEN @@ -0,0 +1,71 @@ +#!/bin/sh + +CF=CREDITS-FILE +tip= + +tree_search () +{ +	head=$1 +	tree=$2 +	for p in $(git rev-list --parents --max-count=1 $head 2>/dev/null) +	do +		test $tree = $(git rev-parse $p^{tree} 2>/dev/null) && +		vn=$(git describe --abbrev=4 $p 2>/dev/null) && +		case "$vn" in +		gitgui-[0-9]*) echo $p; break;; +		esac +	done +} + +generate_credits () +{ +	tip=$1 && +	rm -f "$2" && +	git shortlog -n -s $tip | sed 's/: .*$//' >"$2" || exit +} + +# Always use the tarball credits file if found, just +# in case we are somehow contained in a larger git +# repository that doesn't actually track our state. +# (At least one package manager is doing this.) +# +# We may be a subproject, so try looking for the merge +# commit that supplied this directory content if we are +# not at the toplevel.  We probably will always be the +# second parent in the commit, but we shouldn't rely on +# that fact. +# + +credits_tmp=/var/tmp/gitgui-credits-$$ +trap 'rm -f "$credits_tmp"' 0 + +orig="$credits_tmp" + +if test -f credits +then +	orig=credits +elif prefix="$(git rev-parse --show-prefix 2>/dev/null)" && +   test -n "$prefix" && +   head=$(git rev-list --max-count=1 HEAD -- . 2>/dev/null) && +   tree=$(git rev-parse --verify "HEAD:$prefix" 2>/dev/null) && +   tip=$(tree_search $head $tree) && +   test -n "$tip" +then +	generate_credits $tip "$orig" || exit +elif tip="$(git rev-parse --verify HEAD 2>/dev/null)" && +   test -n "$tip" +then +	generate_credits $tip "$orig" || exit +else +	echo "error: Cannot locate authorship information." >&2 +	exit 1 +fi + +if test -f "$orig" && cmp -s "$orig" "$CF" +then +	: noop +else +	rm -f "$CF" && +	cat "$orig" >"$CF" +fi + diff --git a/git-gui/GIT-VERSION-GEN b/git-gui/GIT-VERSION-GEN index 9966126da2..2741c1e14c 100755 --- a/git-gui/GIT-VERSION-GEN +++ b/git-gui/GIT-VERSION-GEN @@ -20,6 +20,11 @@ tree_search ()  	done  } +# Always use the tarball version file if found, just +# in case we are somehow contained in a larger git +# repository that doesn't actually track our state. +# (At least one package manager is doing this.) +#  # We may be a subproject, so try looking for the merge  # commit that supplied this directory content if we are  # not at the toplevel.  We probably will always be the @@ -27,10 +32,13 @@ tree_search ()  # that fact.  #  # If we are at the toplevel or the merge assumption fails -# try looking for a gitgui-* tag, or fallback onto the -# distributed version file. +# try looking for a gitgui-* tag. -if prefix="$(git rev-parse --show-prefix 2>/dev/null)" +if test -f version && +   VN=$(cat version) +then +	: happy +elif prefix="$(git rev-parse --show-prefix 2>/dev/null)"     test -n "$prefix" &&     head=$(git rev-list --max-count=1 HEAD -- . 2>/dev/null) &&     tree=$(git rev-parse --verify "HEAD:$prefix" 2>/dev/null) && @@ -48,9 +56,6 @@ elif VN=$(git describe --abbrev=4 HEAD 2>/dev/null) &&     esac  then  	VN=$(echo "$VN" | sed -e 's/^gitgui-//;s/-/./g'); -elif test -f version -then -	VN=$(cat version) || VN="$DEF_VER"  else  	VN="$DEF_VER"  fi diff --git a/git-gui/Makefile b/git-gui/Makefile index fd82d9d16d..66538ba1ad 100644 --- a/git-gui/Makefile +++ b/git-gui/Makefile @@ -4,9 +4,8 @@ GIT-VERSION-FILE: .FORCE-GIT-VERSION-FILE  	@$(SHELL_PATH) ./GIT-VERSION-GEN  -include GIT-VERSION-FILE -SCRIPT_SH = git-gui.sh  GITGUI_BUILT_INS = git-citool -ALL_PROGRAMS = $(GITGUI_BUILT_INS) $(patsubst %.sh,%,$(SCRIPT_SH)) +ALL_PROGRAMS = git-gui $(GITGUI_BUILT_INS)  ifndef SHELL_PATH  	SHELL_PATH = /bin/sh @@ -24,20 +23,24 @@ DESTDIR_SQ = $(subst ','\'',$(DESTDIR))  gitexecdir_SQ = $(subst ','\'',$(gitexecdir))  SHELL_PATH_SQ = $(subst ','\'',$(SHELL_PATH)) -$(patsubst %.sh,%,$(SCRIPT_SH)) : % : %.sh +git-gui: git-gui.sh GIT-VERSION-FILE CREDITS-FILE  	rm -f $@ $@+ -	sed -e '1s|#!.*/sh|#!$(SHELL_PATH_SQ)|' \ +	sed -n \ +		-e '1s|#!.*/sh|#!$(SHELL_PATH_SQ)|' \  		-e 's/@@GITGUI_VERSION@@/$(GITGUI_VERSION)/g' \ +		-e '1,/^set gitgui_credits /p' \  		$@.sh >$@+ +	cat CREDITS-FILE >>$@+ +	sed -e '1,/^set gitgui_credits /d' $@.sh >>$@+  	chmod +x $@+  	mv $@+ $@ +CREDITS-FILE: CREDITS-GEN .FORCE-CREDITS-FILE +	$(SHELL_PATH) ./CREDITS-GEN +  $(GITGUI_BUILT_INS): git-gui  	rm -f $@ && ln git-gui $@ -# These can record GITGUI_VERSION -$(patsubst %.sh,%,$(SCRIPT_SH)): GIT-VERSION-FILE -  all:: $(ALL_PROGRAMS)  install: all @@ -45,12 +48,14 @@ install: all  	$(INSTALL) git-gui '$(DESTDIR_SQ)$(gitexecdir_SQ)'  	$(foreach p,$(GITGUI_BUILT_INS), rm -f '$(DESTDIR_SQ)$(gitexecdir_SQ)/$p' && ln '$(DESTDIR_SQ)$(gitexecdir_SQ)/git-gui' '$(DESTDIR_SQ)$(gitexecdir_SQ)/$p' ;) -dist-version: +dist-version: CREDITS-FILE  	@mkdir -p $(TARDIR)  	@echo $(GITGUI_VERSION) > $(TARDIR)/version +	@cat CREDITS-FILE > $(TARDIR)/credits  clean:: -	rm -f $(ALL_PROGRAMS) GIT-VERSION-FILE +	rm -f $(ALL_PROGRAMS) GIT-VERSION-FILE CREDITS-FILE  .PHONY: all install dist-version clean  .PHONY: .FORCE-GIT-VERSION-FILE +.PHONY: .FORCE-CREDITS-FILE diff --git a/git-gui/TODO b/git-gui/TODO deleted file mode 100644 index b95a137322..0000000000 --- a/git-gui/TODO +++ /dev/null @@ -1,44 +0,0 @@ -Items outstanding: - - * Add file to .gitignore or info/excludes. - - * Populate the pull menu with local branches. - - * Make use of the new default merge data stored in repo-config. - - * Checkout a different local branch. - - * Push any local branch to a remote branch. - - * Merge any local branches through a real merge UI. - - * Allow user to define keyboard shortcuts for frequently used fetch -   or merge operations.  Or maybe just define a keyboard shortcut -   for default fetch/default merge of current branch is enough; -   but I do know a few users who merge a couple of common branches -   also into the same branch so one default isn't quite enough. - - * Better organize fetch/push/pull console windows. - - * Clone UI (to download a new repository). - - * Remotes editor (for .git/config format only). - - * Show a shortlog of the last couple of commits in the main window, -   to give the user warm fuzzy feelings that we have their data -   saved.  Actually this may be the set of commits not yet in -   the upstream (aka default merge branch remote repository). - - * GUI configuration editor for options listed in -   git.git/Documentation/config.txt.  Ideally this would -   parse that file and generate the options dialog from -   the documentation itself, and include the help text -   from the documentation as part of the UI somehow. - -Known bugs: - - * git-gui sometimes just closes on Windows with no error message. -   I'm not sure what the problem is here.  I suspect the wish -   process is just terminating due to a segfault or something, -   as the do_quit proc in git-gui doesn't run.  It often seems to -   occur while writing a commit message in the buffer.  Odd. diff --git a/git-gui/git-gui.sh b/git-gui/git-gui.sh index f5010dd47a..f84ba3382b 100755 --- a/git-gui/git-gui.sh +++ b/git-gui/git-gui.sh @@ -4,7 +4,7 @@ exec wish "$0" -- "$@"  set appvers {@@GITGUI_VERSION@@}  set copyright { -Copyright  2006, 2007 Shawn Pearce, Paul Mackerras. +Copyright  2006, 2007 Shawn Pearce, et. al.  This program is free software; you can redistribute it and/or modify  it under the terms of the GNU General Public License as published by @@ -19,6 +19,9 @@ GNU General Public License for more details.  You should have received a copy of the GNU General Public License  along with this program; if not, write to the Free Software  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA} +set gitgui_credits { +Paul Mackerras +}  ######################################################################  ## @@ -46,7 +49,7 @@ proc gitdir {args} {  proc gitexec {args} {  	global _gitexec  	if {$_gitexec eq {}} { -		if {[catch {set _gitexec [exec git --exec-path]} err]} { +		if {[catch {set _gitexec [git --exec-path]} err]} {  			error "Git not installed?\n\n$err"  		}  	} @@ -202,14 +205,14 @@ proc save_config {} {  		set value $global_config_new($name)  		if {$value ne $global_config($name)} {  			if {$value eq $default_config($name)} { -				catch {exec git config --global --unset $name} +				catch {git config --global --unset $name}  			} else {  				regsub -all "\[{}\]" $value {"} value -				exec git config --global $name $value +				git config --global $name $value  			}  			set global_config($name) $value  			if {$value eq $repo_config($name)} { -				catch {exec git config --unset $name} +				catch {git config --unset $name}  				set repo_config($name) $value  			}  		} @@ -219,16 +222,24 @@ proc save_config {} {  		set value $repo_config_new($name)  		if {$value ne $repo_config($name)} {  			if {$value eq $global_config($name)} { -				catch {exec git config --unset $name} +				catch {git config --unset $name}  			} else {  				regsub -all "\[{}\]" $value {"} value -				exec git config $name $value +				git config $name $value  			}  			set repo_config($name) $value  		}  	}  } +###################################################################### +## +## handy utils + +proc git {args} { +	return [eval exec git $args] +} +  proc error_popup {msg} {  	set title [appname]  	if {[reponame] ne {}} { @@ -289,10 +300,42 @@ proc ask_popup {msg} {  ######################################################################  ## +## version check + +set req_maj 1 +set req_min 5 + +if {[catch {set v [git --version]} err]} { +	catch {wm withdraw .} +	error_popup "Cannot determine Git version: + +$err + +[appname] requires Git $req_maj.$req_min or later." +	exit 1 +} +if {[regexp {^git version (\d+)\.(\d+)} $v _junk act_maj act_min]} { +	if {$act_maj < $req_maj +		|| ($act_maj == $req_maj && $act_min < $req_min)} { +		catch {wm withdraw .} +		error_popup "[appname] requires Git $req_maj.$req_min or later. + +You are using $v." +		exit 1 +	} +} else { +	catch {wm withdraw .} +	error_popup "Cannot parse Git version string:\n\n$v" +	exit 1 +} +unset -nocomplain v _junk act_maj act_min req_maj req_min + +###################################################################### +##  ## repository setup  if {   [catch {set _gitdir $env(GIT_DIR)}] -	&& [catch {set _gitdir [exec git rev-parse --git-dir]} err]} { +	&& [catch {set _gitdir [git rev-parse --git-dir]} err]} {  	catch {wm withdraw .}  	error_popup "Cannot find the git directory:\n\n$err"  	exit 1 @@ -321,6 +364,24 @@ set _reponame [lindex [file split \  ######################################################################  ## +## global init + +set current_diff_path {} +set current_diff_side {} +set diff_actions [list] +set ui_status_value {Initializing...} + +set HEAD {} +set PARENT {} +set MERGE_HEAD [list] +set commit_type {} +set empty_tree {} +set current_branch {} +set current_diff_path {} +set selected_commit_type new + +###################################################################### +##  ## task management  set rescan_active 0 @@ -365,7 +426,7 @@ proc repository_state {ctvar hdvar mhvar} {  	set mh [list] -	if {[catch {set current_branch [exec git symbolic-ref HEAD]}]} { +	if {[catch {set current_branch [git symbolic-ref HEAD]}]} {  		set current_branch {}  	} else {  		regsub ^refs/((heads|tags|remotes)/)? \ @@ -374,7 +435,7 @@ proc repository_state {ctvar hdvar mhvar} {  			current_branch  	} -	if {[catch {set hd [exec git rev-parse --verify HEAD]}]} { +	if {[catch {set hd [git rev-parse --verify HEAD]}]} {  		set hd {}  		set ct initial  		return @@ -402,7 +463,7 @@ proc PARENT {} {  		return $p  	}  	if {$empty_tree eq {}} { -		set empty_tree [exec git mktree << {}] +		set empty_tree [git mktree << {}]  	}  	return $empty_tree  } @@ -642,8 +703,9 @@ proc reshow_diff {} {  	global current_diff_path current_diff_side  	set p $current_diff_path -	if {$p eq {} -		|| $current_diff_side eq {} +	if {$p eq {}} { +		# No diff is being shown. +	} elseif {$current_diff_side eq {}  		|| [catch {set s $file_states($p)}]  		|| [lsearch -sorted -exact $file_lists($current_diff_side) $p] == -1} {  		clear_diff @@ -1042,7 +1104,7 @@ proc committer_ident {} {  	global GIT_COMMITTER_IDENT  	if {$GIT_COMMITTER_IDENT eq {}} { -		if {[catch {set me [exec git var GIT_COMMITTER_IDENT]} err]} { +		if {[catch {set me [git var GIT_COMMITTER_IDENT]} err]} {  			error_popup "Unable to obtain your identity:\n\n$err"  			return {}  		} @@ -1256,14 +1318,6 @@ proc commit_committree {fd_wt curHEAD msg} {  		return  	} -	# -- Make sure our current branch exists. -	# -	if {$commit_type eq {initial}} { -		lappend all_heads $current_branch -		set all_heads [lsort -unique $all_heads] -		populate_branch_menu -	} -  	# -- Cleanup after ourselves.  	#  	catch {file delete $msg_p} @@ -1275,7 +1329,7 @@ proc commit_committree {fd_wt curHEAD msg} {  	# -- Let rerere do its thing.  	#  	if {[file isdirectory [gitdir rr-cache]]} { -		catch {exec git rerere} +		catch {git rerere}  	}  	# -- Run the post-commit hook. @@ -1299,6 +1353,14 @@ proc commit_committree {fd_wt curHEAD msg} {  	if {[is_enabled singlecommit]} do_quit +	# -- Make sure our current branch exists. +	# +	if {$commit_type eq {initial}} { +		lappend all_heads $current_branch +		set all_heads [lsort -unique $all_heads] +		populate_branch_menu +	} +  	# -- Update in memory status  	#  	set selected_commit_type new @@ -1876,11 +1938,24 @@ proc all_tracking_branches {} {  	return [lsort -unique $all_trackings]  } +proc load_all_tags {} { +	set all_tags [list] +	set fd [open "| git for-each-ref --format=%(refname) refs/tags" r] +	while {[gets $fd line] > 0} { +		if {![regsub ^refs/tags/ $line {} name]} continue +		lappend all_tags $name +	} +	close $fd + +	return [lsort $all_tags] +} +  proc do_create_branch_action {w} {  	global all_heads null_sha1 repo_config  	global create_branch_checkout create_branch_revtype  	global create_branch_head create_branch_trackinghead  	global create_branch_name create_branch_revexp +	global create_branch_tag  	set newbranch $create_branch_name  	if {$newbranch eq {} @@ -1894,7 +1969,7 @@ proc do_create_branch_action {w} {  		focus $w.desc.name_t  		return  	} -	if {![catch {exec git show-ref --verify -- "refs/heads/$newbranch"}]} { +	if {![catch {git show-ref --verify -- "refs/heads/$newbranch"}]} {  		tk_messageBox \  			-icon error \  			-type ok \ @@ -1904,7 +1979,7 @@ proc do_create_branch_action {w} {  		focus $w.desc.name_t  		return  	} -	if {[catch {exec git check-ref-format "heads/$newbranch"}]} { +	if {[catch {git check-ref-format "heads/$newbranch"}]} {  		tk_messageBox \  			-icon error \  			-type ok \ @@ -1919,9 +1994,10 @@ proc do_create_branch_action {w} {  	switch -- $create_branch_revtype {  	head {set rev $create_branch_head}  	tracking {set rev $create_branch_trackinghead} +	tag {set rev $create_branch_tag}  	expression {set rev $create_branch_revexp}  	} -	if {[catch {set cmt [exec git rev-parse --verify "${rev}^0"]}]} { +	if {[catch {set cmt [git rev-parse --verify "${rev}^0"]}]} {  		tk_messageBox \  			-icon error \  			-type ok \ @@ -1964,6 +2040,8 @@ trace add variable create_branch_head write \  	[list radio_selector create_branch_revtype head]  trace add variable create_branch_trackinghead write \  	[list radio_selector create_branch_revtype tracking] +trace add variable create_branch_tag write \ +	[list radio_selector create_branch_revtype tag]  trace add variable delete_branch_head write \  	[list radio_selector delete_branch_checktype head] @@ -1975,6 +2053,7 @@ proc do_create_branch {} {  	global create_branch_checkout create_branch_revtype  	global create_branch_head create_branch_trackinghead  	global create_branch_name create_branch_revexp +	global create_branch_tag  	set w .branch_editor  	toplevel $w @@ -2038,6 +2117,19 @@ proc do_create_branch {} {  			$all_trackings  		grid $w.from.tracking_r $w.from.tracking_m -sticky w  	} +	set all_tags [load_all_tags] +	if {$all_tags ne {}} { +		set create_branch_tag [lindex $all_tags 0] +		radiobutton $w.from.tag_r \ +			-text {Tag:} \ +			-value tag \ +			-variable create_branch_revtype \ +			-font font_ui +		eval tk_optionMenu $w.from.tag_m \ +			create_branch_tag \ +			$all_tags +		grid $w.from.tag_r $w.from.tag_m -sticky w +	}  	radiobutton $w.from.exp_r \  		-text {Revision Expression:} \  		-value expression \ @@ -2100,7 +2192,7 @@ proc do_delete_branch_action {w} {  	}  	if {$check_rev eq {:none}} {  		set check_cmt {} -	} elseif {[catch {set check_cmt [exec git rev-parse --verify "${check_rev}^0"]}]} { +	} elseif {[catch {set check_cmt [git rev-parse --verify "${check_rev}^0"]}]} {  		tk_messageBox \  			-icon error \  			-type ok \ @@ -2114,10 +2206,10 @@ proc do_delete_branch_action {w} {  	set not_merged [list]  	foreach i [$w.list.l curselection] {  		set b [$w.list.l get $i] -		if {[catch {set o [exec git rev-parse --verify $b]}]} continue +		if {[catch {set o [git rev-parse --verify $b]}]} continue  		if {$check_cmt ne {}} {  			if {$b eq $check_rev} continue -			if {[catch {set m [exec git merge-base $o $check_cmt]}]} continue +			if {[catch {set m [git merge-base $o $check_cmt]}]} continue  			if {$o ne $m} {  				lappend not_merged $b  				continue @@ -2155,7 +2247,7 @@ Delete the selected branches?}  	foreach i $to_delete {  		set b [lindex $i 0]  		set o [lindex $i 1] -		if {[catch {exec git update-ref -d "refs/heads/$b" $o} err]} { +		if {[catch {git update-ref -d "refs/heads/$b" $o} err]} {  			append failed " - $b: $err\n"  		} else {  			set x [lsearch -sorted -exact $all_heads $b] @@ -2366,7 +2458,7 @@ Staying on branch '$current_branch'."  	#    here, it Just Works(tm).  If it doesn't we are in some really ugly  	#    state that is difficult to recover from within git-gui.  	# -	if {[catch {exec git symbolic-ref HEAD "refs/heads/$new_branch"} err]} { +	if {[catch {git symbolic-ref HEAD "refs/heads/$new_branch"} err]} {  		error_popup "Failed to set current branch.  This working directory is only partially switched. @@ -2876,14 +2968,16 @@ proc do_local_merge {} {  	pack $w.source -fill both -expand 1 -pady 5 -padx 5  	set cmd [list git for-each-ref] -	lappend cmd {--format=%(objectname) %(refname)} +	lappend cmd {--format=%(objectname) %(*objectname) %(refname)}  	lappend cmd refs/heads  	lappend cmd refs/remotes +	lappend cmd refs/tags  	set fr_fd [open "| $cmd" r]  	fconfigure $fr_fd -translation binary  	while {[gets $fr_fd line] > 0} {  		set line [split $line { }] -		set sha1([lindex $line 0]) [lindex $line 1] +		set sha1([lindex $line 0]) [lindex $line 2] +		set sha1([lindex $line 1]) [lindex $line 2]  	}  	close $fr_fd @@ -2891,7 +2985,7 @@ proc do_local_merge {} {  	set fr_fd [open "| git rev-list --all --not HEAD"]  	while {[gets $fr_fd line] > 0} {  		if {[catch {set ref $sha1($line)}]} continue -		regsub ^refs/(heads|remotes)/ $ref {} ref +		regsub ^refs/(heads|remotes|tags)/ $ref {} ref  		lappend to_show $ref  	}  	close $fr_fd @@ -2972,7 +3066,14 @@ proc new_browser {commit} {  	global next_browser_id cursor_ptr M1B  	global browser_commit browser_status browser_stack browser_path browser_busy -	set w .browser[incr next_browser_id] +	if {[winfo ismapped .]} { +		set w .browser[incr next_browser_id] +		set tl $w +		toplevel $w +	} else { +		set w {} +		set tl . +	}  	set w_list $w.list.l  	set browser_commit($w_list) $commit  	set browser_status($w_list) {Starting...} @@ -2980,7 +3081,6 @@ proc new_browser {commit} {  	set browser_path($w_list) $browser_commit($w_list):  	set browser_busy($w_list) 1 -	toplevel $w  	label $w.path -textvariable browser_path($w_list) \  		-anchor w \  		-justify left \ @@ -3030,8 +3130,8 @@ proc new_browser {commit} {  	bind $w_list <Left>            break  	bind $w_list <Right>           break -	bind $w <Visibility> "focus $w" -	bind $w <Destroy> " +	bind $tl <Visibility> "focus $w" +	bind $tl <Destroy> "  		array unset browser_buffer $w_list  		array unset browser_files $w_list  		array unset browser_status $w_list @@ -3040,7 +3140,7 @@ proc new_browser {commit} {  		array unset browser_commit $w_list  		array unset browser_busy $w_list  	" -	wm title $w "[appname] ([reponame]): File Browser" +	wm title $tl "[appname] ([reponame]): File Browser"  	ls_tree $w_list $browser_commit($w_list) {}  } @@ -4161,7 +4261,7 @@ proc do_quit {} {  			set rc_geometry {}  		}  		if {$cfg_geometry ne $rc_geometry} { -			catch {exec git config gui.geometry $cfg_geometry} +			catch {git config gui.geometry $cfg_geometry}  		}  	} @@ -4380,6 +4480,61 @@ proc do_commit {} {  	commit_tree  } +proc do_credits {} { +	global gitgui_credits + +	set w .credits_dialog + +	toplevel $w +	wm geometry $w "+[winfo rootx .]+[winfo rooty .]" + +	label $w.header -text {git-gui Contributors} -font font_uibold +	pack $w.header -side top -fill x + +	frame $w.buttons +	button $w.buttons.close -text {Close} \ +		-font font_ui \ +		-command [list destroy $w] +	pack $w.buttons.close -side right +	pack $w.buttons -side bottom -fill x -pady 10 -padx 10 + +	frame $w.credits +	text $w.credits.t \ +		-background [$w.header cget -background] \ +		-yscrollcommand [list $w.credits.sby set] \ +		-width 20 \ +		-height 10 \ +		-wrap none \ +		-borderwidth 1 \ +		-relief solid \ +		-padx 5 -pady 5 \ +		-font font_ui +	scrollbar $w.credits.sby -command [list $w.credits.t yview] +	pack $w.credits.sby -side right -fill y +	pack $w.credits.t -fill both -expand 1 +	pack $w.credits -side top -fill both -expand 1 -padx 5 -pady 5 + +	label $w.desc \ +		-text "All portions are copyrighted by their respective authors +and are distributed under the GNU General Public License." \ +		-padx 5 -pady 5 \ +		-justify left \ +		-anchor w \ +		-borderwidth 1 \ +		-relief solid \ +		-font font_ui +	pack $w.desc -side top -fill x -padx 5 -pady 5 + +	$w.credits.t insert end "[string trim $gitgui_credits]\n" +	$w.credits.t conf -state disabled +	$w.credits.t see 1.0 + +	bind $w <Visibility> "grab $w; focus $w" +	bind $w <Key-Escape> [list destroy $w] +	wm title $w [$w.header cget -text] +	tkwait window $w +} +  proc do_about {} {  	global appvers copyright  	global tcl_patchLevel tk_patchLevel @@ -4396,11 +4551,15 @@ proc do_about {} {  	button $w.buttons.close -text {Close} \  		-font font_ui \  		-command [list destroy $w] +	button $w.buttons.credits -text {Contributors} \ +		-font font_ui \ +		-command do_credits +	pack $w.buttons.credits -side left  	pack $w.buttons.close -side right  	pack $w.buttons -side bottom -fill x -pady 10 -padx 10  	label $w.desc \ -		-text "[appname] - a commit creation tool for Git. +		-text "git-gui - a graphical user interface for Git.  $copyright" \  		-padx 5 -pady 5 \  		-justify left \ @@ -4411,8 +4570,8 @@ $copyright" \  	pack $w.desc -side top -fill x -padx 5 -pady 5  	set v {} -	append v "[appname] version $appvers\n" -	append v "[exec git version]\n" +	append v "git-gui version $appvers\n" +	append v "[git version]\n"  	append v "\n"  	if {$tcl_patchLevel eq $tk_patchLevel} {  		append v "Tcl/Tk version $tcl_patchLevel" @@ -4471,7 +4630,7 @@ proc do_options {} {  	toplevel $w  	wm geometry $w "+[winfo rootx .]+[winfo rooty .]" -	label $w.header -text "[appname] Options" \ +	label $w.header -text "Options" \  		-font font_uibold  	pack $w.header -side top -fill x @@ -4945,6 +5104,9 @@ enable_option branch  enable_option transport  switch -- $subcommand { +--version - +version - +browser -  blame {  	disable_option multicommit  	disable_option branch @@ -5177,7 +5339,7 @@ if {[is_MacOSX]} {  	.mbar.apple add command -label "About [appname]" \  		-command do_about \  		-font font_ui -	.mbar.apple add command -label "[appname] Options..." \ +	.mbar.apple add command -label "Options..." \  		-command do_options \  		-font font_ui  } else { @@ -5236,7 +5398,7 @@ set doc_path [file dirname [gitexec]]  set doc_path [file join $doc_path Documentation index.html]  if {[is_Cygwin]} { -	set doc_path [exec cygpath --windows $doc_path] +	set doc_path [exec cygpath --mixed $doc_path]  }  if {$browser eq {}} { @@ -5280,6 +5442,20 @@ bind all <$M1B-Key-W> {destroy [winfo toplevel %W]}  # -- Not a normal commit type invocation?  Do that instead!  #  switch -- $subcommand { +--version - +version { +	puts "git-gui version $appvers" +	exit +} +browser { +	if {[llength $argv] != 1} { +		puts stderr "usage: $argv0 browser commit" +		exit 1 +	} +	set current_branch [lindex $argv 0] +	new_browser $current_branch +	return +}  blame {  	if {[llength $argv] != 2} {  		puts stderr "usage: $argv0 blame commit path" @@ -5302,7 +5478,7 @@ gui {  	# fall through to setup UI for commits  }  default { -	puts stderr "usage: $argv0 \[{blame|citool}\]" +	puts stderr "usage: $argv0 \[{blame|browser|citool}\]"  	exit 1  }  } @@ -5552,9 +5728,6 @@ bind_button3 $ui_comm "tk_popup $ctxm %X %Y"  # -- Diff Header  # -set current_diff_path {} -set current_diff_side {} -set diff_actions [list]  proc trace_current_diff_path {varname args} {  	global current_diff_path diff_actions file_states  	if {$current_diff_path eq {}} { @@ -5747,7 +5920,6 @@ unset ui_diff_applyhunk  # -- Status Bar  # -set ui_status_value {Initializing...}  label .status -textvariable ui_status_value \  	-anchor w \  	-justify left \ @@ -5821,15 +5993,6 @@ unset i  set file_lists($ui_index) [list]  set file_lists($ui_workdir) [list] -set HEAD {} -set PARENT {} -set MERGE_HEAD [list] -set commit_type {} -set empty_tree {} -set current_branch {} -set current_diff_path {} -set selected_commit_type new -  wm title . "[appname] ([file normalize [file dirname [gitdir]]])"  focus -force $ui_comm @@ -5904,7 +6067,7 @@ if {[is_enabled transport]} {  if {[is_enabled multicommit]} {  	set object_limit 2000  	if {[is_Windows]} {set object_limit 200} -	regexp {^([0-9]+) objects,} [exec git count-objects] _junk objects_current +	regexp {^([0-9]+) objects,} [git count-objects] _junk objects_current  	if {$objects_current >= $object_limit} {  		if {[ask_popup \  			"This repository currently has $objects_current loose objects. diff --git a/git-remote.perl b/git-remote.perl index c56c5a84a4..bd70bf1ddd 100755 --- a/git-remote.perl +++ b/git-remote.perl @@ -67,7 +67,7 @@ sub list_remote {  		$git->command(qw(config --get-regexp), '^remote\.');  	};  	for (@remotes) { -		if (/^remote\.([^.]*)\.(\S*)\s+(.*)$/) { +		if (/^remote\.(\S+?)\.([^.\s]+)\s+(.*)$/) {  			add_remote_config(\%seen, $1, $2, $3);  		}  	} @@ -274,6 +274,31 @@ sub add_remote {  	}  } +sub update_remote { +	my ($name) = @_; + +        my $conf = $git->config("remotes." . $name); +	if (defined($conf)) { +		@remotes = split(' ', $conf); +	} elsif ($name eq 'default') { +		undef @remotes; +		for (sort keys %$remote) { +			my $do_fetch = $git->config_boolean("remote." . $_ . +						    ".skipDefaultUpdate"); +			if (!defined($do_fetch) || $do_fetch ne "true") { +				push @remotes, $_; +			} +		} +	} else { +		print STDERR "Remote group $name does not exists.\n"; +		exit(1); +	} +	for (@remotes) { +		print "Updating $_\n"; +		$git->command('fetch', "$_"); +	} +} +  sub add_usage {  	print STDERR "Usage: git remote add [-f] [-t track]* [-m master] <name> <url>\n";  	exit(1); @@ -303,6 +328,15 @@ elsif ($ARGV[0] eq 'show') {  		show_remote($ARGV[$i], $ls_remote);  	}  } +elsif ($ARGV[0] eq 'update') { +	if (@ARGV <= 1) { +		update_remote("default"); +		exit(1); +	} +	for ($i = 1; $i < @ARGV; $i++) { +		update_remote($ARGV[$i]); +	} +}  elsif ($ARGV[0] eq 'prune') {  	my $ls_remote = 1;  	my $i; @@ -360,5 +394,6 @@ else {  	print STDERR "       git remote add <name> <url>\n";  	print STDERR "       git remote show <name>\n";  	print STDERR "       git remote prune <name>\n"; +	print STDERR "       git remote update [group]\n";  	exit(1);  } diff --git a/git-svn.perl b/git-svn.perl index d792a62d7c..41961b59f6 100755 --- a/git-svn.perl +++ b/git-svn.perl @@ -4,33 +4,21 @@  use warnings;  use strict;  use vars qw/	$AUTHOR $VERSION -		$SVN_URL $SVN_INFO $SVN_WC $SVN_UUID -		$GIT_SVN_INDEX $GIT_SVN -		$GIT_DIR $GIT_SVN_DIR $REVDB/; +		$sha1 $sha1_short $_revision +		$_q $_authors %users/;  $AUTHOR = 'Eric Wong <normalperson@yhbt.net>';  $VERSION = '@@GIT_VERSION@@'; -use Cwd qw/abs_path/; -$GIT_DIR = abs_path($ENV{GIT_DIR} || '.git'); -$ENV{GIT_DIR} = $GIT_DIR; +my $git_dir_user_set = 1 if defined $ENV{GIT_DIR}; +$ENV{GIT_DIR} ||= '.git'; +$Git::SVN::default_repo_id = 'svn'; +$Git::SVN::default_ref_id = $ENV{GIT_SVN_ID} || 'git-svn'; +$Git::SVN::Ra::_log_window_size = 100; -my $LC_ALL = $ENV{LC_ALL}; -my $TZ = $ENV{TZ}; -# make sure the svn binary gives consistent output between locales and TZs: +$Git::SVN::Log::TZ = $ENV{TZ};  $ENV{TZ} = 'UTC'; -$ENV{LC_ALL} = 'C';  $| = 1; # unbuffer STDOUT -# properties that we do not log: -my %SKIP = ( 'svn:wc:ra_dav:version-url' => 1, -             'svn:special' => 1, -             'svn:executable' => 1, -             'svn:entry:committed-rev' => 1, -             'svn:entry:last-author' => 1, -             'svn:entry:uuid' => 1, -             'svn:entry:committed-date' => 1, -); -  sub fatal (@) { print STDERR @_; exit 1 }  require SVN::Core; # use()-ing this causes segfaults for me... *shrug*  require SVN::Ra; @@ -38,120 +26,129 @@ require SVN::Delta;  if ($SVN::Core::VERSION lt '1.1.0') {  	fatal "Need SVN::Core 1.1.0 or better (got $SVN::Core::VERSION)\n";  } +push @Git::SVN::Ra::ISA, 'SVN::Ra';  push @SVN::Git::Editor::ISA, 'SVN::Delta::Editor';  push @SVN::Git::Fetcher::ISA, 'SVN::Delta::Editor'; -*SVN::Git::Fetcher::process_rm = *process_rm;  use Carp qw/croak/;  use IO::File qw//;  use File::Basename qw/dirname basename/;  use File::Path qw/mkpath/;  use Getopt::Long qw/:config gnu_getopt no_ignore_case auto_abbrev pass_through/; -use POSIX qw/strftime/;  use IPC::Open3; -use Memoize; -use Git qw/command command_oneline command_noisy -           command_output_pipe command_input_pipe command_close_pipe/; -memoize('revisions_eq'); -memoize('cmt_metadata'); -memoize('get_commit_time'); +use Git; + +BEGIN { +	my $s; +	foreach (qw/command command_oneline command_noisy command_output_pipe +	            command_input_pipe command_close_pipe/) { +		$s .= "*SVN::Git::Editor::$_ = *SVN::Git::Fetcher::$_ = ". +		      "*Git::SVN::Migration::$_ = ". +		      "*Git::SVN::Log::$_ = *Git::SVN::$_ = *$_ = *Git::$_; "; +	} +	eval $s; +}  my ($SVN); -my $_optimize_commits = 1 unless $ENV{GIT_SVN_NO_OPTIMIZE_COMMITS}; -my $sha1 = qr/[a-f\d]{40}/; -my $sha1_short = qr/[a-f\d]{4,40}/; -my $_esc_color = qr/(?:\033\[(?:(?:\d+;)*\d*)?m)*/; -my ($_revision,$_stdin,$_no_ignore_ext,$_no_stop_copy,$_help,$_rmdir,$_edit, -	$_find_copies_harder, $_l, $_cp_similarity, $_cp_remote, -	$_repack, $_repack_nr, $_repack_flags, $_q, -	$_message, $_file, $_follow_parent, $_no_metadata, -	$_template, $_shared, $_no_default_regex, $_no_graft_copy, -	$_limit, $_verbose, $_incremental, $_oneline, $_l_fmt, $_show_commit, -	$_version, $_upgrade, $_authors, $_branch_all_refs, @_opt_m, -	$_merge, $_strategy, $_dry_run, $_ignore_nodate, $_non_recursive, -	$_username, $_config_dir, $_no_auth_cache, -	$_pager, $_color, $_prefix); -my (@_branch_from, %tree_map, %users, %rusers, %equiv); -my ($_svn_can_do_switch); -my @repo_path_split_cache; - -my %fc_opts = ( 'no-ignore-externals' => \$_no_ignore_ext, -		'branch|b=s' => \@_branch_from, -		'follow-parent|follow' => \$_follow_parent, -		'branch-all-refs|B' => \$_branch_all_refs, +$sha1 = qr/[a-f\d]{40}/; +$sha1_short = qr/[a-f\d]{4,40}/; +my ($_stdin, $_help, $_edit, +	$_message, $_file, +	$_template, $_shared, +	$_version, $_fetch_all, +	$_merge, $_strategy, $_dry_run, +	$_prefix, $_no_checkout, $_verbose); +$Git::SVN::_follow_parent = 1; +my %remote_opts = ( 'username=s' => \$Git::SVN::Prompt::_username, +                    'config-dir=s' => \$Git::SVN::Ra::config_dir, +                    'no-auth-cache' => \$Git::SVN::Prompt::_no_auth_cache ); +my %fc_opts = ( 'follow-parent|follow!' => \$Git::SVN::_follow_parent,  		'authors-file|A=s' => \$_authors, -		'repack:i' => \$_repack, -		'no-metadata' => \$_no_metadata, +		'repack:i' => \$Git::SVN::_repack, +		'noMetadata' => \$Git::SVN::_no_metadata, +		'useSvmProps' => \$Git::SVN::_use_svm_props, +		'useSvnsyncProps' => \$Git::SVN::_use_svnsync_props, +		'log-window-size=i' => \$Git::SVN::Ra::_log_window_size, +		'no-checkout' => \$_no_checkout,  		'quiet|q' => \$_q, -		'username=s' => \$_username, -		'config-dir=s' => \$_config_dir, -		'no-auth-cache' => \$_no_auth_cache, -		'ignore-nodate' => \$_ignore_nodate, -		'repack-flags|repack-args|repack-opts=s' => \$_repack_flags); +		'repack-flags|repack-args|repack-opts=s' => +		   \$Git::SVN::_repack_flags, +		%remote_opts );  my ($_trunk, $_tags, $_branches); -my %multi_opts = ( 'trunk|T=s' => \$_trunk, -		'tags|t=s' => \$_tags, -		'branches|b=s' => \$_branches ); -my %init_opts = ( 'template=s' => \$_template, 'shared' => \$_shared ); +my %icv; +my %init_opts = ( 'template=s' => \$_template, 'shared:s' => \$_shared, +                  'trunk|T=s' => \$_trunk, 'tags|t=s' => \$_tags, +                  'branches|b=s' => \$_branches, 'prefix=s' => \$_prefix, +		  'no-metadata' => sub { $icv{noMetadata} = 1 }, +		  'use-svm-props' => sub { $icv{useSvmProps} = 1 }, +		  'use-svnsync-props' => sub { $icv{useSvnsyncProps} = 1 }, +		  'rewrite-root=s' => sub { $icv{rewriteRoot} = $_[1] }, +                  %remote_opts );  my %cmt_opts = ( 'edit|e' => \$_edit, -		'rmdir' => \$_rmdir, -		'find-copies-harder' => \$_find_copies_harder, -		'l=i' => \$_l, -		'copy-similarity|C=i'=> \$_cp_similarity +		'rmdir' => \$SVN::Git::Editor::_rmdir, +		'find-copies-harder' => \$SVN::Git::Editor::_find_copies_harder, +		'l=i' => \$SVN::Git::Editor::_rename_limit, +		'copy-similarity|C=i'=> \$SVN::Git::Editor::_cp_similarity  );  my %cmd = (  	fetch => [ \&cmd_fetch, "Download new revisions from SVN", -			{ 'revision|r=s' => \$_revision, %fc_opts } ], -	init => [ \&init, "Initialize a repo for tracking" . +			{ 'revision|r=s' => \$_revision, +			  'fetch-all|all' => \$_fetch_all, +			   %fc_opts } ], +	clone => [ \&cmd_clone, "Initialize and fetch revisions", +			{ 'revision|r=s' => \$_revision, +			   %fc_opts, %init_opts } ], +	init => [ \&cmd_init, "Initialize a repo for tracking" .  			  " (requires URL argument)",  			  \%init_opts ], -	dcommit => [ \&dcommit, 'Commit several diffs to merge with upstream', +	'multi-init' => [ \&cmd_multi_init, +	                  "Deprecated alias for ". +			  "'$0 init -T<trunk> -b<branches> -t<tags>'", +			  \%init_opts ], +	dcommit => [ \&cmd_dcommit, +	             'Commit several diffs to merge with upstream',  			{ 'merge|m|M' => \$_merge,  			  'strategy|s=s' => \$_strategy, +			  'verbose|v' => \$_verbose,  			  'dry-run|n' => \$_dry_run, +			  'fetch-all|all' => \$_fetch_all,  			%cmt_opts, %fc_opts } ], -	'set-tree' => [ \&commit, "Set an SVN repository to a git tree-ish", -			{	'stdin|' => \$_stdin, %cmt_opts, %fc_opts, } ], -	'show-ignore' => [ \&show_ignore, "Show svn:ignore listings", +	'set-tree' => [ \&cmd_set_tree, +	                "Set an SVN repository to a git tree-ish", +			{ 'stdin|' => \$_stdin, %cmt_opts, %fc_opts, } ], +	'show-ignore' => [ \&cmd_show_ignore, "Show svn:ignore listings",  			{ 'revision|r=i' => \$_revision } ], -	rebuild => [ \&rebuild, "Rebuild git-svn metadata (after git clone)", -			{ 'no-ignore-externals' => \$_no_ignore_ext, -			  'copy-remote|remote=s' => \$_cp_remote, -			  'upgrade' => \$_upgrade } ], -	'graft-branches' => [ \&graft_branches, -			'Detect merges/branches from already imported history', -			{ 'merge-rx|m' => \@_opt_m, -			  'branch|b=s' => \@_branch_from, -			  'branch-all-refs|B' => \$_branch_all_refs, -			  'no-default-regex' => \$_no_default_regex, -			  'no-graft-copy' => \$_no_graft_copy } ], -	'multi-init' => [ \&multi_init, -			'Initialize multiple trees (like git-svnimport)', -			{ %multi_opts, %init_opts, -			 'revision|r=i' => \$_revision, -			 'username=s' => \$_username, -			 'config-dir=s' => \$_config_dir, -			 'no-auth-cache' => \$_no_auth_cache, -			 'prefix=s' => \$_prefix, -			} ], -	'multi-fetch' => [ \&multi_fetch, -			'Fetch multiple trees (like git-svnimport)', -			\%fc_opts ], -	'log' => [ \&show_log, 'Show commit logs', -			{ 'limit=i' => \$_limit, +	'multi-fetch' => [ \&cmd_multi_fetch, +	                   "Deprecated alias for $0 fetch --all", +			   { 'revision|r=s' => \$_revision, %fc_opts } ], +	'migrate' => [ sub { }, +	               # no-op, we automatically run this anyways, +	               'Migrate configuration/metadata/layout from +		        previous versions of git-svn', +                       { 'minimize' => \$Git::SVN::Migration::_minimize, +			 %remote_opts } ], +	'log' => [ \&Git::SVN::Log::cmd_show_log, 'Show commit logs', +			{ 'limit=i' => \$Git::SVN::Log::limit,  			  'revision|r=s' => \$_revision, -			  'verbose|v' => \$_verbose, -			  'incremental' => \$_incremental, -			  'oneline' => \$_oneline, -			  'show-commit' => \$_show_commit, -			  'non-recursive' => \$_non_recursive, +			  'verbose|v' => \$Git::SVN::Log::verbose, +			  'incremental' => \$Git::SVN::Log::incremental, +			  'oneline' => \$Git::SVN::Log::oneline, +			  'show-commit' => \$Git::SVN::Log::show_commit, +			  'non-recursive' => \$Git::SVN::Log::non_recursive,  			  'authors-file|A=s' => \$_authors, -			  'color' => \$_color, -			  'pager=s' => \$_pager, +			  'color' => \$Git::SVN::Log::color, +			  'pager=s' => \$Git::SVN::Log::pager,  			} ], -	'commit-diff' => [ \&commit_diff, 'Commit a diff between two trees', +	'rebase' => [ \&cmd_rebase, "Fetch and rebase your working directory", +			{ 'merge|m|M' => \$_merge, +			  'verbose|v' => \$_verbose, +			  'strategy|s=s' => \$_strategy, +			  'fetch-all|all' => \$_fetch_all, +			  %fc_opts } ], +	'commit-diff' => [ \&cmd_commit_diff, +	                   'Commit a diff between two trees',  			{ 'message|m=s' => \$_message,  			  'file|F=s' => \$_file,  			  'revision|r=s' => \$_revision, @@ -170,20 +167,50 @@ for (my $i = 0; $i < @ARGV; $i++) {  my %opts = %{$cmd{$cmd}->[2]} if (defined $cmd);  read_repo_config(\%opts); -my $rv = GetOptions(%opts, 'help|H|h' => \$_help, -				'version|V' => \$_version, -				'id|i=s' => \$GIT_SVN); +my $rv = GetOptions(%opts, 'help|H|h' => \$_help, 'version|V' => \$_version, +                    'minimize-connections' => \$Git::SVN::Migration::_minimize, +                    'id|i=s' => \$Git::SVN::default_ref_id, +                    'svn-remote|remote|R=s' => sub { +                       $Git::SVN::no_reuse_existing = 1; +                       $Git::SVN::default_repo_id = $_[1] });  exit 1 if (!$rv && $cmd ne 'log'); -set_default_vals();  usage(0) if $_help;  version() if $_version;  usage(1) unless defined $cmd; -init_vars();  load_authors() if $_authors; -load_all_refs() if $_branch_all_refs; -migration_check() unless $cmd =~ /^(?:init|rebuild|multi-init|commit-diff)$/; -$cmd{$cmd}->[0]->(@ARGV); + +# make sure we're always running +unless ($cmd =~ /(?:clone|init|multi-init)$/) { +	unless (-d $ENV{GIT_DIR}) { +		if ($git_dir_user_set) { +			die "GIT_DIR=$ENV{GIT_DIR} explicitly set, ", +			    "but it is not a directory\n"; +		} +		my $git_dir = delete $ENV{GIT_DIR}; +		chomp(my $cdup = command_oneline(qw/rev-parse --show-cdup/)); +		unless (length $cdup) { +			die "Already at toplevel, but $git_dir ", +			    "not found '$cdup'\n"; +		} +		chdir $cdup or die "Unable to chdir up to '$cdup'\n"; +		unless (-d $git_dir) { +			die "$git_dir still not found after going to ", +			    "'$cdup'\n"; +		} +		$ENV{GIT_DIR} = $git_dir; +	} +} +unless ($cmd =~ /^(?:clone|init|multi-init|commit-diff)$/) { +	Git::SVN::Migration::migration_check(); +} +Git::SVN::init_vars(); +eval { +	Git::SVN::verify_remotes_sanity(); +	$cmd{$cmd}->[0]->(@ARGV); +}; +fatal $@ if $@; +post_fetch_checkout();  exit 0;  ####################### primary functions ###################### @@ -198,6 +225,7 @@ Usage: $0 <command> [options] [arguments]\n  	foreach (sort keys %cmd) {  		next if $cmd && $cmd ne $_; +		next if /^multi-/; # don't show deprecated commands  		print $fd '  ',pack('A17',$_),$cmd{$_}->[1],"\n";  		foreach (keys %{$cmd{$_}->[2]}) {  			# prints out arguments as they should be passed: @@ -221,173 +249,79 @@ sub version {  	exit 0;  } -sub rebuild { -	if (!verify_ref("refs/remotes/$GIT_SVN^0")) { -		copy_remote_ref(); -	} -	$SVN_URL = shift or undef; -	my $newest_rev = 0; -	if ($_upgrade) { -		command_noisy('update-ref',"refs/remotes/$GIT_SVN"," -		              $GIT_SVN-HEAD"); -	} else { -		check_upgrade_needed(); -	} - -	my ($rev_list, $ctx) = command_output_pipe("rev-list", -	                                           "refs/remotes/$GIT_SVN"); -	my $latest; -	while (<$rev_list>) { -		chomp; -		my $c = $_; -		croak "Non-SHA1: $c\n" unless $c =~ /^$sha1$/o; -		my @commit = grep(/^git-svn-id: /, -		                  command(qw/cat-file commit/, $c)); -		next if (!@commit); # skip merges -		my ($url, $rev, $uuid) = extract_metadata($commit[$#commit]); -		if (!defined $rev || !$uuid) { -			croak "Unable to extract revision or UUID from ", -				"$c, $commit[$#commit]\n"; -		} - -		# if we merged or otherwise started elsewhere, this is -		# how we break out of it -		next if (defined $SVN_UUID && ($uuid ne $SVN_UUID)); -		next if (defined $SVN_URL && defined $url && ($url ne $SVN_URL)); - -		unless (defined $latest) { -			if (!$SVN_URL && !$url) { -				croak "SVN repository location required: $url\n"; -			} -			$SVN_URL ||= $url; -			$SVN_UUID ||= $uuid; -			setup_git_svn(); -			$latest = $rev; -		} -		revdb_set($REVDB, $rev, $c); -		print "r$rev = $c\n"; -		$newest_rev = $rev if ($rev > $newest_rev); -	} -	command_close_pipe($rev_list, $ctx); -} - -sub init { -	my $url = shift or die "SVN repository location required " . -				"as a command-line argument\n"; -	$url =~ s!/+$!!; # strip trailing slash - -	if (my $repo_path = shift) { -		unless (-d $repo_path) { -			mkpath([$repo_path]); -		} -		$GIT_DIR = $ENV{GIT_DIR} = $repo_path . "/.git"; -		init_vars(); -	} - -	$SVN_URL = $url; -	unless (-d $GIT_DIR) { +sub do_git_init_db { +	unless (-d $ENV{GIT_DIR}) {  		my @init_db = ('init');  		push @init_db, "--template=$_template" if defined $_template; -		push @init_db, "--shared" if defined $_shared; +		if (defined $_shared) { +			if ($_shared =~ /[a-z]/) { +				push @init_db, "--shared=$_shared"; +			} else { +				push @init_db, "--shared"; +			} +		}  		command_noisy(@init_db);  	} -	setup_git_svn(); +	my $set; +	my $pfx = "svn-remote.$Git::SVN::default_repo_id"; +	foreach my $i (keys %icv) { +		die "'$set' and '$i' cannot both be set\n" if $set; +		next unless defined $icv{$i}; +		command_noisy('config', "$pfx.$i", $icv{$i}); +		$set = $i; +	}  } -sub cmd_fetch { -	fetch_child_id($GIT_SVN, @_); +sub init_subdir { +	my $repo_path = shift or return; +	mkpath([$repo_path]) unless -d $repo_path; +	chdir $repo_path or die "Couldn't chdir to $repo_path: $!\n"; +	$ENV{GIT_DIR} = '.git';  } -sub fetch { -	check_upgrade_needed(); -	$SVN_URL ||= file_to_s("$GIT_SVN_DIR/info/url"); -	my $ret = fetch_lib(@_); -	if ($ret->{commit} && !verify_ref('refs/heads/master^0')) { -		command_noisy(qw(update-ref refs/heads/master),$ret->{commit}); +sub cmd_clone { +	my ($url, $path) = @_; +	if (!defined $path && +	    (defined $_trunk || defined $_branches || defined $_tags) && +	    $url !~ m#^[a-z\+]+://#) { +		$path = $url;  	} -	return $ret; +	$path = basename($url) if !defined $path || !length $path; +	cmd_init($url, $path); +	Git::SVN::fetch_all($Git::SVN::default_repo_id);  } -sub fetch_lib { -	my (@parents) = @_; -	$SVN_URL ||= file_to_s("$GIT_SVN_DIR/info/url"); -	$SVN ||= libsvn_connect($SVN_URL); -	my ($last_rev, $last_commit) = svn_grab_base_rev(); -	my ($base, $head) = libsvn_parse_revision($last_rev); -	if ($base > $head) { -		return { revision => $last_rev, commit => $last_commit } +sub cmd_init { +	if (defined $_trunk || defined $_branches || defined $_tags) { +		return cmd_multi_init(@_);  	} -	my $index = set_index($GIT_SVN_INDEX); +	my $url = shift or die "SVN repository location required ", +	                       "as a command-line argument\n"; +	init_subdir(@_); +	do_git_init_db(); -	# limit ourselves and also fork() since get_log won't release memory -	# after processing a revision and SVN stuff seems to leak -	my $inc = 1000; -	my ($min, $max) = ($base, $head < $base+$inc ? $head : $base+$inc); -	read_uuid(); -	if (defined $last_commit) { -		unless (-e $GIT_SVN_INDEX) { -			command_noisy('read-tree', $last_commit); -		} -		my $x = command_oneline('write-tree'); -		my ($y) = (command(qw/cat-file commit/, $last_commit) -							=~ /^tree ($sha1)/m); -		if ($y ne $x) { -			unlink $GIT_SVN_INDEX or croak $!; -			command_noisy('read-tree', $last_commit); -		} -		$x = command_oneline('write-tree'); -		if ($y ne $x) { -			print STDERR "trees ($last_commit) $y != $x\n", -				 "Something is seriously wrong...\n"; -		} +	Git::SVN->init($url); +} + +sub cmd_fetch { +	if (grep /^\d+=./, @_) { +		die "'<rev>=<commit>' fetch arguments are ", +		    "no longer supported.\n";  	} -	while (1) { -		# fork, because using SVN::Pool with get_log() still doesn't -		# seem to help enough to keep memory usage down. -		defined(my $pid = fork) or croak $!; -		if (!$pid) { -			$SVN::Error::handler = \&libsvn_skip_unknown_revs; - -			# Yes I'm perfectly aware that the fourth argument -			# below is the limit revisions number.  Unfortunately -			# performance sucks with it enabled, so it's much -			# faster to fetch revision ranges instead of relying -			# on the limiter. -			libsvn_get_log(libsvn_dup_ra($SVN), [''], -					$min, $max, 0, 1, 1, -				sub { -					my $log_msg; -					if ($last_commit) { -						$log_msg = libsvn_fetch( -							$last_commit, @_); -						$last_commit = git_commit( -							$log_msg, -							$last_commit, -							@parents); -					} else { -						$log_msg = libsvn_new_tree(@_); -						$last_commit = git_commit( -							$log_msg, @parents); -					} -				}); -			exit 0; -		} -		waitpid $pid, 0; -		croak $? if $?; -		($last_rev, $last_commit) = svn_grab_base_rev(); -		last if ($max >= $head); -		$min = $max + 1; -		$max += $inc; -		$max = $head if ($max > $head); -		$SVN = libsvn_connect($SVN_URL); +	my ($remote) = @_; +	if (@_ > 1) { +		die "Usage: $0 fetch [--all] [svn-remote]\n"; +	} +	$remote ||= $Git::SVN::default_repo_id; +	if ($_fetch_all) { +		cmd_multi_fetch(); +	} else { +		Git::SVN::fetch_all($remote, Git::SVN::read_all_remotes());  	} -	restore_index($index); -	return { revision => $last_rev, commit => $last_commit };  } -sub commit { +sub cmd_set_tree {  	my (@commits) = @_; -	check_upgrade_needed();  	if ($_stdin || !@commits) {  		print "Reading from stdin...\n";  		@commits = (); @@ -405,702 +339,282 @@ sub commit {  		} elsif (scalar @tmp > 1) {  			push @revs, reverse(command('rev-list',@tmp));  		} else { -			die "Failed to rev-parse $c\n"; +			fatal "Failed to rev-parse $c\n";  		}  	} -	commit_lib(@revs); +	my $gs = Git::SVN->new; +	my ($r_last, $cmt_last) = $gs->last_rev_commit; +	$gs->fetch; +	if (defined $gs->{last_rev} && $r_last != $gs->{last_rev}) { +		fatal "There are new revisions that were fetched ", +		      "and need to be merged (or acknowledged) ", +		      "before committing.\nlast rev: $r_last\n", +		      " current: $gs->{last_rev}\n"; +	} +	$gs->set_tree($_) foreach @revs;  	print "Done committing ",scalar @revs," revisions to SVN\n";  } -sub commit_lib { -	my (@revs) = @_; -	my ($r_last, $cmt_last) = svn_grab_base_rev(); -	defined $r_last or die "Must have an existing revision to commit\n"; -	my $fetched = fetch(); -	if ($r_last != $fetched->{revision}) { -		print STDERR "There are new revisions that were fetched ", -				"and need to be merged (or acknowledged) ", -				"before committing.\n", -				"last rev: $r_last\n", -				" current: $fetched->{revision}\n"; -		exit 1; +sub cmd_dcommit { +	my $head = shift; +	$head ||= 'HEAD'; +	my @refs; +	my ($url, $rev, $uuid) = working_head_info($head, \@refs); +	my $c = $refs[-1]; +	unless (defined $url && defined $rev && defined $uuid) { +		die "Unable to determine upstream SVN information from ", +		    "$head history\n";  	} -	read_uuid(); -	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef, 0) : (); -	my $commit_msg = "$GIT_SVN_DIR/.svn-commit.tmp.$$"; - -	my $repo; -	set_svn_commit_env(); -	foreach my $c (@revs) { -		my $log_msg = get_commit_message($c, $commit_msg); - -		# fork for each commit because there's a memory leak I -		# can't track down... (it's probably in the SVN code) -		defined(my $pid = open my $fh, '-|') or croak $!; -		if (!$pid) { -			my $ed = SVN::Git::Editor->new( -					{	r => $r_last, -						ra => libsvn_dup_ra($SVN), -						c => $c, -						svn_path => $SVN->{svn_path}, -					}, -					$SVN->get_commit_editor( -						$log_msg->{msg}, -						sub { -							libsvn_commit_cb( -								@_, $c, -								$log_msg->{msg}, -								$r_last, -								$cmt_last) -						}, -						@lock) -					); -			my $mods = libsvn_checkout_tree($cmt_last, $c, $ed); -			if (@$mods == 0) { -				print "No changes\nr$r_last = $cmt_last\n"; -				$ed->abort_edit; -			} else { -				$ed->close_edit; -			} -			exit 0; -		} -		my ($r_new, $cmt_new, $no); -		while (<$fh>) { -			print $_; -			chomp; -			if (/^r(\d+) = ($sha1)$/o) { -				($r_new, $cmt_new) = ($1, $2); -			} elsif ($_ eq 'No changes') { -				$no = 1; -			} -		} -		close $fh or exit 1; -		if (! defined $r_new && ! defined $cmt_new) { -			unless ($no) { -				die "Failed to parse revision information\n"; -			} -		} else { -			($r_last, $cmt_last) = ($r_new, $cmt_new); -		} -	} -	$ENV{LC_ALL} = 'C'; -	unlink $commit_msg; -} - -sub dcommit { -	my $head = shift || 'HEAD'; -	my $gs = "refs/remotes/$GIT_SVN"; -	my @refs = command(qw/rev-list --no-merges/, "$gs..$head"); +	my $gs = Git::SVN->find_by_url($url);  	my $last_rev; -	foreach my $d (reverse @refs) { +	foreach my $d (@refs) {  		if (!verify_ref("$d~1")) { -			die "Commit $d\n", -			    "has no parent commit, and therefore ", -			    "nothing to diff against.\n", -			    "You should be working from a repository ", -			    "originally created by git-svn\n"; +			fatal "Commit $d\n", +			      "has no parent commit, and therefore ", +			      "nothing to diff against.\n", +			      "You should be working from a repository ", +			      "originally created by git-svn\n";  		}  		unless (defined $last_rev) {  			(undef, $last_rev, undef) = cmt_metadata("$d~1");  			unless (defined $last_rev) { -				die "Unable to extract revision information ", -				    "from commit $d~1\n"; +				fatal "Unable to extract revision information ", +				      "from commit $d~1\n";  			}  		}  		if ($_dry_run) {  			print "diff-tree $d~1 $d\n";  		} else { -			if (my $r = commit_diff("$d~1", $d, undef, $last_rev)) { -				$last_rev = $r; -			} # else: no changes, same $last_rev +			my %ed_opts = ( r => $last_rev, +			                log => get_commit_entry($d)->{log}, +			                ra => Git::SVN::Ra->new($url), +			                tree_a => "$d~1", +			                tree_b => $d, +			                editor_cb => sub { +			                       print "Committed r$_[0]\n"; +			                       $last_rev = $_[0]; }, +			                svn_path => ''); +			if (!SVN::Git::Editor->new(\%ed_opts)->apply_diff) { +				print "No changes\n$d~1 == $d\n"; +			}  		}  	}  	return if $_dry_run; -	fetch(); -	my @diff = command('diff-tree', 'HEAD', $gs, '--'); +	unless ($gs) { +		warn "Could not determine fetch information for $url\n", +		     "Will not attempt to fetch and rebase commits.\n", +		     "This probably means you have useSvmProps and should\n", +		     "now resync your SVN::Mirror repository.\n"; +		return; +	} +	$_fetch_all ? $gs->fetch_all : $gs->fetch; +	# we always want to rebase against the current HEAD, not any +	# head that was passed to us +	my @diff = command('diff-tree', 'HEAD', $gs->refname, '--');  	my @finish;  	if (@diff) { -		@finish = qw/rebase/; -		push @finish, qw/--merge/ if $_merge; -		push @finish, "--strategy=$_strategy" if $_strategy; -		print STDERR "W: HEAD and $gs differ, using @finish:\n", @diff; +		@finish = rebase_cmd(); +		print STDERR "W: HEAD and ", $gs->refname, " differ, ", +		             "using @finish:\n", "@diff";  	} else { -		print "No changes between current HEAD and $gs\n", -		      "Resetting to the latest $gs\n"; +		print "No changes between current HEAD and ", +		      $gs->refname, "\nResetting to the latest ", +		      $gs->refname, "\n";  		@finish = qw/reset --mixed/;  	} -	command_noisy(@finish, $gs); -} - -sub show_ignore { -	$SVN_URL ||= file_to_s("$GIT_SVN_DIR/info/url"); -	my $repo; -	$SVN ||= libsvn_connect($SVN_URL); -	my $r = defined $_revision ? $_revision : $SVN->get_latest_revnum; -	libsvn_traverse_ignore(\*STDOUT, '', $r); +	command_noisy(@finish, $gs->refname);  } -sub graft_branches { -	my $gr_file = "$GIT_DIR/info/grafts"; -	my ($grafts, $comments) = read_grafts($gr_file); -	my $gr_sha1; - -	if (%$grafts) { -		# temporarily disable our grafts file to make this idempotent -		chomp($gr_sha1 = command(qw/hash-object -w/,$gr_file)); -		rename $gr_file, "$gr_file~$gr_sha1" or croak $!; +sub cmd_rebase { +	command_noisy(qw/update-index --refresh/); +	my $url = (working_head_info('HEAD'))[0]; +	if (!defined $url) { +		die "Unable to determine upstream SVN information from ", +		    "working tree history\n";  	} -	my $l_map = read_url_paths(); -	my @re = map { qr/$_/is } @_opt_m if @_opt_m; -	unless ($_no_default_regex) { -		push @re, (qr/\b(?:merge|merging|merged)\s+with\s+([\w\.\-]+)/i, -			qr/\b(?:merge|merging|merged)\s+([\w\.\-]+)/i, -			qr/\b(?:from|of)\s+([\w\.\-]+)/i ); -	} -	foreach my $u (keys %$l_map) { -		if (@re) { -			foreach my $p (keys %{$l_map->{$u}}) { -				graft_merge_msg($grafts,$l_map,$u,$p,@re); -			} -		} -		unless ($_no_graft_copy) { -			graft_file_copy_lib($grafts,$l_map,$u); -		} +	my $gs = Git::SVN->find_by_url($url); +	if (command(qw/diff-index HEAD --/)) { +		print STDERR "Cannot rebase with uncommited changes:\n"; +		command_noisy('status'); +		exit 1;  	} -	graft_tree_joins($grafts); +	$_fetch_all ? $gs->fetch_all : $gs->fetch; +	command_noisy(rebase_cmd(), $gs->refname); +} -	write_grafts($grafts, $comments, $gr_file); -	unlink "$gr_file~$gr_sha1" if $gr_sha1; +sub cmd_show_ignore { +	my $url = (::working_head_info('HEAD'))[0]; +	my $gs = Git::SVN->find_by_url($url) || Git::SVN->new; +	my $r = (defined $_revision ? $_revision : $gs->ra->get_latest_revnum); +	$gs->traverse_ignore(\*STDOUT, '', $r);  } -sub multi_init { +sub cmd_multi_init {  	my $url = shift;  	unless (defined $_trunk || defined $_branches || defined $_tags) {  		usage(1);  	} +	$_prefix = '' unless defined $_prefix; +	if (defined $url) { +		$url =~ s#/+$##; +		init_subdir(@_); +	} +	do_git_init_db();  	if (defined $_trunk) { -		my $trunk_url = complete_svn_url($url, $_trunk); -		my $ch_id; -		if ($GIT_SVN eq 'git-svn') { -			$ch_id = 1; -			$GIT_SVN = $ENV{GIT_SVN_ID} = 'trunk'; -		} -		init_vars(); -		unless (-d $GIT_SVN_DIR) { -			if ($ch_id) { -				print "GIT_SVN_ID set to 'trunk' for ", -				      "$trunk_url ($_trunk)\n"; -			} -			init($trunk_url); -			command_noisy('config', 'svn.trunk', $trunk_url); +		my $trunk_ref = $_prefix . 'trunk'; +		# try both old-style and new-style lookups: +		my $gs_trunk = eval { Git::SVN->new($trunk_ref) }; +		unless ($gs_trunk) { +			my ($trunk_url, $trunk_path) = +			                      complete_svn_url($url, $_trunk); +			$gs_trunk = Git::SVN->init($trunk_url, $trunk_path, +						   undef, $trunk_ref);  		}  	} -	$_prefix = '' unless defined $_prefix; -	complete_url_ls_init($url, $_branches, '--branches/-b', $_prefix); -	complete_url_ls_init($url, $_tags, '--tags/-t', $_prefix . 'tags/'); -} - -sub multi_fetch { -	# try to do trunk first, since branches/tags -	# may be descended from it. -	if (-e "$GIT_DIR/svn/trunk/info/url") { -		fetch_child_id('trunk', @_); -	} -	rec_fetch('', "$GIT_DIR/svn", @_); +	return unless defined $_branches || defined $_tags; +	my $ra = $url ? Git::SVN::Ra->new($url) : undef; +	complete_url_ls_init($ra, $_branches, '--branches/-b', $_prefix); +	complete_url_ls_init($ra, $_tags, '--tags/-t', $_prefix . 'tags/');  } -sub show_log { -	my (@args) = @_; -	my ($r_min, $r_max); -	my $r_last = -1; # prevent dupes -	rload_authors() if $_authors; -	if (defined $TZ) { -		$ENV{TZ} = $TZ; -	} else { -		delete $ENV{TZ}; -	} -	if (defined $_revision) { -		if ($_revision =~ /^(\d+):(\d+)$/) { -			($r_min, $r_max) = ($1, $2); -		} elsif ($_revision =~ /^\d+$/) { -			$r_min = $r_max = $_revision; -		} else { -			print STDERR "-r$_revision is not supported, use ", -				"standard \'git log\' arguments instead\n"; -			exit 1; +sub cmd_multi_fetch { +	my $remotes = Git::SVN::read_all_remotes(); +	foreach my $repo_id (sort keys %$remotes) { +		if ($remotes->{$repo_id}->{url}) { +			Git::SVN::fetch_all($repo_id, $remotes);  		}  	} - -	config_pager(); -	@args = (git_svn_log_cmd($r_min, $r_max), @args); -	my $log = command_output_pipe(@args); -	run_pager(); -	my (@k, $c, $d); - -	while (<$log>) { -		if (/^${_esc_color}commit ($sha1_short)/o) { -			my $cmt = $1; -			if ($c && cmt_showable($c) && $c->{r} != $r_last) { -				$r_last = $c->{r}; -				process_commit($c, $r_min, $r_max, \@k) or -								goto out; -			} -			$d = undef; -			$c = { c => $cmt }; -		} elsif (/^${_esc_color}author (.+) (\d+) ([\-\+]?\d+)$/) { -			get_author_info($c, $1, $2, $3); -		} elsif (/^${_esc_color}(?:tree|parent|committer) /) { -			# ignore -		} elsif (/^${_esc_color}:\d{6} \d{6} $sha1_short/o) { -			push @{$c->{raw}}, $_; -		} elsif (/^${_esc_color}[ACRMDT]\t/) { -			# we could add $SVN->{svn_path} here, but that requires -			# remote access at the moment (repo_path_split)... -			s#^(${_esc_color})([ACRMDT])\t#$1   $2 #; -			push @{$c->{changed}}, $_; -		} elsif (/^${_esc_color}diff /) { -			$d = 1; -			push @{$c->{diff}}, $_; -		} elsif ($d) { -			push @{$c->{diff}}, $_; -		} elsif (/^${_esc_color}    (git-svn-id:.+)$/) { -			($c->{url}, $c->{r}, undef) = extract_metadata($1); -		} elsif (s/^${_esc_color}    //) { -			push @{$c->{l}}, $_; -		} -	} -	if ($c && defined $c->{r} && $c->{r} != $r_last) { -		$r_last = $c->{r}; -		process_commit($c, $r_min, $r_max, \@k); -	} -	if (@k) { -		my $swap = $r_max; -		$r_max = $r_min; -		$r_min = $swap; -		process_commit($_, $r_min, $r_max) foreach reverse @k; -	} -out: -	close $log; -	print '-' x72,"\n" unless $_incremental || $_oneline; -} - -sub commit_diff_usage { -	print STDERR "Usage: $0 commit-diff <tree-ish> <tree-ish> [<URL>]\n"; -	exit 1  } -sub commit_diff { -	my $ta = shift or commit_diff_usage(); -	my $tb = shift or commit_diff_usage(); -	if (!eval { $SVN_URL = shift || file_to_s("$GIT_SVN_DIR/info/url") }) { -		print STDERR "Needed URL or usable git-svn id command-line\n"; -		commit_diff_usage(); -	} -	my $r = shift; -	unless (defined $r) { -		if (defined $_revision) { -			$r = $_revision -		} else { -			die "-r|--revision is a required argument\n"; +# this command is special because it requires no metadata +sub cmd_commit_diff { +	my ($ta, $tb, $url) = @_; +	my $usage = "Usage: $0 commit-diff -r<revision> ". +	            "<tree-ish> <tree-ish> [<URL>]\n"; +	fatal($usage) if (!defined $ta || !defined $tb); +	my $svn_path; +	if (!defined $url) { +		my $gs = eval { Git::SVN->new }; +		if (!$gs) { +			fatal("Needed URL or usable git-svn --id in ", +			      "the command-line\n", $usage);  		} +		$url = $gs->{url}; +		$svn_path = $gs->{path}; +	} +	unless (defined $_revision) { +		fatal("-r|--revision is a required argument\n", $usage);  	}  	if (defined $_message && defined $_file) { -		print STDERR "Both --message/-m and --file/-F specified ", -				"for the commit message.\n", -				"I have no idea what you mean\n"; -		exit 1; +		fatal("Both --message/-m and --file/-F specified ", +		      "for the commit message.\n", +		      "I have no idea what you mean\n");  	}  	if (defined $_file) {  		$_message = file_to_s($_file);  	} else { -		$_message ||= get_commit_message($tb, -					"$GIT_DIR/.svn-commit.tmp.$$")->{msg}; +		$_message ||= get_commit_entry($tb)->{log};  	} -	$SVN ||= libsvn_connect($SVN_URL); +	my $ra ||= Git::SVN::Ra->new($url); +	$svn_path ||= $ra->{svn_path}; +	my $r = $_revision;  	if ($r eq 'HEAD') { -		$r = $SVN->get_latest_revnum; +		$r = $ra->get_latest_revnum;  	} elsif ($r !~ /^\d+$/) {  		die "revision argument: $r not understood by git-svn\n";  	} -	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef, 0) : (); -	my $rev_committed; -	my $ed = SVN::Git::Editor->new({	r => $r, -						ra => libsvn_dup_ra($SVN), -						c => $tb, -						svn_path => $SVN->{svn_path} -					}, -				$SVN->get_commit_editor($_message, -					sub { -						$rev_committed = $_[0]; -						print "Committed $_[0]\n"; -					}, @lock) -				); -	eval { -		my $mods = libsvn_checkout_tree($ta, $tb, $ed); -		if (@$mods == 0) { -			print "No changes\n$ta == $tb\n"; -			$ed->abort_edit; -		} else { -			$ed->close_edit; -		} -	}; -	fatal "$@\n" if $@; -	$_message = $_file = undef; -	return $rev_committed; +	my %ed_opts = ( r => $r, +	                log => $_message, +	                ra => $ra, +	                tree_a => $ta, +	                tree_b => $tb, +	                editor_cb => sub { print "Committed r$_[0]\n" }, +	                svn_path => $svn_path ); +	if (!SVN::Git::Editor->new(\%ed_opts)->apply_diff) { +		print "No changes\n$ta == $tb\n"; +	}  }  ########################### utility functions ######################### -sub cmt_showable { -	my ($c) = @_; -	return 1 if defined $c->{r}; -	if ($c->{l} && $c->{l}->[-1] eq "...\n" && -				$c->{a_raw} =~ /\@([a-f\d\-]+)>$/) { -		my @msg = command(qw/cat-file commit/, $c->{c}); -		shift @msg while ($msg[0] ne "\n"); -		shift @msg; -		@{$c->{l}} = grep !/^git-svn-id: /, @msg; - -		(undef, $c->{r}, undef) = extract_metadata( -				(grep(/^git-svn-id: /, @msg))[-1]); -	} -	return defined $c->{r}; +sub rebase_cmd { +	my @cmd = qw/rebase/; +	push @cmd, '-v' if $_verbose; +	push @cmd, qw/--merge/ if $_merge; +	push @cmd, "--strategy=$_strategy" if $_strategy; +	@cmd;  } -sub log_use_color { -	return 1 if $_color; -	my ($dc, $dcvar); -	$dcvar = 'color.diff'; -	$dc = `git-config --get $dcvar`; -	if ($dc eq '') { -		# nothing at all; fallback to "diff.color" -		$dcvar = 'diff.color'; -		$dc = `git-config --get $dcvar`; -	} -	chomp($dc); -	if ($dc eq 'auto') { -		my $pc; -		$pc = `git-config --get color.pager`; -		if ($pc eq '') { -			# does not have it -- fallback to pager.color -			$pc = `git-config --bool --get pager.color`; -		} -		else { -			$pc = `git-config --bool --get color.pager`; -			if ($?) { -				$pc = 'false'; -			} -		} -		chomp($pc); -		if (-t *STDOUT || (defined $_pager && $pc eq 'true')) { -			return ($ENV{TERM} && $ENV{TERM} ne 'dumb'); -		} -		return 0; -	} -	return 0 if $dc eq 'never'; -	return 1 if $dc eq 'always'; -	chomp($dc = `git-config --bool --get $dcvar`); -	return ($dc eq 'true'); -} +sub post_fetch_checkout { +	return if $_no_checkout; +	my $gs = $Git::SVN::_head or return; +	return if verify_ref('refs/heads/master^0'); -sub git_svn_log_cmd { -	my ($r_min, $r_max) = @_; -	my @cmd = (qw/log --abbrev-commit --pretty=raw -			--default/, "refs/remotes/$GIT_SVN"); -	push @cmd, '-r' unless $_non_recursive; -	push @cmd, qw/--raw --name-status/ if $_verbose; -	push @cmd, '--color' if log_use_color(); -	return @cmd unless defined $r_max; -	if ($r_max == $r_min) { -		push @cmd, '--max-count=1'; -		if (my $c = revdb_get($REVDB, $r_max)) { -			push @cmd, $c; -		} -	} else { -		my ($c_min, $c_max); -		$c_max = revdb_get($REVDB, $r_max); -		$c_min = revdb_get($REVDB, $r_min); -		if (defined $c_min && defined $c_max) { -			if ($r_max > $r_max) { -				push @cmd, "$c_min..$c_max"; -			} else { -				push @cmd, "$c_max..$c_min"; -			} -		} elsif ($r_max > $r_min) { -			push @cmd, $c_max; -		} else { -			push @cmd, $c_min; -		} -	} -	return @cmd; -} +	my $valid_head = verify_ref('HEAD^0'); +	command_noisy(qw(update-ref refs/heads/master), $gs->refname); +	return if ($valid_head || !verify_ref('HEAD^0')); -sub fetch_child_id { -	my $id = shift; -	print "Fetching $id\n"; -	my $ref = "$GIT_DIR/refs/remotes/$id"; -	defined(my $pid = open my $fh, '-|') or croak $!; -	if (!$pid) { -		$GIT_SVN = $ENV{GIT_SVN_ID} = $id; -		init_vars(); -		fetch(@_); -		exit 0; -	} -	while (<$fh>) { -		print $_; -		check_repack() if (/^r\d+ = $sha1/o); -	} -	close $fh or croak $?; -} +	return if $ENV{GIT_DIR} !~ m#^(?:.*/)?\.git$#; +	my $index = $ENV{GIT_INDEX_FILE} || "$ENV{GIT_DIR}/index"; +	return if -f $index; -sub rec_fetch { -	my ($pfx, $p, @args) = @_; -	my @dir; -	foreach (sort <$p/*>) { -		if (-r "$_/info/url") { -			$pfx .= '/' if $pfx && $pfx !~ m!/$!; -			my $id = $pfx . basename $_; -			next if $id eq 'trunk'; -			fetch_child_id($id, @args); -		} elsif (-d $_) { -			push @dir, $_; -		} -	} -	foreach (@dir) { -		my $x = $_; -		$x =~ s!^\Q$GIT_DIR\E/svn/!!; -		rec_fetch($x, $_); -	} +	chomp(my $bare = `git config --bool --get core.bare`); +	return if $bare eq 'true'; +	return if command_oneline(qw/rev-parse --is-inside-git-dir/) eq 'true'; +	command_noisy(qw/read-tree -m -u -v HEAD HEAD/); +	print STDERR "Checked out HEAD:\n  ", +	             $gs->full_url, " r", $gs->last_rev, "\n";  }  sub complete_svn_url {  	my ($url, $path) = @_;  	$path =~ s#/+$##; -	$url =~ s#/+$## if $url;  	if ($path !~ m#^[a-z\+]+://#) { -		$path = '/' . $path if ($path !~ m#^/#);  		if (!defined $url || $url !~ m#^[a-z\+]+://#) {  			fatal("E: '$path' is not a complete URL ",  			      "and a separate URL is not specified\n");  		} -		$path = $url . $path; +		return ($url, $path);  	} -	return $path; +	return ($path, '');  }  sub complete_url_ls_init { -	my ($url, $path, $switch, $pfx) = @_; -	unless ($path) { +	my ($ra, $repo_path, $switch, $pfx) = @_; +	unless ($repo_path) {  		print STDERR "W: $switch not specified\n";  		return;  	} -	my $full_url = complete_svn_url($url, $path); -	my @ls = libsvn_ls_fullurl($full_url); -	defined(my $pid = fork) or croak $!; -	if (!$pid) { -		foreach my $u (map { "$full_url/$_" } (grep m!/$!, @ls)) { -			$u =~ s#/+$##; -			if ($u !~ m!\Q$full_url\E/(.+)$!) { -				print STDERR "W: Unrecognized URL: $u\n"; -				die "This should never happen\n"; -			} -			# don't try to init already existing refs -			my $id = $pfx.$1; -			$GIT_SVN = $ENV{GIT_SVN_ID} = $id; -			init_vars(); -			unless (-d $GIT_SVN_DIR) { -				print "init $u => $id\n"; -				init($u); -			} -		} -		exit 0; -	} -	waitpid $pid, 0; -	croak $? if $?; -	my ($n) = ($switch =~ /^--(\w+)/); -	command_noisy('config', "svn.$n", $full_url); -} - -sub common_prefix { -	my $paths = shift; -	my %common; -	foreach (@$paths) { -		my @tmp = split m#/#, $_; -		my $p = ''; -		while (my $x = shift @tmp) { -			$p .= "/$x"; -			$common{$p} ||= 0; -			$common{$p}++; -		} -	} -	foreach (sort {length $b <=> length $a} keys %common) { -		if ($common{$_} == @$paths) { -			return $_; -		} -	} -	return ''; -} - -# grafts set here are 'stronger' in that they're based on actual tree -# matches, and won't be deleted from merge-base checking in write_grafts() -sub graft_tree_joins { -	my $grafts = shift; -	map_tree_joins() if (@_branch_from && !%tree_map); -	return unless %tree_map; - -	git_svn_each(sub { -		my $i = shift; -		my @args = (qw/rev-list --pretty=raw/, "refs/remotes/$i"); -		my ($fh, $ctx) = command_output_pipe(@args); -		while (<$fh>) { -			next unless /^commit ($sha1)$/o; -			my $c = $1; -			my ($t) = (<$fh> =~ /^tree ($sha1)$/o); -			next unless $tree_map{$t}; - -			my $l; -			do { -				$l = readline $fh; -			} until ($l =~ /^committer (?:.+) (\d+) ([\-\+]?\d+)$/); - -			my ($s, $tz) = ($1, $2); -			if ($tz =~ s/^\+//) { -				$s += tz_to_s_offset($tz); -			} elsif ($tz =~ s/^\-//) { -				$s -= tz_to_s_offset($tz); -			} - -			my ($url_a, $r_a, $uuid_a) = cmt_metadata($c); - -			foreach my $p (@{$tree_map{$t}}) { -				next if $p eq $c; -				my $mb = eval { command('merge-base', $c, $p) }; -				next unless ($@ || $?); -				if (defined $r_a) { -					# see if SVN says it's a relative -					my ($url_b, $r_b, $uuid_b) = -							cmt_metadata($p); -					next if (defined $url_b && -							defined $url_a && -							($url_a eq $url_b) && -							($uuid_a eq $uuid_b)); -					if ($uuid_a eq $uuid_b) { -						if ($r_b < $r_a) { -							$grafts->{$c}->{$p} = 2; -							next; -						} elsif ($r_b > $r_a) { -							$grafts->{$p}->{$c} = 2; -							next; -						} -					} -				} -				my $ct = get_commit_time($p); -				if ($ct < $s) { -					$grafts->{$c}->{$p} = 2; -				} elsif ($ct > $s) { -					$grafts->{$p}->{$c} = 2; -				} -				# what should we do when $ct == $s ? -			} -		} -		command_close_pipe($fh, $ctx); -	}); -} - -sub graft_file_copy_lib { -	my ($grafts, $l_map, $u) = @_; -	my $tree_paths = $l_map->{$u}; -	my $pfx = common_prefix([keys %$tree_paths]); -	my ($repo, $path) = repo_path_split($u.$pfx); -	$SVN = libsvn_connect($repo); - -	my ($base, $head) = libsvn_parse_revision(); -	my $inc = 1000; -	my ($min, $max) = ($base, $head < $base+$inc ? $head : $base+$inc); -	my $eh = $SVN::Error::handler; -	$SVN::Error::handler = \&libsvn_skip_unknown_revs; -	while (1) { -		my $pool = SVN::Pool->new; -		libsvn_get_log(libsvn_dup_ra($SVN), [$path], -		               $min, $max, 0, 2, 1, -			sub { -				libsvn_graft_file_copies($grafts, $tree_paths, -							$path, @_); -			}, $pool); -		$pool->clear; -		last if ($max >= $head); -		$min = $max + 1; -		$max += $inc; -		$max = $head if ($max > $head); -	} -	$SVN::Error::handler = $eh; -} - -sub process_merge_msg_matches { -	my ($grafts, $l_map, $u, $p, $c, @matches) = @_; -	my (@strong, @weak); -	foreach (@matches) { -		# merging with ourselves is not interesting -		next if $_ eq $p; -		if ($l_map->{$u}->{$_}) { -			push @strong, $_; -		} else { -			push @weak, $_; -		} -	} -	foreach my $w (@weak) { -		last if @strong; -		# no exact match, use branch name as regexp. -		my $re = qr/\Q$w\E/i; -		foreach (keys %{$l_map->{$u}}) { -			if (/$re/) { -				push @strong, $l_map->{$u}->{$_}; -				last; -			} -		} -		last if @strong; -		$w = basename($w); -		$re = qr/\Q$w\E/i; -		foreach (keys %{$l_map->{$u}}) { -			if (/$re/) { -				push @strong, $l_map->{$u}->{$_}; -				last; -			} +	$repo_path =~ s#/+$##; +	if ($repo_path =~ m#^[a-z\+]+://#) { +		$ra = Git::SVN::Ra->new($repo_path); +		$repo_path = ''; +	} else { +		$repo_path =~ s#^/+##; +		unless ($ra) { +			fatal("E: '$repo_path' is not a complete URL ", +			      "and a separate URL is not specified\n");  		}  	} -	my ($rev) = ($c->{m} =~ /^git-svn-id:\s(?:\S+?)\@(\d+) -					\s(?:[a-f\d\-]+)$/xsm); -	unless (defined $rev) { -		($rev) = ($c->{m} =~/^git-svn-id:\s(\d+) -					\@(?:[a-f\d\-]+)/xsm); -		return unless defined $rev; -	} -	foreach my $m (@strong) { -		my ($r0, $s0) = find_rev_before($rev, $m, 1); -		$grafts->{$c->{c}}->{$s0} = 1 if defined $s0; +	my $url = $ra->{url}; +	my $gs = Git::SVN->init($url, undef, undef, undef, 1); +	my $k = "svn-remote.$gs->{repo_id}.url"; +	my $orig_url = eval { command_oneline(qw/config --get/, $k) }; +	if ($orig_url && ($orig_url ne $gs->{url})) { +		die "$k already set: $orig_url\n", +		    "wanted to set to: $gs->{url}\n";  	} -} - -sub graft_merge_msg { -	my ($grafts, $l_map, $u, $p, @re) = @_; - -	my $x = $l_map->{$u}->{$p}; -	my $rl = rev_list_raw("refs/remotes/$x"); -	while (my $c = next_rev_list_entry($rl)) { -		foreach my $re (@re) { -			my (@br) = ($c->{m} =~ /$re/g); -			next unless @br; -			process_merge_msg_matches($grafts,$l_map,$u,$p,$c,@br); -		} +	command_oneline('config', $k, $gs->{url}) unless $orig_url; +	my $remote_path = "$ra->{svn_path}/$repo_path/*"; +	$remote_path =~ s#/+#/#g; +	$remote_path =~ s#^/##g; +	my ($n) = ($switch =~ /^--(\w+)/); +	if (length $pfx && $pfx !~ m#/$#) { +		die "--prefix='$pfx' must have a trailing slash '/'\n";  	} -} - -sub read_uuid { -	return if $SVN_UUID; -	my $pool = SVN::Pool->new; -	$SVN_UUID = $SVN->get_uuid($pool); -	$pool->clear; +	command_noisy('config', "svn-remote.$gs->{repo_id}.$n", +				"$remote_path:refs/remotes/$pfx*");  }  sub verify_ref { @@ -1109,37 +623,9 @@ sub verify_ref {  	                       { STDERR => 0 }); };  } -sub repo_path_split { -	my $full_url = shift; -	$full_url =~ s#/+$##; - -	foreach (@repo_path_split_cache) { -		if ($full_url =~ s#$_##) { -			my $u = $1; -			$full_url =~ s#^/+##; -			return ($u, $full_url); -		} -	} -	my $tmp = libsvn_connect($full_url); -	return ($tmp->{repos_root}, $tmp->{svn_path}); -} - -sub setup_git_svn { -	defined $SVN_URL or croak "SVN repository location required\n"; -	unless (-d $GIT_DIR) { -		croak "GIT_DIR=$GIT_DIR does not exist!\n"; -	} -	mkpath([$GIT_SVN_DIR]); -	mkpath(["$GIT_SVN_DIR/info"]); -	open my $fh, '>>',$REVDB or croak $!; -	close $fh; -	s_to_file($SVN_URL,"$GIT_SVN_DIR/info/url"); - -} -  sub get_tree_from_treeish {  	my ($treeish) = @_; -	croak "Not a sha1: $treeish\n" unless $treeish =~ /^$sha1$/o; +	# $treeish can be a symbolic ref, too:  	my $type = command_oneline(qw/cat-file -t/, $treeish);  	my $expected;  	while ($type eq 'tag') { @@ -1148,7 +634,7 @@ sub get_tree_from_treeish {  	if ($type eq 'commit') {  		$expected = (grep /^tree /, command(qw/cat-file commit/,  		                                    $treeish))[0]; -		($expected) = ($expected =~ /^tree ($sha1)$/); +		($expected) = ($expected =~ /^tree ($sha1)$/o);  		die "Unable to get tree from $treeish\n" unless $expected;  	} elsif ($type eq 'tree') {  		$expected = $treeish; @@ -1158,146 +644,44 @@ sub get_tree_from_treeish {  	return $expected;  } -sub get_diff { -	my ($from, $treeish) = @_; -	print "diff-tree $from $treeish\n"; -	my @diff_tree = qw(diff-tree -z -r); -	if ($_cp_similarity) { -		push @diff_tree, "-C$_cp_similarity"; -	} else { -		push @diff_tree, '-C'; -	} -	push @diff_tree, '--find-copies-harder' if $_find_copies_harder; -	push @diff_tree, "-l$_l" if defined $_l; -	push @diff_tree, $from, $treeish; -	my ($diff_fh, $ctx) = command_output_pipe(@diff_tree); -	local $/ = "\0"; -	my $state = 'meta'; -	my @mods; -	while (<$diff_fh>) { -		chomp $_; # this gets rid of the trailing "\0" -		if ($state eq 'meta' && /^:(\d{6})\s(\d{6})\s -					$sha1\s($sha1)\s([MTCRAD])\d*$/xo) { -			push @mods, {	mode_a => $1, mode_b => $2, -					sha1_b => $3, chg => $4 }; -			if ($4 =~ /^(?:C|R)$/) { -				$state = 'file_a'; -			} else { -				$state = 'file_b'; -			} -		} elsif ($state eq 'file_a') { -			my $x = $mods[$#mods] or croak "Empty array\n"; -			if ($x->{chg} !~ /^(?:C|R)$/) { -				croak "Error parsing $_, $x->{chg}\n"; -			} -			$x->{file_a} = $_; -			$state = 'file_b'; -		} elsif ($state eq 'file_b') { -			my $x = $mods[$#mods] or croak "Empty array\n"; -			if (exists $x->{file_a} && $x->{chg} !~ /^(?:C|R)$/) { -				croak "Error parsing $_, $x->{chg}\n"; -			} -			if (!exists $x->{file_a} && $x->{chg} =~ /^(?:C|R)$/) { -				croak "Error parsing $_, $x->{chg}\n"; -			} -			$x->{file_b} = $_; -			$state = 'meta'; -		} else { -			croak "Error parsing $_\n"; -		} -	} -	command_close_pipe($diff_fh, $ctx); -	return \@mods; -} +sub get_commit_entry { +	my ($treeish) = shift; +	my %log_entry = ( log => '', tree => get_tree_from_treeish($treeish) ); +	my $commit_editmsg = "$ENV{GIT_DIR}/COMMIT_EDITMSG"; +	my $commit_msg = "$ENV{GIT_DIR}/COMMIT_MSG"; +	open my $log_fh, '>', $commit_editmsg or croak $!; -sub libsvn_checkout_tree { -	my ($from, $treeish, $ed) = @_; -	my $mods = get_diff($from, $treeish); -	return $mods unless (scalar @$mods); -	my %o = ( D => 1, R => 0, C => -1, A => 3, M => 3, T => 3 ); -	foreach my $m (sort { $o{$a->{chg}} <=> $o{$b->{chg}} } @$mods) { -		my $f = $m->{chg}; -		if (defined $o{$f}) { -			$ed->$f($m, $_q); -		} else { -			croak "Invalid change type: $f\n"; -		} -	} -	$ed->rmdirs($_q) if $_rmdir; -	return $mods; -} - -sub get_commit_message { -	my ($commit, $commit_msg) = (@_); -	my %log_msg = ( msg => '' ); -	open my $msg, '>', $commit_msg or croak $!; - -	my $type = command_oneline(qw/cat-file -t/, $commit); +	my $type = command_oneline(qw/cat-file -t/, $treeish);  	if ($type eq 'commit' || $type eq 'tag') {  		my ($msg_fh, $ctx) = command_output_pipe('cat-file', -		                                         $type, $commit); +		                                         $type, $treeish);  		my $in_msg = 0;  		while (<$msg_fh>) {  			if (!$in_msg) {  				$in_msg = 1 if (/^\s*$/);  			} elsif (/^git-svn-id: /) { -				# skip this, we regenerate the correct one -				# on re-fetch anyways +				# skip this for now, we regenerate the +				# correct one on re-fetch anyways +				# TODO: set *:merge properties or like...  			} else { -				print $msg $_ or croak $!; +				print $log_fh $_ or croak $!;  			}  		}  		command_close_pipe($msg_fh, $ctx);  	} -	close $msg or croak $!; +	close $log_fh or croak $!;  	if ($_edit || ($type eq 'tree')) {  		my $editor = $ENV{VISUAL} || $ENV{EDITOR} || 'vi'; -		system($editor, $commit_msg); -	} - -	# file_to_s removes all trailing newlines, so just use chomp() here: -	open $msg, '<', $commit_msg or croak $!; -	{ local $/; chomp($log_msg{msg} = <$msg>); } -	close $msg or croak $!; - -	return \%log_msg; -} - -sub set_svn_commit_env { -	if (defined $LC_ALL) { -		$ENV{LC_ALL} = $LC_ALL; -	} else { -		delete $ENV{LC_ALL}; -	} -} - -sub rev_list_raw { -	my ($fh, $c) = command_output_pipe(qw/rev-list --pretty=raw/, @_); -	return { fh => $fh, ctx => $c, t => { } }; -} - -sub next_rev_list_entry { -	my $rl = shift; -	my $fh = $rl->{fh}; -	my $x = $rl->{t}; -	while (<$fh>) { -		if (/^commit ($sha1)$/o) { -			if ($x->{c}) { -				$rl->{t} = { c => $1 }; -				return $x; -			} else { -				$x->{c} = $1; -			} -		} elsif (/^parent ($sha1)$/o) { -			$x->{p}->{$1} = 1; -		} elsif (s/^    //) { -			$x->{m} ||= ''; -			$x->{m} .= $_; -		} +		# TODO: strip out spaces, comments, like git-commit.sh +		system($editor, $commit_editmsg);  	} -	command_close_pipe($fh, $rl->{ctx}); -	return ($x != $rl->{t}) ? $x : undef; +	rename $commit_editmsg, $commit_msg or croak $!; +	open $log_fh, '<', $commit_msg or croak $!; +	{ local $/; chomp($log_entry{log} = <$log_fh>); } +	close $log_fh or croak $!; +	unlink $commit_msg; +	\%log_entry;  }  sub s_to_file { @@ -1318,601 +702,1409 @@ sub file_to_s {  	return $ret;  } -sub assert_revision_unknown { -	my $r = shift; -	if (my $c = revdb_get($REVDB, $r)) { -		croak "$r = $c already exists! Why are we refetching it?"; +# '<svn username> = real-name <email address>' mapping based on git-svnimport: +sub load_authors { +	open my $authors, '<', $_authors or die "Can't open $_authors $!\n"; +	my $log = $cmd eq 'log'; +	while (<$authors>) { +		chomp; +		next unless /^(\S+?|\(no author\))\s*=\s*(.+?)\s*<(.+)>\s*$/; +		my ($user, $name, $email) = ($1, $2, $3); +		if ($log) { +			$Git::SVN::Log::rusers{"$name <$email>"} = $user; +		} else { +			$users{$user} = [$name, $email]; +		}  	} +	close $authors or croak $!;  } -sub git_commit { -	my ($log_msg, @parents) = @_; -	assert_revision_unknown($log_msg->{revision}); -	map_tree_joins() if (@_branch_from && !%tree_map); - -	my (@tmp_parents, @exec_parents, %seen_parent); -	if (my $lparents = $log_msg->{parents}) { -		@tmp_parents = @$lparents -	} -	# commit parents can be conditionally bound to a particular -	# svn revision via: "svn_revno=commit_sha1", filter them out here: -	foreach my $p (@parents) { -		next unless defined $p; -		if ($p =~ /^(\d+)=($sha1_short)$/o) { -			if ($1 == $log_msg->{revision}) { -				push @tmp_parents, $2; -			} +# convert GetOpt::Long specs for use by git-config +sub read_repo_config { +	return unless -d $ENV{GIT_DIR}; +	my $opts = shift; +	my @config_only; +	foreach my $o (keys %$opts) { +		# if we have mixedCase and a long option-only, then +		# it's a config-only variable that we don't need for +		# the command-line. +		push @config_only, $o if ($o =~ /[A-Z]/ && $o =~ /^[a-z]+$/i); +		my $v = $opts->{$o}; +		my ($key) = ($o =~ /^([a-zA-Z\-]+)/); +		$key =~ s/-//g; +		my $arg = 'git-config'; +		$arg .= ' --int' if ($o =~ /[:=]i$/); +		$arg .= ' --bool' if ($o !~ /[:=][sfi]$/); +		if (ref $v eq 'ARRAY') { +			chomp(my @tmp = `$arg --get-all svn.$key`); +			@$v = @tmp if @tmp;  		} else { -			push @tmp_parents, $p if $p =~ /$sha1_short/o; -		} -	} -	my $tree = $log_msg->{tree}; -	if (!defined $tree) { -		my $index = set_index($GIT_SVN_INDEX); -		$tree = command_oneline('write-tree'); -		croak $? if $?; -		restore_index($index); -	} -	# just in case we clobber the existing ref, we still want that ref -	# as our parent: -	if (my $cur = verify_ref("refs/remotes/$GIT_SVN^0")) { -		chomp $cur; -		push @tmp_parents, $cur; -	} - -	if (exists $tree_map{$tree}) { -		foreach my $p (@{$tree_map{$tree}}) { -			my $skip; -			foreach (@tmp_parents) { -				# see if a common parent is found -				my $mb = eval { command('merge-base', $_, $p) }; -				next if ($@ || $?); -				$skip = 1; -				last; +			chomp(my $tmp = `$arg --get svn.$key`); +			if ($tmp && !($arg =~ / --bool/ && $tmp eq 'false')) { +				$$v = $tmp;  			} -			next if $skip; -			my ($url_p, $r_p, $uuid_p) = cmt_metadata($p); -			next if (($SVN_UUID eq $uuid_p) && -						($log_msg->{revision} > $r_p)); -			next if (defined $url_p && defined $SVN_URL && -						($SVN_UUID eq $uuid_p) && -						($url_p eq $SVN_URL)); -			push @tmp_parents, $p; -		} -	} -	foreach (@tmp_parents) { -		next if $seen_parent{$_}; -		$seen_parent{$_} = 1; -		push @exec_parents, $_; -		# MAXPARENT is defined to 16 in commit-tree.c: -		last if @exec_parents > 16; +		}  	} +	delete @$opts{@config_only} if @config_only; +} -	set_commit_env($log_msg); -	my @exec = ('git-commit-tree', $tree); -	push @exec, '-p', $_  foreach @exec_parents; -	defined(my $pid = open3(my $msg_fh, my $out_fh, '>&STDERR', @exec)) -								or croak $!; -	print $msg_fh $log_msg->{msg} or croak $!; -	unless ($_no_metadata) { -		print $msg_fh "\ngit-svn-id: $SVN_URL\@$log_msg->{revision}", -					" $SVN_UUID\n" or croak $!; +sub extract_metadata { +	my $id = shift or return (undef, undef, undef); +	my ($url, $rev, $uuid) = ($id =~ /^git-svn-id:\s(\S+?)\@(\d+) +							\s([a-f\d\-]+)$/x); +	if (!defined $rev || !$uuid || !$url) { +		# some of the original repositories I made had +		# identifiers like this: +		($rev, $uuid) = ($id =~/^git-svn-id:\s(\d+)\@([a-f\d\-]+)/);  	} -	$msg_fh->flush == 0 or croak $!; -	close $msg_fh or croak $!; -	chomp(my $commit = do { local $/; <$out_fh> }); -	close $out_fh or croak $!; -	waitpid $pid, 0; -	croak $? if $?; -	if ($commit !~ /^$sha1$/o) { -		die "Failed to commit, invalid sha1: $commit\n"; +	return ($url, $rev, $uuid); +} + +sub cmt_metadata { +	return extract_metadata((grep(/^git-svn-id: /, +		command(qw/cat-file commit/, shift)))[-1]); +} + +sub working_head_info { +	my ($head, $refs) = @_; +	my ($url, $rev, $uuid); +	my ($fh, $ctx) = command_output_pipe('rev-list', $head); +	while (<$fh>) { +		chomp; +		($url, $rev, $uuid) = cmt_metadata($_); +		last if (defined $url && defined $rev && defined $uuid); +		unshift @$refs, $_ if $refs;  	} -	command_noisy('update-ref',"refs/remotes/$GIT_SVN",$commit); -	revdb_set($REVDB, $log_msg->{revision}, $commit); +	close $fh; # break the pipe +	($url, $rev, $uuid); +} -	# this output is read via pipe, do not change: -	print "r$log_msg->{revision} = $commit\n"; -	return $commit; +package Git::SVN; +use strict; +use warnings; +use vars qw/$default_repo_id $default_ref_id $_no_metadata $_follow_parent +            $_repack $_repack_flags $_use_svm_props $_head +            $_use_svnsync_props $no_reuse_existing/; +use Carp qw/croak/; +use File::Path qw/mkpath/; +use File::Copy qw/copy/; +use IPC::Open3; + +my $_repack_nr; +# properties that we do not log: +my %SKIP_PROP; +BEGIN { +	%SKIP_PROP = map { $_ => 1 } qw/svn:wc:ra_dav:version-url +	                                svn:special svn:executable +	                                svn:entry:committed-rev +	                                svn:entry:last-author +	                                svn:entry:uuid +	                                svn:entry:committed-date/; + +	# some options are read globally, but can be overridden locally +	# per [svn-remote "..."] section.  Command-line options will *NOT* +	# override options set in an [svn-remote "..."] section +	my $e; +	foreach (qw/follow_parent no_metadata use_svm_props +	            use_svnsync_props/) { +		my $key = $_; +		$key =~ tr/_//d; +		$e .= "sub $_ { +			my (\$self) = \@_; +			return \$self->{-$_} if exists \$self->{-$_}; +			my \$k = \"svn-remote.\$self->{repo_id}\.$key\"; +			eval { command_oneline(qw/config --get/, \$k) }; +			if (\$@) { +				\$self->{-$_} = \$Git::SVN::_$_; +			} else { +				my \$v = command_oneline(qw/config --bool/,\$k); +				\$self->{-$_} = \$v eq 'false' ? 0 : 1; +			} +			return \$self->{-$_} }\n"; +	} +	$e .= "1;\n"; +	eval $e or die $@; +} + +my %LOCKFILES; +END { unlink keys %LOCKFILES if %LOCKFILES } + +sub resolve_local_globs { +	my ($url, $fetch, $glob_spec) = @_; +	return unless defined $glob_spec; +	my $ref = $glob_spec->{ref}; +	my $path = $glob_spec->{path}; +	foreach (command(qw#for-each-ref --format=%(refname) refs/remotes#)) { +		next unless m#^refs/remotes/$ref->{regex}$#; +		my $p = $1; +		my $pathname = $path->full_path($p); +		my $refname = $ref->full_path($p); +		if (my $existing = $fetch->{$pathname}) { +			if ($existing ne $refname) { +				die "Refspec conflict:\n", +				    "existing: refs/remotes/$existing\n", +				    " globbed: refs/remotes/$refname\n"; +			} +			my $u = (::cmt_metadata("refs/remotes/$refname"))[0]; +			$u =~ s!^\Q$url\E(/|$)!! or die +			  "refs/remotes/$refname: '$url' not found in '$u'\n"; +			if ($pathname ne $u) { +				warn "W: Refspec glob conflict ", +				     "(ref: refs/remotes/$refname):\n", +				     "expected path: $pathname\n", +				     "    real path: $u\n", +				     "Continuing ahead with $u\n"; +				next; +			} +		} else { +			$fetch->{$pathname} = $refname; +		} +	} +} + +sub parse_revision_argument { +	my ($base, $head) = @_; +	if (!defined $::_revision || $::_revision eq 'BASE:HEAD') { +		return ($base, $head); +	} +	return ($1, $2) if ($::_revision =~ /^(\d+):(\d+)$/); +	return ($::_revision, $::_revision) if ($::_revision =~ /^\d+$/); +	return ($head, $head) if ($::_revision eq 'HEAD'); +	return ($base, $1) if ($::_revision =~ /^BASE:(\d+)$/); +	return ($1, $head) if ($::_revision =~ /^(\d+):HEAD$/); +	die "revision argument: $::_revision not understood by git-svn\n"; +} + +sub fetch_all { +	my ($repo_id, $remotes) = @_; +	if (ref $repo_id) { +		my $gs = $repo_id; +		$repo_id = undef; +		$repo_id = $gs->{repo_id}; +	} +	$remotes ||= read_all_remotes(); +	my $remote = $remotes->{$repo_id} or +	             die "[svn-remote \"$repo_id\"] unknown\n"; +	my $fetch = $remote->{fetch}; +	my $url = $remote->{url} or die "svn-remote.$repo_id.url not defined\n"; +	my (@gs, @globs); +	my $ra = Git::SVN::Ra->new($url); +	my $uuid = $ra->get_uuid; +	my $head = $ra->get_latest_revnum; +	my $base = defined $fetch ? $head : 0; + +	# read the max revs for wildcard expansion (branches/*, tags/*) +	foreach my $t (qw/branches tags/) { +		defined $remote->{$t} or next; +		push @globs, $remote->{$t}; +		my $max_rev = eval { tmp_config(qw/--int --get/, +		                         "svn-remote.$repo_id.${t}-maxRev") }; +		if (defined $max_rev && ($max_rev < $base)) { +			$base = $max_rev; +		} elsif (!defined $max_rev) { +			$base = 0; +		} +	} + +	if ($fetch) { +		foreach my $p (sort keys %$fetch) { +			my $gs = Git::SVN->new($fetch->{$p}, $repo_id, $p); +			my $lr = $gs->rev_db_max; +			if (defined $lr) { +				$base = $lr if ($lr < $base); +			} +			push @gs, $gs; +		} +	} + +	($base, $head) = parse_revision_argument($base, $head); +	$ra->gs_fetch_loop_common($base, $head, \@gs, \@globs); +} + +sub read_all_remotes { +	my $r = {}; +	foreach (grep { s/^svn-remote\.// } command(qw/config -l/)) { +		if (m!^(.+)\.fetch=\s*(.*)\s*:\s*refs/remotes/(.+)\s*$!) { +			$r->{$1}->{fetch}->{$2} = $3; +		} elsif (m!^(.+)\.url=\s*(.*)\s*$!) { +			$r->{$1}->{url} = $2; +		} elsif (m!^(.+)\.(branches|tags)= +		           (.*):refs/remotes/(.+)\s*$/!x) { +			my ($p, $g) = ($3, $4); +			my $rs = $r->{$1}->{$2} = { +			                  t => $2, +					  remote => $1, +			                  path => Git::SVN::GlobSpec->new($p), +			                  ref => Git::SVN::GlobSpec->new($g) }; +			if (length($rs->{ref}->{right}) != 0) { +				die "The '*' glob character must be the last ", +				    "character of '$g'\n"; +			} +		} +	} +	$r;  } -sub check_repack { -	if ($_repack && (--$_repack_nr == 0)) { +sub init_vars { +	if (defined $_repack) { +		$_repack = 1000 if ($_repack <= 0);  		$_repack_nr = $_repack; -		# repack doesn't use any arguments with spaces in them, does it? -		command_noisy('repack', split(/\s+/, $_repack_flags)); +		$_repack_flags ||= '-d';  	}  } -sub set_commit_env { -	my ($log_msg) = @_; -	my $author = $log_msg->{author}; -	if (!defined $author || length $author == 0) { -		$author = '(no author)'; +sub verify_remotes_sanity { +	return unless -d $ENV{GIT_DIR}; +	my %seen; +	foreach (command(qw/config -l/)) { +		if (m!^svn-remote\.(?:.+)\.fetch=.*:refs/remotes/(\S+)\s*$!) { +			if ($seen{$1}) { +				die "Remote ref refs/remote/$1 is tracked by", +				    "\n  \"$_\"\nand\n  \"$seen{$1}\"\n", +				    "Please resolve this ambiguity in ", +				    "your git configuration file before ", +				    "continuing\n"; +			} +			$seen{$1} = $_; +		}  	} -	my ($name,$email) = defined $users{$author} ?  @{$users{$author}} -				: ($author,"$author\@$SVN_UUID"); -	$ENV{GIT_AUTHOR_NAME} = $ENV{GIT_COMMITTER_NAME} = $name; -	$ENV{GIT_AUTHOR_EMAIL} = $ENV{GIT_COMMITTER_EMAIL} = $email; -	$ENV{GIT_AUTHOR_DATE} = $ENV{GIT_COMMITTER_DATE} = $log_msg->{date};  } -sub check_upgrade_needed { -	if (!-r $REVDB) { -		-d $GIT_SVN_DIR or mkpath([$GIT_SVN_DIR]); -		open my $fh, '>>',$REVDB or croak $!; -		close $fh; -	} -	return unless eval { -		command([qw/rev-parse --verify/,"$GIT_SVN-HEAD^0"], -		        {STDERR => 0}); -	}; -	my $head = eval { command('rev-parse',"refs/remotes/$GIT_SVN") }; -	if ($@ || !$head) { -		print STDERR "Please run: $0 rebuild --upgrade\n"; -		exit 1; +# we allow more chars than remotes2config.sh... +sub sanitize_remote_name { +	my ($name) = @_; +	$name =~ tr{A-Za-z0-9:,/+-}{.}c; +	$name; +} + +sub find_existing_remote { +	my ($url, $remotes) = @_; +	return undef if $no_reuse_existing; +	my $existing; +	foreach my $repo_id (keys %$remotes) { +		my $u = $remotes->{$repo_id}->{url} or next; +		next if $u ne $url; +		$existing = $repo_id; +		last;  	} +	$existing;  } -# fills %tree_map with a reverse mapping of trees to commits.  Useful -# for finding parents to commit on. -sub map_tree_joins { -	my %seen; -	foreach my $br (@_branch_from) { -		my $pipe = command_output_pipe(qw/rev-list -		                            --topo-order --pretty=raw/, $br); -		while (<$pipe>) { -			if (/^commit ($sha1)$/o) { -				my $commit = $1; - -				# if we've seen a commit, -				# we've seen its parents -				last if $seen{$commit}; -				my ($tree) = (<$pipe> =~ /^tree ($sha1)$/o); -				unless (defined $tree) { -					die "Failed to parse commit $commit\n"; -				} -				push @{$tree_map{$tree}}, $commit; -				$seen{$commit} = 1; +sub init_remote_config { +	my ($self, $url, $no_write) = @_; +	$url =~ s!/+$!!; # strip trailing slash +	my $r = read_all_remotes(); +	my $existing = find_existing_remote($url, $r); +	if ($existing) { +		unless ($no_write) { +			print STDERR "Using existing ", +				     "[svn-remote \"$existing\"]\n"; +		} +		$self->{repo_id} = $existing; +	} else { +		my $min_url = Git::SVN::Ra->new($url)->minimize_url; +		$existing = find_existing_remote($min_url, $r); +		if ($existing) { +			unless ($no_write) { +				print STDERR "Using existing ", +					     "[svn-remote \"$existing\"]\n"; +			} +			$self->{repo_id} = $existing; +		} +		if ($min_url ne $url) { +			unless ($no_write) { +				print STDERR "Using higher level of URL: ", +					     "$url => $min_url\n"; +			} +			my $old_path = $self->{path}; +			$self->{path} = $url; +			$self->{path} =~ s!^\Q$min_url\E(/|$)!!; +			if (length $old_path) { +				$self->{path} .= "/$old_path";  			} +			$url = $min_url; +		} +	} +	my $orig_url; +	if (!$existing) { +		# verify that we aren't overwriting anything: +		$orig_url = eval { +			command_oneline('config', '--get', +					"svn-remote.$self->{repo_id}.url") +		}; +		if ($orig_url && ($orig_url ne $url)) { +			die "svn-remote.$self->{repo_id}.url already set: ", +			    "$orig_url\nwanted to set to: $url\n";  		} -		close $pipe;  	} +	my ($xrepo_id, $xpath) = find_ref($self->refname); +	if (defined $xpath) { +		die "svn-remote.$xrepo_id.fetch already set to track ", +		    "$xpath:refs/remotes/", $self->refname, "\n"; +	} +	unless ($no_write) { +		command_noisy('config', +			      "svn-remote.$self->{repo_id}.url", $url); +		command_noisy('config', '--add', +			      "svn-remote.$self->{repo_id}.fetch", +			      "$self->{path}:".$self->refname); +	} +	$self->{url} = $url;  } -sub load_all_refs { -	if (@_branch_from) { -		print STDERR '--branch|-b parameters are ignored when ', -			"--branch-all-refs|-B is passed\n"; +sub find_by_url { # repos_root and, path are optional +	my ($class, $full_url, $repos_root, $path) = @_; +	return undef unless defined $full_url; +	my $remotes = read_all_remotes(); +	if (defined $full_url && defined $repos_root && !defined $path) { +		$path = $full_url; +		$path =~ s#^\Q$repos_root\E(?:/|$)##;  	} +	foreach my $repo_id (keys %$remotes) { +		my $u = $remotes->{$repo_id}->{url} or next; +		next if defined $repos_root && $repos_root ne $u; -	# don't worry about rev-list on non-commit objects/tags, -	# it shouldn't blow up if a ref is a blob or tree... -	@_branch_from = command(qw/rev-parse --symbolic --all/); +		my $fetch = $remotes->{$repo_id}->{fetch} || {}; +		foreach (qw/branches tags/) { +			resolve_local_globs($u, $fetch, +			                    $remotes->{$repo_id}->{$_}); +		} +		my $p = $path; +		unless (defined $p) { +			$p = $full_url; +			$p =~ s#^\Q$u\E(?:/|$)## or next; +		} +		foreach my $f (keys %$fetch) { +			next if $f ne $p; +			return Git::SVN->new($fetch->{$f}, $repo_id, $f); +		} +	} +	undef;  } -# '<svn username> = real-name <email address>' mapping based on git-svnimport: -sub load_authors { -	open my $authors, '<', $_authors or die "Can't open $_authors $!\n"; -	while (<$authors>) { -		chomp; -		next unless /^(\S+?|\(no author\))\s*=\s*(.+?)\s*<(.+)>\s*$/; -		my ($user, $name, $email) = ($1, $2, $3); -		$users{$user} = [$name, $email]; +sub init { +	my ($class, $url, $path, $repo_id, $ref_id, $no_write) = @_; +	my $self = _new($class, $repo_id, $ref_id, $path); +	if (defined $url) { +		$self->init_remote_config($url, $no_write);  	} -	close $authors or croak $!; +	$self;  } -sub rload_authors { -	open my $authors, '<', $_authors or die "Can't open $_authors $!\n"; -	while (<$authors>) { -		chomp; -		next unless /^(\S+?)\s*=\s*(.+?)\s*<(.+)>\s*$/; -		my ($user, $name, $email) = ($1, $2, $3); -		$rusers{"$name <$email>"} = $user; +sub find_ref { +	my ($ref_id) = @_; +	foreach (command(qw/config -l/)) { +		next unless m!^svn-remote\.(.+)\.fetch= +		              \s*(.*)\s*:\s*refs/remotes/(.+)\s*$!x; +		my ($repo_id, $path, $ref) = ($1, $2, $3); +		if ($ref eq $ref_id) { +			$path = '' if ($path =~ m#^\./?#); +			return ($repo_id, $path); +		}  	} -	close $authors or croak $!; +	(undef, undef, undef);  } -sub git_svn_each { -	my $sub = shift; -	foreach (command(qw/rev-parse --symbolic --all/)) { -		next unless s#^refs/remotes/##; -		chomp $_; -		next unless -f "$GIT_DIR/svn/$_/info/url"; -		&$sub($_); +sub new { +	my ($class, $ref_id, $repo_id, $path) = @_; +	if (defined $ref_id && !defined $repo_id && !defined $path) { +		($repo_id, $path) = find_ref($ref_id); +		if (!defined $repo_id) { +			die "Could not find a \"svn-remote.*.fetch\" key ", +			    "in the repository configuration matching: ", +			    "refs/remotes/$ref_id\n"; +		} +	} +	my $self = _new($class, $repo_id, $ref_id, $path); +	if (!defined $self->{path} || !length $self->{path}) { +		my $fetch = command_oneline('config', '--get', +		                            "svn-remote.$repo_id.fetch", +		                            ":refs/remotes/$ref_id\$") or +		     die "Failed to read \"svn-remote.$repo_id.fetch\" ", +		         "\":refs/remotes/$ref_id\$\" in config\n"; +		($self->{path}, undef) = split(/\s*:\s*/, $fetch); +	} +	$self->{url} = command_oneline('config', '--get', +	                               "svn-remote.$repo_id.url") or +                  die "Failed to read \"svn-remote.$repo_id.url\" in config\n"; +	$self->rebuild; +	$self; +} + +sub refname { "refs/remotes/$_[0]->{ref_id}" } + +sub svm_uuid { +	my ($self) = @_; +	return $self->{svm}->{uuid} if $self->svm; +	$self->ra; +	unless ($self->{svm}) { +		die "SVM UUID not cached, and reading remotely failed\n";  	} +	$self->{svm}->{uuid};  } -sub migrate_revdb { -	git_svn_each(sub { -		my $id = shift; -		defined(my $pid = fork) or croak $!; -		if (!$pid) { -			$GIT_SVN = $ENV{GIT_SVN_ID} = $id; -			init_vars(); -			exit 0 if -r $REVDB; -			print "Upgrading svn => git mapping...\n"; -			-d $GIT_SVN_DIR or mkpath([$GIT_SVN_DIR]); -			open my $fh, '>>',$REVDB or croak $!; -			close $fh; -			rebuild(); -			print "Done upgrading. You may now delete the ", -				"deprecated $GIT_SVN_DIR/revs directory\n"; -			exit 0; +sub svm { +	my ($self) = @_; +	return $self->{svm} if $self->{svm}; +	my $svm; +	# see if we have it in our config, first: +	eval { +		my $section = "svn-remote.$self->{repo_id}"; +		$svm = { +		  source => tmp_config('--get', "$section.svm-source"), +		  uuid => tmp_config('--get', "$section.svm-uuid"), +		  replace => tmp_config('--get', "$section.svm-replace"),  		} -		waitpid $pid, 0; -		croak $? if $?; -	}); -} +	}; +	if ($svm && $svm->{source} && $svm->{uuid} && $svm->{replace}) { +		$self->{svm} = $svm; +	} +	$self->{svm}; +} + +sub _set_svm_vars { +	my ($self, $ra) = @_; +	return $ra if $self->svm; + +	my @err = ( "useSvmProps set, but failed to read SVM properties\n", +		    "(svm:source, svm:uuid) ", +		    "from the following URLs:\n" ); +	sub read_svm_props { +		my ($self, $ra, $path, $r) = @_; +		my $props = ($ra->get_dir($path, $r))[2]; +		my $src = $props->{'svm:source'}; +		my $uuid = $props->{'svm:uuid'}; +		return undef if (!$src || !$uuid); + +		chomp($src, $uuid); + +		$uuid =~ m{^[0-9a-f\-]{30,}$} +		    or die "doesn't look right - svm:uuid is '$uuid'\n"; + +		# the '!' is used to mark the repos_root!/relative/path +		$src =~ s{/?!/?}{/}; +		$src =~ s{/+$}{}; # no trailing slashes please +		# username is of no interest +		$src =~ s{(^[a-z\+]*://)[^/@]*@}{$1}; + +		my $replace = $ra->{url}; +		$replace .= "/$path" if length $path; + +		my $section = "svn-remote.$self->{repo_id}"; +		tmp_config("$section.svm-source", $src); +		tmp_config("$section.svm-replace", $replace); +		tmp_config("$section.svm-uuid", $uuid); +		$self->{svm} = { +			source => $src, +			uuid => $uuid, +			replace => $replace +		}; +	} -sub migration_check { -	migrate_revdb() unless (-e $REVDB); -	return if (-d "$GIT_DIR/svn" || !-d $GIT_DIR); -	print "Upgrading repository...\n"; -	unless (-d "$GIT_DIR/svn") { -		mkdir "$GIT_DIR/svn" or croak $!; +	my $r = $ra->get_latest_revnum; +	my $path = $self->{path}; +	my %tried; +	while (length $path) { +		unless ($tried{"$self->{url}/$path"}) { +			return $ra if $self->read_svm_props($ra, $path, $r); +			$tried{"$self->{url}/$path"} = 1; +		} +		$path =~ s#/?[^/]+$##;  	} -	print "Data from a previous version of git-svn exists, but\n\t", -				"$GIT_SVN_DIR\n\t(required for this version ", -				"($VERSION) of git-svn) does not.\n"; +	die "Path: '$path' should be ''\n" if $path ne ''; +	return $ra if $self->read_svm_props($ra, $path, $r); +	$tried{"$self->{url}/$path"} = 1; -	foreach my $x (command(qw/rev-parse --symbolic --all/)) { -		next unless $x =~ s#^refs/remotes/##; -		chomp $x; -		next unless -f "$GIT_DIR/$x/info/url"; -		my $u = eval { file_to_s("$GIT_DIR/$x/info/url") }; -		next unless $u; -		my $dn = dirname("$GIT_DIR/svn/$x"); -		mkpath([$dn]) unless -d $dn; -		rename "$GIT_DIR/$x", "$GIT_DIR/svn/$x" or croak "$!: $x"; +	if ($ra->{repos_root} eq $self->{url}) { +		die @err, (map { "  $_\n" } keys %tried), "\n";  	} -	migrate_revdb() if (-d $GIT_SVN_DIR && !-w $REVDB); -	print "Done upgrading.\n"; -} -sub find_rev_before { -	my ($r, $id, $eq_ok) = @_; -	my $f = "$GIT_DIR/svn/$id/.rev_db"; -	return (undef,undef) unless -r $f; -	--$r unless $eq_ok; -	while ($r > 0) { -		if (my $c = revdb_get($f, $r)) { -			return ($r, $c); +	# nope, make sure we're connected to the repository root: +	my $ok; +	my @tried_b; +	$path = $ra->{svn_path}; +	$ra = Git::SVN::Ra->new($ra->{repos_root}); +	while (length $path) { +		unless ($tried{"$ra->{url}/$path"}) { +			$ok = $self->read_svm_props($ra, $path, $r); +			last if $ok; +			$tried{"$ra->{url}/$path"} = 1;  		} -		--$r; +		$path =~ s#/?[^/]+$##;  	} -	return (undef, undef); +	die "Path: '$path' should be ''\n" if $path ne ''; +	$ok ||= $self->read_svm_props($ra, $path, $r); +	$tried{"$ra->{url}/$path"} = 1; +	if (!$ok) { +		die @err, (map { "  $_\n" } keys %tried), "\n"; +	} +	Git::SVN::Ra->new($self->{url});  } -sub init_vars { -	$GIT_SVN ||= $ENV{GIT_SVN_ID} || 'git-svn'; -	$GIT_SVN_DIR = "$GIT_DIR/svn/$GIT_SVN"; -	$REVDB = "$GIT_SVN_DIR/.rev_db"; -	$GIT_SVN_INDEX = "$GIT_SVN_DIR/index"; -	$SVN_URL = undef; -	$SVN_WC = "$GIT_SVN_DIR/tree"; -	%tree_map = (); -} +sub svnsync { +	my ($self) = @_; +	return $self->{svnsync} if $self->{svnsync}; -# convert GetOpt::Long specs for use by git-config -sub read_repo_config { -	return unless -d $GIT_DIR; -	my $opts = shift; -	foreach my $o (keys %$opts) { -		my $v = $opts->{$o}; -		my ($key) = ($o =~ /^([a-z\-]+)/); -		$key =~ s/-//g; -		my $arg = 'git-config'; -		$arg .= ' --int' if ($o =~ /[:=]i$/); -		$arg .= ' --bool' if ($o !~ /[:=][sfi]$/); -		if (ref $v eq 'ARRAY') { -			chomp(my @tmp = `$arg --get-all svn.$key`); -			@$v = @tmp if @tmp; -		} else { -			chomp(my $tmp = `$arg --get svn.$key`); -			if ($tmp && !($arg =~ / --bool/ && $tmp eq 'false')) { -				$$v = $tmp; -			} +	if ($self->no_metadata) { +		die "Can't have both 'noMetadata' and ", +		    "'useSvnsyncProps' options set!\n"; +	} +	if ($self->rewrite_root) { +		die "Can't have both 'useSvnsyncProps' and 'rewriteRoot' ", +		    "options set!\n"; +	} + +	my $svnsync; +	# see if we have it in our config, first: +	eval { +		my $section = "svn-remote.$self->{repo_id}"; +		$svnsync = { +		  url => tmp_config('--get', "$section.svnsync-url"), +		  uuid => tmp_config('--get', "$section.svnsync-uuid"),  		} +	}; +	if ($svnsync && $svnsync->{url} && $svnsync->{uuid}) { +		return $self->{svnsync} = $svnsync;  	} + +	my $err = "useSvnsyncProps set, but failed to read " . +	          "svnsync property: svn:sync-from-"; +	my $rp = $self->ra->rev_proplist(0); + +	my $url = $rp->{'svn:sync-from-url'} or die $err . "url\n"; +	$url =~ m{^[a-z\+]+://} or +	           die "doesn't look right - svn:sync-from-url is '$url'\n"; + +	my $uuid = $rp->{'svn:sync-from-uuid'} or die $err . "uuid\n"; +	$uuid =~ m{^[0-9a-f\-]{30,}$} or +	           die "doesn't look right - svn:sync-from-uuid is '$uuid'\n"; + +	my $section = "svn-remote.$self->{repo_id}"; +	tmp_config('--add', "$section.svnsync-uuid", $uuid); +	tmp_config('--add', "$section.svnsync-url", $url); +	return $self->{svnsync} = { url => $url, uuid => $uuid };  } -sub set_default_vals { -	if (defined $_repack) { -		$_repack = 1000 if ($_repack <= 0); -		$_repack_nr = $_repack; -		$_repack_flags ||= '-d'; +# this allows us to memoize our SVN::Ra UUID locally and avoid a +# remote lookup (useful for 'git svn log'). +sub ra_uuid { +	my ($self) = @_; +	unless ($self->{ra_uuid}) { +		my $key = "svn-remote.$self->{repo_id}.uuid"; +		my $uuid = eval { tmp_config('--get', $key) }; +		if (!$@ && $uuid && $uuid =~ /^([a-f\d\-]{30,})$/) { +			$self->{ra_uuid} = $uuid; +		} else { +			die "ra_uuid called without URL\n" unless $self->{url}; +			$self->{ra_uuid} = $self->ra->get_uuid; +			tmp_config('--add', $key, $self->{ra_uuid}); +		}  	} +	$self->{ra_uuid};  } -sub read_grafts { -	my $gr_file = shift; -	my ($grafts, $comments) = ({}, {}); -	if (open my $fh, '<', $gr_file) { -		my @tmp; -		while (<$fh>) { -			if (/^($sha1)\s+/) { -				my $c = $1; -				if (@tmp) { -					@{$comments->{$c}} = @tmp; -					@tmp = (); -				} -				foreach my $p (split /\s+/, $_) { -					$grafts->{$c}->{$p} = 1; -				} -			} else { -				push @tmp, $_; -			} +sub ra { +	my ($self) = shift; +	my $ra = Git::SVN::Ra->new($self->{url}); +	if ($self->use_svm_props && !$self->{svm}) { +		if ($self->no_metadata) { +			die "Can't have both 'noMetadata' and ", +			    "'useSvmProps' options set!\n"; +		} elsif ($self->use_svnsync_props) { +			die "Can't have both 'useSvnsyncProps' and ", +			    "'useSvmProps' options set!\n";  		} -		close $fh or croak $!; -		@{$comments->{'END'}} = @tmp if @tmp; +		$ra = $self->_set_svm_vars($ra); +		$self->{-want_revprops} = 1;  	} -	return ($grafts, $comments); +	$ra;  } -sub write_grafts { -	my ($grafts, $comments, $gr_file) = @_; +sub rel_path { +	my ($self) = @_; +	my $repos_root = $self->ra->{repos_root}; +	return $self->{path} if ($self->{url} eq $repos_root); +	die "BUG: rel_path failed! repos_root: $repos_root, Ra URL: ", +	    $self->ra->{url}, " path: $self->{path},  URL: $self->{url}\n"; +} -	open my $fh, '>', $gr_file or croak $!; -	foreach my $c (sort keys %$grafts) { -		if ($comments->{$c}) { -			print $fh $_ foreach @{$comments->{$c}}; -		} -		my $p = $grafts->{$c}; -		my %x; # real parents -		delete $p->{$c}; # commits are not self-reproducing... -		my $ch = command_output_pipe(qw/cat-file commit/, $c); -		while (<$ch>) { -			if (/^parent ($sha1)/) { -				$x{$1} = $p->{$1} = 1; -			} else { -				last unless /^\S/; -			} -		} -		close $ch; # breaking the pipe - -		# if real parents are the only ones in the grafts, drop it -		next if join(' ',sort keys %$p) eq join(' ',sort keys %x); - -		my (@ip, @jp, $mb); -		my %del = %x; -		@ip = @jp = keys %$p; -		foreach my $i (@ip) { -			next if $del{$i} || $p->{$i} == 2; -			foreach my $j (@jp) { -				next if $i eq $j || $del{$j} || $p->{$j} == 2; -				$mb = eval { command('merge-base', $i, $j) }; -				next unless $mb; -				chomp $mb; -				next if $x{$mb}; -				if ($mb eq $j) { -					delete $p->{$i}; -					$del{$i} = 1; -				} elsif ($mb eq $i) { -					delete $p->{$j}; -					$del{$j} = 1; -				} -			} +sub traverse_ignore { +	my ($self, $fh, $path, $r) = @_; +	$path =~ s#^/+##g; +	my $ra = $self->ra; +	my ($dirent, undef, $props) = $ra->get_dir($path, $r); +	my $p = $path; +	$p =~ s#^\Q$ra->{svn_path}\E/##; +	print $fh length $p ? "\n# $p\n" : "\n# /\n"; +	if (my $s = $props->{'svn:ignore'}) { +		$s =~ s/[\r\n]+/\n/g; +		chomp $s; +		if (length $p == 0) { +			$s =~ s#\n#\n/$p#g; +			print $fh "/$s\n"; +		} else { +			$s =~ s#\n#\n/$p/#g; +			print $fh "/$p/$s\n";  		} +	} +	foreach (sort keys %$dirent) { +		next if $dirent->{$_}->kind != $SVN::Node::dir; +		$self->traverse_ignore($fh, "$path/$_", $r); +	} +} -		# if real parents are the only ones in the grafts, drop it -		next if join(' ',sort keys %$p) eq join(' ',sort keys %x); +sub last_rev { ($_[0]->last_rev_commit)[0] } +sub last_commit { ($_[0]->last_rev_commit)[1] } -		print $fh $c, ' ', join(' ', sort keys %$p),"\n"; +# returns the newest SVN revision number and newest commit SHA1 +sub last_rev_commit { +	my ($self) = @_; +	if (defined $self->{last_rev} && defined $self->{last_commit}) { +		return ($self->{last_rev}, $self->{last_commit}); +	} +	my $c = ::verify_ref($self->refname.'^0'); +	if ($c && !$self->use_svm_props && !$self->no_metadata) { +		my $rev = (::cmt_metadata($c))[1]; +		if (defined $rev) { +			($self->{last_rev}, $self->{last_commit}) = ($rev, $c); +			return ($rev, $c); +		} +	} +	my $db_path = $self->db_path; +	unless (-e $db_path) { +		($self->{last_rev}, $self->{last_commit}) = (undef, undef); +		return (undef, undef); +	} +	my $offset = -41; # from tail +	my $rl; +	open my $fh, '<', $db_path or croak "$db_path not readable: $!\n"; +	sysseek($fh, $offset, 2); # don't care for errors +	sysread($fh, $rl, 41) == 41 or return (undef, undef); +	chomp $rl; +	while (('0' x40) eq $rl && sysseek($fh, 0, 1) != 0) { +		$offset -= 41; +		sysseek($fh, $offset, 2); # don't care for errors +		sysread($fh, $rl, 41) == 41 or return (undef, undef); +		chomp $rl;  	} -	if ($comments->{'END'}) { -		print $fh $_ foreach @{$comments->{'END'}}; +	if ($c && $c ne $rl) { +		die "$db_path and ", $self->refname, +		    " inconsistent!:\n$c != $rl\n";  	} +	my $rev = sysseek($fh, 0, 1) or croak $!; +	$rev =  ($rev - 41) / 41;  	close $fh or croak $!; +	($self->{last_rev}, $self->{last_commit}) = ($rev, $c); +	return ($rev, $c);  } -sub read_url_paths_all { -	my ($l_map, $pfx, $p) = @_; -	my @dir; -	foreach (<$p/*>) { -		if (-r "$_/info/url") { -			$pfx .= '/' if $pfx && $pfx !~ m!/$!; -			my $id = $pfx . basename $_; -			my $url = file_to_s("$_/info/url"); -			my ($u, $p) = repo_path_split($url); -			$l_map->{$u}->{$p} = $id; -		} elsif (-d $_) { -			push @dir, $_; -		} -	} -	foreach (@dir) { -		my $x = $_; -		$x =~ s!^\Q$GIT_DIR\E/svn/!!o; -		read_url_paths_all($l_map, $x, $_); -	} +sub get_fetch_range { +	my ($self, $min, $max) = @_; +	$max ||= $self->ra->get_latest_revnum; +	$min ||= $self->rev_db_max; +	(++$min, $max);  } -# this one only gets ids that have been imported, not new ones -sub read_url_paths { -	my $l_map = {}; -	git_svn_each(sub { my $x = shift; -			my $url = file_to_s("$GIT_DIR/svn/$x/info/url"); -			my ($u, $p) = repo_path_split($url); -			$l_map->{$u}->{$p} = $x; -			}); -	return $l_map; +sub tmp_config { +	my (@args) = @_; +	my $old_def_config = "$ENV{GIT_DIR}/svn/config"; +	my $config = "$ENV{GIT_DIR}/svn/.metadata"; +	if (-e $old_def_config && ! -e $config) { +		rename $old_def_config, $config or +		       die "Failed rename $old_def_config => $config: $!\n"; +	} +	my $old_config = $ENV{GIT_CONFIG}; +	$ENV{GIT_CONFIG} = $config; +	$@ = undef; +	my @ret = eval { +		unless (-f $config) { +			mkfile($config); +			open my $fh, '>', $config or +			    die "Can't open $config: $!\n"; +			print $fh "; This file is used internally by ", +			          "git-svn\n" or die +				  "Couldn't write to $config: $!\n"; +			print $fh "; You should not have to edit it\n" or +			      die "Couldn't write to $config: $!\n"; +			close $fh or die "Couldn't close $config: $!\n"; +		} +		command('config', @args); +	}; +	my $err = $@; +	if (defined $old_config) { +		$ENV{GIT_CONFIG} = $old_config; +	} else { +		delete $ENV{GIT_CONFIG}; +	} +	die $err if $err; +	wantarray ? @ret : $ret[0];  } -sub extract_metadata { -	my $id = shift or return (undef, undef, undef); -	my ($url, $rev, $uuid) = ($id =~ /^git-svn-id:\s(\S+?)\@(\d+) -							\s([a-f\d\-]+)$/x); -	if (!defined $rev || !$uuid || !$url) { -		# some of the original repositories I made had -		# identifiers like this: -		($rev, $uuid) = ($id =~/^git-svn-id:\s(\d+)\@([a-f\d\-]+)/); +sub tmp_index_do { +	my ($self, $sub) = @_; +	my $old_index = $ENV{GIT_INDEX_FILE}; +	$ENV{GIT_INDEX_FILE} = $self->{index}; +	$@ = undef; +	my @ret = eval { +		my ($dir, $base) = ($self->{index} =~ m#^(.*?)/?([^/]+)$#); +		mkpath([$dir]) unless -d $dir; +		&$sub; +	}; +	my $err = $@; +	if (defined $old_index) { +		$ENV{GIT_INDEX_FILE} = $old_index; +	} else { +		delete $ENV{GIT_INDEX_FILE};  	} -	return ($url, $rev, $uuid); +	die $err if $err; +	wantarray ? @ret : $ret[0];  } -sub cmt_metadata { -	return extract_metadata((grep(/^git-svn-id: /, -		command(qw/cat-file commit/, shift)))[-1]); +sub assert_index_clean { +	my ($self, $treeish) = @_; + +	$self->tmp_index_do(sub { +		command_noisy('read-tree', $treeish) unless -e $self->{index}; +		my $x = command_oneline('write-tree'); +		my ($y) = (command(qw/cat-file commit/, $treeish) =~ +		           /^tree ($::sha1)/mo); +		return if $y eq $x; + +		warn "Index mismatch: $y != $x\nrereading $treeish\n"; +		unlink $self->{index} or die "unlink $self->{index}: $!\n"; +		command_noisy('read-tree', $treeish); +		$x = command_oneline('write-tree'); +		if ($y ne $x) { +			::fatal "trees ($treeish) $y != $x\n", +			        "Something is seriously wrong...\n"; +		} +	});  } -sub get_commit_time { -	my $cmt = shift; -	my $fh = command_output_pipe(qw/rev-list --pretty=raw -n1/, $cmt); -	while (<$fh>) { -		/^committer\s(?:.+) (\d+) ([\-\+]?\d+)$/ or next; -		my ($s, $tz) = ($1, $2); -		if ($tz =~ s/^\+//) { -			$s += tz_to_s_offset($tz); -		} elsif ($tz =~ s/^\-//) { -			$s -= tz_to_s_offset($tz); +sub get_commit_parents { +	my ($self, $log_entry) = @_; +	my (%seen, @ret, @tmp); +	# legacy support for 'set-tree'; this is only used by set_tree_cb: +	if (my $ip = $self->{inject_parents}) { +		if (my $commit = delete $ip->{$log_entry->{revision}}) { +			push @tmp, $commit;  		} -		close $fh; -		return $s;  	} -	die "Can't get commit time for commit: $cmt\n"; +	if (my $cur = ::verify_ref($self->refname.'^0')) { +		push @tmp, $cur; +	} +	push @tmp, $_ foreach (@{$log_entry->{parents}}, @tmp); +	while (my $p = shift @tmp) { +		next if $seen{$p}; +		$seen{$p} = 1; +		push @ret, $p; +		# MAXPARENT is defined to 16 in commit-tree.c: +		last if @ret >= 16; +	} +	if (@tmp) { +		die "r$log_entry->{revision}: No room for parents:\n\t", +		    join("\n\t", @tmp), "\n"; +	} +	@ret;  } -sub tz_to_s_offset { -	my ($tz) = @_; -	$tz =~ s/(\d\d)$//; -	return ($1 * 60) + ($tz * 3600); +sub rewrite_root { +	my ($self) = @_; +	return $self->{-rewrite_root} if exists $self->{-rewrite_root}; +	my $k = "svn-remote.$self->{repo_id}.rewriteRoot"; +	my $rwr = eval { command_oneline(qw/config --get/, $k) }; +	if ($rwr) { +		$rwr =~ s#/+$##; +		if ($rwr !~ m#^[a-z\+]+://#) { +			die "$rwr is not a valid URL (key: $k)\n"; +		} +	} +	$self->{-rewrite_root} = $rwr;  } -# adapted from pager.c -sub config_pager { -	$_pager ||= $ENV{GIT_PAGER} || $ENV{PAGER}; -	if (!defined $_pager) { -		$_pager = 'less'; -	} elsif (length $_pager == 0 || $_pager eq 'cat') { -		$_pager = undef; -	} +sub metadata_url { +	my ($self) = @_; +	($self->rewrite_root || $self->{url}) . +	   (length $self->{path} ? '/' . $self->{path} : '');  } -sub run_pager { -	return unless -t *STDOUT; -	pipe my $rfd, my $wfd or return; -	defined(my $pid = fork) or croak $!; -	if (!$pid) { -		open STDOUT, '>&', $wfd or croak $!; -		return; +sub full_url { +	my ($self) = @_; +	$self->{url} . (length $self->{path} ? '/' . $self->{path} : ''); +} + +sub do_git_commit { +	my ($self, $log_entry) = @_; +	my $lr = $self->last_rev; +	if (defined $lr && $lr >= $log_entry->{revision}) { +		die "Last fetched revision of ", $self->refname, +		    " was r$lr, but we are about to fetch: ", +		    "r$log_entry->{revision}!\n";  	} -	open STDIN, '<&', $rfd or croak $!; -	$ENV{LESS} ||= 'FRSX'; -	exec $_pager or croak "Can't run pager: $! ($_pager)\n"; +	if (my $c = $self->rev_db_get($log_entry->{revision})) { +		croak "$log_entry->{revision} = $c already exists! ", +		      "Why are we refetching it?\n"; +	} +	$ENV{GIT_AUTHOR_NAME} = $ENV{GIT_COMMITTER_NAME} = $log_entry->{name}; +	$ENV{GIT_AUTHOR_EMAIL} = $ENV{GIT_COMMITTER_EMAIL} = +	                                                  $log_entry->{email}; +	$ENV{GIT_AUTHOR_DATE} = $ENV{GIT_COMMITTER_DATE} = $log_entry->{date}; + +	my $tree = $log_entry->{tree}; +	if (!defined $tree) { +		$tree = $self->tmp_index_do(sub { +		                            command_oneline('write-tree') }); +	} +	die "Tree is not a valid sha1: $tree\n" if $tree !~ /^$::sha1$/o; + +	my @exec = ('git-commit-tree', $tree); +	foreach ($self->get_commit_parents($log_entry)) { +		push @exec, '-p', $_; +	} +	defined(my $pid = open3(my $msg_fh, my $out_fh, '>&STDERR', @exec)) +	                                                           or croak $!; +	print $msg_fh $log_entry->{log} or croak $!; +	unless ($self->no_metadata) { +		print $msg_fh "\ngit-svn-id: $log_entry->{metadata}\n" +		              or croak $!; +	} +	$msg_fh->flush == 0 or croak $!; +	close $msg_fh or croak $!; +	chomp(my $commit = do { local $/; <$out_fh> }); +	close $out_fh or croak $!; +	waitpid $pid, 0; +	croak $? if $?; +	if ($commit !~ /^$::sha1$/o) { +		die "Failed to commit, invalid sha1: $commit\n"; +	} + +	$self->rev_db_set($log_entry->{revision}, $commit, 1); + +	$self->{last_rev} = $log_entry->{revision}; +	$self->{last_commit} = $commit; +	print "r$log_entry->{revision}"; +	if (defined $log_entry->{svm_revision}) { +		 print " (\@$log_entry->{svm_revision})"; +		 $self->rev_db_set($log_entry->{svm_revision}, $commit, +		                   0, $self->svm_uuid); +	} +	print " = $commit ($self->{ref_id})\n"; +	if (defined $_repack && (--$_repack_nr == 0)) { +		$_repack_nr = $_repack; +		# repack doesn't use any arguments with spaces in them, does it? +		print "Running git repack $_repack_flags ...\n"; +		command_noisy('repack', split(/\s+/, $_repack_flags)); +		print "Done repacking\n"; +	} +	return $commit;  } -sub get_author_info { -	my ($dest, $author, $t, $tz) = @_; -	$author =~ s/(?:^\s*|\s*$)//g; -	$dest->{a_raw} = $author; -	my $_a; -	if ($_authors) { -		$_a = $rusers{$author} || undef; +sub match_paths { +	my ($self, $paths, $r) = @_; +	return 1 if $self->{path} eq ''; +	if (my $path = $paths->{"/$self->{path}"}) { +		return ($path->{action} eq 'D') ? 0 : 1;  	} -	if (!$_a) { -		($_a) = ($author =~ /<([^>]+)\@[^>]+>$/); +	$self->{path_regex} ||= qr/^\/\Q$self->{path}\E\//; +	if (grep /$self->{path_regex}/, keys %$paths) { +		return 1;  	} -	$dest->{t} = $t; -	$dest->{tz} = $tz; -	$dest->{a} = $_a; -	# Date::Parse isn't in the standard Perl distro :( -	if ($tz =~ s/^\+//) { -		$t += tz_to_s_offset($tz); -	} elsif ($tz =~ s/^\-//) { -		$t -= tz_to_s_offset($tz); +	my $c = ''; +	foreach (split m#/#, $self->{path}) { +		$c .= "/$_"; +		next unless ($paths->{$c} && +		             ($paths->{$c}->{action} =~ /^[AR]$/)); +		if ($self->ra->check_path($self->{path}, $r) == +		    $SVN::Node::dir) { +			return 1; +		}  	} -	$dest->{t_utc} = $t; +	return 0;  } -sub process_commit { -	my ($c, $r_min, $r_max, $defer) = @_; -	if (defined $r_min && defined $r_max) { -		if ($r_min == $c->{r} && $r_min == $r_max) { -			show_commit($c); -			return 0; +sub find_parent_branch { +	my ($self, $paths, $rev) = @_; +	return undef unless $self->follow_parent; +	unless (defined $paths) { +		my $err_handler = $SVN::Error::handler; +		$SVN::Error::handler = \&Git::SVN::Ra::skip_unknown_revs; +		$self->ra->get_log([$self->{path}], $rev, $rev, 0, 1, 1, sub { +		                   $paths = +				      Git::SVN::Ra::dup_changed_paths($_[0]) }); +		$SVN::Error::handler = $err_handler; +	} +	return undef unless defined $paths; + +	# look for a parent from another branch: +	my @b_path_components = split m#/#, $self->rel_path; +	my @a_path_components; +	my $i; +	while (@b_path_components) { +		$i = $paths->{'/'.join('/', @b_path_components)}; +		last if $i && defined $i->{copyfrom_path}; +		unshift(@a_path_components, pop(@b_path_components)); +	} +	return undef unless defined $i && defined $i->{copyfrom_path}; +	my $branch_from = $i->{copyfrom_path}; +	if (@a_path_components) { +		print STDERR "branch_from: $branch_from => "; +		$branch_from .= '/'.join('/', @a_path_components); +		print STDERR $branch_from, "\n"; +	} +	my $r = $i->{copyfrom_rev}; +	my $repos_root = $self->ra->{repos_root}; +	my $url = $self->ra->{url}; +	my $new_url = $repos_root . $branch_from; +	print STDERR  "Found possible branch point: ", +	              "$new_url => ", $self->full_url, ", $r\n"; +	$branch_from =~ s#^/##; +	my $gs = Git::SVN->find_by_url($new_url, $repos_root, $branch_from); +	unless ($gs) { +		my $ref_id = $self->{ref_id}; +		$ref_id =~ s/\@\d+$//; +		$ref_id .= "\@$r"; +		# just grow a tail if we're not unique enough :x +		$ref_id .= '-' while find_ref($ref_id); +		print STDERR "Initializing parent: $ref_id\n"; +		$gs = Git::SVN->init($new_url, '', $ref_id, $ref_id, 1); +	} +	my ($r0, $parent) = $gs->find_rev_before($r, 1); +	if (!defined $r0 || !defined $parent) { +		$gs->fetch(0, $r); +		($r0, $parent) = $gs->last_rev_commit; +	} +	if (defined $r0 && defined $parent) { +		print STDERR "Found branch parent: ($self->{ref_id}) $parent\n"; +		my $ed; +		if ($self->ra->can_do_switch) { +			$self->assert_index_clean($parent); +			print STDERR "Following parent with do_switch\n"; +			# do_switch works with svn/trunk >= r22312, but that +			# is not included with SVN 1.4.3 (the latest version +			# at the moment), so we can't rely on it +			$self->{last_commit} = $parent; +			$ed = SVN::Git::Fetcher->new($self); +			$gs->ra->gs_do_switch($r0, $rev, $gs, +					      $self->full_url, $ed) +			  or die "SVN connection failed somewhere...\n"; +		} else { +			print STDERR "Following parent with do_update\n"; +			$ed = SVN::Git::Fetcher->new($self); +			$self->ra->gs_do_update($rev, $rev, $self, $ed) +			  or die "SVN connection failed somewhere...\n";  		} -		return 1 if $r_min == $r_max; -		if ($r_min < $r_max) { -			# we need to reverse the print order -			return 0 if (defined $_limit && --$_limit < 0); -			push @$defer, $c; -			return 1; +		print STDERR "Successfully followed parent\n"; +		return $self->make_log_entry($rev, [$parent], $ed); +	} +	return undef; +} + +sub do_fetch { +	my ($self, $paths, $rev) = @_; +	my $ed; +	my ($last_rev, @parents); +	if (my $lc = $self->last_commit) { +		# we can have a branch that was deleted, then re-added +		# under the same name but copied from another path, in +		# which case we'll have multiple parents (we don't +		# want to break the original ref, nor lose copypath info): +		if (my $log_entry = $self->find_parent_branch($paths, $rev)) { +			push @{$log_entry->{parents}}, $lc; +			return $log_entry; +		} +		$ed = SVN::Git::Fetcher->new($self); +		$last_rev = $self->{last_rev}; +		$ed->{c} = $lc; +		@parents = ($lc); +	} else { +		$last_rev = $rev; +		if (my $log_entry = $self->find_parent_branch($paths, $rev)) { +			return $log_entry;  		} -		if ($r_min != $r_max) { -			return 1 if ($r_min < $c->{r}); -			return 1 if ($r_max > $c->{r}); +		$ed = SVN::Git::Fetcher->new($self); +	} +	unless ($self->ra->gs_do_update($last_rev, $rev, $self, $ed)) { +		die "SVN connection failed somewhere...\n"; +	} +	$self->make_log_entry($rev, \@parents, $ed); +} + +sub get_untracked { +	my ($self, $ed) = @_; +	my @out; +	my $h = $ed->{empty}; +	foreach (sort keys %$h) { +		my $act = $h->{$_} ? '+empty_dir' : '-empty_dir'; +		push @out, "  $act: " . uri_encode($_); +		warn "W: $act: $_\n"; +	} +	foreach my $t (qw/dir_prop file_prop/) { +		$h = $ed->{$t} or next; +		foreach my $path (sort keys %$h) { +			my $ppath = $path eq '' ? '.' : $path; +			foreach my $prop (sort keys %{$h->{$path}}) { +				next if $SKIP_PROP{$prop}; +				my $v = $h->{$path}->{$prop}; +				my $t_ppath_prop = "$t: " . +				                    uri_encode($ppath) . ' ' . +				                    uri_encode($prop); +				if (defined $v) { +					push @out, "  +$t_ppath_prop " . +					           uri_encode($v); +				} else { +					push @out, "  -$t_ppath_prop"; +				} +			}  		}  	} -	return 0 if (defined $_limit && --$_limit < 0); -	show_commit($c); -	return 1; +	foreach my $t (qw/absent_file absent_directory/) { +		$h = $ed->{$t} or next; +		foreach my $parent (sort keys %$h) { +			foreach my $path (sort @{$h->{$parent}}) { +				push @out, "  $t: " . +				           uri_encode("$parent/$path"); +				warn "W: $t: $parent/$path ", +				     "Insufficient permissions?\n"; +			} +		} +	} +	\@out;  } -sub show_commit { -	my $c = shift; -	if ($_oneline) { -		my $x = "\n"; -		if (my $l = $c->{l}) { -			while ($l->[0] =~ /^\s*$/) { shift @$l } -			$x = $l->[0]; +sub parse_svn_date { +	my $date = shift || return '+0000 1970-01-01 00:00:00'; +	my ($Y,$m,$d,$H,$M,$S) = ($date =~ /^(\d{4})\-(\d\d)\-(\d\d)T +	                                    (\d\d)\:(\d\d)\:(\d\d).\d+Z$/x) or +	                                 croak "Unable to parse date: $date\n"; +	"+0000 $Y-$m-$d $H:$M:$S"; +} + +sub check_author { +	my ($author) = @_; +	if (!defined $author || length $author == 0) { +		$author = '(no author)'; +	} +	if (defined $::_authors && ! defined $::users{$author}) { +		die "Author: $author not defined in $::_authors file\n"; +	} +	$author; +} + +sub make_log_entry { +	my ($self, $rev, $parents, $ed) = @_; +	my $untracked = $self->get_untracked($ed); + +	open my $un, '>>', "$self->{dir}/unhandled.log" or croak $!; +	print $un "r$rev\n" or croak $!; +	print $un $_, "\n" foreach @$untracked; +	my %log_entry = ( parents => $parents || [], revision => $rev, +	                  log => ''); + +	my $headrev; +	my $logged = delete $self->{logged_rev_props}; +	if (!$logged || $self->{-want_revprops}) { +		my $rp = $self->ra->rev_proplist($rev); +		foreach (sort keys %$rp) { +			my $v = $rp->{$_}; +			if (/^svn:(author|date|log)$/) { +				$log_entry{$1} = $v; +			} elsif ($_ eq 'svm:headrev') { +				$headrev = $v; +			} else { +				print $un "  rev_prop: ", uri_encode($_), ' ', +					  uri_encode($v), "\n"; +			}  		} -		$_l_fmt ||= 'A' . length($c->{r}); -		print 'r',pack($_l_fmt, $c->{r}),' | '; -		print "$c->{c} | " if $_show_commit; -		print $x;  	} else { -		show_commit_normal($c); +		map { $log_entry{$_} = $logged->{$_} } keys %$logged; +	} +	close $un or croak $!; + +	$log_entry{date} = parse_svn_date($log_entry{date}); +	$log_entry{log} .= "\n"; +	my $author = $log_entry{author} = check_author($log_entry{author}); +	my ($name, $email) = defined $::users{$author} ? @{$::users{$author}} +	                                               : ($author, undef); +	if (defined $headrev && $self->use_svm_props) { +		if ($self->rewrite_root) { +			die "Can't have both 'useSvmProps' and 'rewriteRoot' ", +			    "options set!\n"; +		} +		my ($uuid, $r) = $headrev =~ m{^([a-f\d\-]{30,}):(\d+)$}; +		# we don't want "SVM: initializing mirror for junk" ... +		return undef if $r == 0; +		my $svm = $self->svm; +		if ($uuid ne $svm->{uuid}) { +			die "UUID mismatch on SVM path:\n", +			    "expected: $svm->{uuid}\n", +			    "     got: $uuid\n"; +		} +		my $full_url = $self->full_url; +		$full_url =~ s#^\Q$svm->{replace}\E(/|$)#$svm->{source}$1# or +		             die "Failed to replace '$svm->{replace}' with ", +		                 "'$svm->{source}' in $full_url\n"; +		# throw away username for storing in records +		remove_username($full_url); +		$log_entry{metadata} = "$full_url\@$r $uuid"; +		$log_entry{svm_revision} = $r; +		$email ||= "$author\@$uuid" +	} elsif ($self->use_svnsync_props) { +		my $full_url = $self->svnsync->{url}; +		$full_url .= "/$self->{path}" if length $self->{path}; +		my $uuid = $self->svnsync->{uuid}; +		$log_entry{metadata} = "$full_url\@$rev $uuid"; +		$email ||= "$author\@$uuid" +	} else { +		$log_entry{metadata} = $self->metadata_url. "\@$rev " . +		                       $self->ra->get_uuid; +		$email ||= "$author\@" . $self->ra->get_uuid;  	} +	$log_entry{name} = $name; +	$log_entry{email} = $email; +	\%log_entry;  } -sub show_commit_changed_paths { -	my ($c) = @_; -	return unless $c->{changed}; -	print "Changed paths:\n", @{$c->{changed}}; +sub fetch { +	my ($self, $min_rev, $max_rev, @parents) = @_; +	my ($last_rev, $last_commit) = $self->last_rev_commit; +	my ($base, $head) = $self->get_fetch_range($min_rev, $max_rev); +	$self->ra->gs_fetch_loop_common($base, $head, [$self]);  } -sub show_commit_normal { -	my ($c) = @_; -	print '-' x72, "\nr$c->{r} | "; -	print "$c->{c} | " if $_show_commit; -	print "$c->{a} | ", strftime("%Y-%m-%d %H:%M:%S %z (%a, %d %b %Y)", -				 localtime($c->{t_utc})), ' | '; -	my $nr_line = 0; +sub set_tree_cb { +	my ($self, $log_entry, $tree, $rev, $date, $author) = @_; +	$self->{inject_parents} = { $rev => $tree }; +	$self->fetch(undef, undef); +} -	if (my $l = $c->{l}) { -		while ($l->[$#$l] eq "\n" && $#$l > 0 -		                          && $l->[($#$l - 1)] eq "\n") { -			pop @$l; +sub set_tree { +	my ($self, $tree) = (shift, shift); +	my $log_entry = ::get_commit_entry($tree); +	unless ($self->{last_rev}) { +		fatal("Must have an existing revision to commit\n"); +	} +	my %ed_opts = ( r => $self->{last_rev}, +	                log => $log_entry->{log}, +	                ra => $self->ra, +	                tree_a => $self->{last_commit}, +	                tree_b => $tree, +	                editor_cb => sub { +			       $self->set_tree_cb($log_entry, $tree, @_) }, +	                svn_path => $self->{path} ); +	if (!SVN::Git::Editor->new(\%ed_opts)->apply_diff) { +		print "No changes\nr$self->{last_rev} = $tree\n"; +	} +} + +sub rebuild { +	my ($self) = @_; +	my $db_path = $self->db_path; +	return if (-e $db_path && ! -z $db_path); +	return unless ::verify_ref($self->refname.'^0'); +	if (-f $self->{db_root}) { +		rename $self->{db_root}, $db_path or die +		     "rename $self->{db_root} => $db_path failed: $!\n"; +		my ($dir, $base) = ($db_path =~ m#^(.*?)/?([^/]+)$#); +		symlink $base, $self->{db_root} or die +		     "symlink $base => $self->{db_root} failed: $!\n"; +		return; +	} +	print "Rebuilding $db_path ...\n"; +	my ($rev_list, $ctx) = command_output_pipe("rev-list", $self->refname); +	my $latest; +	my $full_url = $self->full_url; +	remove_username($full_url); +	my $svn_uuid; +	while (<$rev_list>) { +		chomp; +		my $c = $_; +		die "Non-SHA1: $c\n" unless $c =~ /^$::sha1$/o; +		my ($url, $rev, $uuid) = ::cmt_metadata($c); +		remove_username($url); + +		# ignore merges (from set-tree) +		next if (!defined $rev || !$uuid); + +		# if we merged or otherwise started elsewhere, this is +		# how we break out of it +		if ((defined $svn_uuid && ($uuid ne $svn_uuid)) || +		    ($full_url && $url && ($url ne $full_url))) { +			next;  		} -		$nr_line = scalar @$l; -		if (!$nr_line) { -			print "1 line\n\n\n"; -		} else { -			if ($nr_line == 1) { -				$nr_line = '1 line'; -			} else { -				$nr_line .= ' lines'; -			} -			print $nr_line, "\n"; -			show_commit_changed_paths($c); -			print "\n"; -			print $_ foreach @$l; +		$latest ||= $rev; +		$svn_uuid ||= $uuid; + +		$self->rev_db_set($rev, $c); +		print "r$rev = $c\n"; +	} +	command_close_pipe($rev_list, $ctx); +	print "Done rebuilding $db_path\n"; +} + +# rev_db: +# Tie::File seems to be prone to offset errors if revisions get sparse, +# it's not that fast, either.  Tie::File is also not in Perl 5.6.  So +# one of my favorite modules is out :<  Next up would be one of the DBM +# modules, but I'm not sure which is most portable...  So I'll just +# go with something that's plain-text, but still capable of +# being randomly accessed.  So here's my ultra-simple fixed-width +# database.  All records are 40 characters + "\n", so it's easy to seek +# to a revision: (41 * rev) is the byte offset. +# A record of 40 0s denotes an empty revision. +# And yes, it's still pretty fast (faster than Tie::File). +# These files are disposable unless noMetadata or useSvmProps is set + +sub _rev_db_set { +	my ($fh, $rev, $commit) = @_; +	my $offset = $rev * 41; +	# assume that append is the common case: +	seek $fh, 0, 2 or croak $!; +	my $pos = tell $fh; +	if ($pos < $offset) { +		for (1 .. (($offset - $pos) / 41)) { +			print $fh (('0' x 40),"\n") or croak $!;  		} +	} +	seek $fh, $offset, 0 or croak $!; +	print $fh $commit,"\n" or croak $!; +} + +sub mkfile { +	my ($path) = @_; +	unless (-e $path) { +		my ($dir, $base) = ($path =~ m#^(.*?)/?([^/]+)$#); +		mkpath([$dir]) unless -d $dir; +		open my $fh, '>>', $path or die "Couldn't create $path: $!\n"; +		close $fh or die "Couldn't close (create) $path: $!\n"; +	} +} + +sub rev_db_set { +	my ($self, $rev, $commit, $update_ref, $uuid) = @_; +	length $commit == 40 or die "arg3 must be a full SHA1 hexsum\n"; +	my $db = $self->db_path($uuid); +	my $db_lock = "$db.lock"; +	my $sig; +	if ($update_ref) { +		$SIG{INT} = $SIG{HUP} = $SIG{TERM} = $SIG{ALRM} = $SIG{PIPE} = +		            $SIG{USR1} = $SIG{USR2} = sub { $sig = $_[0] }; +	} +	mkfile($db); + +	$LOCKFILES{$db_lock} = 1; +	my $sync; +	# both of these options make our .rev_db file very, very important +	# and we can't afford to lose it because rebuild() won't work +	if ($self->use_svm_props || $self->no_metadata) { +		$sync = 1; +		copy($db, $db_lock) or die "rev_db_set(@_): ", +					   "Failed to copy: ", +					   "$db => $db_lock ($!)\n";  	} else { -		print "1 line\n"; -		show_commit_changed_paths($c); -		print "\n"; +		rename $db, $db_lock or die "rev_db_set(@_): ", +					    "Failed to rename: ", +					    "$db => $db_lock ($!)\n"; +	} +	open my $fh, '+<', $db_lock or die "Couldn't open $db_lock: $!\n"; +	_rev_db_set($fh, $rev, $commit); +	if ($sync) { +		$fh->flush or die "Couldn't flush $db_lock: $!\n"; +		$fh->sync or die "Couldn't sync $db_lock: $!\n"; +	} +	close $fh or croak $!; +	if ($update_ref) { +		$_head = $self; +		command_noisy('update-ref', '-m', "r$rev", +		              $self->refname, $commit); +	} +	rename $db_lock, $db or die "rev_db_set(@_): ", "Failed to rename: ", +	                            "$db_lock => $db ($!)\n"; +	delete $LOCKFILES{$db_lock}; +	if ($update_ref) { +		$SIG{INT} = $SIG{HUP} = $SIG{TERM} = $SIG{ALRM} = $SIG{PIPE} = +		            $SIG{USR1} = $SIG{USR2} = 'DEFAULT'; +		kill $sig, $$ if defined $sig; +	} +} +sub rev_db_max { +	my ($self) = @_; +	$self->rebuild; +	my $db_path = $self->db_path; +	my @stat = stat $db_path or return 0; +	($stat[7] % 41) == 0 or die "$db_path inconsistent size: $stat[7]\n"; +	my $max = $stat[7] / 41; +	(($max > 0) ? $max - 1 : 0); +} + +sub rev_db_get { +	my ($self, $rev, $uuid) = @_; +	my $ret; +	my $offset = $rev * 41; +	my $db_path = $self->db_path($uuid); +	return undef unless -e $db_path; +	open my $fh, '<', $db_path or croak $!; +	if (sysseek($fh, $offset, 0) == $offset) { +		my $read = sysread($fh, $ret, 40); +		$ret = undef if ($read != 40 || $ret eq ('0'x40));  	} -	foreach my $x (qw/raw diff/) { -		if ($c->{$x}) { -			print "\n"; -			print $_ foreach @{$c->{$x}} +	close $fh or croak $!; +	$ret; +} + +sub find_rev_before { +	my ($self, $rev, $eq_ok) = @_; +	--$rev unless $eq_ok; +	while ($rev > 0) { +		if (my $c = $self->rev_db_get($rev)) { +			return ($rev, $c);  		} +		--$rev; +	} +	return (undef, undef); +} + +sub _new { +	my ($class, $repo_id, $ref_id, $path) = @_; +	unless (defined $repo_id && length $repo_id) { +		$repo_id = $Git::SVN::default_repo_id;  	} +	unless (defined $ref_id && length $ref_id) { +		$_[2] = $ref_id = $Git::SVN::default_ref_id; +	} +	$_[1] = $repo_id = sanitize_remote_name($repo_id); +	my $dir = "$ENV{GIT_DIR}/svn/$ref_id"; +	$_[3] = $path = '' unless (defined $path); +	mkpath(["$ENV{GIT_DIR}/svn"]); +	bless { +		ref_id => $ref_id, dir => $dir, index => "$dir/index", +	        path => $path, config => "$ENV{GIT_DIR}/svn/config", +	        db_root => "$dir/.rev_db", repo_id => $repo_id }, $class; +} + +sub db_path { +	my ($self, $uuid) = @_; +	$uuid ||= $self->ra_uuid; +	"$self->{db_root}.$uuid"; +} + +sub uri_encode { +	my ($f) = @_; +	$f =~ s#([^a-zA-Z0-9\*!\:_\./\-])#uc sprintf("%%%02x",ord($1))#eg; +	$f +} + +sub remove_username { +	$_[0] =~ s{^([^:]*://)[^@]+@}{$1};  } -sub _simple_prompt { +package Git::SVN::Prompt; +use strict; +use warnings; +require SVN::Core; +use vars qw/$_no_auth_cache $_username/; + +sub simple {  	my ($cred, $realm, $default_username, $may_save, $pool) = @_;  	$may_save = undef if $_no_auth_cache;  	$default_username = $_username if defined $_username; @@ -1923,7 +2115,7 @@ sub _simple_prompt {  		}  		$cred->username($default_username);  	} else { -		_username_prompt($cred, $realm, $may_save, $pool); +		username($cred, $realm, $may_save, $pool);  	}  	$cred->password(_read_password("Password for '" .  	                               $cred->username . "': ", $realm)); @@ -1931,7 +2123,7 @@ sub _simple_prompt {  	$SVN::_Core::SVN_NO_ERROR;  } -sub _ssl_server_trust_prompt { +sub ssl_server_trust {  	my ($cred, $realm, $failures, $cert_info, $may_save, $pool) = @_;  	$may_save = undef if $_no_auth_cache;  	print STDERR "Error validating server certificate for '$realm':\n"; @@ -1980,7 +2172,7 @@ prompt:  	$SVN::_Core::SVN_NO_ERROR;  } -sub _ssl_client_cert_prompt { +sub ssl_client_cert {  	my ($cred, $realm, $may_save, $pool) = @_;  	$may_save = undef if $_no_auth_cache;  	print STDERR "Client certificate filename: "; @@ -1991,7 +2183,7 @@ sub _ssl_client_cert_prompt {  	$SVN::_Core::SVN_NO_ERROR;  } -sub _ssl_client_cert_pw_prompt { +sub ssl_client_cert_pw {  	my ($cred, $realm, $may_save, $pool) = @_;  	$may_save = undef if $_no_auth_cache;  	$cred->password(_read_password("Password: ", $realm)); @@ -1999,7 +2191,7 @@ sub _ssl_client_cert_pw_prompt {  	$SVN::_Core::SVN_NO_ERROR;  } -sub _username_prompt { +sub username {  	my ($cred, $realm, $may_save, $pool) = @_;  	$may_save = undef if $_no_auth_cache;  	if (defined $realm && length $realm) { @@ -2035,546 +2227,7 @@ sub _read_password {  	$password;  } -sub libsvn_connect { -	my ($url) = @_; -	SVN::_Core::svn_config_ensure($_config_dir, undef); -	my ($baton, $callbacks) = SVN::Core::auth_open_helper([ -	    SVN::Client::get_simple_provider(), -	    SVN::Client::get_ssl_server_trust_file_provider(), -	    SVN::Client::get_simple_prompt_provider( -	      \&_simple_prompt, 2), -	    SVN::Client::get_ssl_client_cert_prompt_provider( -	      \&_ssl_client_cert_prompt, 2), -	    SVN::Client::get_ssl_client_cert_pw_prompt_provider( -	      \&_ssl_client_cert_pw_prompt, 2), -	    SVN::Client::get_username_provider(), -	    SVN::Client::get_ssl_server_trust_prompt_provider( -	      \&_ssl_server_trust_prompt), -	    SVN::Client::get_username_prompt_provider( -	      \&_username_prompt, 2), -	  ]); -	my $config = SVN::Core::config_get_config($_config_dir); -	my $ra = SVN::Ra->new(url => $url, auth => $baton, -	                      config => $config, -	                      pool => SVN::Pool->new, -	                      auth_provider_callbacks => $callbacks); -	$ra->{svn_path} = $url; -	$ra->{repos_root} = $ra->get_repos_root; -	$ra->{svn_path} =~ s#^\Q$ra->{repos_root}\E/*##; -	push @repo_path_split_cache, qr/^(\Q$ra->{repos_root}\E)/; -	return $ra; -} - -sub libsvn_can_do_switch { -	unless (defined $_svn_can_do_switch) { -		my $pool = SVN::Pool->new; -		my $rep = eval { -			$SVN->do_switch(1, '', 0, $SVN->{url}, -			                SVN::Delta::Editor->new, $pool); -		}; -		if ($@) { -			$_svn_can_do_switch = 0; -		} else { -			$rep->abort_report($pool); -			$_svn_can_do_switch = 1; -		} -		$pool->clear; -	} -	$_svn_can_do_switch; -} - -sub libsvn_dup_ra { -	my ($ra) = @_; -	SVN::Ra->new(map { $_ => $ra->{$_} } qw/config url -	             auth auth_provider_callbacks repos_root svn_path/); -} - -sub uri_encode { -	my ($f) = @_; -	$f =~ s#([^a-zA-Z0-9\*!\:_\./\-])#uc sprintf("%%%02x",ord($1))#eg; -	$f -} - -sub uri_decode { -	my ($f) = @_; -	$f =~ tr/+/ /; -	$f =~ s/%([A-F0-9]{2})/chr hex($1)/ge; -	$f -} - -sub libsvn_log_entry { -	my ($rev, $author, $date, $msg, $parents, $untracked) = @_; -	my ($Y,$m,$d,$H,$M,$S) = ($date =~ /^(\d{4})\-(\d\d)\-(\d\d)T -					 (\d\d)\:(\d\d)\:(\d\d).\d+Z$/x) -				or die "Unable to parse date: $date\n"; -	if (defined $author && length $author > 0 && -	    defined $_authors && ! defined $users{$author}) { -		die "Author: $author not defined in $_authors file\n"; -	} -	$msg = '' if ($rev == 0 && !defined $msg); - -	open my $un, '>>', "$GIT_SVN_DIR/unhandled.log" or croak $!; -	my $h; -	print $un "r$rev\n" or croak $!; -	$h = $untracked->{empty}; -	foreach (sort keys %$h) { -		my $act = $h->{$_} ? '+empty_dir' : '-empty_dir'; -		print $un "  $act: ", uri_encode($_), "\n" or croak $!; -		warn "W: $act: $_\n"; -	} -	foreach my $t (qw/dir_prop file_prop/) { -		$h = $untracked->{$t} or next; -		foreach my $path (sort keys %$h) { -			my $ppath = $path eq '' ? '.' : $path; -			foreach my $prop (sort keys %{$h->{$path}}) { -				next if $SKIP{$prop}; -				my $v = $h->{$path}->{$prop}; -				if (defined $v) { -					print $un "  +$t: ", -						  uri_encode($ppath), ' ', -						  uri_encode($prop), ' ', -						  uri_encode($v), "\n" -						  or croak $!; -				} else { -					print $un "  -$t: ", -						  uri_encode($ppath), ' ', -						  uri_encode($prop), "\n" -						  or croak $!; -				} -			} -		} -	} -	foreach my $t (qw/absent_file absent_directory/) { -		$h = $untracked->{$t} or next; -		foreach my $parent (sort keys %$h) { -			foreach my $path (sort @{$h->{$parent}}) { -				print $un "  $t: ", -				      uri_encode("$parent/$path"), "\n" -				      or croak $!; -				warn "W: $t: $parent/$path ", -				     "Insufficient permissions?\n"; -			} -		} -	} - -	# revprops (make this optional? it's an extra network trip...) -	my $pool = SVN::Pool->new; -	my $rp = $SVN->rev_proplist($rev, $pool); -	foreach (sort keys %$rp) { -		next if /^svn:(?:author|date|log)$/; -		print $un "  rev_prop: ", uri_encode($_), ' ', -		          uri_encode($rp->{$_}), "\n"; -	} -	$pool->clear; -	close $un or croak $!; - -	{ revision => $rev, date => "+0000 $Y-$m-$d $H:$M:$S", -	  author => $author, msg => $msg."\n", parents => $parents || [], -	  revprops => $rp } -} - -sub process_rm { -	my ($gui, $last_commit, $f, $q) = @_; -	# remove entire directories. -	if (command('ls-tree',$last_commit,'--',$f) =~ /^040000 tree/) { -		my ($ls, $ctx) = command_output_pipe(qw/ls-tree -		                                     -r --name-only -z/, -				                     $last_commit,'--',$f); -		local $/ = "\0"; -		while (<$ls>) { -			print $gui '0 ',0 x 40,"\t",$_ or croak $!; -			print "\tD\t$_\n" unless $q; -		} -		print "\tD\t$f/\n" unless $q; -		command_close_pipe($ls, $ctx); -		return $SVN::Node::dir; -	} else { -		print $gui '0 ',0 x 40,"\t",$f,"\0" or croak $!; -		print "\tD\t$f\n" unless $q; -		return $SVN::Node::file; -	} -} - -sub libsvn_fetch { -	my ($last_commit, $paths, $rev, $author, $date, $msg) = @_; -	my $pool = SVN::Pool->new; -	my $ed = SVN::Git::Fetcher->new({ c => $last_commit, q => $_q }); -	my $reporter = $SVN->do_update($rev, '', 1, $ed, $pool); -	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef) : (); -	my (undef, $last_rev, undef) = cmt_metadata($last_commit); -	$reporter->set_path('', $last_rev, 0, @lock, $pool); -	$reporter->finish_report($pool); -	$pool->clear; -	unless ($ed->{git_commit_ok}) { -		die "SVN connection failed somewhere...\n"; -	} -	libsvn_log_entry($rev, $author, $date, $msg, [$last_commit], $ed); -} - -sub svn_grab_base_rev { -	my $c = eval { command_oneline([qw/rev-parse --verify/, -	                                "refs/remotes/$GIT_SVN^0"], -				        { STDERR => 0 }) }; -	if (defined $c && length $c) { -		my ($url, $rev, $uuid) = cmt_metadata($c); -		return ($rev, $c) if defined $rev; -	} -	if ($_no_metadata) { -		my $offset = -41; # from tail -		my $rl; -		open my $fh, '<', $REVDB or -			die "--no-metadata specified and $REVDB not readable\n"; -		seek $fh, $offset, 2; -		$rl = readline $fh; -		defined $rl or return (undef, undef); -		chomp $rl; -		while ($c ne $rl && tell $fh != 0) { -			$offset -= 41; -			seek $fh, $offset, 2; -			$rl = readline $fh; -			defined $rl or return (undef, undef); -			chomp $rl; -		} -		my $rev = tell $fh; -		croak $! if ($rev < -1); -		$rev =  ($rev - 41) / 41; -		close $fh or croak $!; -		return ($rev, $c); -	} -	return (undef, undef); -} - -sub libsvn_parse_revision { -	my $base = shift; -	my $head = $SVN->get_latest_revnum(); -	if (!defined $_revision || $_revision eq 'BASE:HEAD') { -		return ($base + 1, $head) if (defined $base); -		return (0, $head); -	} -	return ($1, $2) if ($_revision =~ /^(\d+):(\d+)$/); -	return ($_revision, $_revision) if ($_revision =~ /^\d+$/); -	if ($_revision =~ /^BASE:(\d+)$/) { -		return ($base + 1, $1) if (defined $base); -		return (0, $head); -	} -	return ($1, $head) if ($_revision =~ /^(\d+):HEAD$/); -	die "revision argument: $_revision not understood by git-svn\n", -		"Try using the command-line svn client instead\n"; -} - -sub libsvn_traverse_ignore { -	my ($fh, $path, $r) = @_; -	$path =~ s#^/+##g; -	my $pool = SVN::Pool->new; -	my ($dirent, undef, $props) = $SVN->get_dir($path, $r, $pool); -	my $p = $path; -	$p =~ s#^\Q$SVN->{svn_path}\E/##; -	print $fh length $p ? "\n# $p\n" : "\n# /\n"; -	if (my $s = $props->{'svn:ignore'}) { -		$s =~ s/[\r\n]+/\n/g; -		chomp $s; -		if (length $p == 0) { -			$s =~ s#\n#\n/$p#g; -			print $fh "/$s\n"; -		} else { -			$s =~ s#\n#\n/$p/#g; -			print $fh "/$p/$s\n"; -		} -	} -	foreach (sort keys %$dirent) { -		next if $dirent->{$_}->kind != $SVN::Node::dir; -		libsvn_traverse_ignore($fh, "$path/$_", $r); -	} -	$pool->clear; -} - -sub revisions_eq { -	my ($path, $r0, $r1) = @_; -	return 1 if $r0 == $r1; -	my $nr = 0; -	# should be OK to use Pool here (r1 - r0) should be small -	my $pool = SVN::Pool->new; -	libsvn_get_log($SVN, [$path], $r0, $r1, -			0, 0, 1, sub {$nr++}, $pool); -	$pool->clear; -	return 0 if ($nr > 1); -	return 1; -} - -sub libsvn_find_parent_branch { -	my ($paths, $rev, $author, $date, $msg) = @_; -	my $svn_path = '/'.$SVN->{svn_path}; - -	# look for a parent from another branch: -	my $i = $paths->{$svn_path} or return; -	my $branch_from = $i->copyfrom_path or return; -	my $r = $i->copyfrom_rev; -	print STDERR  "Found possible branch point: ", -				"$branch_from => $svn_path, $r\n"; -	$branch_from =~ s#^/##; -	my $l_map = {}; -	read_url_paths_all($l_map, '', "$GIT_DIR/svn"); -	my $url = $SVN->{repos_root}; -	defined $l_map->{$url} or return; -	my $id = $l_map->{$url}->{$branch_from}; -	if (!defined $id && $_follow_parent) { -		print STDERR "Following parent: $branch_from\@$r\n"; -		# auto create a new branch and follow it -		$id = basename($branch_from); -		$id .= '@'.$r if -r "$GIT_DIR/svn/$id"; -		while (-r "$GIT_DIR/svn/$id") { -			# just grow a tail if we're not unique enough :x -			$id .= '-'; -		} -	} -	return unless defined $id; - -	my ($r0, $parent) = find_rev_before($r,$id,1); -	if ($_follow_parent && (!defined $r0 || !defined $parent)) { -		defined(my $pid = fork) or croak $!; -		if (!$pid) { -			$GIT_SVN = $ENV{GIT_SVN_ID} = $id; -			init_vars(); -			$SVN_URL = "$url/$branch_from"; -			$SVN = undef; -			setup_git_svn(); -			# we can't assume SVN_URL exists at r+1: -			$_revision = "0:$r"; -			fetch_lib(); -			exit 0; -		} -		waitpid $pid, 0; -		croak $? if $?; -		($r0, $parent) = find_rev_before($r,$id,1); -	} -	return unless (defined $r0 && defined $parent); -	if (revisions_eq($branch_from, $r0, $r)) { -		unlink $GIT_SVN_INDEX; -		print STDERR "Found branch parent: ($GIT_SVN) $parent\n"; -		command_noisy('read-tree', $parent); -		unless (libsvn_can_do_switch()) { -			return _libsvn_new_tree($paths, $rev, $author, $date, -			                        $msg, [$parent]); -		} -		# do_switch works with svn/trunk >= r22312, but that is not -		# included with SVN 1.4.2 (the latest version at the moment), -		# so we can't rely on it. -		my $ra = libsvn_connect("$url/$branch_from"); -		my $ed = SVN::Git::Fetcher->new({c => $parent, q => $_q }); -		my $pool = SVN::Pool->new; -		my $reporter = $ra->do_switch($rev, '', 1, $SVN->{url}, -		                              $ed, $pool); -		my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef) : (); -		$reporter->set_path('', $r0, 0, @lock, $pool); -		$reporter->finish_report($pool); -		$pool->clear; -		unless ($ed->{git_commit_ok}) { -			die "SVN connection failed somewhere...\n"; -		} -		return libsvn_log_entry($rev, $author, $date, $msg, [$parent]); -	} -	print STDERR "Nope, branch point not imported or unknown\n"; -	return undef; -} - -sub libsvn_get_log { -	my ($ra, @args) = @_; -	$args[4]-- if $args[4] && ! $_follow_parent; -	if ($SVN::Core::VERSION le '1.2.0') { -		splice(@args, 3, 1); -	} -	$ra->get_log(@args); -} - -sub libsvn_new_tree { -	if (my $log_entry = libsvn_find_parent_branch(@_)) { -		return $log_entry; -	} -	my ($paths, $rev, $author, $date, $msg) = @_; # $pool is last -	_libsvn_new_tree($paths, $rev, $author, $date, $msg, []); -} - -sub _libsvn_new_tree { -	my ($paths, $rev, $author, $date, $msg, $parents) = @_; -	my $pool = SVN::Pool->new; -	my $ed = SVN::Git::Fetcher->new({q => $_q}); -	my $reporter = $SVN->do_update($rev, '', 1, $ed, $pool); -	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef) : (); -	$reporter->set_path('', $rev, 1, @lock, $pool); -	$reporter->finish_report($pool); -	$pool->clear; -	unless ($ed->{git_commit_ok}) { -		die "SVN connection failed somewhere...\n"; -	} -	libsvn_log_entry($rev, $author, $date, $msg, $parents, $ed); -} - -sub find_graft_path_commit { -	my ($tree_paths, $p1, $r1) = @_; -	foreach my $x (keys %$tree_paths) { -		next unless ($p1 =~ /^\Q$x\E/); -		my $i = $tree_paths->{$x}; -		my ($r0, $parent) = find_rev_before($r1,$i,1); -		return $parent if (defined $r0 && $r0 == $r1); -		print STDERR "r$r1 of $i not imported\n"; -		next; -	} -	return undef; -} - -sub find_graft_path_parents { -	my ($grafts, $tree_paths, $c, $p0, $r0) = @_; -	foreach my $x (keys %$tree_paths) { -		next unless ($p0 =~ /^\Q$x\E/); -		my $i = $tree_paths->{$x}; -		my ($r, $parent) = find_rev_before($r0, $i, 1); -		if (defined $r && defined $parent && revisions_eq($x,$r,$r0)) { -			my ($url_b, undef, $uuid_b) = cmt_metadata($c); -			my ($url_a, undef, $uuid_a) = cmt_metadata($parent); -			next if ($url_a && $url_b && $url_a eq $url_b && -							$uuid_b eq $uuid_a); -			$grafts->{$c}->{$parent} = 1; -		} -	} -} - -sub libsvn_graft_file_copies { -	my ($grafts, $tree_paths, $path, $paths, $rev) = @_; -	foreach (keys %$paths) { -		my $i = $paths->{$_}; -		my ($m, $p0, $r0) = ($i->action, $i->copyfrom_path, -					$i->copyfrom_rev); -		next unless (defined $p0 && defined $r0); - -		my $p1 = $_; -		$p1 =~ s#^/##; -		$p0 =~ s#^/##; -		my $c = find_graft_path_commit($tree_paths, $p1, $rev); -		next unless $c; -		find_graft_path_parents($grafts, $tree_paths, $c, $p0, $r0); -	} -} - -sub set_index { -	my $old = $ENV{GIT_INDEX_FILE}; -	$ENV{GIT_INDEX_FILE} = shift; -	return $old; -} - -sub restore_index { -	my ($old) = @_; -	if (defined $old) { -		$ENV{GIT_INDEX_FILE} = $old; -	} else { -		delete $ENV{GIT_INDEX_FILE}; -	} -} - -sub libsvn_commit_cb { -	my ($rev, $date, $committer, $c, $msg, $r_last, $cmt_last) = @_; -	if ($_optimize_commits && $rev == ($r_last + 1)) { -		my $log = libsvn_log_entry($rev,$committer,$date,$msg); -		$log->{tree} = get_tree_from_treeish($c); -		my $cmt = git_commit($log, $cmt_last, $c); -		my @diff = command('diff-tree', $cmt, $c); -		if (@diff) { -			print STDERR "Trees differ: $cmt $c\n", -					join('',@diff),"\n"; -			exit 1; -		} -	} else { -		fetch("$rev=$c"); -	} -} - -sub libsvn_ls_fullurl { -	my $fullurl = shift; -	my $ra = libsvn_connect($fullurl); -	my @ret; -	my $pool = SVN::Pool->new; -	my $r = defined $_revision ? $_revision : $ra->get_latest_revnum; -	my ($dirent, undef, undef) = $ra->get_dir('', $r, $pool); -	foreach my $d (sort keys %$dirent) { -		if ($dirent->{$d}->kind == $SVN::Node::dir) { -			push @ret, "$d/"; # add '/' for compat with cli svn -		} -	} -	$pool->clear; -	return @ret; -} - - -sub libsvn_skip_unknown_revs { -	my $err = shift; -	my $errno = $err->apr_err(); -	# Maybe the branch we're tracking didn't -	# exist when the repo started, so it's -	# not an error if it doesn't, just continue -	# -	# Wonderfully consistent library, eh? -	# 160013 - svn:// and file:// -	# 175002 - http(s):// -	# 175007 - http(s):// (this repo required authorization, too...) -	#   More codes may be discovered later... -	if ($errno == 175007 || $errno == 175002 || $errno == 160013) { -		return; -	} -	croak "Error from SVN, ($errno): ", $err->expanded_message,"\n"; -}; - -# Tie::File seems to be prone to offset errors if revisions get sparse, -# it's not that fast, either.  Tie::File is also not in Perl 5.6.  So -# one of my favorite modules is out :<  Next up would be one of the DBM -# modules, but I'm not sure which is most portable...  So I'll just -# go with something that's plain-text, but still capable of -# being randomly accessed.  So here's my ultra-simple fixed-width -# database.  All records are 40 characters + "\n", so it's easy to seek -# to a revision: (41 * rev) is the byte offset. -# A record of 40 0s denotes an empty revision. -# And yes, it's still pretty fast (faster than Tie::File). -sub revdb_set { -	my ($file, $rev, $commit) = @_; -	length $commit == 40 or croak "arg3 must be a full SHA1 hexsum\n"; -	open my $fh, '+<', $file or croak $!; -	my $offset = $rev * 41; -	# assume that append is the common case: -	seek $fh, 0, 2 or croak $!; -	my $pos = tell $fh; -	if ($pos < $offset) { -		print $fh (('0' x 40),"\n") x (($offset - $pos) / 41); -	} -	seek $fh, $offset, 0 or croak $!; -	print $fh $commit,"\n"; -	close $fh or croak $!; -} - -sub revdb_get { -	my ($file, $rev) = @_; -	my $ret; -	my $offset = $rev * 41; -	open my $fh, '<', $file or croak $!; -	seek $fh, $offset, 0; -	if (tell $fh == $offset) { -		$ret = readline $fh; -		if (defined $ret) { -			chomp $ret; -			$ret = undef if ($ret =~ /^0{40}$/); -		} -	} -	close $fh or croak $!; -	return $ret; -} - -sub copy_remote_ref { -	my $origin = $_cp_remote ? $_cp_remote : 'origin'; -	my $ref = "refs/remotes/$GIT_SVN"; -	if (command('ls-remote', $origin, $ref)) { -		command_noisy('fetch', $origin, "$ref:$ref"); -	} elsif ($_cp_remote && !$_upgrade) { -		die "Unable to find remote reference: ", -				"refs/remotes/$GIT_SVN on $origin\n"; -	} -} +package main;  {  	my $kill_stupid_warnings = $SVN::Node::none.$SVN::Node::file. @@ -2594,27 +2247,28 @@ use strict;  use warnings;  use Carp qw/croak/;  use IO::File qw//; -use Git qw/command command_oneline command_noisy -           command_output_pipe command_input_pipe command_close_pipe/; +use Digest::MD5;  # file baton members: path, mode_a, mode_b, pool, fh, blob, base  sub new {  	my ($class, $git_svn) = @_;  	my $self = SVN::Delta::Editor->new;  	bless $self, $class; -	$self->{c} = $git_svn->{c} if exists $git_svn->{c}; -	$self->{q} = $git_svn->{q}; +	$self->{c} = $git_svn->{last_commit} if exists $git_svn->{last_commit};  	$self->{empty} = {};  	$self->{dir_prop} = {};  	$self->{file_prop} = {};  	$self->{absent_dir} = {};  	$self->{absent_file} = {}; -	($self->{gui}, $self->{ctx}) = command_input_pipe( -	                                     qw/update-index -z --index-info/); -	require Digest::MD5; +	$self->{gii} = $git_svn->tmp_index_do(sub { Git::IndexInfo->new });  	$self;  } +sub set_path_strip { +	my ($self, $path) = @_; +	$self->{path_strip} = qr/^\Q$path\E(\/|$)/ if length $path; +} +  sub open_root {  	{ path => '' };  } @@ -2624,16 +2278,46 @@ sub open_directory {  	{ path => $path };  } +sub git_path { +	my ($self, $path) = @_; +	if ($self->{path_strip}) { +		$path =~ s!$self->{path_strip}!! or +		  die "Failed to strip path '$path' ($self->{path_strip})\n"; +	} +	$path; +} +  sub delete_entry {  	my ($self, $path, $rev, $pb) = @_; -	my $t = process_rm($self->{gui}, $self->{c}, $path, $self->{q}); -	$self->{empty}->{$path} = 0 if $t == $SVN::Node::dir; + +	my $gpath = $self->git_path($path); +	return undef if ($gpath eq ''); + +	# remove entire directories. +	if (command('ls-tree', $self->{c}, '--', $gpath) =~ /^040000 tree/) { +		my ($ls, $ctx) = command_output_pipe(qw/ls-tree +		                                     -r --name-only -z/, +				                     $self->{c}, '--', $gpath); +		local $/ = "\0"; +		while (<$ls>) { +			chomp; +			$self->{gii}->remove($_); +			print "\tD\t$_\n" unless $::_q; +		} +		print "\tD\t$gpath/\n" unless $::_q; +		command_close_pipe($ls, $ctx); +		$self->{empty}->{$path} = 0 +	} else { +		$self->{gii}->remove($gpath); +		print "\tD\t$gpath\n" unless $::_q; +	}  	undef;  }  sub open_file {  	my ($self, $path, $pb, $rev) = @_; -	my ($mode, $blob) = (command('ls-tree', $self->{c}, '--',$path) +	my $gpath = $self->git_path($path); +	my ($mode, $blob) = (command('ls-tree', $self->{c}, '--', $gpath)  	                     =~ /^(\d{6}) blob ([a-f\d]{40})\t/);  	unless (defined $mode && defined $blob) {  		die "$path was not found in commit $self->{c} (r$rev)\n"; @@ -2732,7 +2416,7 @@ sub apply_textdelta {  sub close_file {  	my ($self, $fb, $exp) = @_;  	my $hash; -	my $path = $fb->{path}; +	my $path = $self->git_path($fb->{path});  	if (my $fh = $fb->{fh}) {  		seek($fh, 0, 0) or croak $!;  		my $md5 = Digest::MD5->new; @@ -2760,65 +2444,163 @@ sub close_file {  		$hash = $fb->{blob} or die "no blob information\n";  	}  	$fb->{pool}->clear; -	my $gui = $self->{gui}; -	print $gui "$fb->{mode_b} $hash\t$path\0" or croak $!; -	print "\t$fb->{action}\t$path\n" if $fb->{action} && ! $self->{q}; +	$self->{gii}->update($fb->{mode_b}, $hash, $path) or croak $!; +	print "\t$fb->{action}\t$path\n" if $fb->{action} && ! $::_q;  	undef;  }  sub abort_edit {  	my $self = shift; -	eval { command_close_pipe($self->{gui}, $self->{ctx}) }; +	$self->{nr} = $self->{gii}->{nr}; +	delete $self->{gii};  	$self->SUPER::abort_edit(@_);  }  sub close_edit {  	my $self = shift; -	command_close_pipe($self->{gui}, $self->{ctx});  	$self->{git_commit_ok} = 1; +	$self->{nr} = $self->{gii}->{nr}; +	delete $self->{gii};  	$self->SUPER::close_edit(@_);  }  package SVN::Git::Editor; -use vars qw/@ISA/; +use vars qw/@ISA $_rmdir $_cp_similarity $_find_copies_harder $_rename_limit/;  use strict;  use warnings;  use Carp qw/croak/;  use IO::File; -use Git qw/command command_oneline command_noisy -           command_output_pipe command_input_pipe command_close_pipe/; +use Digest::MD5;  sub new { -	my $class = shift; -	my $git_svn = shift; -	my $self = SVN::Delta::Editor->new(@_); +	my ($class, $opts) = @_; +	foreach (qw/svn_path r ra tree_a tree_b log editor_cb/) { +		die "$_ required!\n" unless (defined $opts->{$_}); +	} + +	my $pool = SVN::Pool->new; +	my $mods = generate_diff($opts->{tree_a}, $opts->{tree_b}); +	my $types = check_diff_paths($opts->{ra}, $opts->{svn_path}, +	                             $opts->{r}, $mods); + +	# $opts->{ra} functions should not be used after this: +	my @ce  = $opts->{ra}->get_commit_editor($opts->{log}, +	                                        $opts->{editor_cb}, $pool); +	my $self = SVN::Delta::Editor->new(@ce, $pool);  	bless $self, $class; -	foreach (qw/svn_path c r ra /) { -		die "$_ required!\n" unless (defined $git_svn->{$_}); -		$self->{$_} = $git_svn->{$_}; +	foreach (qw/svn_path r tree_a tree_b/) { +		$self->{$_} = $opts->{$_};  	} -	$self->{pool} = SVN::Pool->new; +	$self->{url} = $opts->{ra}->{url}; +	$self->{mods} = $mods; +	$self->{types} = $types; +	$self->{pool} = $pool;  	$self->{bat} = { '' => $self->open_root($self->{r}, $self->{pool}) };  	$self->{rm} = { }; -	require Digest::MD5; +	$self->{path_prefix} = length $self->{svn_path} ? +	                       "$self->{svn_path}/" : '';  	return $self;  } +sub generate_diff { +	my ($tree_a, $tree_b) = @_; +	my @diff_tree = qw(diff-tree -z -r); +	if ($_cp_similarity) { +		push @diff_tree, "-C$_cp_similarity"; +	} else { +		push @diff_tree, '-C'; +	} +	push @diff_tree, '--find-copies-harder' if $_find_copies_harder; +	push @diff_tree, "-l$_rename_limit" if defined $_rename_limit; +	push @diff_tree, $tree_a, $tree_b; +	my ($diff_fh, $ctx) = command_output_pipe(@diff_tree); +	local $/ = "\0"; +	my $state = 'meta'; +	my @mods; +	while (<$diff_fh>) { +		chomp $_; # this gets rid of the trailing "\0" +		if ($state eq 'meta' && /^:(\d{6})\s(\d{6})\s +					$::sha1\s($::sha1)\s +					([MTCRAD])\d*$/xo) { +			push @mods, {	mode_a => $1, mode_b => $2, +					sha1_b => $3, chg => $4 }; +			if ($4 =~ /^(?:C|R)$/) { +				$state = 'file_a'; +			} else { +				$state = 'file_b'; +			} +		} elsif ($state eq 'file_a') { +			my $x = $mods[$#mods] or croak "Empty array\n"; +			if ($x->{chg} !~ /^(?:C|R)$/) { +				croak "Error parsing $_, $x->{chg}\n"; +			} +			$x->{file_a} = $_; +			$state = 'file_b'; +		} elsif ($state eq 'file_b') { +			my $x = $mods[$#mods] or croak "Empty array\n"; +			if (exists $x->{file_a} && $x->{chg} !~ /^(?:C|R)$/) { +				croak "Error parsing $_, $x->{chg}\n"; +			} +			if (!exists $x->{file_a} && $x->{chg} =~ /^(?:C|R)$/) { +				croak "Error parsing $_, $x->{chg}\n"; +			} +			$x->{file_b} = $_; +			$state = 'meta'; +		} else { +			croak "Error parsing $_\n"; +		} +	} +	command_close_pipe($diff_fh, $ctx); +	\@mods; +} + +sub check_diff_paths { +	my ($ra, $pfx, $rev, $mods) = @_; +	my %types; +	$pfx .= '/' if length $pfx; + +	sub type_diff_paths { +		my ($ra, $types, $path, $rev) = @_; +		my @p = split m#/+#, $path; +		my $c = shift @p; +		unless (defined $types->{$c}) { +			$types->{$c} = $ra->check_path($c, $rev); +		} +		while (@p) { +			$c .= '/' . shift @p; +			next if defined $types->{$c}; +			$types->{$c} = $ra->check_path($c, $rev); +		} +	} + +	foreach my $m (@$mods) { +		foreach my $f (qw/file_a file_b/) { +			next unless defined $m->{$f}; +			my ($dir) = ($m->{$f} =~ m#^(.*?)/?(?:[^/]+)$#); +			if (length $pfx.$dir && ! defined $types{$dir}) { +				type_diff_paths($ra, \%types, $pfx.$dir, $rev); +			} +		} +	} +	\%types; +} +  sub split_path {  	return ($_[0] =~ m#^(.*?)/?([^/]+)$#);  }  sub repo_path { -	(defined $_[1] && length $_[1]) ? $_[1] : '' +	my ($self, $path) = @_; +	$self->{path_prefix}.(defined $path ? $path : '');  }  sub url_path {  	my ($self, $path) = @_; -	$self->{ra}->{url} . '/' . $self->repo_path($path); +	$self->{url} . '/' . $self->repo_path($path);  }  sub rmdirs { -	my ($self, $q) = @_; +	my ($self) = @_;  	my $rm = $self->{rm};  	delete $rm->{''}; # we never delete the url we're tracking  	return unless %$rm; @@ -2836,8 +2618,8 @@ sub rmdirs {  	delete $rm->{''}; # we never delete the url we're tracking  	return unless %$rm; -	my ($fh, $ctx) = command_output_pipe( -	                           qw/ls-tree --name-only -r -z/, $self->{c}); +	my ($fh, $ctx) = command_output_pipe(qw/ls-tree --name-only -r -z/, +	                                     $self->{tree_b});  	local $/ = "\0";  	while (<$fh>) {  		chomp; @@ -2856,7 +2638,7 @@ sub rmdirs {  	foreach my $d (sort { $b =~ tr#/#/# <=> $a =~ tr#/#/# } keys %$rm) {  		$self->close_directory($bat->{$d}, $p);  		my ($dn) = ($d =~ m#^(.*?)/?(?:[^/]+)$#); -		print "\tD+\t$d/\n" unless $q; +		print "\tD+\t$d/\n" unless $::_q;  		$self->SUPER::delete_entry($d, $r, $bat->{$dn}, $p);  		delete $bat->{$d};  	} @@ -2864,9 +2646,10 @@ sub rmdirs {  sub open_or_add_dir {  	my ($self, $full_path, $baton) = @_; -	my $p = SVN::Pool->new; -	my $t = $self->{ra}->check_path($full_path, $self->{r}, $p); -	$p->clear; +	my $t = $self->{types}->{$full_path}; +	if (!defined $t) { +		die "$full_path not known in r$self->{r} or we have a bug!\n"; +	}  	if ($t == $SVN::Node::none) {  		return $self->add_directory($full_path, $baton,  						undef, -1, $self->{pool}); @@ -2883,9 +2666,9 @@ sub open_or_add_dir {  sub ensure_path {  	my ($self, $path) = @_;  	my $bat = $self->{bat}; -	$path = $self->repo_path($path); -	return $bat->{''} unless (length $path); -	my @p = split m#/+#, $path; +	my $repo_path = $self->repo_path($path); +	return $bat->{''} unless (length $repo_path); +	my @p = split m#/+#, $repo_path;  	my $c = shift @p;  	$bat->{$c} ||= $self->open_or_add_dir($c, $bat->{''});  	while (@p) { @@ -2897,23 +2680,23 @@ sub ensure_path {  }  sub A { -	my ($self, $m, $q) = @_; +	my ($self, $m) = @_;  	my ($dir, $file) = split_path($m->{file_b});  	my $pbat = $self->ensure_path($dir);  	my $fbat = $self->add_file($self->repo_path($m->{file_b}), $pbat,  					undef, -1); -	print "\tA\t$m->{file_b}\n" unless $q; +	print "\tA\t$m->{file_b}\n" unless $::_q;  	$self->chg_file($fbat, $m);  	$self->close_file($fbat,undef,$self->{pool});  }  sub C { -	my ($self, $m, $q) = @_; +	my ($self, $m) = @_;  	my ($dir, $file) = split_path($m->{file_b});  	my $pbat = $self->ensure_path($dir);  	my $fbat = $self->add_file($self->repo_path($m->{file_b}), $pbat,  				$self->url_path($m->{file_a}), $self->{r}); -	print "\tC\t$m->{file_a} => $m->{file_b}\n" unless $q; +	print "\tC\t$m->{file_a} => $m->{file_b}\n" unless $::_q;  	$self->chg_file($fbat, $m);  	$self->close_file($fbat,undef,$self->{pool});  } @@ -2927,12 +2710,12 @@ sub delete_entry {  }  sub R { -	my ($self, $m, $q) = @_; +	my ($self, $m) = @_;  	my ($dir, $file) = split_path($m->{file_b});  	my $pbat = $self->ensure_path($dir);  	my $fbat = $self->add_file($self->repo_path($m->{file_b}), $pbat,  				$self->url_path($m->{file_a}), $self->{r}); -	print "\tR\t$m->{file_a} => $m->{file_b}\n" unless $q; +	print "\tR\t$m->{file_a} => $m->{file_b}\n" unless $::_q;  	$self->chg_file($fbat, $m);  	$self->close_file($fbat,undef,$self->{pool}); @@ -2942,12 +2725,12 @@ sub R {  }  sub M { -	my ($self, $m, $q) = @_; +	my ($self, $m) = @_;  	my ($dir, $file) = split_path($m->{file_b});  	my $pbat = $self->ensure_path($dir);  	my $fbat = $self->open_file($self->repo_path($m->{file_b}),  				$pbat,$self->{r},$self->{pool}); -	print "\t$m->{chg}\t$m->{file_b}\n" unless $q; +	print "\t$m->{chg}\t$m->{file_b}\n" unless $::_q;  	$self->chg_file($fbat, $m);  	$self->close_file($fbat,undef,$self->{pool});  } @@ -2998,10 +2781,10 @@ sub chg_file {  }  sub D { -	my ($self, $m, $q) = @_; +	my ($self, $m) = @_;  	my ($dir, $file) = split_path($m->{file_b});  	my $pbat = $self->ensure_path($dir); -	print "\tD\t$m->{file_b}\n" unless $q; +	print "\tD\t$m->{file_b}\n" unless $::_q;  	$self->delete_entry($m->{file_b}, $pbat);  } @@ -3018,22 +2801,1109 @@ sub close_edit {  sub abort_edit {  	my ($self) = @_;  	$self->SUPER::abort_edit($self->{pool}); +} + +sub DESTROY { +	my $self = shift; +	$self->SUPER::DESTROY(@_);  	$self->{pool}->clear;  } +# this drives the editor +sub apply_diff { +	my ($self) = @_; +	my $mods = $self->{mods}; +	my %o = ( D => 1, R => 0, C => -1, A => 3, M => 3, T => 3 ); +	foreach my $m (sort { $o{$a->{chg}} <=> $o{$b->{chg}} } @$mods) { +		my $f = $m->{chg}; +		if (defined $o{$f}) { +			$self->$f($m); +		} else { +			fatal("Invalid change type: $f\n"); +		} +	} +	$self->rmdirs if $_rmdir; +	if (@$mods == 0) { +		$self->abort_edit; +	} else { +		$self->close_edit; +	} +	return scalar @$mods; +} + +package Git::SVN::Ra; +use vars qw/@ISA $config_dir $_log_window_size/; +use strict; +use warnings; +my ($can_do_switch); +my $RA; + +BEGIN { +	# enforce temporary pool usage for some simple functions +	my $e; +	foreach (qw/get_latest_revnum get_uuid get_repos_root/) { +		$e .= "sub $_ { +			my \$self = shift; +			my \$pool = SVN::Pool->new; +			my \@ret = \$self->SUPER::$_(\@_,\$pool); +			\$pool->clear; +			wantarray ? \@ret : \$ret[0]; }\n"; +	} + +	# get_dir needs $pool held in cache for dirents to work, +	# check_path is cacheable and rev_proplist is close enough +	# for our purposes. +	foreach (qw/check_path get_dir rev_proplist/) { +		$e .= "my \%${_}_cache; my \$${_}_rev = 0; sub $_ { +			my \$self = shift; +			my \$r = pop; +			my \$k = join(\"\\0\", \@_); +			if (my \$x = \$${_}_cache{\$r}->{\$k}) { +				return wantarray ? \@\$x : \$x->[0]; +			} +			my \$pool = SVN::Pool->new; +			my \@ret = \$self->SUPER::$_(\@_, \$r, \$pool); +			if (\$r != \$${_}_rev) { +				\%${_}_cache = ( pool => [] ); +				\$${_}_rev = \$r; +			} +			\$${_}_cache{\$r}->{\$k} = \\\@ret; +			push \@{\$${_}_cache{pool}}, \$pool; +			wantarray ? \@ret : \$ret[0]; }\n"; +	} +	$e .= "\n1;"; +	eval $e or die $@; +} + +sub new { +	my ($class, $url) = @_; +	$url =~ s!/+$!!; +	return $RA if ($RA && $RA->{url} eq $url); +	$RA->{pool}->clear if $RA; + +	SVN::_Core::svn_config_ensure($config_dir, undef); +	my ($baton, $callbacks) = SVN::Core::auth_open_helper([ +	    SVN::Client::get_simple_provider(), +	    SVN::Client::get_ssl_server_trust_file_provider(), +	    SVN::Client::get_simple_prompt_provider( +	      \&Git::SVN::Prompt::simple, 2), +	    SVN::Client::get_ssl_client_cert_prompt_provider( +	      \&Git::SVN::Prompt::ssl_client_cert, 2), +	    SVN::Client::get_ssl_client_cert_pw_prompt_provider( +	      \&Git::SVN::Prompt::ssl_client_cert_pw, 2), +	    SVN::Client::get_username_provider(), +	    SVN::Client::get_ssl_server_trust_prompt_provider( +	      \&Git::SVN::Prompt::ssl_server_trust), +	    SVN::Client::get_username_prompt_provider( +	      \&Git::SVN::Prompt::username, 2), +	  ]); +	my $config = SVN::Core::config_get_config($config_dir); +	my $self = SVN::Ra->new(url => $url, auth => $baton, +	                      config => $config, +			      pool => SVN::Pool->new, +	                      auth_provider_callbacks => $callbacks); +	$self->{svn_path} = $url; +	$self->{repos_root} = $self->get_repos_root; +	$self->{svn_path} =~ s#^\Q$self->{repos_root}\E(/|$)##; +	$RA = bless $self, $class; +} + +sub DESTROY { +	# do not call the real DESTROY since we store ourselves in $RA +} + +sub get_log { +	my ($self, @args) = @_; +	my $pool = SVN::Pool->new; +	splice(@args, 3, 1) if ($SVN::Core::VERSION le '1.2.0'); +	my $ret = $self->SUPER::get_log(@args, $pool); +	$pool->clear; +	$ret; +} + +sub get_commit_editor { +	my ($self, $log, $cb, $pool) = @_; +	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef, 0) : (); +	$self->SUPER::get_commit_editor($log, $cb, @lock, $pool); +} + +sub gs_do_update { +	my ($self, $rev_a, $rev_b, $gs, $editor) = @_; +	my $new = ($rev_a == $rev_b); +	my $path = $gs->{path}; + +	if ($new && -e $gs->{index}) { +		unlink $gs->{index} or die +		  "Couldn't unlink index: $gs->{index}: $!\n"; +	} +	my $pool = SVN::Pool->new; +	$editor->set_path_strip($path); +	my (@pc) = split m#/#, $path; +	my $reporter = $self->do_update($rev_b, (@pc ? shift @pc : ''), +	                                1, $editor, $pool); +	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef) : (); + +	# Since we can't rely on svn_ra_reparent being available, we'll +	# just have to do some magic with set_path to make it so +	# we only want a partial path. +	my $sp = ''; +	my $final = join('/', @pc); +	while (@pc) { +		$reporter->set_path($sp, $rev_b, 0, @lock, $pool); +		$sp .= '/' if length $sp; +		$sp .= shift @pc; +	} +	die "BUG: '$sp' != '$final'\n" if ($sp ne $final); + +	$reporter->set_path($sp, $rev_a, $new, @lock, $pool); + +	$reporter->finish_report($pool); +	$pool->clear; +	$editor->{git_commit_ok}; +} + +# this requires SVN 1.4.3 or later (do_switch didn't work before 1.4.3, and +# svn_ra_reparent didn't work before 1.4) +sub gs_do_switch { +	my ($self, $rev_a, $rev_b, $gs, $url_b, $editor) = @_; +	my $path = $gs->{path}; +	my $pool = SVN::Pool->new; + +	my $full_url = $self->{url}; +	my $old_url = $full_url; +	$full_url .= "/$path" if length $path; +	my ($ra, $reparented); +	if ($old_url ne $full_url) { +		if ($old_url !~ m#^svn(\+ssh)?://#) { +			SVN::_Ra::svn_ra_reparent($self->{session}, $full_url, +			                          $pool); +			$self->{url} = $full_url; +			$reparented = 1; +		} else { +			$ra = Git::SVN::Ra->new($full_url); +		} +	} +	$ra ||= $self; +	my $reporter = $ra->do_switch($rev_b, '', 1, $url_b, $editor, $pool); +	my @lock = $SVN::Core::VERSION ge '1.2.0' ? (undef) : (); +	$reporter->set_path('', $rev_a, 0, @lock, $pool); +	$reporter->finish_report($pool); + +	if ($reparented) { +		SVN::_Ra::svn_ra_reparent($self->{session}, $old_url, $pool); +		$self->{url} = $old_url; +	} + +	$pool->clear; +	$editor->{git_commit_ok}; +} + +sub gs_fetch_loop_common { +	my ($self, $base, $head, $gsv, $globs) = @_; +	return if ($base > $head); +	my $inc = $_log_window_size; +	my ($min, $max) = ($base, $head < $base + $inc ? $head : $base + $inc); +	my %common; +	my $common_max = scalar @$gsv; + +	foreach my $gs (@$gsv) { +		my @tmp = split m#/#, $gs->{path}; +		my $p = ''; +		foreach (@tmp) { +			$p .= length($p) ? "/$_" : $_; +			$common{$p} ||= 0; +			$common{$p}++; +		} +	} +	$globs ||= []; +	$common_max += scalar @$globs; +	foreach my $glob (@$globs) { +		my @tmp = split m#/#, $glob->{path}->{left}; +		my $p = ''; +		foreach (@tmp) { +			$p .= length($p) ? "/$_" : $_; +			$common{$p} ||= 0; +			$common{$p}++; +		} +	} + +	my $longest_path = ''; +	foreach (sort {length $b <=> length $a} keys %common) { +		if ($common{$_} == $common_max) { +			$longest_path = $_; +			last; +		} +	} +	while (1) { +		my %revs; +		my $err; +		my $err_handler = $SVN::Error::handler; +		$SVN::Error::handler = sub { +			($err) = @_; +			skip_unknown_revs($err); +		}; +		sub _cb { +			my ($paths, $r, $author, $date, $log) = @_; +			[ dup_changed_paths($paths), +			  { author => $author, date => $date, log => $log } ]; +		} +		$self->get_log([$longest_path], $min, $max, 0, 1, 1, +		               sub { $revs{$_[1]} = _cb(@_) }); +		if ($err && $max >= $head) { +			print STDERR "Path '$longest_path' ", +				     "was probably deleted:\n", +				     $err->expanded_message, +				     "\nWill attempt to follow ", +				     "revisions r$min .. r$max ", +				     "committed before the deletion\n"; +			my $hi = $max; +			while (--$hi >= $min) { +				my $ok; +				$self->get_log([$longest_path], $min, $hi, +				               0, 1, 1, sub { +				               $ok ||= $_[1]; +				               $revs{$_[1]} = _cb(@_) }); +				if ($ok) { +					print STDERR "r$min .. r$ok OK\n"; +					last; +				} +			} +		} +		$SVN::Error::handler = $err_handler; + +		my %exists = map { $_->{path} => $_ } @$gsv; +		foreach my $r (sort {$a <=> $b} keys %revs) { +			my ($paths, $logged) = @{$revs{$r}}; + +			foreach my $gs ($self->match_globs(\%exists, $paths, +			                                   $globs, $r)) { +				if ($gs->rev_db_max >= $r) { +					next; +				} +				next unless $gs->match_paths($paths, $r); +				$gs->{logged_rev_props} = $logged; +				if (my $last_commit = $gs->last_commit) { +					$gs->assert_index_clean($last_commit); +				} +				my $log_entry = $gs->do_fetch($paths, $r); +				if ($log_entry) { +					$gs->do_git_commit($log_entry); +				} +			} +			foreach my $g (@$globs) { +				my $k = "svn-remote.$g->{remote}." . +				        "$g->{t}-maxRev"; +				Git::SVN::tmp_config($k, $r); +			} +		} +		# pre-fill the .rev_db since it'll eventually get filled in +		# with '0' x40 if something new gets committed +		foreach my $gs (@$gsv) { +			next if defined $gs->rev_db_get($max); +			$gs->rev_db_set($max, 0 x40); +		} +		foreach my $g (@$globs) { +			my $k = "svn-remote.$g->{remote}.$g->{t}-maxRev"; +			Git::SVN::tmp_config($k, $max); +		} +		last if $max >= $head; +		$min = $max + 1; +		$max += $inc; +		$max = $head if ($max > $head); +	} +} + +sub match_globs { +	my ($self, $exists, $paths, $globs, $r) = @_; + +	sub get_dir_check { +		my ($self, $exists, $g, $r) = @_; +		my @x = eval { $self->get_dir($g->{path}->{left}, $r) }; +		return unless scalar @x == 3; +		my $dirents = $x[0]; +		foreach my $de (keys %$dirents) { +			next if $dirents->{$de}->kind != $SVN::Node::dir; +			my $p = $g->{path}->full_path($de); +			next if $exists->{$p}; +			next if (length $g->{path}->{right} && +				 ($self->check_path($p, $r) != +				  $SVN::Node::dir)); +			$exists->{$p} = Git::SVN->init($self->{url}, $p, undef, +					 $g->{ref}->full_path($de), 1); +		} +	} +	foreach my $g (@$globs) { +		if (my $path = $paths->{"/$g->{path}->{left}"}) { +			if ($path->{action} =~ /^[AR]$/) { +				get_dir_check($self, $exists, $g, $r); +			} +		} +		foreach (keys %$paths) { +			if (/$g->{path}->{left_regex}/ && +			    !/$g->{path}->{regex}/) { +				next if $paths->{$_}->{action} !~ /^[AR]$/; +				get_dir_check($self, $exists, $g, $r); +			} +			next unless /$g->{path}->{regex}/; +			my $p = $1; +			my $pathname = $g->{path}->full_path($p); +			next if $exists->{$pathname}; +			$exists->{$pathname} = Git::SVN->init( +			                      $self->{url}, $pathname, undef, +			                      $g->{ref}->full_path($p), 1); +		} +		my $c = ''; +		foreach (split m#/#, $g->{path}->{left}) { +			$c .= "/$_"; +			next unless ($paths->{$c} && +			             ($paths->{$c}->{action} =~ /^[AR]$/)); +			get_dir_check($self, $exists, $g, $r); +		} +	} +	values %$exists; +} + +sub minimize_url { +	my ($self) = @_; +	return $self->{url} if ($self->{url} eq $self->{repos_root}); +	my $url = $self->{repos_root}; +	my @components = split(m!/!, $self->{svn_path}); +	my $c = ''; +	do { +		$url .= "/$c" if length $c; +		eval { (ref $self)->new($url)->get_latest_revnum }; +	} while ($@ && ($c = shift @components)); +	$url; +} + +sub can_do_switch { +	my $self = shift; +	unless (defined $can_do_switch) { +		my $pool = SVN::Pool->new; +		my $rep = eval { +			$self->do_switch(1, '', 0, $self->{url}, +			                 SVN::Delta::Editor->new, $pool); +		}; +		if ($@) { +			$can_do_switch = 0; +		} else { +			$rep->abort_report($pool); +			$can_do_switch = 1; +		} +		$pool->clear; +	} +	$can_do_switch; +} + +sub skip_unknown_revs { +	my ($err) = @_; +	my $errno = $err->apr_err(); +	# Maybe the branch we're tracking didn't +	# exist when the repo started, so it's +	# not an error if it doesn't, just continue +	# +	# Wonderfully consistent library, eh? +	# 160013 - svn:// and file:// +	# 175002 - http(s):// +	# 175007 - http(s):// (this repo required authorization, too...) +	#   More codes may be discovered later... +	if ($errno == 175007 || $errno == 175002 || $errno == 160013) { +		warn "W: Ignoring error from SVN, path probably ", +		     "does not exist: ($errno): ", +		     $err->expanded_message,"\n"; +		return; +	} +	die "Error from SVN, ($errno): ", $err->expanded_message,"\n"; +} + +# svn_log_changed_path_t objects passed to get_log are likely to be +# overwritten even if only the refs are copied to an external variable, +# so we should dup the structures in their entirety.  Using an externally +# passed pool (instead of our temporary and quickly cleared pool in +# Git::SVN::Ra) does not help matters at all... +sub dup_changed_paths { +	my ($paths) = @_; +	return undef unless $paths; +	my %ret; +	foreach my $p (keys %$paths) { +		my $i = $paths->{$p}; +		my %s = map { $_ => $i->$_ } +		              qw/copyfrom_path copyfrom_rev action/; +		$ret{$p} = \%s; +	} +	\%ret; +} + +package Git::SVN::Log; +use strict; +use warnings; +use POSIX qw/strftime/; +use vars qw/$TZ $limit $color $pager $non_recursive $verbose $oneline +            %rusers $show_commit $incremental/; +my $l_fmt; + +sub cmt_showable { +	my ($c) = @_; +	return 1 if defined $c->{r}; +	if ($c->{l} && $c->{l}->[-1] eq "...\n" && +				$c->{a_raw} =~ /\@([a-f\d\-]+)>$/) { +		my @log = command(qw/cat-file commit/, $c->{c}); +		shift @log while ($log[0] ne "\n"); +		shift @log; +		@{$c->{l}} = grep !/^git-svn-id: /, @log; + +		(undef, $c->{r}, undef) = ::extract_metadata( +				(grep(/^git-svn-id: /, @log))[-1]); +	} +	return defined $c->{r}; +} + +sub log_use_color { +	return 1 if $color; +	my ($dc, $dcvar); +	$dcvar = 'color.diff'; +	$dc = `git-config --get $dcvar`; +	if ($dc eq '') { +		# nothing at all; fallback to "diff.color" +		$dcvar = 'diff.color'; +		$dc = `git-config --get $dcvar`; +	} +	chomp($dc); +	if ($dc eq 'auto') { +		my $pc; +		$pc = `git-config --get color.pager`; +		if ($pc eq '') { +			# does not have it -- fallback to pager.color +			$pc = `git-config --bool --get pager.color`; +		} +		else { +			$pc = `git-config --bool --get color.pager`; +			if ($?) { +				$pc = 'false'; +			} +		} +		chomp($pc); +		if (-t *STDOUT || (defined $pager && $pc eq 'true')) { +			return ($ENV{TERM} && $ENV{TERM} ne 'dumb'); +		} +		return 0; +	} +	return 0 if $dc eq 'never'; +	return 1 if $dc eq 'always'; +	chomp($dc = `git-config --bool --get $dcvar`); +	return ($dc eq 'true'); +} + +sub git_svn_log_cmd { +	my ($r_min, $r_max, @args) = @_; +	my $head = 'HEAD'; +	foreach my $x (@args) { +		last if $x eq '--'; +		next unless ::verify_ref("$x^0"); +		$head = $x; +		last; +	} + +	my $url = (::working_head_info($head))[0]; +	my $gs = Git::SVN->find_by_url($url) || Git::SVN->_new; +	my @cmd = (qw/log --abbrev-commit --pretty=raw --default/, +	           $gs->refname); +	push @cmd, '-r' unless $non_recursive; +	push @cmd, qw/--raw --name-status/ if $verbose; +	push @cmd, '--color' if log_use_color(); +	return @cmd unless defined $r_max; +	if ($r_max == $r_min) { +		push @cmd, '--max-count=1'; +		if (my $c = $gs->rev_db_get($r_max)) { +			push @cmd, $c; +		} +	} else { +		my ($c_min, $c_max); +		$c_max = $gs->rev_db_get($r_max); +		$c_min = $gs->rev_db_get($r_min); +		if (defined $c_min && defined $c_max) { +			if ($r_max > $r_max) { +				push @cmd, "$c_min..$c_max"; +			} else { +				push @cmd, "$c_max..$c_min"; +			} +		} elsif ($r_max > $r_min) { +			push @cmd, $c_max; +		} else { +			push @cmd, $c_min; +		} +	} +	return @cmd; +} + +# adapted from pager.c +sub config_pager { +	$pager ||= $ENV{GIT_PAGER} || $ENV{PAGER}; +	if (!defined $pager) { +		$pager = 'less'; +	} elsif (length $pager == 0 || $pager eq 'cat') { +		$pager = undef; +	} +} + +sub run_pager { +	return unless -t *STDOUT; +	pipe my $rfd, my $wfd or return; +	defined(my $pid = fork) or ::fatal "Can't fork: $!\n"; +	if (!$pid) { +		open STDOUT, '>&', $wfd or +		                     ::fatal "Can't redirect to stdout: $!\n"; +		return; +	} +	open STDIN, '<&', $rfd or ::fatal "Can't redirect stdin: $!\n"; +	$ENV{LESS} ||= 'FRSX'; +	exec $pager or ::fatal "Can't run pager: $! ($pager)\n"; +} + +sub tz_to_s_offset { +	my ($tz) = @_; +	$tz =~ s/(\d\d)$//; +	return ($1 * 60) + ($tz * 3600); +} + +sub get_author_info { +	my ($dest, $author, $t, $tz) = @_; +	$author =~ s/(?:^\s*|\s*$)//g; +	$dest->{a_raw} = $author; +	my $au; +	if ($::_authors) { +		$au = $rusers{$author} || undef; +	} +	if (!$au) { +		($au) = ($author =~ /<([^>]+)\@[^>]+>$/); +	} +	$dest->{t} = $t; +	$dest->{tz} = $tz; +	$dest->{a} = $au; +	# Date::Parse isn't in the standard Perl distro :( +	if ($tz =~ s/^\+//) { +		$t += tz_to_s_offset($tz); +	} elsif ($tz =~ s/^\-//) { +		$t -= tz_to_s_offset($tz); +	} +	$dest->{t_utc} = $t; +} + +sub process_commit { +	my ($c, $r_min, $r_max, $defer) = @_; +	if (defined $r_min && defined $r_max) { +		if ($r_min == $c->{r} && $r_min == $r_max) { +			show_commit($c); +			return 0; +		} +		return 1 if $r_min == $r_max; +		if ($r_min < $r_max) { +			# we need to reverse the print order +			return 0 if (defined $limit && --$limit < 0); +			push @$defer, $c; +			return 1; +		} +		if ($r_min != $r_max) { +			return 1 if ($r_min < $c->{r}); +			return 1 if ($r_max > $c->{r}); +		} +	} +	return 0 if (defined $limit && --$limit < 0); +	show_commit($c); +	return 1; +} + +sub show_commit { +	my $c = shift; +	if ($oneline) { +		my $x = "\n"; +		if (my $l = $c->{l}) { +			while ($l->[0] =~ /^\s*$/) { shift @$l } +			$x = $l->[0]; +		} +		$l_fmt ||= 'A' . length($c->{r}); +		print 'r',pack($l_fmt, $c->{r}),' | '; +		print "$c->{c} | " if $show_commit; +		print $x; +	} else { +		show_commit_normal($c); +	} +} + +sub show_commit_changed_paths { +	my ($c) = @_; +	return unless $c->{changed}; +	print "Changed paths:\n", @{$c->{changed}}; +} + +sub show_commit_normal { +	my ($c) = @_; +	print '-' x72, "\nr$c->{r} | "; +	print "$c->{c} | " if $show_commit; +	print "$c->{a} | ", strftime("%Y-%m-%d %H:%M:%S %z (%a, %d %b %Y)", +				 localtime($c->{t_utc})), ' | '; +	my $nr_line = 0; + +	if (my $l = $c->{l}) { +		while ($l->[$#$l] eq "\n" && $#$l > 0 +		                          && $l->[($#$l - 1)] eq "\n") { +			pop @$l; +		} +		$nr_line = scalar @$l; +		if (!$nr_line) { +			print "1 line\n\n\n"; +		} else { +			if ($nr_line == 1) { +				$nr_line = '1 line'; +			} else { +				$nr_line .= ' lines'; +			} +			print $nr_line, "\n"; +			show_commit_changed_paths($c); +			print "\n"; +			print $_ foreach @$l; +		} +	} else { +		print "1 line\n"; +		show_commit_changed_paths($c); +		print "\n"; + +	} +	foreach my $x (qw/raw stat diff/) { +		if ($c->{$x}) { +			print "\n"; +			print $_ foreach @{$c->{$x}} +		} +	} +} + +sub cmd_show_log { +	my (@args) = @_; +	my ($r_min, $r_max); +	my $r_last = -1; # prevent dupes +	if (defined $TZ) { +		$ENV{TZ} = $TZ; +	} else { +		delete $ENV{TZ}; +	} +	if (defined $::_revision) { +		if ($::_revision =~ /^(\d+):(\d+)$/) { +			($r_min, $r_max) = ($1, $2); +		} elsif ($::_revision =~ /^\d+$/) { +			$r_min = $r_max = $::_revision; +		} else { +			::fatal "-r$::_revision is not supported, use ", +				"standard \'git log\' arguments instead\n"; +		} +	} + +	config_pager(); +	@args = (git_svn_log_cmd($r_min, $r_max, @args), @args); +	my $log = command_output_pipe(@args); +	run_pager(); +	my (@k, $c, $d, $stat); +	my $esc_color = qr/(?:\033\[(?:(?:\d+;)*\d*)?m)*/; +	while (<$log>) { +		if (/^${esc_color}commit ($::sha1_short)/o) { +			my $cmt = $1; +			if ($c && cmt_showable($c) && $c->{r} != $r_last) { +				$r_last = $c->{r}; +				process_commit($c, $r_min, $r_max, \@k) or +								goto out; +			} +			$d = undef; +			$c = { c => $cmt }; +		} elsif (/^${esc_color}author (.+) (\d+) ([\-\+]?\d+)$/o) { +			get_author_info($c, $1, $2, $3); +		} elsif (/^${esc_color}(?:tree|parent|committer) /o) { +			# ignore +		} elsif (/^${esc_color}:\d{6} \d{6} $::sha1_short/o) { +			push @{$c->{raw}}, $_; +		} elsif (/^${esc_color}[ACRMDT]\t/) { +			# we could add $SVN->{svn_path} here, but that requires +			# remote access at the moment (repo_path_split)... +			s#^(${esc_color})([ACRMDT])\t#$1   $2 #o; +			push @{$c->{changed}}, $_; +		} elsif (/^${esc_color}diff /o) { +			$d = 1; +			push @{$c->{diff}}, $_; +		} elsif ($d) { +			push @{$c->{diff}}, $_; +		} elsif (/^\ .+\ \|\s*\d+\ $esc_color[\+\-]* +		          $esc_color*[\+\-]*$esc_color$/x) { +			$stat = 1; +			push @{$c->{stat}}, $_; +		} elsif ($stat && /^ \d+ files changed, \d+ insertions/) { +			push @{$c->{stat}}, $_; +			$stat = undef; +		} elsif (/^${esc_color}    (git-svn-id:.+)$/o) { +			($c->{url}, $c->{r}, undef) = ::extract_metadata($1); +		} elsif (s/^${esc_color}    //o) { +			push @{$c->{l}}, $_; +		} +	} +	if ($c && defined $c->{r} && $c->{r} != $r_last) { +		$r_last = $c->{r}; +		process_commit($c, $r_min, $r_max, \@k); +	} +	if (@k) { +		my $swap = $r_max; +		$r_max = $r_min; +		$r_min = $swap; +		process_commit($_, $r_min, $r_max) foreach reverse @k; +	} +out: +	close $log; +	print '-' x72,"\n" unless $incremental || $oneline; +} + +package Git::SVN::Migration; +# these version numbers do NOT correspond to actual version numbers +# of git nor git-svn.  They are just relative. +# +# v0 layout: .git/$id/info/url, refs/heads/$id-HEAD +# +# v1 layout: .git/$id/info/url, refs/remotes/$id +# +# v2 layout: .git/svn/$id/info/url, refs/remotes/$id +# +# v3 layout: .git/svn/$id, refs/remotes/$id +#            - info/url may remain for backwards compatibility +#            - this is what we migrate up to this layout automatically, +#            - this will be used by git svn init on single branches +# v3.1 layout (auto migrated): +#            - .rev_db => .rev_db.$UUID, .rev_db will remain as a symlink +#              for backwards compatibility +# +# v4 layout: .git/svn/$repo_id/$id, refs/remotes/$repo_id/$id +#            - this is only created for newly multi-init-ed +#              repositories.  Similar in spirit to the +#              --use-separate-remotes option in git-clone (now default) +#            - we do not automatically migrate to this (following +#              the example set by core git) +use strict; +use warnings; +use Carp qw/croak/; +use File::Path qw/mkpath/; +use File::Basename qw/dirname basename/; +use vars qw/$_minimize/; + +sub migrate_from_v0 { +	my $git_dir = $ENV{GIT_DIR}; +	return undef unless -d $git_dir; +	my ($fh, $ctx) = command_output_pipe(qw/rev-parse --symbolic --all/); +	my $migrated = 0; +	while (<$fh>) { +		chomp; +		my ($id, $orig_ref) = ($_, $_); +		next unless $id =~ s#^refs/heads/(.+)-HEAD$#$1#; +		next unless -f "$git_dir/$id/info/url"; +		my $new_ref = "refs/remotes/$id"; +		if (::verify_ref("$new_ref^0")) { +			print STDERR "W: $orig_ref is probably an old ", +			             "branch used by an ancient version of ", +				     "git-svn.\n", +				     "However, $new_ref also exists.\n", +				     "We will not be able ", +				     "to use this branch until this ", +				     "ambiguity is resolved.\n"; +			next; +		} +		print STDERR "Migrating from v0 layout...\n" if !$migrated; +		print STDERR "Renaming ref: $orig_ref => $new_ref\n"; +		command_noisy('update-ref', $new_ref, $orig_ref); +		command_noisy('update-ref', '-d', $orig_ref, $orig_ref); +		$migrated++; +	} +	command_close_pipe($fh, $ctx); +	print STDERR "Done migrating from v0 layout...\n" if $migrated; +	$migrated; +} + +sub migrate_from_v1 { +	my $git_dir = $ENV{GIT_DIR}; +	my $migrated = 0; +	return $migrated unless -d $git_dir; +	my $svn_dir = "$git_dir/svn"; + +	# just in case somebody used 'svn' as their $id at some point... +	return $migrated if -d $svn_dir && ! -f "$svn_dir/info/url"; + +	print STDERR "Migrating from a git-svn v1 layout...\n"; +	mkpath([$svn_dir]); +	print STDERR "Data from a previous version of git-svn exists, but\n\t", +	             "$svn_dir\n\t(required for this version ", +	             "($::VERSION) of git-svn) does not. exist\n"; +	my ($fh, $ctx) = command_output_pipe(qw/rev-parse --symbolic --all/); +	while (<$fh>) { +		my $x = $_; +		next unless $x =~ s#^refs/remotes/##; +		chomp $x; +		next unless -f "$git_dir/$x/info/url"; +		my $u = eval { ::file_to_s("$git_dir/$x/info/url") }; +		next unless $u; +		my $dn = dirname("$git_dir/svn/$x"); +		mkpath([$dn]) unless -d $dn; +		if ($x eq 'svn') { # they used 'svn' as GIT_SVN_ID: +			mkpath(["$git_dir/svn/svn"]); +			print STDERR " - $git_dir/$x/info => ", +			                "$git_dir/svn/$x/info\n"; +			rename "$git_dir/$x/info", "$git_dir/svn/$x/info" or +			       croak "$!: $x"; +			# don't worry too much about these, they probably +			# don't exist with repos this old (save for index, +			# and we can easily regenerate that) +			foreach my $f (qw/unhandled.log index .rev_db/) { +				rename "$git_dir/$x/$f", "$git_dir/svn/$x/$f"; +			} +		} else { +			print STDERR " - $git_dir/$x => $git_dir/svn/$x\n"; +			rename "$git_dir/$x", "$git_dir/svn/$x" or +			       croak "$!: $x"; +		} +		$migrated++; +	} +	command_close_pipe($fh, $ctx); +	print STDERR "Done migrating from a git-svn v1 layout\n"; +	$migrated; +} + +sub read_old_urls { +	my ($l_map, $pfx, $path) = @_; +	my @dir; +	foreach (<$path/*>) { +		if (-r "$_/info/url") { +			$pfx .= '/' if $pfx && $pfx !~ m!/$!; +			my $ref_id = $pfx . basename $_; +			my $url = ::file_to_s("$_/info/url"); +			$l_map->{$ref_id} = $url; +		} elsif (-d $_) { +			push @dir, $_; +		} +	} +	foreach (@dir) { +		my $x = $_; +		$x =~ s!^\Q$ENV{GIT_DIR}\E/svn/!!o; +		read_old_urls($l_map, $x, $_); +	} +} + +sub migrate_from_v2 { +	my @cfg = command(qw/config -l/); +	return if grep /^svn-remote\..+\.url=/, @cfg; +	my %l_map; +	read_old_urls(\%l_map, '', "$ENV{GIT_DIR}/svn"); +	my $migrated = 0; + +	foreach my $ref_id (sort keys %l_map) { +		eval { Git::SVN->init($l_map{$ref_id}, '', undef, $ref_id) }; +		if ($@) { +			Git::SVN->init($l_map{$ref_id}, '', $ref_id, $ref_id); +		} +		$migrated++; +	} +	$migrated; +} + +sub minimize_connections { +	my $r = Git::SVN::read_all_remotes(); +	my $new_urls = {}; +	my $root_repos = {}; +	foreach my $repo_id (keys %$r) { +		my $url = $r->{$repo_id}->{url} or next; +		my $fetch = $r->{$repo_id}->{fetch} or next; +		my $ra = Git::SVN::Ra->new($url); + +		# skip existing cases where we already connect to the root +		if (($ra->{url} eq $ra->{repos_root}) || +		    (Git::SVN::sanitize_remote_name($ra->{repos_root}) eq +		     $repo_id)) { +			$root_repos->{$ra->{url}} = $repo_id; +			next; +		} + +		my $root_ra = Git::SVN::Ra->new($ra->{repos_root}); +		my $root_path = $ra->{url}; +		$root_path =~ s#^\Q$ra->{repos_root}\E(/|$)##; +		foreach my $path (keys %$fetch) { +			my $ref_id = $fetch->{$path}; +			my $gs = Git::SVN->new($ref_id, $repo_id, $path); + +			# make sure we can read when connecting to +			# a higher level of a repository +			my ($last_rev, undef) = $gs->last_rev_commit; +			if (!defined $last_rev) { +				$last_rev = eval { +					$root_ra->get_latest_revnum; +				}; +				next if $@; +			} +			my $new = $root_path; +			$new .= length $path ? "/$path" : ''; +			eval { +				$root_ra->get_log([$new], $last_rev, $last_rev, +			                          0, 0, 1, sub { }); +			}; +			next if $@; +			$new_urls->{$ra->{repos_root}}->{$new} = +			        { ref_id => $ref_id, +				  old_repo_id => $repo_id, +				  old_path => $path }; +		} +	} + +	my @emptied; +	foreach my $url (keys %$new_urls) { +		# see if we can re-use an existing [svn-remote "repo_id"] +		# instead of creating a(n ugly) new section: +		my $repo_id = $root_repos->{$url} || +		              Git::SVN::sanitize_remote_name($url); + +		my $fetch = $new_urls->{$url}; +		foreach my $path (keys %$fetch) { +			my $x = $fetch->{$path}; +			Git::SVN->init($url, $path, $repo_id, $x->{ref_id}); +			my $pfx = "svn-remote.$x->{old_repo_id}"; + +			my $old_fetch = quotemeta("$x->{old_path}:". +			                          "refs/remotes/$x->{ref_id}"); +			command_noisy(qw/config --unset/, +			              "$pfx.fetch", '^'. $old_fetch . '$'); +			delete $r->{$x->{old_repo_id}}-> +			       {fetch}->{$x->{old_path}}; +			if (!keys %{$r->{$x->{old_repo_id}}->{fetch}}) { +				command_noisy(qw/config --unset/, +				              "$pfx.url"); +				push @emptied, $x->{old_repo_id} +			} +		} +	} +	if (@emptied) { +		my $file = $ENV{GIT_CONFIG} || $ENV{GIT_CONFIG_LOCAL} || +		           "$ENV{GIT_DIR}/config"; +		print STDERR <<EOF; +The following [svn-remote] sections in your config file ($file) are empty +and can be safely removed: +EOF +		print STDERR "[svn-remote \"$_\"]\n" foreach @emptied; +	} +} + +sub migration_check { +	migrate_from_v0(); +	migrate_from_v1(); +	migrate_from_v2(); +	minimize_connections() if $_minimize; +} + +package Git::IndexInfo; +use strict; +use warnings; +use Git qw/command_input_pipe command_close_pipe/; + +sub new { +	my ($class) = @_; +	my ($gui, $ctx) = command_input_pipe(qw/update-index -z --index-info/); +	bless { gui => $gui, ctx => $ctx, nr => 0}, $class; +} + +sub remove { +	my ($self, $path) = @_; +	if (print { $self->{gui} } '0 ', 0 x 40, "\t", $path, "\0") { +		return ++$self->{nr}; +	} +	undef; +} + +sub update { +	my ($self, $mode, $hash, $path) = @_; +	if (print { $self->{gui} } $mode, ' ', $hash, "\t", $path, "\0") { +		return ++$self->{nr}; +	} +	undef; +} + +sub DESTROY { +	my ($self) = @_; +	command_close_pipe($self->{gui}, $self->{ctx}); +} + +package Git::SVN::GlobSpec; +use strict; +use warnings; + +sub new { +	my ($class, $glob) = @_; +	my $re = $glob; +	$re =~ s!/+$!!g; # no need for trailing slashes +	my $nr = ($re =~ s!^(.*)\*(.*)$!\(\[^/\]+\)!g); +	my ($left, $right) = ($1, $2); +	if ($nr > 1) { +		die "Only one '*' wildcard expansion ", +		    "is supported (got $nr): '$glob'\n"; +	} elsif ($nr == 0) { +		die "One '*' is needed for glob: '$glob'\n"; +	} +	$re = quotemeta($left) . $re . quotemeta($right); +	if (length $left && !($left =~ s!/+$!!g)) { +		die "Missing trailing '/' on left side of: '$glob' ($left)\n"; +	} +	if (length $right && !($right =~ s!^/+!!g)) { +		die "Missing leading '/' on right side of: '$glob' ($right)\n"; +	} +	my $left_re = qr/^\/\Q$left\E(\/|$)/; +	bless { left => $left, right => $right, left_regex => $left_re, +	        regex => qr/$re/, glob => $glob }, $class; +} + +sub full_path { +	my ($self, $path) = @_; +	return (length $self->{left} ? "$self->{left}/" : '') . +	       $path . (length $self->{right} ? "/$self->{right}" : ''); +} +  __END__  Data structures: -$log_msg hashref as returned by libsvn_log_entry() + +$remotes = { # returned by read_all_remotes() +	'svn' => { +		# svn-remote.svn.url=https://svn.musicpd.org +		url => 'https://svn.musicpd.org', +		# svn-remote.svn.fetch=mpd/trunk:trunk +		fetch => { +			'mpd/trunk' => 'trunk', +		}, +		# svn-remote.svn.tags=mpd/tags/*:tags/* +		tags => { +			path => { +				left => 'mpd/tags', +				right => '', +				regex => qr!mpd/tags/([^/]+)$!, +				glob => 'tags/*', +			}, +			ref => { +				left => 'tags', +				right => '', +				regex => qr!tags/([^/]+)$!, +				glob => 'tags/*', +			}, +		} +	} +}; + +$log_entry hashref as returned by libsvn_log_entry()  { -	msg => 'whitespace-formatted log entry +	log => 'whitespace-formatted log entry  ',						# trailing newline is preserved  	revision => '8',			# integer  	date => '2004-02-24T17:01:44.108345Z',	# commit date  	author => 'committer name'  }; + +# this is generated by generate_diff();  @mods = array of diff-index line hashes, each element represents one line  	of diff-index output @@ -48,7 +48,7 @@ static int handle_options(const char*** argv, int* argc)  		/*  		 * Check remaining flags.  		 */ -		if (!strncmp(cmd, "--exec-path", 11)) { +		if (!prefixcmp(cmd, "--exec-path")) {  			cmd += 11;  			if (*cmd == '=')  				git_set_exec_path(cmd + 1); @@ -66,7 +66,7 @@ static int handle_options(const char*** argv, int* argc)  			setenv(GIT_DIR_ENVIRONMENT, (*argv)[1], 1);  			(*argv)++;  			(*argc)--; -		} else if (!strncmp(cmd, "--git-dir=", 10)) { +		} else if (!prefixcmp(cmd, "--git-dir=")) {  			setenv(GIT_DIR_ENVIRONMENT, cmd + 10, 1);  		} else if (!strcmp(cmd, "--bare")) {  			static char git_dir[PATH_MAX+1]; @@ -88,7 +88,7 @@ static char *alias_string;  static int git_alias_config(const char *var, const char *value)  { -	if (!strncmp(var, "alias.", 6) && !strcmp(var + 6, alias_command)) { +	if (!prefixcmp(var, "alias.") && !strcmp(var + 6, alias_command)) {  		alias_string = xstrdup(value);  	}  	return 0; @@ -247,7 +247,7 @@ static void handle_internal_command(int argc, const char **argv, char **envp)  		{ "fsck", cmd_fsck, RUN_SETUP },  		{ "fsck-objects", cmd_fsck, RUN_SETUP },  		{ "get-tar-commit-id", cmd_get_tar_commit_id }, -		{ "grep", cmd_grep, RUN_SETUP }, +		{ "grep", cmd_grep, RUN_SETUP | USE_PAGER },  		{ "help", cmd_help },  		{ "init", cmd_init_db },  		{ "init-db", cmd_init_db }, @@ -348,7 +348,7 @@ int main(int argc, const char **argv, char **envp)  	 * So we just directly call the internal command handler, and  	 * die if that one cannot handle it.  	 */ -	if (!strncmp(cmd, "git-", 4)) { +	if (!prefixcmp(cmd, "git-")) {  		cmd += 4;  		argv[0] = cmd;  		handle_internal_command(argc, argv, envp); @@ -360,7 +360,7 @@ int main(int argc, const char **argv, char **envp)  	argc--;  	handle_options(&argv, &argc);  	if (argc > 0) { -		if (!strncmp(argv[0], "--", 2)) +		if (!prefixcmp(argv[0], "--"))  			argv[0] += 2;  	} else {  		/* Default command: "help" */ @@ -130,7 +130,7 @@ static void list_commands(const char *exec_path, const char *pattern)  		struct stat st;  		int entlen; -		if (strncmp(de->d_name, "git-", 4)) +		if (prefixcmp(de->d_name, "git-"))  			continue;  		strcpy(path+dirlen, de->d_name);  		if (stat(path, &st) || /* stat, not lstat */ @@ -179,7 +179,7 @@ static void show_man_page(const char *git_cmd)  {  	const char *page; -	if (!strncmp(git_cmd, "git", 3)) +	if (!prefixcmp(git_cmd, "git"))  		page = git_cmd;  	else {  		int page_len = strlen(git_cmd) + 4; diff --git a/http-fetch.c b/http-fetch.c index 9f790a08e5..e6cd11db73 100644 --- a/http-fetch.c +++ b/http-fetch.c @@ -717,8 +717,8 @@ static int fetch_indices(struct alt_base *repo)  		case 'P':  			i++;  			if (i + 52 <= buffer.posn && -			    !strncmp(data + i, " pack-", 6) && -			    !strncmp(data + i + 46, ".pack\n", 6)) { +			    !prefixcmp(data + i, " pack-") && +			    !prefixcmp(data + i + 46, ".pack\n")) {  				get_sha1_hex(data + i + 6, sha1);  				setup_index(repo, sha1);  				i += 51; diff --git a/http-push.c b/http-push.c index b128c0146c..9ad6fd00b0 100644 --- a/http-push.c +++ b/http-push.c @@ -1060,8 +1060,8 @@ static int fetch_indices(void)  		case 'P':  			i++;  			if (i + 52 < buffer.posn && -			    !strncmp(data + i, " pack-", 6) && -			    !strncmp(data + i + 46, ".pack\n", 6)) { +			    !prefixcmp(data + i, " pack-") && +			    !prefixcmp(data + i + 46, ".pack\n")) {  				get_sha1_hex(data + i + 6, sha1);  				setup_index(sha1);  				i += 51; @@ -1206,11 +1206,11 @@ static void handle_new_lock_ctx(struct xml_ctx *ctx, int tag_closed)  			lock->owner = xmalloc(strlen(ctx->cdata) + 1);  			strcpy(lock->owner, ctx->cdata);  		} else if (!strcmp(ctx->name, DAV_ACTIVELOCK_TIMEOUT)) { -			if (!strncmp(ctx->cdata, "Second-", 7)) +			if (!prefixcmp(ctx->cdata, "Second-"))  				lock->timeout =  					strtol(ctx->cdata + 7, NULL, 10);  		} else if (!strcmp(ctx->name, DAV_ACTIVELOCK_TOKEN)) { -			if (!strncmp(ctx->cdata, "opaquelocktoken:", 16)) { +			if (!prefixcmp(ctx->cdata, "opaquelocktoken:")) {  				lock->token = xmalloc(strlen(ctx->cdata) - 15);  				strcpy(lock->token, ctx->cdata + 16);  			} @@ -2168,7 +2168,7 @@ static void fetch_symref(const char *path, char **symref, unsigned char *sha1)  		return;  	/* If it's a symref, set the refname; otherwise try for a sha1 */ -	if (!strncmp((char *)buffer.buffer, "ref: ", 5)) { +	if (!prefixcmp((char *)buffer.buffer, "ref: ")) {  		*symref = xmalloc(buffer.posn - 5);  		strlcpy(*symref, (char *)buffer.buffer + 5, buffer.posn - 5);  	} else { diff --git a/imap-send.c b/imap-send.c index 3eaf025720..84df2fabb7 100644 --- a/imap-send.c +++ b/imap-send.c @@ -1192,7 +1192,7 @@ count_messages( msg_data_t *msg )  	char *p = msg->data;  	while (1) { -		if (!strncmp( "From ", p, 5 )) { +		if (!prefixcmp(p, "From ")) {  			count++;  			p += 5;  		} @@ -1216,7 +1216,7 @@ split_msg( msg_data_t *all_msgs, msg_data_t *msg, int *ofs )  	data = &all_msgs->data[ *ofs ];  	msg->len = all_msgs->len - *ofs; -	if (msg->len < 5 || strncmp( data, "From ", 5 )) +	if (msg->len < 5 || prefixcmp(data, "From "))  		return 0;  	p = strchr( data, '\n' ); @@ -1267,12 +1267,12 @@ git_imap_config(const char *key, const char *val)  		imap_folder = xstrdup( val );  	} else if (!strcmp( "host", key )) {  		{ -			if (!strncmp( "imap:", val, 5 )) +			if (!prefixcmp(val, "imap:"))  				val += 5;  			if (!server.port)  				server.port = 143;  		} -		if (!strncmp( "//", val, 2 )) +		if (!prefixcmp(val, "//"))  			val += 2;  		server.host = xstrdup( val );  	} diff --git a/index-pack.c b/index-pack.c index 72e0962415..fa9a0e7489 100644 --- a/index-pack.c +++ b/index-pack.c @@ -849,9 +849,9 @@ int main(int argc, char **argv)  				fix_thin_pack = 1;  			} else if (!strcmp(arg, "--keep")) {  				keep_msg = ""; -			} else if (!strncmp(arg, "--keep=", 7)) { +			} else if (!prefixcmp(arg, "--keep=")) {  				keep_msg = arg + 7; -			} else if (!strncmp(arg, "--pack_header=", 14)) { +			} else if (!prefixcmp(arg, "--pack_header=")) {  				struct pack_header *hdr;  				char *c; diff --git a/merge-index.c b/merge-index.c index a9983dd78a..7027d78659 100644 --- a/merge-index.c +++ b/merge-index.c @@ -60,7 +60,7 @@ static int merge_entry(int pos, const char *path)  			break;  		found++;  		strcpy(hexbuf[stage], sha1_to_hex(ce->sha1)); -		sprintf(ownbuf[stage], "%o", ntohl(ce->ce_mode) & (~S_IFMT)); +		sprintf(ownbuf[stage], "%o", ntohl(ce->ce_mode));  		arguments[stage] = hexbuf[stage];  		arguments[stage + 4] = ownbuf[stage];  	} while (++pos < active_nr); diff --git a/merge-recursive.c b/merge-recursive.c index 58989424d7..397a7ad85b 100644 --- a/merge-recursive.c +++ b/merge-recursive.c @@ -589,7 +589,7 @@ static void update_file_flags(const unsigned char *sha,  			memcpy(lnk, buf, size);  			lnk[size] = '\0';  			mkdir_p(path, 0777); -			unlink(lnk); +			unlink(path);  			symlink(lnk, path);  		} else  			die("do not know what to do with %06o %s '%s'", diff --git a/peek-remote.c b/peek-remote.c index ef3c76ce52..96bfac498b 100644 --- a/peek-remote.c +++ b/peek-remote.c @@ -35,11 +35,11 @@ int main(int argc, char **argv)  		char *arg = argv[i];  		if (*arg == '-') { -			if (!strncmp("--upload-pack=", arg, 14)) { +			if (!prefixcmp(arg, "--upload-pack=")) {  				uploadpack = arg + 14;  				continue;  			} -			if (!strncmp("--exec=", arg, 7)) { +			if (!prefixcmp(arg, "--exec=")) {  				uploadpack = arg + 7;  				continue;  			} diff --git a/perl/Git.pm b/perl/Git.pm index f2c156cde9..b5b1cf5edc 100644 --- a/perl/Git.pm +++ b/perl/Git.pm @@ -516,6 +516,36 @@ sub config {  } +=item config_boolean ( VARIABLE ) + +Retrieve the boolean configuration C<VARIABLE>. + +Must be called on a repository instance. + +This currently wraps command('config') so it is not so fast. + +=cut + +sub config_boolean { +	my ($self, $var) = @_; +	$self->repo_path() +		or throw Error::Simple("not a repository"); + +	try { +		return $self->command_oneline('config', '--bool', '--get', +					      $var); +	} catch Git::Error::Command with { +		my $E = shift; +		if ($E->value() == 1) { +			# Key not found. +			return undef; +		} else { +			throw $E; +		} +	}; +} + +  =item ident ( TYPE | IDENTSTR )  =item ident_person ( TYPE | IDENTSTR | IDENTARRAY ) diff --git a/receive-pack.c b/receive-pack.c index 7311c822dd..7f1dcc045c 100644 --- a/receive-pack.c +++ b/receive-pack.c @@ -109,7 +109,7 @@ static int update(struct command *cmd)  	struct ref_lock *lock;  	cmd->error_string = NULL; -	if (!strncmp(name, "refs/", 5) && check_ref_format(name + 5)) { +	if (!prefixcmp(name, "refs/") && check_ref_format(name + 5)) {  		cmd->error_string = "funny refname";  		return error("refusing to create funny ref '%s' locally",  			     name); @@ -125,7 +125,7 @@ static int update(struct command *cmd)  	}  	if (deny_non_fast_forwards && !is_null_sha1(new_sha1) &&  	    !is_null_sha1(old_sha1) && -	    !strncmp(name, "refs/heads/", 11)) { +	    !prefixcmp(name, "refs/heads/")) {  		struct commit *old_commit, *new_commit;  		struct commit_list *bases, *ent; @@ -828,8 +828,8 @@ int rename_ref(const char *oldref, const char *newref, const char *logmsg)  		goto rollback;  	} -	if (!strncmp(oldref, "refs/heads/", 11) && -			!strncmp(newref, "refs/heads/", 11)) { +	if (!prefixcmp(oldref, "refs/heads/") && +			!prefixcmp(newref, "refs/heads/")) {  		char oldsection[1024], newsection[1024];  		snprintf(oldsection, 1024, "branch.%s", oldref + 11); @@ -894,8 +894,8 @@ static int log_ref_write(const char *ref_name, const unsigned char *old_sha1,  	log_file = git_path("logs/%s", ref_name);  	if (log_all_ref_updates && -	    (!strncmp(ref_name, "refs/heads/", 11) || -	     !strncmp(ref_name, "refs/remotes/", 13) || +	    (!prefixcmp(ref_name, "refs/heads/") || +	     !prefixcmp(ref_name, "refs/remotes/") ||  	     !strcmp(ref_name, "HEAD"))) {  		if (safe_create_leading_directories(log_file) < 0)  			return error("unable to create directory for %s", diff --git a/revision.c b/revision.c index 5b1794b0ef..4cf697e2c1 100644 --- a/revision.c +++ b/revision.c @@ -813,11 +813,11 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  		const char *arg = argv[i];  		if (*arg == '-') {  			int opts; -			if (!strncmp(arg, "--max-count=", 12)) { +			if (!prefixcmp(arg, "--max-count=")) {  				revs->max_count = atoi(arg + 12);  				continue;  			} -			if (!strncmp(arg, "--skip=", 7)) { +			if (!prefixcmp(arg, "--skip=")) {  				revs->skip_count = atoi(arg + 7);  				continue;  			} @@ -832,31 +832,31 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  				revs->max_count = atoi(argv[++i]);  				continue;  			} -			if (!strncmp(arg,"-n",2)) { +			if (!prefixcmp(arg, "-n")) {  				revs->max_count = atoi(arg + 2);  				continue;  			} -			if (!strncmp(arg, "--max-age=", 10)) { +			if (!prefixcmp(arg, "--max-age=")) {  				revs->max_age = atoi(arg + 10);  				continue;  			} -			if (!strncmp(arg, "--since=", 8)) { +			if (!prefixcmp(arg, "--since=")) {  				revs->max_age = approxidate(arg + 8);  				continue;  			} -			if (!strncmp(arg, "--after=", 8)) { +			if (!prefixcmp(arg, "--after=")) {  				revs->max_age = approxidate(arg + 8);  				continue;  			} -			if (!strncmp(arg, "--min-age=", 10)) { +			if (!prefixcmp(arg, "--min-age=")) {  				revs->min_age = atoi(arg + 10);  				continue;  			} -			if (!strncmp(arg, "--before=", 9)) { +			if (!prefixcmp(arg, "--before=")) {  				revs->min_age = approxidate(arg + 9);  				continue;  			} -			if (!strncmp(arg, "--until=", 8)) { +			if (!prefixcmp(arg, "--until=")) {  				revs->min_age = approxidate(arg + 8);  				continue;  			} @@ -944,7 +944,7 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  				revs->num_ignore_packed = 0;  				continue;  			} -			if (!strncmp(arg, "--unpacked=", 11)) { +			if (!prefixcmp(arg, "--unpacked=")) {  				revs->unpacked = 1;  				add_ignore_packed(revs, arg+11);  				continue; @@ -980,7 +980,7 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  				revs->verbose_header = 1;  				continue;  			} -			if (!strncmp(arg, "--pretty", 8)) { +			if (!prefixcmp(arg, "--pretty")) {  				revs->verbose_header = 1;  				revs->commit_format = get_commit_format(arg+8);  				continue; @@ -1005,7 +1005,7 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  				revs->abbrev = DEFAULT_ABBREV;  				continue;  			} -			if (!strncmp(arg, "--abbrev=", 9)) { +			if (!prefixcmp(arg, "--abbrev=")) {  				revs->abbrev = strtoul(arg + 9, NULL, 10);  				if (revs->abbrev < MINIMUM_ABBREV)  					revs->abbrev = MINIMUM_ABBREV; @@ -1034,15 +1034,15 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  			/*  			 * Grepping the commit log  			 */ -			if (!strncmp(arg, "--author=", 9)) { +			if (!prefixcmp(arg, "--author=")) {  				add_header_grep(revs, "author", arg+9);  				continue;  			} -			if (!strncmp(arg, "--committer=", 12)) { +			if (!prefixcmp(arg, "--committer=")) {  				add_header_grep(revs, "committer", arg+12);  				continue;  			} -			if (!strncmp(arg, "--grep=", 7)) { +			if (!prefixcmp(arg, "--grep=")) {  				add_message_grep(revs, arg+7);  				continue;  			} @@ -1050,7 +1050,7 @@ int setup_revisions(int argc, const char **argv, struct rev_info *revs, const ch  				all_match = 1;  				continue;  			} -			if (!strncmp(arg, "--encoding=", 11)) { +			if (!prefixcmp(arg, "--encoding=")) {  				arg += 11;  				if (strcmp(arg, "none"))  					git_log_output_encoding = strdup(arg); @@ -1233,9 +1233,15 @@ static struct commit *get_revision_1(struct rev_info *revs)  		 */  		if (!revs->limited) {  			if (revs->max_age != -1 && -			    (commit->date < revs->max_age)) -				continue; -			add_parents_to_list(revs, commit, &revs->commits); +			    (commit->date < revs->max_age)) { +				if (revs->boundary) +					commit->object.flags |= +						BOUNDARY_SHOW | BOUNDARY; +				else +					continue; +			} else +				add_parents_to_list(revs, commit, +						&revs->commits);  		}  		if (commit->object.flags & SHOWN)  			continue; @@ -1336,7 +1342,18 @@ struct commit *get_revision(struct rev_info *revs)  	case -1:  		break;  	case 0: -		return NULL; +		if (revs->boundary) { +			struct commit_list *list = revs->commits; +			while (list) { +				list->item->object.flags |= +					BOUNDARY_SHOW | BOUNDARY; +				list = list->next; +			} +			/* all remaining commits are boundary commits */ +			revs->max_count = -1; +			revs->limited = 1; +		} else +			return NULL;  	default:  		revs->max_count--;  	} diff --git a/send-pack.c b/send-pack.c index 33e69dbe18..512b660e99 100644 --- a/send-pack.c +++ b/send-pack.c @@ -379,11 +379,11 @@ int main(int argc, char **argv)  		char *arg = *argv;  		if (*arg == '-') { -			if (!strncmp(arg, "--receive-pack=", 15)) { +			if (!prefixcmp(arg, "--receive-pack=")) {  				receivepack = arg + 15;  				continue;  			} -			if (!strncmp(arg, "--exec=", 7)) { +			if (!prefixcmp(arg, "--exec=")) {  				receivepack = arg + 7;  				continue;  			} @@ -251,7 +251,7 @@ const char *setup_git_directory_gently(int *nongit_ok)  	offset++;  	cwd[len++] = '/';  	cwd[len] = 0; -	inside_git_dir = !strncmp(cwd + offset, ".git/", 5); +	inside_git_dir = !prefixcmp(cwd + offset, ".git/");  	return cwd + offset;  } diff --git a/sha1_file.c b/sha1_file.c index 2c870314d5..9a1dee051a 100644 --- a/sha1_file.c +++ b/sha1_file.c @@ -2082,7 +2082,7 @@ int index_fd(unsigned char *sha1, int fd, struct stat *st, int write_object, con  {  	unsigned long size = st->st_size;  	void *buf; -	int ret; +	int ret, re_allocated = 0;  	buf = "";  	if (size) @@ -2091,11 +2091,30 @@ int index_fd(unsigned char *sha1, int fd, struct stat *st, int write_object, con  	if (!type)  		type = blob_type; -	/* FIXME: CRLF -> LF conversion here for blobs! We'll need the path! */ + +	/* +	 * Convert blobs to git internal format +	 */ +	if (!strcmp(type, blob_type)) { +		unsigned long nsize = size; +		char *nbuf = buf; +		if (convert_to_git(NULL, &nbuf, &nsize)) { +			if (size) +				munmap(buf, size); +			size = nsize; +			buf = nbuf; +			re_allocated = 1; +		} +	} +  	if (write_object)  		ret = write_sha1_file(buf, size, type, sha1);  	else  		ret = hash_sha1_file(buf, size, type, sha1); +	if (re_allocated) { +		free(buf); +		return ret; +	}  	if (size)  		munmap(buf, size);  	return ret; @@ -8,7 +8,7 @@ static int do_generic_cmd(const char *me, char *arg)  	if (!arg || !(arg = sq_dequote(arg)))  		die("bad argument"); -	if (strncmp(me, "git-", 4)) +	if (prefixcmp(me, "git-"))  		die("bad command");  	my_argv[0] = me + 4; diff --git a/t/lib-git-svn.sh b/t/lib-git-svn.sh index 67d08cf740..f6fe78cd27 100644 --- a/t/lib-git-svn.sh +++ b/t/lib-git-svn.sh @@ -42,9 +42,9 @@ then  	exit  fi +rawsvnrepo="$svnrepo"  svnrepo="file://$svnrepo" -  poke() { -	perl -e '@x = stat($ARGV[0]); utime($x[8], $x[9] + 1, $ARGV[0])' "$1" +	test-chmtime +1 "$1"  } diff --git a/t/t0020-crlf.sh b/t/t0020-crlf.sh new file mode 100755 index 0000000000..723b29ad17 --- /dev/null +++ b/t/t0020-crlf.sh @@ -0,0 +1,217 @@ +#!/bin/sh + +test_description='CRLF conversion' + +. ./test-lib.sh + +append_cr () { +	sed -e 's/$/Q/' | tr Q '\015' +} + +remove_cr () { +	tr '\015' Q <"$1" | grep Q >/dev/null && +	tr '\015' Q <"$1" | sed -ne 's/Q$//p' +} + +test_expect_success setup ' + +	git repo-config core.autocrlf false && + +	for w in Hello world how are you; do echo $w; done >one && +	mkdir dir && +	for w in I am very very fine thank you; do echo $w; done >dir/two && +	git add . && + +	git commit -m initial && + +	one=`git rev-parse HEAD:one` && +	dir=`git rev-parse HEAD:dir` && +	two=`git rev-parse HEAD:dir/two` && + +	for w in Some extra lines here; do echo $w; done >>one && +	git diff >patch.file && +	patched=`git hash-object --stdin <one` && +	git read-tree --reset -u HEAD && + +	echo happy. +' + +test_expect_success 'update with autocrlf=input' ' + +	rm -f tmp one dir/two && +	git read-tree --reset -u HEAD && +	git repo-config core.autocrlf input && + +	for f in one dir/two +	do +		append_cr <$f >tmp && mv -f tmp $f && +		git update-index -- $f || { +			echo Oops +			false +			break +		} +	done && + +	differs=`git diff-index --cached HEAD` && +	test -z "$differs" || { +		echo Oops "$differs" +		false +	} + +' + +test_expect_success 'update with autocrlf=true' ' + +	rm -f tmp one dir/two && +	git read-tree --reset -u HEAD && +	git repo-config core.autocrlf true && + +	for f in one dir/two +	do +		append_cr <$f >tmp && mv -f tmp $f && +		git update-index -- $f || { +			echo "Oops $f" +			false +			break +		} +	done && + +	differs=`git diff-index --cached HEAD` && +	test -z "$differs" || { +		echo Oops "$differs" +		false +	} + +' + +test_expect_success 'checkout with autocrlf=true' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf true && +	git read-tree --reset -u HEAD && + +	for f in one dir/two +	do +		remove_cr "$f" >tmp && mv -f tmp $f && +		git update-index -- $f || { +			echo "Eh? $f" +			false +			break +		} +	done && +	test "$one" = `git hash-object --stdin <one` && +	test "$two" = `git hash-object --stdin <dir/two` && +	differs=`git diff-index --cached HEAD` && +	test -z "$differs" || { +		echo Oops "$differs" +		false +	} +' + +test_expect_success 'checkout with autocrlf=input' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf input && +	git read-tree --reset -u HEAD && + +	for f in one dir/two +	do +		if remove_cr "$f" >/dev/null +		then +			echo "Eh? $f" +			false +			break +		else +			git update-index -- $f +		fi +	done && +	test "$one" = `git hash-object --stdin <one` && +	test "$two" = `git hash-object --stdin <dir/two` && +	differs=`git diff-index --cached HEAD` && +	test -z "$differs" || { +		echo Oops "$differs" +		false +	} +' + +test_expect_success 'apply patch (autocrlf=input)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf input && +	git read-tree --reset -u HEAD && + +	git apply patch.file && +	test "$patched" = "`git hash-object --stdin <one`" || { +		echo "Eh?  apply without index" +		false +	} +' + +test_expect_success 'apply patch --cached (autocrlf=input)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf input && +	git read-tree --reset -u HEAD && + +	git apply --cached patch.file && +	test "$patched" = `git rev-parse :one` || { +		echo "Eh?  apply with --cached" +		false +	} +' + +test_expect_success 'apply patch --index (autocrlf=input)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf input && +	git read-tree --reset -u HEAD && + +	git apply --index patch.file && +	test "$patched" = `git rev-parse :one` && +	test "$patched" = `git hash-object --stdin <one` || { +		echo "Eh?  apply with --index" +		false +	} +' + +test_expect_success 'apply patch (autocrlf=true)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf true && +	git read-tree --reset -u HEAD && + +	git apply patch.file && +	test "$patched" = "`remove_cr one | git hash-object --stdin`" || { +		echo "Eh?  apply without index" +		false +	} +' + +test_expect_success 'apply patch --cached (autocrlf=true)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf true && +	git read-tree --reset -u HEAD && + +	git apply --cached patch.file && +	test "$patched" = `git rev-parse :one` || { +		echo "Eh?  apply without index" +		false +	} +' + +test_expect_success 'apply patch --index (autocrlf=true)' ' + +	rm -f tmp one dir/two && +	git repo-config core.autocrlf true && +	git read-tree --reset -u HEAD && + +	git apply --index patch.file && +	test "$patched" = `git rev-parse :one` && +	test "$patched" = "`remove_cr one | git hash-object --stdin`" || { +		echo "Eh?  apply with --index" +		false +	} +' + +test_done diff --git a/t/t4016-diff-quote.sh b/t/t4016-diff-quote.sh index edde8f5568..2e7cd5f255 100755 --- a/t/t4016-diff-quote.sh +++ b/t/t4016-diff-quote.sh @@ -13,6 +13,10 @@ P1='pathname	with HT'  P2='pathname with SP'  P3='pathname  with LF' +: >"$P1" 2>&1 && test -f "$P1" && rm -f "$P1" || { +	echo >&2 'Filesystem does not support tabs in names' +	test_done +}  test_expect_success setup '  	echo P0.0 >"$P0.0" && diff --git a/t/t4119-apply-config.sh b/t/t4119-apply-config.sh new file mode 100755 index 0000000000..620a9207bf --- /dev/null +++ b/t/t4119-apply-config.sh @@ -0,0 +1,162 @@ +#!/bin/sh +# +# Copyright (c) 2007 Junio C Hamano +# + +test_description='git-apply --whitespace=strip and configuration file. + +' + +. ./test-lib.sh + +test_expect_success setup ' +	mkdir sub && +	echo A >sub/file1 && +	cp sub/file1 saved && +	git add sub/file1 && +	echo "B " >sub/file1 && +	git diff >patch.file +' + +# Also handcraft GNU diff output; note this has trailing whitespace. +cat >gpatch.file <<\EOF && +--- file1	2007-02-21 01:04:24.000000000 -0800 ++++ file1+	2007-02-21 01:07:44.000000000 -0800 +@@ -1 +1 @@ +-A ++B  +EOF + +sed -e 's|file1|sub/&|' gpatch.file >gpatch-sub.file && +sed -e ' +	/^--- /s|file1|a/sub/&| +	/^+++ /s|file1|b/sub/&| +' gpatch.file >gpatch-ab-sub.file && + +check_result () { +	if grep " " "$1" +	then +		echo "Eh?" +		false +	elif grep B "$1" +	then +		echo Happy +	else +		echo "Huh?" +		false +	fi +} + +test_expect_success 'apply --whitespace=strip' ' + +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	git apply --whitespace=strip patch.file && +	check_result sub/file1 +' + +test_expect_success 'apply --whitespace=strip from config' ' + +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	git config apply.whitespace strip && +	git apply patch.file && +	check_result sub/file1 +' + +D=`pwd` + +test_expect_success 'apply --whitespace=strip in subdir' ' + +	cd "$D" && +	git config --unset-all apply.whitespace +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	cd sub && +	git apply --whitespace=strip ../patch.file && +	check_result file1 +' + +test_expect_success 'apply --whitespace=strip from config in subdir' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	cd sub && +	git apply ../patch.file && +	check_result file1 +' + +test_expect_success 'same in subdir but with traditional patch input' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	cd sub && +	git apply ../gpatch.file && +	check_result file1 +' + +test_expect_success 'same but with traditional patch input of depth 1' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	cd sub && +	git apply ../gpatch-sub.file && +	check_result file1 +' + +test_expect_success 'same but with traditional patch input of depth 2' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	cd sub && +	git apply ../gpatch-ab-sub.file && +	check_result file1 +' + +test_expect_success 'same but with traditional patch input of depth 1' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	git apply -p0 gpatch-sub.file && +	check_result sub/file1 +' + +test_expect_success 'same but with traditional patch input of depth 2' ' + +	cd "$D" && +	git config apply.whitespace strip && +	rm -f sub/file1 && +	cp saved sub/file1 && +	git update-index --refresh && + +	git apply gpatch-ab-sub.file && +	check_result sub/file1 +' + +test_done diff --git a/t/t4200-rerere.sh b/t/t4200-rerere.sh index c571a1bd74..639d45fcec 100755 --- a/t/t4200-rerere.sh +++ b/t/t4200-rerere.sh @@ -112,39 +112,26 @@ rr2=.git/rr-cache/$sha2  mkdir $rr2  echo Hello > $rr2/preimage -case "$(date -d @11111111 +%s 2>/dev/null)" in -11111111) -	# 'date' must be able to take arbitrary input with @11111111 notation. -	# for this test to succeed.  We should fix this part using more -	# portable script someday. - -	now=$(date +%s) -	almost_15_days_ago=$(($now+60-15*86400)) -	just_over_15_days_ago=$(($now-1-15*86400)) -	almost_60_days_ago=$(($now+60-60*86400)) -	just_over_60_days_ago=$(($now-1-60*86400)) -	predate1="$(date -d "@$almost_60_days_ago" +%Y%m%d%H%M.%S)" -	predate2="$(date -d "@$almost_15_days_ago" +%Y%m%d%H%M.%S)" -	postdate1="$(date -d "@$just_over_60_days_ago" +%Y%m%d%H%M.%S)" -	postdate2="$(date -d "@$just_over_15_days_ago" +%Y%m%d%H%M.%S)" - -	touch -m -t "$predate1" $rr/preimage -	touch -m -t "$predate2" $rr2/preimage - -	test_expect_success 'garbage collection (part1)' 'git rerere gc' - -	test_expect_success 'young records still live' \ -		"test -f $rr/preimage -a -f $rr2/preimage" - -	touch -m -t "$postdate1" $rr/preimage -	touch -m -t "$postdate2" $rr2/preimage - -	test_expect_success 'garbage collection (part2)' 'git rerere gc' - -	test_expect_success 'old records rest in peace' \ -		"test ! -f $rr/preimage -a ! -f $rr2/preimage" -	;; -esac +almost_15_days_ago=$((60-15*86400)) +just_over_15_days_ago=$((-1-15*86400)) +almost_60_days_ago=$((60-60*86400)) +just_over_60_days_ago=$((-1-60*86400)) + +test-chmtime =$almost_60_days_ago $rr/preimage +test-chmtime =$almost_15_days_ago $rr2/preimage + +test_expect_success 'garbage collection (part1)' 'git rerere gc' + +test_expect_success 'young records still live' \ +	"test -f $rr/preimage && test -f $rr2/preimage" + +test-chmtime =$just_over_60_days_ago $rr/preimage +test-chmtime =$just_over_15_days_ago $rr2/preimage + +test_expect_success 'garbage collection (part2)' 'git rerere gc' + +test_expect_success 'old records rest in peace' \ +	"test ! -f $rr/preimage && test ! -f $rr2/preimage"  test_done diff --git a/t/t9100-git-svn-basic.sh b/t/t9100-git-svn-basic.sh index 040da92756..7dcfc7e7db 100755 --- a/t/t9100-git-svn-basic.sh +++ b/t/t9100-git-svn-basic.sh @@ -211,8 +211,58 @@ tree d667270a1f7b109f5eb3aaea21ede14b56bfdd6e  tree 8f51f74cf0163afc9ad68a4b1537288c4558b5a4  EOF -echo tree 4b825dc642cb6eb9a060e54bf8d69288fbee4904 >> expected -  test_expect_success "$name" "diff -u a expected" +test_expect_failure 'exit if remote refs are ambigious' " +        git-config --add svn-remote.svn.fetch \ +                              bar:refs/remotes/git-svn && +        git-svn migrate +        " + +test_expect_failure 'exit if init-ing a would clobber a URL' " +        svnadmin create ${PWD}/svnrepo2 && +        svn mkdir -m 'mkdir bar' ${svnrepo}2/bar && +        git-config --unset svn-remote.svn.fetch \ +                                '^bar:refs/remotes/git-svn$' && +        git-svn init ${svnrepo}2/bar +        " + +test_expect_success \ +  'init allows us to connect to another directory in the same repo' " +        git-svn init -i bar $svnrepo/bar && +        git config --get svn-remote.svn.fetch \ +                              '^bar:refs/remotes/bar$' && +        git config --get svn-remote.svn.fetch \ +                              '^:refs/remotes/git-svn$' +        " + +test_expect_success 'able to dcommit to a subdirectory' " +	git-svn fetch -i bar && +	git checkout -b my-bar refs/remotes/bar && +	echo abc > d && +	git update-index --add d && +	git commit -m '/bar/d should be in the log' && +	git-svn dcommit -i bar && +	test -z \"\`git diff refs/heads/my-bar refs/remotes/bar\`\" && +	mkdir newdir && +	echo new > newdir/dir && +	git update-index --add newdir/dir && +	git commit -m 'add a new directory' && +	git-svn dcommit -i bar && +	test -z \"\`git diff refs/heads/my-bar refs/remotes/bar\`\" && +	echo foo >> newdir/dir && +	git update-index newdir/dir && +	git commit -m 'modify a file in new directory' && +	git-svn dcommit -i bar && +	test -z \"\`git diff refs/heads/my-bar refs/remotes/bar\`\" +	" + +test_expect_success 'able to set-tree to a subdirectory' " +	echo cba > d && +	git update-index d && +	git commit -m 'update /bar/d' && +	git-svn set-tree -i bar HEAD && +	test -z \"\`git diff refs/heads/my-bar refs/remotes/bar\`\" +	" +  test_done diff --git a/t/t9101-git-svn-props.sh b/t/t9101-git-svn-props.sh index e8133d81cb..622ea1c0df 100755 --- a/t/t9101-git-svn-props.sh +++ b/t/t9101-git-svn-props.sh @@ -121,4 +121,30 @@ b_ne_cr="`git-hash-object ne_cr`"  test_expect_success 'CRLF + $Id$' "test '$a_cr' = '$b_cr'"  test_expect_success 'CRLF + $Id$ (no newline)' "test '$a_ne_cr' = '$b_ne_cr'" +cat > show-ignore.expect <<\EOF + +# / +/no-such-file* + +# deeply +/deeply/no-such-file* + +# deeply/nested +/deeply/nested/no-such-file* + +# deeply/nested/directory +/deeply/nested/directory/no-such-file* +EOF + +test_expect_success 'test show-ignore' " +	cd test_wc && +	mkdir -p deeply/nested/directory && +	svn add deeply && +	svn propset -R svn:ignore 'no-such-file*' . +	svn commit -m 'propset svn:ignore' +	cd .. && +	git-svn show-ignore > show-ignore.got && +	cmp show-ignore.expect show-ignore.got +	" +  test_done diff --git a/t/t9103-git-svn-graft-branches.sh b/t/t9103-git-svn-graft-branches.sh deleted file mode 100755 index 183ae3b1c2..0000000000 --- a/t/t9103-git-svn-graft-branches.sh +++ /dev/null @@ -1,61 +0,0 @@ -#!/bin/sh -test_description='git-svn graft-branches' -. ./lib-git-svn.sh - -svnrepo="$svnrepo/test-git-svn" - -test_expect_success 'initialize repo' " -	mkdir import && -	cd import && -	mkdir -p trunk branches tags && -	echo hello > trunk/readme && -	svn import -m 'import for git-svn' . $svnrepo && -	cd .. && -	svn cp -m 'tag a' $svnrepo/trunk $svnrepo/tags/a && -	svn cp -m 'branch a' $svnrepo/trunk $svnrepo/branches/a && -	svn co $svnrepo wc && -	cd wc && -	echo feedme >> branches/a/readme && -	poke branches/a/readme && -	svn commit -m hungry && -	cd trunk && -	svn merge -r3:4 $svnrepo/branches/a && -	svn commit -m 'merge with a' && -	cd ../.. && -	git-svn multi-init $svnrepo -T trunk -b branches -t tags && -	git-svn multi-fetch -	" - -r1=`git-rev-list remotes/trunk | tail -n1` -r2=`git-rev-list remotes/tags/a | tail -n1` -r3=`git-rev-list remotes/a | tail -n1` -r4=`git-rev-parse remotes/a` -r5=`git-rev-parse remotes/trunk` - -test_expect_success 'test graft-branches regexes and copies' " -	test -n "$r1" && -	test -n "$r2" && -	test -n "$r3" && -	test -n "$r4" && -	test -n "$r5" && -	git-svn graft-branches && -	grep '^$r2 $r1' $GIT_DIR/info/grafts && -	grep '^$r3 $r1' $GIT_DIR/info/grafts && -	grep '^$r5 ' $GIT_DIR/info/grafts | grep '$r4' | grep '$r1' -	" - -test_debug 'gitk --all & sleep 1' - -test_expect_success 'test graft-branches with tree-joins' " -	rm $GIT_DIR/info/grafts && -	git-svn graft-branches --no-default-regex --no-graft-copy -B && -	grep '^$r3 ' $GIT_DIR/info/grafts | grep '$r1' | grep '$r2' && -	grep '^$r2 $r1' $GIT_DIR/info/grafts && -	grep '^$r5 ' $GIT_DIR/info/grafts | grep '$r1' | grep '$r4' -	" - -# the result of this is kinda funky, we have a strange history and -# this is just a test :) -test_debug 'gitk --all &' - -test_done diff --git a/t/t9104-git-svn-follow-parent.sh b/t/t9104-git-svn-follow-parent.sh index 405b555368..bd4f366e86 100755 --- a/t/t9104-git-svn-follow-parent.sh +++ b/t/t9104-git-svn-follow-parent.sh @@ -3,7 +3,7 @@  # Copyright (c) 2006 Eric Wong  # -test_description='git-svn --follow-parent fetching' +test_description='git-svn fetching'  . ./lib-git-svn.sh  test_expect_success 'initialize repo' " @@ -27,11 +27,141 @@ test_expect_success 'initialize repo' "  	cd ..  	" -test_expect_success 'init and fetch --follow-parent a moved directory' " +test_expect_success 'init and fetch a moved directory' "  	git-svn init -i thunk $svnrepo/thunk && -	git-svn fetch --follow-parent -i thunk && -	git-rev-parse --verify refs/remotes/trunk && -	test '$?' -eq '0' +	git-svn fetch -i thunk && +	test \"\`git-rev-parse --verify refs/remotes/thunk@2\`\" \ +           = \"\`git-rev-parse --verify refs/remotes/thunk~1\`\" && +        test \"\`git-cat-file blob refs/remotes/thunk:readme |\ +                 sed -n -e '3p'\`\" = goodbye && +	test -z \"\`git-config --get svn-remote.svn.fetch \ +	         '^trunk:refs/remotes/thunk@2$'\`\" +	" + +test_expect_success 'init and fetch from one svn-remote' " +        git-config svn-remote.svn.url $svnrepo && +        git-config --add svn-remote.svn.fetch \ +          trunk:refs/remotes/svn/trunk && +        git-config --add svn-remote.svn.fetch \ +          thunk:refs/remotes/svn/thunk && +        git-svn fetch -i svn/thunk && +	test \"\`git-rev-parse --verify refs/remotes/svn/trunk\`\" \ +           = \"\`git-rev-parse --verify refs/remotes/svn/thunk~1\`\" && +        test \"\`git-cat-file blob refs/remotes/svn/thunk:readme |\ +                 sed -n -e '3p'\`\" = goodbye +        " + +test_expect_success 'follow deleted parent' " +        svn cp -m 'resurrecting trunk as junk' \ +               -r2 $svnrepo/trunk $svnrepo/junk && +        git-config --add svn-remote.svn.fetch \ +          junk:refs/remotes/svn/junk && +        git-svn fetch -i svn/thunk && +        git-svn fetch -i svn/junk && +        test -z \"\`git diff svn/junk svn/trunk\`\" && +        test \"\`git merge-base svn/junk svn/trunk\`\" \ +           = \"\`git rev-parse svn/trunk\`\" +        " + +test_expect_success 'follow larger parent' " +        mkdir -p import/trunk/thunk/bump/thud && +        echo hi > import/trunk/thunk/bump/thud/file && +        svn import -m 'import a larger parent' import $svnrepo/larger-parent && +        svn cp -m 'hi' $svnrepo/larger-parent $svnrepo/another-larger && +        git-svn init -i larger $svnrepo/another-larger/trunk/thunk/bump/thud && +        git-svn fetch -i larger && +        git-rev-parse --verify refs/remotes/larger && +        git-rev-parse --verify \ +           refs/remotes/larger-parent/trunk/thunk/bump/thud && +        test \"\`git-merge-base \ +                 refs/remotes/larger-parent/trunk/thunk/bump/thud \ +                 refs/remotes/larger\`\" = \ +             \"\`git-rev-parse refs/remotes/larger\`\" +        true +        " + +test_expect_success 'follow higher-level parent' " +        svn mkdir -m 'follow higher-level parent' $svnrepo/blob && +        svn co $svnrepo/blob blob && +        cd blob && +                echo hi > hi && +                svn add hi && +                svn commit -m 'hihi' && +                cd .. +        svn mkdir -m 'new glob at top level' $svnrepo/glob && +        svn mv -m 'move blob down a level' $svnrepo/blob $svnrepo/glob/blob && +        git-svn init -i blob $svnrepo/glob/blob && +        git-svn fetch -i blob +        " + +test_expect_success 'follow deleted directory' " +	svn mv -m 'bye!' $svnrepo/glob/blob/hi $svnrepo/glob/blob/bye && +	svn rm -m 'remove glob' $svnrepo/glob && +	git-svn init -i glob $svnrepo/glob && +	git-svn fetch -i glob && +	test \"\`git cat-file blob refs/remotes/glob:blob/bye\`\" = hi && +	test \"\`git ls-tree refs/remotes/glob | wc -l \`\" -eq 1 +	" + +# ref: r9270 of the Subversion repository: (http://svn.collab.net/repos/svn) +# in trunk/subversion/bindings/swig/perl +test_expect_success 'follow-parent avoids deleting relevant info' " +	mkdir -p import/trunk/subversion/bindings/swig/perl/t && +	for i in a b c ; do \ +	  echo \$i > import/trunk/subversion/bindings/swig/perl/\$i.pm && +	  echo _\$i > import/trunk/subversion/bindings/swig/perl/t/\$i.t; \ +	done && +	  echo 'bad delete test' > \ +	   import/trunk/subversion/bindings/swig/perl/t/larger-parent && +	  echo 'bad delete test 2' > \ +	   import/trunk/subversion/bindings/swig/perl/another-larger && +	cd import && +	  svn import -m 'r9270 test' . $svnrepo/r9270 && +	cd .. && +	svn co $svnrepo/r9270/trunk/subversion/bindings/swig/perl r9270 && +	cd r9270 && +	  svn mkdir native && +	  svn mv t native/t && +	  for i in a b c; do svn mv \$i.pm native/\$i.pm; done && +	  echo z >> native/t/c.t && +	  poke native/t/c.t && +	  svn commit -m 'reorg test' && +	cd .. && +	git-svn init -i r9270-t \ +	  $svnrepo/r9270/trunk/subversion/bindings/swig/perl/native/t && +	git-svn fetch -i r9270-t && +	test \`git rev-list r9270-t | wc -l\` -eq 2 && +	test \"\`git ls-tree --name-only r9270-t~1\`\" = \ +	     \"\`git ls-tree --name-only r9270-t\`\" +	" + +test_expect_success "track initial change if it was only made to parent" " +	svn cp -m 'wheee!' $svnrepo/r9270/trunk $svnrepo/r9270/drunk && +	git-svn init -i r9270-d \ +	  $svnrepo/r9270/drunk/subversion/bindings/swig/perl/native/t && +	git-svn fetch -i r9270-d && +	test \`git rev-list r9270-d | wc -l\` -eq 3 && +	test \"\`git ls-tree --name-only r9270-t\`\" = \ +	     \"\`git ls-tree --name-only r9270-d\`\" && +	test \"\`git rev-parse r9270-t\`\" = \ +	     \"\`git rev-parse r9270-d~1\`\" +	" + +test_expect_success "track multi-parent paths" " +	svn cp -m 'resurrect /glob' $svnrepo/r9270 $svnrepo/glob && +	git-svn multi-fetch && +	test \`git cat-file commit refs/remotes/glob | \ +	       grep '^parent ' | wc -l\` -eq 2 +	" + +test_expect_success "multi-fetch continues to work" " +	git-svn multi-fetch +	" + +test_expect_success "multi-fetch works off a 'clean' repository" " +	rm -r $GIT_DIR/svn $GIT_DIR/refs/remotes $GIT_DIR/logs && +	mkdir $GIT_DIR/svn && +	git-svn multi-fetch  	"  test_debug 'gitk --all &' diff --git a/t/t9105-git-svn-commit-diff.sh b/t/t9105-git-svn-commit-diff.sh index 6323c7e3ac..c668dd1270 100755 --- a/t/t9105-git-svn-commit-diff.sh +++ b/t/t9105-git-svn-commit-diff.sh @@ -31,4 +31,13 @@ test_expect_success 'test the commit-diff command' "  	cmp readme wc/readme  	" +test_expect_success 'commit-diff to a sub-directory (with git-svn config)' " +	svn import -m 'sub-directory' import $svnrepo/subdir && +	git-svn init $svnrepo/subdir && +	git-svn fetch && +	git-svn commit-diff -r3 '$prev' '$head' && +	svn cat $svnrepo/subdir/readme > readme.2 && +	cmp readme readme.2 +	" +  test_done diff --git a/t/t9107-git-svn-migrate.sh b/t/t9107-git-svn-migrate.sh new file mode 100755 index 0000000000..dc2afdaa45 --- /dev/null +++ b/t/t9107-git-svn-migrate.sh @@ -0,0 +1,112 @@ +#!/bin/sh +# Copyright (c) 2006 Eric Wong +test_description='git-svn metadata migrations from previous versions' +. ./lib-git-svn.sh + +test_expect_success 'setup old-looking metadata' " +	cp $GIT_DIR/config $GIT_DIR/config-old-git-svn && +	mkdir import && +	cd import && +		for i in trunk branches/a branches/b \ +		         tags/0.1 tags/0.2 tags/0.3; do +			mkdir -p \$i && \ +			echo hello >> \$i/README || exit 1 +		done && \ +		svn import -m test . $svnrepo +		cd .. && +	git-svn init $svnrepo && +	git-svn fetch && +	mv $GIT_DIR/svn/* $GIT_DIR/ && +	mv $GIT_DIR/svn/.metadata $GIT_DIR/ && +	rmdir $GIT_DIR/svn && +	git-update-ref refs/heads/git-svn-HEAD refs/remotes/git-svn && +	git-update-ref refs/heads/svn-HEAD refs/remotes/git-svn && +	git-update-ref -d refs/remotes/git-svn refs/remotes/git-svn +	" + +head=`git rev-parse --verify refs/heads/git-svn-HEAD^0` +test_expect_success 'git-svn-HEAD is a real HEAD' "test -n '$head'" + +test_expect_success 'initialize old-style (v0) git-svn layout' " +	mkdir -p $GIT_DIR/git-svn/info $GIT_DIR/svn/info && +	echo $svnrepo > $GIT_DIR/git-svn/info/url && +	echo $svnrepo > $GIT_DIR/svn/info/url && +	git-svn migrate && +	! test -d $GIT_DIR/git-svn && +	git-rev-parse --verify refs/remotes/git-svn^0 && +	git-rev-parse --verify refs/remotes/svn^0 && +	test \`git config --get svn-remote.svn.url\` = '$svnrepo' && +	test \`git config --get svn-remote.svn.fetch\` = \ +             ':refs/remotes/git-svn' +	" + +test_expect_success 'initialize a multi-repository repo' " +	git-svn init $svnrepo -T trunk -t tags -b branches && +	git-config --get-all svn-remote.svn.fetch > fetch.out && +	grep '^trunk:refs/remotes/trunk$' fetch.out && +	test -n \"\`git-config --get svn-remote.svn.branches \ +	            '^branches/\*:refs/remotes/\*$'\`\" && +	test -n \"\`git-config --get svn-remote.svn.tags \ +	            '^tags/\*:refs/remotes/tags/\*$'\`\" && +	git config --unset svn-remote.svn.branches \ +	                        '^branches/\*:refs/remotes/\*$' && +	git config --unset svn-remote.svn.tags \ +	                        '^tags/\*:refs/remotes/tags/\*$' && +	git-config --add svn-remote.svn.fetch 'branches/a:refs/remotes/a' && +	git-config --add svn-remote.svn.fetch 'branches/b:refs/remotes/b' && +	for i in tags/0.1 tags/0.2 tags/0.3; do +		git-config --add svn-remote.svn.fetch \ +		                 \$i:refs/remotes/\$i || exit 1; done +	" + +# refs should all be different, but the trees should all be the same: +test_expect_success 'multi-fetch works on partial urls + paths' " +	git-svn multi-fetch && +	for i in trunk a b tags/0.1 tags/0.2 tags/0.3; do +		git rev-parse --verify refs/remotes/\$i^0 >> refs.out || exit 1; +	    done && +	test -z \"\`sort < refs.out | uniq -d\`\" && +	for i in trunk a b tags/0.1 tags/0.2 tags/0.3; do +	  for j in trunk a b tags/0.1 tags/0.2 tags/0.3; do +		if test \$j != \$i; then continue; fi +	    test -z \"\`git diff refs/remotes/\$i \ +	                         refs/remotes/\$j\`\" ||exit 1; done; done +	" + +test_expect_success 'migrate --minimize on old inited layout' " +	git config --unset-all svn-remote.svn.fetch && +	git config --unset-all svn-remote.svn.url && +	rm -rf $GIT_DIR/svn && +	for i in \`cat fetch.out\`; do +		path=\`expr \$i : '\\([^:]*\\):.*$'\` +		ref=\`expr \$i : '[^:]*:refs/remotes/\\(.*\\)$'\` +		if test -z \"\$ref\"; then continue; fi +		if test -n \"\$path\"; then path=\"/\$path\"; fi +		( mkdir -p $GIT_DIR/svn/\$ref/info/ && +		echo $svnrepo\$path > $GIT_DIR/svn/\$ref/info/url ) || exit 1; +	done && +	git-svn migrate --minimize && +	test -z \"\`git-config -l |grep -v '^svn-remote\.git-svn\.'\`\" && +	git-config --get-all svn-remote.svn.fetch > fetch.out && +	grep '^trunk:refs/remotes/trunk$' fetch.out && +	grep '^branches/a:refs/remotes/a$' fetch.out && +	grep '^branches/b:refs/remotes/b$' fetch.out && +	grep '^tags/0\.1:refs/remotes/tags/0\.1$' fetch.out && +	grep '^tags/0\.2:refs/remotes/tags/0\.2$' fetch.out && +	grep '^tags/0\.3:refs/remotes/tags/0\.3$' fetch.out +	grep '^:refs/remotes/git-svn' fetch.out +	" + +test_expect_success  ".rev_db auto-converted to .rev_db.UUID" " +	git-svn fetch -i trunk && +	expect=$GIT_DIR/svn/trunk/.rev_db.* && +	test -n \"\$expect\" && +	mv \$expect $GIT_DIR/svn/trunk/.rev_db && +	git-svn fetch -i trunk && +	test -L $GIT_DIR/svn/trunk/.rev_db && +	test -f \$expect && +	cmp \$expect $GIT_DIR/svn/trunk/.rev_db +	" + +test_done + diff --git a/t/t9108-git-svn-glob.sh b/t/t9108-git-svn-glob.sh new file mode 100755 index 0000000000..db4344cc84 --- /dev/null +++ b/t/t9108-git-svn-glob.sh @@ -0,0 +1,86 @@ +#!/bin/sh +# Copyright (c) 2007 Eric Wong +test_description='git-svn globbing refspecs' +. ./lib-git-svn.sh + +cat > expect.end <<EOF +the end +hi +start a new branch +initial +EOF + +test_expect_success 'test refspec globbing' " +	mkdir -p trunk/src/a trunk/src/b trunk/doc && +	echo 'hello world' > trunk/src/a/readme && +	echo 'goodbye world' > trunk/src/b/readme && +	svn import -m 'initial' trunk $svnrepo/trunk && +	svn co $svnrepo tmp && +	cd tmp && +		mkdir branches tags && +		svn add branches tags && +		svn cp trunk branches/start && +		svn commit -m 'start a new branch' && +		svn up && +		echo 'hi' >> branches/start/src/b/readme && +		poke branches/start/src/b/readme && +		echo 'hey' >> branches/start/src/a/readme && +		poke branches/start/src/a/readme && +		svn commit -m 'hi' && +		svn up && +		svn cp branches/start tags/end && +		echo 'bye' >> tags/end/src/b/readme && +		poke tags/end/src/b/readme && +		echo 'aye' >> tags/end/src/a/readme && +		poke tags/end/src/a/readme && +		svn commit -m 'the end' && +		echo 'byebye' >> tags/end/src/b/readme && +		poke tags/end/src/b/readme && +		svn commit -m 'nothing to see here' +		cd .. && +	git config --add svn-remote.svn.url $svnrepo && +	git config --add svn-remote.svn.fetch \ +	                 'trunk/src/a:refs/remotes/trunk' && +	git config --add svn-remote.svn.branches \ +	                 'branches/*/src/a:refs/remotes/branches/*' && +	git config --add svn-remote.svn.tags\ +	                 'tags/*/src/a:refs/remotes/tags/*' && +	git-svn multi-fetch && +	git log --pretty=oneline refs/remotes/tags/end | \ +	    sed -e 's/^.\{41\}//' > output.end && +	cmp expect.end output.end && +	test \"\`git rev-parse refs/remotes/tags/end~1\`\" = \ +		\"\`git rev-parse refs/remotes/branches/start\`\" && +	test \"\`git rev-parse refs/remotes/branches/start~2\`\" = \ +		\"\`git rev-parse refs/remotes/trunk\`\" +	" + +echo try to try > expect.two +echo nothing to see here >> expect.two +cat expect.end >> expect.two + +test_expect_success 'test left-hand-side only globbing' " +	git config --add svn-remote.two.url $svnrepo && +	git config --add svn-remote.two.fetch trunk:refs/remotes/two/trunk && +	git config --add svn-remote.two.branches \ +	                 'branches/*:refs/remotes/two/branches/*' && +	git config --add svn-remote.two.tags \ +	                 'tags/*:refs/remotes/two/tags/*' && +	cd tmp && +		echo 'try try' >> tags/end/src/b/readme && +		poke tags/end/src/b/readme && +		svn commit -m 'try to try' +		cd .. && +	git-svn fetch two && +	test \`git rev-list refs/remotes/two/tags/end | wc -l\` -eq 6 && +	test \`git rev-list refs/remotes/two/branches/start | wc -l\` -eq 3 && +	test \`git rev-parse refs/remotes/two/branches/start~2\` = \ +	     \`git rev-parse refs/remotes/two/trunk\` && +	test \`git rev-parse refs/remotes/two/tags/end~3\` = \ +	     \`git rev-parse refs/remotes/two/branches/start\` && +	git log --pretty=oneline refs/remotes/two/tags/end | \ +	    sed -e 's/^.\{41\}//' > output.two && +	cmp expect.two output.two +	" + +test_done diff --git a/t/t9110-git-svn-use-svm-props.sh b/t/t9110-git-svn-use-svm-props.sh new file mode 100755 index 0000000000..9db0d8fd8d --- /dev/null +++ b/t/t9110-git-svn-use-svm-props.sh @@ -0,0 +1,51 @@ +#!/bin/sh +# +# Copyright (c) 2007 Eric Wong +# + +test_description='git-svn useSvmProps test' + +. ./lib-git-svn.sh + +test_expect_success 'load svm repo' " +	svnadmin load -q $rawsvnrepo < ../t9110/svm.dump && +	git-svn init -R arr -i bar $svnrepo/mirror/arr && +	git-svn init -R argh -i dir $svnrepo/mirror/argh && +	git-svn init -R argh -i e $svnrepo/mirror/argh/a/b/c/d/e && +	git-config svn.useSvmProps true && +	git-svn fetch --all +	" + +uuid=161ce429-a9dd-4828-af4a-52023f968c89 + +bar_url=http://mayonaise/svnrepo/bar +test_expect_success 'verify metadata for /bar' " +	git-cat-file commit refs/remotes/bar | \ +	   grep '^git-svn-id: $bar_url@12 $uuid$' && +	git-cat-file commit refs/remotes/bar~1 | \ +	   grep '^git-svn-id: $bar_url@11 $uuid$' && +	git-cat-file commit refs/remotes/bar~2 | \ +	   grep '^git-svn-id: $bar_url@10 $uuid$' && +	git-cat-file commit refs/remotes/bar~3 | \ +	   grep '^git-svn-id: $bar_url@9 $uuid$' && +	git-cat-file commit refs/remotes/bar~4 | \ +	   grep '^git-svn-id: $bar_url@6 $uuid$' && +	git-cat-file commit refs/remotes/bar~5 | \ +	   grep '^git-svn-id: $bar_url@1 $uuid$' +	" + +e_url=http://mayonaise/svnrepo/dir/a/b/c/d/e +test_expect_success 'verify metadata for /dir/a/b/c/d/e' " +	git-cat-file commit refs/remotes/e | \ +	   grep '^git-svn-id: $e_url@1 $uuid$' +	" + +dir_url=http://mayonaise/svnrepo/dir +test_expect_success 'verify metadata for /dir' " +	git-cat-file commit refs/remotes/dir | \ +	   grep '^git-svn-id: $dir_url@2 $uuid$' && +	git-cat-file commit refs/remotes/dir~1 | \ +	   grep '^git-svn-id: $dir_url@1 $uuid$' +	" + +test_done diff --git a/t/t9110/svm.dump b/t/t9110/svm.dump new file mode 100644 index 0000000000..cc799c238d --- /dev/null +++ b/t/t9110/svm.dump @@ -0,0 +1,511 @@ +SVN-fs-dump-format-version: 2 + +UUID: de5973c6-545d-41da-aded-c265f9039e74 + +Revision-number: 0 +Prop-content-length: 56 +Content-length: 56 + +K 8 +svn:date +V 27 +2007-02-17T06:54:59.793104Z +PROPS-END + +Revision-number: 1 +Prop-content-length: 200 +Content-length: 200 + +K 7 +svn:log +V 40 +SVM: initializing mirror for /mirror/arr +K 10 +svn:author +V 3 +svm +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:0 + +K 8 +svn:date +V 27 +2007-02-17T06:55:00.121647Z +PROPS-END + +Node-path:  +Node-kind: dir +Node-action: change +Prop-content-length: 44 +Content-length: 44 + +K 10 +svm:mirror +V 12 +/mirror/arr + +PROPS-END + + +Node-path: mirror +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/arr +Node-kind: dir +Node-action: add +Prop-content-length: 116 +Content-length: 116 + +K 10 +svm:source +V 29 +http://mayonaise/svnrepo!/bar +K 8 +svm:uuid +V 36 +161ce429-a9dd-4828-af4a-52023f968c89 +PROPS-END + + +Revision-number: 2 +Prop-content-length: 182 +Content-length: 182 + +K 7 +svn:log +V 18 +import for git-svn +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:1 + +K 8 +svn:date +V 27 +2007-02-17T05:10:52.108847Z +PROPS-END + +Node-path: mirror/arr +Node-kind: dir +Node-action: change +Prop-content-length: 116 +Content-length: 116 + +K 10 +svm:source +V 29 +http://mayonaise/svnrepo!/bar +K 8 +svm:uuid +V 36 +161ce429-a9dd-4828-af4a-52023f968c89 +PROPS-END + + +Node-path: mirror/arr/zzz +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 33b02bc15ce9557d2dd8484d58f95ac4 +Content-length: 14 + +PROPS-END +zzz + + +Revision-number: 3 +Prop-content-length: 230 +Content-length: 230 + +K 7 +svn:log +V 66 +new symlink is added to a file that was also just made executable + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:6 + +K 8 +svn:date +V 27 +2007-02-17T05:11:01.686891Z +PROPS-END + +Node-path: mirror/arr/zzz +Node-kind: file +Node-action: change +Prop-content-length: 36 +Text-content-length: 4 +Text-content-md5: 33b02bc15ce9557d2dd8484d58f95ac4 +Content-length: 40 + +K 14 +svn:executable +V 1 +* +PROPS-END +zzz + + +Revision-number: 4 +Prop-content-length: 192 +Content-length: 192 + +K 7 +svn:log +V 28 +/bar/d should be in the log + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:9 + +K 8 +svn:date +V 27 +2007-02-17T05:11:07.686552Z +PROPS-END + +Node-path: mirror/arr/d +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 0bee89b07a248e27c83fc3d5951213c1 +Content-length: 14 + +PROPS-END +abc + + +Revision-number: 5 +Prop-content-length: 185 +Content-length: 185 + +K 7 +svn:log +V 20 +add a new directory + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 40 +161ce429-a9dd-4828-af4a-52023f968c89:10 + +K 8 +svn:date +V 27 +2007-02-17T05:11:08.405953Z +PROPS-END + +Node-path: mirror/arr/newdir +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/arr/newdir/dir +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 9cd599a3523898e6a12e13ec787da50a +Content-length: 14 + +PROPS-END +new + + +Revision-number: 6 +Prop-content-length: 196 +Content-length: 196 + +K 7 +svn:log +V 31 +modify a file in new directory + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 40 +161ce429-a9dd-4828-af4a-52023f968c89:11 + +K 8 +svn:date +V 27 +2007-02-17T05:11:09.126645Z +PROPS-END + +Node-path: mirror/arr/newdir/dir +Node-kind: file +Node-action: change +Text-content-length: 8 +Text-content-md5: a950e20332358e523a5e9d571e47fa64 +Content-length: 8 + +new +foo + + +Revision-number: 7 +Prop-content-length: 179 +Content-length: 179 + +K 7 +svn:log +V 14 +update /bar/d + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 40 +161ce429-a9dd-4828-af4a-52023f968c89:12 + +K 8 +svn:date +V 27 +2007-02-17T05:11:09.846221Z +PROPS-END + +Node-path: mirror/arr/d +Node-kind: file +Node-action: change +Text-content-length: 4 +Text-content-md5: 7abb78de7f2756ca8b511cbc879fd5e7 +Content-length: 4 + +cba + + +Revision-number: 8 +Prop-content-length: 201 +Content-length: 201 + +K 7 +svn:log +V 41 +SVM: initializing mirror for /mirror/argh +K 10 +svn:author +V 3 +svm +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:0 + +K 8 +svn:date +V 27 +2007-02-17T06:56:03.703677Z +PROPS-END + +Node-path:  +Node-kind: dir +Node-action: change +Prop-content-length: 57 +Content-length: 57 + +K 10 +svm:mirror +V 25 +/mirror/argh +/mirror/arr + +PROPS-END + + +Node-path: mirror/argh +Node-kind: dir +Node-action: add +Prop-content-length: 116 +Content-length: 116 + +K 10 +svm:source +V 29 +http://mayonaise/svnrepo!/dir +K 8 +svm:uuid +V 36 +161ce429-a9dd-4828-af4a-52023f968c89 +PROPS-END + + +Revision-number: 9 +Prop-content-length: 182 +Content-length: 182 + +K 7 +svn:log +V 18 +import for git-svn +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:1 + +K 8 +svn:date +V 27 +2007-02-17T05:10:52.108847Z +PROPS-END + +Node-path: mirror/argh +Node-kind: dir +Node-action: change +Prop-content-length: 116 +Content-length: 116 + +K 10 +svm:source +V 29 +http://mayonaise/svnrepo!/dir +K 8 +svm:uuid +V 36 +161ce429-a9dd-4828-af4a-52023f968c89 +PROPS-END + + +Node-path: mirror/argh/a +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/argh/a/b +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/argh/a/b/c +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/argh/a/b/c/d +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/argh/a/b/c/d/e +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: mirror/argh/a/b/c/d/e/file +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 9 +Text-content-md5: 3fd46fe46fcdcf062c802ca60dc826d5 +Content-length: 19 + +PROPS-END +deep dir + + +Revision-number: 10 +Prop-content-length: 197 +Content-length: 197 + +K 7 +svn:log +V 33 +try a deep --rmdir with a commit + +K 10 +svn:author +V 7 +svnsync +K 11 +svm:headrev +V 39 +161ce429-a9dd-4828-af4a-52023f968c89:2 + +K 8 +svn:date +V 27 +2007-02-17T05:10:54.847015Z +PROPS-END + +Node-path: mirror/argh/file +Node-kind: file +Node-action: add +Node-copyfrom-rev: 9 +Node-copyfrom-path: mirror/argh/a/b/c/d/e/file +Text-content-length: 9 +Text-content-md5: 3fd46fe46fcdcf062c802ca60dc826d5 +Content-length: 9 + +deep dir + + +Node-path: mirror/argh/a +Node-action: delete + + diff --git a/t/t9111-git-svn-use-svnsync-props.sh b/t/t9111-git-svn-use-svnsync-props.sh new file mode 100755 index 0000000000..483d7f8159 --- /dev/null +++ b/t/t9111-git-svn-use-svnsync-props.sh @@ -0,0 +1,51 @@ +#!/bin/sh +# +# Copyright (c) 2007 Eric Wong +# + +test_description='git-svn useSvnsyncProps test' + +. ./lib-git-svn.sh + +test_expect_success 'load svnsync repo' " +	svnadmin load -q $rawsvnrepo < ../t9111/svnsync.dump && +	git-svn init -R arr -i bar $svnrepo/bar && +	git-svn init -R argh -i dir $svnrepo/dir && +	git-svn init -R argh -i e $svnrepo/dir/a/b/c/d/e && +	git-config svn.useSvnsyncProps true && +	git-svn fetch --all +	" + +uuid=161ce429-a9dd-4828-af4a-52023f968c89 + +bar_url=http://mayonaise/svnrepo/bar +test_expect_success 'verify metadata for /bar' " +	git-cat-file commit refs/remotes/bar | \ +	   grep '^git-svn-id: $bar_url@12 $uuid$' && +	git-cat-file commit refs/remotes/bar~1 | \ +	   grep '^git-svn-id: $bar_url@11 $uuid$' && +	git-cat-file commit refs/remotes/bar~2 | \ +	   grep '^git-svn-id: $bar_url@10 $uuid$' && +	git-cat-file commit refs/remotes/bar~3 | \ +	   grep '^git-svn-id: $bar_url@9 $uuid$' && +	git-cat-file commit refs/remotes/bar~4 | \ +	   grep '^git-svn-id: $bar_url@6 $uuid$' && +	git-cat-file commit refs/remotes/bar~5 | \ +	   grep '^git-svn-id: $bar_url@1 $uuid$' +	" + +e_url=http://mayonaise/svnrepo/dir/a/b/c/d/e +test_expect_success 'verify metadata for /dir/a/b/c/d/e' " +	git-cat-file commit refs/remotes/e | \ +	   grep '^git-svn-id: $e_url@1 $uuid$' +	" + +dir_url=http://mayonaise/svnrepo/dir +test_expect_success 'verify metadata for /dir' " +	git-cat-file commit refs/remotes/dir | \ +	   grep '^git-svn-id: $dir_url@2 $uuid$' && +	git-cat-file commit refs/remotes/dir~1 | \ +	   grep '^git-svn-id: $dir_url@1 $uuid$' +	" + +test_done diff --git a/t/t9111/svnsync.dump b/t/t9111/svnsync.dump new file mode 100644 index 0000000000..a9a46eeb29 --- /dev/null +++ b/t/t9111/svnsync.dump @@ -0,0 +1,562 @@ +SVN-fs-dump-format-version: 2 + +UUID: b4bfe35e-f256-4096-874c-08c5639ecad7 + +Revision-number: 0 +Prop-content-length: 240 +Content-length: 240 + +K 18 +svn:sync-from-uuid +V 36 +161ce429-a9dd-4828-af4a-52023f968c89 +K 10 +svn:author +V 7 +svnsync +K 24 +svn:sync-last-merged-rev +V 2 +12 +K 8 +svn:date +V 27 +2007-02-17T05:10:52.017552Z +K 17 +svn:sync-from-url +V 24 +http://mayonaise/svnrepo +PROPS-END + +Revision-number: 1 +Prop-content-length: 120 +Content-length: 120 + +K 7 +svn:log +V 18 +import for git-svn +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:10:52.108847Z +PROPS-END + +Node-path: bar +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: bar/zzz +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 33b02bc15ce9557d2dd8484d58f95ac4 +Content-length: 14 + +PROPS-END +zzz + + +Node-path: dir +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a/b +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a/b/c +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a/b/c/d +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a/b/c/d/e +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: dir/a/b/c/d/e/file +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 9 +Text-content-md5: 3fd46fe46fcdcf062c802ca60dc826d5 +Content-length: 19 + +PROPS-END +deep dir + + +Node-path: exec.sh +Node-kind: file +Node-action: add +Prop-content-length: 35 +Text-content-length: 10 +Text-content-md5: 3e2b31c72181b87149ff995e7202c0e3 +Content-length: 45 + +K 14 +svn:executable +V 0 + +PROPS-END +#!/bin/sh + + +Node-path: foo +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: d3b07384d113edec49eaa6238ad5ff00 +Content-length: 14 + +PROPS-END +foo + + +Node-path: foo.link +Node-kind: file +Node-action: add +Prop-content-length: 33 +Text-content-length: 8 +Text-content-md5: 1043146e49ef02cab12eef865cb34ff3 +Content-length: 41 + +K 11 +svn:special +V 1 +* +PROPS-END +link foo + +Revision-number: 2 +Prop-content-length: 135 +Content-length: 135 + +K 7 +svn:log +V 33 +try a deep --rmdir with a commit + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:10:54.847015Z +PROPS-END + +Node-path: dir/file +Node-kind: file +Node-action: add +Node-copyfrom-rev: 1 +Node-copyfrom-path: dir/a/b/c/d/e/file +Text-content-length: 9 +Text-content-md5: 3fd46fe46fcdcf062c802ca60dc826d5 +Content-length: 9 + +deep dir + + +Node-path: dir/a +Node-action: delete + + +Node-path: file +Node-kind: file +Node-action: add +Node-copyfrom-rev: 1 +Node-copyfrom-path: dir/a/b/c/d/e/file +Text-content-length: 9 +Text-content-md5: 3fd46fe46fcdcf062c802ca60dc826d5 +Content-length: 9 + +deep dir + + +Revision-number: 3 +Prop-content-length: 136 +Content-length: 136 + +K 7 +svn:log +V 34 +remove executable bit from a file + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:10:58.232691Z +PROPS-END + +Node-path: exec.sh +Node-kind: file +Node-action: change +Prop-content-length: 10 +Text-content-length: 10 +Text-content-md5: 3e2b31c72181b87149ff995e7202c0e3 +Content-length: 20 + +PROPS-END +#!/bin/sh + + +Revision-number: 4 +Prop-content-length: 131 +Content-length: 131 + +K 7 +svn:log +V 29 +add executable bit back file + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:10:59.666560Z +PROPS-END + +Node-path: exec.sh +Node-kind: file +Node-action: change +Prop-content-length: 36 +Text-content-length: 10 +Text-content-md5: 3e2b31c72181b87149ff995e7202c0e3 +Content-length: 46 + +K 14 +svn:executable +V 1 +* +PROPS-END +#!/bin/sh + + +Revision-number: 5 +Prop-content-length: 154 +Content-length: 154 + +K 7 +svn:log +V 52 +executable file becomes a symlink to bar/zzz (file) + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:00.676495Z +PROPS-END + +Node-path: exec.sh +Node-kind: file +Node-action: change +Prop-content-length: 33 +Text-content-length: 12 +Text-content-md5: f138693371665cc117742508761d684d +Content-length: 45 + +K 11 +svn:special +V 1 +* +PROPS-END +link bar/zzz + +Revision-number: 6 +Prop-content-length: 168 +Content-length: 168 + +K 7 +svn:log +V 66 +new symlink is added to a file that was also just made executable + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:01.686891Z +PROPS-END + +Node-path: bar/zzz +Node-kind: file +Node-action: change +Prop-content-length: 36 +Text-content-length: 4 +Text-content-md5: 33b02bc15ce9557d2dd8484d58f95ac4 +Content-length: 40 + +K 14 +svn:executable +V 1 +* +PROPS-END +zzz + + +Node-path: exec-2.sh +Node-kind: file +Node-action: add +Node-copyfrom-rev: 5 +Node-copyfrom-path: exec.sh +Text-content-length: 12 +Text-content-md5: f138693371665cc117742508761d684d +Content-length: 12 + +link bar/zzz + +Revision-number: 7 +Prop-content-length: 136 +Content-length: 136 + +K 7 +svn:log +V 34 +modify a symlink to become a file + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:02.677035Z +PROPS-END + +Node-path: exec-2.sh +Node-kind: file +Node-action: change +Prop-content-length: 10 +Text-content-length: 9 +Text-content-md5: 8e92eff9e911886cede27d420f89c735 +Content-length: 19 + +PROPS-END +git help + + +Revision-number: 8 +Prop-content-length: 109 +Content-length: 109 + +K 7 +svn:log +V 8 +éï∏ + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:03.676862Z +PROPS-END + +Node-path: exec-2.sh +Node-kind: file +Node-action: change +Text-content-length: 17 +Text-content-md5: 49881954063cf26ca48c212396a957ca +Content-length: 17 + +git help +# hello + + +Revision-number: 9 +Prop-content-length: 130 +Content-length: 130 + +K 7 +svn:log +V 28 +/bar/d should be in the log + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:07.686552Z +PROPS-END + +Node-path: bar/d +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 0bee89b07a248e27c83fc3d5951213c1 +Content-length: 14 + +PROPS-END +abc + + +Revision-number: 10 +Prop-content-length: 122 +Content-length: 122 + +K 7 +svn:log +V 20 +add a new directory + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:08.405953Z +PROPS-END + +Node-path: bar/newdir +Node-kind: dir +Node-action: add +Prop-content-length: 10 +Content-length: 10 + +PROPS-END + + +Node-path: bar/newdir/dir +Node-kind: file +Node-action: add +Prop-content-length: 10 +Text-content-length: 4 +Text-content-md5: 9cd599a3523898e6a12e13ec787da50a +Content-length: 14 + +PROPS-END +new + + +Revision-number: 11 +Prop-content-length: 133 +Content-length: 133 + +K 7 +svn:log +V 31 +modify a file in new directory + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:09.126645Z +PROPS-END + +Node-path: bar/newdir/dir +Node-kind: file +Node-action: change +Text-content-length: 8 +Text-content-md5: a950e20332358e523a5e9d571e47fa64 +Content-length: 8 + +new +foo + + +Revision-number: 12 +Prop-content-length: 116 +Content-length: 116 + +K 7 +svn:log +V 14 +update /bar/d + +K 10 +svn:author +V 7 +svnsync +K 8 +svn:date +V 27 +2007-02-17T05:11:09.846221Z +PROPS-END + +Node-path: bar/d +Node-kind: file +Node-action: change +Text-content-length: 4 +Text-content-md5: 7abb78de7f2756ca8b511cbc879fd5e7 +Content-length: 4 + +cba + + diff --git a/t/test-lib.sh b/t/test-lib.sh index 37822fc13d..c0754747fb 100755 --- a/t/test-lib.sh +++ b/t/test-lib.sh @@ -255,8 +255,8 @@ test_done () {  PATH=$(pwd)/..:$PATH  GIT_EXEC_PATH=$(pwd)/..  GIT_TEMPLATE_DIR=$(pwd)/../templates/blt -HOME=$(pwd)/trash -export PATH GIT_EXEC_PATH GIT_TEMPLATE_DIR HOME +GIT_CONFIG=.git/config +export PATH GIT_EXEC_PATH GIT_TEMPLATE_DIR GIT_CONFIG  GITPERLLIB=$(pwd)/../perl/blib/lib:$(pwd)/../perl/blib/arch/auto/Git  export GITPERLLIB @@ -264,6 +264,12 @@ test -d ../templates/blt || {  	error "You haven't built things yet, have you?"  } +if ! test -x ../test-chmtime; then +	echo >&2 'You need to build test-chmtime:' +	echo >&2 'Run "make test-chmtime" in the source (toplevel) directory' +	exit 1 +fi +  # Test repository  test=trash  rm -fr "$test" diff --git a/test-chmtime.c b/test-chmtime.c new file mode 100644 index 0000000000..90da448ebe --- /dev/null +++ b/test-chmtime.c @@ -0,0 +1,61 @@ +#include "git-compat-util.h" +#include <utime.h> + +static const char usage_str[] = "(+|=|=+|=-|-)<seconds> <file>..."; + +int main(int argc, const char *argv[]) +{ +	int i; +	int set_eq; +	long int set_time; +	char *test; +	const char *timespec; + +	if (argc < 3) +		goto usage; + +	timespec = argv[1]; +	set_eq = (*timespec == '=') ? 1 : 0; +	if (set_eq) { +		timespec++; +		if (*timespec == '+') { +			set_eq = 2; /* relative "in the future" */ +			timespec++; +		} +	} +	set_time = strtol(timespec, &test, 10); +	if (*test) { +		fprintf(stderr, "Not a base-10 integer: %s\n", argv[1] + 1); +		goto usage; +	} +	if ((set_eq && set_time < 0) || set_eq == 2) { +		time_t now = time(NULL); +		set_time += now; +	} + +	for (i = 2; i < argc; i++) { +		struct stat sb; +		struct utimbuf utb; + +		if (stat(argv[i], &sb) < 0) { +			fprintf(stderr, "Failed to stat %s: %s\n", +			        argv[i], strerror(errno)); +			return -1; +		} + +		utb.actime = sb.st_atime; +		utb.modtime = set_eq ? set_time : sb.st_mtime + set_time; + +		if (utime(argv[i], &utb) < 0) { +			fprintf(stderr, "Failed to modify time on %s: %s\n", +			        argv[i], strerror(errno)); +			return -1; +		} +	} + +	return 0; + +usage: +	fprintf(stderr, "Usage: %s %s\n", argv[0], usage_str); +	return -1; +} diff --git a/upload-pack.c b/upload-pack.c index 3648aae1a7..804bbb6c9e 100644 --- a/upload-pack.c +++ b/upload-pack.c @@ -455,7 +455,7 @@ static int get_common_commits(void)  			continue;  		}  		len = strip(line, len); -		if (!strncmp(line, "have ", 5)) { +		if (!prefixcmp(line, "have ")) {  			switch (got_sha1(line+5, sha1)) {  			case -1: /* they have what we do not */  				if (multi_ack && ok_to_give_up()) @@ -502,7 +502,7 @@ static void receive_needs(void)  		if (!len)  			break; -		if (!strncmp("shallow ", line, 8)) { +		if (!prefixcmp(line, "shallow ")) {  			unsigned char sha1[20];  			struct object *object;  			use_thin_pack = 0; @@ -515,7 +515,7 @@ static void receive_needs(void)  			add_object_array(object, NULL, &shallows);  			continue;  		} -		if (!strncmp("deepen ", line, 7)) { +		if (!prefixcmp(line, "deepen ")) {  			char *end;  			use_thin_pack = 0;  			depth = strtol(line + 7, &end, 0); @@ -523,7 +523,7 @@ static void receive_needs(void)  				die("Invalid deepen: %s", line);  			continue;  		} -		if (strncmp("want ", line, 5) || +		if (prefixcmp(line, "want ") ||  		    get_sha1_hex(line+5, sha1_buf))  			die("git-upload-pack: protocol error, "  			    "expected to get sha, not '%s'", line); @@ -656,7 +656,7 @@ int main(int argc, char **argv)  			strict = 1;  			continue;  		} -		if (!strncmp(arg, "--timeout=", 10)) { +		if (!prefixcmp(arg, "--timeout=")) {  			timeout = atoi(arg+10);  			continue;  		} diff --git a/wt-status.c b/wt-status.c index e346511153..a25632bc87 100644 --- a/wt-status.c +++ b/wt-status.c @@ -307,7 +307,7 @@ void wt_status_print(struct wt_status *s)  	if (s->branch) {  		const char *on_what = "On branch ";  		const char *branch_name = s->branch; -		if (!strncmp(branch_name, "refs/heads/", 11)) +		if (!prefixcmp(branch_name, "refs/heads/"))  			branch_name += 11;  		else if (!strcmp(branch_name, "HEAD")) {  			branch_name = ""; @@ -352,7 +352,7 @@ int git_status_config(const char *k, const char *v)  		wt_status_use_color = git_config_colorbool(k, v);  		return 0;  	} -	if (!strncmp(k, "status.color.", 13) || !strncmp(k, "color.status.", 13)) { +	if (!prefixcmp(k, "status.color.") || !prefixcmp(k, "color.status.")) {  		int slot = parse_status_slot(k, 13);  		color_parse(v, k, wt_status_colors[slot]);  	} | 
