mirror of
https://github.com/YMFE/yapi.git
synced 2025-01-06 12:45:22 +08:00
517 lines
17 KiB
HTML
517 lines
17 KiB
HTML
<!DOCTYPE html>
|
||
<html lang="zh-CN">
|
||
<head>
|
||
<meta charset="utf-8">
|
||
<meta name="viewport" content="initial-scale=1,maximum-scale=1,minimum-scale=1,user-scalable=no" />
|
||
<title>yapi : ./server/controllers/project.js</title>
|
||
<link type="text/css" rel="stylesheet" href="../../../source/code.css"/>
|
||
<script type="text/javascript" src="../../../source/shCore.js"></script>
|
||
<script type="text/javascript" src="../../../source/shBrush-js.js"></script>
|
||
<style>
|
||
.syntaxhighlighter .number1 .spaces,.syntaxhighlighter .toolbar{ display: none;}
|
||
.syntaxhighlighter table td.gutter .line.highlight { background-color: #6ce26c !important; color: white; }
|
||
</style>
|
||
</head>
|
||
<body>
|
||
<div class="ydoc">
|
||
<div class="ydoc-banner-bg">
|
||
<div class="ydoc-banner" id="content" tabindex="-1">
|
||
<div class="ydoc-banner-area">
|
||
<h1>yapi : ./server/controllers/project.js</h1>
|
||
<p>源代码</p>
|
||
</div>
|
||
</div>
|
||
<div class="ydoc-container">
|
||
<div class="ydoc-container-content">
|
||
<div class="static-code-content" role="main">
|
||
<pre class="brush: js;">
|
||
import projectModel from '../models/project.js'
|
||
import yapi from '../yapi.js'
|
||
import baseController from './base.js'
|
||
import interfaceModel from '../models/interface.js'
|
||
import groupModel from '../models/group'
|
||
import commons from '../utils/commons.js'
|
||
import userModel from '../models/user.js'
|
||
|
||
class projectController extends baseController {
|
||
|
||
constructor(ctx){
|
||
super(ctx)
|
||
this.Model = yapi.getInst(projectModel);
|
||
this.groupModel = yapi.getInst(groupModel);
|
||
}
|
||
|
||
handleBasepath(basepath){
|
||
if(!basepath) return false;
|
||
if(basepath[0] !== '/') basepath = '/' + basepath;
|
||
if(basepath[basepath.length -1] === '/') basepath = basepath.substr(0, basepath.length -1)
|
||
if(!this.verifyPath(basepath)){
|
||
return false;
|
||
}
|
||
return basepath;
|
||
}
|
||
|
||
verifyPath(path){
|
||
if(/^[a-zA-Z0-9\-\/_:]+$/.test(path)){
|
||
return true;
|
||
}else{
|
||
return false;
|
||
}
|
||
}
|
||
|
||
verifyDomain(domain){
|
||
if(!domain) return false;
|
||
if(/^[a-zA-Z0-9\-_\.]+[a-zA-Z]{2,6}$/.test(domain)){
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* 添加项目分组
|
||
* @interface /project/add
|
||
* @method POST
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {String} name 项目名称,不能为空
|
||
* @param {String} basepath 项目基本路径,不能为空
|
||
* @param {String} prd_host 项目线上域名,不能为空。可通过配置的域名访问到mock数据
|
||
* @param {String} protocol 线上域名协议,不能为空
|
||
* @param {Number} group_id 项目分组id,不能为空
|
||
* @param {String} [desc] 项目描述
|
||
* @returns {Object}
|
||
* @example ./api/project/add.json
|
||
*/
|
||
async add(ctx) {
|
||
let params = ctx.request.body;
|
||
|
||
if(!params.group_id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目分组id不能为空');
|
||
}
|
||
|
||
if(!params.name){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目名不能为空');
|
||
}
|
||
|
||
let checkRepeat = await this.Model.checkNameRepeat(params.name);
|
||
if(checkRepeat > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '已存在的项目名');
|
||
}
|
||
|
||
if(!params.basepath){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目basepath不能为空');
|
||
}
|
||
if(!params.prd_host){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目domain不能为空');
|
||
}
|
||
|
||
if((params.basepath = this.handleBasepath(params.basepath)) === false){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, 'basepath格式有误')
|
||
}
|
||
|
||
if(!this.verifyDomain(params.prd_host)){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '线上域名格式有误')
|
||
}
|
||
|
||
let checkRepeatDomain = await this.Model.checkDomainRepeat(params.prd_host, params.basepath);
|
||
if(checkRepeatDomain > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '已存在domain和basepath');
|
||
}
|
||
|
||
let data = {
|
||
name: params.name,
|
||
desc: params.desc,
|
||
prd_host: params.prd_host,
|
||
basepath: params.basepath,
|
||
protocol: params.protocol || 'http',
|
||
members: [this.getUid()],
|
||
uid: this.getUid(),
|
||
group_id: params.group_id,
|
||
add_time: yapi.commons.time(),
|
||
up_time: yapi.commons.time()
|
||
}
|
||
|
||
try{
|
||
let result = await this.Model.save(data);
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
}catch(e){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
|
||
}
|
||
/**
|
||
* 添加项目
|
||
* @interface /project/add_member
|
||
* @method POST
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @param {String} member_uid 项目成员uid,不能为空
|
||
* @returns {Object}
|
||
* @example ./api/project/add_member.json
|
||
*/
|
||
async addMember(ctx){
|
||
let params = ctx.request.body;
|
||
if(!params.member_uid){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目成员uid不能为空');
|
||
}
|
||
if(!params.id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目id不能为空');
|
||
}
|
||
|
||
var check = await this.Model.checkMemberRepeat(params.id, params.member_uid);
|
||
if(check > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目成员已存在');
|
||
}
|
||
try{
|
||
let result = await this.Model.addMember(params.id, params.member_uid);
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
}catch(e){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
|
||
}
|
||
/**
|
||
* 添加项目
|
||
* @interface /project/del_member
|
||
* @method POST
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @param {member_uid} uid 项目成员uid,不能为空
|
||
* @returns {Object}
|
||
* @example ./api/project/del_member.json
|
||
*/
|
||
|
||
async delMember(ctx){
|
||
let params = ctx.request.body;
|
||
if(!params.member_uid){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目成员uid不能为空');
|
||
}
|
||
if(!params.id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目id不能为空');
|
||
}
|
||
var check = await this.Model.checkMemberRepeat(params.id, params.member_uid);
|
||
if(check === 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目成员不存在');
|
||
}
|
||
|
||
try{
|
||
let result = await this.Model.delMember(params.id, params.member_uid);
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
}catch(e){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 获取项目成员列表
|
||
* @interface /project/get_member_list
|
||
* @method GET
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @return {Object}
|
||
* @example ./api/project/get_member_list.json
|
||
*/
|
||
|
||
async getMemberList(ctx) {
|
||
let params = ctx.request.query;
|
||
if(!params.id) {
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目id不能为空');
|
||
}
|
||
|
||
try {
|
||
let project = await this.Model.get(params.id);
|
||
let userInst = yapi.getInst(userModel);
|
||
let result = await userInst.findByUids(project.members);
|
||
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
} catch(e) {
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 添加项目
|
||
* @interface /project/get
|
||
* @method GET
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @returns {Object}
|
||
* @example ./api/project/get.json
|
||
*/
|
||
|
||
async get(ctx){
|
||
let params = ctx.request.query;
|
||
if(!params.id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目id不能为空');
|
||
}
|
||
try{
|
||
let result = await this.Model.get(params.id);
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
}catch(e){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 获取项目列表
|
||
* @interface /project/list
|
||
* @method GET
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} group_id 项目group_id,不能为空
|
||
* @param {Number} [page] 分页页码
|
||
* @param {Number} [limit] 每页数据条目,默认为10
|
||
* @returns {Object}
|
||
* @example ./api/project/list.json
|
||
*/
|
||
|
||
async list(ctx) {
|
||
let group_id = ctx.request.query.group_id,
|
||
page = ctx.request.query.page || 1,
|
||
limit = ctx.request.query.limit || 10;
|
||
|
||
if(!group_id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目分组id不能为空');
|
||
}
|
||
|
||
try{
|
||
let result = await this.Model.listWithPaging(group_id, page, limit);
|
||
let count = await this.Model.listCount(group_id);
|
||
let uids = [];
|
||
result.forEach( (item)=> {
|
||
if(uids.indexOf(item.uid) !== -1){
|
||
uids.push(item.uid)
|
||
}
|
||
|
||
} )
|
||
|
||
let _users = {}, users = await yapi.getInst(userModel).findByUids(uids);
|
||
users.forEach((item)=> {
|
||
_users[item._id] = item;
|
||
} )
|
||
ctx.body = yapi.commons.resReturn({
|
||
total: Math.ceil(count / limit),
|
||
list: result,
|
||
userinfo: _users
|
||
})
|
||
}catch(err){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 删除项目
|
||
* @interface /project/del
|
||
* @method POST
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @returns {Object}
|
||
* @example ./api/project/del.json
|
||
*/
|
||
|
||
async del(ctx){
|
||
try{
|
||
let id = ctx.request.body.id;
|
||
if(!id){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '项目id不能为空');
|
||
}
|
||
let interfaceInst = yapi.getInst(interfaceModel);
|
||
let count = await interfaceInst.countByProjectId(id);
|
||
if(count > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 400, '请先删除该项目下所有接口');
|
||
}
|
||
|
||
if(await this.jungeProjectAuth(id) !== true){
|
||
return ctx.body = yapi.commons.resReturn(null, 405, '没有权限');
|
||
}
|
||
let result = await this.Model.del(id);
|
||
ctx.body = yapi.commons.resReturn(result);
|
||
}catch(err){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 编辑项目
|
||
* @interface /project/up
|
||
* @method POST
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {Number} id 项目id,不能为空
|
||
* @param {String} name 项目名称,不能为空
|
||
* @param {String} basepath 项目基本路径,不能为空
|
||
* @param {String} prd_host 项目线上域名,不能为空。可通过配置的域名访问到mock数据
|
||
* @param {String} [desc] 项目描述
|
||
* @param {Array} [env] 项目环境配置
|
||
* @param {String} [env[].name] 环境名称
|
||
* @param {String} [env[].domain] 环境域名
|
||
* @returns {Object}
|
||
* @example ./api/project/up.json
|
||
*/
|
||
|
||
async up(ctx){
|
||
try{
|
||
let id = ctx.request.body.id;
|
||
let params = ctx.request.body;
|
||
if(!id){
|
||
return ctx.body = yapi.commons.resReturn(null, 405, '项目id不能为空');
|
||
}
|
||
|
||
if(await this.jungeMemberAuth(id, this.getUid()) !== true){
|
||
return ctx.body = yapi.commons.resReturn(null, 405, '没有权限');
|
||
}
|
||
|
||
let projectData = await this.Model.get(id);
|
||
|
||
if(params.basepath = (this.handleBasepath(params.basepath)) === false){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, 'basepath格式有误')
|
||
}
|
||
|
||
if(!this.verifyDomain(params.prd_host)){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '线上域名格式有误')
|
||
}
|
||
|
||
if(projectData.name === params.name){
|
||
delete params.name;
|
||
}
|
||
if(projectData.basepath === params.basepath && projectData.prd_host === params.prd_host){
|
||
delete params.basepath
|
||
delete params.prd_host
|
||
}
|
||
|
||
if(params.name){
|
||
let checkRepeat = await this.Model.checkNameRepeat(params.name);
|
||
if(checkRepeat > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '已存在的项目名');
|
||
}
|
||
}
|
||
|
||
if(params.basepath && params.prd_host){
|
||
let checkRepeatDomain = await this.Model.checkDomainRepeat(params.prd_host, params.basepath);
|
||
if(checkRepeatDomain > 0){
|
||
return ctx.body = yapi.commons.resReturn(null, 401, '已存在domain和basepath');
|
||
}
|
||
}
|
||
|
||
let data= {
|
||
uid: this.getUid(),
|
||
up_time: yapi.commons.time()
|
||
}
|
||
|
||
if(params.name) data.name = params.name;
|
||
if(params.desc) data.desc = params.desc;
|
||
if(params.prd_host && params.basepath){
|
||
data.prd_host = params.prd_host;
|
||
data.basepath = params.basepath;
|
||
}
|
||
if(params.protocol) data.protocol = params.protocol;
|
||
if(params.env) data.env = params.env;
|
||
|
||
let result = await this.Model.up(id, data);
|
||
ctx.body = yapi.commons.resReturn(result)
|
||
}catch(e){
|
||
ctx.body = yapi.commons.resReturn(null, 402, e.message)
|
||
}
|
||
}
|
||
|
||
/**
|
||
* 模糊搜索项目名称或者组名称
|
||
* @interface /project/search
|
||
* @method GET
|
||
* @category project
|
||
* @foldnumber 10
|
||
* @param {String} q
|
||
* @return {Object}
|
||
* @example ./api/project/search.json
|
||
*/
|
||
async search(ctx) {
|
||
const { q } = ctx.request.query;
|
||
|
||
if (!q) {
|
||
return ctx.body = yapi.commons.resReturn(void 0, 400, 'No keyword.')
|
||
}
|
||
|
||
if (!yapi.commons.validateSearchKeyword(q)) {
|
||
return ctx.body = yapi.commons.resReturn(void 0, 400, 'Bad query.')
|
||
}
|
||
|
||
let projectList = await this.Model.search(q);
|
||
let groupList = await this.groupModel.search(q);
|
||
let projectRules = [
|
||
'_id',
|
||
'name',
|
||
'basepath',
|
||
'uid',
|
||
'env',
|
||
'members',
|
||
{ key: 'group_id', alias: 'groupId' },
|
||
{ key: 'up_time', alias: 'upTime' },
|
||
{ key: 'prd_host', alias: 'prdHost' },
|
||
{ key: 'add_time', alias: 'addTime' }
|
||
];
|
||
let groupRules = [
|
||
'_id',
|
||
'uid',
|
||
{ key: 'group_name', alias: 'groupName'},
|
||
{ key: 'group_desc', alias: 'groupDesc' },
|
||
{ key: 'add_time', alias: 'addTime' },
|
||
{ key: 'up_time', alias: 'upTime' }
|
||
];
|
||
|
||
projectList = commons.filterRes(projectList, projectRules);
|
||
groupList = commons.filterRes(groupList, groupRules);
|
||
|
||
let queryList = {
|
||
project: projectList,
|
||
group: groupList
|
||
};
|
||
|
||
return ctx.body = yapi.commons.resReturn(queryList, 0, 'ok')
|
||
}
|
||
}
|
||
|
||
module.exports = projectController;
|
||
</pre>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<!-- <div class="docs-header" id="content" tabindex="-1">
|
||
<div class="container">
|
||
<h1>yapi : ./server/controllers/project.js</h1>
|
||
<p>源代码</p>
|
||
</div>
|
||
</div> -->
|
||
|
||
<footer class="docs-footer" role="contentinfo">
|
||
<div class="container">
|
||
<p></p>
|
||
</div>
|
||
</footer>
|
||
</div>
|
||
<script type="text/javascript">
|
||
SyntaxHighlighter.all();
|
||
|
||
function getTop(node){
|
||
return node.offsetTop + (node.offsetParent ? getTop(node.offsetParent) : 0);
|
||
}
|
||
|
||
document.addEventListener('DOMContentLoaded', function() {
|
||
setTimeout(function() {
|
||
try {
|
||
var lineNum = (parseInt(location.hash.replace(/#/g, '')) - 1) || 0,
|
||
node = document.querySelectorAll('div.line')[lineNum];
|
||
document.body.scrollTop = getTop(node);
|
||
node.className += ' highlight';
|
||
} catch(e) {}
|
||
}, 500);
|
||
}, false);
|
||
</script>
|
||
</body>
|
||
</html>
|