* @package release * @version $Id: model.php 4129 2013-01-18 01:58:14Z wwccss $ * @link http://www.zentao.net */ ?> dao->select('t1.*, t2.id as buildID, t2.filePath, t2.scmPath, t2.name as buildName, t2.project, t3.name as productName, t3.type as productType') ->from(TABLE_RELEASE)->alias('t1') ->leftJoin(TABLE_BUILD)->alias('t2')->on('t1.build = t2.id') ->leftJoin(TABLE_PRODUCT)->alias('t3')->on('t1.product = t3.id') ->where('t1.id')->eq((int)$releaseID) ->orderBy('t1.id DESC') ->fetch(); if(!$release) return false; $this->loadModel('file'); $release = $this->file->replaceImgURL($release, 'desc'); $release->files = $this->file->getByObject('release', $releaseID); if(empty($release->files))$release->files = $this->file->getByObject('build', $release->buildID); if($setImgSize) $release->desc = $this->file->setImgSize($release->desc); return $release; } /** * Get list of releases. * * @param int $productID * @param int $branch * @param string $type * @param string $orderBy * @access public * @return array */ public function getList($productID, $branch = 'all', $type = 'all', $orderBy = 't1.date_desc') { return $this->dao->select('t1.*, t2.name as productName, t3.id as buildID, t3.name as buildName, t3.project, t4.name as projectName') ->from(TABLE_RELEASE)->alias('t1') ->leftJoin(TABLE_PRODUCT)->alias('t2')->on('t1.product = t2.id') ->leftJoin(TABLE_BUILD)->alias('t3')->on('t1.build = t3.id') ->leftJoin(TABLE_PROJECT)->alias('t4')->on('t1.project = t4.id') ->where('t1.product')->eq((int)$productID) ->beginIF($branch !== 'all')->andWhere('t1.branch')->eq($branch)->fi() ->beginIF($type != 'all')->andWhere('t1.status')->eq($type)->fi() ->andWhere('t1.deleted')->eq(0) ->orderBy($orderBy) ->fetchAll(); } /** * Get last release. * * @param int $productID * @param int $branch * @access public * @return bool | object */ public function getLast($productID, $branch = 0) { return $this->dao->select('id, name')->from(TABLE_RELEASE) ->where('product')->eq((int)$productID) ->beginIF($branch)->andWhere('branch')->eq($branch)->fi() ->orderBy('date DESC') ->limit(1) ->fetch(); } /** * Get released builds from product. * * @param int $productID * @param string|int $branch * @access public * @return void */ public function getReleasedBuilds($productID, $branch = 'all') { $releases = $this->dao->select('build')->from(TABLE_RELEASE) ->where('deleted')->eq(0) ->andWhere('product')->eq($productID) ->beginIF($branch !== 'all')->andWhere('branch')->eq($branch)->fi() ->fetchAll('build'); return array_keys($releases); } /** * Create a release. * * @param int $productID * @param int $branch * @access public * @return int */ public function create($productID, $branch = 0) { /* Init vars. */ $productID = (int)$productID; $branch = (int)$branch; $buildID = 0; /* Check build if build is required. */ if(strpos($this->config->release->create->requiredFields, 'build') !== false and $this->post->build == false) return dao::$errors[] = sprintf($this->lang->error->notempty, $this->lang->release->build); /* Check date must be not more than today. */ if($this->post->date > date('Y-m-d')) return dao::$errors[] = $this->lang->release->errorDate; $release = fixer::input('post') ->add('product', (int)$productID) ->add('branch', (int)$branch) ->setDefault('stories', '') ->join('stories', ',') ->join('bugs', ',') ->join('mailto', ',') ->setIF($this->post->build == false, 'build', $buildID) ->stripTags($this->config->release->editor->create['id'], $this->config->allowedTags) ->remove('allchecker,files,labels,uid') ->get(); /* Auto create build when release is not link build. */ if(empty($release->build) and $release->name) { $build = $this->dao->select('*')->from(TABLE_BUILD) ->where('deleted')->eq('0') ->andWhere('name')->eq($release->name) ->andWhere('product')->eq($productID) ->andWhere('branch')->eq($branch) ->fetch(); if($build) { return dao::$errors['build'] = sprintf($this->lang->release->existBuild, $release->name); } else { $build = new stdclass(); $build->product = (int)$productID; $build->branch = (int)$branch; $build->name = $release->name; $build->date = $release->date; $build->builder = $this->app->user->account; $build->desc = $release->desc; $build->execution = 0; $build = $this->loadModel('file')->processImgURL($build, $this->config->release->editor->create['id']); $this->app->loadLang('build'); $this->dao->insert(TABLE_BUILD)->data($build) ->autoCheck() ->check('name', 'unique', "product = {$productID} AND branch = {$branch} AND deleted = '0'") ->batchCheck($this->config->release->create->requiredFields, 'notempty') ->exec(); if(dao::isError()) return false; $buildID = $this->dao->lastInsertID(); $release->build = $buildID; } } if($release->build) { $buildInfo = $this->dao->select('project, branch')->from(TABLE_BUILD)->where('id')->eq($release->build)->fetch(); $release->branch = $buildInfo->branch; $release->project = $buildInfo->project; } $release = $this->loadModel('file')->processImgURL($release, $this->config->release->editor->create['id'], $this->post->uid); $this->dao->insert(TABLE_RELEASE)->data($release) ->autoCheck() ->batchCheck($this->config->release->create->requiredFields, 'notempty') ->check('name', 'unique', "product = '{$release->product}' AND branch = '{$release->branch}' AND deleted = '0'"); if(dao::isError()) { if(!empty($buildID)) $this->dao->delete()->from(TABLE_BUILD)->where('id')->eq($buildID)->exec(); return false; } $this->dao->exec(); if(dao::isError()) { if(!empty($buildID)) $this->dao->delete()->from(TABLE_BUILD)->where('id')->eq($buildID)->exec(); } else { $releaseID = $this->dao->lastInsertID(); $this->file->updateObjectID($this->post->uid, $releaseID, 'release'); $this->file->saveUpload('release', $releaseID); $this->loadModel('score')->create('release', 'create', $releaseID); return $releaseID; } return false; } /** * Update a release. * * @param int $releaseID * @access public * @return void */ public function update($releaseID) { /* Init vars. */ $releaseID = (int)$releaseID; $oldRelease = $this->dao->select('*')->from(TABLE_RELEASE)->where('id')->eq($releaseID)->fetch(); $branch = $this->dao->select('branch')->from(TABLE_BUILD)->where('id')->eq((int)$this->post->build)->fetch('branch'); $release = fixer::input('post')->stripTags($this->config->release->editor->edit['id'], $this->config->allowedTags) ->add('branch', (int)$branch) ->join('mailto', ',') ->setIF(!$this->post->marker, 'marker', 0) ->cleanInt('product') ->remove('files,labels,allchecker,uid') ->get(); $release = $this->loadModel('file')->processImgURL($release, $this->config->release->editor->edit['id'], $this->post->uid); /* update release project and branch */ if($release->build) { $buildInfo = $this->dao->select('project, branch')->from(TABLE_BUILD)->where('id')->eq($release->build)->fetch(); $release->branch = $buildInfo->branch; $release->project = $buildInfo->project; } $this->dao->update(TABLE_RELEASE)->data($release) ->autoCheck() ->batchCheck($this->config->release->edit->requiredFields, 'notempty') ->check('name', 'unique', "id != '$releaseID' AND product = '{$release->product}' AND branch = '$branch' AND deleted = '0'") ->where('id')->eq((int)$releaseID) ->exec(); if(!dao::isError()) { $this->file->updateObjectID($this->post->uid, $releaseID, 'release'); return common::createChanges($oldRelease, $release); } } /** * Get notify persons. * * @param string $notfiyList * @param int $productID * @param int $buildID * @param int $releaseID * @access public * @return array */ public function getNotifyPersons($notifyList = '', $productID = 0, $buildID = 0, $releaseID = 0) { if(empty($notifyList)) return array(); /* Init vars. */ $notifyPersons = array(); $managerFields = ''; $notifyList = explode(',', $notifyList); foreach($notifyList as $notify) { if($notify == 'PO' or $notify == 'QD' or $notify == 'feedback') { $managerFields .= $notify . ','; } elseif($notify == 'SC' and !empty($buildID)) { $stories = $this->dao->select('stories')->from(TABLE_BUILD)->where('id')->eq($buildID)->fetch('stories'); $stories .= $this->dao->select('stories')->from(TABLE_RELEASE)->where('id')->eq($releaseID)->fetch('stories'); $stories = trim($stories, ','); if(empty($stories)) continue; $openedByList = $this->dao->select('openedBy')->from(TABLE_STORY)->where('id')->in($stories)->fetchPairs(); $notifyPersons += $openedByList; } elseif(($notify == 'ET' or $notify == 'PT') and !empty($buildID)) { $type = $notify == 'ET' ? 'execution' : 'project'; $members = $this->dao->select('t2.account')->from(TABLE_BUILD)->alias('t1') ->leftJoin(TABLE_TEAM)->alias('t2')->on('t1.' . $type . '=t2.root') ->where('t2.type')->eq($type) ->fetchPairs(); if(empty($members)) continue; $notifyPersons += $members; } } if(!empty($managerFields)) { $managerFields = trim($managerFields, ','); $managerUsers = $this->dao->select($managerFields)->from(TABLE_PRODUCT)->where('id')->eq($productID)->fetch(); foreach($managerUsers as $account) { if(!isset($notifyPersons[$account])) $notifyPersons[$account] = $account; } } return $notifyPersons; } /** * Link stories * * @param int $releaseID * @access public * @return void */ public function linkStory($releaseID) { $release = $this->getByID($releaseID); $product = $this->loadModel('product')->getByID($release->product); foreach($this->post->stories as $i => $storyID) { if(strpos(",{$release->stories},", ",{$storyID},") !== false) unset($_POST['stories'][$i]); } $release->stories .= ',' . join(',', $this->post->stories); $this->dao->update(TABLE_RELEASE)->set('stories')->eq($release->stories)->where('id')->eq((int)$releaseID)->exec(); if($release->stories) { $this->loadModel('story'); $this->loadModel('action'); foreach($this->post->stories as $storyID) { /* Reset story stagedBy field for auto compute stage. */ $this->dao->update(TABLE_STORY)->set('stagedBy')->eq('')->where('id')->eq($storyID)->exec(); if($product->type != 'normal') $this->dao->update(TABLE_STORYSTAGE)->set('stagedBy')->eq('')->where('story')->eq($storyID)->andWhere('branch')->eq($release->branch)->exec(); $this->story->setStage($storyID); $this->action->create('story', $storyID, 'linked2release', '', $releaseID); } } } /** * Unlink story * * @param int $releaseID * @param int $storyID * @access public * @return void */ public function unlinkStory($releaseID, $storyID) { $release = $this->getByID($releaseID); $release->stories = trim(str_replace(",$storyID,", ',', ",$release->stories,"), ','); $this->dao->update(TABLE_RELEASE)->set('stories')->eq($release->stories)->where('id')->eq((int)$releaseID)->exec(); $this->loadModel('action')->create('story', $storyID, 'unlinkedfromrelease', '', $releaseID); } /** * Batch unlink story. * * @param int $releaseID * @access public * @return void */ public function batchUnlinkStory($releaseID) { $storyList = $this->post->storyIdList; if(empty($storyList)) return true; $release = $this->getByID($releaseID); $release->stories = ",$release->stories,"; foreach($storyList as $storyID) $release->stories = str_replace(",$storyID,", ',', $release->stories); $release->stories = trim($release->stories, ','); $this->dao->update(TABLE_RELEASE)->set('stories')->eq($release->stories)->where('id')->eq((int)$releaseID)->exec(); $this->loadModel('action'); foreach($this->post->storyIdList as $unlinkStoryID) $this->action->create('story', $unlinkStoryID, 'unlinkedfromrelease', '', $releaseID); } /** * Link bugs. * * @param int $releaseID * @param string $type * @access public * @return void */ public function linkBug($releaseID, $type = 'bug') { $release = $this->getByID($releaseID); $field = $type == 'bug' ? 'bugs' : 'leftBugs'; foreach($this->post->bugs as $i => $bugID) { if(strpos(",{$release->$field},", ",{$bugID},") !== false) unset($_POST['bugs'][$i]); } $release->$field .= ',' . join(',', $this->post->bugs); $this->dao->update(TABLE_RELEASE)->set($field)->eq($release->$field)->where('id')->eq((int)$releaseID)->exec(); $this->loadModel('action'); foreach($this->post->bugs as $bugID) $this->action->create('bug', $bugID, 'linked2release', '', $releaseID); } /** * Unlink bug. * * @param int $releaseID * @param int $bugID * @param string $type * @access public * @return void */ public function unlinkBug($releaseID, $bugID, $type = 'bug') { $release = $this->getByID($releaseID); $field = $type == 'bug' ? 'bugs' : 'leftBugs'; $release->{$field} = trim(str_replace(",$bugID,", ',', ",{$release->$field},"), ','); $this->dao->update(TABLE_RELEASE)->set($field)->eq($release->$field)->where('id')->eq((int)$releaseID)->exec(); $this->loadModel('action')->create('bug', $bugID, 'unlinkedfromrelease', '', $releaseID); } /** * Batch unlink bug. * * @param int $releaseID * @param string $type * @access public * @return void */ public function batchUnlinkBug($releaseID, $type = 'bug') { $bugList = $this->post->unlinkBugs; if(empty($bugList)) return true; $release = $this->getByID($releaseID); $field = $type == 'bug' ? 'bugs' : 'leftBugs'; $release->$field = ",{$release->$field},"; foreach($bugList as $bugID) $release->$field = str_replace(",$bugID,", ',', $release->$field); $release->$field = trim($release->$field, ','); $this->dao->update(TABLE_RELEASE)->set($field)->eq($release->$field)->where('id')->eq((int)$releaseID)->exec(); $this->loadModel('action'); foreach($this->post->unlinkBugs as $unlinkBugID) $this->action->create('bug', $unlinkBugID, 'unlinkedfromrelease', '', $releaseID); } /** * Change status. * * @param int $releaseID * @param string $status * @access public * @return bool */ public function changeStatus($releaseID, $status) { $this->dao->update(TABLE_RELEASE)->set('status')->eq($status)->where('id')->eq($releaseID)->exec(); return !dao::isError(); } /** * Judge btn is clickable or not. * * @param int $release * @param string $action * @static * @access public * @return bool */ public static function isClickable($release, $action) { $action = strtolower($action); if($action == 'notify') return $release->bugs or $release->stories; return true; } /** * Send mail to release related users. * * @param int $releaseID * @access public * @return void */ public function sendmail($releaseID) { if(empty($releaseID)) return; $this->app->loadConfig('mail'); /* Load module and get vars. */ $users = $this->loadModel('user')->getPairs('noletter'); $release = $this->getByID($releaseID); $suffix = empty($release->product) ? '' : ' - ' . $this->loadModel('product')->getById($release->product)->name; $subject = 'Release #' . $release->id . ' ' . $release->name . $suffix; /* Get mail content. */ $modulePath = $this->app->getModulePath($appName = '', 'release'); $oldcwd = getcwd(); $viewFile = $modulePath . 'view/sendmail.html.php'; chdir($modulePath . 'view'); if(file_exists($modulePath . 'ext/view/sendmail.html.php')) { $viewFile = $modulePath . 'ext/view/sendmail.html.php'; chdir($modulePath . 'ext/view'); } ob_start(); include $viewFile; foreach(glob($modulePath . 'ext/view/sendmail.*.html.hook.php') as $hookFile) include $hookFile; $mailContent = ob_get_contents(); ob_end_clean(); chdir($oldcwd); if(strpos(",{$release->notify},", ',FB,') !== false) $this->sendMail2Feedback($release, $subject); /* Get the sender. */ $sendUsers = $this->getToAndCcList($release); if(!$sendUsers) return; list($toList, $ccList) = $sendUsers; /* Send it. */ $this->loadModel('mail')->send($toList, $subject, $mailContent, $ccList); } /** * Get toList and ccList. * * @param object $release * @access public * @return bool|array */ public function getToAndCcList($release) { /* Set toList and ccList. */ $toList = $this->app->user->account; $ccList = $release->mailto . ','; /* Get notifiy persons. */ $notifyPersons = array(); if(!empty($release->notify)) $notifyPersons = $this->getNotifyPersons($release->notify, $release->product, $release->build, $release->id); foreach($notifyPersons as $account) { if(strpos($ccList, ",{$account},") === false) $ccList .= ",$account,"; } $ccList = trim($ccList, ','); if(empty($toList)) { if(empty($ccList)) return false; if(strpos($ccList, ',') === false) { $toList = $ccList; $ccList = ''; } else { $commaPos = strpos($ccList, ','); $toList = substr($ccList, 0, $commaPos); $ccList = substr($ccList, $commaPos + 1); } } return array($toList, $ccList); } /** * Send mail to feedback user. * * @param object $release * @param string $subject * @access public * @return void */ public function sendMail2Feedback($release, $subject) { if(!$release->stories && !$release->bugs) return; $stories = explode(',', trim($release->stories, ',')); $bugs = explode(',', trim($release->bugs, ',')); $storyNotifyList = $this->dao->select('id,title,notifyEmail')->from(TABLE_STORY) ->where('id')->in($stories) ->andWhere('notifyEmail')->ne('') ->fetchGroup('notifyEmail', 'id'); $bugNotifyList = $this->dao->select('id,title,notifyEmail')->from(TABLE_BUG) ->where('id')->in($bugs) ->andWhere('notifyEmail')->ne('') ->fetchGroup('notifyEmail', 'id'); $toList = array(); $emails = array(); $storyNames = array(); $bugNames = array(); foreach($storyNotifyList as $notifyEmail => $storyList) { $email = new stdClass(); $email->account = $notifyEmail; $email->email = $notifyEmail; $email->realname = ''; $emails[$notifyEmail] = $email; $toList[$notifyEmail] = $notifyEmail; foreach($storyList as $story) $storyNames[] = $story->title; } foreach($bugNotifyList as $notifyEmail => $bugList) { $email = new stdClass(); $email->account = $notifyEmail; $email->email = $notifyEmail; $email->realname = ''; $emails[$notifyEmail] = $email; $toList[$notifyEmail] = $notifyEmail; foreach($bugList as $bug) $bugNames[] = $bug->title; } if(!empty($toList)) { $storyNames = implode(',', $storyNames); $bugNames = implode(',', $bugNames); $mailContent = sprintf($this->lang->release->mailContent, $release->name); if($storyNames) $mailContent .= sprintf($this->lang->release->storyList, $storyNames); if($bugNames) $mailContent .= sprintf($this->lang->release->bugList, $bugNames); $this->loadModel('mail')->send(implode(',', $toList), $subject, $mailContent, '', false, $emails); } } }