egg基础功能2 -- egg操作mongodb

egg操作Mongodb

今天将给大家带来通过egg连接mongodb的相关总结,具体包括:

  • egg如何连接mongodb
  • 通过保存数据验证连接mongodb是成功的

egg连接mongodb

  • 1、 我们通过插件“egg-mongoose”连接mongodb数据库,我们先安装好插件:
npm i --save egg-mongoose

首先,我们通过egg的脚手架的方式创建egg工程,这一步不再细述,请参阅之前的文章。

  • 2、 添加egg-mongoose 插件:在egg工程的config/plugin.js中添加插件egg-mongoose:
'use strict';

/** @type Egg.EggPlugin */
module.exports = {
  mongoose: {
    enable: true,
    package: 'egg-mongoose',
  },
};
  • 3、 添加mongodb的连接信息: 在config/config.default.js中添加mongodb的连接信息:
/* eslint valid-jsdoc: "off" */

'use strict';

/**
 * @param {Egg.EggAppInfo} appInfo app info
 */
module.exports = appInfo => {
  /**
   * built-in config
   * @type {Egg.EggAppConfig}
   **/
  const config = exports = {};

  // use for cookie sign key, should change to your own and keep security
  config.keys = appInfo.name + '_1564639326024_3672';

  // add your middleware config here
  config.middleware = [];

  // 连接mongodb的配置
  config.mongoose = {
    client: {
      url: 'mongodb://127.0.0.1/jianshu',
      options: {},
    },
  };

  // config.cluster = {
  //   listen: {
  //     port: 9999,
  //   },
  // };

  // add your user config here
  const userConfig = {
    // myAppName: 'egg',
  };

  return {
    ...config,
    ...userConfig,
  };
};

到这里,我们就可以通过egg连接mongodb了。下面来验证一下连接!

验证连接

准备工作: 创建集合对象

'use strict';

module.exports = app => {
  const { mongoose } = app;

  return mongoose.model(
    'User',
    new mongoose.Schema({
      user_id: { type: String, unique: true },
      user_name: { type: String },
      age: { type: Number },
      description: { type: String },
      status: { type: Number },
    }),
    'user'
  );
};

我们可以先去数据库中查看相关数据情况, 可以看到:目前仅仅只有user集合,且集合中无数据:


此外,我们在app/目录下创建core目录,并封装BaseController和BaseService:

'use strict';

const { Controller } = require('egg');

class BaseController extends Controller {
  async returnService(promise) {
    const [ error, data ] = await this.wapperError(promise);
    if (error) {
      this.error(error);
    } else {
      this.success(data);
    }
  }

  success(data) {
    this.ctx.body = { status: 'OK', data };
  }

  error(error) {
    this.ctx.body = { status: 'NG', error: error.message || error };
  }

  wapperError(promise) {
    return promise
      .then(data => {
        return [ undefined, data ];
      })
      .catch(err => [ err ]);
  }
}
module.exports = BaseController;

'use strict';

const Service = require('egg').Service;

class BaseMongooseService extends Service {
  get document() {
    throw Error("BaseMongooseService need override property <document>'s get method!");
  }

  /**
     * 分页返回数据
     * @param {Object} option 查询参数
     * @param {String} next 下一条数据的id
     * @param {Number} limit 一页包含多少数据
     * @param {Array} includs 返回数据包含字段数组,为空返回全部字段
     */
  async page(option, next, limit, includs, sort) {
    limit = limit || 50;
    const findLimit = limit + 1;
    const projection = {};
    if (includs && includs instanceof Array) {
      includs.forEach(item => {
        projection[item] = 1;
      });
    }
    if (next) {
      option._id = { $lte: next };
    }

    // const sortPrama ={ _id : -1 } ;
    const sortPrama = (sort ? sort : { _id: -1 });
    const data = await this.document
      .find(option, projection)
      .sort(sortPrama)
      .limit(findLimit);

    if (data.length === findLimit) {
      return { next: data.pop()._id, data, total: data.length };
    }
    return { data, total: data.length };
  }

  /**
     * 分页返回数据
     * @param {Object} option 查询参数
     * @param {number} next 下一条数据的id
     * @param {Number} limit 一页包含多少数据
     * @param {Object} sort 排序
     * @param {Array} includs 返回数据包含字段数组,为空返回全部字段
     */
  async pageList(option, next, limit, includs, sort) {
    limit = limit || 50;
    next = parseInt(next) || 1;
    const projection = {};
    if (includs && includs instanceof Array) {
      includs.forEach(item => {
        projection[item] = 1;
      });
    }

    if (!sort) { throw Error('sort is not find '); }
    const data = await this.document
      .find(option, projection).skip((next - 1) * limit)
      .sort(sort)
      .limit(limit);
    return { next: (next + 1), data, total: data.length };
  }

  /**
     * 查询
     * @param {Object} option 查询参数
     * @param {Array} includs 返回数据包含字段数组,为空返回全部字段
     * @return {Array} 查询结果
     */
  async find(option, includs) {
    const projection = {};
    if (includs && includs instanceof Array) {
      includs.forEach(item => {
        projection[item] = 1;
      });
    }
    return await this.document.find(option, projection);
  }
  async findById(_id, includs) {
    return await this.document.findOne({ _id }, includs);
  }
  async findOne(option, includs) {
    const projection = {};
    if (includs && includs instanceof Array) {
      includs.forEach(item => {
        projection[item] = 1;
      });
    }
    return await this.document.findOne(option, projection).orFail();
  }

  async create(detail) {
    // const Document = this.document;
    const now = new Date().getTime();
    const _create = {
      create_stamp: now,
      update_stamp: now,
    };
    return await new this.document(Object.assign(_create, detail)).save();
  }

  async update(option, detail) {
    const now = new Date().getTime();
    const _update = {
      update_stamp: now,
    };
    await this.document.updateOne(option, Object.assign(_update, detail)).orFail();
  }
  async updateById(_id, detail) {
    return await this.update({ _id }, detail);
  }


}

module.exports = BaseMongooseService;

保存数据,验证连接成功

  • 1、我们在app/controller/目录下定义UserController.js,并且包含add()接口,接收user_name, age参数信息,调用UserService完成对user数据的保存。
'use strict';

const Controller = require('../core/BaseController');

class UserController extends Controller {

  async add() {
    const { userService } = this.ctx.service;
    const { user_name, age } = this.ctx.request.body;
    console.log('Received: user_name = ' + user_name + ', age = ' + age);

    await this.returnService(userService.add(user_name, age));
  }
}

module.exports = UserController;

  • 2、创建app/service目录,创建UserService.js,接收Controller传输过来的数据,完成对user的保存。
'use strict';

const Service = require('../core/BaseMongooseService');
const uuidv1 = require('uuid/v1');

class UserService extends Service {
  get document() {
    return this.ctx.model.User;
  }

  /**
   * 添加用户
   * @param user_name
   * @param age
   * @returns {Promise<void>}
   */
  async add(user_name, age) {
    const { User } = this.ctx.model;
    const userMap = new User();
    userMap.user_name = user_name;
    userMap.age = age;
    userMap.user_id = uuidv1();
    userMap.description = '用户';
    userMap.status = 0;

    userMap.save();
  }
}
module.exports = UserService;

  • 3、定义路由:在router.js下定义add接口的路由:
router.post('/add', controller.userController.add);
  • 4、使用postman调用接口,完成保存


  • 5、调用接口,异常现象解决
    在第4步,我们直接调用postman之后,程序会发生异常。异常如下:


这是因为egg的csrf防范的安全,我们在config.default.js中添加如下代码即可解决:

config.security = {
   csrf: {
     enable: false,
     ignoreJSON: true,
   },
   domainWhiteList: [ '*' ],
 };
  • 6、 正常调用结果:

postman:


console控制台:


image.png

mongo数据库:


至此,我们完成了egg连接mongodb步骤。后面将详细总结egg+mongo的增删改查及多数据源情况下的配置。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容