This commit is contained in:
yi-ge 2017-01-23 04:59:10 +08:00
parent f5dfeaa123
commit 8ed5794348
11 changed files with 140 additions and 39 deletions

View File

@ -79,7 +79,7 @@ http://pm2.keymetrics.io/docs/usage/docker-pm2-nodejs/
### Docker部署说明 ### Docker部署说明
``` ```
$ docker pull node $ docker pull node
$ docker run -itd --name RESTfulAPI -v '$PWD':/usr/src/app -w /usr/src/app node node ./dist/app.js $ docker run -itd --name RESTfulAPI -v `pwd`:/usr/src/app -w /usr/src/app node node ./dist/app.js
``` ```
通过'docker ps'查看是否运行成功及运行状态 通过'docker ps'查看是否运行成功及运行状态
@ -146,17 +146,15 @@ $ docker run -itd -p 80:80 -p 443:443 -v `pwd`/nginx_config:/etc/nginx/conf.d ng
> 引入插件的版本将会持续更新 > 引入插件的版本将会持续更新
引入的插件: 引入的插件:
`` koa@2 koa-bodyparser@next koa-router@next koa-session2 koa-static2 koa-compose require-directory babel-cli babel-register babel-plugin-transform-runtime babel-preset-es2015 babel-preset-stage-2 gulp gulp-eslint eslint eslint-config-standard eslint-friendly-formatter eslint-plugin-html eslint-plugin-promise nodemailer promise-mysql `` `` koa@2 koa-body@2 koa-router@next koa-session2 koa-static2 koa-compose koa-multer require-directory babel-cli babel-register babel-plugin-transform-runtime babel-preset-es2015 babel-preset-stage-2 gulp gulp-eslint eslint eslint-config-standard eslint-friendly-formatter eslint-plugin-html eslint-plugin-promise nodemailer promise-mysql ``
koa-multer': '^1.0.0',
**koa2**: HTTP框架 **koa2**: HTTP框架
 Synopsis: HTTP framework.  Synopsis: HTTP framework.
 From: https://github.com/koajs/koa v2  From: https://github.com/koajs/koa v2
**koa-bodyparser**: body解析器 **koa-body**: body解析器
 Synopsis: A body parser for koa, base on co-body. support json, form and text type body.  Synopsis: A full-feature koa body parser middleware.
 From: https://github.com/koajs/logger  From: https://github.com/dlau/koa-body
**koa-router**: Koa路由 **koa-router**: Koa路由
 Synopsis: Router middleware for koa.  Synopsis: Router middleware for koa.
@ -174,6 +172,10 @@ koa-multer': '^1.0.0',
 Synopsis: Compose several middleware into one.  Synopsis: Compose several middleware into one.
 From: https://github.com/koajs/compose  From: https://github.com/koajs/compose
**koa-multer**: 处理数据中间件
 Synopsis: Multer is a node.js middleware for handling multipart/form-data for koa.
 From: https://github.com/koa-modules/multer
**require-directory**: 递归遍历指定目录 **require-directory**: 递归遍历指定目录
 Synopsis: Recursively iterates over specified directory.  Synopsis: Recursively iterates over specified directory.
 From: https://github.com/troygoode/node-require-directory  From: https://github.com/troygoode/node-require-directory
@ -234,6 +236,14 @@ koa-multer': '^1.0.0',
 Synopsis: Promise Mysql.  Synopsis: Promise Mysql.
 From: https://github.com/lukeb-uk/node-promise-mysql  From: https://github.com/lukeb-uk/node-promise-mysql
**sequelize**: 关系型数据库ORM
 Synopsis: Sequelize is a promise-based ORM for Node.js.
 From: https://github.com/sequelize/sequelize
**mysql**: MySQL库
 Synopsis: A pure node.js JavaScript Client implementing the MySql protocol.
 From: https://github.com/mysqljs/mysql
支持Koa2的中间件列表https://github.com/koajs/koa/wiki 支持Koa2的中间件列表https://github.com/koajs/koa/wiki
**其它经常配合Koa2的插件** **其它经常配合Koa2的插件**
@ -255,6 +265,9 @@ HTTP2推送中间件https://github.com/silenceisgolden/koa-server-push
 Synopsis: Development style logger middleware for Koa.  Synopsis: Development style logger middleware for Koa.
 From: https://github.com/koajs/logger  From: https://github.com/koajs/logger
**koa-onerror**:
Koa的错误拦截中间件需要配合上面的插件使用https://github.com/koajs/onerror
## 目录结构说明 ## 目录结构说明
```bash ```bash

View File

@ -2,10 +2,11 @@ import path from 'path'
export let SystemConfig = { export let SystemConfig = {
HTTP_server_type: 'http://', // HTTP服务器地址,包含"http://"或"https://" HTTP_server_type: 'http://', // HTTP服务器地址,包含"http://"或"https://"
HTTP_server_host: 'localhost',// HTTP服务器地址,请勿添加"http://" HTTP_server_host: 'localhost', // HTTP服务器地址,请勿添加"http://"
HTTP_server_port: '3000',// HTTP服务器端口号 HTTP_server_port: '3000', // HTTP服务器端口号
System_country: 'zh-cn', // 所在国家的国家代码 System_country: 'zh-cn', // 所在国家的国家代码
System_plugin_path: path.join(__dirname, "plugins/"), // 插件路径 System_plugin_path: path.join(__dirname, "../src/plugins"), // 插件路径
Session_Key: 'RESTfulAPI', // 生产环境务必随机设置一个值
mysql_host: 'localhost', // MySQL服务器地址 mysql_host: 'localhost', // MySQL服务器地址
mysql_user: 'root', // 数据库用户名 mysql_user: 'root', // 数据库用户名
mysql_password: 'root', // 数据库密码 mysql_password: 'root', // 数据库密码
@ -13,3 +14,10 @@ export let SystemConfig = {
mysql_port: 3306, // 数据库端口号 mysql_port: 3306, // 数据库端口号
mysql_prefix: 'api_' // 默认"api_" mysql_prefix: 'api_' // 默认"api_"
} }
export let SendEmail = {
service: 'smtp.abcd.com', // SMTP服务提供商域名
username: 'postmaster%40abcd.com', // 用户名/用户邮箱
password: 'password', // 邮箱密码
sender_address: '"XX平台 👥" <postmaster@abcd.com>'
}

View File

@ -11,13 +11,16 @@
}, },
"dependencies": { "dependencies": {
"koa": "^2.0.0", "koa": "^2.0.0",
"koa-bodyparser": "^3.2.0", "koa-body": "^2.0.0",
"koa-compose": "^2.5.1", "koa-compose": "^2.5.1",
"koa-multer": "^1.0.1",
"koa-router": "^7.0.1", "koa-router": "^7.0.1",
"koa-session2": "^1.0.8", "koa-session2": "^1.0.8",
"mysql": "^2.12.0",
"nodemailer": "^2.7.2", "nodemailer": "^2.7.2",
"promise-mysql": "^3.0.0", "promise-mysql": "^3.0.0",
"require-directory": "^2.1.1" "require-directory": "^2.1.1",
"sequelize": "^3.29.0"
}, },
"devDependencies": { "devDependencies": {
"babel-cli": "^6.22.2", "babel-cli": "^6.22.2",

View File

@ -1,32 +1,23 @@
import Koa2 from 'koa' import Koa2 from 'koa'
import KoaBodyParser from 'koa-bodyparser' import KoaBody from 'koa-body'
import KoaSession from 'koa-session2' import KoaSession from 'koa-session2'
import KoaStatic from 'koa-static2' import KoaStatic from 'koa-static2'
import { SystemConfig } from '../config/main.js' import { SystemConfig } from '../config/main.js'
import path from 'path' import path from 'path'
import MainRoutes from './routes/main-routes' import MainRoutes from './routes/main-routes'
import ErrorRoutes from './routes/error-routes' import ErrorRoutes from './routes/error-routes'
import PluginLoader from './tool/PluginLoader' // import PluginLoader from './tool/PluginLoader'
const app = new Koa2() const app = new Koa2()
const BodyParser = new KoaBodyParser()
const env = process.env.NODE_ENV || 'development' // Current mode const env = process.env.NODE_ENV || 'development' // Current mode
app.use(BodyParser({ app.use(KoaBody({
detectJSON: function (ctx) { multipart: true
return /\.json$/i.test(ctx.path) // formidable: {uploadDir: path.join(__dirname, '../../../assets/uploads/')}
},
extendTypes: {
json: ['application/x-javascript'] // will parse application/x-javascript type body as a JSON string
},
onerror: function (err, ctx) {
ctx.throw('body parse error:' + err, 422)
}
})) // Processing request })) // Processing request
.use(KoaStatic('assets', path.resolve(__dirname, '../assets'))) // Static resource .use(KoaStatic('assets', path.resolve(__dirname, '../assets'))) // Static resource
.use(KoaSession({key: 'RESTfulAPI'})) // Set Session 生产环境务必随机设置一个值 .use(KoaSession({key: SystemConfig.Session_Key})) // Set Session
.use(PluginLoader(SystemConfig.System_plugin_path)) // .use(PluginLoader(SystemConfig.System_plugin_path))
.use((ctx, next) => { .use((ctx, next) => {
if (ctx.request.header.host.split(':')[0] === 'api.XXX.com' || ctx.request.header.host.split(':')[0] === '127.0.0.1') { if (ctx.request.header.host.split(':')[0] === 'api.XXX.com' || ctx.request.header.host.split(':')[0] === '127.0.0.1') {
ctx.set('Access-Control-Allow-Origin', '*') ctx.set('Access-Control-Allow-Origin', '*')

View File

@ -0,0 +1,30 @@
export let Get = (ctx) => {
ctx.body = {
result: 'get',
name: ctx.params.name
}
}
export let Post = (ctx) => {
ctx.body = {
result: 'post',
name: ctx.params.name,
para: ctx.request.body.para
}
}
export let Put = (ctx) => {
ctx.body = {
result: 'put',
name: ctx.params.name,
para: ctx.request.body.para
}
}
export let Delect = (ctx) => {
ctx.body = {
result: 'delect',
name: ctx.params.name,
para: ctx.request.body.para
}
}

35
src/controllers/upload.js Normal file
View File

@ -0,0 +1,35 @@
import multer from 'koa-multer'
import path from 'path'
/***
* 说明该方法只支持`multipart/form-data`方式的上传postman可以调试
*/
let storage = multer.diskStorage({
// 设置上传后文件路径
destination: function (req, file, cb) {
cb(null, path.join(__dirname, '../../../assets/uploads/'))
},
// 给上传文件重命名,获取添加后缀名
filename: function (req, file, cb) {
let fileFormat = (file.originalname).split('.')
cb(null, Date.now() + '.' + fileFormat[fileFormat.length - 1])
}
})
export let upload = multer({
storage: storage,
// dest: 'uploads/', //上传文件保存的路径,如果你想在上传时进行更多的控制, 你可以使用storage选项替代dest. Multer 具有 DiskStorage 和 MemoryStorage 两个存储引擎; 另外还可以从第三方获得更多可用的引擎.
// fileFilter: function (req, file, cb) {
// let mimetypes = (['text/*', 'image/*', 'video/*', 'audio/*', 'application/zip']).join(',');
// let testItems = file.mimetype.split('/');
// if ((new RegExp('\b' + testItems[0] + '/\*', 'i')).test(mimetypes) || (new RegExp('\*/' + testItems[1] + '\b', 'i')).test(mimetypes) || (new RegExp('\b' + testItems[0] + '/' + testItems[1] + '\b', 'i')).test(mimetypes)) {
// cb(null, true);
// } else {
// return cb(new Error('Only image, plain text, audio, video and zip format files are allowed!'), false);
// }
// }, // fileFilter要在这里声明才行用instance.fileFilter = funciton(){};是不管用的,
limits: {
fileSize: 200 * 1024 * 1024 // Max file size in bytes (20 MB) 限制上传的文件大小,不设置则是无限
}
})

12
src/lib/sequelize.js Normal file
View File

@ -0,0 +1,12 @@
import Sequelize from 'sequelize'
import { SystemConfig } from '../config/main'
export default new Sequelize(SystemConfig.mysql_database, SystemConfig.mysql_user, SystemConfig.mysql_password, {
host: SystemConfig.mysql_host,
dialect: 'mysql',
pool: {
max: 50,
min: 0,
idle: 10000
}
})

View File

@ -1,4 +1,5 @@
import nodemailer from 'nodemailer' import nodemailer from 'nodemailer'
import { SendEmail } from '../../../config/main'
// 发送Email目前使用的是阿里云SMTP发送邮件 // 发送Email目前使用的是阿里云SMTP发送邮件
// receivers 目标邮箱,可以用英文逗号分隔多个。(我没试过) // receivers 目标邮箱,可以用英文逗号分隔多个。(我没试过)
@ -10,11 +11,11 @@ import nodemailer from 'nodemailer'
// info 是返回的消息,可能是结果的文本,也可能是对象。(这个错误不要暴露给用户) // info 是返回的消息,可能是结果的文本,也可能是对象。(这个错误不要暴露给用户)
export let sendemail = (receivers, subject, text, html) => { export let sendemail = (receivers, subject, text, html) => {
return new Promise(function (resolve) { return new Promise(function (resolve) {
let transporter = nodemailer.createTransport('smtp://postmaster%40abcd.com:password@smtp.abcd.com') let transporter = nodemailer.createTransport('smtp://' + SendEmail.username + ':' + SendEmail.password + '@' + SendEmail.service)
// setup e-mail data with unicode symbols // setup e-mail data with unicode symbols
let mailOptions = { let mailOptions = {
from: '"XX平台 👥" <postmaster@abcd.com>', // sender address from: SendEmail.sender_address, // sender address
to: receivers, to: receivers,
subject: subject, subject: subject,
text: text || 'Hello world 🐴', // plaintext body text: text || 'Hello world 🐴', // plaintext body

View File

@ -1,5 +1,5 @@
import KoaRouter from 'koa-router' import KoaRouter from 'koa-router'
// import controllers from '../controllers/index.js' import controllers from '../controllers/index.js'
const router = new KoaRouter() const router = new KoaRouter()
@ -7,9 +7,17 @@ router
.get('/', function (ctx, next) { .get('/', function (ctx, next) {
ctx.body = '禁止访问!' ctx.body = '禁止访问!'
}) // HOME 路由 }) // HOME 路由
// .get('/api/:api_type/:name', controllers.api.api_get) .all('/upload', controllers.upload.upload.array('file'), function (ctx, next) { // 上传到本地示例
// .put('/api/:api_type/:name', controllers.api_put.api_put return new Promise(function (resolve, reject) {
// .post('/api/:api_type/:name', controllers.api.default) // 允许跨域,正式环境要改为对应域名
// .delect('/api/:api_type/:name', controllers.api.default) ctx.set('Access-Control-Allow-Origin', '*')
// ctx.req.files.filename = ctx.req.files.path
resolve(ctx.body = ctx.req.files)
})
})
.get('/api/:name', controllers.api.Get)
.post('/api/:name', controllers.api.Post)
.put('/api/:name', controllers.api.Put)
.del('/api/:name', controllers.api.Delect)
module.exports = router module.exports = router

View File

@ -34,8 +34,8 @@ export let OptionFormat = (GetOptions) => {
// 替换SQL字符串中的前缀 // 替换SQL字符串中的前缀
export let SqlFormat = (str) => { export let SqlFormat = (str) => {
if (SystemConfig.mysql_prefix !== 'bm_') { if (SystemConfig.mysql_prefix !== 'api_') {
str = str.replace(/bm_/g, SystemConfig.mysql_prefix) str = str.replace(/api_/g, SystemConfig.mysql_prefix)
} }
return str return str
} }