3 final class PhabricatorEditorURIEngine
10 private $repositoryTokens;
12 public static function newForViewer(PhabricatorUser
$viewer) {
13 if (!$viewer->isLoggedIn()) {
17 $pattern = $viewer->getUserSetting(PhabricatorEditorSetting
::SETTINGKEY
);
19 if ($pattern === null ||
!strlen(trim($pattern))) {
23 $engine = id(new self())
25 ->setPattern($pattern);
27 // If there's a problem with the pattern,
30 $engine->validatePattern();
31 } catch (PhabricatorEditorURIParserException
$ex) {
38 public function setViewer(PhabricatorUser
$viewer) {
39 $this->viewer
= $viewer;
43 public function getViewer() {
47 public function setRepository(PhabricatorRepository
$repository) {
48 $this->repository
= $repository;
52 public function getRepository() {
53 return $this->repository
;
56 public function setPattern($pattern) {
57 $this->pattern
= $pattern;
61 public function getPattern() {
62 return $this->pattern
;
65 public function validatePattern() {
66 $this->getRawURITokens();
70 public function getURIForPath($path, $line) {
71 $tokens = $this->getURITokensForRepository($path);
74 'f' => $this->escapeToken($path),
75 'l' => $this->escapeToken($line),
78 $tokens = $this->newTokensWithVariables($tokens, $variables);
80 return $this->newStringFromTokens($tokens);
83 public function getURITokensForPath($path) {
84 $tokens = $this->getURITokensForRepository($path);
87 'f' => $this->escapeToken($path),
90 return $this->newTokensWithVariables($tokens, $variables);
93 public static function getVariableDefinitions() {
96 'name' => pht('File Name'),
97 'example' => pht('path/to/source.c'),
100 'name' => pht('Line Number'),
104 'name' => pht('Repository Short Name'),
105 'example' => 'arcanist',
108 'name' => pht('Repository ID'),
112 'name' => pht('Repository PHID'),
113 'example' => 'PHID-REPO-abcdefghijklmnopqrst',
116 'name' => pht('Repository Callsign'),
120 'name' => pht('Literal Percent Symbol'),
126 private function getURITokensForRepository() {
127 if (!$this->repositoryTokens
) {
128 $this->repositoryTokens
= $this->newURITokensForRepository();
131 return $this->repositoryTokens
;
134 private function newURITokensForRepository() {
135 $tokens = $this->getRawURITokens();
137 $repository = $this->getRepository();
139 throw new PhutilInvalidStateException('setRepository');
143 'r' => $this->escapeToken($repository->getCallsign()),
144 'n' => $this->escapeToken($repository->getRepositorySlug()),
145 'd' => $this->escapeToken($repository->getID()),
146 'p' => $this->escapeToken($repository->getPHID()),
149 return $this->newTokensWithVariables($tokens, $variables);
152 private function getRawURITokens() {
153 if (!$this->rawTokens
) {
154 $this->rawTokens
= $this->newRawURITokens();
156 return $this->rawTokens
;
159 private function newRawURITokens() {
160 $raw_pattern = $this->getPattern();
161 $raw_tokens = self
::newPatternTokens($raw_pattern);
163 $variable_definitions = self
::getVariableDefinitions();
165 foreach ($raw_tokens as $token) {
166 if ($token['type'] !== 'variable') {
170 $value = $token['value'];
172 if (isset($variable_definitions[$value])) {
176 throw new PhabricatorEditorURIParserException(
178 'Editor pattern "%s" is invalid: the pattern contains an '.
179 'unrecognized variable ("%s"). Use "%%%%" to encode a literal '.
189 $tokens = $this->newTokensWithVariables($raw_tokens, $variables);
191 $first_literal = null;
193 foreach ($tokens as $token) {
194 if ($token['type'] === 'literal') {
195 $first_literal = $token['value'];
200 if ($first_literal === null) {
201 throw new PhabricatorEditorURIParserException(
203 'Editor pattern "%s" is invalid: the pattern must begin with '.
204 'a valid editor protocol, but begins with a variable. This is '.
205 'very sneaky and also very forbidden.',
210 $uri = new PhutilURI($first_literal);
211 $editor_protocol = $uri->getProtocol();
213 if (!$editor_protocol) {
214 throw new PhabricatorEditorURIParserException(
216 'Editor pattern "%s" is invalid: the pattern must begin with '.
217 'a valid editor protocol, but does not begin with a recognized '.
222 $allowed_key = 'uri.allowed-editor-protocols';
223 $allowed_protocols = PhabricatorEnv
::getEnvConfig($allowed_key);
224 if (empty($allowed_protocols[$editor_protocol])) {
225 throw new PhabricatorEditorURIParserException(
227 'Editor pattern "%s" is invalid: the pattern must begin with '.
228 'a valid editor protocol, but the protocol "%s://" is not allowed.',
236 private function newTokensWithVariables(array $tokens, array $variables) {
237 // Replace all "variable" tokens that we have replacements for with
238 // the literal value.
239 foreach ($tokens as $key => $token) {
240 $type = $token['type'];
242 if ($type == 'variable') {
243 $variable = $token['value'];
244 if (isset($variables[$variable])) {
245 $tokens[$key] = array(
247 'value' => $variables[$variable],
253 // Now, merge sequences of adjacent "literal" tokens into a single token.
254 $last_literal = null;
255 foreach ($tokens as $key => $token) {
256 $is_literal = ($token['type'] === 'literal');
259 $last_literal = null;
263 if ($last_literal !== null) {
264 $tokens[$key]['value'] =
265 $tokens[$last_literal]['value'].$token['value'];
266 unset($tokens[$last_literal]);
269 $last_literal = $key;
272 $tokens = array_values($tokens);
277 private function escapeToken($token) {
278 // Paths are user controlled, so a clever user could potentially make
279 // editor links do surprising things with paths containing "/../".
281 // Find anything that looks like "/../" and mangle it.
283 $token = preg_replace('((^|/)\.\.(/|\z))', '\1dot-dot\2', $token);
285 return phutil_escape_uri($token);
288 private function newStringFromTokens(array $tokens) {
291 foreach ($tokens as $token) {
292 $token_type = $token['type'];
293 $token_value = $token['value'];
295 $is_literal = ($token_type === 'literal');
299 'Editor pattern token list can not be converted into a string: '.
300 'it still contains a non-literal token ("%s", of type "%s").',
305 $result[] = $token_value;
308 $result = implode('', $result);
313 public static function newPatternTokens($raw_pattern) {
314 $token_positions = array();
316 $len = strlen($raw_pattern);
318 for ($ii = 0; $ii < $len; $ii++
) {
319 $c = $raw_pattern[$ii];
321 if (!isset($raw_pattern[$ii +
1])) {
322 throw new PhabricatorEditorURIParserException(
324 'Editor pattern "%s" is invalid: the final character in a '.
325 'pattern may not be an unencoded percent symbol ("%%"). '.
326 'Use "%%%%" to encode a literal percent symbol.',
330 $token_positions[] = $ii;
335 // Add a final marker past the end of the string, so we'll collect any
336 // trailing literal bytes.
337 $token_positions[] = $len;
341 foreach ($token_positions as $pos) {
342 $token_len = ($pos - $cursor);
344 if ($token_len > 0) {
347 'value' => substr($raw_pattern, $cursor, $token_len),
353 if ($cursor < $len) {
355 'type' => 'variable',
356 'value' => substr($raw_pattern, $cursor +
1, 1),