14
14
# along with this program; if not, write to the Free Software
15
15
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18
17
"""Benchmark test suite for bzr."""
21
from bzrlib.bzrdir import BzrDir
22
from bzrlib.tests import TestLoader
22
from bzrlib import plugin as _mod_plugin
24
from bzrlib.tests.TestUtil import TestLoader
23
25
from bzrlib.tests.blackbox import ExternalBase
25
28
class Benchmark(ExternalBase):
29
"""A Test class which provides helpers for writing benchmark tests."""
27
def make_kernel_like_tree(self, url=None):
31
def make_kernel_like_tree(self, url=None, root='.',
28
33
"""Setup a temporary tree roughly like a kernel tree.
30
35
:param url: Creat the kernel like tree as a lightweight checkout
31
of a new branch created at url.
33
# a kernel tree has ~10000 and 500 directory, with most files around
35
# we simulate this by three levels of dirs named 0-7, givin 512 dirs,
38
b = bzrlib.bzrdir.BzrDir.create_branch_convenience(url)
39
d = bzrlib.bzrdir.BzrDir.create('.')
40
bzrlib.branch.BranchReferenceFormat().initialize(d, b)
41
d.create_workingtree()
45
for outer in range(8):
46
files.append("%s/" % outer)
47
for middle in range(8):
48
files.append("%s/%s/" % (outer, middle))
49
for inner in range(8):
50
prefix = "%s/%s/%s/" % (outer, middle, inner)
52
files.extend([prefix + str(foo) for foo in range(20)])
53
self.build_tree(files)
55
def make_many_commit_tree(self, directory_name='.'):
56
"""Create a tree with an egregious number of commits.
36
of a new branch created at url.
37
:param root: Path where the tree will be created.
38
:param link_working: instead of creating a new copy of all files
39
just hardlink the working tree. Tests must request this, because
40
they must break links if they want to change the files
41
:return: A newly created tree.
43
from bzrlib.benchmarks.tree_creator.kernel_like import (
44
KernelLikeTreeCreator,
46
creator = KernelLikeTreeCreator(self, link_working=link_working,
48
return creator.create(root=root)
50
def make_kernel_like_added_tree(self, root='.',
53
"""Make a kernel like tree, with all files added
55
:param root: Where to create the files
56
:param link_working: Instead of copying all of the working tree
57
files, just hardlink them to the cached files. Tests can unlink
58
files that they will change.
59
:param hot_cache: Run through the newly created tree and make sure
60
the stat-cache is correct. The old way of creating a freshly
61
added tree always had a hot cache.
63
from bzrlib.benchmarks.tree_creator.kernel_like import (
64
KernelLikeAddedTreeCreator,
66
creator = KernelLikeAddedTreeCreator(self, link_working=link_working,
68
return creator.create(root=root)
70
def make_kernel_like_committed_tree(self, root='.',
74
"""Make a kernel like tree, with all files added and committed
76
:param root: Where to create the files
77
:param link_working: Instead of copying all of the working tree
78
files, just hardlink them to the cached files. Tests can unlink
79
files that they will change.
80
:param link_bzr: Hardlink the .bzr directory. For readonly
81
operations this is safe, and shaves off a lot of setup time
83
from bzrlib.benchmarks.tree_creator.kernel_like import (
84
KernelLikeCommittedTreeCreator,
86
creator = KernelLikeCommittedTreeCreator(self,
87
link_working=link_working,
90
return creator.create(root=root)
92
def make_kernel_like_inventory(self):
93
"""Create an inventory with the properties of a kernel-like tree
95
This should be equivalent to a committed kernel like tree, not
98
from bzrlib.benchmarks.tree_creator.kernel_like import (
99
KernelLikeInventoryCreator,
101
creator = KernelLikeInventoryCreator(self)
102
return creator.create()
104
def make_many_commit_tree(self, directory_name='.',
106
"""Create a tree with many commits.
58
No files change are included.
60
tree = BzrDir.create_standalone_workingtree(directory_name)
62
tree.branch.lock_write()
63
tree.branch.repository.lock_write()
65
for i in xrange(1000):
66
tree.commit('no-changes commit %d' % i)
108
No file changes are included. Not hardlinking the working tree,
109
because there are no working tree files.
111
from bzrlib.benchmarks.tree_creator.simple_many_commit import (
112
SimpleManyCommitTreeCreator,
114
creator = SimpleManyCommitTreeCreator(self, link_bzr=hardlink)
115
return creator.create(root=directory_name)
117
def make_heavily_merged_tree(self, directory_name='.',
119
"""Create a tree in which almost every commit is a merge.
121
No file changes are included. This produces two trees,
122
one of which is returned. Except for the first commit, every
123
commit in its revision-history is a merge another commit in the other
124
tree. Not hardlinking the working tree, because there are no working
127
from bzrlib.benchmarks.tree_creator.heavily_merged import (
128
HeavilyMergedTreeCreator,
130
creator = HeavilyMergedTreeCreator(self, link_bzr=hardlink)
131
return creator.create(root=directory_name)
133
def create_with_commits(self, num_files, num_commits, directory_name='.',
135
"""Create a tree with many files and many commits. Every commit changes
138
:param num_files: number of files to be created
139
:param num_commits: number of commits in the newly created tree
141
from bzrlib.benchmarks.tree_creator.many_commit import (
142
ManyCommitTreeCreator,
144
creator = ManyCommitTreeCreator(self, link_bzr=hardlink,
146
num_commits=num_commits)
147
tree = creator.create(root=directory_name)
148
files = ["%s/%s" % (directory_name, fn) for fn in creator.files]
151
def commit_some_revisions(self, tree, files, num_commits,
153
"""Commit a specified number of revisions to some files in a tree,
154
makeing a specified number of changes per commit.
156
:param tree: The tree in which the changes happen.
157
:param files: The list of files where changes should occur.
158
:param num_commits: The number of commits
159
:param changes_per_commit: The number of files that are touched in
162
for j in range(num_commits):
163
for i in range(changes_per_commit):
164
fn = files[(i + j) % changes_per_commit]
165
content = range(i) + [i, i, i, '']
70
tree.branch.repository.unlock()
168
f.write("\n".join([str(k) for k in content]))
171
tree.commit("new revision")
80
176
testmod_names = [ \
81
177
'bzrlib.benchmarks.bench_add',
82
178
'bzrlib.benchmarks.bench_bench',
179
'bzrlib.benchmarks.bench_bundle',
180
'bzrlib.benchmarks.bench_cache_utf8',
83
181
'bzrlib.benchmarks.bench_checkout',
84
182
'bzrlib.benchmarks.bench_commit',
183
'bzrlib.benchmarks.bench_dirstate',
184
'bzrlib.benchmarks.bench_info',
85
185
'bzrlib.benchmarks.bench_inventory',
186
'bzrlib.benchmarks.bench_knit',
86
187
'bzrlib.benchmarks.bench_log',
188
'bzrlib.benchmarks.bench_pack',
87
189
'bzrlib.benchmarks.bench_osutils',
88
190
'bzrlib.benchmarks.bench_rocks',
191
'bzrlib.benchmarks.bench_startup',
89
192
'bzrlib.benchmarks.bench_status',
90
193
'bzrlib.benchmarks.bench_transform',
91
194
'bzrlib.benchmarks.bench_workingtree',
195
'bzrlib.benchmarks.bench_sftp',
196
'bzrlib.benchmarks.bench_xml',
93
return TestLoader().loadTestsFromModuleNames(testmod_names)
198
suite = TestLoader().loadTestsFromModuleNames(testmod_names)
200
# Load any benchmarks from plugins
201
for name, plugin in _mod_plugin.plugins().items():
202
if getattr(plugin.module, 'bench_suite', None) is not None:
203
suite.addTest(plugin.module.bench_suite())