~bzr/ubuntu/lucid/bzr/beta-ppa

« back to all changes in this revision

Viewing changes to bzrlib/tests/per_repository/test_check_reconcile.py

  • Committer: Martin Pool
  • Date: 2010-07-02 07:29:40 UTC
  • mfrom: (129.1.7 packaging-karmic)
  • Revision ID: mbp@sourcefrog.net-20100702072940-hpzq5elg8wjve8rh
* PPA rebuild.
* PPA rebuild for Karmic.
* PPA rebuild for Jaunty.
* PPA rebuild for Hardy.
* From postinst, actually remove the example bash completion scripts.
  (LP: #249452)
* New upstream release.
* New upstream release.
* New upstream release.
* Revert change to Build-depends: Dapper does not have python-central.
  Should be python-support..
* Target ppa..
* Target ppa..
* Target ppa..
* Target ppa..
* New upstream release.
* Switch to dpkg-source 3.0 (quilt) format.
* Bump standards version to 3.8.4.
* Remove embedded copy of python-configobj. Closes: #555336
* Remove embedded copy of python-elementtree. Closes: #555343
* Change section from 'Devel' to 'Vcs'..
* Change section from 'Devel' to 'Vcs'..
* Change section from 'Devel' to 'Vcs'..
* Change section from 'Devel' to 'Vcs'..
* Change section from 'Devel' to 'Vcs'..
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* debian/control: Fix obsolete-relation-form-in-source
  lintian warning. 
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Split out docs into bzr-doc package.
* New upstream release.
* Added John Francesco Ferlito to Uploaders.
* Fix install path to quick-reference guide
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Fix FTBFS due to path changes, again.
* Fix FTBFS due to doc paths changing
* New upstream release.
* Fix FTBFS due to path changes, again.
* Fix FTBFS due to doc paths changing
* New upstream release.
* Fix FTBFS due to path changes, again.
* Fix FTBFS due to doc paths changing
* New upstream release.
* Fix FTBFS due to path changes, again, again.
* Fix FTBFS due to path changes, again.
* Fix FTBFS due to path changes.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Bump standards version to 3.8.3.
* Remove unused patch system.
* New upstream release.
* New upstream release.
* New upstream release.
* Fix copy and paste tab error in .install file
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
 + Fixes compatibility with Python 2.4. Closes: #537708
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream version.
* Bump standards version to 3.8.2.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Add python-pyrex to build-deps to ensure C extensions are always build.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Split documentation into bzr-doc package. ((LP: #385074)
* Multiple packaging changes to make us more linitan clean.
* New upstream release.
* Split documentation into bzr-doc package. ((LP: #385074)
* Multiple packaging changes to make us more linitan clean.
* New upstream release.
* Split documentation into bzr-doc package. ((LP: #385074)
* Multiple packaging changes to make us more linitan clean.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Fix API compatibility version. (Closes: #526233)
* New upstream release.
  + Fixes default format for upgrade command. (Closes: #464688)
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Add missing dependency on zlib development library. (Closes:
  #523595)
* Add zlib build-depends.
* Add zlib build-depends.
* Add zlib build-depends.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Move to section vcs.
* Bump standards version to 3.8.1.
* New upstream release.
* Remove temporary patch for missing .c files from distribution
* New upstream release.
* Remove temporary patch for missing .c files from distribution
* New upstream release.
* Remove temporary patch for missing .c files from distribution
* Add temporary patch for missing .c files from distribution
* Add temporary patch for missing .c files from distribution
* Add temporary patch for missing .c files from distribution
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Recommend ca-certificates. (Closes: #452024)
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Update watch file. bazaar now uses launchpad to host its sources.
* Remove patch for inventory root revision copy, applied upstream.
* New upstream release.
* New upstream release.
* New upstream release
* Force removal of files installed in error to /etc/bash_completion.d/
  (LP: #249452)
* New upstream release.
* New upstream release
* New upstream release.
* Bump standards version.
* Include patch for inventory root revision copy, required for bzr-svn.
* New upstream release.
* Remove unused lintian overrides.
* Correct the package version not to be native.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* New upstream release.
* Final 1.5 release.
* New upstream release.
* New upstream release.
* New upstream release.
* Add myself as a co-maintainer.
* Add a Dm-Upload-Allowed: yes header.
* New upstream bugfix release.
* New upstream release.
* Final 1.3 release.
* New upstream release.
* First release candidate of the upcoming 1.3 release.
* Rebuild to fix the problem caused by a build with a broken python-central.
* New upstream release.
* Rebuild for dapper PPA.
* Apply Lamont's patches to fix build-dependencies on dapper.
  (See: https://bugs.launchpad.net/bzr/+bug/189915)

Show diffs side-by-side

added added

removed removed

Lines of Context:
 
1
# Copyright (C) 2007-2010 Canonical Ltd
 
2
#
 
3
# This program is free software; you can redistribute it and/or modify
 
4
# it under the terms of the GNU General Public License as published by
 
5
# the Free Software Foundation; either version 2 of the License, or
 
6
# (at your option) any later version.
 
7
#
 
8
# This program is distributed in the hope that it will be useful,
 
9
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 
10
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 
11
# GNU General Public License for more details.
 
12
#
 
13
# You should have received a copy of the GNU General Public License
 
14
# along with this program; if not, write to the Free Software
 
15
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 
16
 
 
17
"""Tests that use BrokenRepoScenario objects.
 
18
 
 
19
That is, tests for reconcile and check.
 
20
"""
 
21
 
 
22
from bzrlib import osutils
 
23
 
 
24
from bzrlib.inventory import Inventory, InventoryFile
 
25
from bzrlib.revision import Revision
 
26
from bzrlib.tests import TestNotApplicable
 
27
from bzrlib.tests.per_repository import TestCaseWithRepository
 
28
 
 
29
 
 
30
class TestFileParentReconciliation(TestCaseWithRepository):
 
31
    """Tests for how reconcile corrects errors in parents of file versions."""
 
32
 
 
33
    def make_populated_repository(self, factory):
 
34
        """Create a new repository populated by the given factory."""
 
35
        repo = self.make_repository('broken-repo')
 
36
        repo.lock_write()
 
37
        try:
 
38
            repo.start_write_group()
 
39
            try:
 
40
                factory(repo)
 
41
                repo.commit_write_group()
 
42
                return repo
 
43
            except:
 
44
                repo.abort_write_group()
 
45
                raise
 
46
        finally:
 
47
            repo.unlock()
 
48
 
 
49
    def add_revision(self, repo, revision_id, inv, parent_ids):
 
50
        """Add a revision with a given inventory and parents to a repository.
 
51
 
 
52
        :param repo: a repository.
 
53
        :param revision_id: the revision ID for the new revision.
 
54
        :param inv: an inventory (such as created by
 
55
            `make_one_file_inventory`).
 
56
        :param parent_ids: the parents for the new revision.
 
57
        """
 
58
        inv.revision_id = revision_id
 
59
        inv.root.revision = revision_id
 
60
        if repo.supports_rich_root():
 
61
            root_id = inv.root.file_id
 
62
            repo.texts.add_lines((root_id, revision_id), [], [])
 
63
        repo.add_inventory(revision_id, inv, parent_ids)
 
64
        revision = Revision(revision_id, committer='jrandom@example.com',
 
65
            timestamp=0, inventory_sha1='', timezone=0, message='foo',
 
66
            parent_ids=parent_ids)
 
67
        repo.add_revision(revision_id,revision, inv)
 
68
 
 
69
    def make_one_file_inventory(self, repo, revision, parents,
 
70
                                inv_revision=None, root_revision=None,
 
71
                                file_contents=None, make_file_version=True):
 
72
        """Make an inventory containing a version of a file with ID 'a-file'.
 
73
 
 
74
        The file's ID will be 'a-file', and its filename will be 'a file name',
 
75
        stored at the tree root.
 
76
 
 
77
        :param repo: a repository to add the new file version to.
 
78
        :param revision: the revision ID of the new inventory.
 
79
        :param parents: the parents for this revision of 'a-file'.
 
80
        :param inv_revision: if not None, the revision ID to store in the
 
81
            inventory entry.  Otherwise, this defaults to revision.
 
82
        :param root_revision: if not None, the inventory's root.revision will
 
83
            be set to this.
 
84
        :param file_contents: if not None, the contents of this file version.
 
85
            Otherwise a unique default (based on revision ID) will be
 
86
            generated.
 
87
        """
 
88
        inv = Inventory(revision_id=revision)
 
89
        if root_revision is not None:
 
90
            inv.root.revision = root_revision
 
91
        file_id = 'a-file-id'
 
92
        entry = InventoryFile(file_id, 'a file name', 'TREE_ROOT')
 
93
        if inv_revision is not None:
 
94
            entry.revision = inv_revision
 
95
        else:
 
96
            entry.revision = revision
 
97
        entry.text_size = 0
 
98
        if file_contents is None:
 
99
            file_contents = '%sline\n' % entry.revision
 
100
        entry.text_sha1 = osutils.sha(file_contents).hexdigest()
 
101
        inv.add(entry)
 
102
        if make_file_version:
 
103
            repo.texts.add_lines((file_id, revision),
 
104
                [(file_id, parent) for parent in parents], [file_contents])
 
105
        return inv
 
106
 
 
107
    def require_repo_suffers_text_parent_corruption(self, repo):
 
108
        if not repo._reconcile_fixes_text_parents:
 
109
            raise TestNotApplicable(
 
110
                    "Format does not support text parent reconciliation")
 
111
 
 
112
    def file_parents(self, repo, revision_id):
 
113
        key = ('a-file-id', revision_id)
 
114
        parent_map = repo.texts.get_parent_map([key])
 
115
        return tuple(parent[-1] for parent in parent_map[key])
 
116
 
 
117
    def assertFileVersionAbsent(self, repo, revision_id):
 
118
        self.assertEqual({},
 
119
            repo.texts.get_parent_map([('a-file-id', revision_id)]))
 
120
 
 
121
    def assertParentsMatch(self, expected_parents_for_versions, repo,
 
122
                           when_description):
 
123
        for expected_parents, version in expected_parents_for_versions:
 
124
            if expected_parents is None:
 
125
                self.assertFileVersionAbsent(repo, version)
 
126
            else:
 
127
                found_parents = self.file_parents(repo, version)
 
128
                self.assertEqual(expected_parents, found_parents,
 
129
                    "%s reconcile %s has parents %s, should have %s."
 
130
                    % (when_description, version, found_parents,
 
131
                       expected_parents))
 
132
 
 
133
    def prepare_test_repository(self):
 
134
        """Prepare a repository to test with from the test scenario.
 
135
 
 
136
        :return: A repository, and the scenario instance.
 
137
        """
 
138
        scenario = self.scenario_class(self)
 
139
        repo = self.make_populated_repository(scenario.populate_repository)
 
140
        self.require_repo_suffers_text_parent_corruption(repo)
 
141
        return repo, scenario
 
142
 
 
143
    def shas_for_versions_of_file(self, repo, versions):
 
144
        """Get the SHA-1 hashes of the versions of 'a-file' in the repository.
 
145
 
 
146
        :param repo: the repository to get the hashes from.
 
147
        :param versions: a list of versions to get hashes for.
 
148
 
 
149
        :returns: A dict of `{version: hash}`.
 
150
        """
 
151
        keys = [('a-file-id', version) for version in versions]
 
152
        return repo.texts.get_sha1s(keys)
 
153
 
 
154
    def test_reconcile_behaviour(self):
 
155
        """Populate a repository and reconcile it, verifying the state before
 
156
        and after.
 
157
        """
 
158
        repo, scenario = self.prepare_test_repository()
 
159
        repo.lock_read()
 
160
        try:
 
161
            self.assertParentsMatch(scenario.populated_parents(), repo,
 
162
                'before')
 
163
            vf_shas = self.shas_for_versions_of_file(
 
164
                repo, scenario.all_versions_after_reconcile())
 
165
        finally:
 
166
            repo.unlock()
 
167
        result = repo.reconcile(thorough=True)
 
168
        repo.lock_read()
 
169
        try:
 
170
            self.assertParentsMatch(scenario.corrected_parents(), repo,
 
171
                'after')
 
172
            # The contents of the versions in the versionedfile should be the
 
173
            # same after the reconcile.
 
174
            self.assertEqual(
 
175
                vf_shas,
 
176
                self.shas_for_versions_of_file(
 
177
                    repo, scenario.all_versions_after_reconcile()))
 
178
 
 
179
            # Scenario.corrected_fulltexts contains texts which the test wants
 
180
            # to assert are now fulltexts. However this is an abstraction
 
181
            # violation; really we care that:
 
182
            # - the text is reconstructable
 
183
            # - it has an empty parents list
 
184
            # (we specify it this way because a store can use arbitrary
 
185
            # compression pointers in principle.
 
186
            for file_version in scenario.corrected_fulltexts():
 
187
                key = ('a-file-id', file_version)
 
188
                self.assertEqual({key:()}, repo.texts.get_parent_map([key]))
 
189
                self.assertIsInstance(
 
190
                    repo.texts.get_record_stream([key], 'unordered',
 
191
                        True).next().get_bytes_as('fulltext'),
 
192
                    str)
 
193
        finally:
 
194
            repo.unlock()
 
195
 
 
196
    def test_check_behaviour(self):
 
197
        """Populate a repository and check it, and verify the output."""
 
198
        repo, scenario = self.prepare_test_repository()
 
199
        check_result = repo.check()
 
200
        check_result.report_results(verbose=True)
 
201
        log = self.get_log()
 
202
        for pattern in scenario.check_regexes(repo):
 
203
            self.assertContainsRe(log, pattern)
 
204
 
 
205
    def test_find_text_key_references(self):
 
206
        """Test that find_text_key_references finds erroneous references."""
 
207
        repo, scenario = self.prepare_test_repository()
 
208
        repo.lock_read()
 
209
        self.addCleanup(repo.unlock)
 
210
        self.assertEqual(scenario.repository_text_key_references(),
 
211
            repo.find_text_key_references())
 
212
 
 
213
    def test__generate_text_key_index(self):
 
214
        """Test that the generated text key index has all entries."""
 
215
        repo, scenario = self.prepare_test_repository()
 
216
        repo.lock_read()
 
217
        self.addCleanup(repo.unlock)
 
218
        self.assertEqual(scenario.repository_text_key_index(),
 
219
            repo._generate_text_key_index())