mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-25 14:08:19 +01:00
Improve performance of repository discovery in repositories with >65K refs
Summary: Ref T13593. The commit cache in this Engine has a maximum fixed size (currently 65,535 entries). If we execute discovery in a repository with more refs than this (e.g., 180K), we get fast lookups for the first 65,535 refs and slow lookups for the remaining refs. Instead, divide the refs into chunks no larger than the cache size, and perform an explicit cache fill before each chunk is processed. Test Plan: - Created a repository with 1K refs. Set cache size to 256. Ran discovery. - Before patch: saw one large cache fill and then ~750 single-gets. - After patch: saw four large cache fills. - Compared `bin/repository discover ... --verbose` output before and after patch for overall effect; saw no differences. Maniphest Tasks: T13593 Differential Revision: https://secure.phabricator.com/D21521
This commit is contained in:
parent
888604c9dd
commit
ed86c42b26
1 changed files with 69 additions and 44 deletions
|
@ -142,61 +142,86 @@ final class PhabricatorRepositoryDiscoveryEngine
|
||||||
return array();
|
return array();
|
||||||
}
|
}
|
||||||
|
|
||||||
$heads = $this->sortRefs($heads);
|
|
||||||
$head_commits = mpull($heads, 'getCommitIdentifier');
|
|
||||||
|
|
||||||
$this->log(
|
$this->log(
|
||||||
pht(
|
pht(
|
||||||
'Discovering commits in repository "%s".',
|
'Discovering commits in repository "%s".',
|
||||||
$repository->getDisplayName()));
|
$repository->getDisplayName()));
|
||||||
|
|
||||||
$this->fillCommitCache($head_commits);
|
$ref_lists = array();
|
||||||
|
|
||||||
$refs = array();
|
$head_groups = $this->getRefGroupsForDiscovery($heads);
|
||||||
foreach ($heads as $ref) {
|
foreach ($head_groups as $head_group) {
|
||||||
$name = $ref->getShortName();
|
|
||||||
$commit = $ref->getCommitIdentifier();
|
|
||||||
|
|
||||||
$this->log(
|
$group_identifiers = mpull($head_group, 'getCommitIdentifier');
|
||||||
pht(
|
$group_identifiers = array_fuse($group_identifiers);
|
||||||
'Examining "%s" (%s) at "%s".',
|
$this->fillCommitCache($group_identifiers);
|
||||||
$name,
|
|
||||||
$ref->getRefType(),
|
|
||||||
$commit));
|
|
||||||
|
|
||||||
if (!$repository->shouldTrackRef($ref)) {
|
foreach ($head_group as $ref) {
|
||||||
$this->log(pht('Skipping, ref is untracked.'));
|
$name = $ref->getShortName();
|
||||||
continue;
|
$commit = $ref->getCommitIdentifier();
|
||||||
|
|
||||||
|
$this->log(
|
||||||
|
pht(
|
||||||
|
'Examining "%s" (%s) at "%s".',
|
||||||
|
$name,
|
||||||
|
$ref->getRefType(),
|
||||||
|
$commit));
|
||||||
|
|
||||||
|
if (!$repository->shouldTrackRef($ref)) {
|
||||||
|
$this->log(pht('Skipping, ref is untracked.'));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->isKnownCommit($commit)) {
|
||||||
|
$this->log(pht('Skipping, HEAD is known.'));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// In Git, it's possible to tag anything. We just skip tags that don't
|
||||||
|
// point to a commit. See T11301.
|
||||||
|
$fields = $ref->getRawFields();
|
||||||
|
$ref_type = idx($fields, 'objecttype');
|
||||||
|
$tag_type = idx($fields, '*objecttype');
|
||||||
|
if ($ref_type != 'commit' && $tag_type != 'commit') {
|
||||||
|
$this->log(pht('Skipping, this is not a commit.'));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log(pht('Looking for new commits.'));
|
||||||
|
|
||||||
|
$head_refs = $this->discoverStreamAncestry(
|
||||||
|
new PhabricatorGitGraphStream($repository, $commit),
|
||||||
|
$commit,
|
||||||
|
$publisher->isPermanentRef($ref));
|
||||||
|
|
||||||
|
$this->didDiscoverRefs($head_refs);
|
||||||
|
|
||||||
|
$ref_lists[] = $head_refs;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->isKnownCommit($commit)) {
|
|
||||||
$this->log(pht('Skipping, HEAD is known.'));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
// In Git, it's possible to tag anything. We just skip tags that don't
|
|
||||||
// point to a commit. See T11301.
|
|
||||||
$fields = $ref->getRawFields();
|
|
||||||
$ref_type = idx($fields, 'objecttype');
|
|
||||||
$tag_type = idx($fields, '*objecttype');
|
|
||||||
if ($ref_type != 'commit' && $tag_type != 'commit') {
|
|
||||||
$this->log(pht('Skipping, this is not a commit.'));
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->log(pht('Looking for new commits.'));
|
|
||||||
|
|
||||||
$head_refs = $this->discoverStreamAncestry(
|
|
||||||
new PhabricatorGitGraphStream($repository, $commit),
|
|
||||||
$commit,
|
|
||||||
$publisher->isPermanentRef($ref));
|
|
||||||
|
|
||||||
$this->didDiscoverRefs($head_refs);
|
|
||||||
|
|
||||||
$refs[] = $head_refs;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return array_mergev($refs);
|
$refs = array_mergev($ref_lists);
|
||||||
|
|
||||||
|
return $refs;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @task git
|
||||||
|
*/
|
||||||
|
private function getRefGroupsForDiscovery(array $heads) {
|
||||||
|
$heads = $this->sortRefs($heads);
|
||||||
|
|
||||||
|
// See T13593. We hold a commit cache with a fixed maximum size. Split the
|
||||||
|
// refs into chunks no larger than the cache size, so we don't overflow the
|
||||||
|
// cache when testing them.
|
||||||
|
|
||||||
|
$array_iterator = new ArrayIterator($heads);
|
||||||
|
|
||||||
|
$chunk_iterator = new PhutilChunkedIterator(
|
||||||
|
$array_iterator,
|
||||||
|
self::MAX_COMMIT_CACHE_SIZE);
|
||||||
|
|
||||||
|
return $chunk_iterator;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
Loading…
Add table
Reference in a new issue