zoukankan      html  css  js  c++  java
  • koa2+mongodb搭建后台


    title: koa2+mongodb搭建后台
    date: 2018-12-28
    categories:

    • frontend
      tags:
    • koa2
    • mongodb
    • nodejs

    前言

    毫无疑问,目前nodejs里面用来开发后台的首选就是koa2+mongodb的组合了。参考过很多资料,都是零零碎碎不齐全,要么很简单只是教你如何运行一个demo,要么只讲了简单的一方面,要么就是一个复杂的koa项目生成器,我设想的一个最基础的后台应该具有以下内容:

    此处只讨论前后端分离,后台项目提供接口,不考虑模板渲染之类的,毕竟你都用nodejs做后台了,还不做前后端分离也太说不过去了

    • 对传入、返回及错误数据做统一处理
    • 支持跨域
    • 使用 token 做身份验证
    • 完善的日志记录
    • 支持发送邮件
    • 上传文件
    • 常见的数据库操作,对列表数据分页,返回指定行数据做封装
    • 调试

    所以在折腾完之后,就想把整个过程记录一下,如果你正好是刚开始摸索,应该能让你避免不少弯路。

    创建基础项目

    • 创建koa-mongo目录,并运行 npm init 创建package.json
    mkdir koa-mongo
    cd ./koa-mongo
    npm init
    
    • 安装基础包
    npm install koa # koa,必须的
    npm install koa-router  # 路由,必须的,这里要注意的是还有一个koa-route,这两个是不同的,不要用koa-route
    npm install koa-static  # 静态资源,必须的
    
    • 创建app.js,填入以下内容:
    // koa
    const Koa = require('koa')
    const app = new Koa()
    
    // midleware
    const serve = require('koa-static')
    
    app.use(serve('./assets'))
    
    var server = app.listen(3000, function (){
        const host = server.address().address;
        const port = server.address().port;
        console.log('app start listening at http://%s:%s', host, port);
    });
    

    此处创建了一个实例,监听3000端口,将assets目录作为静态资源运行,我们创建一个assets目录,里面创建一个index.html,然后我们运行起来试试:

    node ./app.js
    

    此时会打印一行日志:app start listening at http://:::3000,让我们来访问试试:

    ok,koa启动一个项目就是这么简单。。

    添加路由

    在koa-router的使用说明中,我们可以看到是这样使用的:

    var router = new Router();
     
    router.get('/', (ctx, next) => {
      // ctx.router available
    });
     
    app
      .use(router.routes())
      .use(router.allowedMethods());
    

    因为正常项目中,controller肯定不止一个的,所以我把目录写成这样:

    ├─ controller
        ├─ test-controller
    ├─ router.js
    

    controller目录用来放置所有的controller,在router.js中统一汇总,app.js中只需要使用router.js即可。
    test-controller:

    const hello = async (ctx, next) => {
        ctx.body = 'hello world'
        ctx.status = 200;
    }
    
    module.exports = {
        'test/hello': hello,
    }
    

    router.js:

    const Router = require('koa-router')
    const router = new Router({
        prefix: '/api', // 统一前缀,接口全部为 /api/xxx 格式
    })
    
    const testController = require('../controller/test-controller')
    
    Object.keys(testController).forEach(key=>{
        router.all("/"+key, testController[key]);   // router.all是允许所有的访问方式,如果需要限定则改为指定方式即可
    })
    
    module.exports = router;
    

    app.js:

    // router
    const router = require('./router')
    app.use(router.routes()).use(router.allowedMethods())
    

    这个时候让我们重新启动一下,访问localhost:3000/api/test/hello试试:

    可以看到正确返回了hello world。

    调试

    到这里了,有没有感觉哪里不对劲。每修改一次,都需要手动敲命令重启一次,这简直太烦了好嘛,我们程序猿哪能忍受这个。答案就是使用nodemon,这个玩意儿能监听我们的文件变更,自动运行命令重启应用。使用方式也很简单,这个我们直接全局安装就好了:

    npm install -g nodemon
    

    然后去package.json的scripts中添加一行脚本:

    "dev": "nodemon ./app.js",
    

    然后 npm run dev,把hello函数修改一下返回值,保存,就会看到nodemon自动帮我们重启应用了。

    除了nodemon,类似的工具还有很多,这里就不展开说了,现在的你只需要知道开发用nodemon,线上用pm2就ok了

    解决了自动运行之后,我们来说一下调试。其实nodejs自带了调试的,只需要一个inspect参数,调试的时候就跟我们在chrome中调试是一模一样的。
    我们先去package.json的scripts中添加一行脚本:

    "debug": "nodemon --inspect ./app.js",
    

    然后我们运行 npm run debug,刷新我们的网页,用f12打开,此时我们能看到开发者工具上面多了一个nodejs的图标:

    点击这个图标,就可以跟调试网页一样调试nodejs代码了:

    配置化

    代码就是一步步总结,边写边优化,重构。到目前为止,我们会发现有不少配置性的东西是散乱在不同文件中,比如说项目启动时监听的端口,接口的统一前缀,考虑到我们还会有很多配置项,我们应该把这些写到一个配置文件中集中管理。
    新建一个config.js:

    module.exports= {
        port: 3000,
        apiPrefix: '/api',
    }
    

    然后把用到的地方全部更改为变量

    跨域

    为了防止跨域问题,我们需要使用koa2-cors类库,使用方式很简单:

    npm install koa2-cors
    

    然后在app.js中添加以下内容:

    const cors = require('koa2-cors')
    app.use(cors())
    

    一行代码搞定,cors的具体配置此处就不细说了,有兴趣的可以自己去看看

    处理参数,上传文件

    一般来说,我们是使用koa-bodyparser 和 koa-multer来分别处理表单数据和文件数据的。这两个分别集成也没什么问题,但我们可以直接使用koa-body来完成。koa-body是基于co-body和formidable做了封装,同时支持参数解析和文件上传。最后是把参数和文件分别放到ctx.request.body和ctx.request.files变量中。
    我这里是把上传文件统一放到assets/upload目录中:

    const uploadDir = path.join(__dirname, 'assets/upload/')
    // 此处还需要判断文件夹是否存在,不存在的话就创建
    app.use(koaBody({
        multipart: true,
        encoding: 'utf-8',
        formidable:{
            uploadDir: uploadDir,
            keepExtensions: true,
            maxFieldsSize: 5*1024*1024,
            onFileBegin:(name, file)=>{
    
            }
        }
    }))
    

    然后添加一个upload:

    const upload = async (ctx, next)=>{
       const files = ctx.request.files || {};   # 文件会被解析到ctx.request.files中,是个object
       let fileNames = Object.keys(files);
       if(fileNames.length<=0){
           throw new ApiError('上传文件不能为空')
       }else{
           if(fileNames.length===1){
               ctx.body = files[fileNames[0]].path.replace(config.baseDir+'/assets', "")
           }else {
               ctx.body = fileNames.map(key => files[key].path.replace(config.baseDir+'/assets', ""))
           }
       }
    }
    

    token验证

    使用jwt,待续

    统一返回格式

    待续

    mongodb

    待续

    日志模块

    待续

    微信支付

    待续

  • 相关阅读:
    javascript第七章--DOM
    javascript第六章--BOM
    javascript第五章--函数表达式
    javascript第四章--面向对象的程序设计
    javascript第三章--引用类型
    javascript第二章--变量、作用域和内存问题
    Java 成员变量的区分
    equals 与"=="的区别
    java 基础数据类型
    一个带倒计时按钮的代码片段
  • 原文地址:https://www.cnblogs.com/thyong/p/10192484.html
Copyright © 2011-2022 走看看