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

319 lines
12 KiB
PHP
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

<?php
/**
* The model file of score module of ZenTaoPMS.
* @copyright Copyright 2009-2015 禅道软件(青岛)有限公司(ZenTao Software (Qingdao) Co., Ltd. www.cnezsoft.com)
* @author Memory <lvtao@cnezsoft.com>
* @package score
* @version $Id: model.php $
* @link http://www.zentao.net
*/
class scoreModel extends model
{
/**
* Get user score list.
*
* @param string $account
* @param object $pager
*
* @access public
* @return array
*/
public function getListByAccount($account, $pager)
{
return $this->dao->select('*')->from(TABLE_SCORE)->where('account')->eq($account)->orderBy('time_desc, id_desc')->page($pager)->fetchAll();
}
/**
* Add score logs.
*
* @param string $module
* @param string $method
* @param string $param
* @param string $account
* @param string $time
*
* @access public
* @return bool|object
*/
public function create($module = '', $method = '', $param = '', $account = '', $time = '')
{
if(empty($this->config->score->rule->{$module}->{$method}) || empty($this->config->global->scoreStatus)) return true;
$rule = $this->config->score->rule->{$module}->{$method};
$desc = $this->lang->score->modules[$module];
$user = empty($account) ? $this->app->user->account : $account;
$time = empty($time) ? helper::now() : $time;
$extended = isset($this->config->score->ruleExtended[$module][$method]) ? $this->config->score->ruleExtended[$module][$method] : array();
if(is_numeric($param)) $desc .= 'ID:' . $param;
$object = '';
switch($module)
{
case 'user':
if($method == 'login') $desc = $this->lang->score->methods[$module][$method];
if($method == 'changePassword')
{
if(!empty($extended['strength'][$param]))
{
$rule['score'] = $rule['score'] + $extended['strength'][$param];
}
$desc = $this->lang->score->methods[$module][$method];
}
break;
case 'story':
if($method == 'close')
{
$openedBy = $this->dao->findById($param)->from(TABLE_STORY)->fetch('openedBy');
$object = true;
if(!empty($openedBy))
{
$newRule = $rule;
$newRule['score'] = $extended['createID'];
$object = $this->saveScore($openedBy, $newRule, $module, $method, $desc, $time);
unset($newRule);
}
}
break;
case 'task':
if($method == 'finish')
{
$desc = $this->lang->score->methods[$module][$method] . 'ID:' . $param;
/* Check child task. */
$parentTask = $this->dao->select('id')->from(TABLE_TASK)->where('parent')->eq($param)->fetch('id');
if(!empty($parentTask)) return true;
$task = $this->loadModel('task')->getById($param);
if(!empty($extended['pri'][$task->pri]))
{
$rule['score'] = $rule['score'] + $extended['pri'][$task->pri];
}
if(!empty($task->estimate))
{
$rule['score'] = $rule['score'] + (empty($task->consumed) ? 0 : round($task->consumed / 10.0 * $task->estimate / $task->consumed));
}
}
break;
case 'bug':
if($method == 'createFormCase')
{
$desc = $this->lang->score->modules['testcase'] . 'ID:' . $param;
$openedBy = $this->dao->findById($param)->from(TABLE_CASE)->fetch('openedBy');
if(!empty($openedBy)) $user = $openedBy;
}
if($method == 'saveTplModal') $desc = $this->lang->score->methods[$module][$method] . 'ID:' . $param;
if($method == 'confirmBug')
{
$user = $param->openedBy;
$desc .= 'ID:' . $param->id;
if(!empty($extended['severity'][$param->severity]))
{
$rule['score'] = $rule['score'] + $extended['severity'][$param->severity];
}
}
if($method == 'resolve' && !empty($extended['severity'][$param->severity]))
{
$rule['score'] = $rule['score'] + $extended['severity'][$param->severity];
}
break;
case 'testTask':
if($method == 'runCase') $desc = $this->lang->score->methods[$module][$method] . 'ID:' . $param;
break;
case 'execution':
if($method == 'close')
{
$desc = $this->lang->score->methods[$module][$method] . ',' . $desc . 'ID:' . $param->id;
$timestamp = empty($time) ? time() : strtotime($time);
$object = true;
/* Project PM. */
if(!empty($param->PM))
{
$rule['score'] = $extended['manager']['close'];
if($param->end > date('Y-m-d', $timestamp))
{
$rule['score'] += $extended['manager']['onTime'];
}
$object = $this->saveScore($param->PM, $rule, $module, $method, $desc, $time);
}
/* Project team user. */
$teams = $this->dao->select('account')->from(TABLE_TEAM)->where('root')->eq($param->id)->andWhere('type')->eq('execution')->fetchPairs();
if(!empty($teams))
{
$rule['score'] = $extended['member']['close'];
if($param->end > date('Y-m-d', $timestamp))
{
$rule['score'] += $extended['member']['onTime'];
}
foreach($teams as $user)
{
if($user != $param->PM) $object = $this->saveScore($user, $rule, $module, $method, $desc, $time);
}
}
}
break;
case 'search':
if($method == 'saveQueryAdvanced') $desc = $this->lang->score->methods[$module][$method];
break;
case 'ajax':
$desc = $this->lang->score->methods[$module][$method];
break;
}
$object = $object === '' ? $this->saveScore($user, $rule, $module, $method, $desc, $time) : $object;
return $object;
}
/**
* Save user score.
*
* @param string $account
* @param array $rule
* @param string $module
* @param string $method
* @param string $desc
* @param string $time
*
* @access public
* @return object|bool
*/
public function saveScore($account = '', $rule = array(), $module = '', $method = '', $desc = '', $time = '')
{
if($rule['score'] == 0) return true;
if(!empty($rule['times']) || !empty($rule['hour']))
{
if(empty($rule['hour']))
{
$count = $this->dao->select('id')->from(TABLE_SCORE)->where('account')->eq($account)->andWhere('module')->eq($module)->andWhere('method')->eq($method)->count();
if($count >= $rule['times']) return true;
}
else
{
$timestamp = empty($time) ? time() : strtotime($time);
$count = $this->dao->select('id')->from(TABLE_SCORE)->where('account')->eq($account)
->andWhere('time')->between(date('Y-m-d 00:00:00', $timestamp), date('Y-m-d 23:59:59', $timestamp))
->andWhere('module')->eq($module)
->andWhere('method')->eq($method)
->count();
if($count >= $rule['times']) return true;
}
}
$user = $this->loadModel('user')->getById($account);
if(empty($user)) return false;
$data = new stdClass();
$data->account = $account;
$data->module = $module;
$data->method = $method;
$data->desc = $desc;
$data->before = $user->score;
$data->score = $rule['score'];
$data->after = $user->score + $rule['score'];
$data->time = empty($time) ? helper::now() : $time;
$this->dao->insert(TABLE_SCORE)->data($data)->exec();
$this->dao->update(TABLE_USER)->set("`score`=`score` + " . (int)$rule['score'])->set("`scoreLevel`=`scoreLevel` + " . (int)$rule['score'])->where('account')->eq($account)->exec();
return $data;
}
/**
* Score reset.
*
* @param int $lastID
*
* @access public
* @return array
*/
public function reset($lastID = 0)
{
if($lastID == 0)
{
$this->dao->query("UPDATE " . TABLE_USER . " SET `score`=0, `scoreLevel`=0");
$this->dao->delete()->from(TABLE_SCORE)->exec();
try
{
$this->dbh->exec('ALTER TABLE ' . TABLE_SCORE . ' auto_increment=1');
}
catch(Exception $e){}
}
$actions = $this->dao->select('*')->from(TABLE_ACTION)->where('id')->gt($lastID)->orderBy('id_asc')->limit(100)->fetchAll('id');
if(empty($actions)) return array('number' => 0, 'status' => 'finish');
foreach($actions as $action)
{
$param = $action->objectID;
if($action->objectType == 'execution' && $action->action == 'closed') $param = $this->dao->findById($action->objectID)->from(TABLE_PROJECT)->fetch();
if($action->objectType == 'bug')
{
$bug = $this->dao->findById($action->objectID)->from(TABLE_BUG)->fetch();
if(!empty($bug->case)) $action->action = 'createFormCase';
if($action->action == 'bugconfirmed' || $action->action == 'resolved') $param = $bug;
}
if($action->objectType == 'case') $action->objectType = 'testcase';
$this->create($action->objectType, $this->fixKey($action->action), $param, $action->actor, $action->date);
}
return array('status' => 'more', 'lastID' => max(array_keys($actions)), 'number' => count($actions));
}
/**
* Fix action type for score.
*
* @param $string
*
* @access public
* @return string
*/
public function fixKey($string)
{
$strings = array('created' => 'create', 'opened' => 'create', 'closed' => 'close', 'finished' => 'finish', 'bugconfirmed' => 'confirmBug', 'resolved' => 'resolve');
return isset($strings[$string]) ? $strings[$string] : $string;
}
/**
* Get yesterday's score for user.
*
* @access public
* @return string
*/
public function getNotice()
{
if(empty($this->config->global->scoreStatus)) return '';
if(date('Y-m-d', $this->app->user->lastTime) == helper::today()) return '';
$this->app->user->lastTime = time();
$score = $this->dao->select("SUM(score) AS score")->from(TABLE_SCORE)
->where('time')->between(date('Y-m-d 00:00:00', strtotime('-1 day')), date('Y-m-d 23:59:59', strtotime('-1 day')))
->andWhere('account')->eq($this->app->user->account)
->fetch('score');
if(!$score) return '';
$notice = sprintf($this->lang->score->tips, $score, $this->app->user->score);
$fullNotice = <<<EOT
<div id='noticeAttend' class='alert alert-success with-icon alert-dismissable' style='width:310px; position:fixed; bottom:25px; right:15px; z-index: 9999;' id='planInfo'>
<i class='icon icon-diamond'> </i>
<div class='content'>{$notice}</div>
<button type="button" class="close" data-dismiss="alert">×</button>
</div>
EOT;
return $fullNotice;
}
}