* @package release * @version $Id: control.php 4178 2013-01-20 09:32:11Z wwccss $ * @link http://www.zentao.net */ class release extends control { /** * Common actions. * * @param int $productID * @param int $branch * @access public * @return void */ public function commonAction($productID, $branch = 0) { $this->loadModel('product')->setMenu($productID, $branch); $product = $this->product->getById($productID); if(empty($product)) $this->locate($this->createLink('product', 'create')); $this->view->product = $product; $this->view->branch = $branch; $this->view->branches = $product->type == 'normal' ? array() : $this->loadModel('branch')->getPairs($product->id); $this->view->position[] = html::a($this->createLink('product', 'browse', "productID={$this->view->product->id}&branch=$branch"), $this->view->product->name); } /** * Browse releases. * * @param int $productID * @param int $branch * @param string $type * @param string $orderBy * @access public * @return void */ public function browse($productID, $branch = 'all', $type = 'all', $orderBy = 't1.date_desc', $recTotal = 0, $recPerPage = 20, $pageID = 1) { $this->app->loadClass('pager', $static = true); $pager = new pager($recTotal, $recPerPage, $pageID); $this->commonAction($productID, $branch); $uri = $this->app->getURI(true); $this->session->set('releaseList', $uri, 'product'); $this->session->set('buildList', $uri); $releases = $this->release->getList($productID, $branch, $type, $orderBy, $pager); $showBranch = false; foreach($releases as $release) { if($release->productType != 'normal') { $showBranch = true; break; } } $this->view->title = $this->view->product->name . $this->lang->colon . $this->lang->release->browse; $this->view->position[] = $this->lang->release->browse; $this->view->releases = $releases; $this->view->type = $type; $this->view->pager = $pager; $this->view->showBranch = $showBranch; $this->display(); } /** * Create a release. * * @param int $productID * @param string|int $branch * @access public * @return void */ public function create($productID, $branch = 'all') { if(!empty($_POST)) { $releaseID = $this->release->create($productID, $branch); if(dao::isError()) return $this->send(array('result' => 'fail', 'message' => dao::getError())); $this->loadModel('action')->create('release', $releaseID, 'opened'); $result = $this->executeHooks($releaseID); $message = $result ? $result : $this->lang->saveSuccess; if($this->viewType == 'json') return $this->send(array('result' => 'success', 'message' => $message, 'id' => $releaseID)); if(isonlybody()) return $this->send(array('result' => 'success', 'message' => $message, 'closeModal' => true, 'callback' => "parent.loadProductBuilds($productID)")); return $this->send(array('result' => 'success', 'message' => $message, 'locate' => inlink('view', "releaseID=$releaseID"))); } $builds = $this->loadModel('build')->getBuildPairs($productID, $branch, 'notrunk|withbranch|hasproject', 0, 'execution', '', false); $releasedBuilds = $this->release->getReleasedBuilds($productID, $branch); foreach($releasedBuilds as $build) unset($builds[$build]); $this->commonAction($productID, $branch); $this->view->title = $this->view->product->name . $this->lang->colon . $this->lang->release->create; $this->view->position[] = $this->lang->release->create; $this->view->productID = $productID; $this->view->builds = $builds; $this->view->users = $this->loadModel('user')->getPairs('noclosed'); $this->view->lastRelease = $this->release->getLast($productID, $branch); $this->display(); } /** * Edit a release. * * @param int $releaseID * @access public * @return void */ public function edit($releaseID) { if(!empty($_POST)) { $changes = $this->release->update($releaseID); if(dao::isError()) return $this->send(array('result' => 'fail', 'message' => dao::getError())); if($changes) { $actionID = $this->loadModel('action')->create('release', $releaseID, 'Edited'); if(!empty($changes)) $this->action->logHistory($actionID, $changes); } $result = $this->executeHooks($releaseID); $message = $result ? $result : $this->lang->saveSuccess; return $this->send(array('result' => 'success', 'message' => $message, 'locate' => inlink('view', "releaseID=$releaseID"))); } $this->loadModel('story'); $this->loadModel('bug'); $this->loadModel('build'); /* Get release and build. */ $release = $this->release->getById((int)$releaseID); $this->commonAction($release->product); $builds = $this->loadModel('build')->getBuildPairs($release->product, $release->branch, 'notrunk|withbranch|hasproject', 0, 'project', $release->build, false); $releasedBuilds = $this->release->getReleasedBuilds($release->product); foreach($releasedBuilds as $releasedBuild) { if(strpos(',' . trim($release->build, ',') . ',', ",{$releasedBuild},") === false) unset($builds[$releasedBuild]); } $this->view->title = $this->view->product->name . $this->lang->colon . $this->lang->release->edit; $this->view->position[] = $this->lang->release->edit; $this->view->release = $release; $this->view->builds = $builds; $this->view->users = $this->loadModel('user')->getPairs('noclosed'); $this->display(); } /** * View a release. * * @param int $releaseID * @param string $type * @param string $link * @param string $param * @param string $orderBy * @param int $recTotal * @param int $recPerPage * @param int $pageID * @access public * @return void */ public function view($releaseID, $type = 'story', $link = 'false', $param = '', $orderBy = 'id_desc', $recTotal = 0, $recPerPage = 100, $pageID = 1) { $releaseID = (int)$releaseID; $release = $this->release->getByID($releaseID, true); if(!$release) return print(js::error($this->lang->notFound) . js::locate($this->createLink('product', 'index'))); $uri = $this->app->getURI(true); if(!empty($release->build)) $this->session->set('buildList', $uri, 'project'); if($type == 'story') $this->session->set('storyList', $uri, 'product'); if($type == 'bug' or $type == 'leftBug') $this->session->set('bugList', $uri, 'qa'); $this->loadModel('story'); $this->loadModel('bug'); /* Load pager. */ $this->app->loadClass('pager', $static = true); if($this->app->getViewType() == 'mhtml') $recPerPage = 10; $sort = common::appendOrder($orderBy); if(strpos($sort, 'pri_') !== false) $sort = str_replace('pri_', 'priOrder_', $sort); $sort .= ',buildID_asc'; $storyPager = new pager($type == 'story' ? $recTotal : 0, $recPerPage, $type == 'story' ? $pageID : 1); $stories = $this->dao->select("t1.*,t2.id as buildID, t2.name as buildName, IF(t1.`pri` = 0, {$this->config->maxPriValue}, t1.`pri`) as priOrder")->from(TABLE_STORY)->alias('t1') ->leftJoin(TABLE_BUILD)->alias('t2')->on("FIND_IN_SET(t1.id, t2.stories)") ->where('t1.id')->in($release->stories) ->andWhere('t1.deleted')->eq(0) ->beginIF($type == 'story')->orderBy($sort)->fi() ->page($storyPager) ->fetchAll('id'); $this->loadModel('common')->saveQueryCondition($this->dao->get(), 'story', false); $stages = $this->dao->select('*')->from(TABLE_STORYSTAGE)->where('story')->in(array_keys($stories))->andWhere('branch')->eq($release->branch)->fetchPairs('story', 'stage'); foreach($stages as $storyID => $stage) $stories[$storyID]->stage = $stage; $bugPager = new pager($type == 'bug' ? $recTotal : 0, $recPerPage, $type == 'bug' ? $pageID : 1); $sort = common::appendOrder($orderBy); $bugs = $this->dao->select('*')->from(TABLE_BUG) ->where('id')->in($release->bugs) ->andWhere('deleted')->eq(0) ->beginIF($type == 'bug')->orderBy($sort)->fi() ->page($bugPager) ->fetchAll(); $this->loadModel('common')->saveQueryCondition($this->dao->get(), 'linkedBug'); $sort = common::appendOrder($orderBy); if($type == 'leftBug' and strpos($orderBy, 'severity_') !== false) $sort = str_replace('severity_', 'severityOrder_', $sort); $leftBugPager = new pager($type == 'leftBug' ? $recTotal : 0, $recPerPage, $type == 'leftBug' ? $pageID : 1); $leftBugs = $this->dao->select("*, IF(`severity` = 0, {$this->config->maxPriValue}, `severity`) as severityOrder")->from(TABLE_BUG) ->where('id')->in($release->leftBugs) ->andWhere('deleted')->eq(0) ->beginIF($type == 'leftBug')->orderBy($sort)->fi() ->page($leftBugPager) ->fetchAll(); $this->loadModel('common')->saveQueryCondition($this->dao->get(), 'leftBugs'); $this->commonAction($release->product); $product = $this->product->getById($release->product); $this->executeHooks($releaseID); $this->view->title = "RELEASE #$release->id $release->name/" . $product->name; $this->view->position[] = $this->lang->release->view; $this->view->release = $release; $this->view->stories = $stories; $this->view->bugs = $bugs; $this->view->leftBugs = $leftBugs; $this->view->actions = $this->loadModel('action')->getList('release', $releaseID); $this->view->users = $this->loadModel('user')->getPairs('noletter'); $this->view->type = $type; $this->view->link = $link; $this->view->param = $param; $this->view->orderBy = $orderBy; $this->view->branchName = $release->productType == 'normal' ? '' : $this->loadModel('branch')->getById($release->branch); $this->view->storyPager = $storyPager; $this->view->bugPager = $bugPager; $this->view->leftBugPager = $leftBugPager; $this->view->builds = $this->loadModel('build')->getBuildPairs($release->product, 'all', 'withbranch|hasproject', 0, 'execution', '', false); $this->view->summary = $this->product->summary($stories); $this->view->storyCases = $this->loadModel('testcase')->getStoryCaseCounts(array_keys($stories)); if($this->app->getViewType() == 'json') { unset($this->view->storyPager); unset($this->view->bugPager); unset($this->view->leftBugPager); } $this->display(); } /** * Notify for release. * * @param int $releaseID * @access public * @return void */ public function notify($releaseID) { if($_POST) { if(isset($_POST['notify'])) { $notify = implode(',', $this->post->notify); $this->dao->update(TABLE_RELEASE)->set('notify')->eq($notify)->where('id')->eq($releaseID)->exec(); $this->release->sendmail($releaseID); $this->loadModel('action')->create('release', $releaseID, 'notified'); } $this->send(array('result' => 'success', 'message' => $this->lang->saveSuccess, 'locate' => 'parent')); } $this->view->release = $this->release->getById($releaseID); $this->view->actions = $this->loadModel('action')->getList('release', $releaseID); $this->view->users = $this->loadModel('user')->getPairs('noletter|noclosed'); $this->display(); } /** * Delete a release. * * @param int $releaseID * @param string $confirm yes|no * @access public * @return void */ public function delete($releaseID, $confirm = 'no') { if($confirm == 'no') { return print(js::confirm($this->lang->release->confirmDelete, $this->createLink('release', 'delete', "releaseID=$releaseID&confirm=yes"))); } else { $this->release->delete(TABLE_RELEASE, $releaseID); $release = $this->dao->select('*')->from(TABLE_RELEASE)->where('id')->eq((int)$releaseID)->fetch(); $build = $this->dao->select('*')->from(TABLE_BUILD)->where('id')->eq((int)$release->build)->fetch(); $this->dao->update(TABLE_BUILD)->set('deleted')->eq(1)->where('id')->eq($release->shadow)->exec(); foreach($builds as $build) { if(empty($build->execution) and $build->createdDate == $release->createdDate) $this->build->delete(TABLE_BUILD, $build->id); } $message = $this->executeHooks($releaseID); if($message) $response['message'] = $message; /* if ajax request, send result. */ if($this->server->ajax) { if(dao::isError()) { $response['result'] = 'fail'; $response['message'] = dao::getError(); } else { $response['result'] = 'success'; $response['message'] = ''; $release = $this->release->getById($releaseID); } return $this->send($response); } if($this->viewType == 'json') return $this->send(array('result' => 'success', 'message' => $this->lang->saveSuccess)); $locateLink = $this->session->releaseList ? $this->session->releaseList : inlink('browse', "productID={$release->product}"); return print(js::locate($locateLink, 'parent')); } } /** * Export the stories of release to HTML. * * @access public * @return void */ public function export() { if(!empty($_POST)) { $type = $this->post->type; $html = ''; if($type == 'story' or $type == 'all') { $html .= "

{$this->lang->release->stories}

"; $this->loadModel('story'); $stories = $this->dbh->query($this->session->storyQueryCondition . " ORDER BY " . strtr($this->session->storyOrderBy, '_', ' '))->fetchAll(); foreach($stories as $story) $story->title = "id") . "' target='_blank'>$story->title"; $fields = array('id' => $this->lang->story->id, 'title' => $this->lang->story->title); $rows = $stories; $html .= ''; foreach($fields as $fieldLabel) $html .= "\n"; $html .= ''; foreach($rows as $row) { $html .= "\n"; foreach($fields as $fieldName => $fieldLabel) { $fieldValue = isset($row->$fieldName) ? $row->$fieldName : ''; $html .= "\n"; } $html .= "\n"; } $html .= '
$fieldLabel
$fieldValue
'; } if($type == 'bug' or $type == 'all') { $html .= "

{$this->lang->release->bugs}

"; $this->loadModel('bug'); $bugs = $this->dao->select('id, title')->from(TABLE_BUG)->where($this->session->linkedBugQueryCondition) ->beginIF($this->session->bugOrderBy != false)->orderBy($this->session->bugOrderBy)->fi() ->fetchAll('id'); foreach($bugs as $bug) $bug->title = "id") . "' target='_blank'>$bug->title"; $fields = array('id' => $this->lang->bug->id, 'title' => $this->lang->bug->title); $rows = $bugs; $html .= ''; foreach($fields as $fieldLabel) $html .= "\n"; $html .= ''; foreach($rows as $row) { $html .= "\n"; foreach($fields as $fieldName => $fieldLabel) { $fieldValue = isset($row->$fieldName) ? $row->$fieldName : ''; $html .= "\n"; } $html .= "\n"; } $html .= '
$fieldLabel
$fieldValue
'; } if($type == 'leftbug' or $type == 'all') { $html .= "

{$this->lang->release->generatedBugs}

"; $bugs = $this->dao->select('id, title')->from(TABLE_BUG)->where($this->session->leftBugsQueryCondition) ->beginIF($this->session->bugOrderBy != false)->orderBy($this->session->bugOrderBy)->fi() ->fetchAll('id'); foreach($bugs as $bug) $bug->title = "id") . "' target='_blank'>$bug->title"; $fields = array('id' => $this->lang->bug->id, 'title' => $this->lang->bug->title); $rows = $bugs; $html .= ''; foreach($fields as $fieldLabel) $html .= "\n"; $html .= ''; foreach($rows as $row) { $html .= "\n"; foreach($fields as $fieldName => $fieldLabel) { $fieldValue = isset($row->$fieldName) ? $row->$fieldName : ''; $html .= "\n"; } $html .= "\n"; } $html .= '
$fieldLabel
$fieldValue
'; } $html = "{$this->post->fileName}$html"; return print($this->fetch('file', 'sendDownHeader', array('fileName' => $this->post->fileName, 'html', $html))); } $this->display(); } /** * Link stories * * @param int $releaseID * @param string $browseType * @param int $param * @param int $recTotal * @param int $recPerPage * @param int $pageID * @access public * @return void */ public function linkStory($releaseID = 0, $browseType = '', $param = 0, $recTotal = 0, $recPerPage = 100, $pageID = 1) { if(!empty($_POST['stories'])) { $this->release->linkStory($releaseID); return print(js::locate(inlink('view', "releaseID=$releaseID&type=story"), 'parent')); } $this->session->set('storyList', inlink('view', "releaseID=$releaseID&type=story&link=true¶m=" . helper::safe64Encode("&browseType=$browseType&queryID=$param")), 'product'); $release = $this->release->getById($releaseID); $this->commonAction($release->product); $this->loadModel('story'); $this->loadModel('tree'); $this->loadModel('product'); /* Load pager. */ $this->app->loadClass('pager', $static = true); $pager = new pager($recTotal, $recPerPage, $pageID); /* Build search form. */ unset($this->config->product->search['fields']['product']); unset($this->config->product->search['fields']['project']); $queryID = ($browseType == 'bySearch') ? (int)$param : 0; $this->config->product->search['actionURL'] = $this->createLink('release', 'view', "releaseID=$releaseID&type=story&link=true¶m=" . helper::safe64Encode('&browseType=bySearch&queryID=myQueryID')); $this->config->product->search['queryID'] = $queryID; $this->config->product->search['style'] = 'simple'; $this->config->product->search['params']['plan']['values'] = $this->loadModel('productplan')->getPairsForStory($release->product, $release->branch, 'skipParent|withMainPlan'); $this->config->product->search['params']['status'] = array('operator' => '=', 'control' => 'select', 'values' => $this->lang->story->statusList); $searchModules = array(); $moduleGroups = $this->loadModel('tree')->getOptionMenu($release->product, 'story', 0, explode(',', $release->branch));; foreach($moduleGroups as $modules) $searchModules += $modules; $this->config->product->search['params']['module']['values'] = $searchModules; if($this->session->currentProductType == 'normal') { unset($this->config->product->search['fields']['branch']); unset($this->config->product->search['params']['branch']); } else { $allBranchs = $this->loadModel('branch')->getPairs($release->product); $branches = array('' => '', BRANCH_MAIN => $this->lang->branch->main); foreach(explode(',', trim($release->branch, ',')) as $branchID) $branches[$branchID] = zget($allBranchs, $branchID); $this->config->product->search['fields']['branch'] = sprintf($this->lang->product->branch, $this->lang->product->branchName[$release->productType]); $this->config->product->search['params']['branch']['values'] = $branches; } $this->loadModel('search')->setSearchParams($this->config->product->search); $builds = $this->loadModel('build')->getByList($release->build); $executionIdList = array(); foreach($builds as $build) { if(!empty($build->execution)) { $executionIdList[$build->execution] = $build->execution; } elseif(!empty($build->project)) { $executionIdList[$build->project] = $build->project; } } if($browseType == 'bySearch') { $allStories = $this->story->getBySearch($release->product, $release->branch, $queryID, 'id', $executionIdList, 'story', $release->stories, $pager); } else { $allStories = $this->story->getExecutionStories($executionIdList, $release->product, 0, 't1.`order`_desc', 'byBranch', $release->branch, 'story', $release->stories, $pager); } $this->view->allStories = $allStories; $this->view->release = $release; $this->view->releaseStories = empty($release->stories) ? array() : $this->story->getByList($release->stories); $this->view->users = $this->loadModel('user')->getPairs('noletter'); $this->view->browseType = $browseType; $this->view->param = $param; $this->view->pager = $pager; $this->display(); } /** * Unlink story * * @param int $releaseID * @param int $storyID * @access public * @return void */ public function unlinkStory($releaseID, $storyID, $confirm = 'no') { if($confirm == 'no') { return print(js::confirm($this->lang->release->confirmUnlinkStory, inlink('unlinkstory', "releaseID=$releaseID&storyID=$storyID&confirm=yes"))); } else { $this->release->unlinkStory($releaseID, $storyID); return print(js::reload('parent')); } } /** * Batch unlink story. * * @param int $releaseID * @access public * @return void */ public function batchUnlinkStory($releaseID) { $this->release->batchUnlinkStory($releaseID); echo js::locate($this->createLink('release', 'view', "releaseID=$releaseID&type=story"), 'parent'); } /** * Link bugs. * * @param int $releaseID * @param string $browseType * @param int $param * @param string $type * @param int $recTotal * @param int $recPerPage * @param int $pageID * @access public * @return void */ public function linkBug($releaseID = 0, $browseType = '', $param = 0, $type = 'bug', $recTotal = 0, $recPerPage = 100, $pageID = 1) { if(!empty($_POST['bugs'])) { $this->release->linkBug($releaseID, $type); return print(js::locate(inlink('view', "releaseID=$releaseID&type=$type"), 'parent')); } $this->session->set('bugList', inlink('view', "releaseID=$releaseID&type=$type&link=true¶m=" . helper::safe64Encode("&browseType=$browseType&queryID=$param")), 'qa'); /* Set menu. */ $release = $this->release->getByID($releaseID); $this->commonAction($release->product); /* Load pager. */ $this->app->loadClass('pager', $static = true); $pager = new pager($recTotal, $recPerPage, $pageID); /* Build the search form. */ $this->loadModel('bug'); unset($this->config->bug->search['fields']['product']); unset($this->config->bug->search['fields']['project']); $queryID = ($browseType == 'bysearch') ? (int)$param : 0; $this->config->bug->search['actionURL'] = $this->createLink('release', 'view', "releaseID=$releaseID&type=$type&link=true¶m=" . helper::safe64Encode('&browseType=bySearch&queryID=myQueryID')); $this->config->bug->search['queryID'] = $queryID; $this->config->bug->search['style'] = 'simple'; $this->config->bug->search['params']['plan']['values'] = $this->loadModel('productplan')->getPairsForStory($release->product, $release->branch, 'skipParent|withMainPlan'); $this->config->bug->search['params']['execution']['values'] = $this->loadModel('product')->getExecutionPairsByProduct($release->product, $release->branch); $this->config->bug->search['params']['openedBuild']['values'] = $this->loadModel('build')->getBuildPairs($release->product, $branch = 'all', 'releasetag'); $this->config->bug->search['params']['resolvedBuild']['values'] = $this->config->bug->search['params']['openedBuild']['values']; $searchModules = array(); $moduleGroups = $this->loadModel('tree')->getOptionMenu($release->product, 'bug', 0, explode(',', $release->branch)); foreach($moduleGroups as $modules) $searchModules += $modules; $this->config->bug->search['params']['module']['values'] = $searchModules; if($this->session->currentProductType == 'normal') { unset($this->config->bug->search['fields']['branch']); unset($this->config->bug->search['params']['branch']); } else { $allBranchs = $this->loadModel('branch')->getPairs($release->product); $branches = array('' => '', BRANCH_MAIN => $this->lang->branch->main); foreach(explode(',', trim($release->branch, ',')) as $branchID) $branches[$branchID] = zget($allBranchs, $branchID); $this->config->bug->search['fields']['branch'] = sprintf($this->lang->product->branch, $this->lang->product->branchName[$release->productType]); $this->config->bug->search['params']['branch']['values'] = $branches; } $this->loadModel('search')->setSearchParams($this->config->bug->search); $builds = $this->loadModel('build')->getByList($release->build); $allBugs = array(); $releaseBugs = $type == 'bug' ? $release->bugs : $release->leftBugs; if($browseType == 'bySearch') { $allBugs = $this->bug->getBySearch($release->product, $release->branch, $queryID, 'id_desc', $releaseBugs, $pager); } else { if($type == 'bug') { $allBugs = $this->bug->getReleaseBugs(array_keys($builds), $release->product, $release->branch, $releaseBugs, $pager); } elseif($type == 'leftBug') { $allBugs = $this->bug->getProductLeftBugs(array_keys($builds), $release->product, $release->branch, $releaseBugs, $pager); } } $this->view->allBugs = $allBugs; $this->view->releaseBugs = empty($releaseBugs) ? array() : $this->bug->getByList($releaseBugs); $this->view->release = $release; $this->view->users = $this->loadModel('user')->getPairs('noletter'); $this->view->browseType = $browseType; $this->view->param = $param; $this->view->type = $type; $this->view->pager = $pager; $this->display(); } /** * Unlink story * * @param int $releaseID * @param int $bugID * @param string $type * @access public * @return void */ public function unlinkBug($releaseID, $bugID, $type = 'bug') { $this->release->unlinkBug($releaseID, $bugID, $type); /* if ajax request, send result. */ if($this->server->ajax) { if(dao::isError()) { $response['result'] = 'fail'; $response['message'] = dao::getError(); } else { $response['result'] = 'success'; $response['message'] = ''; } return $this->send($response); } echo js::reload('parent'); } /** * Batch unlink story. * * @param int $releaseID * @param string $type * @access public * @return void */ public function batchUnlinkBug($releaseID, $type = 'bug') { $this->release->batchUnlinkBug($releaseID, $type); echo js::locate($this->createLink('release', 'view', "releaseID=$releaseID&type=$type"), 'parent'); } /** * Change status. * * @param int $releaseID * @param string $status * @access public * @return void */ public function changeStatus($releaseID, $status) { $this->release->changeStatus($releaseID, $status); if(dao::isError()) return print(js::error(dao::getError())); $actionID = $this->loadModel('action')->create('release', $releaseID, 'changestatus', '', $status); echo js::reload('parent'); } }