Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
freebsd
GitHub Repository: freebsd/phabricator
Path: blob/master/src/applications/diffusion/conduit/DiffusionBranchQueryConduitAPIMethod.php
12242 views
1
<?php
2
3
final class DiffusionBranchQueryConduitAPIMethod
4
extends DiffusionQueryConduitAPIMethod {
5
6
public function getAPIMethodName() {
7
return 'diffusion.branchquery';
8
}
9
10
public function getMethodDescription() {
11
return pht('Determine what branches exist for a repository.');
12
}
13
14
protected function defineReturnType() {
15
return 'list<dict>';
16
}
17
18
protected function defineCustomParamTypes() {
19
return array(
20
'closed' => 'optional bool',
21
'limit' => 'optional int',
22
'offset' => 'optional int',
23
'contains' => 'optional string',
24
'patterns' => 'optional list<string>',
25
);
26
}
27
28
protected function getGitResult(ConduitAPIRequest $request) {
29
$drequest = $this->getDiffusionRequest();
30
$repository = $drequest->getRepository();
31
32
$contains = $request->getValue('contains');
33
if ($contains !== null && strlen($contains)) {
34
35
// See PHI958 (and, earlier, PHI720). If "patterns" are provided, pass
36
// them to "git branch ..." to let callers test for reachability from
37
// particular branch heads.
38
$patterns_argv = $request->getValue('patterns', array());
39
PhutilTypeSpec::checkMap(
40
array(
41
'patterns' => $patterns_argv,
42
),
43
array(
44
'patterns' => 'list<string>',
45
));
46
47
// NOTE: We can't use DiffusionLowLevelGitRefQuery here because
48
// `git for-each-ref` does not support `--contains`.
49
list($stdout) = $repository->execxLocalCommand(
50
'branch --verbose --no-abbrev --contains %s -- %Ls',
51
$contains,
52
$patterns_argv);
53
$ref_map = DiffusionGitBranch::parseLocalBranchOutput(
54
$stdout);
55
56
$refs = array();
57
foreach ($ref_map as $ref => $commit) {
58
$refs[] = id(new DiffusionRepositoryRef())
59
->setShortName($ref)
60
->setCommitIdentifier($commit);
61
}
62
} else {
63
$refs = id(new DiffusionLowLevelGitRefQuery())
64
->setRepository($repository)
65
->withRefTypes(
66
array(
67
PhabricatorRepositoryRefCursor::TYPE_BRANCH,
68
))
69
->execute();
70
}
71
72
return $this->processBranchRefs($request, $refs);
73
}
74
75
protected function getMercurialResult(ConduitAPIRequest $request) {
76
$drequest = $this->getDiffusionRequest();
77
$repository = $drequest->getRepository();
78
79
$query = id(new DiffusionLowLevelMercurialBranchesQuery())
80
->setRepository($repository);
81
82
$contains = $request->getValue('contains');
83
if ($contains !== null && strlen($contains)) {
84
$query->withContainsCommit($contains);
85
}
86
87
$refs = $query->execute();
88
89
return $this->processBranchRefs($request, $refs);
90
}
91
92
protected function getSVNResult(ConduitAPIRequest $request) {
93
// Since SVN doesn't have meaningful branches, just return nothing for all
94
// queries.
95
return array();
96
}
97
98
private function processBranchRefs(ConduitAPIRequest $request, array $refs) {
99
$drequest = $this->getDiffusionRequest();
100
$repository = $drequest->getRepository();
101
$offset = $request->getValue('offset');
102
$limit = $request->getValue('limit');
103
104
foreach ($refs as $key => $ref) {
105
if (!$repository->shouldTrackBranch($ref->getShortName())) {
106
unset($refs[$key]);
107
}
108
}
109
110
$with_closed = $request->getValue('closed');
111
if ($with_closed !== null) {
112
foreach ($refs as $key => $ref) {
113
$fields = $ref->getRawFields();
114
if (idx($fields, 'closed') != $with_closed) {
115
unset($refs[$key]);
116
}
117
}
118
}
119
120
// NOTE: We can't apply the offset or limit until here, because we may have
121
// filtered untrackable branches out of the result set.
122
123
if ($offset) {
124
$refs = array_slice($refs, $offset);
125
}
126
127
if ($limit) {
128
$refs = array_slice($refs, 0, $limit);
129
}
130
131
$refs = array_values($refs);
132
133
return mpull($refs, 'toDictionary');
134
}
135
136
}
137
138