mirror of
				https://github.com/mnauw/git-remote-hg.git
				synced 2025-10-31 00:25:48 +01:00 
			
		
		
		
	Compare commits
	
		
			112 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 426ed618b2 | ||
|  | 5f34d049b9 | ||
|  | ea7e9bf31a | ||
|  | a3a36883c5 | ||
|  | 0fdd28319a | ||
|  | dfa3ad5fca | ||
|  | ebd5bdb111 | ||
|  | 00ac711fb2 | ||
|  | aadc899982 | ||
|  | 4d38bff053 | ||
|  | a8cd6a92b3 | ||
|  | a08ad9d2b4 | ||
|  | 8c08b6de21 | ||
|  | 949345fb11 | ||
|  | 0d49f75131 | ||
|  | 7159e4a030 | ||
|  | cdcd70b453 | ||
|  | f5c38f3a59 | ||
|  | 3b11156e69 | ||
|  | afc1f3a2c2 | ||
|  | e596a5f457 | ||
|  | ec654d4682 | ||
|  | 7913920a97 | ||
|  | da60201ae3 | ||
|  | 704869df29 | ||
|  | 5769e965eb | ||
|  | 1ee28bd233 | ||
|  | 1796289df3 | ||
|  | 929ae262f5 | ||
|  | 4328aa1c19 | ||
|  | 919678be4a | ||
|  | ac8f659620 | ||
|  | 28ed63b707 | ||
|  | dc91c58e1c | ||
|  | 46178f546a | ||
|  | 4c0e8e6439 | ||
|  | 59b5a8c848 | ||
|  | be2445963b | ||
|  | 8b4bfe7e87 | ||
|  | 741b440fcc | ||
|  | 03b1ac9845 | ||
|  | f3a8546406 | ||
|  | ccc9e55b7e | ||
|  | b516aa9326 | ||
|  | 08626200d0 | ||
|  | b60eb47173 | ||
|  | 76162ce148 | ||
|  | 7ae03f7640 | ||
|  | 95da53badd | ||
|  | a0608664ca | ||
|  | 9d6d135855 | ||
|  | 60a6c7b36d | ||
|  | 74d1aa14ac | ||
|  | 1d85449b0b | ||
|  | fe8b8c1a61 | ||
|  | 510441bba9 | ||
|  | fa3484e08b | ||
|  | d1544e2ccd | ||
|  | 153a216f47 | ||
|  | b3cdbe8e96 | ||
|  | d11509cab7 | ||
|  | 4d01165b1b | ||
|  | a030d603ac | ||
|  | 3d4a5a6d7e | ||
|  | 7fb9d9b642 | ||
|  | fad59f53eb | ||
|  | e17e147fb1 | ||
|  | 4878023a8b | ||
|  | 0dfae24d21 | ||
|  | c7dbb5612b | ||
|  | cc4e5659d9 | ||
|  | 2c993b3433 | ||
|  | 4944a384cd | ||
|  | f29c42c645 | ||
|  | 7b7c65f72d | ||
|  | cee3ed7c00 | ||
|  | 01c9a981c7 | ||
|  | d0a5888580 | ||
|  | a5dfc9025b | ||
|  | 4d337cff06 | ||
|  | 5cc271ef18 | ||
|  | c8fff2cd06 | ||
|  | c95fba3c18 | ||
|  | aaef56a2a3 | ||
|  | a16c69a99c | ||
|  | 00e95fd8df | ||
|  | 5bf7aad6e3 | ||
|  | 9b8e0ec2c0 | ||
|  | b309562574 | ||
|  | e25d3d78cd | ||
|  | ed5a70706a | ||
|  | 0faf2c9189 | ||
|  | ada49422a7 | ||
|  | 580cea0d31 | ||
|  | 1f376e437f | ||
|  | 4108665799 | ||
|  | fc28115a53 | ||
|  | e3009683f8 | ||
|  | 54cec85f94 | ||
|  | 5ad322c54d | ||
|  | 13bbc8a342 | ||
|  | 673b50d3f4 | ||
|  | 35ecb45fda | ||
|  | 1456e68129 | ||
|  | de95133416 | ||
|  | e0b752be8f | ||
|  | f050de1bcc | ||
|  | 0bf3db826b | ||
|  | 3698638e98 | ||
|  | 765f9ae287 | ||
|  | 5ddcdd33ec | ||
|  | 435373ee83 | 
							
								
								
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | /build/ | ||||||
|  | /dist/ | ||||||
|  | /git_remote_hg.egg-info/ | ||||||
							
								
								
									
										36
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										36
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,28 +1,20 @@ | |||||||
| language: python | dist: xenial | ||||||
|  | language: minimal | ||||||
|  |  | ||||||
| install: | cache: | ||||||
|   - if [ "$HG_VERSION" != "dev" ]; |   directories: | ||||||
|     then pip install -q Mercurial${HG_VERSION+==$HG_VERSION}; |     - $HOME/.cache/git-remote-hg | ||||||
|     else pip install -q http://selenic.com/repo/hg/archive/tip.tar.gz; |  | ||||||
|     fi |  | ||||||
|   - pip install -q dulwich hg-git==0.6.1 || true |  | ||||||
|  |  | ||||||
| before_script: |  | ||||||
|   - hg --version || true |  | ||||||
|   - pip show hg-git dulwich |  | ||||||
|  |  | ||||||
| script: | script: | ||||||
|   - make test |   - ./tools/check-versions hg:$HG_VERSION | ||||||
|  |  | ||||||
| matrix: | matrix: | ||||||
|   include: |   include: | ||||||
|     - env: HG_VERSION=2.9.1 |     - env: | ||||||
|     - env: HG_VERSION=2.8.2 |     - env: HG_VERSION=@ | ||||||
|     - env: HG_VERSION=2.7.2 |     - env: HG_VERSION=5.0 | ||||||
|     - env: HG_VERSION=3.0 |     - env: HG_VERSION=4.9 | ||||||
|     - env: HG_VERSION=3.5.2 |     - env: HG_VERSION=4.8 | ||||||
|     - env: HG_VERSION=3.6.3 |     - env: HG_VERSION=4.7 | ||||||
|     - env: HG_VERSION=3.7 |     - env: HG_VERSION=4.6 | ||||||
|     - env: HG_VERSION=dev |     - env: HG_VERSION=4.5 | ||||||
|     - python: 2.7 |  | ||||||
|     - python: 2.6 |  | ||||||
|   | |||||||
							
								
								
									
										43
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										43
									
								
								Makefile
									
									
									
									
									
								
							| @@ -3,7 +3,28 @@ prefix := $(HOME) | |||||||
| bindir := $(prefix)/bin | bindir := $(prefix)/bin | ||||||
| mandir := $(prefix)/share/man/man1 | mandir := $(prefix)/share/man/man1 | ||||||
|  |  | ||||||
| all: doc | all: build doc | ||||||
|  |  | ||||||
|  | build: | ||||||
|  | 	if [ -n "$$PYTHON" ] && "$$PYTHON" -c 'import mercurial' 2> /dev/null ; then \ | ||||||
|  | 		: Use chosen Python version ; \ | ||||||
|  | 	elif python3 -c 'import mercurial' 2> /dev/null ; then \ | ||||||
|  | 		PYTHON=python3 ; \ | ||||||
|  | 	elif python2 -c 'import mercurial' 2> /dev/null ; then \ | ||||||
|  | 		PYTHON=python2 ; \ | ||||||
|  | 	elif python -c 'import mercurial' 2> /dev/null ; then \ | ||||||
|  | 		PYTHON=python ; \ | ||||||
|  | 	else \ | ||||||
|  | 		echo 'Python with Mercurial not available' >&2 ; \ | ||||||
|  | 		exit 1 ; \ | ||||||
|  | 	fi ; \ | ||||||
|  | 	mkdir -p bin ; \ | ||||||
|  | 	for s in git-remote-hg git-hg-helper ; do \ | ||||||
|  | 		printf "%s\n" "#!/usr/bin/env $$PYTHON" > "bin/$$s" ; \ | ||||||
|  | 		tail -n +2 "./$$s" >> "bin/$$s" ; \ | ||||||
|  | 		chmod 755 "bin/$$s" ; \ | ||||||
|  | 		touch -r "./$$s" "bin/$$s" ; \ | ||||||
|  | 	done | ||||||
|  |  | ||||||
| doc: doc/git-remote-hg.1 | doc: doc/git-remote-hg.1 | ||||||
|  |  | ||||||
| @@ -15,15 +36,29 @@ doc/git-remote-hg.1: doc/git-remote-hg.txt | |||||||
|  |  | ||||||
| clean: | clean: | ||||||
| 	$(RM) doc/git-remote-hg.1 | 	$(RM) doc/git-remote-hg.1 | ||||||
|  | 	$(RM) -r bin/ | ||||||
|  |  | ||||||
| D = $(DESTDIR) | D = $(DESTDIR) | ||||||
|  |  | ||||||
| install: | install: build | ||||||
| 	install -d -m 755 $(D)$(bindir)/ | 	install -d -m 755 $(D)$(bindir)/ | ||||||
| 	install -m 755 git-remote-hg $(D)$(bindir)/git-remote-hg | 	install -m 755 bin/git-remote-hg $(D)$(bindir)/git-remote-hg | ||||||
|  | 	install -m 755 bin/git-hg-helper $(D)$(bindir)/git-hg-helper | ||||||
|  |  | ||||||
| install-doc: doc | install-doc: doc | ||||||
| 	install -d -m 755 $(D)$(mandir)/ | 	install -d -m 755 $(D)$(mandir)/ | ||||||
| 	install -m 644 doc/git-remote-hg.1 $(D)$(mandir)/git-remote-hg.1 | 	install -m 644 doc/git-remote-hg.1 $(D)$(mandir)/git-remote-hg.1 | ||||||
|  |  | ||||||
| .PHONY: all test install install-doc clean | pypi: | ||||||
|  | 	version=`git describe --tags ${REV}` && \ | ||||||
|  | 		sed -i "s/version = .*/version = '$$version'[1:]/" setup.py | ||||||
|  | 	-rm -rf dist build | ||||||
|  | 	python setup.py sdist bdist_wheel | ||||||
|  |  | ||||||
|  | pypi-upload: | ||||||
|  | 	twine upload dist/* | ||||||
|  |  | ||||||
|  | pypi-test: | ||||||
|  | 	twine upload --repository-url https://test.pypi.org/legacy/ dist/* | ||||||
|  |  | ||||||
|  | .PHONY: all build test install install-doc clean pypy pypy-upload | ||||||
|   | |||||||
| @@ -13,6 +13,13 @@ wget https://raw.github.com/mnauw/git-remote-hg/master/git-remote-hg -O ~/bin/gi | |||||||
| chmod +x ~/bin/git-remote-hg | chmod +x ~/bin/git-remote-hg | ||||||
| -------------------------------------- | -------------------------------------- | ||||||
|  |  | ||||||
|  | In Windows, you also need to put and an NTFS symbolic link named `python2.exe` somewhere | ||||||
|  | on your `$PATH` pointing to your Python 2 executable: | ||||||
|  |  | ||||||
|  | -------------------------------------- | ||||||
|  | mklink <path to link> <path to python.exe> | ||||||
|  | -------------------------------------- | ||||||
|  |  | ||||||
| That's it :) | That's it :) | ||||||
|  |  | ||||||
| Obviously you will need Mercurial installed. | Obviously you will need Mercurial installed. | ||||||
| @@ -146,8 +153,6 @@ Limitations of the remote-helpers' framework apply. In particular, these | |||||||
| commands don't work: | commands don't work: | ||||||
|  |  | ||||||
| * `git push origin :branch-to-delete` | * `git push origin :branch-to-delete` | ||||||
| * `git push origin old:new` (it will push 'old') (patches available) |  | ||||||
| * `git push --dry-run origin branch` (it will push) (patches available) |  | ||||||
|  |  | ||||||
| **** | **** | ||||||
| Another limitation is that if `git log` reports a rename, this will not survive | Another limitation is that if `git log` reports a rename, this will not survive | ||||||
| @@ -247,9 +252,11 @@ to fully complete the conversion (prior to subsequent pushing). | |||||||
| Some Mercurial names (of branches, bookmarks, tags) may not be a valid git | Some Mercurial names (of branches, bookmarks, tags) may not be a valid git | ||||||
| refname. See e.g. `man git-check-ref-format` for a rather involved set of rules. | refname. See e.g. `man git-check-ref-format` for a rather involved set of rules. | ||||||
| Moreover, while a slash `/` is allowed, it is not supported to have both a `parent` | Moreover, while a slash `/` is allowed, it is not supported to have both a `parent` | ||||||
| and `parent/child` branch (though only the latter is allowed). So, pending some | and `parent/child` branch (though only the latter is allowed). Even though | ||||||
| "nice" bidirectional encoding (not even e.g. URL encoding is safe actually), it is more | it is not quite (bidirectionally) safe, a (percentage) URL encoding | ||||||
| likely that only a few instances (out of a whole Mercurial repo) are | (with some additional twist) is performed to obtain sane git refnames, at least | ||||||
|  | so for most cases.  If some nasty cases still slip through, then likely only | ||||||
|  | a few instances (out of a whole Mercurial repo) are | ||||||
| problematic.  This could be handled by a single-branch clone and/or configuring | problematic.  This could be handled by a single-branch clone and/or configuring | ||||||
| a suitable refspec.  However, it might be more convenient to simply filter out a | a suitable refspec.  However, it might be more convenient to simply filter out a | ||||||
| few unimportant pesky cases, which can be done by configuring a regural | few unimportant pesky cases, which can be done by configuring a regural | ||||||
|   | |||||||
							
								
								
									
										300
									
								
								git-hg-helper
									
									
									
									
									
								
							
							
						
						
									
										300
									
								
								git-hg-helper
									
									
									
									
									
								
							| @@ -1,10 +1,15 @@ | |||||||
| #!/usr/bin/env python2 | #!/usr/bin/env python | ||||||
| # | # | ||||||
| # Copyright (c) 2016 Mark Nauwelaerts | # Copyright (c) 2016 Mark Nauwelaerts | ||||||
| # | # | ||||||
|  |  | ||||||
| from mercurial import hg, ui, commands, util | from mercurial import hg, ui, commands, util | ||||||
| from mercurial import context, subrepo | from mercurial import context, subrepo | ||||||
|  | try: | ||||||
|  |     # hg >= 5.8 | ||||||
|  |     from mercurial.utils import urlutil | ||||||
|  | except ImportError: | ||||||
|  |     from mercurial import util as urlutil | ||||||
|  |  | ||||||
| import re | import re | ||||||
| import sys | import sys | ||||||
| @@ -16,13 +21,72 @@ import logging | |||||||
| import threading | import threading | ||||||
|  |  | ||||||
| # thanks go to git-remote-helper for some helper functions | # thanks go to git-remote-helper for some helper functions | ||||||
|  | # likewise so for python2/3 compatibility | ||||||
|  |  | ||||||
| def die(msg, *args): | # generic | ||||||
|     sys.stderr.write('ERROR: %s\n' % (msg % args)) | class basecompat: | ||||||
|  |     @staticmethod | ||||||
|  |     def char(c): | ||||||
|  |       assert len(c) == 1 | ||||||
|  |       return c[0] | ||||||
|  |  | ||||||
|  | if sys.version_info[0] == 3: | ||||||
|  |     import locale | ||||||
|  |     class compat(basecompat): | ||||||
|  |         # sigh ... wonderful python3 ... as taken from Mercurial's pycompat | ||||||
|  |         @staticmethod | ||||||
|  |         def decode_sysarg(arg): | ||||||
|  |             if os.name == r'nt': | ||||||
|  |                 return arg.encode("mbcs", "ignore") | ||||||
|  |             else: | ||||||
|  |                 enc = ( | ||||||
|  |                     locale.getlocale()[1] | ||||||
|  |                     or locale.getdefaultlocale()[1] | ||||||
|  |                     or sys.getfilesystemencoding() | ||||||
|  |                 ) | ||||||
|  |                 return arg.encode(enc, "surrogateescape") | ||||||
|  |         # mostly used for straight 'cast' (not real unicode content) | ||||||
|  |         @staticmethod | ||||||
|  |         def to_b(s, *args): | ||||||
|  |             if isinstance(s, str): | ||||||
|  |                 args = args or ['latin-1'] | ||||||
|  |                 return s.encode(*args) | ||||||
|  |             return s | ||||||
|  |         stdin = sys.stdin.buffer | ||||||
|  |         stdout = sys.stdout.buffer | ||||||
|  |         stderr = sys.stderr.buffer | ||||||
|  |         getcwd = os.getcwdb | ||||||
|  |         @staticmethod | ||||||
|  |         def getenvb(val, default): | ||||||
|  |             result = os.getenv(val.decode(), default.decode() if hasattr(default, 'decode') else default) | ||||||
|  |              # if result is a string, get bytes instead | ||||||
|  |             result = result.encode() if hasattr(result, 'encode') else result | ||||||
|  |             return result | ||||||
|  |         getenv = os.getenvb if os.supports_bytes_environ else getenvb | ||||||
|  | else: | ||||||
|  |     class compat(basecompat): | ||||||
|  |         # life was simple in those days ... | ||||||
|  |         @staticmethod | ||||||
|  |         def to_b(s, *args): | ||||||
|  |             return s | ||||||
|  |         decode_sysarg = to_b | ||||||
|  |         stdin = sys.stdin | ||||||
|  |         stdout = sys.stdout | ||||||
|  |         stderr = sys.stderr | ||||||
|  |         getcwd = staticmethod(os.getcwd) | ||||||
|  |         getenv = staticmethod(os.getenv) | ||||||
|  |  | ||||||
|  | def puts(msg = b''): | ||||||
|  |     compat.stdout.write(msg) | ||||||
|  |     compat.stdout.write(b'\n') | ||||||
|  |  | ||||||
|  | def die(msg): | ||||||
|  |     compat.stderr.write(b'ERROR: %s\n' % compat.to_b(msg, 'utf-8')) | ||||||
|     sys.exit(1) |     sys.exit(1) | ||||||
|  |  | ||||||
| def warn(msg, *args): | def warn(msg): | ||||||
|     sys.stderr.write('WARNING: %s\n' % (msg % args)) |     compat.stderr.write(b'WARNING: %s\n' % compat.to_b(msg, 'utf-8')) | ||||||
|  |     compat.stderr.flush() | ||||||
|  |  | ||||||
| def info(msg, *args): | def info(msg, *args): | ||||||
|     logger.info(msg, *args) |     logger.info(msg, *args) | ||||||
| @@ -44,7 +108,7 @@ class GitHgRepo: | |||||||
|     def __init__(self, topdir=None, gitdir=None): |     def __init__(self, topdir=None, gitdir=None): | ||||||
|         if gitdir != None: |         if gitdir != None: | ||||||
|             self.gitdir = gitdir |             self.gitdir = gitdir | ||||||
|             self.topdir = os.path.join(gitdir, '..') # will have to do |             self.topdir = os.path.join(gitdir, b'..') # will have to do | ||||||
|         else: |         else: | ||||||
|             self.topdir = None |             self.topdir = None | ||||||
|             if not topdir: |             if not topdir: | ||||||
| @@ -53,7 +117,7 @@ class GitHgRepo: | |||||||
|                     if not os.path.exists('.git'): |                     if not os.path.exists('.git'): | ||||||
|                         # now we lost where we are |                         # now we lost where we are | ||||||
|                         raise Exception('failed to determine topdir') |                         raise Exception('failed to determine topdir') | ||||||
|                     topdir = '.' |                     topdir = b'.' | ||||||
|             self.topdir = topdir |             self.topdir = topdir | ||||||
|             self.gitdir = self.run_cmd(['rev-parse', '--git-dir']).strip() |             self.gitdir = self.run_cmd(['rev-parse', '--git-dir']).strip() | ||||||
|             if not self.gitdir: |             if not self.gitdir: | ||||||
| @@ -64,7 +128,7 @@ class GitHgRepo: | |||||||
|         self.hg_repos = {} |         self.hg_repos = {} | ||||||
|  |  | ||||||
|     def identity(self): |     def identity(self): | ||||||
|         return '[%s|%s]' % (os.getcwd(), self.topdir) |         return b'[%s|%s]' % (compat.getcwd(), self.topdir or b'') | ||||||
|  |  | ||||||
|     def start_cmd(self, args, **kwargs): |     def start_cmd(self, args, **kwargs): | ||||||
|         cmd = ['git'] + args |         cmd = ['git'] + args | ||||||
| @@ -82,7 +146,7 @@ class GitHgRepo: | |||||||
|         process = self.start_cmd(args, **kwargs) |         process = self.start_cmd(args, **kwargs) | ||||||
|         output = process.communicate()[0] |         output = process.communicate()[0] | ||||||
|         if check and process.returncode != 0: |         if check and process.returncode != 0: | ||||||
|             die('command failed: %s', ' '.join(cmd)) |             die(b'command failed: %s' % b' '.join([compat.to_b(a) for a in cmd])) | ||||||
|         return output |         return output | ||||||
|  |  | ||||||
|     def get_config(self, config, getall=False): |     def get_config(self, config, getall=False): | ||||||
| @@ -91,17 +155,17 @@ class GitHgRepo: | |||||||
|         return self.run_cmd(['config', get[getall] , config], stderr=None) |         return self.run_cmd(['config', get[getall] , config], stderr=None) | ||||||
|  |  | ||||||
|     def get_config_bool(self, config, default=False): |     def get_config_bool(self, config, default=False): | ||||||
|         value = self.get_config(config).rstrip('\n') |         value = self.get_config(config).rstrip() | ||||||
|         if value == "true": |         if value == b"true": | ||||||
|             return True |             return True | ||||||
|         elif value == "false": |         elif value == b"false": | ||||||
|             return False |             return False | ||||||
|         else: |         else: | ||||||
|             return default |             return default | ||||||
|  |  | ||||||
|     def get_hg_repo_url(self, remote): |     def get_hg_repo_url(self, remote): | ||||||
|         url = self.get_config('remote.%s.url' % (remote)) |         url = self.get_config(b'remote.%s.url' % (remote)) | ||||||
|         if url and url[0:4] == 'hg::': |         if url and url[0:4] == b'hg::': | ||||||
|             url = url[4:].strip() |             url = url[4:].strip() | ||||||
|         else: |         else: | ||||||
|             url = None |             url = None | ||||||
| @@ -129,9 +193,9 @@ class GitHgRepo: | |||||||
|         for r in self.get_hg_repos(): |         for r in self.get_hg_repos(): | ||||||
|             try: |             try: | ||||||
|                 hgpath = remotehg.select_marks_dir(r, self.gitdir, False) |                 hgpath = remotehg.select_marks_dir(r, self.gitdir, False) | ||||||
|                 m = remotehg.Marks(os.path.join(hgpath, 'marks-hg'), None) |                 m = remotehg.Marks(os.path.join(hgpath, b'marks-hg'), None) | ||||||
|                 mark = m.from_rev(rev) |                 mark = m.from_rev(rev) | ||||||
|                 m = GitMarks(os.path.join(hgpath, 'marks-git')) |                 m = GitMarks(os.path.join(hgpath, b'marks-git')) | ||||||
|                 return m.to_rev(mark) |                 return m.to_rev(mark) | ||||||
|             except: |             except: | ||||||
|                 pass |                 pass | ||||||
| @@ -143,28 +207,28 @@ class GitHgRepo: | |||||||
|             return self.hg_repos |             return self.hg_repos | ||||||
|  |  | ||||||
|         # check any local hg repo to see if rev is in there |         # check any local hg repo to see if rev is in there | ||||||
|         shared_path = os.path.join(self.gitdir, 'hg') |         shared_path = os.path.join(self.gitdir, b'hg') | ||||||
|         hg_path = os.path.join(shared_path, '.hg') |         hg_path = os.path.join(shared_path, b'.hg') | ||||||
|         if os.path.exists(shared_path): |         if os.path.exists(shared_path): | ||||||
|             repos = os.listdir(shared_path) |             repos = os.listdir(shared_path) | ||||||
|             for r in repos: |             for r in repos: | ||||||
|                 # skip the shared repo |                 # skip the shared repo | ||||||
|                 if r == '.hg': |                 if r == b'.hg': | ||||||
|                     continue |                     continue | ||||||
|                 # only dirs |                 # only dirs | ||||||
|                 if not os.path.isdir(os.path.join(shared_path, r)): |                 if not os.path.isdir(os.path.join(shared_path, r)): | ||||||
|                     continue |                     continue | ||||||
|                 local_path = os.path.join(shared_path, r, 'clone') |                 local_path = os.path.join(shared_path, r, b'clone') | ||||||
|                 local_hg = os.path.join(local_path, '.hg') |                 local_hg = os.path.join(local_path, b'.hg') | ||||||
|                 if not os.path.exists(local_hg): |                 if not os.path.exists(local_hg): | ||||||
|                     # could be a local repo without proxy, fetch url |                     # could be a local repo without proxy, fetch url | ||||||
|                     local_path = self.get_hg_repo_url(r) |                     local_path = self.get_hg_repo_url(r) | ||||||
|                     if not local_path: |                     if not local_path: | ||||||
|                         warn('failed to find local hg for remote %s', r) |                         warn(b'failed to find local hg for remote %s' % (r)) | ||||||
|                         continue |                         continue | ||||||
|                 else: |                 else: | ||||||
|                     # make sure the shared path is always up-to-date |                     # make sure the shared path is always up-to-date | ||||||
|                     util.writefile(os.path.join(local_hg, 'sharedpath'), |                     util.writefile(os.path.join(local_hg, b'sharedpath'), | ||||||
|                         os.path.abspath(hg_path)) |                         os.path.abspath(hg_path)) | ||||||
|                 self.hg_repos[r] = os.path.join(local_path) |                 self.hg_repos[r] = os.path.join(local_path) | ||||||
|  |  | ||||||
| @@ -176,9 +240,9 @@ class GitHgRepo: | |||||||
|         repos = self.get_hg_repos() |         repos = self.get_hg_repos() | ||||||
|         if r in repos: |         if r in repos: | ||||||
|             local_path = repos[r] |             local_path = repos[r] | ||||||
|             hushui = ui.ui() |             hushui = ui.ui.load() if hasattr(ui.ui, 'load') else ui.ui() | ||||||
|             hushui.setconfig('ui', 'interactive', 'off') |             hushui.setconfig(b'ui', b'interactive', b'off') | ||||||
|             hushui.fout = open(os.devnull, 'w') |             hushui.fout = open(os.devnull, 'wb') | ||||||
|             return hg.repository(hushui, local_path) |             return hg.repository(hushui, local_path) | ||||||
|  |  | ||||||
|     def find_hg_repo(self, rev): |     def find_hg_repo(self, rev): | ||||||
| @@ -201,7 +265,7 @@ class GitHgRepo: | |||||||
|         def __init__(self, repo, files): |         def __init__(self, repo, files): | ||||||
|             p1, p2, data = repo[None], '0' * 40, '' |             p1, p2, data = repo[None], '0' * 40, '' | ||||||
|             context.memctx.__init__(self, repo, (p1, p2), |             context.memctx.__init__(self, repo, (p1, p2), | ||||||
|                 data, files.keys(), self.getfilectx) |                 data, list(files.keys()), self.getfilectx) | ||||||
|             self.files = files |             self.files = files | ||||||
|             self.remotehg = import_sibling('remotehg', 'git-remote-hg') |             self.remotehg = import_sibling('remotehg', 'git-remote-hg') | ||||||
|             self.remotehg.hg_version = hg_version |             self.remotehg.hg_version = hg_version | ||||||
| @@ -215,13 +279,13 @@ class GitHgRepo: | |||||||
|                 is_link, is_exec, rename) |                 is_link, is_exec, rename) | ||||||
|  |  | ||||||
|     def read(self, relpath, rev=None): |     def read(self, relpath, rev=None): | ||||||
|         rev = rev if rev else ':0' |         rev = rev if rev else b':0' | ||||||
|         obj = '%s:%s' % (rev, relpath) |         obj = b'%s:%s' % (rev, relpath) | ||||||
|         # might complain bitterly to stderr if no subrepos so let's not show that |         # might complain bitterly to stderr if no subrepos so let's not show that | ||||||
|         return self.run_cmd(['show', obj]) |         return self.run_cmd(['show', obj]) | ||||||
|  |  | ||||||
|     # see also subrepo.state |     # see also subrepo.state | ||||||
|     def state(self, remote='origin', rev=None): |     def state(self, remote=b'origin', rev=None): | ||||||
|         """return a state dict, mapping subrepo paths configured in .hgsub |         """return a state dict, mapping subrepo paths configured in .hgsub | ||||||
|         to tuple: (source from .hgsub, revision from .hgsubstate, kind |         to tuple: (source from .hgsub, revision from .hgsubstate, kind | ||||||
|         (key in types dict)) |         (key in types dict)) | ||||||
| @@ -229,7 +293,7 @@ class GitHgRepo: | |||||||
|  |  | ||||||
|         # obtain relevant files' content from specified revision |         # obtain relevant files' content from specified revision | ||||||
|         files = { } |         files = { } | ||||||
|         for f in ('.hgsub', '.hgsubstate'): |         for f in (b'.hgsub', b'.hgsubstate'): | ||||||
|             files[f] = self.read(f) |             files[f] = self.read(f) | ||||||
|         log('state files for %s in revision %s:\n%s', remote, rev, files) |         log('state files for %s in revision %s:\n%s', remote, rev, files) | ||||||
|  |  | ||||||
| @@ -237,7 +301,7 @@ class GitHgRepo: | |||||||
|         # (rather than duplicating the admittedly simple parsing here) |         # (rather than duplicating the admittedly simple parsing here) | ||||||
|         repo = self.get_hg_repo(remote) |         repo = self.get_hg_repo(remote) | ||||||
|         if not repo: |         if not repo: | ||||||
|             die('no hg repo for alias %s' % remote) |             die(b'no hg repo for alias %s' % remote) | ||||||
|         ctx = self.dictmemctx(repo, files) |         ctx = self.dictmemctx(repo, files) | ||||||
|         # helpers moved around 4.6 |         # helpers moved around 4.6 | ||||||
|         if hasattr(subrepo, 'state'): |         if hasattr(subrepo, 'state'): | ||||||
| @@ -252,21 +316,21 @@ class GitHgRepo: | |||||||
|         resolved = {} |         resolved = {} | ||||||
|         for s in state: |         for s in state: | ||||||
|             src, rev, kind = state[s] |             src, rev, kind = state[s] | ||||||
|             if not kind in ('hg', 'git'): |             if not kind in (b'hg', b'git'): | ||||||
|                 warn('skipping unsupported subrepo type %s' % kind) |                 warn('skipping unsupported subrepo type %s' % kind) | ||||||
|                 continue |                 continue | ||||||
|             if not util.url(src).isabs(): |             if not urlutil.url(src).isabs(): | ||||||
|                 parent = self.get_hg_repo_url(remote) |                 parent = self.get_hg_repo_url(remote) | ||||||
|                 if not parent: |                 if not parent: | ||||||
|                     die('could not determine repo url of %s' % remote) |                     die(b'could not determine repo url of %s' % remote) | ||||||
|                 parent = util.url(parent) |                 parent = urlutil.url(parent) | ||||||
|                 parent.path = posixpath.join(parent.path or '', src) |                 parent.path = posixpath.join(parent.path or b'', src) | ||||||
|                 parent.path = posixpath.normpath(parent.path) |                 parent.path = posixpath.normpath(parent.path) | ||||||
|                 src = str(parent) |                 src = bytes(parent) | ||||||
|             # translate to git view url |             # translate to git view url | ||||||
|             if kind == 'hg': |             if kind == b'hg': | ||||||
|                 src = 'hg::' + src |                 src = b'hg::' + src | ||||||
|             resolved[s] = (src.strip(), rev or '', kind) |             resolved[s] = (src.strip(), rev or b'', kind) | ||||||
|         log('resolved state %s', resolved) |         log('resolved state %s', resolved) | ||||||
|         return resolved |         return resolved | ||||||
|  |  | ||||||
| @@ -277,12 +341,15 @@ class SubCommand: | |||||||
|         self.subcommand = subcmdname |         self.subcommand = subcmdname | ||||||
|         self.githgrepo = githgrepo |         self.githgrepo = githgrepo | ||||||
|         self.argparser = self.argumentparser() |         self.argparser = self.argumentparser() | ||||||
|  |         # list of str | ||||||
|  |         self.args = [] | ||||||
|  |  | ||||||
|     def argumentparser(self): |     def argumentparser(self): | ||||||
|         return argparse.ArgumentParser() |         return argparse.ArgumentParser() | ||||||
|  |  | ||||||
|     def get_remote(self, args): |     def get_remote(self, args): | ||||||
|         if len(args): |         if len(args): | ||||||
|  |             assert isinstance(args[0], bytes) | ||||||
|             return (args[0], args[1:]) |             return (args[0], args[1:]) | ||||||
|         else: |         else: | ||||||
|             self.usage('missing argument: <remote-alias>') |             self.usage('missing argument: <remote-alias>') | ||||||
| @@ -295,7 +362,7 @@ class SubCommand: | |||||||
|  |  | ||||||
|     def execute(self, args): |     def execute(self, args): | ||||||
|         (self.options, self.args) = self.argparser.parse_known_args(args) |         (self.options, self.args) = self.argparser.parse_known_args(args) | ||||||
|         self.do(self.options, self.args) |         self.do(self.options, [compat.decode_sysarg(a) for a in self.args]) | ||||||
|  |  | ||||||
|     def usage(self, msg): |     def usage(self, msg): | ||||||
|         if msg: |         if msg: | ||||||
| @@ -304,6 +371,7 @@ class SubCommand: | |||||||
|             self.argparser.print_usage(sys.stderr) |             self.argparser.print_usage(sys.stderr) | ||||||
|             sys.exit(2) |             sys.exit(2) | ||||||
|  |  | ||||||
|  |     # args: list of bytes | ||||||
|     def do(self, options, args): |     def do(self, options, args): | ||||||
|         pass |         pass | ||||||
|  |  | ||||||
| @@ -322,7 +390,7 @@ class HgRevCommand(SubCommand): | |||||||
|         if len(args): |         if len(args): | ||||||
|             hgrev = self.githgrepo.get_hg_rev(args[0]) |             hgrev = self.githgrepo.get_hg_rev(args[0]) | ||||||
|             if hgrev: |             if hgrev: | ||||||
|                 print hgrev |                 puts(hgrev) | ||||||
|  |  | ||||||
|  |  | ||||||
| class GitMarks: | class GitMarks: | ||||||
| @@ -340,24 +408,24 @@ class GitMarks: | |||||||
|         if not os.path.exists(self.path): |         if not os.path.exists(self.path): | ||||||
|             return |             return | ||||||
|  |  | ||||||
|         for l in file(self.path): |         for l in open(self.path, 'rb'): | ||||||
|             m, c = l.strip().split(' ', 2) |             m, c = l.strip().split(b' ', 2) | ||||||
|             m = int(m[1:]) |             m = int(m[1:]) | ||||||
|             self.marks[c] = m |             self.marks[c] = m | ||||||
|             self.rev_marks[m] = c |             self.rev_marks[m] = c | ||||||
|  |  | ||||||
|     def store(self): |     def store(self): | ||||||
|         marks = self.rev_marks.keys() |         marks = list(self.rev_marks.keys()) | ||||||
|         marks.sort() |         marks.sort() | ||||||
|         with open(self.path, 'w') as f: |         with open(self.path, 'wb') as f: | ||||||
|             for m in marks: |             for m in marks: | ||||||
|                 f.write(':%d %s\n' % (m, self.rev_marks[m])) |                 f.write(b':%d %s\n' % (m, self.rev_marks[m])) | ||||||
|  |  | ||||||
|     def from_rev(self, rev): |     def from_rev(self, rev): | ||||||
|         return self.marks[rev] |         return self.marks[rev] | ||||||
|  |  | ||||||
|     def to_rev(self, mark): |     def to_rev(self, mark): | ||||||
|         return str(self.rev_marks[mark]) |         return self.rev_marks[mark] | ||||||
|  |  | ||||||
|  |  | ||||||
| class GitRevCommand(SubCommand): | class GitRevCommand(SubCommand): | ||||||
| @@ -375,7 +443,7 @@ class GitRevCommand(SubCommand): | |||||||
|             rev = args[0] |             rev = args[0] | ||||||
|             gitcommit = self.githgrepo.get_git_commit(rev) |             gitcommit = self.githgrepo.get_git_commit(rev) | ||||||
|             if gitcommit: |             if gitcommit: | ||||||
|                 print gitcommit |                 puts(gitcommit) | ||||||
|  |  | ||||||
|  |  | ||||||
| class GcCommand(SubCommand): | class GcCommand(SubCommand): | ||||||
| @@ -408,7 +476,7 @@ class GcCommand(SubCommand): | |||||||
|  |  | ||||||
|     def print_commits(self, gm, dest): |     def print_commits(self, gm, dest): | ||||||
|         for c in gm.marks.keys(): |         for c in gm.marks.keys(): | ||||||
|             dest.write(c + '\n') |             dest.write(c + b'\n') | ||||||
|         dest.flush() |         dest.flush() | ||||||
|         dest.close() |         dest.close() | ||||||
|  |  | ||||||
| @@ -422,27 +490,27 @@ class GcCommand(SubCommand): | |||||||
|             if not remote in hg_repos: |             if not remote in hg_repos: | ||||||
|                 self.usage('%s is not a valid hg remote' % (remote)) |                 self.usage('%s is not a valid hg remote' % (remote)) | ||||||
|             hgpath = remotehg.select_marks_dir(remote, self.githgrepo.gitdir, False) |             hgpath = remotehg.select_marks_dir(remote, self.githgrepo.gitdir, False) | ||||||
|             print "Loading hg marks ..." |             puts(b"Loading hg marks ...") | ||||||
|             hgm = remotehg.Marks(os.path.join(hgpath, 'marks-hg'), None) |             hgm = remotehg.Marks(os.path.join(hgpath, b'marks-hg'), None) | ||||||
|             print "Loading git marks ..." |             puts(b"Loading git marks ...") | ||||||
|             gm = GitMarks(os.path.join(hgpath, 'marks-git')) |             gm = GitMarks(os.path.join(hgpath, b'marks-git')) | ||||||
|             repo = hg.repository(ui.ui(), hg_repos[remote]) if options.check_hg else None |             repo = hg.repository(ui.ui(), hg_repos[remote]) if options.check_hg else None | ||||||
|             # git-gc may have dropped unreachable commits |             # git-gc may have dropped unreachable commits | ||||||
|             # (in particular due to multiple hg head cases) |             # (in particular due to multiple hg head cases) | ||||||
|             # need to drop those so git-fast-export or git-fast-import does not complain |             # need to drop those so git-fast-export or git-fast-import does not complain | ||||||
|             print "Performing garbage collection on git commits ..." |             puts(b"Performing garbage collection on git commits ...") | ||||||
|             process = self.githgrepo.start_cmd(['cat-file', '--batch-check'], \ |             process = self.githgrepo.start_cmd(['cat-file', '--batch-check'], \ | ||||||
|                 stdin=subprocess.PIPE) |                 stdin=subprocess.PIPE) | ||||||
|             thread = threading.Thread(target=self.print_commits, args=(gm, process.stdin)) |             thread = threading.Thread(target=self.print_commits, args=(gm, process.stdin)) | ||||||
|             thread.start() |             thread.start() | ||||||
|             git_marks = set({}) |             git_marks = set({}) | ||||||
|             for l in process.stdout: |             for l in process.stdout: | ||||||
|                 sp = l.strip().split(' ', 2) |                 sp = l.strip().split(b' ', 2) | ||||||
|                 if sp[1] == 'commit': |                 if sp[1] == 'commit': | ||||||
|                     git_marks.add(gm.from_rev(sp[0])) |                     git_marks.add(gm.from_rev(sp[0])) | ||||||
|             thread.join() |             thread.join() | ||||||
|             # reduce down to marks that are common to both |             # reduce down to marks that are common to both | ||||||
|             print "Computing marks intersection ..." |             puts(b"Computing marks intersection ...") | ||||||
|             common_marks = set(hgm.rev_marks.keys()).intersection(git_marks) |             common_marks = set(hgm.rev_marks.keys()).intersection(git_marks) | ||||||
|             hg_rev_marks = {} |             hg_rev_marks = {} | ||||||
|             git_rev_marks = {} |             git_rev_marks = {} | ||||||
| @@ -453,7 +521,7 @@ class GcCommand(SubCommand): | |||||||
|                 git_rev_marks[m] = gm.rev_marks[m] |                 git_rev_marks[m] = gm.rev_marks[m] | ||||||
|             # common marks will not not include any refs/notes/hg |             # common marks will not not include any refs/notes/hg | ||||||
|             # let's not discard those casually, though they are not vital |             # let's not discard those casually, though they are not vital | ||||||
|             print "Including notes commits ..." |             puts(b"Including notes commits ...") | ||||||
|             revlist = self.githgrepo.start_cmd(['rev-list', 'refs/notes/hg']) |             revlist = self.githgrepo.start_cmd(['rev-list', 'refs/notes/hg']) | ||||||
|             for l in revlist.stdout.readlines(): |             for l in revlist.stdout.readlines(): | ||||||
|                 c = l.strip() |                 c = l.strip() | ||||||
| @@ -465,24 +533,24 @@ class GcCommand(SubCommand): | |||||||
|                 git_rev_marks[hgm.last_note] = gm.rev_marks[hgm.last_note] |                 git_rev_marks[hgm.last_note] = gm.rev_marks[hgm.last_note] | ||||||
|             # some status report |             # some status report | ||||||
|             if len(hgm.rev_marks) != len(hg_rev_marks): |             if len(hgm.rev_marks) != len(hg_rev_marks): | ||||||
|                 print "Trimmed hg marks from #%d down to #%d" % (len(hgm.rev_marks), len(hg_rev_marks)) |                 puts(b"Trimmed hg marks from #%d down to #%d" % (len(hgm.rev_marks), len(hg_rev_marks))) | ||||||
|             if len(gm.rev_marks) != len(git_rev_marks): |             if len(gm.rev_marks) != len(git_rev_marks): | ||||||
|                 print "Trimmed git marks from #%d down to #%d" % (len(gm.rev_marks), len(git_rev_marks)) |                 puts(b"Trimmed git marks from #%d down to #%d" % (len(gm.rev_marks), len(git_rev_marks))) | ||||||
|             # marks-hg tips irrelevant nowadays |             # marks-hg tips irrelevant nowadays | ||||||
|             # now update and store |             # now update and store | ||||||
|             if not options.dry_run: |             if not options.dry_run: | ||||||
|                 # hg marks |                 # hg marks | ||||||
|                 print "Writing hg marks ..." |                 puts(b"Writing hg marks ...") | ||||||
|                 hgm.rev_marks = hg_rev_marks |                 hgm.rev_marks = hg_rev_marks | ||||||
|                 hgm.marks = {} |                 hgm.marks = {} | ||||||
|                 for mark, rev in hg_rev_marks.iteritems(): |                 for mark, rev in hg_rev_marks.items(): | ||||||
|                     hgm.marks[rev] = mark |                     hgm.marks[rev] = mark | ||||||
|                 hgm.store() |                 hgm.store() | ||||||
|                 # git marks |                 # git marks | ||||||
|                 print "Writing git marks ..." |                 puts(b"Writing git marks ...") | ||||||
|                 gm.rev_marks = git_rev_marks |                 gm.rev_marks = git_rev_marks | ||||||
|                 gm.marks = {} |                 gm.marks = {} | ||||||
|                 for mark, rev in git_rev_marks.iteritems(): |                 for mark, rev in git_rev_marks.items(): | ||||||
|                     gm.marks[rev] = mark |                     gm.marks[rev] = mark | ||||||
|                 gm.store() |                 gm.store() | ||||||
|  |  | ||||||
| @@ -491,9 +559,9 @@ class SubRepoCommand(SubCommand): | |||||||
|  |  | ||||||
|     def writestate(repo, state): |     def writestate(repo, state): | ||||||
|         """rewrite .hgsubstate in (outer) repo with these subrepo states""" |         """rewrite .hgsubstate in (outer) repo with these subrepo states""" | ||||||
|         lines = ['%s %s\n' % (state[s][1], s) for s in sorted(state) |         lines = [b'%s %s\n' % (state[s][1], s) for s in sorted(state) | ||||||
|                                                     if state[s][1] != nullstate[1]] |                                                     if state[s][1] != nullstate[1]] | ||||||
|         repo.wwrite('.hgsubstate', ''.join(lines), '') |         repo.wwrite(b'.hgsubstate', b''.join(lines), b'') | ||||||
|  |  | ||||||
|     def argumentparser(self): |     def argumentparser(self): | ||||||
|         #usage = '%%(prog)s %s [options] <remote>...' % (self.subcommand) |         #usage = '%%(prog)s %s [options] <remote>...' % (self.subcommand) | ||||||
| @@ -630,7 +698,8 @@ class SubRepoCommand(SubCommand): | |||||||
|         if args: |         if args: | ||||||
|             # all arguments are registered, so should not have leftover |             # all arguments are registered, so should not have leftover | ||||||
|             # could be that main arguments were given to subcommands |             # could be that main arguments were given to subcommands | ||||||
|             warn('unparsed arguments: %s' % ' '.join(args)) |             warn(b'unparsed arguments: %s' % b' '.join(args)) | ||||||
|  |         options.remote = compat.decode_sysarg(options.remote) | ||||||
|         log('running subcmd options %s, args %s', options, args) |         log('running subcmd options %s, args %s', options, args) | ||||||
|         # establish initial operation ctx |         # establish initial operation ctx | ||||||
|         ctx = self.subcontext(self.githgrepo) |         ctx = self.subcontext(self.githgrepo) | ||||||
| @@ -640,10 +709,10 @@ class SubRepoCommand(SubCommand): | |||||||
|         log('running %s with options %s in context %s', \ |         log('running %s with options %s in context %s', \ | ||||||
|             options.func, options, ctx) |             options.func, options, ctx) | ||||||
|         subrepos = ctx.repo.state(options.remote) |         subrepos = ctx.repo.state(options.remote) | ||||||
|         paths = subrepos.keys() |         paths = list(subrepos.keys()) | ||||||
|         selabspaths = None |         selabspaths = None | ||||||
|         if ctx.level == 0 and hasattr(options, 'paths') and options.paths: |         if ctx.level == 0 and hasattr(options, 'paths') and options.paths: | ||||||
|             selabspaths = [ os.path.abspath(p) for p in options.paths ] |             selabspaths = [ os.path.abspath(compat.decode_sysarg(p)) for p in options.paths ] | ||||||
|         log('level %s selected paths %s', ctx.level, selabspaths) |         log('level %s selected paths %s', ctx.level, selabspaths) | ||||||
|         for p in paths: |         for p in paths: | ||||||
|             # prep context |             # prep context | ||||||
| @@ -662,17 +731,17 @@ class SubRepoCommand(SubCommand): | |||||||
|             if not ctx.subrepo: |             if not ctx.subrepo: | ||||||
|                 ctx.subrepo = self.git_hg_repo_try(ctx.subpath) |                 ctx.subrepo = self.git_hg_repo_try(ctx.subpath) | ||||||
|             # prep recursion (only into git-hg subrepos) |             # prep recursion (only into git-hg subrepos) | ||||||
|             if ctx.subrepo and options.recursive and ctx.state[2] == 'hg': |             if ctx.subrepo and options.recursive and ctx.state[2] == b'hg': | ||||||
|                 newctx = self.subcontext(ctx.subrepo) |                 newctx = self.subcontext(ctx.subrepo) | ||||||
|                 newctx.level = ctx.level + 1 |                 newctx.level = ctx.level + 1 | ||||||
|                 self.do_operation(options, args, newctx) |                 self.do_operation(options, args, newctx) | ||||||
|  |  | ||||||
|     def get_git_commit(self, ctx): |     def get_git_commit(self, ctx): | ||||||
|         src, rev, kind = ctx.state |         src, rev, kind = ctx.state | ||||||
|         if kind == 'hg': |         if kind == b'hg': | ||||||
|             gitcommit = ctx.subrepo.get_git_commit(rev) |             gitcommit = ctx.subrepo.get_git_commit(rev) | ||||||
|             if not gitcommit: |             if not gitcommit: | ||||||
|                 die('could not determine git commit for %s; a fetch may update notes' % rev) |                 die(b'could not determine git commit for %s; a fetch may update notes' % rev) | ||||||
|         else: |         else: | ||||||
|             gitcommit = rev |             gitcommit = rev | ||||||
|         return gitcommit |         return gitcommit | ||||||
| @@ -681,28 +750,28 @@ class SubRepoCommand(SubCommand): | |||||||
|         if not ctx.subrepo: |         if not ctx.subrepo: | ||||||
|             return |             return | ||||||
|         src, orig, kind = ctx.state |         src, orig, kind = ctx.state | ||||||
|         gitcommit = ctx.subrepo.rev_parse('HEAD') |         gitcommit = ctx.subrepo.rev_parse(b'HEAD') | ||||||
|         if not gitcommit: |         if not gitcommit: | ||||||
|             die('could not determine current HEAD state in %s' % ctx.subrepo.topdir) |             die(b'could not determine current HEAD state in %s' % ctx.subrepo.topdir) | ||||||
|         rev = gitcommit |         rev = gitcommit | ||||||
|         if kind == 'hg': |         if kind == b'hg': | ||||||
|             rev = ctx.subrepo.get_hg_rev(gitcommit) |             rev = ctx.subrepo.get_hg_rev(gitcommit) | ||||||
|             if not rev: |             if not rev: | ||||||
|                 die('could not determine hg changeset for commit %s' % gitcommit) |                 die(b'could not determine hg changeset for commit %s' % gitcommit) | ||||||
|         else: |         else: | ||||||
|             rev = gitcommit |             rev = gitcommit | ||||||
|         # obtain state from index |         # obtain state from index | ||||||
|         state_path = os.path.join(ctx.repo.topdir, '.hgsubstate') |         state_path = os.path.join(ctx.repo.topdir, b'.hgsubstate') | ||||||
|         # should have this, since we have subrepo (state) in the first place ... |         # should have this, since we have subrepo (state) in the first place ... | ||||||
|         if not os.path.exists(state_path): |         if not os.path.exists(state_path): | ||||||
|             die('no .hgsubstate found in repo %s' % ctx.repo.topdir) |             die(b'no .hgsubstate found in repo %s' % ctx.repo.topdir) | ||||||
|         if orig != rev: |         if orig != rev: | ||||||
|             short = ctx.subrepo.rev_parse(['--short', gitcommit]) |             short = ctx.subrepo.rev_parse(['--short', gitcommit]) | ||||||
|             print "Updating %s to %s [git %s]" % (ctx.subpath, rev, short) |             puts(b"Updating %s to %s [git %s]" % (ctx.subpath, rev, short)) | ||||||
|             # replace and update index |             # replace and update index | ||||||
|             with open(state_path, 'r') as f: |             with open(state_path, 'rb') as f: | ||||||
|                 state = f.read() |                 state = f.read() | ||||||
|             state = re.sub('.{40} %s' % (ctx.relpath), '%s %s' % (rev, ctx.relpath), state) |             state = re.sub(b'.{40} %s' % (ctx.relpath), b'%s %s' % (rev, ctx.relpath), state) | ||||||
|             with open(state_path, 'wb') as f: |             with open(state_path, 'wb') as f: | ||||||
|                 f.write(state) |                 f.write(state) | ||||||
|  |  | ||||||
| @@ -710,7 +779,7 @@ class SubRepoCommand(SubCommand): | |||||||
|         if not ctx.subrepo: |         if not ctx.subrepo: | ||||||
|             return |             return | ||||||
|         if not options.quiet: |         if not options.quiet: | ||||||
|             print 'Entering %s' % ctx.subpath |             puts(b'Entering %s' % ctx.subpath) | ||||||
|             sys.stdout.flush() |             sys.stdout.flush() | ||||||
|         newenv = os.environ.copy() |         newenv = os.environ.copy() | ||||||
|         newenv['path'] = ctx.relpath |         newenv['path'] = ctx.relpath | ||||||
| @@ -721,7 +790,7 @@ class SubRepoCommand(SubCommand): | |||||||
|         proc = subprocess.Popen(options.command, shell=True, cwd=ctx.subpath, env=newenv) |         proc = subprocess.Popen(options.command, shell=True, cwd=ctx.subpath, env=newenv) | ||||||
|         proc.wait() |         proc.wait() | ||||||
|         if proc.returncode != 0: |         if proc.returncode != 0: | ||||||
|             die('stopping at %s; script returned non-zero status' % ctx.subpath) |             die(b'stopping at %s; script returned non-zero status' % ctx.subpath) | ||||||
|  |  | ||||||
|     def cmd_update(self, options, args, ctx): |     def cmd_update(self, options, args, ctx): | ||||||
|         if not ctx.subrepo: |         if not ctx.subrepo: | ||||||
| @@ -729,7 +798,7 @@ class SubRepoCommand(SubCommand): | |||||||
|             self.run_cmd(options, ctx.repo, ['clone', src, ctx.subpath], cwd=None) |             self.run_cmd(options, ctx.repo, ['clone', src, ctx.subpath], cwd=None) | ||||||
|             ctx.subrepo = self.git_hg_repo_try(ctx.subpath) |             ctx.subrepo = self.git_hg_repo_try(ctx.subpath) | ||||||
|             if not ctx.subrepo: |             if not ctx.subrepo: | ||||||
|                 die('subrepo %s setup clone failed', ctx.subpath) |                 die(b'subrepo %s setup clone failed' % ctx.subpath) | ||||||
|             # force (detached) checkout of target commit following clone |             # force (detached) checkout of target commit following clone | ||||||
|             cmd = [ 'checkout', '-q' ] |             cmd = [ 'checkout', '-q' ] | ||||||
|         else: |         else: | ||||||
| @@ -757,32 +826,32 @@ class SubRepoCommand(SubCommand): | |||||||
|  |  | ||||||
|     def cmd_status(self, options, args, ctx): |     def cmd_status(self, options, args, ctx): | ||||||
|         if not ctx.subrepo: |         if not ctx.subrepo: | ||||||
|             state = '-' |             state = b'-' | ||||||
|             revname = '' |             revname = b'' | ||||||
|             _, gitcommit, kind = ctx.state |             _, gitcommit, kind = ctx.state | ||||||
|             if kind != 'git': |             if kind != b'git': | ||||||
|                 gitcommit += '[hg] ' |                 gitcommit += b'[hg] ' | ||||||
|         else: |         else: | ||||||
|             gitcommit = self.get_git_commit(ctx) |             gitcommit = self.get_git_commit(ctx) | ||||||
|             head = ctx.subrepo.rev_parse('HEAD') |             head = ctx.subrepo.rev_parse(b'HEAD') | ||||||
|             if head == gitcommit: |             if head == gitcommit: | ||||||
|                 state = ' ' |                 state = b' ' | ||||||
|             else: |             else: | ||||||
|                 state = '+' |                 state = b'+' | ||||||
|                 # option determines what to print |                 # option determines what to print | ||||||
|                 if not options.cached: |                 if not options.cached: | ||||||
|                     gitcommit = head |                     gitcommit = head | ||||||
|             revname = ctx.subrepo.rev_describe(gitcommit) |             revname = ctx.subrepo.rev_describe(gitcommit) | ||||||
|             if revname: |             if revname: | ||||||
|                 revname = ' (%s)' % revname |                 revname = b' (%s)' % revname | ||||||
|         print "%s%s %s%s" % (state, gitcommit, ctx.subpath, revname) |         puts(b"%s%s %s%s" % (state, gitcommit, ctx.subpath, revname)) | ||||||
|  |  | ||||||
|     def cmd_sync(self, options, args, ctx): |     def cmd_sync(self, options, args, ctx): | ||||||
|         if not ctx.subrepo: |         if not ctx.subrepo: | ||||||
|             return |             return | ||||||
|         src, _, _ = ctx.state |         src, _, _ = ctx.state | ||||||
|         self.run_cmd(options, ctx.subrepo, \ |         self.run_cmd(options, ctx.subrepo, \ | ||||||
|             ['config', 'remote.%s.url' % (options.remote), src]) |             ['config', b'remote.%s.url' % (options.remote), src]) | ||||||
|  |  | ||||||
|  |  | ||||||
| class RepoCommand(SubCommand): | class RepoCommand(SubCommand): | ||||||
| @@ -801,7 +870,7 @@ class RepoCommand(SubCommand): | |||||||
|         (remote, args) = self.get_remote(args) |         (remote, args) = self.get_remote(args) | ||||||
|         repos = self.githgrepo.get_hg_repos() |         repos = self.githgrepo.get_hg_repos() | ||||||
|         if remote in repos: |         if remote in repos: | ||||||
|             print repos[remote].rstrip('/') |             puts(repos[remote].rstrip(b'/')) | ||||||
|  |  | ||||||
|  |  | ||||||
| class HgCommand(SubCommand): | class HgCommand(SubCommand): | ||||||
| @@ -821,8 +890,8 @@ class HgCommand(SubCommand): | |||||||
|         remote = self.subcommand |         remote = self.subcommand | ||||||
|         repos = self.githgrepo.get_hg_repos() |         repos = self.githgrepo.get_hg_repos() | ||||||
|         if len(args) and remote in repos: |         if len(args) and remote in repos: | ||||||
|             if args[0].find('hg') < 0: |             if args[0].find(b'hg') < 0: | ||||||
|                 args.insert(0, 'hg') |                 args.insert(0, b'hg') | ||||||
|             args[1:1] = ['-R', repos[remote]] |             args[1:1] = ['-R', repos[remote]] | ||||||
|             p = subprocess.Popen(args, stdout=None) |             p = subprocess.Popen(args, stdout=None) | ||||||
|             p.wait() |             p.wait() | ||||||
| @@ -847,12 +916,12 @@ class HelpCommand(SubCommand): | |||||||
|  |  | ||||||
| def get_subcommands(): | def get_subcommands(): | ||||||
|     commands = { |     commands = { | ||||||
|         'hg-rev': HgRevCommand, |         b'hg-rev': HgRevCommand, | ||||||
|         'git-rev': GitRevCommand, |         b'git-rev': GitRevCommand, | ||||||
|         'repo': RepoCommand, |         b'repo': RepoCommand, | ||||||
|         'gc':  GcCommand, |         b'gc':  GcCommand, | ||||||
|         'sub': SubRepoCommand, |         b'sub': SubRepoCommand, | ||||||
|         'help' : HelpCommand |         b'help' : HelpCommand | ||||||
|     } |     } | ||||||
|     # add remote named subcommands |     # add remote named subcommands | ||||||
|     repos = githgrepo.get_hg_repos() |     repos = githgrepo.get_hg_repos() | ||||||
| @@ -885,11 +954,12 @@ def do_usage(): | |||||||
|  |  | ||||||
|     Available hg remotes: |     Available hg remotes: | ||||||
|     """) |     """) | ||||||
|  |     usage = compat.to_b(usage) | ||||||
|     for r in githgrepo.get_hg_repos(): |     for r in githgrepo.get_hg_repos(): | ||||||
|         usage += '\t%s\n' % (r) |         usage += b'\t%s\n' % (r) | ||||||
|     usage += '\n' |     usage += b'\n' | ||||||
|     sys.stderr.write(usage) |     compat.stderr.write(usage) | ||||||
|     sys.stderr.flush() |     compat.stderr.flush() | ||||||
|     sys.exit(2) |     sys.exit(2) | ||||||
|  |  | ||||||
| def init_git(gitdir=None): | def init_git(gitdir=None): | ||||||
| @@ -897,7 +967,7 @@ def init_git(gitdir=None): | |||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         githgrepo = GitHgRepo(gitdir=gitdir) |         githgrepo = GitHgRepo(gitdir=gitdir) | ||||||
|     except Exception, e: |     except Exception as e: | ||||||
|         die(str(e)) |         die(str(e)) | ||||||
|  |  | ||||||
| def init_logger(): | def init_logger(): | ||||||
| @@ -916,8 +986,8 @@ def init_version(): | |||||||
|     global hg_version |     global hg_version | ||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         version, _, extra = util.version().partition('+') |         version, _, extra = util.version().partition(b'+') | ||||||
|         version = list(int(e) for e in version.split('.')) |         version = list(int(e) for e in version.split(b'.')) | ||||||
|         if extra: |         if extra: | ||||||
|             version[-1] += 1 |             version[-1] += 1 | ||||||
|         hg_version = tuple(version) |         hg_version = tuple(version) | ||||||
| @@ -933,19 +1003,21 @@ def main(argv): | |||||||
|     global subcommands |     global subcommands | ||||||
|  |  | ||||||
|     # as an alias, cwd is top dir, change again to original directory |     # as an alias, cwd is top dir, change again to original directory | ||||||
|     reldir = os.environ.get('GIT_PREFIX') |     reldir = compat.getenv(b'GIT_PREFIX', None) | ||||||
|     if reldir: |     if reldir: | ||||||
|         os.chdir(reldir) |         os.chdir(reldir) | ||||||
|  |  | ||||||
|     # init repo dir |     # init repo dir | ||||||
|     # we will take over dir management ... |     # we will take over dir management ... | ||||||
|     init_git(os.environ.pop('GIT_DIR', None)) |     gitdir = compat.getenv(b'GIT_DIR', None) | ||||||
|  |     os.environ.pop('GIT_DIR', None) | ||||||
|  |     init_git(gitdir) | ||||||
|  |  | ||||||
|     subcommands = get_subcommands() |     subcommands = get_subcommands() | ||||||
|  |  | ||||||
|     cmd = '' |     cmd = '' | ||||||
|     if len(argv) > 1: |     if len(argv) > 1: | ||||||
|         cmd = argv[1] |         cmd = compat.decode_sysarg(argv[1]) | ||||||
|     argv = argv[2:] |     argv = argv[2:] | ||||||
|     if cmd in subcommands: |     if cmd in subcommands: | ||||||
|         c = subcommands[cmd] |         c = subcommands[cmd] | ||||||
|   | |||||||
							
								
								
									
										971
									
								
								git-remote-hg
									
									
									
									
									
								
							
							
						
						
									
										971
									
								
								git-remote-hg
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										45
									
								
								setup.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								setup.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | |||||||
|  | # git-remote-hg setuptools script | ||||||
|  |  | ||||||
|  | import setuptools | ||||||
|  |  | ||||||
|  | # strip leading v | ||||||
|  | version = 'v1.0.4'[1:] | ||||||
|  |  | ||||||
|  | # check for released version | ||||||
|  | assert (len(version) > 0) | ||||||
|  | assert (version.find('-') < 0) | ||||||
|  |  | ||||||
|  | long_description = \ | ||||||
|  | """ | ||||||
|  | 'git-remote-hg' is a gitremote protocol helper for Mercurial. | ||||||
|  | It allows you to clone, fetch and push to and from Mercurial repositories as if | ||||||
|  | they were Git ones using a hg::some-url URL. | ||||||
|  |  | ||||||
|  | See the homepage for much more explanation. | ||||||
|  | """ | ||||||
|  |  | ||||||
|  | CLASSIFIERS = [ | ||||||
|  |     "Programming Language :: Python", | ||||||
|  |     "Programming Language :: Python :: 2", | ||||||
|  |     "Programming Language :: Python :: 2.7", | ||||||
|  |     "Programming Language :: Python :: 3", | ||||||
|  |     "Programming Language :: Python :: 3.6", | ||||||
|  |     "License :: OSI Approved", | ||||||
|  |     "License :: OSI Approved :: GNU General Public License v2 (GPLv2)", | ||||||
|  |     "Development Status :: 5 - Production/Stable", | ||||||
|  |     "Intended Audience :: Developers", | ||||||
|  | ] | ||||||
|  |  | ||||||
|  | setuptools.setup(name="git-remote-hg", | ||||||
|  |       version=version, | ||||||
|  |       author="Mark Nauwelaerts", | ||||||
|  |       author_email="mnauw@users.sourceforge.net", | ||||||
|  |       url="http://github.com/mnauw/git-remote-hg", | ||||||
|  |       description="access hg repositories as git remotes", | ||||||
|  |       long_description=long_description, | ||||||
|  |       license="GPLv2", | ||||||
|  |       keywords="git hg mercurial", | ||||||
|  |       scripts=["git-remote-hg", "git-hg-helper"], | ||||||
|  |       classifiers=CLASSIFIERS | ||||||
|  |      ) | ||||||
|  |  | ||||||
							
								
								
									
										46
									
								
								test/bidi.t
									
									
									
									
									
								
							
							
						
						
									
										46
									
								
								test/bidi.t
									
									
									
									
									
								
							| @@ -13,13 +13,7 @@ test -n "$TEST_DIRECTORY" || TEST_DIRECTORY=$(dirname $0)/ | |||||||
|  |  | ||||||
| if ! test_have_prereq PYTHON | if ! test_have_prereq PYTHON | ||||||
| then | then | ||||||
| 	skip_all='skipping remote-hg tests; python not available' | 	skip_all='skipping remote-hg tests; python with mercurial not available' | ||||||
| 	test_done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if ! python2 -c 'import mercurial' > /dev/null 2>&1 |  | ||||||
| then |  | ||||||
| 	skip_all='skipping remote-hg tests; mercurial not available' |  | ||||||
| 	test_done | 	test_done | ||||||
| fi | fi | ||||||
|  |  | ||||||
| @@ -243,4 +237,42 @@ test_expect_success 'hg tags' ' | |||||||
| 	test_cmp expected actual | 	test_cmp expected actual | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'test timezones' ' | ||||||
|  | 	test_when_finished "rm -rf gitrepo* hgrepo*" && | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	git init -q gitrepo && | ||||||
|  | 	cd gitrepo && | ||||||
|  |  | ||||||
|  | 	echo alpha > alpha && | ||||||
|  | 	git add alpha && | ||||||
|  | 	git commit -m "add alpha" --date="2007-01-01 00:00:00 +0000" && | ||||||
|  |  | ||||||
|  | 	echo beta > beta && | ||||||
|  | 	git add beta && | ||||||
|  | 	git commit -m "add beta" --date="2007-01-01 00:00:00 +0100" && | ||||||
|  |  | ||||||
|  | 	echo gamma > gamma && | ||||||
|  | 	git add gamma && | ||||||
|  | 	git commit -m "add gamma" --date="2007-01-01 00:00:00 -0100" && | ||||||
|  |  | ||||||
|  | 	echo delta > delta && | ||||||
|  | 	git add delta && | ||||||
|  | 	git commit -m "add delta" --date="2007-01-01 00:00:00 +0130" && | ||||||
|  |  | ||||||
|  | 	echo epsilon > epsilon && | ||||||
|  | 	git add epsilon && | ||||||
|  | 	git commit -m "add epsilon" --date="2007-01-01 00:00:00 -0130" | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	hg_clone gitrepo hgrepo && | ||||||
|  | 	git_clone hgrepo gitrepo2 && | ||||||
|  | 	hg_clone gitrepo2 hgrepo2 && | ||||||
|  |  | ||||||
|  | 	hg_log hgrepo > expected && | ||||||
|  | 	hg_log hgrepo2 > actual && | ||||||
|  |  | ||||||
|  | 	test_cmp expected actual | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_done | test_done | ||||||
|   | |||||||
| @@ -13,13 +13,7 @@ test -n "$TEST_DIRECTORY" || TEST_DIRECTORY=$(dirname $0)/ | |||||||
|  |  | ||||||
| if ! test_have_prereq PYTHON | if ! test_have_prereq PYTHON | ||||||
| then | then | ||||||
| 	skip_all='skipping remote-hg tests; python not available' | 	skip_all='skipping remote-hg tests; python with mercurial not available' | ||||||
| 	test_done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if ! python2 -c 'import mercurial' > /dev/null 2>&1 |  | ||||||
| then |  | ||||||
| 	skip_all='skipping remote-hg tests; mercurial not available' |  | ||||||
| 	test_done | 	test_done | ||||||
| fi | fi | ||||||
|  |  | ||||||
| @@ -521,8 +515,8 @@ test_expect_success 'subcommand sub status' ' | |||||||
|  |  | ||||||
| 	( | 	( | ||||||
| 	cd gitrepo && | 	cd gitrepo && | ||||||
| 	git-hg-helper sub update sub_hg_a --force && | 	git-hg-helper sub update --force sub_hg_a  && | ||||||
| 	git-hg-helper sub update sub_git --force && | 	git-hg-helper sub update --force sub_git && | ||||||
| 		( | 		( | ||||||
| 		# advance and add a tag to the git repo | 		# advance and add a tag to the git repo | ||||||
| 		cd sub_git && | 		cd sub_git && | ||||||
| @@ -544,4 +538,4 @@ test_expect_success 'subcommand sub status' ' | |||||||
| 	) | 	) | ||||||
| ' | ' | ||||||
|  |  | ||||||
| test_done | test_done | ||||||
|   | |||||||
| @@ -13,20 +13,14 @@ test -n "$TEST_DIRECTORY" || TEST_DIRECTORY=$(dirname $0)/ | |||||||
|  |  | ||||||
| if ! test_have_prereq PYTHON | if ! test_have_prereq PYTHON | ||||||
| then | then | ||||||
| 	skip_all='skipping remote-hg tests; python not available' | 	skip_all='skipping remote-hg tests; python with mercurial not available' | ||||||
| 	test_done | 	test_done | ||||||
| fi | fi | ||||||
|  |  | ||||||
| if ! python2 -c 'import mercurial' > /dev/null 2>&1 | if "$PYTHON" -c 'import hggit' > /dev/null 2>&1 | ||||||
| then |  | ||||||
| 	skip_all='skipping remote-hg tests; mercurial not available' |  | ||||||
| 	test_done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if python2 -c 'import hggit' > /dev/null 2>&1 |  | ||||||
| then | then | ||||||
| 	hggit=hggit | 	hggit=hggit | ||||||
| elif python2 -c 'import hgext.git' > /dev/null 2>&1 | elif "$PYTHON" -c 'import hgext.git' > /dev/null 2>&1 | ||||||
| then | then | ||||||
| 	hggit=hgext.git | 	hggit=hgext.git | ||||||
| else | else | ||||||
| @@ -34,15 +28,6 @@ else | |||||||
| 	test_done | 	test_done | ||||||
| fi | fi | ||||||
|  |  | ||||||
| hg_version=$(python2 -c 'from mercurial import util; print util.version()') |  | ||||||
|  |  | ||||||
| case $hg_version in |  | ||||||
| 3.0*+*) |  | ||||||
| 	skip_all='skipping remote-hg tests; unsuported version of hg by hg-git' |  | ||||||
| 	test_done |  | ||||||
| 	;; |  | ||||||
| esac |  | ||||||
|  |  | ||||||
| # clone to a git repo with git | # clone to a git repo with git | ||||||
| git_clone_git () { | git_clone_git () { | ||||||
| 	git clone -q "hg::$1" $2 && | 	git clone -q "hg::$1" $2 && | ||||||
| @@ -117,10 +102,13 @@ setup () { | |||||||
| 	[extensions] | 	[extensions] | ||||||
| 	$hggit = | 	$hggit = | ||||||
| 	graphlog = | 	graphlog = | ||||||
|  | 	[git] | ||||||
|  | 	debugextrainmessage = 1 | ||||||
| 	EOF | 	EOF | ||||||
| 	git config --global receive.denycurrentbranch warn | 	git config --global receive.denycurrentbranch warn | ||||||
| 	git config --global remote-hg.hg-git-compat true | 	git config --global remote-hg.hg-git-compat true | ||||||
| 	git config --global remote-hg.track-branches false | 	git config --global remote-hg.track-branches false | ||||||
|  | 	git config --global remote-hg.shared-marks false | ||||||
|  |  | ||||||
| 	HGEDITOR=true | 	HGEDITOR=true | ||||||
| 	HGMERGE=true | 	HGMERGE=true | ||||||
| @@ -132,6 +120,31 @@ setup () { | |||||||
|  |  | ||||||
| setup | setup | ||||||
|  |  | ||||||
|  | test_expect_success 'rename' ' | ||||||
|  | 	test_when_finished "rm -rf gitrepo* hgrepo*" && | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	hg init hgrepo1 && | ||||||
|  | 	cd hgrepo1 && | ||||||
|  | 	echo alpha > alpha && | ||||||
|  | 	hg add alpha && | ||||||
|  | 	hg commit -m "add alpha" && | ||||||
|  | 	hg mv alpha beta && | ||||||
|  | 	hg commit -m "rename alpha to beta" | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	for x in hg git | ||||||
|  | 	do | ||||||
|  | 		git_clone_$x hgrepo1 gitrepo-$x && | ||||||
|  | 		hg_clone_$x gitrepo-$x hgrepo2-$x && | ||||||
|  | 		hg_log hgrepo2-$x > "hg-log-$x" && | ||||||
|  | 		git_log gitrepo-$x > "git-log-$x" | ||||||
|  | 	done && | ||||||
|  |  | ||||||
|  | 	test_cmp hg-log-hg hg-log-git && | ||||||
|  | 	test_cmp git-log-hg git-log-git | ||||||
|  | ' | ||||||
|  |  | ||||||
| test_expect_success 'executable bit' ' | test_expect_success 'executable bit' ' | ||||||
| 	test_when_finished "rm -rf gitrepo* hgrepo*" && | 	test_when_finished "rm -rf gitrepo* hgrepo*" && | ||||||
|  |  | ||||||
|   | |||||||
| @@ -270,10 +270,10 @@ test_expect_success 'push with renamed executable preserves executable bit' ' | |||||||
| 	) && | 	) && | ||||||
|  |  | ||||||
| 	( | 	( | ||||||
|  | 	umask 0 && | ||||||
| 	cd hgrepo && | 	cd hgrepo && | ||||||
| 	hg update && | 	hg update && | ||||||
| 	stat content2 >expected && | 	stat content2 >expected && | ||||||
| 	# umask mileage might vary |  | ||||||
| 	grep -- -r.xr.xr.x expected | 	grep -- -r.xr.xr.x expected | ||||||
| 	) | 	) | ||||||
| ' | ' | ||||||
|   | |||||||
							
								
								
									
										163
									
								
								test/main.t
									
									
									
									
									
								
							
							
						
						
									
										163
									
								
								test/main.t
									
									
									
									
									
								
							| @@ -22,13 +22,7 @@ fi | |||||||
|  |  | ||||||
| if ! test_have_prereq PYTHON | if ! test_have_prereq PYTHON | ||||||
| then | then | ||||||
| 	skip_all='skipping remote-hg tests; python not available' | 	skip_all='skipping remote-hg tests; python with mercurial not available' | ||||||
| 	test_done |  | ||||||
| fi |  | ||||||
|  |  | ||||||
| if ! python2 -c 'import mercurial' > /dev/null 2>&1 |  | ||||||
| then |  | ||||||
| 	skip_all='skipping remote-hg tests; mercurial not available' |  | ||||||
| 	test_done | 	test_done | ||||||
| fi | fi | ||||||
|  |  | ||||||
| @@ -96,14 +90,14 @@ check_push () { | |||||||
| 		'') | 		'') | ||||||
| 			grep "^   [a-f0-9]*\.\.[a-f0-9]* *${branch} -> ${branch}$" error || ref_ret=1 | 			grep "^   [a-f0-9]*\.\.[a-f0-9]* *${branch} -> ${branch}$" error || ref_ret=1 | ||||||
| 			;; | 			;; | ||||||
|  | 		*) | ||||||
|  | 			echo "BUG: wrong kind '$kind'" && return 3 | ||||||
|  | 			;; | ||||||
| 		esac | 		esac | ||||||
| 		test $ref_ret -ne 0 && echo "match for '$branch' failed" && break | 		test $ref_ret -ne 0 && echo "match for '$branch' failed" && return 2 | ||||||
| 	done | 	done | ||||||
|  |  | ||||||
| 	if test $expected_ret -ne $ret || test $ref_ret -ne 0 | 	test $expected_ret -ne $ret && return 1 | ||||||
| 	then |  | ||||||
| 		return 1 |  | ||||||
| 	fi |  | ||||||
|  |  | ||||||
| 	return 0 | 	return 0 | ||||||
| } | } | ||||||
| @@ -686,11 +680,11 @@ test_expect_success 'remote big push' ' | |||||||
| 		check_branch hgrepo default one && | 		check_branch hgrepo default one && | ||||||
| 		check_branch hgrepo good_branch "good branch" && | 		check_branch hgrepo good_branch "good branch" && | ||||||
| 		check_branch hgrepo bad_branch "bad branch" && | 		check_branch hgrepo bad_branch "bad branch" && | ||||||
| 		check_branch hgrepo new_branch '' && | 		check_branch hgrepo new_branch && | ||||||
| 		check_bookmark hgrepo good_bmark one && | 		check_bookmark hgrepo good_bmark one && | ||||||
| 		check_bookmark hgrepo bad_bmark1 one && | 		check_bookmark hgrepo bad_bmark1 one && | ||||||
| 		check_bookmark hgrepo bad_bmark2 one && | 		check_bookmark hgrepo bad_bmark2 one && | ||||||
| 		check_bookmark hgrepo new_bmark '' | 		check_bookmark hgrepo new_bmark | ||||||
| 	fi | 	fi | ||||||
| ' | ' | ||||||
|  |  | ||||||
| @@ -755,7 +749,6 @@ test_expect_success 'remote big push non fast forward' ' | |||||||
| 		# so it will already have pushed some above previously | 		# so it will already have pushed some above previously | ||||||
| 		# (and master is a fake one that jumps around a bit) | 		# (and master is a fake one that jumps around a bit) | ||||||
| 		check_push 1 --all <<-\EOF | 		check_push 1 --all <<-\EOF | ||||||
| 		master:non-fast-forward |  | ||||||
| 		bad_bmark:non-fast-forward | 		bad_bmark:non-fast-forward | ||||||
| 		branches/bad_branch:non-fast-forward | 		branches/bad_branch:non-fast-forward | ||||||
| 		EOF | 		EOF | ||||||
| @@ -770,7 +763,7 @@ test_expect_success 'remote big push non fast forward' ' | |||||||
| 	) | 	) | ||||||
| ' | ' | ||||||
|  |  | ||||||
| test_expect_failure 'remote big push force' ' | test_expect_success 'remote big push force' ' | ||||||
| 	test_when_finished "rm -rf hgrepo gitrepo*" && | 	test_when_finished "rm -rf hgrepo gitrepo*" && | ||||||
|  |  | ||||||
| 	setup_big_push | 	setup_big_push | ||||||
| @@ -778,19 +771,32 @@ test_expect_failure 'remote big push force' ' | |||||||
| 	( | 	( | ||||||
| 	cd gitrepo && | 	cd gitrepo && | ||||||
|  |  | ||||||
| 	check_push 0 --force --all <<-\EOF | 	if test "$CAPABILITY_PUSH" = "t" | ||||||
| 	master | 	then | ||||||
| 	good_bmark | 		check_push 0 --force --all <<-\EOF | ||||||
| 	branches/good_branch | 		master:forced-update | ||||||
| 	new_bmark:new | 		good_bmark:forced-update | ||||||
| 	branches/new_branch:new | 		branches/good_branch:forced-update | ||||||
| 	bad_bmark1:forced-update | 		new_bmark:new | ||||||
| 	bad_bmark2:forced-update | 		branches/new_branch:new | ||||||
| 	branches/bad_branch:forced-update | 		bad_bmark1:forced-update | ||||||
| 	EOF | 		bad_bmark2:forced-update | ||||||
|  | 		branches/bad_branch:forced-update | ||||||
|  | 		EOF | ||||||
|  | 	else | ||||||
|  | 		check_push 0 --force --all <<-\EOF | ||||||
|  | 		master | ||||||
|  | 		good_bmark | ||||||
|  | 		branches/good_branch | ||||||
|  | 		new_bmark:new | ||||||
|  | 		branches/new_branch:new | ||||||
|  | 		bad_bmark1:forced-update | ||||||
|  | 		bad_bmark2:forced-update | ||||||
|  | 		branches/bad_branch:forced-update | ||||||
|  | 		EOF | ||||||
|  | 	fi | ||||||
| 	) && | 	) && | ||||||
|  |  | ||||||
| 	check_branch hgrepo default six && |  | ||||||
| 	check_branch hgrepo good_branch eight && | 	check_branch hgrepo good_branch eight && | ||||||
| 	check_branch hgrepo bad_branch nine && | 	check_branch hgrepo bad_branch nine && | ||||||
| 	check_branch hgrepo new_branch ten && | 	check_branch hgrepo new_branch ten && | ||||||
| @@ -831,11 +837,55 @@ test_expect_success 'remote big push dry-run' ' | |||||||
| 	check_branch hgrepo default one && | 	check_branch hgrepo default one && | ||||||
| 	check_branch hgrepo good_branch "good branch" && | 	check_branch hgrepo good_branch "good branch" && | ||||||
| 	check_branch hgrepo bad_branch "bad branch" && | 	check_branch hgrepo bad_branch "bad branch" && | ||||||
| 	check_branch hgrepo new_branch '' && | 	check_branch hgrepo new_branch && | ||||||
| 	check_bookmark hgrepo good_bmark one && | 	check_bookmark hgrepo good_bmark one && | ||||||
| 	check_bookmark hgrepo bad_bmark1 one && | 	check_bookmark hgrepo bad_bmark1 one && | ||||||
| 	check_bookmark hgrepo bad_bmark2 one && | 	check_bookmark hgrepo bad_bmark2 one && | ||||||
| 	check_bookmark hgrepo new_bmark '' | 	check_bookmark hgrepo new_bmark | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'remote big push force dry-run' ' | ||||||
|  | 	test_when_finished "rm -rf hgrepo gitrepo*" && | ||||||
|  |  | ||||||
|  | 	setup_big_push | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	cd gitrepo && | ||||||
|  |  | ||||||
|  | 	if test "$CAPABILITY_PUSH" = "t" | ||||||
|  | 	then | ||||||
|  | 		check_push 0 --force --dry-run --all <<-\EOF | ||||||
|  | 		master:forced-update | ||||||
|  | 		good_bmark:forced-update | ||||||
|  | 		branches/good_branch:forced-update | ||||||
|  | 		new_bmark:new | ||||||
|  | 		branches/new_branch:new | ||||||
|  | 		bad_bmark1:forced-update | ||||||
|  | 		bad_bmark2:forced-update | ||||||
|  | 		branches/bad_branch:forced-update | ||||||
|  | 		EOF | ||||||
|  | 	else | ||||||
|  | 		check_push 0 --force --dry-run --all <<-\EOF | ||||||
|  | 		master | ||||||
|  | 		good_bmark | ||||||
|  | 		branches/good_branch | ||||||
|  | 		new_bmark:new | ||||||
|  | 		branches/new_branch:new | ||||||
|  | 		bad_bmark1:forced-update | ||||||
|  | 		bad_bmark2:forced-update | ||||||
|  | 		branches/bad_branch:forced-update | ||||||
|  | 		EOF | ||||||
|  | 	fi | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	check_branch hgrepo default one && | ||||||
|  | 	check_branch hgrepo good_branch "good branch" && | ||||||
|  | 	check_branch hgrepo bad_branch "bad branch" && | ||||||
|  | 	check_branch hgrepo new_branch && | ||||||
|  | 	check_bookmark hgrepo good_bmark one && | ||||||
|  | 	check_bookmark hgrepo bad_bmark1 one && | ||||||
|  | 	check_bookmark hgrepo bad_bmark2 one && | ||||||
|  | 	check_bookmark hgrepo new_bmark | ||||||
| ' | ' | ||||||
|  |  | ||||||
| test_expect_success 'remote double failed push' ' | test_expect_success 'remote double failed push' ' | ||||||
| @@ -995,7 +1045,7 @@ testpushupdatesnotes=' | |||||||
| 	( | 	( | ||||||
| 	cd gitrepo && | 	cd gitrepo && | ||||||
| 	echo two > content && | 	echo two > content && | ||||||
| 	git commit -a -m two | 	git commit -a -m two && | ||||||
| 	git push | 	git push | ||||||
| 	) && | 	) && | ||||||
|  |  | ||||||
| @@ -1081,7 +1131,7 @@ test_expect_success 'push merged named branch' ' | |||||||
| 	git push | 	git push | ||||||
| 	) && | 	) && | ||||||
|  |  | ||||||
| 	cat > expected <<-EOF | 	cat > expected <<-EOF && | ||||||
| 	Merge | 	Merge | ||||||
| 	three | 	three | ||||||
| 	two | 	two | ||||||
| @@ -1122,7 +1172,7 @@ test_expect_success 'push tag different branch' ' | |||||||
| 	cd hgrepo && | 	cd hgrepo && | ||||||
| 	echo one > content && | 	echo one > content && | ||||||
| 	hg add content && | 	hg add content && | ||||||
| 	hg commit -m one | 	hg commit -m one && | ||||||
| 	hg branch feature && | 	hg branch feature && | ||||||
| 	echo two > content && | 	echo two > content && | ||||||
| 	hg commit -m two | 	hg commit -m two | ||||||
| @@ -1229,6 +1279,55 @@ test_expect_success 'clone can ignore invalid refnames' ' | |||||||
| 	check_files gitrepo "test.txt" | 	check_files gitrepo "test.txt" | ||||||
| ' | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'push annotated tag' ' | ||||||
|  | 	test_when_finished "rm -rf hgrepo gitrepo" && | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	hg init hgrepo && | ||||||
|  | 	cd hgrepo && | ||||||
|  | 	echo one > content && | ||||||
|  | 	hg add content && | ||||||
|  | 	hg commit -m one | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	git clone "hg::hgrepo" gitrepo && | ||||||
|  | 	cd gitrepo && | ||||||
|  | 	git tag -m "Version 1.0" v1.0 && | ||||||
|  | 	git push --tags | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	cat > expected <<-\EOF && | ||||||
|  | 	tip:Version 1.0:C O Mitter <committer@example.com> | ||||||
|  | 	v1.0:one:H G Wells <wells@example.com> | ||||||
|  | 	EOF | ||||||
|  |  | ||||||
|  | 	hg -R hgrepo log --template "{tags}:{desc}:{author}\n" > actual && | ||||||
|  |  | ||||||
|  | 	test_cmp expected actual | ||||||
|  | ' | ||||||
|  |  | ||||||
|  | test_expect_success 'timezone issues with negative offsets' ' | ||||||
|  | 	test_when_finished "rm -rf hgrepo gitrepo1 gitrepo2" && | ||||||
|  |  | ||||||
|  | 	hg init hgrepo && | ||||||
|  |  | ||||||
|  | 	( | ||||||
|  | 	git clone "hg::hgrepo" gitrepo1 && | ||||||
|  | 	cd gitrepo1 && | ||||||
|  | 	echo two >> content && | ||||||
|  | 	git add content && | ||||||
|  | 	git commit -m two --date="2016-09-26 00:00:00 -0230" && | ||||||
|  | 	git push | ||||||
|  | 	) && | ||||||
|  |  | ||||||
|  | 	git clone "hg::hgrepo" gitrepo2 && | ||||||
|  |  | ||||||
|  | 	git --git-dir=gitrepo1/.git log -1 --format="%ai" > expected && | ||||||
|  | 	git --git-dir=gitrepo2/.git log -1 --format="%ai" > actual && | ||||||
|  | 	test_cmp expected actual | ||||||
|  | ' | ||||||
|  |  | ||||||
| if test "$CAPABILITY_PUSH" != "t" | if test "$CAPABILITY_PUSH" != "t" | ||||||
| then | then | ||||||
| test_done | test_done | ||||||
|   | |||||||
							
								
								
									
										396
									
								
								test/sharness.sh
									
									
									
									
									
								
							
							
						
						
									
										396
									
								
								test/sharness.sh
									
									
									
									
									
								
							| @@ -18,33 +18,80 @@ | |||||||
| # along with this program.  If not, see http://www.gnu.org/licenses/ . | # along with this program.  If not, see http://www.gnu.org/licenses/ . | ||||||
|  |  | ||||||
| # Public: Current version of Sharness. | # Public: Current version of Sharness. | ||||||
| SHARNESS_VERSION="0.3.0" | SHARNESS_VERSION="1.1.0" | ||||||
| export SHARNESS_VERSION | export SHARNESS_VERSION | ||||||
|  |  | ||||||
| # Public: The file extension for tests.  By default, it is set to "t". | # Public: The file extension for tests.  By default, it is set to "t". | ||||||
| : ${SHARNESS_TEST_EXTENSION:=t} | : "${SHARNESS_TEST_EXTENSION:=t}" | ||||||
| export SHARNESS_TEST_EXTENSION | export SHARNESS_TEST_EXTENSION | ||||||
|  |  | ||||||
| # Keep the original TERM for say_color | # Public: Root directory containing tests. Tests can override this variable, | ||||||
| ORIGINAL_TERM=$TERM | # e.g. for testing Sharness itself. | ||||||
|  | if test -z "$SHARNESS_TEST_DIRECTORY" | ||||||
|  | then | ||||||
|  | 	SHARNESS_TEST_DIRECTORY=$(pwd) | ||||||
|  | else | ||||||
|  | 	# ensure that SHARNESS_TEST_DIRECTORY is an absolute path so that it | ||||||
|  | 	# is valid even if the current working directory is changed | ||||||
|  | 	SHARNESS_TEST_DIRECTORY=$(cd "$SHARNESS_TEST_DIRECTORY" && pwd) || exit 1 | ||||||
|  | fi | ||||||
|  | export SHARNESS_TEST_DIRECTORY | ||||||
|  |  | ||||||
|  | if test -z "$SHARNESS_TEST_OUTPUT_DIRECTORY" | ||||||
|  | then | ||||||
|  | 	# Similarly, override this to store the test-results subdir | ||||||
|  | 	# elsewhere | ||||||
|  | 	SHARNESS_TEST_OUTPUT_DIRECTORY=$SHARNESS_TEST_DIRECTORY | ||||||
|  | fi | ||||||
|  |  | ||||||
|  | #  Reset TERM to original terminal if found, otherwise save original TERM | ||||||
|  | [ "x" = "x$SHARNESS_ORIG_TERM" ] && | ||||||
|  | 		SHARNESS_ORIG_TERM="$TERM" || | ||||||
|  | 		TERM="$SHARNESS_ORIG_TERM" | ||||||
|  | # Public: The unsanitized TERM under which sharness is originally run | ||||||
|  | export SHARNESS_ORIG_TERM | ||||||
|  |  | ||||||
|  | # Export SHELL_PATH | ||||||
|  | : "${SHELL_PATH:=$SHELL}" | ||||||
|  | export SHELL_PATH | ||||||
|  |  | ||||||
|  | # if --tee was passed, write the output not only to the terminal, but | ||||||
|  | # additionally to the file test-results/$BASENAME.out, too. | ||||||
|  | case "$SHARNESS_TEST_TEE_STARTED, $* " in | ||||||
|  | done,*) | ||||||
|  | 	# do not redirect again | ||||||
|  | 	;; | ||||||
|  | *' --tee '*|*' --verbose-log '*) | ||||||
|  | 	mkdir -p "$SHARNESS_TEST_OUTPUT_DIRECTORY/test-results" | ||||||
|  | 	BASE="$SHARNESS_TEST_OUTPUT_DIRECTORY/test-results/$(basename "$0" ".$SHARNESS_TEST_EXTENSION")" | ||||||
|  |  | ||||||
|  | 	# Make this filename available to the sub-process in case it is using | ||||||
|  | 	# --verbose-log. | ||||||
|  | 	SHARNESS_TEST_TEE_OUTPUT_FILE="$BASE.out" | ||||||
|  | 	export SHARNESS_TEST_TEE_OUTPUT_FILE | ||||||
|  |  | ||||||
|  | 	# Truncate before calling "tee -a" to get rid of the results | ||||||
|  | 	# from any previous runs. | ||||||
|  | 	: >"$SHARNESS_TEST_TEE_OUTPUT_FILE" | ||||||
|  |  | ||||||
|  | 	(SHARNESS_TEST_TEE_STARTED="done" ${SHELL_PATH} "$0" "$@" 2>&1; | ||||||
|  | 	 echo $? >"$BASE.exit") | tee -a "$SHARNESS_TEST_TEE_OUTPUT_FILE" | ||||||
|  | 	test "$(cat "$BASE.exit")" = 0 | ||||||
|  | 	exit | ||||||
|  | 	;; | ||||||
|  | esac | ||||||
|  |  | ||||||
| # For repeatability, reset the environment to a known state. | # For repeatability, reset the environment to a known state. | ||||||
|  | # TERM is sanitized below, after saving color control sequences. | ||||||
| LANG=C | LANG=C | ||||||
| LC_ALL=C | LC_ALL=C | ||||||
| PAGER=cat | PAGER="cat" | ||||||
| TZ=UTC | TZ=UTC | ||||||
| TERM=dumb |  | ||||||
| EDITOR=: | EDITOR=: | ||||||
| export LANG LC_ALL PAGER TZ TERM EDITOR | export LANG LC_ALL PAGER TZ EDITOR | ||||||
| unset VISUAL CDPATH GREP_OPTIONS | unset VISUAL CDPATH GREP_OPTIONS | ||||||
|  |  | ||||||
| # Line feed | [ "x$TERM" != "xdumb" ] && ( | ||||||
| LF=' |  | ||||||
| ' |  | ||||||
|  |  | ||||||
| [ "x$ORIGINAL_TERM" != "xdumb" ] && ( |  | ||||||
| 		TERM=$ORIGINAL_TERM && |  | ||||||
| 		export TERM && |  | ||||||
| 		[ -t 1 ] && | 		[ -t 1 ] && | ||||||
| 		tput bold >/dev/null 2>&1 && | 		tput bold >/dev/null 2>&1 && | ||||||
| 		tput setaf 1 >/dev/null 2>&1 && | 		tput setaf 1 >/dev/null 2>&1 && | ||||||
| @@ -60,6 +107,8 @@ while test "$#" -ne 0; do | |||||||
| 		immediate=t; shift ;; | 		immediate=t; shift ;; | ||||||
| 	-l|--l|--lo|--lon|--long|--long-|--long-t|--long-te|--long-tes|--long-test|--long-tests) | 	-l|--l|--lo|--lon|--long|--long-|--long-t|--long-te|--long-tes|--long-test|--long-tests) | ||||||
| 		TEST_LONG=t; export TEST_LONG; shift ;; | 		TEST_LONG=t; export TEST_LONG; shift ;; | ||||||
|  | 	--in|--int|--inte|--inter|--intera|--interac|--interact|--interacti|--interactiv|--interactive|--interactive-|--interactive-t|--interactive-te|--interactive-tes|--interactive-test|--interactive-tests): | ||||||
|  | 		TEST_INTERACTIVE=t; export TEST_INTERACTIVE; verbose=t; shift ;; | ||||||
| 	-h|--h|--he|--hel|--help) | 	-h|--h|--he|--hel|--help) | ||||||
| 		help=t; shift ;; | 		help=t; shift ;; | ||||||
| 	-v|--v|--ve|--ver|--verb|--verbo|--verbos|--verbose) | 	-v|--v|--ve|--ver|--verb|--verbo|--verbos|--verbose) | ||||||
| @@ -68,49 +117,69 @@ while test "$#" -ne 0; do | |||||||
| 		# Ignore --quiet under a TAP::Harness. Saying how many tests | 		# Ignore --quiet under a TAP::Harness. Saying how many tests | ||||||
| 		# passed without the ok/not ok details is always an error. | 		# passed without the ok/not ok details is always an error. | ||||||
| 		test -z "$HARNESS_ACTIVE" && quiet=t; shift ;; | 		test -z "$HARNESS_ACTIVE" && quiet=t; shift ;; | ||||||
|  | 	--chain-lint) | ||||||
|  | 		chain_lint=t; shift ;; | ||||||
|  | 	--no-chain-lint) | ||||||
|  | 		chain_lint=; shift ;; | ||||||
| 	--no-color) | 	--no-color) | ||||||
| 		color=; shift ;; | 		color=; shift ;; | ||||||
|  | 	--tee) | ||||||
|  | 		shift ;; # was handled already | ||||||
| 	--root=*) | 	--root=*) | ||||||
| 		root=$(expr "z$1" : 'z[^=]*=\(.*\)') | 		root=$(expr "z$1" : 'z[^=]*=\(.*\)') | ||||||
| 		shift ;; | 		shift ;; | ||||||
|  | 	--verbose-log) | ||||||
|  | 		verbose_log=t | ||||||
|  | 		shift ;; | ||||||
| 	*) | 	*) | ||||||
| 		echo "error: unknown test option '$1'" >&2; exit 1 ;; | 		echo "error: unknown test option '$1'" >&2; exit 1 ;; | ||||||
| 	esac | 	esac | ||||||
| done | done | ||||||
|  |  | ||||||
| if test -n "$color"; then | if test -n "$color"; then | ||||||
|  | 	# Save the color control sequences now rather than run tput | ||||||
|  | 	# each time say_color() is called.  This is done for two | ||||||
|  | 	# reasons: | ||||||
|  | 	#   * TERM will be changed to dumb | ||||||
|  | 	#   * HOME will be changed to a temporary directory and tput | ||||||
|  | 	#     might need to read ~/.terminfo from the original HOME | ||||||
|  | 	#     directory to get the control sequences | ||||||
|  | 	# Note:  This approach assumes the control sequences don't end | ||||||
|  | 	# in a newline for any terminal of interest (command | ||||||
|  | 	# substitutions strip trailing newlines).  Given that most | ||||||
|  | 	# (all?) terminals in common use are related to ECMA-48, this | ||||||
|  | 	# shouldn't be a problem. | ||||||
|  | 	say_color_error=$(tput bold; tput setaf 1) # bold red | ||||||
|  | 	say_color_skip=$(tput setaf 4) # blue | ||||||
|  | 	say_color_warn=$(tput setaf 3) # brown/yellow | ||||||
|  | 	say_color_pass=$(tput setaf 2) # green | ||||||
|  | 	say_color_info=$(tput setaf 6) # cyan | ||||||
|  | 	say_color_reset=$(tput sgr0) | ||||||
|  | 	say_color_raw="" # no formatting for normal text | ||||||
| 	say_color() { | 	say_color() { | ||||||
| 		( | 		test -z "$1" && test -n "$quiet" && return | ||||||
| 		TERM=$ORIGINAL_TERM |  | ||||||
| 		export TERM |  | ||||||
| 		case "$1" in | 		case "$1" in | ||||||
| 		error) | 			error) say_color_color=$say_color_error ;; | ||||||
| 			tput bold; tput setaf 1;; # bold red | 			skip) say_color_color=$say_color_skip ;; | ||||||
| 		skip) | 			warn) say_color_color=$say_color_warn ;; | ||||||
| 			tput setaf 4;; # blue | 			pass) say_color_color=$say_color_pass ;; | ||||||
| 		warn) | 			info) say_color_color=$say_color_info ;; | ||||||
| 			tput setaf 3;; # brown/yellow | 			*) say_color_color=$say_color_raw ;; | ||||||
| 		pass) |  | ||||||
| 			tput setaf 2;; # green |  | ||||||
| 		info) |  | ||||||
| 			tput setaf 6;; # cyan |  | ||||||
| 		*) |  | ||||||
| 			test -n "$quiet" && return;; |  | ||||||
| 		esac | 		esac | ||||||
| 		shift | 		shift | ||||||
| 		printf "%s" "$*" | 		printf '%s%s%s\n' "$say_color_color" "$*" "$say_color_reset" | ||||||
| 		tput sgr0 |  | ||||||
| 		echo |  | ||||||
| 		) |  | ||||||
| 	} | 	} | ||||||
| else | else | ||||||
| 	say_color() { | 	say_color() { | ||||||
| 		test -z "$1" && test -n "$quiet" && return | 		test -z "$1" && test -n "$quiet" && return | ||||||
| 		shift | 		shift | ||||||
| 		printf "%s\n" "$*" | 		printf '%s\n' "$*" | ||||||
| 	} | 	} | ||||||
| fi | fi | ||||||
|  |  | ||||||
|  | TERM=dumb | ||||||
|  | export TERM | ||||||
|  |  | ||||||
| error() { | error() { | ||||||
| 	say_color error "error: $*" | 	say_color error "error: $*" | ||||||
| 	EXIT_OK=t | 	EXIT_OK=t | ||||||
| @@ -121,7 +190,7 @@ say() { | |||||||
| 	say_color info "$*" | 	say_color info "$*" | ||||||
| } | } | ||||||
|  |  | ||||||
| test -n "$test_description" || error "Test script did not set test_description." | test -n "${test_description:-}" || error "Test script did not set test_description." | ||||||
|  |  | ||||||
| if test "$help" = "t"; then | if test "$help" = "t"; then | ||||||
| 	echo "$test_description" | 	echo "$test_description" | ||||||
| @@ -130,7 +199,11 @@ fi | |||||||
|  |  | ||||||
| exec 5>&1 | exec 5>&1 | ||||||
| exec 6<&0 | exec 6<&0 | ||||||
| if test "$verbose" = "t"; then | if test "$verbose_log" = "t" | ||||||
|  | then | ||||||
|  | 	exec 3>>"$SHARNESS_TEST_TEE_OUTPUT_FILE" 4>&3 | ||||||
|  | elif test "$verbose" = "t" | ||||||
|  | then | ||||||
| 	exec 4>&2 3>&1 | 	exec 4>&2 3>&1 | ||||||
| else | else | ||||||
| 	exec 4>/dev/null 3>/dev/null | 	exec 4>/dev/null 3>/dev/null | ||||||
| @@ -161,7 +234,7 @@ trap 'die' EXIT | |||||||
| # implicitly by specifying the prerequisite name in calls to test_expect_success | # implicitly by specifying the prerequisite name in calls to test_expect_success | ||||||
| # or test_expect_failure. | # or test_expect_failure. | ||||||
| # | # | ||||||
| # $1 - Name of prerequiste (a simple word, in all capital letters by convention) | # $1 - Name of prerequisite (a simple word, in all capital letters by convention) | ||||||
| # | # | ||||||
| # Examples | # Examples | ||||||
| # | # | ||||||
| @@ -198,7 +271,7 @@ test_have_prereq() { | |||||||
| 	# prerequisites can be concatenated with ',' | 	# prerequisites can be concatenated with ',' | ||||||
| 	save_IFS=$IFS | 	save_IFS=$IFS | ||||||
| 	IFS=, | 	IFS=, | ||||||
| 	set -- $* | 	set -- $@ | ||||||
| 	IFS=$save_IFS | 	IFS=$save_IFS | ||||||
|  |  | ||||||
| 	total_prereq=0 | 	total_prereq=0 | ||||||
| @@ -215,7 +288,7 @@ test_have_prereq() { | |||||||
| 			negative_prereq= | 			negative_prereq= | ||||||
| 		esac | 		esac | ||||||
|  |  | ||||||
| 		total_prereq=$(($total_prereq + 1)) | 		total_prereq=$((total_prereq + 1)) | ||||||
| 		case "$satisfied_prereq" in | 		case "$satisfied_prereq" in | ||||||
| 		*" $prerequisite "*) | 		*" $prerequisite "*) | ||||||
| 			satisfied_this_prereq=t | 			satisfied_this_prereq=t | ||||||
| @@ -226,7 +299,7 @@ test_have_prereq() { | |||||||
|  |  | ||||||
| 		case "$satisfied_this_prereq,$negative_prereq" in | 		case "$satisfied_this_prereq,$negative_prereq" in | ||||||
| 		t,|,t) | 		t,|,t) | ||||||
| 			ok_prereq=$(($ok_prereq + 1)) | 			ok_prereq=$((ok_prereq + 1)) | ||||||
| 			;; | 			;; | ||||||
| 		*) | 		*) | ||||||
| 			# Keep a list of missing prerequisites; restore | 			# Keep a list of missing prerequisites; restore | ||||||
| @@ -247,12 +320,12 @@ test_have_prereq() { | |||||||
| # the text_expect_* functions instead. | # the text_expect_* functions instead. | ||||||
|  |  | ||||||
| test_ok_() { | test_ok_() { | ||||||
| 	test_success=$(($test_success + 1)) | 	test_success=$((test_success + 1)) | ||||||
| 	say_color "" "ok $test_count - $@" | 	say_color "" "ok $test_count - $*" | ||||||
| } | } | ||||||
|  |  | ||||||
| test_failure_() { | test_failure_() { | ||||||
| 	test_failure=$(($test_failure + 1)) | 	test_failure=$((test_failure + 1)) | ||||||
| 	say_color error "not ok $test_count - $1" | 	say_color error "not ok $test_count - $1" | ||||||
| 	shift | 	shift | ||||||
| 	echo "$@" | sed -e 's/^/#	/' | 	echo "$@" | sed -e 's/^/#	/' | ||||||
| @@ -260,13 +333,13 @@ test_failure_() { | |||||||
| } | } | ||||||
|  |  | ||||||
| test_known_broken_ok_() { | test_known_broken_ok_() { | ||||||
| 	test_fixed=$(($test_fixed + 1)) | 	test_fixed=$((test_fixed + 1)) | ||||||
| 	say_color error "ok $test_count - $@ # TODO known breakage vanished" | 	say_color error "ok $test_count - $* # TODO known breakage vanished" | ||||||
| } | } | ||||||
|  |  | ||||||
| test_known_broken_failure_() { | test_known_broken_failure_() { | ||||||
| 	test_broken=$(($test_broken + 1)) | 	test_broken=$((test_broken + 1)) | ||||||
| 	say_color warn "not ok $test_count - $@ # TODO known breakage" | 	say_color warn "not ok $test_count - $* # TODO known breakage" | ||||||
| } | } | ||||||
|  |  | ||||||
| # Public: Execute commands in debug mode. | # Public: Execute commands in debug mode. | ||||||
| @@ -287,10 +360,29 @@ test_debug() { | |||||||
| 	test "$debug" = "" || eval "$1" | 	test "$debug" = "" || eval "$1" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | # Public: Stop execution and start a shell. | ||||||
|  | # | ||||||
|  | # This is useful for debugging tests and only makes sense together with "-v". | ||||||
|  | # Be sure to remove all invocations of this command before submitting. | ||||||
|  | test_pause() { | ||||||
|  | 	if test "$verbose" = t; then | ||||||
|  | 		"$SHELL_PATH" <&6 >&3 2>&4 | ||||||
|  | 	else | ||||||
|  | 		error >&5 "test_pause requires --verbose" | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
| test_eval_() { | test_eval_() { | ||||||
| 	# This is a separate function because some tests use | 	# This is a separate function because some tests use | ||||||
| 	# "return" to end a test_expect_success block early. | 	# "return" to end a test_expect_success block early. | ||||||
| 	eval </dev/null >&3 2>&4 "$*" | 	case ",$test_prereq," in | ||||||
|  | 	*,INTERACTIVE,*) | ||||||
|  | 		eval "$*" | ||||||
|  | 		;; | ||||||
|  | 	*) | ||||||
|  | 		eval </dev/null >&3 2>&4 "$*" | ||||||
|  | 		;; | ||||||
|  | 	esac | ||||||
| } | } | ||||||
|  |  | ||||||
| test_run_() { | test_run_() { | ||||||
| @@ -299,6 +391,13 @@ test_run_() { | |||||||
| 	test_eval_ "$1" | 	test_eval_ "$1" | ||||||
| 	eval_ret=$? | 	eval_ret=$? | ||||||
|  |  | ||||||
|  | 	if test "$chain_lint" = "t"; then | ||||||
|  | 		test_eval_ "(exit 117) && $1" | ||||||
|  | 		if test "$?" != 117; then | ||||||
|  | 			error "bug in the test script: broken &&-chain: $1" | ||||||
|  | 		fi | ||||||
|  | 	fi | ||||||
|  |  | ||||||
| 	if test -z "$immediate" || test $eval_ret = 0 || test -n "$expecting_failure"; then | 	if test -z "$immediate" || test $eval_ret = 0 || test -n "$expecting_failure"; then | ||||||
| 		test_eval_ "$test_cleanup" | 		test_eval_ "$test_cleanup" | ||||||
| 	fi | 	fi | ||||||
| @@ -309,7 +408,7 @@ test_run_() { | |||||||
| } | } | ||||||
|  |  | ||||||
| test_skip_() { | test_skip_() { | ||||||
| 	test_count=$(($test_count + 1)) | 	test_count=$((test_count + 1)) | ||||||
| 	to_skip= | 	to_skip= | ||||||
| 	for skp in $SKIP_TESTS; do | 	for skp in $SKIP_TESTS; do | ||||||
| 		case $this_test.$test_count in | 		case $this_test.$test_count in | ||||||
| @@ -328,7 +427,7 @@ test_skip_() { | |||||||
| 			of_prereq=" of $test_prereq" | 			of_prereq=" of $test_prereq" | ||||||
| 		fi | 		fi | ||||||
|  |  | ||||||
| 		say_color skip >&3 "skipping test: $@" | 		say_color skip >&3 "skipping test: $*" | ||||||
| 		say_color skip "ok $test_count # skip $1 (missing $missing_prereq${of_prereq})" | 		say_color skip "ok $test_count # skip $1 (missing $missing_prereq${of_prereq})" | ||||||
| 		: true | 		: true | ||||||
| 		;; | 		;; | ||||||
| @@ -426,6 +525,44 @@ test_expect_failure() { | |||||||
| 	echo >&3 "" | 	echo >&3 "" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | # Public: Run test commands and expect anything from them. Used when a | ||||||
|  | # test is not stable or not finished for some reason. | ||||||
|  | # | ||||||
|  | # When the test passed, an "ok" message is printed, but the number of | ||||||
|  | # fixed tests is not incremented. | ||||||
|  | # | ||||||
|  | # When it failed, a "not ok ... # TODO known breakage" message is | ||||||
|  | # printed, and the number of tests still broken is incremented. | ||||||
|  | # | ||||||
|  | # Failures from these tests won't cause --immediate to stop. | ||||||
|  | # | ||||||
|  | # Usually takes two arguments: | ||||||
|  | # $1 - Test description | ||||||
|  | # $2 - Commands to be executed. | ||||||
|  | # | ||||||
|  | # With three arguments, the first will be taken to be a prerequisite: | ||||||
|  | # $1 - Comma-separated list of test prerequisites. The test will be skipped if | ||||||
|  | #      not all of the given prerequisites are set. To negate a prerequisite, | ||||||
|  | #      put a "!" in front of it. | ||||||
|  | # $2 - Test description | ||||||
|  | # $3 - Commands to be executed. | ||||||
|  | # | ||||||
|  | # Returns nothing. | ||||||
|  | test_expect_unstable() { | ||||||
|  | 	test "$#" = 3 && { test_prereq=$1; shift; } || test_prereq= | ||||||
|  | 	test "$#" = 2 || error "bug in the test script: not 2 or 3 parameters to test_expect_unstable" | ||||||
|  | 	export test_prereq | ||||||
|  | 	if ! test_skip_ "$@"; then | ||||||
|  | 		say >&3 "checking unstable test: $2" | ||||||
|  | 		if test_run_ "$2" unstable; then | ||||||
|  | 			test_ok_ "$1" | ||||||
|  | 		else | ||||||
|  | 			test_known_broken_failure_ "$1" | ||||||
|  | 		fi | ||||||
|  | 	fi | ||||||
|  | 	echo >&3 "" | ||||||
|  | } | ||||||
|  |  | ||||||
| # Public: Run command and ensure that it fails in a controlled way. | # Public: Run command and ensure that it fails in a controlled way. | ||||||
| # | # | ||||||
| # Use it instead of "! <command>". For example, when <command> dies due to a | # Use it instead of "! <command>". For example, when <command> dies due to a | ||||||
| @@ -518,7 +655,7 @@ test_expect_code() { | |||||||
| 	shift | 	shift | ||||||
| 	"$@" | 	"$@" | ||||||
| 	exit_code=$? | 	exit_code=$? | ||||||
| 	if test $exit_code = $want_code; then | 	if test "$exit_code" = "$want_code"; then | ||||||
| 		return 0 | 		return 0 | ||||||
| 	fi | 	fi | ||||||
|  |  | ||||||
| @@ -528,7 +665,7 @@ test_expect_code() { | |||||||
|  |  | ||||||
| # Public: Compare two files to see if expected output matches actual output. | # Public: Compare two files to see if expected output matches actual output. | ||||||
| # | # | ||||||
| # The TEST_CMP variable defines the command used for the comparision; it | # The TEST_CMP variable defines the command used for the comparison; it | ||||||
| # defaults to "diff -u". Only when the test script was started with --verbose, | # defaults to "diff -u". Only when the test script was started with --verbose, | ||||||
| # will the command's output, the diff, be printed to the standard output. | # will the command's output, the diff, be printed to the standard output. | ||||||
| # | # | ||||||
| @@ -551,6 +688,79 @@ test_cmp() { | |||||||
| 	${TEST_CMP:-diff -u} "$@" | 	${TEST_CMP:-diff -u} "$@" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | # Public: portably print a sequence of numbers. | ||||||
|  | # | ||||||
|  | # seq is not in POSIX and GNU seq might not be available everywhere, | ||||||
|  | # so it is nice to have a seq implementation, even a very simple one. | ||||||
|  | # | ||||||
|  | # $1 - Starting number. | ||||||
|  | # $2 - Ending number. | ||||||
|  | # | ||||||
|  | # Examples | ||||||
|  | # | ||||||
|  | #   test_expect_success 'foo works 10 times' ' | ||||||
|  | #       for i in $(test_seq 1 10) | ||||||
|  | #       do | ||||||
|  | #           foo || return | ||||||
|  | #       done | ||||||
|  | #   ' | ||||||
|  | # | ||||||
|  | # Returns 0 if all the specified numbers can be displayed. | ||||||
|  | test_seq() { | ||||||
|  | 	i="$1" | ||||||
|  | 	j="$2" | ||||||
|  | 	while test "$i" -le "$j" | ||||||
|  | 	do | ||||||
|  | 		echo "$i" || return | ||||||
|  | 		i=$(("$i" + 1)) | ||||||
|  | 	done | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Public: Check if the file expected to be empty is indeed empty, and barfs | ||||||
|  | # otherwise. | ||||||
|  | # | ||||||
|  | # $1 - File to check for emptiness. | ||||||
|  | # | ||||||
|  | # Returns 0 if file is empty, 1 otherwise. | ||||||
|  | test_must_be_empty() { | ||||||
|  | 	if test -s "$1" | ||||||
|  | 	then | ||||||
|  | 		echo "'$1' is not empty, it contains:" | ||||||
|  | 		cat "$1" | ||||||
|  | 		return 1 | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # debugging-friendly alternatives to "test [-f|-d|-e]" | ||||||
|  | # The commands test the existence or non-existence of $1. $2 can be | ||||||
|  | # given to provide a more precise diagnosis. | ||||||
|  | test_path_is_file () { | ||||||
|  | 	if ! test -f "$1" | ||||||
|  | 	then | ||||||
|  | 		echo "File $1 doesn't exist. $2" | ||||||
|  | 		false | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | test_path_is_dir () { | ||||||
|  | 	if ! test -d "$1" | ||||||
|  | 	then | ||||||
|  | 		echo "Directory $1 doesn't exist. $2" | ||||||
|  | 		false | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
|  | # Check if the directory exists and is empty as expected, barf otherwise. | ||||||
|  | test_dir_is_empty () { | ||||||
|  | 	test_path_is_dir "$1" && | ||||||
|  | 	if test -n "$(find "$1" -mindepth 1 -maxdepth 1)" | ||||||
|  | 	then | ||||||
|  | 		echo "Directory '$1' is not empty, it contains:" | ||||||
|  | 		ls -la "$1" | ||||||
|  | 		return 1 | ||||||
|  | 	fi | ||||||
|  | } | ||||||
|  |  | ||||||
| # Public: Schedule cleanup commands to be run unconditionally at the end of a | # Public: Schedule cleanup commands to be run unconditionally at the end of a | ||||||
| # test. | # test. | ||||||
| # | # | ||||||
| @@ -576,6 +786,23 @@ test_when_finished() { | |||||||
| 		} && (exit \"\$eval_ret\"); eval_ret=\$?; $test_cleanup" | 		} && (exit \"\$eval_ret\"); eval_ret=\$?; $test_cleanup" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | # Public: Schedule cleanup commands to be run unconditionally when all tests | ||||||
|  | # have run. | ||||||
|  | # | ||||||
|  | # This can be used to clean up things like test databases. It is not needed to | ||||||
|  | # clean up temporary files, as test_done already does that. | ||||||
|  | # | ||||||
|  | # Examples: | ||||||
|  | # | ||||||
|  | #   cleanup mysql -e "DROP DATABASE mytest" | ||||||
|  | # | ||||||
|  | # Returns the exit code of the last cleanup command executed. | ||||||
|  | final_cleanup= | ||||||
|  | cleanup() { | ||||||
|  | 	final_cleanup="{ $* | ||||||
|  | 		} && (exit \"\$eval_ret\"); eval_ret=\$?; $final_cleanup" | ||||||
|  | } | ||||||
|  |  | ||||||
| # Public: Summarize test results and exit with an appropriate error code. | # Public: Summarize test results and exit with an appropriate error code. | ||||||
| # | # | ||||||
| # Must be called at the end of each test script. | # Must be called at the end of each test script. | ||||||
| @@ -600,9 +827,9 @@ test_done() { | |||||||
| 	EXIT_OK=t | 	EXIT_OK=t | ||||||
|  |  | ||||||
| 	if test -z "$HARNESS_ACTIVE"; then | 	if test -z "$HARNESS_ACTIVE"; then | ||||||
| 		test_results_dir="$SHARNESS_TEST_DIRECTORY/test-results" | 		test_results_dir="$SHARNESS_TEST_OUTPUT_DIRECTORY/test-results" | ||||||
| 		mkdir -p "$test_results_dir" | 		mkdir -p "$test_results_dir" | ||||||
| 		test_results_path="$test_results_dir/${SHARNESS_TEST_FILE%.$SHARNESS_TEST_EXTENSION}.$$.counts" | 		test_results_path="$test_results_dir/$this_test.$$.counts" | ||||||
|  |  | ||||||
| 		cat >>"$test_results_path" <<-EOF | 		cat >>"$test_results_path" <<-EOF | ||||||
| 		total $test_count | 		total $test_count | ||||||
| @@ -621,7 +848,7 @@ test_done() { | |||||||
| 		say_color warn "# still have $test_broken known breakage(s)" | 		say_color warn "# still have $test_broken known breakage(s)" | ||||||
| 	fi | 	fi | ||||||
| 	if test "$test_broken" != 0 || test "$test_fixed" != 0; then | 	if test "$test_broken" != 0 || test "$test_fixed" != 0; then | ||||||
| 		test_remaining=$(( $test_count - $test_broken - $test_fixed )) | 		test_remaining=$((test_count - test_broken - test_fixed)) | ||||||
| 		msg="remaining $test_remaining test(s)" | 		msg="remaining $test_remaining test(s)" | ||||||
| 	else | 	else | ||||||
| 		test_remaining=$test_count | 		test_remaining=$test_count | ||||||
| @@ -641,6 +868,8 @@ test_done() { | |||||||
| 		fi | 		fi | ||||||
| 		say "1..$test_count$skip_all" | 		say "1..$test_count$skip_all" | ||||||
|  |  | ||||||
|  | 		test_eval_ "$final_cleanup" | ||||||
|  |  | ||||||
| 		test -d "$remove_trash" && | 		test -d "$remove_trash" && | ||||||
| 		cd "$(dirname "$remove_trash")" && | 		cd "$(dirname "$remove_trash")" && | ||||||
| 		rm -rf "$(basename "$remove_trash")" | 		rm -rf "$(basename "$remove_trash")" | ||||||
| @@ -656,14 +885,15 @@ test_done() { | |||||||
| 	esac | 	esac | ||||||
| } | } | ||||||
|  |  | ||||||
| # Public: Root directory containing tests. Tests can override this variable, | # Public: Source directory of test code and sharness library. | ||||||
| # e.g. for testing Sharness itself. | # This directory may be different from the directory in which tests are | ||||||
| : ${SHARNESS_TEST_DIRECTORY:=$(pwd)} | # being run. | ||||||
| export SHARNESS_TEST_DIRECTORY | : "${SHARNESS_TEST_SRCDIR:=$(cd "$(dirname "$0")" && pwd)}" | ||||||
|  | export SHARNESS_TEST_SRCDIR | ||||||
|  |  | ||||||
| # Public: Build directory that will be added to PATH. By default, it is set to | # Public: Build directory that will be added to PATH. By default, it is set to | ||||||
| # the parent directory of SHARNESS_TEST_DIRECTORY. | # the parent directory of SHARNESS_TEST_DIRECTORY. | ||||||
| : ${SHARNESS_BUILD_DIRECTORY:="$SHARNESS_TEST_DIRECTORY/.."} | : "${SHARNESS_BUILD_DIRECTORY:="$SHARNESS_TEST_DIRECTORY/.."}" | ||||||
| PATH="$SHARNESS_BUILD_DIRECTORY:$PATH" | PATH="$SHARNESS_BUILD_DIRECTORY:$PATH" | ||||||
| export PATH SHARNESS_BUILD_DIRECTORY | export PATH SHARNESS_BUILD_DIRECTORY | ||||||
|  |  | ||||||
| @@ -672,19 +902,43 @@ SHARNESS_TEST_FILE="$0" | |||||||
| export SHARNESS_TEST_FILE | export SHARNESS_TEST_FILE | ||||||
|  |  | ||||||
| # Prepare test area. | # Prepare test area. | ||||||
| test_dir="trash directory.$(basename "$SHARNESS_TEST_FILE" ".$SHARNESS_TEST_EXTENSION")" | SHARNESS_TRASH_DIRECTORY="trash directory.$(basename "$SHARNESS_TEST_FILE" ".$SHARNESS_TEST_EXTENSION")" | ||||||
| test -n "$root" && test_dir="$root/$test_dir" | test -n "$root" && SHARNESS_TRASH_DIRECTORY="$root/$SHARNESS_TRASH_DIRECTORY" | ||||||
| case "$test_dir" in | case "$SHARNESS_TRASH_DIRECTORY" in | ||||||
| /*) SHARNESS_TRASH_DIRECTORY="$test_dir" ;; | /*) ;; # absolute path is good | ||||||
|  *) SHARNESS_TRASH_DIRECTORY="$SHARNESS_TEST_DIRECTORY/$test_dir" ;; |  *) SHARNESS_TRASH_DIRECTORY="$SHARNESS_TEST_OUTPUT_DIRECTORY/$SHARNESS_TRASH_DIRECTORY" ;; | ||||||
| esac | esac | ||||||
| test "$debug" = "t" || remove_trash="$SHARNESS_TRASH_DIRECTORY" | test "$debug" = "t" || remove_trash="$SHARNESS_TRASH_DIRECTORY" | ||||||
| rm -rf "$test_dir" || { | rm -rf "$SHARNESS_TRASH_DIRECTORY" || { | ||||||
| 	EXIT_OK=t | 	EXIT_OK=t | ||||||
| 	echo >&5 "FATAL: Cannot prepare test area" | 	echo >&5 "FATAL: Cannot prepare test area" | ||||||
| 	exit 1 | 	exit 1 | ||||||
| } | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | # | ||||||
|  | #  Load any extensions in $srcdir/sharness.d/*.sh | ||||||
|  | # | ||||||
|  | if test -d "${SHARNESS_TEST_SRCDIR}/sharness.d" | ||||||
|  | then | ||||||
|  | 	for file in "${SHARNESS_TEST_SRCDIR}"/sharness.d/*.sh | ||||||
|  | 	do | ||||||
|  | 		# Ensure glob was not an empty match: | ||||||
|  | 		test -e "${file}" || break | ||||||
|  |  | ||||||
|  | 		if test -n "$debug" | ||||||
|  | 		then | ||||||
|  | 			echo >&5 "sharness: loading extensions from ${file}" | ||||||
|  | 		fi | ||||||
|  | 		. "${file}" | ||||||
|  | 		if test $? != 0 | ||||||
|  | 		then | ||||||
|  | 			echo >&5 "sharness: Error loading ${file}. Aborting." | ||||||
|  | 			exit 1 | ||||||
|  | 		fi | ||||||
|  | 	done | ||||||
|  | fi | ||||||
|  |  | ||||||
| # Public: Empty trash directory, the test area, provided for each test. The HOME | # Public: Empty trash directory, the test area, provided for each test. The HOME | ||||||
| # variable is set to that directory too. | # variable is set to that directory too. | ||||||
| export SHARNESS_TRASH_DIRECTORY | export SHARNESS_TRASH_DIRECTORY | ||||||
| @@ -692,10 +946,10 @@ export SHARNESS_TRASH_DIRECTORY | |||||||
| HOME="$SHARNESS_TRASH_DIRECTORY" | HOME="$SHARNESS_TRASH_DIRECTORY" | ||||||
| export HOME | export HOME | ||||||
|  |  | ||||||
| mkdir -p "$test_dir" || exit 1 | mkdir -p "$SHARNESS_TRASH_DIRECTORY" || exit 1 | ||||||
| # Use -P to resolve symlinks in our working directory so that the cwd | # Use -P to resolve symlinks in our working directory so that the cwd | ||||||
| # in subprocesses like git equals our $PWD (for pathname comparisons). | # in subprocesses like git equals our $PWD (for pathname comparisons). | ||||||
| cd -P "$test_dir" || exit 1 | cd -P "$SHARNESS_TRASH_DIRECTORY" || exit 1 | ||||||
|  |  | ||||||
| this_test=${SHARNESS_TEST_FILE##*/} | this_test=${SHARNESS_TEST_FILE##*/} | ||||||
| this_test=${this_test%.$SHARNESS_TEST_EXTENSION} | this_test=${this_test%.$SHARNESS_TEST_EXTENSION} | ||||||
| @@ -708,4 +962,10 @@ for skp in $SKIP_TESTS; do | |||||||
| 	esac | 	esac | ||||||
| done | done | ||||||
|  |  | ||||||
|  | test -n "$TEST_LONG" && test_set_prereq EXPENSIVE | ||||||
|  | test -n "$TEST_INTERACTIVE" && test_set_prereq INTERACTIVE | ||||||
|  |  | ||||||
|  | # Make sure this script ends with code 0 | ||||||
|  | : | ||||||
|  |  | ||||||
| # vi: set ts=4 sw=4 noet : | # vi: set ts=4 sw=4 noet : | ||||||
|   | |||||||
| @@ -1,8 +1,60 @@ | |||||||
| #!/bin/sh | #!/bin/sh | ||||||
|  |  | ||||||
| . ./sharness.sh | if [ -z "$SHARNESS" ] ; then | ||||||
|  | 	for d in \ | ||||||
|  | 		"." \ | ||||||
|  | 		"$HOME/share/sharness" \ | ||||||
|  | 		"/usr/local/share/sharness" \ | ||||||
|  | 		"/usr/share/sharness" | ||||||
|  | 	do | ||||||
|  | 		f="$d/sharness.sh" | ||||||
|  | 		if [ -f "$f" ] ; then | ||||||
|  | 			SHARNESS="$f" | ||||||
|  | 		fi | ||||||
|  | 	done | ||||||
|  | fi | ||||||
|  | if [ -z "$SHARNESS" ] || [ ! -f "$SHARNESS" ] ; then | ||||||
|  | 	echo "sharness.sh not found" >&2 | ||||||
|  | 	exit 1 | ||||||
|  | fi | ||||||
|  |  | ||||||
| test_set_prereq PYTHON | # Prevent sharness from adding the source directory to PATH | ||||||
|  | # since the scripts use unversioned python for their shebang | ||||||
|  | # but tests should run under the python with mercurial support | ||||||
|  | # so create an empty directory and strip it from PATH afterwards | ||||||
|  | SHARNESS_BUILD_DIRECTORY="$(mktemp -d)" | ||||||
|  | . "$SHARNESS" | ||||||
|  | export PATH="${PATH#*:}" | ||||||
|  | rmdir "$SHARNESS_BUILD_DIRECTORY" | ||||||
|  |  | ||||||
|  | if [ -z "$TEST_INSTALLED_SCRIPTS" ] ; then | ||||||
|  | 	if [ -n "$PYTHON" ] && "$PYTHON" -c 'import mercurial' 2> /dev/null ; then | ||||||
|  | 		: Use chosen Python version | ||||||
|  | 	elif python3 -c 'import mercurial' 2> /dev/null ; then | ||||||
|  | 		PYTHON=python3 | ||||||
|  | 	elif python2 -c 'import mercurial' 2> /dev/null ; then | ||||||
|  | 		PYTHON=python2 | ||||||
|  | 	elif python -c 'import mercurial' 2> /dev/null ; then | ||||||
|  | 		PYTHON=python | ||||||
|  | 	fi | ||||||
|  | 	if [ -n "$PYTHON" ] ; then | ||||||
|  | 		test_set_prereq PYTHON | ||||||
|  |  | ||||||
|  | 		# Change shebang on a copy of scripts to chosen Python version | ||||||
|  | 		TEST_BIN="$SHARNESS_TRASH_DIRECTORY/bin" | ||||||
|  | 		mkdir -p "$TEST_BIN" | ||||||
|  | 		for s in git-remote-hg git-hg-helper ; do | ||||||
|  | 			printf "%s\n" "#!/usr/bin/env $PYTHON" > "$TEST_BIN/$s" | ||||||
|  | 			tail -n +2 "$SHARNESS_TEST_DIRECTORY/../$s" >> "$TEST_BIN/$s" | ||||||
|  | 			chmod u+x "$TEST_BIN/$s" | ||||||
|  | 		done | ||||||
|  | 		export PATH="$TEST_BIN${PATH:+:$PATH}" | ||||||
|  | 		unset TEST_BIN | ||||||
|  | 	fi | ||||||
|  | else | ||||||
|  | 	# The build/install process ensures Python is available | ||||||
|  | 	test_set_prereq PYTHON | ||||||
|  | fi | ||||||
|  |  | ||||||
| GIT_AUTHOR_EMAIL=author@example.com | GIT_AUTHOR_EMAIL=author@example.com | ||||||
| GIT_AUTHOR_NAME='A U Thor' | GIT_AUTHOR_NAME='A U Thor' | ||||||
| @@ -10,3 +62,7 @@ GIT_COMMITTER_EMAIL=committer@example.com | |||||||
| GIT_COMMITTER_NAME='C O Mitter' | GIT_COMMITTER_NAME='C O Mitter' | ||||||
| export GIT_AUTHOR_EMAIL GIT_AUTHOR_NAME | export GIT_AUTHOR_EMAIL GIT_AUTHOR_NAME | ||||||
| export GIT_COMMITTER_EMAIL GIT_COMMITTER_NAME | export GIT_COMMITTER_EMAIL GIT_COMMITTER_NAME | ||||||
|  | # maintain backwards compatible default | ||||||
|  | # (as used in remote helper) | ||||||
|  | git config --global init.defaultBranch master | ||||||
|  | git config --global protocol.file.allow always | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								tools/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								tools/.gitignore
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | results.txt | ||||||
							
								
								
									
										303
									
								
								tools/check-versions
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										303
									
								
								tools/check-versions
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,303 @@ | |||||||
|  | #!/usr/bin/env ruby | ||||||
|  |  | ||||||
|  | # | ||||||
|  | # Copyright (c) 2019 Felipe Contreras | ||||||
|  | # | ||||||
|  | # This script runs the tests for all versions of the components: | ||||||
|  | #   hg, hggit and dulwich | ||||||
|  | # | ||||||
|  | # You can run it without arguments, in which case it reads the file | ||||||
|  | # 'versions.txt' and executes all those checks. | ||||||
|  | # | ||||||
|  | # Or you can pass the versions to check manually, like: | ||||||
|  | # | ||||||
|  | #   ./check-versions hg:4.7 hggit:0.8.12 dulwich:0.19.7 | ||||||
|  | # | ||||||
|  | # Or you can pass just the hg version, the other versions are fetched from | ||||||
|  | # 'versions.txt': | ||||||
|  | # | ||||||
|  | #   ./check-versions hg:5.0 | ||||||
|  | # | ||||||
|  |  | ||||||
|  | require 'fileutils' | ||||||
|  | require 'tmpdir' | ||||||
|  |  | ||||||
|  | $tests = %w[main.t bidi.t hg-git.t] | ||||||
|  | $workdir = "#{Dir.home}/.cache/git-remote-hg" | ||||||
|  | $builddir = Dir.mktmpdir("git-remote-hg-build-") | ||||||
|  | $testoutdir = Dir.mktmpdir("git-remote-hg-tests-") | ||||||
|  |  | ||||||
|  | at_exit { | ||||||
|  |   FileUtils.remove_entry($builddir) | ||||||
|  |   FileUtils.remove_entry($testoutdir) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | QUIET, LOW, HIGH = (1..3).to_a | ||||||
|  | $verbosity = LOW | ||||||
|  |  | ||||||
|  | # Util {{{1 | ||||||
|  |  | ||||||
|  | def section(text) | ||||||
|  |   puts [nil, text, '=' * text.size] | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def title(text) | ||||||
|  |   puts [nil, text, '-' * text.size] unless $verbosity < HIGH | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def run_cmd(cmd, fatal: true) | ||||||
|  |   puts cmd.join(' ') unless $verbosity < HIGH | ||||||
|  |   result = system(*cmd) | ||||||
|  |   unless result or not fatal | ||||||
|  |     STDERR.puts "Failed to run command '%s'" % cmd.join(' ') | ||||||
|  |     exit -1 | ||||||
|  |   end | ||||||
|  |   result | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def check_version(a, b) | ||||||
|  |   return true if a == '@' | ||||||
|  |   a = a.split('.').map(&:to_i) | ||||||
|  |   b = b.split('.').map(&:to_i) | ||||||
|  |   (a <=> b) >= 0 | ||||||
|  | end | ||||||
|  |  | ||||||
|  | # Component {{{1 | ||||||
|  |  | ||||||
|  | class Component | ||||||
|  |  | ||||||
|  |   attr_reader :id | ||||||
|  |  | ||||||
|  |   def initialize(id, url, kind: nil, **args) | ||||||
|  |     @id = id | ||||||
|  |     @url = url | ||||||
|  |     @kind = kind || (url.start_with?('git') ? :git : :hg) | ||||||
|  |     @tool = @kind.to_s | ||||||
|  |     @checkout_fix = args[:checkout_fix] | ||||||
|  |     @version_format = args[:version_format] | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def dir | ||||||
|  |     "#{$workdir}/#{@id}" | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def get_version(version) | ||||||
|  |     return @kind == :hg ? 'tip' : '@' if version == '@' | ||||||
|  |     @version_format ? @version_format % version : version | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def clone | ||||||
|  |     run_cmd [@tool, 'clone', '-q', @url, dir] | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def checkout(version) | ||||||
|  |     Dir.chdir(dir) do | ||||||
|  |       case @kind | ||||||
|  |       when :hg | ||||||
|  |         cmd = %w[update --clean] | ||||||
|  |       when :git | ||||||
|  |         cmd = %w[reset --hard] | ||||||
|  |       else | ||||||
|  |         cmd = %w[checkout] | ||||||
|  |       end | ||||||
|  |       run_cmd [@tool] + cmd + ['-q', get_version(version)] | ||||||
|  |       @checkout_fix.call(version) if @checkout_fix | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   def build | ||||||
|  |     Dir.chdir(dir) do | ||||||
|  |       targets = %w[build_py build_ext].map { |e| [e, '--build-lib', "#{$builddir}/python"] } | ||||||
|  |       run_cmd %w[python setup.py --quiet] + targets.flatten | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  |  | ||||||
|  | end | ||||||
|  |  | ||||||
|  | # Functions {{{1 | ||||||
|  |  | ||||||
|  | def setup | ||||||
|  |   dirs = %w[bin python] | ||||||
|  |   FileUtils.mkdir_p(dirs.map { |e| "#{$builddir}/#{e}" }) | ||||||
|  |   FileUtils.mkdir_p($workdir) | ||||||
|  |  | ||||||
|  |   $components.each do |id, component| | ||||||
|  |     next if File.exists?(component.dir) | ||||||
|  |  | ||||||
|  |     if $verbosity < HIGH | ||||||
|  |       puts "Cloning #{component.id}" | ||||||
|  |     else | ||||||
|  |       title "Cloning #{component.id}" | ||||||
|  |     end | ||||||
|  |     component.clone | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def test_env(paths: nil) | ||||||
|  |   old = ENV.to_h | ||||||
|  |   paths.each do |id, path| | ||||||
|  |     name = id.to_s | ||||||
|  |     ENV[name] = "#{path}:#{ENV[name]}" | ||||||
|  |   end | ||||||
|  |   r = yield | ||||||
|  |   ENV.replace(old) | ||||||
|  |   return r | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def run_tests(tests) | ||||||
|  |   title "Running tests" | ||||||
|  |  | ||||||
|  |   Dir.chdir("#{__dir__}/../test") do | ||||||
|  |     case $verbosity | ||||||
|  |     when QUIET | ||||||
|  |       tests_opt = tests.join(' ') | ||||||
|  |       cmd = "prove -q #{tests_opt} :: -i" | ||||||
|  |     when LOW | ||||||
|  |       tests_opt = "T='%s'" % tests.join(' ') | ||||||
|  |       cmd = "make -j1 #{tests_opt}" | ||||||
|  |     else | ||||||
|  |       tests_opt = "T='%s'" % tests.join(' ') | ||||||
|  |       cmd = "TEST_OPTS='-v -i' make -j1 #{tests_opt}" | ||||||
|  |     end | ||||||
|  |     system(cmd) | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def versions_to_s(versions) | ||||||
|  |   versions.map { |k,v| "#{k}:#{v}" }.join(' ') | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def versions_from_args(args) | ||||||
|  |   args.map { |e| k, v = e.split(':'); [k.to_sym, v] }.to_h | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def versions_from_s(str) | ||||||
|  |   versions_from_args(str.split(' ')) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def check(versions) | ||||||
|  |   section versions_to_s(versions) | ||||||
|  |  | ||||||
|  |   versions.each do |id, version| | ||||||
|  |     component = $components[id] | ||||||
|  |     next unless component | ||||||
|  |  | ||||||
|  |     title "Checking out #{component.id} #{version}" | ||||||
|  |     component.checkout(version) | ||||||
|  |  | ||||||
|  |     title "Building #{component.id}" | ||||||
|  |     component.build | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   paths = { | ||||||
|  |     PATH: "#{$builddir}/bin", | ||||||
|  |     PYTHONPATH: "#{$builddir}/python", | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   test_env(paths: paths) do | ||||||
|  |     ENV['SHARNESS_TEST_OUTPUT_DIRECTORY'] = $testoutdir | ||||||
|  |     run_tests($tests) | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | # Add components {{{1 | ||||||
|  |  | ||||||
|  | $components = {} | ||||||
|  |  | ||||||
|  | def add_component(id, url, **args) | ||||||
|  |   $components[id] = Component.new(id, url, **args) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | hg_checkout_fix = lambda do |version| | ||||||
|  |   FileUtils.cp('hg', "#{$builddir}/bin/") | ||||||
|  |  | ||||||
|  |   return if check_version(version, '4.3') | ||||||
|  |  | ||||||
|  |   if run_cmd %W[hg import -q --no-commit #{__dir__}/hg_setup_hack_2.4.patch], fatal: false | ||||||
|  |     File.write('.hg_force_version', "%s\n" % version) | ||||||
|  |   else | ||||||
|  |     File.write('mercurial/__version__.py', "version = \"%s\"\n" % version) | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | add_component(:hg, 'https://www.mercurial-scm.org/repo/hg', checkout_fix: hg_checkout_fix) | ||||||
|  |  | ||||||
|  | hggit_checkout_fix = lambda do |version| | ||||||
|  |   return unless check_version(version, '0.8.0') | ||||||
|  |  | ||||||
|  |   run_cmd %W[hg import -q --no-commit #{__dir__}/hggit_rename_fix_0.8.0.patch], fatal: false | ||||||
|  | end | ||||||
|  |  | ||||||
|  | add_component(:hggit, 'https://bitbucket.org/durin42/hg-git', checkout_fix: hggit_checkout_fix) | ||||||
|  |  | ||||||
|  | add_component(:dulwich, 'https://github.com/dulwich/dulwich.git', version_format: 'dulwich-%s', kind: :git) | ||||||
|  |  | ||||||
|  | def load_checks(file) | ||||||
|  |   file.each do |e| | ||||||
|  |     e.chomp! | ||||||
|  |     next if e.empty? or e.start_with?('#') | ||||||
|  |     content, comment = e.split(' # ') | ||||||
|  |     versions = versions_from_s(content) | ||||||
|  |     $checks << versions | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | def store_results(file) | ||||||
|  |   $results.each do |versions, result| | ||||||
|  |     content = versions_to_s(versions) | ||||||
|  |     comment = result ? 'OK' : 'FAIL' | ||||||
|  |     file.puts '%s # %s' % [content, comment] | ||||||
|  |   end | ||||||
|  | end | ||||||
|  |  | ||||||
|  | # Main {{{1 | ||||||
|  |  | ||||||
|  | setup | ||||||
|  |  | ||||||
|  | $checks = [] | ||||||
|  | $results = [] | ||||||
|  |  | ||||||
|  | $versions = versions_from_args(ARGV) | ||||||
|  |  | ||||||
|  | File.open("#{__dir__}/versions.txt") do |f| | ||||||
|  |   load_checks(f) | ||||||
|  | end | ||||||
|  |  | ||||||
|  | if $versions.size == 1 and $versions.key?(:hg) | ||||||
|  |   # mode 1 | ||||||
|  |   $verbosity = LOW | ||||||
|  |  | ||||||
|  |   if ['@', nil].include?($versions[:hg]) | ||||||
|  |     versions = $checks.last | ||||||
|  |     versions[:hg] = $versions[:hg] if $versions[:hg] | ||||||
|  |   else | ||||||
|  |     versions = $checks.find { |e| e[:hg] == $versions[:hg] } | ||||||
|  |     exit 1 unless versions | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   exit check(versions) ? 0 : 1 | ||||||
|  | elsif not $versions.empty? | ||||||
|  |   # mode 2 | ||||||
|  |   $verbosity = HIGH | ||||||
|  |  | ||||||
|  |   exit check(versions) ? 0 : 1 | ||||||
|  | else | ||||||
|  |   # mode 3 | ||||||
|  |   $verbosity = QUIET | ||||||
|  |  | ||||||
|  |   at_exit do | ||||||
|  |     File.open("#{__dir__}/results.txt", 'w') do |f| | ||||||
|  |       store_results(f) | ||||||
|  |     end | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   failures = 0 | ||||||
|  |  | ||||||
|  |   $checks.each do |versions| | ||||||
|  |     result = check(versions) | ||||||
|  |     failures += 1 unless result | ||||||
|  |     $results << [versions, result] | ||||||
|  |   end | ||||||
|  |  | ||||||
|  |   exit 1 unless failures == 0 | ||||||
|  | end | ||||||
							
								
								
									
										15
									
								
								tools/hg_setup_hack_2.4.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								tools/hg_setup_hack_2.4.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | diff --git a/setup.py b/setup.py | ||||||
|  | --- a/setup.py | ||||||
|  | +++ b/setup.py | ||||||
|  | @@ -181,7 +181,10 @@ | ||||||
|  |      # error 0xc0150004. See: http://bugs.python.org/issue3440 | ||||||
|  |      env['SystemRoot'] = os.environ['SystemRoot'] | ||||||
|  |   | ||||||
|  | -if os.path.isdir('.hg'): | ||||||
|  | +if os.path.exists('.hg_force_version'): | ||||||
|  | +    with open('.hg_force_version') as f: | ||||||
|  | +        version = f.read().rstrip('\n') | ||||||
|  | +elif os.path.isdir('.hg'): | ||||||
|  |      cmd = [sys.executable, 'hg', 'log', '-r', '.', '--template', '{tags}\n'] | ||||||
|  |      numerictags = [t for t in runhg(cmd, env).split() if t[0].isdigit()] | ||||||
|  |      hgid = runhg([sys.executable, 'hg', 'id', '-i'], env).strip() | ||||||
							
								
								
									
										22
									
								
								tools/hggit_rename_fix_0.8.0.patch
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								tools/hggit_rename_fix_0.8.0.patch
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | diff --git a/hggit/git_handler.py b/hggit/git_handler.py | ||||||
|  | --- a/hggit/git_handler.py | ||||||
|  | +++ b/hggit/git_handler.py | ||||||
|  | @@ -693,6 +693,8 @@ | ||||||
|  |      def import_git_commit(self, commit): | ||||||
|  |          self.ui.debug(_("importing: %s\n") % commit.id) | ||||||
|  |   | ||||||
|  | +        extra_in_message = self.ui.configbool('git', 'debugextrainmessage', False) | ||||||
|  | + | ||||||
|  |          detect_renames = False | ||||||
|  |          (strip_message, hg_renames, | ||||||
|  |           hg_branch, extra) = git2hg.extract_hg_metadata( | ||||||
|  | @@ -703,7 +705,8 @@ | ||||||
|  |              # renames detected from Git. This is because we export an extra | ||||||
|  |              # 'HG:rename-source' Git parameter when this isn't set, which will | ||||||
|  |              # break bidirectionality. | ||||||
|  | -            extra['hg-git-rename-source'] = 'git' | ||||||
|  | +            if not extra_in_message: | ||||||
|  | +                extra['hg-git-rename-source'] = 'git' | ||||||
|  |          else: | ||||||
|  |              renames = hg_renames | ||||||
|  |   | ||||||
							
								
								
									
										33
									
								
								tools/versions.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								tools/versions.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | |||||||
|  | # vi: ft=ruby | ||||||
|  |  | ||||||
|  | hg:2.4 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  | hg:2.5 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  | hg:2.6 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  | hg:2.7 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  | hg:2.8 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  | hg:2.9 hggit:0.4.0 dulwich:0.9.0 # 2013_02 | ||||||
|  |  | ||||||
|  | hg:3.0 hggit:0.7.0 dulwich:0.10.0 # 2014_11 | ||||||
|  | hg:3.1 hggit:0.7.0 dulwich:0.10.0 # 2014_11 | ||||||
|  | hg:3.2 hggit:0.7.0 dulwich:0.10.0 # 2014_11 | ||||||
|  |  | ||||||
|  | hg:3.3 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.4 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.5 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.6 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.7 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.8 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  | hg:3.9 hggit:0.8.4 dulwich:0.13.0 # 2016_01 | ||||||
|  |  | ||||||
|  | hg:4.0 hggit:0.8.10 dulwich:0.18.0 # 2017_11 | ||||||
|  | hg:4.1 hggit:0.8.10 dulwich:0.18.0 # 2017_11 | ||||||
|  | hg:4.2 hggit:0.8.10 dulwich:0.18.0 # 2017_11 | ||||||
|  | hg:4.3 hggit:0.8.10 dulwich:0.18.0 # 2017_11 | ||||||
|  | hg:4.4 hggit:0.8.10 dulwich:0.18.0 # 2017_11 | ||||||
|  |  | ||||||
|  | hg:4.5 hggit:0.8.11 dulwich:0.18.0 # 2018_02 | ||||||
|  | hg:4.6 hggit:0.8.12 dulwich:0.19.7 # 2018_10 | ||||||
|  | hg:4.7 hggit:0.8.12 dulwich:0.19.7 # 2018_10 | ||||||
|  | hg:4.8 hggit:@ dulwich:0.19.11 | ||||||
|  | hg:4.9 hggit:@ dulwich:0.19.11 | ||||||
|  | hg:5.0 hggit:@ dulwich:0.19.11 | ||||||
		Reference in New Issue
	
	Block a user