| # Copyright (C) 2009 Google Inc. All rights reserved. |
| # Copyright (C) 2009, 2016 Apple Inc. All rights reserved. |
| # Copyright (C) 2011 Daniel Bates (dbates@intudata.com). All rights reserved. |
| # |
| # Redistribution and use in source and binary forms, with or without |
| # modification, are permitted provided that the following conditions are |
| # met: |
| # |
| # * Redistributions of source code must retain the above copyright |
| # notice, this list of conditions and the following disclaimer. |
| # * Redistributions in binary form must reproduce the above |
| # copyright notice, this list of conditions and the following disclaimer |
| # in the documentation and/or other materials provided with the |
| # distribution. |
| # * Neither the name of Google Inc. nor the names of its |
| # contributors may be used to endorse or promote products derived from |
| # this software without specific prior written permission. |
| # |
| # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| # "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| # LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| # A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| # OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| # SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| # LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| # DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| # THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| # (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| # OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| import atexit |
| import base64 |
| import codecs |
| import getpass |
| import os |
| import os.path |
| import re |
| import stat |
| import sys |
| import subprocess |
| import tempfile |
| import time |
| import urllib |
| import shutil |
| import unittest |
| |
| from datetime import date |
| from webkitpy.common.checkout.checkout import Checkout |
| from webkitpy.common.config.committers import Committer # FIXME: This should not be needed |
| from webkitpy.common.net.bugzilla import Attachment # FIXME: This should not be needed |
| from webkitpy.common.system.executive import Executive, ScriptError |
| from webkitpy.common.system.filesystem_mock import MockFileSystem |
| from webkitpy.common.system.outputcapture import OutputCapture |
| from webkitpy.common.system.executive_mock import MockExecutive |
| from webkitpy.common.version import Version |
| from webkitpy.common.checkout.scm.git import Git, AmbiguousCommitError |
| from webkitpy.common.checkout.scm.detection import detect_scm_system |
| from webkitpy.common.checkout.scm.scm import SCM, CheckoutNeedsUpdate, commit_error_handler, AuthenticationError |
| from webkitpy.common.checkout.scm.svn import SVN |
| from webkitpy.common.unicode_compatibility import decode_if_necessary |
| |
| |
| # We cache the mock SVN repo so that we don't create it again for each call to an SVNTest or GitTest test_ method. |
| # We store it in a global variable so that we can delete this cached repo on exit(3). |
| # FIXME: Remove this once test-webkitpy supports class and module fixtures (i.e. setUpClass()/setUpModule() |
| # are called exactly once per class/module). |
| cached_svn_repo_path = None |
| |
| |
| def remove_dir(path): |
| # Change directory to / to ensure that we aren't in the directory we want to delete. |
| os.chdir('/') |
| shutil.rmtree(path) |
| |
| |
| # FIXME: Remove this once test-webkitpy supports class and module fixtures (i.e. setUpClass()/setUpModule() |
| # are called exactly once per class/module). |
| @atexit.register |
| def delete_cached_mock_repo_at_exit(): |
| if cached_svn_repo_path: |
| remove_dir(cached_svn_repo_path) |
| |
| # Eventually we will want to write tests which work for both scms. (like update_webkit, changed_files, etc.) |
| # Perhaps through some SCMTest base-class which both SVNTest and GitTest inherit from. |
| |
| |
| def run_command(*args, **kwargs): |
| # FIXME: This should not be a global static. |
| # New code should use Executive.run_command directly instead |
| return Executive().run_command(*args, **kwargs) |
| |
| |
| # FIXME: This should be unified into one of the executive.py commands! |
| # Callers could use run_and_throw_if_fail(args, cwd=cwd, quiet=True) |
| def run_silent(args, cwd=None): |
| # Note: Not thread safe: http://bugs.python.org/issue2320 |
| process = subprocess.Popen(args, stdout=subprocess.PIPE, stderr=subprocess.PIPE, cwd=cwd) |
| process.communicate() # ignore output |
| exit_code = process.wait() |
| if exit_code: |
| raise ScriptError('Failed to run "%s" exit_code: %d cwd: %s' % (args, exit_code, cwd)) |
| |
| |
| def write_into_file_at_path(file_path, contents, encoding="utf-8"): |
| if encoding: |
| with codecs.open(file_path, "w", encoding) as file: |
| return file.write(contents) |
| with open(file_path, "wb") as file: |
| return file.write(contents) |
| |
| |
| def read_from_path(file_path, encoding="utf-8"): |
| if not encoding: |
| with codecs.open(file_path, "rb") as file: |
| return file.read() |
| |
| with codecs.open(file_path, "r", encoding) as file: |
| return file.read() |
| |
| |
| def _make_diff(command, *args): |
| # We use this wrapper to disable output decoding. diffs should be treated as |
| # binary files since they may include text files of multiple differnet encodings. |
| # FIXME: This should use an Executive. |
| return run_command([command, "diff"] + list(args), decode_output=False) |
| |
| |
| def _svn_diff(*args): |
| return _make_diff("svn", *args) |
| |
| |
| def _git_diff(*args): |
| return _make_diff("git", *args) |
| |
| |
| # Exists to share svn repository creation code between the git and svn tests |
| class SVNTestRepository(object): |
| @classmethod |
| def _svn_add(cls, path): |
| run_command(["svn", "add", path]) |
| |
| @classmethod |
| def _svn_commit(cls, message): |
| run_command(["svn", "commit", "--quiet", "--message", message]) |
| |
| @classmethod |
| def _setup_test_commits(cls, svn_repo_url): |
| |
| svn_checkout_path = tempfile.mkdtemp(suffix="svn_test_checkout") |
| run_command(['svn', 'checkout', '--quiet', svn_repo_url, svn_checkout_path]) |
| |
| # Add some test commits |
| os.chdir(svn_checkout_path) |
| |
| write_into_file_at_path("test_file", "test1") |
| cls._svn_add("test_file") |
| cls._svn_commit("initial commit") |
| |
| write_into_file_at_path("test_file", "test1test2") |
| # This used to be the last commit, but doing so broke |
| # GitTest.test_apply_git_patch which use the inverse diff of the last commit. |
| # svn-apply fails to remove directories in Git, see: |
| # https://bugs.webkit.org/show_bug.cgi?id=34871 |
| os.mkdir("test_dir") |
| # Slash should always be the right path separator since we use cygwin on Windows. |
| test_file3_path = "test_dir/test_file3" |
| write_into_file_at_path(test_file3_path, "third file") |
| cls._svn_add("test_dir") |
| cls._svn_commit("second commit") |
| |
| write_into_file_at_path("test_file", "test1test2test3\n") |
| write_into_file_at_path("test_file2", "second file") |
| cls._svn_add("test_file2") |
| cls._svn_commit("third commit") |
| |
| # This 4th commit is used to make sure that our patch file handling |
| # code correctly treats patches as binary and does not attempt to |
| # decode them assuming they're utf-8. |
| write_into_file_at_path("test_file", u"latin1 test: \u00A0\n", "latin1") |
| write_into_file_at_path("test_file2", u"utf-8 test: \u00A0\n", "utf-8") |
| cls._svn_commit("fourth commit") |
| |
| # svn does not seem to update after commit as I would expect. |
| run_command(['svn', 'update']) |
| remove_dir(svn_checkout_path) |
| |
| # This is a hot function since it's invoked by unittest before calling each test_ method in SVNTest and |
| # GitTest. We create a mock SVN repo once and then perform an SVN checkout from a filesystem copy of |
| # it since it's expensive to create the mock repo. |
| @classmethod |
| def setup(cls, test_object): |
| global cached_svn_repo_path |
| if not cached_svn_repo_path: |
| cached_svn_repo_path = cls._setup_mock_repo() |
| |
| test_object.temp_directory = os.path.realpath(tempfile.mkdtemp(suffix="svn_test")) |
| test_object.svn_repo_path = os.path.join(test_object.temp_directory, "repo") |
| test_object.svn_repo_url = "file://%s" % test_object.svn_repo_path |
| test_object.svn_checkout_path = os.path.join(test_object.temp_directory, "checkout") |
| shutil.copytree(cached_svn_repo_path, test_object.svn_repo_path) |
| run_command(['svn', 'checkout', '--quiet', test_object.svn_repo_url + "/trunk", test_object.svn_checkout_path]) |
| |
| @classmethod |
| def _setup_mock_repo(cls): |
| # Create an test SVN repository |
| svn_repo_path = tempfile.mkdtemp(suffix="svn_test_repo") |
| svn_repo_url = "file://%s" % svn_repo_path # Not sure this will work on windows |
| # git svn complains if we don't pass --pre-1.5-compatible, not sure why: |
| # Expected FS format '2'; found format '3' at /usr/local/libexec/git-core//git-svn line 1477 |
| run_command(['svnadmin', 'create', '--pre-1.5-compatible', svn_repo_path]) |
| |
| # Create a test svn checkout |
| svn_checkout_path = tempfile.mkdtemp(suffix="svn_test_checkout") |
| run_command(['svn', 'checkout', '--quiet', svn_repo_url, svn_checkout_path]) |
| |
| # Create and checkout a trunk dir to match the standard svn configuration to match git-svn's expectations |
| os.chdir(svn_checkout_path) |
| os.mkdir('trunk') |
| cls._svn_add('trunk') |
| # We can add tags and branches as well if we ever need to test those. |
| cls._svn_commit('add trunk') |
| |
| # Change directory out of the svn checkout so we can delete the checkout directory. |
| remove_dir(svn_checkout_path) |
| |
| cls._setup_test_commits(svn_repo_url + "/trunk") |
| return svn_repo_path |
| |
| @classmethod |
| def tear_down(cls, test_object): |
| remove_dir(test_object.temp_directory) |
| |
| # Now that we've deleted the checkout paths, cwddir may be invalid |
| # Change back to a valid directory so that later calls to os.getcwd() do not fail. |
| if os.path.isabs(__file__): |
| path = os.path.dirname(__file__) |
| else: |
| path = sys.path[0] |
| os.chdir(detect_scm_system(path).checkout_root) |
| |
| |
| # For testing the SCM baseclass directly. |
| class SCMClassTests(unittest.TestCase): |
| def setUp(self): |
| self.dev_null = open(os.devnull, "w") # Used to make our Popen calls quiet. |
| |
| def tearDown(self): |
| self.dev_null.close() |
| |
| def test_run_command_with_pipe(self): |
| input_process = subprocess.Popen(['echo', 'foo\nbar'], stdout=subprocess.PIPE, stderr=self.dev_null) |
| self.assertEqual(run_command(['grep', 'bar'], input=input_process.stdout), "bar\n") |
| |
| # Test the non-pipe case too: |
| self.assertEqual(run_command(['grep', 'bar'], input="foo\nbar"), "bar\n") |
| |
| command_returns_non_zero = ['/bin/sh', '--invalid-option'] |
| # Test when the input pipe process fails. |
| input_process = subprocess.Popen(command_returns_non_zero, stdout=subprocess.PIPE, stderr=self.dev_null) |
| self.assertNotEqual(input_process.poll(), 0) |
| self.assertRaises(ScriptError, run_command, ['grep', 'bar'], input=input_process.stdout) |
| |
| # Test when the run_command process fails. |
| input_process = subprocess.Popen(['echo', 'foo\nbar'], stdout=subprocess.PIPE, stderr=self.dev_null) # grep shows usage and calls exit(2) when called w/o arguments. |
| self.assertRaises(ScriptError, run_command, command_returns_non_zero, input=input_process.stdout) |
| |
| def test_error_handlers(self): |
| git_failure_message = "Merge conflict during commit: Your file or directory 'WebCore/ChangeLog' is probably out-of-date: resource out of date; try updating at /usr/local/libexec/git-core//git-svn line 469" |
| svn_failure_message = """svn: Commit failed (details follow): |
| svn: File or directory 'ChangeLog' is out of date; try updating |
| svn: resource out of date; try updating |
| """ |
| command_does_not_exist = ['does_not_exist', 'invalid_option'] |
| self.assertRaises(OSError, run_command, command_does_not_exist) |
| self.assertRaises(OSError, run_command, command_does_not_exist, ignore_errors=True) |
| |
| command_returns_non_zero = ['/bin/sh', '--invalid-option'] |
| self.assertRaises(ScriptError, run_command, command_returns_non_zero) |
| # Check if returns error text: |
| self.assertTrue(run_command(command_returns_non_zero, ignore_errors=True)) |
| |
| self.assertRaises(CheckoutNeedsUpdate, commit_error_handler, ScriptError(output=git_failure_message)) |
| self.assertRaises(CheckoutNeedsUpdate, commit_error_handler, ScriptError(output=svn_failure_message)) |
| self.assertRaises(ScriptError, commit_error_handler, ScriptError(output='blah blah blah')) |
| |
| |
| # GitTest and SVNTest inherit from this so any test_ methods here will be run once for this class and then once for each subclass. |
| class SCMTest(unittest.TestCase): |
| def __init__(self, *args, **kwargs): |
| super(SCMTest, self).__init__(*args, **kwargs) |
| self.scm = None |
| |
| def _create_patch(self, patch_contents): |
| # FIXME: This code is brittle if the Attachment API changes. |
| attachment = Attachment({"bug_id": 12345}, None) |
| attachment.contents = lambda: patch_contents |
| |
| joe_cool = Committer("Joe Cool", "joe@cool.com") |
| attachment.reviewer = lambda: joe_cool |
| |
| return attachment |
| |
| def _setup_webkittools_scripts_symlink(self, local_scm): |
| webkit_scm = detect_scm_system(os.path.dirname(os.path.abspath(__file__))) |
| webkit_scripts_directory = Checkout(webkit_scm, webkit_scm._executive, webkit_scm._filesystem).scripts_directory() |
| local_scripts_directory = Checkout(local_scm, local_scm._executive, local_scm._filesystem).scripts_directory() |
| os.mkdir(os.path.dirname(local_scripts_directory)) |
| os.symlink(webkit_scripts_directory, local_scripts_directory) |
| |
| # Tests which both GitTest and SVNTest should run. |
| # FIXME: There must be a simpler way to add these w/o adding a wrapper method to both subclasses |
| |
| def _shared_test_changed_files(self): |
| write_into_file_at_path("test_file", "changed content") |
| self.assertEqual(self.scm.changed_files(), ["test_file"]) |
| write_into_file_at_path("test_dir/test_file3", "new stuff") |
| self.assertEqual(sorted(self.scm.changed_files()), sorted(["test_dir/test_file3", "test_file"])) |
| old_cwd = os.getcwd() |
| os.chdir("test_dir") |
| # Validate that changed_files does not change with our cwd, see bug 37015. |
| self.assertEqual(sorted(self.scm.changed_files()), sorted(["test_dir/test_file3", "test_file"])) |
| os.chdir(old_cwd) |
| |
| def _shared_test_untracked_files(self, scm): |
| write_into_file_at_path("test_file_new", "new content") |
| self.assertEqual(scm.untracked_files(), ["test_file_new"]) |
| |
| os.mkdir("test_dir_new") |
| write_into_file_at_path(os.path.join("test_dir_new", "test_file_new"), "new stuff") |
| self.assertEqual(sorted(scm.untracked_files()), sorted(["test_dir_new", "test_file_new"])) |
| |
| old_cwd = os.getcwd() |
| os.chdir("test_dir_new") |
| # Validate that untracked_files do not change with our cwd. |
| self.assertEqual(scm.untracked_files(), ["test_dir_new", "test_file_new"]) |
| |
| os.chdir(old_cwd) |
| |
| write_into_file_at_path("test_file_new.pyc", "new ignored file") |
| self.assertEqual(sorted(scm.untracked_files()), sorted(["test_dir_new", "test_file_new"])) |
| self.assertEqual(sorted(scm.untracked_files(include_ignored_files=True)), sorted(["test_file_new.pyc", "test_dir_new", "test_file_new"])) |
| |
| shutil.rmtree("test_dir_new") |
| os.remove("test_file_new") |
| os.remove("test_file_new.pyc") |
| |
| def _shared_test_discard_untracked_files(self, scm): |
| write_into_file_at_path("test_file_new", "new content") |
| os.mkdir("test_dir_new") |
| write_into_file_at_path("test_dir_new/test_file_new", "new stuff") |
| self.assertEqual(sorted(scm.untracked_files()), sorted(["test_dir_new", "test_file_new"])) |
| scm.discard_untracked_files() |
| self.assertEqual(scm.untracked_files(), []) |
| |
| write_into_file_at_path("test_file_new.pyc", "new content") |
| self.assertEqual(scm.untracked_files(True), ["test_file_new.pyc"]) |
| scm.discard_untracked_files(discard_ignored_files=False) |
| self.assertEqual(scm.untracked_files(True), ["test_file_new.pyc"]) |
| scm.discard_untracked_files(discard_ignored_files=True) |
| self.assertEqual(scm.untracked_files(True), []) |
| |
| os.mkdir("WebKitBuild") |
| self.assertEqual(scm.untracked_files(True), ["WebKitBuild"]) |
| scm.discard_untracked_files(discard_ignored_files=True, keep_webkitbuild_directory=True) |
| self.assertEqual(scm.untracked_files(True), ["WebKitBuild"]) |
| scm.discard_untracked_files(discard_ignored_files=True, keep_webkitbuild_directory=False) |
| self.assertEqual(scm.untracked_files(True), []) |
| |
| if os.path.isdir("test_dir_new"): |
| shutil.rmtree("test_dir_new") |
| if os.path.isfile("test_file_new"): |
| os.remove("test_file_new") |
| if os.path.isfile("test_file_new.pyc"): |
| os.remove("test_file_new.pyc") |
| |
| def _shared_test_added_files(self): |
| write_into_file_at_path("test_file", "changed content") |
| self.assertEqual(self.scm.added_files(), []) |
| |
| write_into_file_at_path("added_file", "new stuff") |
| self.scm.add("added_file") |
| |
| write_into_file_at_path("added_file3", "more new stuff") |
| write_into_file_at_path("added_file4", "more new stuff") |
| self.scm.add_list(["added_file3", "added_file4"]) |
| |
| os.mkdir("added_dir") |
| write_into_file_at_path("added_dir/added_file2", "new stuff") |
| self.scm.add("added_dir") |
| |
| # SVN reports directory changes, Git does not. |
| added_files = self.scm.added_files() |
| if "added_dir" in added_files: |
| added_files.remove("added_dir") |
| self.assertEqual(sorted(added_files), sorted(["added_dir/added_file2", "added_file", "added_file3", "added_file4"])) |
| |
| # Test also to make sure discard_working_directory_changes removes added files |
| self.scm.discard_working_directory_changes() |
| self.assertEqual(self.scm.added_files(), []) |
| self.assertFalse(os.path.exists("added_file")) |
| self.assertFalse(os.path.exists("added_file3")) |
| self.assertFalse(os.path.exists("added_file4")) |
| self.assertFalse(os.path.exists("added_dir")) |
| |
| def _shared_test_changed_files_for_revision(self): |
| # SVN reports directory changes, Git does not. |
| changed_files = self.scm.changed_files_for_revision(3) |
| if "test_dir" in changed_files: |
| changed_files.remove("test_dir") |
| self.assertEqual(sorted(changed_files), sorted(["test_dir/test_file3", "test_file"])) |
| self.assertEqual(sorted(self.scm.changed_files_for_revision(4)), sorted(["test_file", "test_file2"])) # Git and SVN return different orders. |
| self.assertEqual(self.scm.changed_files_for_revision(2), ["test_file"]) |
| |
| def _shared_test_contents_at_revision(self): |
| self.assertEqual(self.scm.contents_at_revision("test_file", 3), b"test1test2") |
| self.assertEqual(self.scm.contents_at_revision("test_file", 4), b"test1test2test3\n") |
| |
| # Verify that contents_at_revision returns a byte array, aka str(): |
| self.assertEqual(self.scm.contents_at_revision("test_file", 5), u"latin1 test: \u00A0\n".encode("latin1")) |
| self.assertEqual(self.scm.contents_at_revision("test_file2", 5), u"utf-8 test: \u00A0\n".encode("utf-8")) |
| |
| self.assertEqual(self.scm.contents_at_revision("test_file2", 4), b"second file") |
| # Files which don't exist: |
| # Currently we raise instead of returning None because detecting the difference between |
| # "file not found" and any other error seems impossible with svn (git seems to expose such through the return code). |
| self.assertRaises(ScriptError, self.scm.contents_at_revision, b"test_file2", 2) |
| self.assertRaises(ScriptError, self.scm.contents_at_revision, b"does_not_exist", 2) |
| |
| def _shared_test_revisions_changing_file(self): |
| self.assertEqual(sorted(self.scm.revisions_changing_file("test_file")), sorted([5, 4, 3, 2])) |
| self.assertEqual(self.scm.revisions_changing_file("non_existent_file"), []) |
| |
| def _shared_test_committer_email_for_revision(self): |
| self.assertEqual(self.scm.committer_email_for_revision(3), getpass.getuser()) # Committer "email" will be the current user |
| |
| def _shared_test_reverse_diff(self): |
| self._setup_webkittools_scripts_symlink(self.scm) # Git's apply_reverse_diff uses resolve-ChangeLogs. |
| # Only test the simple case, as any other will end up with conflict markers. |
| self.scm.apply_reverse_diff('5') |
| self.assertEqual(read_from_path('test_file'), "test1test2test3\n") |
| |
| def _shared_test_diff_for_revision(self): |
| # Patch formats are slightly different between svn and git, so just regexp for things we know should be there. |
| r3_patch = self.scm.diff_for_revision(4) |
| self.assertRegexpMatches(r3_patch, b'test3') |
| self.assertNotRegexpMatches(r3_patch, b'test4') |
| self.assertRegexpMatches(r3_patch, b'test2') |
| self.assertRegexpMatches(self.scm.diff_for_revision(3), b'test2') |
| |
| def _shared_test_svn_apply_git_patch(self): |
| self._setup_webkittools_scripts_symlink(self.scm) |
| git_binary_addition = """diff --git a/fizzbuzz7.gif b/fizzbuzz7.gif |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..64a9532e7794fcd791f6f12157406d90 |
| 60151690 |
| GIT binary patch |
| literal 512 |
| zcmZ?wbhEHbRAx|MU|?iW{Kxc~?KofD;ckY;H+&5HnHl!!GQMD7h+sU{_)e9f^V3c? |
| zhJP##HdZC#4K}7F68@!1jfWQg2daCm-gs#3|JREDT>c+pG4L<_2;w##WMO#ysPPap |
| zLqpAf1OE938xAsSp4!5f-o><?VKe(#0jEcwfHGF4%M1^kRs14oVBp2ZEL{E1N<-zJ |
| zsfLmOtKta;2_;2c#^S1-8cf<nb!QnGl>c!Xe6RXvrEtAWBvSDTgTO1j3vA31Puw!A |
| zs(87q)j_mVDTqBo-P+03-P5mHCEnJ+x}YdCuS7#bCCyePUe(ynK+|4b-3qK)T?Z&) |
| zYG+`tl4h?GZv_$t82}X4*DTE|$;{DEiPyF@)U-1+FaX++T9H{&%cag`W1|zVP@`%b |
| zqiSkp6{BTpWTkCr!=<C6Q=?#~R8^JfrliAF6Q^gV9Iup8RqCXqqhqC`qsyhk<-nlB |
| z00f{QZvfK&|Nm#oZ0TQl`Yr$BIa6A@16O26ud7H<QM=xl`toLKnz-3h@9c9q&wm|X |
| z{89I|WPyD!*M?gv?q`;L=2YFeXrJQNti4?}s!zFo=5CzeBxC69xA<zrjP<wUcCRh4 |
| ptUl-ZG<%a~#LwkIWv&q!KSCH7tQ8cJDiw+|GV?MN)RjY50RTb-xvT&H |
| |
| literal 0 |
| HcmV?d00001 |
| |
| """ |
| self.checkout.apply_patch(self._create_patch(git_binary_addition)) |
| added = read_from_path('fizzbuzz7.gif', encoding=None) |
| self.assertEqual(512, len(added)) |
| self.assertTrue(added.startswith(b'GIF89a')) |
| self.assertIn('fizzbuzz7.gif', self.scm.changed_files()) |
| |
| # The file already exists. |
| self.assertRaises(ScriptError, self.checkout.apply_patch, self._create_patch(git_binary_addition)) |
| |
| git_binary_modification = """diff --git a/fizzbuzz7.gif b/fizzbuzz7.gif |
| index 64a9532e7794fcd791f6f12157406d9060151690..323fae03f4606ea9991df8befbb2fca7 |
| GIT binary patch |
| literal 7 |
| OcmYex&reD$;sO8*F9L)B |
| |
| literal 512 |
| zcmZ?wbhEHbRAx|MU|?iW{Kxc~?KofD;ckY;H+&5HnHl!!GQMD7h+sU{_)e9f^V3c? |
| zhJP##HdZC#4K}7F68@!1jfWQg2daCm-gs#3|JREDT>c+pG4L<_2;w##WMO#ysPPap |
| zLqpAf1OE938xAsSp4!5f-o><?VKe(#0jEcwfHGF4%M1^kRs14oVBp2ZEL{E1N<-zJ |
| zsfLmOtKta;2_;2c#^S1-8cf<nb!QnGl>c!Xe6RXvrEtAWBvSDTgTO1j3vA31Puw!A |
| zs(87q)j_mVDTqBo-P+03-P5mHCEnJ+x}YdCuS7#bCCyePUe(ynK+|4b-3qK)T?Z&) |
| zYG+`tl4h?GZv_$t82}X4*DTE|$;{DEiPyF@)U-1+FaX++T9H{&%cag`W1|zVP@`%b |
| zqiSkp6{BTpWTkCr!=<C6Q=?#~R8^JfrliAF6Q^gV9Iup8RqCXqqhqC`qsyhk<-nlB |
| z00f{QZvfK&|Nm#oZ0TQl`Yr$BIa6A@16O26ud7H<QM=xl`toLKnz-3h@9c9q&wm|X |
| z{89I|WPyD!*M?gv?q`;L=2YFeXrJQNti4?}s!zFo=5CzeBxC69xA<zrjP<wUcCRh4 |
| ptUl-ZG<%a~#LwkIWv&q!KSCH7tQ8cJDiw+|GV?MN)RjY50RTb-xvT&H |
| |
| """ |
| self.checkout.apply_patch(self._create_patch(git_binary_modification)) |
| modified = read_from_path('fizzbuzz7.gif', encoding=None) |
| self.assertEqual(b'foobar\n', modified) |
| self.assertIn('fizzbuzz7.gif', self.scm.changed_files()) |
| |
| # Applying the same modification should fail. |
| self.assertRaises(ScriptError, self.checkout.apply_patch, self._create_patch(git_binary_modification)) |
| |
| git_binary_deletion = """diff --git a/fizzbuzz7.gif b/fizzbuzz7.gif |
| deleted file mode 100644 |
| index 323fae0..0000000 |
| GIT binary patch |
| literal 0 |
| HcmV?d00001 |
| |
| literal 7 |
| OcmYex&reD$;sO8*F9L)B |
| |
| """ |
| self.checkout.apply_patch(self._create_patch(git_binary_deletion)) |
| self.assertFalse(os.path.exists('fizzbuzz7.gif')) |
| self.assertNotIn('fizzbuzz7.gif', self.scm.changed_files()) |
| |
| # Cannot delete again. |
| self.assertRaises(ScriptError, self.checkout.apply_patch, self._create_patch(git_binary_deletion)) |
| |
| def _shared_test_add_recursively(self): |
| os.mkdir("added_dir") |
| write_into_file_at_path("added_dir/added_file", "new stuff") |
| self.scm.add("added_dir/added_file") |
| self.assertIn("added_dir/added_file", self.scm.added_files()) |
| |
| def _shared_test_delete_recursively(self): |
| os.mkdir("added_dir") |
| write_into_file_at_path("added_dir/added_file", "new stuff") |
| self.scm.add("added_dir/added_file") |
| self.assertIn("added_dir/added_file", self.scm.added_files()) |
| self.scm.delete("added_dir/added_file") |
| self.assertNotIn("added_dir", self.scm.added_files()) |
| |
| def _shared_test_delete_recursively_or_not(self): |
| os.mkdir("added_dir") |
| write_into_file_at_path("added_dir/added_file", "new stuff") |
| write_into_file_at_path("added_dir/another_added_file", "more new stuff") |
| self.scm.add("added_dir/added_file") |
| self.scm.add("added_dir/another_added_file") |
| self.assertIn("added_dir/added_file", self.scm.added_files()) |
| self.assertIn("added_dir/another_added_file", self.scm.added_files()) |
| self.scm.delete("added_dir/added_file") |
| self.assertIn("added_dir/another_added_file", self.scm.added_files()) |
| |
| def _shared_test_exists(self, scm, commit_function): |
| os.chdir(scm.checkout_root) |
| self.assertFalse(scm.exists('foo.txt')) |
| write_into_file_at_path('foo.txt', 'some stuff') |
| self.assertFalse(scm.exists('foo.txt')) |
| scm.add('foo.txt') |
| commit_function('adding foo') |
| self.assertTrue(scm.exists('foo.txt')) |
| scm.delete('foo.txt') |
| commit_function('deleting foo') |
| self.assertFalse(scm.exists('foo.txt')) |
| |
| def _shared_test_head_svn_revision(self): |
| self.assertEqual(self.scm.head_svn_revision(), '5') |
| |
| def _shared_test_svn_revision(self, scm): |
| self.assertEqual(scm.svn_revision(scm.checkout_root), '5') |
| |
| def _shared_test_svn_branch(self, scm): |
| self.assertEqual(scm.svn_branch(scm.checkout_root), 'trunk') |
| |
| # Context manager that overrides the current timezone. |
| class TimezoneOverride(object): |
| def __init__(self, timezone_string): |
| self._timezone_string = timezone_string |
| |
| def __enter__(self): |
| if hasattr(time, 'tzset'): |
| self._saved_timezone = os.environ.get('TZ', None) |
| os.environ['TZ'] = self._timezone_string |
| time.tzset() |
| |
| def __exit__(self, type, value, traceback): |
| if hasattr(time, 'tzset'): |
| if self._saved_timezone: |
| os.environ['TZ'] = self._saved_timezone |
| else: |
| del os.environ['TZ'] |
| time.tzset() |
| |
| |
| class SVNTest(SCMTest): |
| |
| @staticmethod |
| def _set_date_and_reviewer(changelog_entry): |
| # Joe Cool matches the reviewer set in SCMTest._create_patch |
| changelog_entry = changelog_entry.replace('REVIEWER_HERE', 'Joe Cool') |
| # svn-apply will update ChangeLog entries with today's date (as in Cupertino, CA, US) |
| with TimezoneOverride('PST8PDT'): |
| return changelog_entry.replace('DATE_HERE', date.today().isoformat()) |
| |
| def test_svn_apply(self): |
| first_entry = """2009-10-26 Eric Seidel <eric@webkit.org> |
| |
| Reviewed by Foo Bar. |
| |
| Most awesome change ever. |
| |
| * scm_unittest.py: |
| """ |
| intermediate_entry = """2009-10-27 Eric Seidel <eric@webkit.org> |
| |
| Reviewed by Baz Bar. |
| |
| A more awesomer change yet! |
| |
| * scm_unittest.py: |
| """ |
| one_line_overlap_patch = """Index: ChangeLog |
| =================================================================== |
| --- ChangeLog (revision 5) |
| +++ ChangeLog (working copy) |
| @@ -1,5 +1,13 @@ |
| 2009-10-26 Eric Seidel <eric@webkit.org> |
| %(whitespace)s |
| + Reviewed by NOBODY (OOPS!). |
| + |
| + Second most awesome change ever. |
| + |
| + * scm_unittest.py: |
| + |
| +2009-10-26 Eric Seidel <eric@webkit.org> |
| + |
| Reviewed by Foo Bar. |
| %(whitespace)s |
| Most awesome change ever. |
| """ % {'whitespace': ' '} |
| one_line_overlap_entry = """DATE_HERE Eric Seidel <eric@webkit.org> |
| |
| Reviewed by REVIEWER_HERE. |
| |
| Second most awesome change ever. |
| |
| * scm_unittest.py: |
| """ |
| two_line_overlap_patch = """Index: ChangeLog |
| =================================================================== |
| --- ChangeLog (revision 5) |
| +++ ChangeLog (working copy) |
| @@ -2,6 +2,14 @@ |
| %(whitespace)s |
| Reviewed by Foo Bar. |
| %(whitespace)s |
| + Second most awesome change ever. |
| + |
| + * scm_unittest.py: |
| + |
| +2009-10-26 Eric Seidel <eric@webkit.org> |
| + |
| + Reviewed by Foo Bar. |
| + |
| Most awesome change ever. |
| %(whitespace)s |
| * scm_unittest.py: |
| """ % {'whitespace': ' '} |
| two_line_overlap_entry = """DATE_HERE Eric Seidel <eric@webkit.org> |
| |
| Reviewed by Foo Bar. |
| |
| Second most awesome change ever. |
| |
| * scm_unittest.py: |
| """ |
| write_into_file_at_path('ChangeLog', first_entry) |
| run_command(['svn', 'add', 'ChangeLog']) |
| run_command(['svn', 'commit', '--quiet', '--message', 'ChangeLog commit']) |
| |
| # Patch files were created against just 'first_entry'. |
| # Add a second commit to make svn-apply have to apply the patches with fuzz. |
| changelog_contents = "%s\n%s" % (intermediate_entry, first_entry) |
| write_into_file_at_path('ChangeLog', changelog_contents) |
| run_command(['svn', 'commit', '--quiet', '--message', 'Intermediate commit']) |
| |
| self._setup_webkittools_scripts_symlink(self.scm) |
| self.checkout.apply_patch(self._create_patch(one_line_overlap_patch)) |
| expected_changelog_contents = "%s\n%s" % (self._set_date_and_reviewer(one_line_overlap_entry), changelog_contents) |
| self.assertEqual(read_from_path('ChangeLog'), expected_changelog_contents) |
| |
| self.scm.revert_files(['ChangeLog']) |
| self.checkout.apply_patch(self._create_patch(two_line_overlap_patch)) |
| expected_changelog_contents = "%s\n%s" % (self._set_date_and_reviewer(two_line_overlap_entry), changelog_contents) |
| self.assertEqual(read_from_path('ChangeLog'), expected_changelog_contents) |
| |
| def setUp(self): |
| SVNTestRepository.setup(self) |
| os.chdir(self.svn_checkout_path) |
| self.scm = detect_scm_system(self.svn_checkout_path) |
| self.scm.svn_server_realm = None |
| # For historical reasons, we test some checkout code here too. |
| self.checkout = Checkout(self.scm) |
| |
| def tearDown(self): |
| SVNTestRepository.tear_down(self) |
| |
| def test_detect_scm_system_relative_url(self): |
| scm = detect_scm_system(".") |
| # I wanted to assert that we got the right path, but there was some |
| # crazy magic with temp folder names that I couldn't figure out. |
| self.assertTrue(scm.checkout_root) |
| |
| def test_create_patch_is_full_patch(self): |
| test_dir_path = os.path.join(self.svn_checkout_path, "test_dir2") |
| os.mkdir(test_dir_path) |
| test_file_path = os.path.join(test_dir_path, 'test_file2') |
| write_into_file_at_path(test_file_path, 'test content') |
| run_command(['svn', 'add', 'test_dir2']) |
| |
| # create_patch depends on 'svn-create-patch', so make a dummy version. |
| scripts_path = os.path.join(self.svn_checkout_path, 'Tools', 'Scripts') |
| os.makedirs(scripts_path) |
| create_patch_path = os.path.join(scripts_path, 'svn-create-patch') |
| write_into_file_at_path(create_patch_path, '#!/bin/sh\necho $PWD') # We could pass -n to prevent the \n, but not all echo accept -n. |
| os.chmod(create_patch_path, stat.S_IXUSR | stat.S_IRUSR) |
| |
| # Change into our test directory and run the create_patch command. |
| os.chdir(test_dir_path) |
| scm = detect_scm_system(test_dir_path) |
| self.assertEqual(scm.checkout_root, self.svn_checkout_path) # Sanity check that detection worked right. |
| actual_patch_contents = scm.create_patch() |
| expected_patch_contents = b"""Index: test_dir2/test_file2 |
| =================================================================== |
| --- test_dir2/test_file2\t(nonexistent) |
| +++ test_dir2/test_file2\t(working copy) |
| @@ -0,0 +1 @@\n+test content |
| \\ No newline at end of file |
| """ |
| self.assertEqual(expected_patch_contents, actual_patch_contents) |
| |
| def test_detection(self): |
| self.assertEqual(self.scm.display_name(), "svn") |
| self.assertEqual(self.scm.supports_local_commits(), False) |
| |
| def test_apply_small_binary_patch(self): |
| patch_contents = """Index: test_file.swf |
| =================================================================== |
| Cannot display: file marked as a binary type. |
| svn:mime-type = application/octet-stream |
| |
| Property changes on: test_file.swf |
| ___________________________________________________________________ |
| Name: svn:mime-type |
| + application/octet-stream |
| |
| |
| Q1dTBx0AAAB42itg4GlgYJjGwMDDyODMxMDw34GBgQEAJPQDJA== |
| """ |
| expected_contents = base64.b64decode("Q1dTBx0AAAB42itg4GlgYJjGwMDDyODMxMDw34GBgQEAJPQDJA==") |
| self._setup_webkittools_scripts_symlink(self.scm) |
| patch_file = self._create_patch(patch_contents) |
| self.checkout.apply_patch(patch_file) |
| actual_contents = read_from_path("test_file.swf", encoding=None) |
| self.assertEqual(actual_contents, expected_contents) |
| |
| def test_apply_svn_patch(self): |
| patch = self._create_patch(_svn_diff("-r5:4")) |
| self._setup_webkittools_scripts_symlink(self.scm) |
| Checkout(self.scm).apply_patch(patch) |
| |
| def test_commit_logs(self): |
| # Commits have dates and usernames in them, so we can't just direct compare. |
| self.assertRegexpMatches(self.scm.last_svn_commit_log(), 'fourth commit') |
| self.assertRegexpMatches(self.scm.svn_commit_log(3), 'second commit') |
| |
| def _shared_test_commit_with_message(self, username=None): |
| write_into_file_at_path('test_file', 'more test content') |
| commit_text = self.scm.commit_with_message("another test commit", username) |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| def test_commit_in_subdir(self, username=None): |
| write_into_file_at_path('test_dir/test_file3', 'more test content') |
| os.chdir("test_dir") |
| commit_text = self.scm.commit_with_message("another test commit", username) |
| os.chdir("..") |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| def test_commit_text_parsing(self): |
| self._shared_test_commit_with_message() |
| |
| def test_commit_with_username(self): |
| self._shared_test_commit_with_message("dbates@webkit.org") |
| |
| def test_commit_without_authorization(self): |
| # FIXME: https://bugs.webkit.org/show_bug.cgi?id=111669 |
| # This test ends up looking in the actal $HOME/.subversion for authorization, |
| # which makes it fragile. For now, set it to use a realm that won't be authorized, |
| # but we should really plumb through a fake_home_dir here like we do in |
| # test_has_authorization_for_realm. |
| self.scm.svn_server_realm = '<http://svn.example.com:80> Example' |
| self.assertRaises(AuthenticationError, self._shared_test_commit_with_message) |
| |
| def test_has_authorization_for_realm_using_credentials_with_passtype(self): |
| credentials = """ |
| K 8 |
| passtype |
| V 8 |
| keychain |
| K 15 |
| svn:realmstring |
| V 39 |
| <http://svn.webkit.org:80> Mac OS Forge |
| K 8 |
| username |
| V 17 |
| dbates@webkit.org |
| END |
| """ |
| self.assertTrue(self._test_has_authorization_for_realm_using_credentials(SVN.svn_server_realm, credentials)) |
| |
| def test_has_authorization_for_realm_using_credentials_with_password(self): |
| credentials = """ |
| K 15 |
| svn:realmstring |
| V 39 |
| <http://svn.webkit.org:80> Mac OS Forge |
| K 8 |
| username |
| V 17 |
| dbates@webkit.org |
| K 8 |
| password |
| V 4 |
| blah |
| END |
| """ |
| self.assertTrue(self._test_has_authorization_for_realm_using_credentials(SVN.svn_server_realm, credentials)) |
| |
| def _test_has_authorization_for_realm_using_credentials(self, realm, credentials): |
| fake_home_dir = tempfile.mkdtemp(suffix="fake_home_dir") |
| svn_config_dir_path = os.path.join(fake_home_dir, ".subversion") |
| os.mkdir(svn_config_dir_path) |
| fake_webkit_auth_file = os.path.join(svn_config_dir_path, "fake_webkit_auth_file") |
| write_into_file_at_path(fake_webkit_auth_file, credentials) |
| result = self.scm.has_authorization_for_realm(realm, home_directory=fake_home_dir) |
| os.remove(fake_webkit_auth_file) |
| os.rmdir(svn_config_dir_path) |
| os.rmdir(fake_home_dir) |
| return result |
| |
| def test_not_have_authorization_for_realm_with_credentials_missing_password_and_passtype(self): |
| credentials = """ |
| K 15 |
| svn:realmstring |
| V 39 |
| <http://svn.webkit.org:80> Mac OS Forge |
| K 8 |
| username |
| V 17 |
| dbates@webkit.org |
| END |
| """ |
| self.assertFalse(self._test_has_authorization_for_realm_using_credentials(SVN.svn_server_realm, credentials)) |
| |
| def test_not_have_authorization_for_realm_when_missing_credentials_file(self): |
| fake_home_dir = tempfile.mkdtemp(suffix="fake_home_dir") |
| svn_config_dir_path = os.path.join(fake_home_dir, ".subversion") |
| os.mkdir(svn_config_dir_path) |
| self.assertFalse(self.scm.has_authorization_for_realm(SVN.svn_server_realm, home_directory=fake_home_dir)) |
| os.rmdir(svn_config_dir_path) |
| os.rmdir(fake_home_dir) |
| |
| def test_reverse_diff(self): |
| self._shared_test_reverse_diff() |
| |
| def test_diff_for_revision(self): |
| self._shared_test_diff_for_revision() |
| |
| def test_svn_apply_git_patch(self): |
| self._shared_test_svn_apply_git_patch() |
| |
| def test_changed_files(self): |
| self._shared_test_changed_files() |
| |
| def test_untracked_files(self): |
| self._shared_test_untracked_files(self.scm) |
| |
| def test_discard_untracked_files(self): |
| self._shared_test_discard_untracked_files(self.scm) |
| |
| def test_changed_files_for_revision(self): |
| self._shared_test_changed_files_for_revision() |
| |
| def test_added_files(self): |
| self._shared_test_added_files() |
| |
| def test_contents_at_revision(self): |
| self._shared_test_contents_at_revision() |
| |
| def test_revisions_changing_file(self): |
| self._shared_test_revisions_changing_file() |
| |
| def test_committer_email_for_revision(self): |
| self._shared_test_committer_email_for_revision() |
| |
| def test_add_recursively(self): |
| self._shared_test_add_recursively() |
| |
| def test_delete(self): |
| os.chdir(self.svn_checkout_path) |
| self.scm.delete("test_file") |
| self.assertIn("test_file", self.scm.deleted_files()) |
| |
| def test_delete_list(self): |
| os.chdir(self.svn_checkout_path) |
| self.scm.delete_list(["test_file", "test_file2"]) |
| self.assertIn("test_file", self.scm.deleted_files()) |
| self.assertIn("test_file2", self.scm.deleted_files()) |
| |
| def test_delete_recursively(self): |
| self._shared_test_delete_recursively() |
| |
| def test_delete_recursively_or_not(self): |
| self._shared_test_delete_recursively_or_not() |
| |
| def test_head_svn_revision(self): |
| self._shared_test_head_svn_revision() |
| |
| def test_svn_revision(self): |
| self._shared_test_svn_revision(self.scm) |
| |
| def test_svn_branch(self): |
| self._shared_test_svn_branch(self.scm) |
| |
| def test_native_revision(self): |
| self.assertEqual(self.scm.head_svn_revision(), self.scm.native_revision('.')) |
| self.assertEqual(self.scm.native_revision('.'), '5') |
| |
| def test_native_branch(self): |
| self.assertEqual(self.scm.native_branch('.'), 'trunk') |
| |
| def test_propset_propget(self): |
| filepath = os.path.join(self.svn_checkout_path, "test_file") |
| self.scm.propset("svn:mime-type", "x-application/foo-bar", filepath) |
| self.assertEqual(b"x-application/foo-bar", self.scm.propget("svn:mime-type", filepath)) |
| |
| def test_show_head(self): |
| write_into_file_at_path("test_file", u"Hello!", "utf-8") |
| SVNTestRepository._svn_commit("fourth commit") |
| self.assertEqual(b"Hello!", self.scm.show_head('test_file')) |
| |
| def test_show_head_binary(self): |
| data = b"\244" |
| write_into_file_at_path("binary_file", data, encoding=None) |
| self.scm.add("binary_file") |
| self.scm.commit_with_message("a test commit") |
| self.assertEqual(data, self.scm.show_head('binary_file')) |
| |
| def do_test_diff_for_file(self): |
| write_into_file_at_path('test_file', 'some content') |
| self.scm.commit_with_message("a test commit") |
| diff = self.scm.diff_for_file('test_file') |
| self.assertEqual(diff, "") |
| |
| write_into_file_at_path("test_file", "changed content") |
| diff = self.scm.diff_for_file('test_file') |
| self.assertIn("-some content", diff) |
| self.assertIn("+changed content", diff) |
| |
| def clean_bogus_dir(self): |
| self.bogus_dir = self.scm._bogus_dir_name() |
| if os.path.exists(self.bogus_dir): |
| shutil.rmtree(self.bogus_dir) |
| |
| def test_diff_for_file_with_existing_bogus_dir(self): |
| self.clean_bogus_dir() |
| os.mkdir(self.bogus_dir) |
| self.do_test_diff_for_file() |
| self.assertTrue(os.path.exists(self.bogus_dir)) |
| shutil.rmtree(self.bogus_dir) |
| |
| def test_diff_for_file_with_missing_bogus_dir(self): |
| self.clean_bogus_dir() |
| self.do_test_diff_for_file() |
| self.assertFalse(os.path.exists(self.bogus_dir)) |
| |
| def test_svn_lock(self): |
| if self.scm.svn_version() >= Version(1, 7): |
| # the following technique with .svn/lock then svn update doesn't work with subversion client 1.7 or later |
| pass |
| else: |
| svn_root_lock_path = ".svn/lock" |
| write_into_file_at_path(svn_root_lock_path, "", "utf-8") |
| # webkit-patch uses a Checkout object and runs update-webkit, just use svn update here. |
| self.assertRaises(ScriptError, run_command, ['svn', 'update']) |
| self.scm.discard_working_directory_changes() |
| self.assertFalse(os.path.exists(svn_root_lock_path)) |
| run_command(['svn', 'update']) # Should succeed and not raise. |
| |
| def test_exists(self): |
| self._shared_test_exists(self.scm, self.scm.commit_with_message) |
| |
| |
| class GitTest(SCMTest): |
| |
| def setUp(self): |
| """Sets up fresh git repository with one commit. Then setups a second git |
| repo that tracks the first one.""" |
| # FIXME: We should instead clone a git repo that is tracking an SVN repo. |
| # That better matches what we do with WebKit. |
| self.original_dir = os.getcwd() |
| |
| self.untracking_checkout_path = tempfile.mkdtemp(suffix="git_test_checkout2") |
| run_command(['git', 'init', self.untracking_checkout_path]) |
| |
| os.chdir(self.untracking_checkout_path) |
| write_into_file_at_path('foo_file', 'foo') |
| run_command(['git', 'add', 'foo_file']) |
| write_into_file_at_path('.gitignore', '*.pyc') |
| run_command(['git', 'add', '.gitignore']) |
| run_command(['git', 'commit', '-am', 'dummy commit']) |
| self.untracking_scm = detect_scm_system(self.untracking_checkout_path) |
| |
| self.tracking_git_checkout_path = tempfile.mkdtemp(suffix="git_test_checkout") |
| run_command(['git', 'clone', '--quiet', self.untracking_checkout_path, self.tracking_git_checkout_path]) |
| os.chdir(self.tracking_git_checkout_path) |
| self.tracking_scm = detect_scm_system(self.tracking_git_checkout_path) |
| |
| def tearDown(self): |
| # Change back to a valid directory so that later calls to os.getcwd() do not fail. |
| os.chdir(self.original_dir) |
| run_command(['rm', '-rf', self.tracking_git_checkout_path]) |
| run_command(['rm', '-rf', self.untracking_checkout_path]) |
| |
| def test_remote_branch_ref(self): |
| self.assertEqual(self.tracking_scm.remote_branch_ref(), 'refs/remotes/origin/master') |
| |
| os.chdir(self.untracking_checkout_path) |
| self.assertRaises(ScriptError, self.untracking_scm.remote_branch_ref) |
| |
| def test_multiple_remotes(self): |
| run_command(['git', 'config', '--add', 'svn-remote.svn.fetch', 'trunk:remote1']) |
| run_command(['git', 'config', '--add', 'svn-remote.svn.fetch', 'trunk:remote2']) |
| self.assertEqual(self.tracking_scm.remote_branch_ref(), 'remote1') |
| |
| def test_create_patch(self): |
| write_into_file_at_path('test_file_commit1', 'contents') |
| run_command(['git', 'add', 'test_file_commit1']) |
| scm = self.tracking_scm |
| scm.commit_locally_with_message('message') |
| |
| patch = scm.create_patch() |
| self.assertNotRegexpMatches(patch, b'Subversion Revision:') |
| |
| def test_create_patch_with_git_index(self): |
| # First change. Committed. |
| write_into_file_at_path('test_file_commit1', 'first cat') |
| run_command(['git', 'add', 'test_file_commit1']) |
| scm = self.tracking_scm |
| scm.commit_locally_with_message('message') |
| |
| # Second change. Staged but not committed. |
| write_into_file_at_path('test_file_commit1', 'second dog') |
| run_command(['git', 'add', 'test_file_commit1']) |
| |
| # Third change. Not even staged. |
| write_into_file_at_path('test_file_commit1', 'third unicorn') |
| |
| patch = scm.create_patch(None, None, True) |
| self.assertRegexpMatches(patch, b'-first cat') |
| self.assertRegexpMatches(patch, b'\\+second dog') |
| self.assertNotRegexpMatches(patch, b'third') |
| self.assertNotRegexpMatches(patch, b'unicorn') |
| |
| def test_orderfile(self): |
| os.mkdir("Tools") |
| os.mkdir("Source") |
| os.mkdir("LayoutTests") |
| os.mkdir("Websites") |
| |
| # Slash should always be the right path separator since we use cygwin on Windows. |
| Tools_ChangeLog = "Tools/ChangeLog" |
| write_into_file_at_path(Tools_ChangeLog, "contents") |
| Source_ChangeLog = "Source/ChangeLog" |
| write_into_file_at_path(Source_ChangeLog, "contents") |
| LayoutTests_ChangeLog = "LayoutTests/ChangeLog" |
| write_into_file_at_path(LayoutTests_ChangeLog, "contents") |
| Websites_ChangeLog = "Websites/ChangeLog" |
| write_into_file_at_path(Websites_ChangeLog, "contents") |
| |
| Tools_ChangeFile = "Tools/ChangeFile" |
| write_into_file_at_path(Tools_ChangeFile, "contents") |
| Source_ChangeFile = "Source/ChangeFile" |
| write_into_file_at_path(Source_ChangeFile, "contents") |
| LayoutTests_ChangeFile = "LayoutTests/ChangeFile" |
| write_into_file_at_path(LayoutTests_ChangeFile, "contents") |
| Websites_ChangeFile = "Websites/ChangeFile" |
| write_into_file_at_path(Websites_ChangeFile, "contents") |
| |
| run_command(['git', 'add', 'Tools/ChangeLog']) |
| run_command(['git', 'add', 'LayoutTests/ChangeLog']) |
| run_command(['git', 'add', 'Source/ChangeLog']) |
| run_command(['git', 'add', 'Websites/ChangeLog']) |
| run_command(['git', 'add', 'Tools/ChangeFile']) |
| run_command(['git', 'add', 'LayoutTests/ChangeFile']) |
| run_command(['git', 'add', 'Source/ChangeFile']) |
| run_command(['git', 'add', 'Websites/ChangeFile']) |
| scm = self.tracking_scm |
| scm.commit_locally_with_message('message') |
| |
| patch = scm.create_patch() |
| self.assertTrue(re.search(b'Tools/ChangeLog', patch).start() < re.search(b'Tools/ChangeFile', patch).start()) |
| self.assertTrue(re.search(b'Websites/ChangeLog', patch).start() < re.search(b'Websites/ChangeFile', patch).start()) |
| self.assertTrue(re.search(b'Source/ChangeLog', patch).start() < re.search(b'Source/ChangeFile', patch).start()) |
| self.assertTrue(re.search(b'LayoutTests/ChangeLog', patch).start() < re.search(b'LayoutTests/ChangeFile', patch).start()) |
| |
| self.assertTrue(re.search(b'Source/ChangeLog', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| self.assertTrue(re.search(b'Tools/ChangeLog', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| self.assertTrue(re.search(b'Websites/ChangeLog', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| |
| self.assertTrue(re.search(b'Source/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| self.assertTrue(re.search(b'Tools/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| self.assertTrue(re.search(b'Websites/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeLog', patch).start()) |
| |
| self.assertTrue(re.search(b'Source/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeFile', patch).start()) |
| self.assertTrue(re.search(b'Tools/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeFile', patch).start()) |
| self.assertTrue(re.search(b'Websites/ChangeFile', patch).start() < re.search(b'LayoutTests/ChangeFile', patch).start()) |
| |
| def test_exists(self): |
| scm = self.untracking_scm |
| self._shared_test_exists(scm, scm.commit_locally_with_message) |
| |
| def test_head_svn_revision(self): |
| scm = detect_scm_system(self.untracking_checkout_path) |
| # If we cloned a git repo tracking an SVN repo, this would give the same result as |
| # self._shared_test_head_svn_revision(). |
| self.assertEqual(scm.head_svn_revision(), '') |
| |
| def test_native_revision(self): |
| scm = self.tracking_scm |
| command = ['git', '-C', scm.checkout_root, 'rev-parse', 'HEAD'] |
| self.assertEqual(scm.native_revision(scm.checkout_root), run_command(command).strip()) |
| |
| def test_native_branch(self): |
| scm = self.tracking_scm |
| self.assertEqual('master', scm.native_branch(scm.checkout_root)) |
| |
| def test_rename_files(self): |
| scm = self.tracking_scm |
| |
| run_command(['git', 'mv', 'foo_file', 'bar_file']) |
| scm.commit_locally_with_message('message') |
| |
| patch = scm.create_patch() |
| self.assertNotRegexpMatches(patch, b'rename from ') |
| self.assertNotRegexpMatches(patch, b'rename to ') |
| |
| def test_untracked_files(self): |
| self._shared_test_untracked_files(self.tracking_scm) |
| |
| |
| class GitSVNTest(SCMTest): |
| |
| def _setup_git_checkout(self): |
| self.git_checkout_path = tempfile.mkdtemp(suffix="git_test_checkout") |
| # --quiet doesn't make git svn silent, so we use run_silent to redirect output |
| run_silent(['git', 'svn', 'clone', '-T', 'trunk', '--prefix', '', self.svn_repo_url, self.git_checkout_path]) |
| os.chdir(self.git_checkout_path) |
| run_silent(['git', 'branch', '-m', 'trunk']) |
| |
| def _tear_down_git_checkout(self): |
| # Change back to a valid directory so that later calls to os.getcwd() do not fail. |
| os.chdir(self.original_dir) |
| run_command(['rm', '-rf', self.git_checkout_path]) |
| |
| def setUp(self): |
| self.original_dir = os.getcwd() |
| |
| SVNTestRepository.setup(self) |
| self._setup_git_checkout() |
| self.scm = detect_scm_system(self.git_checkout_path) |
| self.scm.svn_server_realm = None |
| # For historical reasons, we test some checkout code here too. |
| self.checkout = Checkout(self.scm) |
| |
| def tearDown(self): |
| SVNTestRepository.tear_down(self) |
| self._tear_down_git_checkout() |
| |
| def test_detection(self): |
| self.assertEqual(self.scm.display_name(), "git") |
| self.assertEqual(self.scm.supports_local_commits(), True) |
| |
| def test_read_git_config(self): |
| key = 'test.git-config' |
| value = 'git-config value' |
| run_command(['git', 'config', key, value]) |
| self.assertEqual(self.scm.read_git_config(key), value) |
| |
| def test_local_commits(self): |
| test_file = os.path.join(self.git_checkout_path, 'test_file') |
| write_into_file_at_path(test_file, 'foo') |
| run_command(['git', 'commit', '-a', '-m', 'local commit']) |
| |
| self.assertEqual(len(self.scm.local_commits()), 1) |
| |
| def test_discard_local_commits(self): |
| test_file = os.path.join(self.git_checkout_path, 'test_file') |
| write_into_file_at_path(test_file, 'foo') |
| run_command(['git', 'commit', '-a', '-m', 'local commit']) |
| |
| self.assertEqual(len(self.scm.local_commits()), 1) |
| self.scm.discard_local_commits() |
| self.assertEqual(len(self.scm.local_commits()), 0) |
| |
| def test_delete_branch(self): |
| new_branch = 'foo' |
| |
| run_command(['git', 'checkout', '-b', new_branch]) |
| self.assertEqual(run_command(['git', 'symbolic-ref', 'HEAD']).strip(), 'refs/heads/' + new_branch) |
| |
| run_command(['git', 'checkout', '-b', 'bar']) |
| self.scm.delete_branch(new_branch) |
| |
| self.assertNotRegexpMatches(run_command(['git', 'branch']), r'foo') |
| |
| def test_remote_merge_base(self): |
| # Diff to merge-base should include working-copy changes, |
| # which the diff to svn_branch.. doesn't. |
| test_file = os.path.join(self.git_checkout_path, 'test_file') |
| write_into_file_at_path(test_file, 'foo') |
| |
| diff_to_common_base = _git_diff(self.scm.remote_branch_ref() + '..') |
| diff_to_merge_base = _git_diff(self.scm.remote_merge_base()) |
| |
| self.assertNotRegexpMatches(diff_to_common_base, b'foo') |
| self.assertRegexpMatches(diff_to_merge_base, b'foo') |
| |
| def test_rebase_in_progress(self): |
| svn_test_file = os.path.join(self.svn_checkout_path, 'test_file') |
| write_into_file_at_path(svn_test_file, "svn_checkout") |
| run_command(['svn', 'commit', '--message', 'commit to conflict with git commit'], cwd=self.svn_checkout_path) |
| |
| git_test_file = os.path.join(self.git_checkout_path, 'test_file') |
| write_into_file_at_path(git_test_file, "git_checkout") |
| run_command(['git', 'commit', '-a', '-m', 'commit to be thrown away by rebase abort']) |
| |
| # --quiet doesn't make git svn silent, so use run_silent to redirect output |
| self.assertRaises(ScriptError, run_silent, ['git', 'svn', '--quiet', 'rebase']) # Will fail due to a conflict leaving us mid-rebase. |
| |
| self.assertTrue(self.scm.rebase_in_progress()) |
| |
| # Make sure our cleanup works. |
| self.scm.discard_working_directory_changes() |
| self.assertFalse(self.scm.rebase_in_progress()) |
| |
| # Make sure cleanup doesn't throw when no rebase is in progress. |
| self.scm.discard_working_directory_changes() |
| |
| def test_commitish_parsing(self): |
| # Multiple revisions are cherry-picked. |
| self.assertEqual(len(self.scm.commit_ids_from_commitish_arguments(['HEAD~2'])), 1) |
| self.assertEqual(len(self.scm.commit_ids_from_commitish_arguments(['HEAD', 'HEAD~2'])), 2) |
| |
| # ... is an invalid range specifier |
| self.assertRaises(ScriptError, self.scm.commit_ids_from_commitish_arguments, ['trunk...HEAD']) |
| |
| def test_commitish_order(self): |
| commit_range = 'HEAD~3..HEAD' |
| |
| actual_commits = self.scm.commit_ids_from_commitish_arguments([commit_range]) |
| expected_commits = [] |
| expected_commits += reversed(run_command(['git', 'rev-list', commit_range]).splitlines()) |
| |
| self.assertEqual(actual_commits, expected_commits) |
| |
| def test_apply_git_patch(self): |
| # We carefullly pick a diff which does not have a directory addition |
| # as currently svn-apply will error out when trying to remove directories |
| # in Git: https://bugs.webkit.org/show_bug.cgi?id=34871 |
| patch = self._create_patch(_git_diff('HEAD..HEAD^')) |
| self._setup_webkittools_scripts_symlink(self.scm) |
| Checkout(self.scm).apply_patch(patch) |
| |
| def test_commit_text_parsing(self): |
| write_into_file_at_path('test_file', 'more test content') |
| commit_text = self.scm.commit_with_message("another test commit") |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| def test_commit_with_message_working_copy_only(self): |
| write_into_file_at_path('test_file_commit1', 'more test content') |
| run_command(['git', 'add', 'test_file_commit1']) |
| commit_text = self.scm.commit_with_message("yet another test commit") |
| |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def _local_commit(self, filename, contents, message): |
| write_into_file_at_path(filename, contents) |
| run_command(['git', 'add', filename]) |
| self.scm.commit_locally_with_message(message) |
| |
| def _one_local_commit(self): |
| self._local_commit('test_file_commit1', 'more test content', 'another test commit') |
| |
| def _one_local_commit_plus_working_copy_changes(self): |
| self._one_local_commit() |
| write_into_file_at_path('test_file_commit2', 'still more test content') |
| run_command(['git', 'add', 'test_file_commit2']) |
| |
| def _second_local_commit(self): |
| self._local_commit('test_file_commit2', 'still more test content', 'yet another test commit') |
| |
| def _two_local_commits(self): |
| self._one_local_commit() |
| self._second_local_commit() |
| |
| def _three_local_commits(self): |
| self._local_commit('test_file_commit0', 'more test content', 'another test commit') |
| self._two_local_commits() |
| |
| def test_revisions_changing_files_with_local_commit(self): |
| self._one_local_commit() |
| self.assertEqual(self.scm.revisions_changing_file('test_file_commit1'), []) |
| |
| def test_commit_with_message(self): |
| self._one_local_commit_plus_working_copy_changes() |
| self.assertRaises(AmbiguousCommitError, self.scm.commit_with_message, "yet another test commit") |
| commit_text = self.scm.commit_with_message("yet another test commit", force_squash=True) |
| |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_git_commit(self): |
| self._two_local_commits() |
| |
| commit_text = self.scm.commit_with_message("another test commit", git_commit="HEAD^") |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| self.assertNotRegexpMatches(svn_log, r'test_file_commit2') |
| |
| def test_commit_with_message_git_commit_range(self): |
| self._three_local_commits() |
| |
| commit_text = self.scm.commit_with_message("another test commit", git_commit="HEAD~2..HEAD") |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertNotRegexpMatches(svn_log, r'test_file_commit0') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| |
| def test_commit_with_message_only_local_commit(self): |
| self._one_local_commit() |
| commit_text = self.scm.commit_with_message("another test commit") |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_multiple_local_commits_and_working_copy(self): |
| self._two_local_commits() |
| write_into_file_at_path('test_file_commit1', 'working copy change') |
| |
| self.assertRaises(AmbiguousCommitError, self.scm.commit_with_message, "another test commit") |
| commit_text = self.scm.commit_with_message("another test commit", force_squash=True) |
| |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_git_commit_and_working_copy(self): |
| self._two_local_commits() |
| write_into_file_at_path('test_file_commit1', 'working copy change') |
| self.assertRaises(ScriptError, self.scm.commit_with_message, "another test commit", git_commit="HEAD^") |
| |
| def test_commit_with_message_multiple_local_commits_always_squash(self): |
| run_command(['git', 'config', 'webkit-patch.commit-should-always-squash', 'true']) |
| self._two_local_commits() |
| commit_text = self.scm.commit_with_message("yet another test commit") |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_multiple_local_commits(self): |
| self._two_local_commits() |
| self.assertRaises(AmbiguousCommitError, self.scm.commit_with_message, "yet another test commit") |
| commit_text = self.scm.commit_with_message("yet another test commit", force_squash=True) |
| |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_not_synced(self): |
| run_command(['git', 'checkout', '-b', 'my-branch', 'trunk~3']) |
| self._two_local_commits() |
| self.assertRaises(AmbiguousCommitError, self.scm.commit_with_message, "another test commit") |
| commit_text = self.scm.commit_with_message("another test commit", force_squash=True) |
| |
| self.assertEqual(self.scm.svn_revision_from_commit_text(commit_text), '6') |
| |
| svn_log = run_command(['git', 'svn', 'log', '--limit=1', '--verbose']) |
| self.assertNotRegexpMatches(svn_log, r'test_file2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit2') |
| self.assertRegexpMatches(svn_log, r'test_file_commit1') |
| |
| def test_commit_with_message_not_synced_with_conflict(self): |
| run_command(['git', 'checkout', '-b', 'my-branch', 'trunk~3']) |
| self._local_commit('test_file2', 'asdf', 'asdf commit') |
| |
| # There's a conflict between trunk and the test_file2 modification. |
| self.assertRaises(ScriptError, self.scm.commit_with_message, "another test commit", force_squash=True) |
| |
| def test_upstream_branch(self): |
| run_command(['git', 'checkout', '-t', '-b', 'my-branch']) |
| run_command(['git', 'checkout', '-t', '-b', 'my-second-branch']) |
| self.assertEqual(self.scm._upstream_branch(), 'my-branch') |
| |
| def test_remote_branch_ref(self): |
| self.assertEqual(self.scm.remote_branch_ref(), 'refs/remotes/trunk') |
| |
| def test_reverse_diff(self): |
| self._shared_test_reverse_diff() |
| |
| def test_diff_for_revision(self): |
| self._shared_test_diff_for_revision() |
| |
| def test_svn_apply_git_patch(self): |
| self._shared_test_svn_apply_git_patch() |
| |
| def test_create_patch_local_plus_working_copy(self): |
| self._one_local_commit_plus_working_copy_changes() |
| patch = self.scm.create_patch() |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| |
| def test_create_patch(self): |
| self._one_local_commit_plus_working_copy_changes() |
| patch = self.scm.create_patch() |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| self.assertRegexpMatches(patch, b'Subversion Revision: 5') |
| |
| def test_create_patch_after_merge(self): |
| run_command(['git', 'checkout', '-b', 'dummy-branch', 'trunk~3']) |
| self._one_local_commit() |
| run_command(['git', 'merge', 'trunk']) |
| |
| patch = self.scm.create_patch() |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| self.assertRegexpMatches(patch, b'Subversion Revision: 5') |
| |
| def test_create_patch_with_changed_files(self): |
| self._one_local_commit_plus_working_copy_changes() |
| patch = self.scm.create_patch(changed_files=['test_file_commit2']) |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| |
| def test_create_patch_with_rm_and_changed_files(self): |
| self._one_local_commit_plus_working_copy_changes() |
| os.remove('test_file_commit1') |
| patch = self.scm.create_patch() |
| patch_with_changed_files = self.scm.create_patch(changed_files=['test_file_commit1', 'test_file_commit2']) |
| self.assertEqual(patch, patch_with_changed_files) |
| |
| def test_create_patch_git_commit(self): |
| self._two_local_commits() |
| patch = self.scm.create_patch(git_commit="HEAD^") |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| self.assertNotRegexpMatches(patch, b'test_file_commit2') |
| |
| def test_create_patch_git_commit_range(self): |
| self._three_local_commits() |
| patch = self.scm.create_patch(git_commit="HEAD~2..HEAD") |
| self.assertNotRegexpMatches(patch, b'test_file_commit0') |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| |
| def test_create_patch_working_copy_only(self): |
| self._one_local_commit_plus_working_copy_changes() |
| patch = self.scm.create_patch(git_commit="HEAD....") |
| self.assertNotRegexpMatches(patch, b'test_file_commit1') |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| |
| def test_create_patch_multiple_local_commits(self): |
| self._two_local_commits() |
| patch = self.scm.create_patch() |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| |
| def test_create_patch_not_synced(self): |
| run_command(['git', 'checkout', '-b', 'my-branch', 'trunk~3']) |
| self._two_local_commits() |
| patch = self.scm.create_patch() |
| self.assertNotRegexpMatches(patch, b'test_file2') |
| self.assertRegexpMatches(patch, b'test_file_commit2') |
| self.assertRegexpMatches(patch, b'test_file_commit1') |
| |
| def test_create_binary_patch(self): |
| # Create a git binary patch and check the contents. |
| test_file_name = 'binary_file' |
| test_file_path = os.path.join(self.git_checkout_path, test_file_name) |
| |
| if sys.version_info < (3, 0): |
| file_contents = ''.join(map(chr, range(256))) |
| else: |
| file_contents = bytes(list(range(256))) |
| |
| write_into_file_at_path(test_file_path, file_contents, encoding=None) |
| run_command(['git', 'add', test_file_name]) |
| patch = self.scm.create_patch() |
| self.assertRegexpMatches(patch, b'\nliteral 0\n') |
| self.assertRegexpMatches(patch, b'\nliteral 256\n') |
| |
| # Check if we can apply the created patch. |
| run_command(['git', 'rm', '-f', test_file_name]) |
| self._setup_webkittools_scripts_symlink(self.scm) |
| self.checkout.apply_patch(self._create_patch(patch)) |
| self.assertEqual(file_contents, read_from_path(test_file_path, encoding=None)) |
| |
| # Check if we can create a patch from a local commit. |
| write_into_file_at_path(test_file_path, file_contents, encoding=None) |
| run_command(['git', 'add', test_file_name]) |
| run_command(['git', 'commit', '-m', 'binary diff']) |
| |
| patch_from_local_commit = self.scm.create_patch('HEAD') |
| self.assertRegexpMatches(patch_from_local_commit, b'\nliteral 0\n') |
| self.assertRegexpMatches(patch_from_local_commit, b'\nliteral 256\n') |
| |
| def test_changed_files_local_plus_working_copy(self): |
| self._one_local_commit_plus_working_copy_changes() |
| files = self.scm.changed_files() |
| self.assertIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| |
| # working copy should *not* be in the list. |
| files = self.scm.changed_files('trunk..') # git diff trunk..HEAD |
| self.assertNotIn('test_file_commit1', files) |
| self.assertNotIn('test_file_commit2', files) |
| |
| files = self.scm.changed_files('trunk^..') # git diff trunk^..HEAD |
| self.assertIn('test_file_commit1', files) |
| self.assertNotIn('test_file_commit2', files) |
| |
| # working copy *should* be in the list. |
| # .... is a hack for working copy changes to be included |
| files = self.scm.changed_files('trunk....') # git diff trunk |
| self.assertNotIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| |
| files = self.scm.changed_files('trunk^....') # git diff trunk^ |
| self.assertIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| |
| def test_changed_files_git_commit(self): |
| self._two_local_commits() |
| files = self.scm.changed_files(git_commit="HEAD^") |
| self.assertIn('test_file_commit1', files) |
| self.assertNotIn('test_file_commit2', files) |
| |
| def test_changed_files_git_commit_range(self): |
| self._three_local_commits() |
| files = self.scm.changed_files(git_commit="HEAD~2..HEAD") |
| self.assertNotIn('test_file_commit0', files) |
| self.assertIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| |
| def test_changed_files_working_copy_only(self): |
| self._one_local_commit_plus_working_copy_changes() |
| files = self.scm.changed_files(git_commit="HEAD....") |
| self.assertNotIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| |
| def test_changed_files_multiple_local_commits(self): |
| self._two_local_commits() |
| files = self.scm.changed_files() |
| self.assertIn('test_file_commit2', files) |
| self.assertIn('test_file_commit1', files) |
| |
| def test_changed_files_not_synced(self): |
| run_command(['git', 'checkout', '-b', 'my-branch', 'trunk~3']) |
| self._two_local_commits() |
| files = self.scm.changed_files() |
| self.assertNotIn('test_file2', files) |
| self.assertIn('test_file_commit2', files) |
| self.assertIn('test_file_commit1', files) |
| |
| def test_changed_files_not_synced(self): |
| run_command(['git', 'checkout', '-b', 'my-branch', 'trunk~3']) |
| self._two_local_commits() |
| files = self.scm.changed_files() |
| self.assertNotIn('test_file2', files) |
| self.assertIn('test_file_commit2', files) |
| self.assertIn('test_file_commit1', files) |
| |
| def test_changed_files(self): |
| self._shared_test_changed_files() |
| |
| def test_changed_files_for_revision(self): |
| self._shared_test_changed_files_for_revision() |
| |
| def test_changed_files_upstream(self): |
| run_command(['git', 'checkout', '-t', '-b', 'my-branch']) |
| self._one_local_commit() |
| run_command(['git', 'checkout', '-t', '-b', 'my-second-branch']) |
| self._second_local_commit() |
| write_into_file_at_path('test_file_commit0', 'more test content') |
| run_command(['git', 'add', 'test_file_commit0']) |
| |
| # equivalent to 'git diff my-branch..HEAD, should not include working changes |
| files = self.scm.changed_files(git_commit='UPSTREAM..') |
| self.assertNotIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| self.assertNotIn('test_file_commit0', files) |
| |
| # equivalent to 'git diff my-branch', *should* include working changes |
| files = self.scm.changed_files(git_commit='UPSTREAM....') |
| self.assertNotIn('test_file_commit1', files) |
| self.assertIn('test_file_commit2', files) |
| self.assertIn('test_file_commit0', files) |
| |
| def test_contents_at_revision(self): |
| self._shared_test_contents_at_revision() |
| |
| def test_revisions_changing_file(self): |
| self._shared_test_revisions_changing_file() |
| |
| def test_added_files(self): |
| self._shared_test_added_files() |
| |
| def test_committer_email_for_revision(self): |
| self._shared_test_committer_email_for_revision() |
| |
| def test_add_recursively(self): |
| self._shared_test_add_recursively() |
| |
| def test_delete(self): |
| self._two_local_commits() |
| self.scm.delete('test_file_commit1') |
| self.assertIn("test_file_commit1", self.scm.deleted_files()) |
| |
| def test_delete_list(self): |
| self._two_local_commits() |
| self.scm.delete_list(["test_file_commit1", "test_file_commit2"]) |
| self.assertIn("test_file_commit1", self.scm.deleted_files()) |
| self.assertIn("test_file_commit2", self.scm.deleted_files()) |
| |
| def test_delete_recursively(self): |
| self._shared_test_delete_recursively() |
| |
| def test_delete_recursively_or_not(self): |
| self._shared_test_delete_recursively_or_not() |
| |
| def test_head_svn_revision(self): |
| self._shared_test_head_svn_revision() |
| |
| def test_native_revision(self): |
| command = ['git', '-C', self.git_checkout_path, 'rev-parse', 'HEAD'] |
| self.assertEqual(self.scm.native_revision(self.git_checkout_path), run_command(command).strip()) |
| |
| def test_native_branch(self): |
| self.assertEqual('trunk', self.scm.native_branch(self.git_checkout_path)) |
| |
| def test_to_object_name(self): |
| relpath = 'test_file_commit1' |
| fullpath = os.path.realpath(os.path.join(self.git_checkout_path, relpath)) |
| self.assertEqual(relpath, self.scm.to_object_name(fullpath)) |
| |
| def test_show_head(self): |
| self._two_local_commits() |
| self.assertEqual(b"more test content", self.scm.show_head('test_file_commit1')) |
| |
| def test_show_head_binary(self): |
| self._two_local_commits() |
| data = b"\244" |
| write_into_file_at_path("binary_file", data, encoding=None) |
| self.scm.add("binary_file") |
| self.scm.commit_locally_with_message("a test commit") |
| self.assertEqual(data, self.scm.show_head('binary_file')) |
| |
| def test_diff_for_file(self): |
| self._two_local_commits() |
| write_into_file_at_path('test_file_commit1', b"Updated", encoding=None) |
| |
| diff = self.scm.diff_for_file('test_file_commit1') |
| cached_diff = self.scm.diff_for_file('test_file_commit1') |
| self.assertIn("+Updated", diff) |
| self.assertIn("-more test content", diff) |
| |
| self.scm.add('test_file_commit1') |
| |
| cached_diff = self.scm.diff_for_file('test_file_commit1') |
| self.assertIn("+Updated", cached_diff) |
| self.assertIn("-more test content", cached_diff) |
| |
| def test_exists(self): |
| self._shared_test_exists(self.scm, self.scm.commit_locally_with_message) |
| |
| |
| # We need to split off more of these SCM tests to use mocks instead of the filesystem. |
| # This class is the first part of that. |
| class GitTestWithMock(unittest.TestCase): |
| maxDiff = None |
| |
| def make_scm(self, logging_executive=False): |
| # We do this should_log dance to avoid logging when Git.__init__ runs sysctl on mac to check for 64-bit support. |
| scm = Git(cwd=".", patch_directories=None, executive=MockExecutive(), filesystem=MockFileSystem()) |
| scm.read_git_config = lambda *args, **kw: "MOCKKEY:MOCKVALUE" |
| scm._executive._should_log = logging_executive |
| return scm |
| |
| def test_create_patch(self): |
| scm = self.make_scm(logging_executive=True) |
| expected_stderr = """\ |
| MOCK run_command: ['git', 'merge-base', 'MOCKVALUE', 'HEAD'], cwd=%(checkout)s |
| MOCK run_command: ['git', 'diff', '--binary', '--no-color', '--no-ext-diff', '--full-index', '--no-renames', '', 'MOCK output of child process', '--'], cwd=%(checkout)s |
| MOCK run_command: ['git', 'rev-parse', '--show-toplevel'], cwd=%(checkout)s |
| MOCK run_command: ['git', 'log', '-1', '--grep=git-svn-id:', '--date=iso', './MOCK output of child process/MOCK output of child process'], cwd=%(checkout)s |
| """ % {'checkout': scm.checkout_root} |
| OutputCapture().assert_outputs(self, scm.create_patch, expected_logs=expected_stderr) |
| |
| def test_push_local_commits_to_server_with_username_and_password(self): |
| self.assertEqual(self.make_scm().push_local_commits_to_server(username='dbates@webkit.org', password='blah'), "MOCK output of child process") |
| |
| def test_push_local_commits_to_server_without_username_and_password(self): |
| self.assertRaises(AuthenticationError, self.make_scm().push_local_commits_to_server) |
| |
| def test_push_local_commits_to_server_with_username_and_without_password(self): |
| self.assertRaises(AuthenticationError, self.make_scm().push_local_commits_to_server, {'username': 'dbates@webkit.org'}) |
| |
| def test_push_local_commits_to_server_without_username_and_with_password(self): |
| self.assertRaises(AuthenticationError, self.make_scm().push_local_commits_to_server, {'password': 'blah'}) |
| |
| def test_timestamp_of_revision(self): |
| scm = self.make_scm() |
| scm.find_checkout_root = lambda path: '' |
| scm._run_git = lambda args: 'Date: 2013-02-08 08:05:49 +0000' |
| self.assertEqual(scm.timestamp_of_revision('some-path', '12345'), '2013-02-08T08:05:49Z') |
| |
| scm._run_git = lambda args: 'Date: 2013-02-08 01:02:03 +0130' |
| self.assertEqual(scm.timestamp_of_revision('some-path', '12345'), '2013-02-07T23:32:03Z') |
| |
| scm._run_git = lambda args: 'Date: 2013-02-08 01:55:21 -0800' |
| self.assertEqual(scm.timestamp_of_revision('some-path', '12345'), '2013-02-08T09:55:21Z') |
| |
| def test_timestamp_of_native_revision(self): |
| scm = self.make_scm() |
| scm.find_checkout_root = lambda path: '' |
| scm._run_git = lambda args: '1360310749' |
| self.assertEqual(scm.timestamp_of_native_revision('some-path', '1a1c3b08814bc2a8c15b0f6db63cdce68f2aaa7a'), '2013-02-08T08:05:49Z') |
| |
| scm._run_git = lambda args: '1360279923' |
| self.assertEqual(scm.timestamp_of_native_revision('some-path', '1a1c3b08814bc2a8c15b0f6db63cdce68f2aaa7a'), '2013-02-07T23:32:03Z') |
| |
| scm._run_git = lambda args: '1360317321' |
| self.assertEqual(scm.timestamp_of_native_revision('some-path', '1a1c3b08814bc2a8c15b0f6db63cdce68f2aaa7a'), '2013-02-08T09:55:21Z') |
| |
| def test_svn_data_from_git_svn_id(self): |
| scm = self.make_scm() |
| scm.find_checkout_root = lambda path: '' |
| scm._most_recent_log_matching = lambda grep_str, path: 'git-svn-id: http://svn.webkit.org/repository/webkit/trunk@258024 268f45cc-cd09-0410-ab3c-d52691b4dbfc' |
| self.assertEqual(scm.svn_revision(scm.checkout_root), '258024') |
| self.assertEqual(scm.svn_branch(scm.checkout_root), 'trunk') |
| self.assertEqual(scm.svn_url(scm.checkout_root), 'http://svn.webkit.org/repository/webkit/trunk') |
| |
| scm._most_recent_log_matching = lambda grep_str, path: 'git-svn-id: http://svn.webkit.org/repository/webkit/branch/specialSubmission@258000 268f45cc-cd09-0410-ab3c-d52691b4dbfc' |
| self.assertEqual(scm.svn_revision(scm.checkout_root), '258000') |
| self.assertEqual(scm.svn_branch(scm.checkout_root), 'specialSubmission') |
| self.assertEqual(scm.svn_url(scm.checkout_root), 'http://svn.webkit.org/repository/webkit/branch/specialSubmission') |