zentaopms/module/git/model.php
2023-05-16 10:47:08 +08:00

416 lines
12 KiB
PHP

<?php
/**
* The model file of git module of ZenTaoPMS.
*
* @copyright Copyright 2009-2015 禅道软件(青岛)有限公司(ZenTao Software (Qingdao) Co., Ltd. www.cnezsoft.com)
* @license ZPL(http://zpl.pub/page/zplv12.html) or AGPL(https://www.gnu.org/licenses/agpl-3.0.en.html)
* @author Chunsheng Wang <chunsheng@cnezsoft.com>
* @package git
* @version $Id$
* @link http://www.zentao.net
*/
?>
<?php
class gitModel extends model
{
/**
* The git binary client.
*
* @var int
* @access public
*/
public $client;
/**
* Repos.
*
* @var array
* @access public
*/
public $repos = array();
/**
* The root path of a repo
*
* @var string
* @access public
*/
public $repoRoot = '';
/**
* Users
*
* @var array
* @access public
*/
public $users = array();
/**
* Construct function.
*
* @access public
* @return void
*/
public function __construct()
{
parent::__construct();
$this->loadModel('action');
$this->loadModel('repo');
}
/**
* Run.
*
* @access public
* @return void
*/
public function run()
{
/* Get repos and load module. */
$this->setRepos();
$this->loadModel('job');
if(empty($this->repos)) return false;
/* Get commit triggerType jobs by repoIdList. */
$commentGroup = $this->job->getTriggerGroup('commit', array_keys($this->repos));
/* Get tag triggerType jobs by repoIdList. */
$tagGroup = $this->job->getTriggerGroup('tag', array_keys($this->repos));
foreach($this->repos as $repoID => $repo)
{
$this->updateCommit($repo, $commentGroup, true);
/* Create compile by tag. */
$jobs = zget($tagGroup, $repoID, array());
foreach($jobs as $job)
{
$tags = $this->getRepoTags($repo);
$isNew = empty($job->lastTag) ? true : false;
$lastTag = '';
foreach($tags as $tag)
{
if(empty($tag)) continue;
if(!$isNew and $tag == $job->lastTag)
{
$isNew = true;
continue;
}
if(!$isNew) continue;
$lastTag = $tag;
if($lastTag) $this->loadModel('compile')->createByJob($job->id, $lastTag, 'tag');
}
if($lastTag) $this->dao->update(TABLE_JOB)->set('lastTag')->eq($lastTag)->where('id')->eq($job->id)->exec();
}
}
}
/**
* Update commit.
*
* @param object $repo
* @param array $commentGroup
* @param bool $printLog
* @access public
* @return void
*/
public function updateCommit($repo, $commentGroup, $printLog = true)
{
/* Load module and print log. */
$this->loadModel('repo');
if($printLog) $this->printLog("begin repo $repo->id");
if(!$this->setRepo($repo)) return false;
/* Get branches and commits. */
$branches = $this->repo->getBranches($repo);
$commits = $repo->commits;
$accountPairs = array();
if($repo->SCM == 'Gitlab')
{
$userList = $this->loadModel('gitlab')->apiGetUsers($repo->gitService);
$acountIDPairs = $this->gitlab->getUserIdAccountPairs($repo->gitService);
foreach($userList as $gitlabUser) $accountPairs[$gitlabUser->realname] = zget($acountIDPairs, $gitlabUser->id, '');
}
elseif($repo->SCM == 'Gitea')
{
$userList = $this->loadModel('gitea')->apiGetUsers($repo->gitService);
$acountIDPairs = $this->gitea->getUserAccountIdPairs($repo->gitService, 'openID,account');
foreach($userList as $gitlabUser) $accountPairs[$gitlabUser->realname] = zget($acountIDPairs, $gitlabUser->id, '');
}
elseif($repo->SCM == 'Gogs')
{
$userList = $this->loadModel('gogs')->apiGetUsers($repo->gitService);
$acountIDPairs = $this->gogs->getUserAccountIdPairs($repo->gitService, 'openID,account');
foreach($userList as $gitlabUser) $accountPairs[$gitlabUser->realname] = zget($acountIDPairs, $gitlabUser->id, '');
}
/* Update code commit history. */
foreach($branches as $branch)
{
if($printLog) $this->printLog("sync branch $branch logs.");
$_COOKIE['repoBranch'] = $branch;
if($printLog) $this->printLog("get this repo logs.");
$lastInDB = $this->repo->getLatestCommit($repo->id);
/* Ignore unsynced branch. */
if($repo->synced != 1)
{
if($printLog) $this->printLog("Please init repo {$repo->name}");
continue;
}
$version = isset($lastInDB->commit) ? (int)$lastInDB->commit + 1 : 1;
$logs = $this->repo->getUnsyncedCommits($repo);
$objects = array();
if(!empty($logs))
{
if($printLog) $this->printLog("get " . count($logs) . " logs");
if($printLog) $this->printLog('begin parsing logs');
foreach($logs as $log)
{
if($printLog) $this->printLog("parsing log {$log->revision}");
if($printLog) $this->printLog("comment is\n----------\n" . trim($log->msg) . "\n----------");
$objects = $this->repo->parseComment($log->msg);
$lastVersion = $version;
$version = $this->repo->saveOneCommit($repo->id, $log, $version, $branch);
if($objects)
{
if($printLog) $this->printLog('extract' .
' story:' . join(' ', $objects['stories']) .
' task:' . join(' ', $objects['tasks']) .
' bug:' . join(',', $objects['bugs']));
if($lastVersion != $version)
{
$this->repo->saveAction2PMS($objects, $log, $this->repoRoot, $repo->encoding, 'git', $accountPairs);
/* Objects link commit. */
foreach($objects as $objectType => $objectIDs)
{
$objectTypeMap = array('stories' => 'story', 'bugs' => 'bug', 'tasks' => 'task');
if(empty($objectIDs) or !isset($objectTypeMap[$objectType])) continue;
$this->post->$objectType = $objectIDs;
$this->repo->link($repo->id, $log->revision, $objectTypeMap[$objectType]);
}
}
}
else
{
if($printLog) $this->printLog('no objects found' . "\n");
}
/* Create compile by comment. */
$jobs = zget($commentGroup, $repo->id, array());
foreach($jobs as $job)
{
foreach(explode(',', $job->comment) as $comment)
{
if(strpos($log->msg, $comment) !== false) $this->loadModel('compile')->createByJob($job->id);
}
}
$commits += count($logs);
}
}
}
$this->repo->updateCommitCount($repo->id, $commits);
$this->dao->update(TABLE_REPO)->set('lastSync')->eq(helper::now())->where('id')->eq($repo->id)->exec();
if($printLog) $this->printLog("\n\nrepo #" . $repo->id . ': ' . $repo->path . " finished");
}
/**
* Set the repos.
*
* @access public
* @return mixed
*/
public function setRepos()
{
$repos = $this->loadModel('repo')->getListBySCM('Git,Gitlab,Gogs,Gitea');
$gitRepos = array();
$paths = array();
foreach($repos as $repo)
{
if(!isset($paths[$repo->path]))
{
unset($repo->acl);
unset($repo->desc);
$gitRepos[$repo->id] = $repo;
$paths[$repo->path] = $repo->path;
}
}
if(empty($gitRepos)) echo "You must set one git repo.\n";
$this->repos = $gitRepos;
return true;
}
/**
* Get repos.
*
* @access public
* @return array
*/
public function getRepos()
{
$this->setRepos();
$repoPairs = array();
foreach($this->repos as $repo) $repoPairs[] = $repo->path;
return $repoPairs;
}
/**
* Set repo.
*
* @param object $repo
* @access public
* @return bool
*/
public function setRepo($repo)
{
$this->setClient($repo);
if(empty($this->client)) return false;
$this->setRepoRoot($repo);
return true;
}
/**
* Set the git binary client of a repo.
*
* @param object $repo
* @access public
* @return bool
*/
public function setClient($repo)
{
$this->client = $repo->client;
return true;
}
/**
* set the root path of a repo.
*
* @param object $repo
* @access public
* @return void
*/
public function setRepoRoot($repo)
{
$this->repoRoot = $repo->path;
}
/**
* get tags histories for repo.
*
* @param object $repo
* @access public
* @return mixed
*/
public function getRepoTags($repo)
{
if(empty($repo->client) or empty($repo->path) or !isset($repo->account) or !isset($repo->password) or !isset($repo->encoding)) return false;
$scm = $this->app->loadClass('scm');
$scm->setEngine($repo);
return $scm->tags('');
}
/**
* Get repo logs.
*
* @param object $repo
* @param int $fromRevision
* @access public
* @return array
*/
public function getRepoLogs($repo, $fromRevision)
{
if(empty($repo->client) or empty($repo->path) or !isset($repo->account) or !isset($repo->password) or !isset($repo->encoding)) return false;
$scm = $this->app->loadClass('scm');
$scm->setEngine($repo);
$logs = $scm->log('', $fromRevision);
if(empty($logs)) return false;
foreach($logs as $log)
{
$log->author = $log->committer;
$log->msg = $log->comment;
$log->date = $log->time;
/* Process files. */
$log->files = array();
foreach($log->change as $file => $info) $log->files[$info['action']][] = $file;
}
return $logs;
}
/**
* Convert log from xml format to object.
*
* @param object $log
* @access public
* @return object
*/
public function convertLog($log)
{
list($hash, $account, $date) = $log;
$account = preg_replace('/^Author:/', '', $account);
$account = trim(preg_replace('/<[a-zA-Z0-9_\-\.]+@[a-zA-Z0-9_\-\.]+>/', '', $account));
$date = trim(preg_replace('/^Date:/', '', $date));
$count = count($log);
$comment = '';
$files = array();
for($i = 3; $i < $count; $i++)
{
$line = $log[$i];
if(preg_match('/^\s{2,}/', $line))
{
$comment .= $line;
}
elseif(strpos($line, "\t") !== false)
{
list($action, $entry) = explode("\t", $line);
$entry = '/' . trim($entry);
$files[$action][] = $entry;
}
}
$parsedLog = new stdClass();
$parsedLog->author = $account;
$parsedLog->revision = trim(preg_replace('/^commit/', '', $hash));
$parsedLog->msg = trim($comment);
$parsedLog->date = date('Y-m-d H:i:s', strtotime($date));
$parsedLog->files = $files;
return $parsedLog;
}
/**
* Pring log.
*
* @param sting $log
* @access public
* @return void
*/
public function printLog($log)
{
echo helper::now() . " $log\n";
}
}