Introduce a more subtle concept of validity when it comes to revision
indentifiers in IDF - the SCM function isValidRevision has been replaced by a validateRevision() method which returns one of three states, valid, invalid or ambiguous. The source view can then act accordingly and display disambiguate view for the latter, so the user can select for which revision he actually wants to execute the requested action. Also, invalid revisions now lead to another separate view, telling the user that it is invalid / does not exist and pointing him optionally to the help page where he can read further how to access his repository to push the first changes into. (partially resolves issue 525)
This commit is contained in:
parent
5d263e78e0
commit
21cdf60c31
@ -63,7 +63,7 @@ class IDF_Scm
|
|||||||
public $project = null;
|
public $project = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cache storage.
|
* Cache storage.
|
||||||
*
|
*
|
||||||
* It must only be used to store data for the lifetime of the
|
* It must only be used to store data for the lifetime of the
|
||||||
* object. For example if you need to get the list of branches in
|
* object. For example if you need to get the list of branches in
|
||||||
@ -166,13 +166,28 @@ class IDF_Scm
|
|||||||
throw new Pluf_Exception_NotImplemented();
|
throw new Pluf_Exception_NotImplemented();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const REVISION_VALID = 0;
|
||||||
|
const REVISION_INVALID = 1;
|
||||||
|
const REVISION_AMBIGUOUS = 2;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if a revision or commit is valid.
|
* Check if a revision or commit is valid, invalid or ambiguous.
|
||||||
*
|
*
|
||||||
* @param string Revision or commit
|
* @param string Revision or commit
|
||||||
* @return bool
|
* @return int One of REVISION_VALID, REVISION_INVALID or REVISION_AMBIGIOUS
|
||||||
*/
|
*/
|
||||||
public function isValidRevision($rev)
|
public function validateRevision($rev)
|
||||||
|
{
|
||||||
|
throw new Pluf_Exception_NotImplemented();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns an array of single commit objects for ambiguous commit identifiers
|
||||||
|
*
|
||||||
|
* @param string Ambiguous commit identifier
|
||||||
|
* @return array of objects
|
||||||
|
*/
|
||||||
|
public function disambiguateRevision($commit)
|
||||||
{
|
{
|
||||||
throw new Pluf_Exception_NotImplemented();
|
throw new Pluf_Exception_NotImplemented();
|
||||||
}
|
}
|
||||||
@ -217,7 +232,7 @@ class IDF_Scm
|
|||||||
* 'foo-branch' => 'branches/foo-branch',)
|
* 'foo-branch' => 'branches/foo-branch',)
|
||||||
* </pre>
|
* </pre>
|
||||||
*
|
*
|
||||||
* @return array Branches
|
* @return array Branches
|
||||||
*/
|
*/
|
||||||
public function getBranches()
|
public function getBranches()
|
||||||
{
|
{
|
||||||
@ -282,7 +297,7 @@ class IDF_Scm
|
|||||||
* @param string Revision or commit
|
* @param string Revision or commit
|
||||||
* @param string Folder ('/')
|
* @param string Folder ('/')
|
||||||
* @param string Branch (null)
|
* @param string Branch (null)
|
||||||
* @return array
|
* @return array
|
||||||
*/
|
*/
|
||||||
public function getTree($rev, $folder='/', $branch=null)
|
public function getTree($rev, $folder='/', $branch=null)
|
||||||
{
|
{
|
||||||
@ -396,7 +411,7 @@ class IDF_Scm
|
|||||||
public static function syncTimeline($project, $force=false)
|
public static function syncTimeline($project, $force=false)
|
||||||
{
|
{
|
||||||
$cache = Pluf_Cache::factory();
|
$cache = Pluf_Cache::factory();
|
||||||
$key = 'IDF_Scm:'.$project->shortname.':lastsync';
|
$key = 'IDF_Scm:'.$project->shortname.':lastsync';
|
||||||
if ($force or null === ($res=$cache->get($key))) {
|
if ($force or null === ($res=$cache->get($key))) {
|
||||||
$scm = IDF_Scm::get($project);
|
$scm = IDF_Scm::get($project);
|
||||||
if ($scm->isAvailable()) {
|
if ($scm->isAvailable()) {
|
||||||
|
@ -296,10 +296,12 @@ class IDF_Scm_Git extends IDF_Scm
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public function isValidRevision($commit)
|
public function validateRevision($commit)
|
||||||
{
|
{
|
||||||
$type = $this->testHash($commit);
|
$type = $this->testHash($commit);
|
||||||
return ('commit' == $type || 'tag' == $type);
|
if ('commit' == $type || 'tag' == $type)
|
||||||
|
return IDF_Scm::REVISION_VALID;
|
||||||
|
return IDF_Scm::REVISION_INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -87,14 +87,19 @@ class IDF_Scm_Mercurial extends IDF_Scm
|
|||||||
return sprintf(Pluf::f('mercurial_remote_url'), $project->shortname);
|
return sprintf(Pluf::f('mercurial_remote_url'), $project->shortname);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function isValidRevision($rev)
|
public function validateRevision($rev)
|
||||||
{
|
{
|
||||||
$cmd = sprintf(Pluf::f('hg_path', 'hg').' log -R %s -r %s',
|
$cmd = sprintf(Pluf::f('hg_path', 'hg').' log -R %s -r %s',
|
||||||
escapeshellarg($this->repo),
|
escapeshellarg($this->repo),
|
||||||
escapeshellarg($rev));
|
escapeshellarg($rev));
|
||||||
$cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
|
$cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
|
||||||
self::exec('IDF_Scm_Mercurial::isValidRevision', $cmd, $out, $ret);
|
self::exec('IDF_Scm_Mercurial::isValidRevision', $cmd, $out, $ret);
|
||||||
return ($ret == 0) && (count($out) > 0);
|
|
||||||
|
// FIXME: apparently a given hg revision can also be ambigious -
|
||||||
|
// handle this case here sometime
|
||||||
|
if ($ret == 0 && count($out) > 0)
|
||||||
|
return IDF_Scm::REVISION_VALID;
|
||||||
|
return IDF_Scm::REVISION_INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -425,12 +425,53 @@ class IDF_Scm_Monotone extends IDF_Scm
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @see IDF_Scm::isValidRevision()
|
* @see IDF_Scm::validateRevision()
|
||||||
*/
|
*/
|
||||||
public function isValidRevision($commit)
|
public function validateRevision($commit)
|
||||||
{
|
{
|
||||||
$revs = $this->_resolveSelector($commit);
|
$revs = $this->_resolveSelector($commit);
|
||||||
return count($revs) == 1;
|
if (count($revs) == 0)
|
||||||
|
return IDF_Scm::REVISION_INVALID;
|
||||||
|
|
||||||
|
if (count($revs) > 1)
|
||||||
|
return IDF_Scm::REVISION_AMBIGUOUS;
|
||||||
|
|
||||||
|
return IDF_Scm::REVISION_VALID;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see IDF_Scm::disambiguateRevision
|
||||||
|
*/
|
||||||
|
public function disambiguateRevision($commit)
|
||||||
|
{
|
||||||
|
$revs = $this->_resolveSelector($commit);
|
||||||
|
|
||||||
|
$out = array();
|
||||||
|
foreach ($revs as $rev)
|
||||||
|
{
|
||||||
|
$certs = $this->_getCerts($rev);
|
||||||
|
|
||||||
|
$log = array();
|
||||||
|
$log['author'] = implode(', ', $certs['author']);
|
||||||
|
|
||||||
|
$log['branch'] = implode(', ', $certs['branch']);
|
||||||
|
|
||||||
|
$dates = array();
|
||||||
|
foreach ($certs['date'] as $date)
|
||||||
|
$dates[] = date('Y-m-d H:i:s', strtotime($date));
|
||||||
|
$log['date'] = implode(', ', $dates);
|
||||||
|
|
||||||
|
$combinedChangelog = implode("\n---\n", $certs['changelog']);
|
||||||
|
$split = preg_split("/[\n\r]/", $combinedChangelog, 2);
|
||||||
|
$log['title'] = $split[0];
|
||||||
|
$log['full_message'] = (isset($split[1])) ? trim($split[1]) : '';
|
||||||
|
|
||||||
|
$log['commit'] = $rev;
|
||||||
|
|
||||||
|
$out[] = (object)$log;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $out;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -630,7 +671,7 @@ class IDF_Scm_Monotone extends IDF_Scm
|
|||||||
--$n;
|
--$n;
|
||||||
|
|
||||||
$log = array();
|
$log = array();
|
||||||
$log['author'] = implode(", ", $certs['author']);
|
$log['author'] = implode(', ', $certs['author']);
|
||||||
|
|
||||||
$dates = array();
|
$dates = array();
|
||||||
foreach ($certs['date'] as $date)
|
foreach ($certs['date'] as $date)
|
||||||
|
@ -138,7 +138,7 @@ class IDF_Scm_Svn extends IDF_Scm
|
|||||||
/**
|
/**
|
||||||
* Subversion revisions are either a number or 'HEAD'.
|
* Subversion revisions are either a number or 'HEAD'.
|
||||||
*/
|
*/
|
||||||
public function isValidRevision($rev)
|
public function validateRevision($rev)
|
||||||
{
|
{
|
||||||
if ($rev == 'HEAD') {
|
if ($rev == 'HEAD') {
|
||||||
return true;
|
return true;
|
||||||
@ -149,8 +149,11 @@ class IDF_Scm_Svn extends IDF_Scm
|
|||||||
escapeshellarg($this->repo),
|
escapeshellarg($this->repo),
|
||||||
escapeshellarg($rev));
|
escapeshellarg($rev));
|
||||||
$cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
|
$cmd = Pluf::f('idf_exec_cmd_prefix', '').$cmd;
|
||||||
self::exec('IDF_Scm_Svn::isValidRevision', $cmd, $out, $ret);
|
self::exec('IDF_Scm_Svn::validateRevision', $cmd, $out, $ret);
|
||||||
return (0 == $ret);
|
|
||||||
|
if ($ret == 0)
|
||||||
|
return IDF_Scm::REVISION_VALID;
|
||||||
|
return IDF_Scm::REVISION_INVALID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -59,30 +59,56 @@ class IDF_Views_Source
|
|||||||
$params, $request);
|
$params, $request);
|
||||||
}
|
}
|
||||||
|
|
||||||
public $changeLog_precond = array('IDF_Precondition::accessSource');
|
/**
|
||||||
|
* Is displayed in case an invalid revision is requested
|
||||||
|
*/
|
||||||
|
public $invalidRevision_precond = array('IDF_Precondition::accessSource');
|
||||||
|
public function invalidRevision($request, $match)
|
||||||
|
{
|
||||||
|
$title = sprintf(__('%s Invalid Revision'), (string) $request->project);
|
||||||
|
$commit = $match[2];
|
||||||
|
$params = array(
|
||||||
|
'page_title' => $title,
|
||||||
|
'title' => $title,
|
||||||
|
'commit' => $commit,
|
||||||
|
);
|
||||||
|
return Pluf_Shortcuts_RenderToResponse('idf/source/invalid_revision.html',
|
||||||
|
$params, $request);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Is displayed in case a revision identifier cannot be uniquely resolved
|
||||||
|
* to one single revision
|
||||||
|
*/
|
||||||
|
public $disambiguateRevision_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable');
|
||||||
|
public function disambiguateRevision($request, $match)
|
||||||
|
{
|
||||||
|
$title = sprintf(__('%s Ambiguous Revision'), (string) $request->project);
|
||||||
|
$commit = $match[2];
|
||||||
|
$redirect = $match[3];
|
||||||
|
$scm = IDF_Scm::get($request->project);
|
||||||
|
$revisions = $scm->disambiguateRevision($commit);
|
||||||
|
$params = array(
|
||||||
|
'page_title' => $title,
|
||||||
|
'title' => $title,
|
||||||
|
'commit' => $commit,
|
||||||
|
'revisions' => $revisions,
|
||||||
|
'redirect' => $redirect,
|
||||||
|
);
|
||||||
|
return Pluf_Shortcuts_RenderToResponse('idf/source/disambiguate_revision.html',
|
||||||
|
$params, $request);
|
||||||
|
}
|
||||||
|
|
||||||
|
public $changeLog_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function changeLog($request, $match)
|
public function changeLog($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
if (!$scm->isAvailable()) {
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::help',
|
|
||||||
array($request->project->shortname));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$branches = $scm->getBranches();
|
$branches = $scm->getBranches();
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
if (count($branches) == 0) {
|
|
||||||
// Redirect to the project source help
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::help',
|
|
||||||
array($request->project->shortname));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
// Redirect to the first branch
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::changeLog',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$title = sprintf(__('%1$s %2$s Change Log'), (string) $request->project,
|
$title = sprintf(__('%1$s %2$s Change Log'), (string) $request->project,
|
||||||
$this->getScmType($request));
|
$this->getScmType($request));
|
||||||
$changes = $scm->getChangeLog($commit, 25);
|
$changes = $scm->getChangeLog($commit, 25);
|
||||||
@ -111,22 +137,17 @@ class IDF_Views_Source
|
|||||||
$request);
|
$request);
|
||||||
}
|
}
|
||||||
|
|
||||||
public $treeBase_precond = array('IDF_Precondition::accessSource');
|
public $treeBase_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function treeBase($request, $match)
|
public function treeBase($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
if (!$scm->isAvailable()) {
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::help',
|
|
||||||
array($request->project->shortname));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
|
|
||||||
$cobject = $scm->getCommit($commit);
|
$cobject = $scm->getCommit($commit);
|
||||||
if (!$cobject) {
|
if (!$cobject) {
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
throw new Exception('could not retrieve commit object for '. $commit);
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
}
|
||||||
$title = sprintf(__('%1$s %2$s Source Tree'),
|
$title = sprintf(__('%1$s %2$s Source Tree'),
|
||||||
$request->project, $this->getScmType($request));
|
$request->project, $this->getScmType($request));
|
||||||
@ -159,20 +180,14 @@ class IDF_Views_Source
|
|||||||
$request);
|
$request);
|
||||||
}
|
}
|
||||||
|
|
||||||
public $tree_precond = array('IDF_Precondition::accessSource');
|
public $tree_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function tree($request, $match)
|
public function tree($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
|
|
||||||
if (!$scm->isAvailable()) {
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::help',
|
|
||||||
array($request->project->shortname));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$fburl = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
$request_file = $match[3];
|
$request_file = $match[3];
|
||||||
if (substr($request_file, -1) == '/') {
|
if (substr($request_file, -1) == '/') {
|
||||||
$request_file = substr($request_file, 0, -1);
|
$request_file = substr($request_file, 0, -1);
|
||||||
@ -181,13 +196,13 @@ class IDF_Views_Source
|
|||||||
$request_file));
|
$request_file));
|
||||||
return new Pluf_HTTP_Response_Redirect($url, 301);
|
return new Pluf_HTTP_Response_Redirect($url, 301);
|
||||||
}
|
}
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
// Redirect to the first branch
|
|
||||||
return new Pluf_HTTP_Response_Redirect($fburl);
|
|
||||||
}
|
|
||||||
$request_file_info = $scm->getPathInfo($request_file, $commit);
|
$request_file_info = $scm->getPathInfo($request_file, $commit);
|
||||||
if (!$request_file_info) {
|
if (!$request_file_info) {
|
||||||
// Redirect to the first branch
|
// Redirect to the main branch
|
||||||
|
$fburl = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
||||||
|
array($request->project->shortname,
|
||||||
|
$scm->getMainBranch()));
|
||||||
return new Pluf_HTTP_Response_Redirect($fburl);
|
return new Pluf_HTTP_Response_Redirect($fburl);
|
||||||
}
|
}
|
||||||
$branches = $scm->getBranches();
|
$branches = $scm->getBranches();
|
||||||
@ -277,26 +292,17 @@ class IDF_Views_Source
|
|||||||
return '<span class="breadcrumb">'.implode('<span class="sep">'.$sep.'</span>', $out).'</span>';
|
return '<span class="breadcrumb">'.implode('<span class="sep">'.$sep.'</span>', $out).'</span>';
|
||||||
}
|
}
|
||||||
|
|
||||||
public $commit_precond = array('IDF_Precondition::accessSource');
|
public $commit_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function commit($request, $match)
|
public function commit($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
// Redirect to the first branch
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$large = $scm->isCommitLarge($commit);
|
$large = $scm->isCommitLarge($commit);
|
||||||
$cobject = $scm->getCommit($commit, !$large);
|
$cobject = $scm->getCommit($commit, !$large);
|
||||||
if (!$cobject) {
|
if (!$cobject) {
|
||||||
// Redirect to the first branch
|
throw new Exception('could not retrieve commit object for '. $commit);
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
}
|
||||||
$title = sprintf(__('%s Commit Details'), (string) $request->project);
|
$title = sprintf(__('%s Commit Details'), (string) $request->project);
|
||||||
$page_title = sprintf(__('%s Commit Details - %s'), (string) $request->project, $commit);
|
$page_title = sprintf(__('%s Commit Details - %s'), (string) $request->project, $commit);
|
||||||
@ -326,19 +332,17 @@ class IDF_Views_Source
|
|||||||
$request);
|
$request);
|
||||||
}
|
}
|
||||||
|
|
||||||
public $downloadDiff_precond = array('IDF_Precondition::accessSource');
|
public $downloadDiff_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function downloadDiff($request, $match)
|
public function downloadDiff($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
// Redirect to the first branch
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$cobject = $scm->getCommit($commit, true);
|
$cobject = $scm->getCommit($commit, true);
|
||||||
|
if (!$cobject) {
|
||||||
|
throw new Exception('could not retrieve commit object for '. $commit);
|
||||||
|
}
|
||||||
$rep = new Pluf_HTTP_Response($cobject->changes, 'text/plain');
|
$rep = new Pluf_HTTP_Response($cobject->changes, 'text/plain');
|
||||||
$rep->headers['Content-Disposition'] = 'attachment; filename="'.$commit.'.diff"';
|
$rep->headers['Content-Disposition'] = 'attachment; filename="'.$commit.'.diff"';
|
||||||
return $rep;
|
return $rep;
|
||||||
@ -394,19 +398,14 @@ class IDF_Views_Source
|
|||||||
* Get a given file at a given commit.
|
* Get a given file at a given commit.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public $getFile_precond = array('IDF_Precondition::accessSource');
|
public $getFile_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function getFile($request, $match)
|
public function getFile($request, $match)
|
||||||
{
|
{
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
$commit = $match[2];
|
$commit = $match[2];
|
||||||
$request_file = $match[3];
|
$request_file = $match[3];
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
// Redirect to the first branch
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$request_file_info = $scm->getPathInfo($request_file, $commit);
|
$request_file_info = $scm->getPathInfo($request_file, $commit);
|
||||||
if (!$request_file_info or $request_file_info->type == 'tree') {
|
if (!$request_file_info or $request_file_info->type == 'tree') {
|
||||||
// Redirect to the first branch
|
// Redirect to the first branch
|
||||||
@ -427,18 +426,13 @@ class IDF_Views_Source
|
|||||||
* Get a zip archive of the current commit.
|
* Get a zip archive of the current commit.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public $download_precond = array('IDF_Precondition::accessSource');
|
public $download_precond = array('IDF_Precondition::accessSource',
|
||||||
|
'IDF_Views_Source_Precondition::scmAvailable',
|
||||||
|
'IDF_Views_Source_Precondition::revisionValid');
|
||||||
public function download($request, $match)
|
public function download($request, $match)
|
||||||
{
|
{
|
||||||
$commit = trim($match[2]);
|
$commit = trim($match[2]);
|
||||||
$scm = IDF_Scm::get($request->project);
|
$scm = IDF_Scm::get($request->project);
|
||||||
if (!$scm->isValidRevision($commit)) {
|
|
||||||
// Redirect to the first branch
|
|
||||||
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::treeBase',
|
|
||||||
array($request->project->shortname,
|
|
||||||
$scm->getMainBranch()));
|
|
||||||
return new Pluf_HTTP_Response_Redirect($url);
|
|
||||||
}
|
|
||||||
$base = $request->project->shortname.'-'.$commit;
|
$base = $request->project->shortname.'-'.$commit;
|
||||||
$cmd = $scm->getArchiveCommand($commit, $base.'/');
|
$cmd = $scm->getArchiveCommand($commit, $base.'/');
|
||||||
$rep = new Pluf_HTTP_Response_CommandPassThru($cmd, 'application/x-zip');
|
$rep = new Pluf_HTTP_Response_CommandPassThru($cmd, 'application/x-zip');
|
||||||
@ -447,7 +441,6 @@ class IDF_Views_Source
|
|||||||
return $rep;
|
return $rep;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find the mime type of a requested file.
|
* Find the mime type of a requested file.
|
||||||
*
|
*
|
||||||
@ -495,7 +488,6 @@ class IDF_Views_Source
|
|||||||
return $res;
|
return $res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Find the mime type of a file.
|
* Find the mime type of a file.
|
||||||
*
|
*
|
||||||
@ -610,3 +602,55 @@ function IDF_Views_Source_ShortenString($string, $length)
|
|||||||
substr($string, -($length - $preflen - mb_strlen($ellipse)));
|
substr($string, -($length - $preflen - mb_strlen($ellipse)));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class IDF_Views_Source_Precondition
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Ensures that the configured SCM for the project is available
|
||||||
|
*
|
||||||
|
* @param $request
|
||||||
|
* @return true | Pluf_HTTP_Response_Redirect
|
||||||
|
*/
|
||||||
|
static public function scmAvailable($request)
|
||||||
|
{
|
||||||
|
$scm = IDF_Scm::get($request->project);
|
||||||
|
if (!$scm->isAvailable()) {
|
||||||
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::help',
|
||||||
|
array($request->project->shortname));
|
||||||
|
return new Pluf_HTTP_Response_Redirect($url);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the revision given in the URL path and acts accordingly
|
||||||
|
*
|
||||||
|
* @param $request
|
||||||
|
* @return true | Pluf_HTTP_Response_Redirect
|
||||||
|
* @throws Exception
|
||||||
|
*/
|
||||||
|
static public function revisionValid($request)
|
||||||
|
{
|
||||||
|
list($url_info, $url_matches) = $request->view;
|
||||||
|
list(, $project, $commit) = $url_matches;
|
||||||
|
|
||||||
|
$scm = IDF_Scm::get($request->project);
|
||||||
|
$res = $scm->validateRevision($commit);
|
||||||
|
switch ($res) {
|
||||||
|
case IDF_Scm::REVISION_VALID:
|
||||||
|
return true;
|
||||||
|
case IDF_Scm::REVISION_INVALID:
|
||||||
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::invalidRevision',
|
||||||
|
array($request->project->shortname, $commit));
|
||||||
|
return new Pluf_HTTP_Response_Redirect($url);
|
||||||
|
case IDF_Scm::REVISION_AMBIGUOUS:
|
||||||
|
$url = Pluf_HTTP_URL_urlForView('IDF_Views_Source::disambiguateRevision',
|
||||||
|
array($request->project->shortname,
|
||||||
|
$commit,
|
||||||
|
$url_info['model'].'::'.$url_info['method']));
|
||||||
|
return new Pluf_HTTP_Response_Redirect($url);
|
||||||
|
default:
|
||||||
|
throw new Exception('unknown validation result: '. $res);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@ -148,6 +148,16 @@ $ctl[] = array('regex' => '#^/p/([\-\w]+)/source/help/$#',
|
|||||||
'model' => 'IDF_Views_Source',
|
'model' => 'IDF_Views_Source',
|
||||||
'method' => 'help');
|
'method' => 'help');
|
||||||
|
|
||||||
|
$ctl[] = array('regex' => '#^/p/([\-\w]+)/source/invalid/([^/]+)/$#',
|
||||||
|
'base' => $base,
|
||||||
|
'model' => 'IDF_Views_Source',
|
||||||
|
'method' => 'invalidRevision');
|
||||||
|
|
||||||
|
$ctl[] = array('regex' => '#^/p/([\-\w]+)/source/disambiguate/([^/]+)/from/([^/]+)/$#',
|
||||||
|
'base' => $base,
|
||||||
|
'model' => 'IDF_Views_Source',
|
||||||
|
'method' => 'disambiguateRevision');
|
||||||
|
|
||||||
$ctl[] = array('regex' => '#^/p/([\-\w]+)/source/tree/([^/]+)/$#',
|
$ctl[] = array('regex' => '#^/p/([\-\w]+)/source/tree/([^/]+)/$#',
|
||||||
'base' => $base,
|
'base' => $base,
|
||||||
'model' => 'IDF_Views_Source',
|
'model' => 'IDF_Views_Source',
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{extends "idf/base.html"}
|
{extends "idf/base.html"}
|
||||||
{block tabsource} class="active"{/block}
|
{block tabsource} class="active"{/block}
|
||||||
{block subtabs}
|
{block subtabs}
|
||||||
{if !$inHelp and (in_array($commit, $tree_in) or (in_array($commit, $tags_in)))}{assign $currentCommit = $commit}{else}{assign $currentCommit = $project.getScmRoot()}{/if}
|
{if !$inHelp and !$inError and (in_array($commit, $tree_in) or (in_array($commit, $tags_in)))}{assign $currentCommit = $commit}{else}{assign $currentCommit = $project.getScmRoot()}{/if}
|
||||||
<div id="sub-tabs">
|
<div id="sub-tabs">
|
||||||
<a {if $inSourceTree}class="active" {/if}href="{url 'IDF_Views_Source::treeBase', array($project.shortname, $currentCommit)}">{trans 'Source Tree'}</a> |
|
<a {if $inSourceTree}class="active" {/if}href="{url 'IDF_Views_Source::treeBase', array($project.shortname, $currentCommit)}">{trans 'Source Tree'}</a> |
|
||||||
<a {if $inChangeLog}class="active" {/if}href="{url 'IDF_Views_Source::changeLog', array($project.shortname, $currentCommit)}">{trans 'Change Log'}</a>
|
<a {if $inChangeLog}class="active" {/if}href="{url 'IDF_Views_Source::changeLog', array($project.shortname, $currentCommit)}">{trans 'Change Log'}</a>
|
||||||
|
33
src/IDF/templates/idf/source/disambiguate_revision.html
Normal file
33
src/IDF/templates/idf/source/disambiguate_revision.html
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
{extends "idf/source/base.html"}
|
||||||
|
{block docclass}yui-t2{assign $inError=true}{/block}
|
||||||
|
{block body}
|
||||||
|
|
||||||
|
<p>{blocktrans}The revision identifier <b>{$commit}</b> is ambiguous and can be
|
||||||
|
expanded to multiple valid revisions - please choose one:{/blocktrans}</p>
|
||||||
|
|
||||||
|
<table summary="" class="tree-list">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>{trans 'Title'}</th>
|
||||||
|
<th>{trans 'Author'}</th>
|
||||||
|
<th>{trans 'Date'}</th>
|
||||||
|
<th>{trans 'Branch'}</th>
|
||||||
|
<th>{trans 'Revision'}</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{foreach $revisions as $revision}
|
||||||
|
{aurl 'url', $redirect, array($project.shortname, $revision.commit)}
|
||||||
|
<tr class="log">
|
||||||
|
<td>{$revision.title}</td>
|
||||||
|
<td>{$revision.author}</td>
|
||||||
|
<td>{$revision.date}</td>
|
||||||
|
<td>{$revision.branch}</td>
|
||||||
|
<td><a href="{$url}">{$revision.commit}</a></td>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/foreach}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{/block}
|
||||||
|
|
17
src/IDF/templates/idf/source/invalid_revision.html
Normal file
17
src/IDF/templates/idf/source/invalid_revision.html
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
{extends "idf/source/base.html"}
|
||||||
|
{block docclass}yui-t2{assign $inError=true}{/block}
|
||||||
|
{block body}
|
||||||
|
|
||||||
|
<p>{blocktrans}The revision <b>{$commit}</b> is not valid or does not exist
|
||||||
|
in this repository.{/blocktrans}</p>
|
||||||
|
|
||||||
|
{if $isOwner or $isMember}
|
||||||
|
{aurl 'url', 'IDF_Views_Source::help', array($project.shortname)}
|
||||||
|
<p>{blocktrans}If this is a new repository, the reason for this error
|
||||||
|
could be that you have not committed and / or pushed any change so far.
|
||||||
|
In this case please take a look at the <a href="{$url}">Help</a> page
|
||||||
|
how to access your repository.{/blocktrans}</p>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{/block}
|
||||||
|
|
Loading…
Reference in New Issue
Block a user