2011-01-09 15:22:25 -08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright 2011 Facebook, Inc.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2011-02-19 11:36:08 -08:00
|
|
|
/**
|
|
|
|
* Interfaces with Subversion working copies.
|
|
|
|
*
|
|
|
|
* @group workingcopy
|
|
|
|
*/
|
2011-01-09 15:22:25 -08:00
|
|
|
class ArcanistSubversionAPI extends ArcanistRepositoryAPI {
|
|
|
|
|
|
|
|
protected $svnStatus;
|
|
|
|
protected $svnBaseRevisions;
|
|
|
|
protected $svnInfo = array();
|
|
|
|
|
|
|
|
protected $svnInfoRaw = array();
|
|
|
|
protected $svnDiffRaw = array();
|
2011-02-24 16:34:27 -08:00
|
|
|
|
2011-02-23 12:06:22 -08:00
|
|
|
private $svnBaseRevisionNumber;
|
2011-01-09 15:22:25 -08:00
|
|
|
|
|
|
|
public function getSourceControlSystemName() {
|
|
|
|
return 'svn';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function hasMergeConflicts() {
|
|
|
|
foreach ($this->getSVNStatus() as $path => $mask) {
|
|
|
|
if ($mask & self::FLAG_CONFLICT) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getWorkingCopyStatus() {
|
|
|
|
return $this->getSVNStatus();
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSVNBaseRevisions() {
|
|
|
|
if ($this->svnBaseRevisions === null) {
|
|
|
|
$this->getSVNStatus();
|
|
|
|
}
|
|
|
|
return $this->svnBaseRevisions;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSVNStatus($with_externals = false) {
|
|
|
|
if ($this->svnStatus === null) {
|
|
|
|
list($status) = execx('(cd %s && svn --xml status)', $this->getPath());
|
|
|
|
$xml = new SimpleXMLElement($status);
|
|
|
|
|
|
|
|
if (count($xml->target) != 1) {
|
|
|
|
throw new Exception("Expected exactly one XML status target.");
|
|
|
|
}
|
|
|
|
|
|
|
|
$externals = array();
|
|
|
|
$files = array();
|
|
|
|
|
|
|
|
$target = $xml->target[0];
|
|
|
|
$this->svnBaseRevisions = array();
|
|
|
|
foreach ($target->entry as $entry) {
|
|
|
|
$path = (string)$entry['path'];
|
|
|
|
$mask = 0;
|
|
|
|
|
|
|
|
$props = (string)($entry->{'wc-status'}[0]['props']);
|
|
|
|
$item = (string)($entry->{'wc-status'}[0]['item']);
|
|
|
|
|
|
|
|
$base = (string)($entry->{'wc-status'}[0]['revision']);
|
|
|
|
$this->svnBaseRevisions[$path] = $base;
|
|
|
|
|
|
|
|
switch ($props) {
|
|
|
|
case 'none':
|
|
|
|
case 'normal':
|
|
|
|
break;
|
|
|
|
case 'modified':
|
|
|
|
$mask |= self::FLAG_MODIFIED;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Exception("Unrecognized property status '{$props}'.");
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ($item) {
|
|
|
|
case 'normal':
|
|
|
|
break;
|
|
|
|
case 'external':
|
|
|
|
$mask |= self::FLAG_EXTERNALS;
|
|
|
|
$externals[] = $path;
|
|
|
|
break;
|
|
|
|
case 'unversioned':
|
|
|
|
$mask |= self::FLAG_UNTRACKED;
|
|
|
|
break;
|
|
|
|
case 'obstructed':
|
|
|
|
$mask |= self::FLAG_OBSTRUCTED;
|
|
|
|
break;
|
|
|
|
case 'missing':
|
|
|
|
$mask |= self::FLAG_MISSING;
|
|
|
|
break;
|
|
|
|
case 'added':
|
|
|
|
$mask |= self::FLAG_ADDED;
|
|
|
|
break;
|
2011-03-07 19:03:27 -08:00
|
|
|
case 'replaced':
|
|
|
|
// This is the result of "svn rm"-ing a file, putting another one
|
|
|
|
// in place of it, and then "svn add"-ing the new file. Just treat
|
|
|
|
// this as equivalent to "modified".
|
|
|
|
$mask |= self::FLAG_MODIFIED;
|
|
|
|
break;
|
2011-01-09 15:22:25 -08:00
|
|
|
case 'modified':
|
|
|
|
$mask |= self::FLAG_MODIFIED;
|
|
|
|
break;
|
|
|
|
case 'deleted':
|
|
|
|
$mask |= self::FLAG_DELETED;
|
|
|
|
break;
|
2011-02-21 16:59:37 -08:00
|
|
|
case 'conflicted':
|
|
|
|
$mask |= self::FLAG_CONFLICT;
|
|
|
|
break;
|
2011-03-12 17:57:35 -08:00
|
|
|
case 'incomplete':
|
|
|
|
$mask |= self::FLAG_INCOMPLETE;
|
|
|
|
break;
|
2011-01-09 15:22:25 -08:00
|
|
|
default:
|
|
|
|
throw new Exception("Unrecognized item status '{$item}'.");
|
|
|
|
}
|
|
|
|
|
2011-03-20 15:06:55 -07:00
|
|
|
// This is new in or around Subversion 1.6.
|
|
|
|
$tree_conflicts = (string)($entry->{'wc-status'}[0]['tree-conflicted']);
|
|
|
|
if ($tree_conflicts) {
|
|
|
|
$mask |= self::FLAG_CONFLICT;
|
|
|
|
}
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
$files[$path] = $mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ($files as $path => $mask) {
|
|
|
|
foreach ($externals as $external) {
|
|
|
|
if (!strncmp($path, $external, strlen($external))) {
|
|
|
|
$files[$path] |= self::FLAG_EXTERNALS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->svnStatus = $files;
|
|
|
|
}
|
|
|
|
|
|
|
|
$status = $this->svnStatus;
|
|
|
|
if (!$with_externals) {
|
|
|
|
foreach ($status as $path => $mask) {
|
|
|
|
if ($mask & ArcanistRepositoryAPI::FLAG_EXTERNALS) {
|
|
|
|
unset($status[$path]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $status;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSVNProperty($path, $property) {
|
|
|
|
list($stdout) = execx(
|
|
|
|
'svn propget %s %s@',
|
|
|
|
$property,
|
|
|
|
$this->getPath($path));
|
|
|
|
return trim($stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSourceControlPath() {
|
|
|
|
return idx($this->getSVNInfo('/'), 'URL');
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSourceControlBaseRevision() {
|
|
|
|
$info = $this->getSVNInfo('/');
|
2011-02-23 12:06:22 -08:00
|
|
|
return $info['URL'].'@'.$this->getSVNBaseRevisionNumber();
|
|
|
|
}
|
2011-02-24 16:34:27 -08:00
|
|
|
|
2011-02-23 12:06:22 -08:00
|
|
|
public function getSVNBaseRevisionNumber() {
|
|
|
|
if ($this->svnBaseRevisionNumber) {
|
|
|
|
return $this->svnBaseRevisionNumber;
|
|
|
|
}
|
|
|
|
$info = $this->getSVNInfo('/');
|
|
|
|
return $info['Revision'];
|
|
|
|
}
|
2011-02-24 16:34:27 -08:00
|
|
|
|
2011-02-23 12:06:22 -08:00
|
|
|
public function overrideSVNBaseRevisionNumber($effective_base_revision) {
|
|
|
|
$this->svnBaseRevisionNumber = $effective_base_revision;
|
|
|
|
return $this;
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
public function getBranchName() {
|
|
|
|
return 'svn';
|
|
|
|
}
|
|
|
|
|
|
|
|
public function buildInfoFuture($path) {
|
2011-01-11 14:26:21 -08:00
|
|
|
if ($path == '/') {
|
|
|
|
// When the root of a working copy is referenced by a symlink and you
|
|
|
|
// execute 'svn info' on that symlink, svn fails. This is a longstanding
|
|
|
|
// bug in svn:
|
|
|
|
//
|
|
|
|
// See http://subversion.tigris.org/issues/show_bug.cgi?id=2305
|
|
|
|
//
|
|
|
|
// To reproduce, do:
|
|
|
|
//
|
|
|
|
// $ ln -s working_copy working_link
|
2011-01-12 15:45:17 -08:00
|
|
|
// $ svn info working_copy # ok
|
2011-01-11 14:26:21 -08:00
|
|
|
// $ svn info working_link # fails
|
|
|
|
//
|
|
|
|
// Work around this by cd-ing into the directory before executing
|
|
|
|
// 'svn info'.
|
|
|
|
return new ExecFuture(
|
|
|
|
'(cd %s && svn info .)',
|
|
|
|
$this->getPath());
|
|
|
|
} else {
|
|
|
|
// Note: here and elsewhere we need to append "@" to the path because if
|
|
|
|
// a file has a literal "@" in it, everything after that will be
|
|
|
|
// interpreted as a revision. By appending "@" with no argument, SVN
|
|
|
|
// parses it properly.
|
|
|
|
return new ExecFuture(
|
|
|
|
'svn info %s@',
|
|
|
|
$this->getPath($path));
|
|
|
|
}
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
public function buildDiffFuture($path) {
|
|
|
|
// The "--depth empty" flag prevents us from picking up changes in
|
|
|
|
// children when we run 'diff' against a directory.
|
|
|
|
return new ExecFuture(
|
|
|
|
'(cd %s; svn diff --depth empty --diff-cmd diff -x -U%d %s)',
|
|
|
|
$this->getPath(),
|
|
|
|
$this->getDiffLinesOfContext(),
|
|
|
|
$path);
|
|
|
|
}
|
|
|
|
|
|
|
|
public function primeSVNInfoResult($path, $result) {
|
|
|
|
$this->svnInfoRaw[$path] = $result;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function primeSVNDiffResult($path, $result) {
|
|
|
|
$this->svnDiffRaw[$path] = $result;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getSVNInfo($path) {
|
|
|
|
|
|
|
|
if (empty($this->svnInfo[$path])) {
|
|
|
|
|
|
|
|
if (empty($this->svnInfoRaw[$path])) {
|
|
|
|
$this->svnInfoRaw[$path] = $this->buildInfoFuture($path)->resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
list($err, $stdout) = $this->svnInfoRaw[$path];
|
|
|
|
if ($err) {
|
|
|
|
throw new Exception(
|
|
|
|
"Error #{$err} executing svn info against '{$path}'.");
|
|
|
|
}
|
|
|
|
|
|
|
|
$patterns = array(
|
|
|
|
'/^(URL): (\S+)$/m',
|
|
|
|
'/^(Revision): (\d+)$/m',
|
|
|
|
'/^(Last Changed Author): (\S+)$/m',
|
|
|
|
'/^(Last Changed Rev): (\d+)$/m',
|
|
|
|
'/^(Last Changed Date): (.+) \(.+\)$/m',
|
|
|
|
'/^(Copied From URL): (\S+)$/m',
|
|
|
|
'/^(Copied From Rev): (\d+)$/m',
|
2011-04-05 22:27:32 -07:00
|
|
|
'/^(Repository UUID): (\S+)$/m',
|
2011-01-09 15:22:25 -08:00
|
|
|
);
|
|
|
|
|
|
|
|
$result = array();
|
|
|
|
foreach ($patterns as $pattern) {
|
|
|
|
$matches = null;
|
|
|
|
if (preg_match($pattern, $stdout, $matches)) {
|
|
|
|
$result[$matches[1]] = $matches[2];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isset($result['Last Changed Date'])) {
|
|
|
|
$result['Last Changed Date'] = strtotime($result['Last Changed Date']);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($result)) {
|
|
|
|
throw new Exception('Unable to parse SVN info.');
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->svnInfo[$path] = $result;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->svnInfo[$path];
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public function getRawDiffText($path) {
|
|
|
|
$status = $this->getSVNStatus();
|
|
|
|
if (!isset($status[$path])) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$status = $status[$path];
|
|
|
|
|
|
|
|
// Build meaningful diff text for "svn copy" operations.
|
|
|
|
if ($status & ArcanistRepositoryAPI::FLAG_ADDED) {
|
|
|
|
$info = $this->getSVNInfo($path);
|
|
|
|
if (!empty($info['Copied From URL'])) {
|
|
|
|
return $this->buildSyntheticAdditionDiff(
|
|
|
|
$path,
|
|
|
|
$info['Copied From URL'],
|
|
|
|
$info['Copied From Rev']);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we run "diff" on a binary file which doesn't have the "svn:mime-type"
|
|
|
|
// of "application/octet-stream", `diff' will explode in a rain of
|
|
|
|
// unhelpful hellfire as it tries to build a textual diff of the two
|
|
|
|
// files. We just fix this inline since it's pretty unambiguous.
|
|
|
|
// TODO: Move this to configuration?
|
|
|
|
$matches = null;
|
|
|
|
if (preg_match('/\.(gif|png|jpe?g|swf|pdf|ico)$/i', $path, $matches)) {
|
|
|
|
$mime = $this->getSVNProperty($path, 'svn:mime-type');
|
|
|
|
if ($mime != 'application/octet-stream') {
|
|
|
|
execx(
|
|
|
|
'svn propset svn:mime-type application/octet-stream %s',
|
|
|
|
$this->getPath($path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($this->svnDiffRaw[$path])) {
|
|
|
|
$this->svnDiffRaw[$path] = $this->buildDiffFuture($path)->resolve();
|
|
|
|
}
|
|
|
|
|
|
|
|
list($err, $stdout, $stderr) = $this->svnDiffRaw[$path];
|
|
|
|
|
|
|
|
// Note: GNU Diff returns 2 when SVN hands it binary files to diff and they
|
|
|
|
// differ. This is not an error; it is documented behavior. But SVN isn't
|
|
|
|
// happy about it. SVN will exit with code 1 and return the string below.
|
|
|
|
if ($err != 0 && $stderr !== "svn: 'diff' returned 2\n") {
|
|
|
|
throw new Exception(
|
|
|
|
"svn diff returned unexpected error code: $err\n".
|
|
|
|
"stdout: $stdout\n".
|
|
|
|
"stderr: $stderr");
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($err == 0 && empty($stdout)) {
|
|
|
|
// If there are no changes, 'diff' exits with no output, but that means
|
|
|
|
// we can not distinguish between empty and unmodified files. Build a
|
|
|
|
// synthetic "diff" without any changes in it.
|
|
|
|
return $this->buildSyntheticUnchangedDiff($path);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $stdout;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function buildSyntheticAdditionDiff($path, $source, $rev) {
|
|
|
|
$type = $this->getSVNProperty($path, 'svn:mime-type');
|
|
|
|
if ($type == 'application/octet-stream') {
|
|
|
|
return <<<EODIFF
|
|
|
|
Index: {$path}
|
|
|
|
===================================================================
|
|
|
|
Cannot display: file marked as a binary type.
|
|
|
|
svn:mime-type = application/octet-stream
|
|
|
|
|
|
|
|
EODIFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (is_dir($this->getPath($path))) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$data = Filesystem::readFile($this->getPath($path));
|
|
|
|
list($orig) = execx('svn cat %s@%s', $source, $rev);
|
|
|
|
|
|
|
|
$src = new TempFile();
|
|
|
|
$dst = new TempFile();
|
|
|
|
Filesystem::writeFile($src, $orig);
|
|
|
|
Filesystem::writeFile($dst, $data);
|
|
|
|
|
|
|
|
list($err, $diff) = exec_manual(
|
|
|
|
'diff -L a/%s -L b/%s -U%d %s %s',
|
|
|
|
str_replace($this->getSourceControlPath().'/', '', $source),
|
|
|
|
$path,
|
|
|
|
$this->getDiffLinesOfContext(),
|
|
|
|
$src,
|
|
|
|
$dst);
|
|
|
|
|
|
|
|
if ($err == 1) { // 1 means there are differences.
|
|
|
|
return <<<EODIFF
|
|
|
|
Index: {$path}
|
|
|
|
===================================================================
|
|
|
|
{$diff}
|
|
|
|
|
|
|
|
EODIFF;
|
|
|
|
} else {
|
|
|
|
return $this->buildSyntheticUnchangedDiff($path);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function buildSyntheticUnchangedDiff($path) {
|
|
|
|
$full_path = $this->getPath($path);
|
|
|
|
if (is_dir($full_path)) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$data = Filesystem::readFile($full_path);
|
|
|
|
$lines = explode("\n", $data);
|
|
|
|
$len = count($lines);
|
|
|
|
foreach ($lines as $key => $line) {
|
|
|
|
$lines[$key] = ' '.$line;
|
|
|
|
}
|
|
|
|
$lines = implode("\n", $lines);
|
|
|
|
return <<<EODIFF
|
|
|
|
Index: {$path}
|
|
|
|
===================================================================
|
|
|
|
--- {$path} (synthetic)
|
|
|
|
+++ {$path} (synthetic)
|
|
|
|
@@ -1,{$len} +1,{$len} @@
|
|
|
|
{$lines}
|
|
|
|
|
|
|
|
EODIFF;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getBlame($path) {
|
|
|
|
$blame = array();
|
|
|
|
|
|
|
|
list($stdout) = execx(
|
|
|
|
'(cd %s && svn blame %s)',
|
|
|
|
$this->getPath(),
|
|
|
|
$path);
|
|
|
|
|
|
|
|
$stdout = trim($stdout);
|
|
|
|
if (!strlen($stdout)) {
|
|
|
|
// Empty file.
|
|
|
|
return $blame;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (explode("\n", $stdout) as $line) {
|
|
|
|
$m = array();
|
|
|
|
if (!preg_match('/^\s*(\d+)\s+(\S+)/', $line, $m)) {
|
|
|
|
throw new Exception("Bad blame? `{$line}'");
|
|
|
|
}
|
|
|
|
$revision = $m[1];
|
|
|
|
$author = $m[2];
|
|
|
|
$blame[] = array($author, $revision);
|
|
|
|
}
|
|
|
|
|
|
|
|
return $blame;
|
|
|
|
}
|
2011-01-12 15:45:17 -08:00
|
|
|
|
2011-01-11 13:02:38 -08:00
|
|
|
public function getOriginalFileData($path) {
|
|
|
|
// SVN issues warnings for nonexistent paths, directories, etc., but still
|
|
|
|
// returns no error code. However, for new paths in the working copy it
|
|
|
|
// fails. Assume that failure means the original file does not exist.
|
|
|
|
list($err, $stdout) = exec_manual(
|
|
|
|
'(cd %s && svn cat %s@)',
|
|
|
|
$this->getPath(),
|
|
|
|
$path);
|
|
|
|
if ($err) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return $stdout;
|
|
|
|
}
|
2011-01-12 15:45:17 -08:00
|
|
|
|
2011-01-11 13:02:38 -08:00
|
|
|
public function getCurrentFileData($path) {
|
|
|
|
$full_path = $this->getPath($path);
|
|
|
|
if (Filesystem::pathExists($full_path)) {
|
|
|
|
return Filesystem::readFile($full_path);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
2011-01-09 15:22:25 -08:00
|
|
|
|
2011-04-05 20:12:37 -07:00
|
|
|
public function getRepositorySVNUUID() {
|
|
|
|
$info = $this->getSVNInfo('/');
|
|
|
|
return $info['Repository UUID'];
|
|
|
|
}
|
|
|
|
|
2011-08-23 18:48:55 -07:00
|
|
|
public function getLocalCommitInformation() {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2011-09-25 16:48:18 -07:00
|
|
|
public function supportsRelativeLocalCommits() {
|
2011-09-14 18:44:54 -07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
Add an "arc merge" workflow
Summary:
This should support conservative rewrite policies in git fairly well, under an
assumed workflow of:
- Develop in local branches, never rewrite history.
- Commit with "-m" or by typing a brief, non-template commit message
describing the checkpoint.
- Provide rich information in the web console (reviewers, etc.)
- Finalize with "git checkout master && arc merge branch && git push" or some
flavor thereof.
This supports Mercurial somewhat. The major problem is that "hg merge" fails if
the local is a fastforward of the remote, at which point there's nowhere we can
throw the commit message. Oh well. Just push it and we'll do our best to link
them up based on local commit info.
I am increasingly forming an opinion that Mercurial is "saftey-scissors git".
But also maybe I have no clue what I'm doing. I just don't understand why anyone
would think it's a good idea to have a trunk consisting of ~50% known-broken
revisions, random checkpoint parts, whitespace changes, typo fixes, etc. If you
use git with branching you can avoid this by making a trunk out of merges or
with rebase/amend, but there seems to be no way to have "one commit = one idea"
in any real sense in Mercurial.
Test Plan: Execute "arc merge" in git and mercurial.
Reviewers: fratrik, Makinde, aran, jungejason, tuomaspelkonen
Reviewed By: Makinde
CC: aran, epriestley, Makinde
Differential Revision: 860
2011-08-25 16:02:03 -07:00
|
|
|
public function supportsLocalBranchMerge() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-01-09 15:22:25 -08:00
|
|
|
}
|