| 描述:
一个独立完整的技术团队简历及面试系统。

项目价值:
通过逐年推广形成公司技术人才库,沉淀简历资产及过程信息
●便于未来简历再次捞出,挖掘候选人
●便于做年度招聘分析与复盘,优化招聘效果

技术栈:
Node.js + Egg + MySQL + React

| | —- |

功能说明

1. 用户管理

  1. 账号信息管理
  2. 团队成员的权限说明
    1. 超级管理员 - 管理一切,可增删团队,及设置团队管理员
      1. 皆为软删除(软删除是打标记、硬删除是表数据清除)
    2. 团队管理员 - 管理成员增删与简历增删(皆为软删除)
      1. 可为团队成员手动重置密码(密码符合一定强度)
    3. 普通成员 - 查看上传历史,上传简历,其他信息不可见
    4. 面试官 - 简历增改删、评论、指派(皆为软删除)
      1. 第一个情况,是团队管理者,能参与相关技术面
      2. 第二个情况,是 HR,能参与面试前后的沟通
  3. 账号登录/登出

    2. 团队管理

    超级管理员 可以做 团队增删与修改

  4. 可以新增团队,或修改既有团队

  5. 新增团队时默认创建团队简历池
    1. 简历池存放待筛选简历
    2. 简历可被废弃(软删除)
  6. 设置团队管理员,从管理员池中捞取指定

    3. 职位管理

    团队管理员 和 超级管理员 可以做 职位岗位增删与修改

  7. 可以新增公司的职位

  8. 可以新增职位下的岗位要求
  9. 将岗位关联到团队下

    4. 简历管理

    简历新增与指派

  10. 简历新增默认到简历池

    1. 填写简历描述字段
    2. 上传简历附件
    3. 简历上传完毕,自动进入当前登录人上传历史页面
  11. 简历进简历池后,默认指派给团队管理员筛选
    1. 团队管理员拿到简历之后,做一下基础筛选(是否有重复简历,是否符合 HC 对应要求…)
    2. 筛选通过,团队管理员指派给适合的面试官,进入面试流程
    3. 筛选失败,团队管理员直接给简历标记为不符合。
  12. 简历面试指派

    1. 由团队管理员进行面试官的指派
    2. 面试官可修改指派人,将候选人给其他面试官面试

      5. 面试管理

  13. 面试官面试阶段最多支持到 10 面

    1. 一面、二面…到十面
    2. 每一轮面试需要填写了评价之后,才能进行下一环节指派
    3. 如果当时确认当前候选人不适合,面试官可以标记为不合适,直接指派给 HR ,结束当前面试流程
  14. 面试评价
    1. 每一轮面试评价中需要包含候选人基础评价和职能评级
    2. 单个环节面试结果可以为不通过,不适合,适合
    3. 单环节面试结果如果为不通过,或者不适合,也可以进入到下一个流程

页面访问链路

上传简历流程(所有人)

image.png

上传简历环节,简洁版可以只填写字段可以只用填写姓名,手机号,推荐岗位,推荐类型(校招或社招)简历和备注

面试评价流程(所有人)

image.png

简历筛选及面试(团队管理员)

image.png

人员新增(团队管理员)

image.png

  • 团队管理员 可以新增类型为面试官和普通成员 的成员
  • 超级管理员 可以新增类型为 团队管理员,面试官和普通成员 的成员
  • 多个部门的团队管理员可以是一个

    岗位新增(团队管理员)

    image.png

    新增团队及设置管理员(超级管理员)

    image.png

    只有角色是团队管理员的成员,才可以关联为团队的团队管理员

技术架构

image.png

项目里程碑

里程碑 1 - 项目初始化及页面可渲染

  • 完成项目环境搭建
  • 完成项目的目录文件结构初始化
  • 基于 Egg 的前后端分离项目,能基础跑起来
  • 首页/简历详情页的样式完成,可以渲染出登录首页

    注意事项

  • 项目目录拆分合理

  • 项目的各种配置文件完整
  • 后台登录首页可以正常渲染出来

里程碑 2 - 数据建模及成员注册登录

  • 完成 MySQL 环境搭建与数据库连接
  • 完成 用户/简历/面试 等相关数据表结构的 Schema 设计
  • 完成用户相关数据表增删改查 API 接口设计
  • 完成后台管理页面的开发与渲染
    • 后台首页
    • 新增成员页
    • 成员列表页
    • 新增团队页
    • 团队列表页

注意事项

  • 数据建模合理
  • 后台页面正常渲染
  • 管理员可注册登录
  • 团队可被新增、删除、设置管理员
  • 成员可以被新增、删除、指派身份、修改密码

里程碑 3 - 简历上传及面试管理

  • 完成岗位新增
  • 完成简历新增
  • 完成面试的筛选功能和指派功能
  • 完成面试的评价流程功能
  • 完成个人后台中心的关联任务
    • 待筛选的简历
    • 待面试的简历
    • 待评价的简历

注意事项

  • 职位可被新增
  • 岗位可被新增
  • 简历可被上传
  • 简历可被指派
  • 简历可被评价
  • 简历可被删除
  • 面试可被指派
  • 简历及面试列表、详情可被展示

公司的部门及岗位

  1. 主要部门维度:技术部、产品部、运营部、人力资源部等
  2. 职位维度:技术、产品、运营、销售、行政、财务、物流、人力资源等
  3. 岗位维度:前端实习生、前端工程师、高级前端工程师、资深前端工程师、前端专家、高级前端专家等

简历的几个阶段

  1. 简历上传完毕,等待评估
  2. 简历过审,状态如下
    1. 等待电话邀约
    2. 等待安排面试
    3. 候选人联系不上
    4. 候选人拒邀

业务逻辑控制

放到 Egg/app/controller 下面

  1. const Controller = require('../core/controller')
  2. class IntervieweeController extends Controller {
  3. async list () {
  4. const {
  5. currPage = 1,
  6. pageSize = 10,
  7. funcId,
  8. jobId,
  9. // 状态,可以传入多个以 , 隔开 status=1,2,3
  10. status
  11. } = this.ctx.query
  12. const res = await this.service.interviewee.pagination({
  13. currPage,
  14. pageSize,
  15. funcId,
  16. jobId,
  17. status
  18. })
  19. if (res.errMsg) {
  20. this.fail(res)
  21. } else {
  22. this.success(res)
  23. }
  24. }
  25. /**
  26. * 评价简历,评价后,面试流程将往后推一步
  27. */
  28. async comment (ctx) {
  29. const {
  30. // 面试评语
  31. remark,
  32. // 求职者ID
  33. intervieweeId,
  34. // 第几面
  35. step
  36. } = ctx.request.body
  37. // 当前登录用户为面试官
  38. const currentUser = ctx.user
  39. if (!currentUser) {
  40. this.fail('请登录后再试')
  41. return
  42. }
  43. // 面试官ID
  44. const userId = currentUser.id
  45. const viewUserInfo = {
  46. user_id: userId,
  47. interviewee_id: intervieweeId,
  48. status: step,
  49. comment: remark
  50. }
  51. const res = await this.service.interviewee.nextProgress(viewUserInfo)
  52. this.checkRes(res)
  53. }
  54. }
  55. module.exports = IntervieweeController

MySQL 表字段定义

表结构初始化入库 - 可放到 Egg/database/migrations 下面

  1. 'use strict'
  2. const tableName = 'interviewee'
  3. module.exports = {
  4. up: async (queryInterface, Sequelize) => {
  5. const { STRING, INTEGER, DATE, NOW } = Sequelize
  6. return await queryInterface.createTable(tableName, {
  7. id: {
  8. type: INTEGER,
  9. primaryKey: true,
  10. autoIncrement: true,
  11. comment: '主键,自增'
  12. },
  13. name: {
  14. type: STRING,
  15. allowNull: false,
  16. comment: '候选人姓名'
  17. },
  18. phone: {
  19. type: STRING,
  20. allowNull: false,
  21. unique: 'intervieweePhone',
  22. is: /^1[3456789]\d{9}$/,
  23. comment: '候选人电话号码'
  24. },
  25. email: {
  26. type: STRING,
  27. allowNull: true,
  28. comment: '候选人电子邮箱',
  29. isEmail: true
  30. },
  31. address: {
  32. type: STRING,
  33. allowNull: false,
  34. comment: '候选人地址'
  35. },
  36. education: {
  37. type: STRING,
  38. allowNull: false,
  39. comment: '候选人学历,一般为 高中,大专,本科,研究生,博士,其他'
  40. },
  41. type: {
  42. type: INTEGER(1),
  43. comment: '招聘类型,0 社招,1校招,默认 0',
  44. defaultValue: 0
  45. },
  46. is_internship: {
  47. type: INTEGER(1),
  48. comment: '是否实习 0 试用 1 实习,默认 0',
  49. defaultValue: 0
  50. },
  51. job_id: {
  52. type: INTEGER,
  53. allowNull: true,
  54. comment: '推荐岗位外键'
  55. },
  56. reason: {
  57. type: STRING,
  58. allowNull: false,
  59. comment: '推荐理由'
  60. },
  61. resume_path: {
  62. type: STRING,
  63. allowNull: false,
  64. comment: '简历路径'
  65. },
  66. note: {
  67. type: STRING,
  68. allowNull: true,
  69. comment: '推荐备注'
  70. },
  71. channel: {
  72. type: INTEGER,
  73. allowNull: false,
  74. comment: '招聘渠道 0 外部 1 内部,默认 0',
  75. defaultValue: 1
  76. },
  77. status: {
  78. type: STRING,
  79. allowNull: false,
  80. comment: '面试状态,一面,二面,三面,四面,五面,六面,发放 offer,正式入职'
  81. },
  82. state: {
  83. type: INTEGER(1),
  84. defaultValue: 1,
  85. comment: '数据是否有效,0 无效,1 有效,默认 1'
  86. },
  87. is_success: {
  88. type: INTEGER(1),
  89. defaultValue: 0,
  90. comment: '是否成功 0 不成功 1 成功'
  91. },
  92. viewer_id: {
  93. type: INTEGER,
  94. allowNull: true,
  95. comment: '面试官外键'
  96. },
  97. recommender_id: {
  98. type: INTEGER,
  99. allowNull: false,
  100. comment: '内推人外键'
  101. },
  102. created_at: {
  103. type: DATE,
  104. allowNull: true,
  105. defaultValue: NOW,
  106. comment: '创建日期'
  107. },
  108. updated_at: {
  109. type: DATE,
  110. allowNull: true,
  111. defaultValue: NOW,
  112. comment: '更新日期'
  113. },
  114. deleted_at: {
  115. type: DATE,
  116. allowNull: true,
  117. comment: '删除日期'
  118. }
  119. }, {
  120. paranoid: true,
  121. deletedAt: true,
  122. underscored: true,
  123. freezeTableName: true,
  124. tableName: tableName,
  125. version: true,
  126. comment: '候选人表'
  127. })
  128. .then(() => {
  129. queryInterface.addIndex(tableName, {
  130. name: 'name_index',
  131. fields: ['name']
  132. })
  133. queryInterface.addIndex(tableName, {
  134. name: 'phone_index',
  135. fields: ['phone']
  136. })
  137. queryInterface.addIndex(tableName, {
  138. name: 'status_index',
  139. fields: ['status']
  140. })
  141. })
  142. },
  143. down: async (queryInterface, Sequelize) => {
  144. await queryInterface.dropTable(tableName)
  145. }
  146. }

表结构模型定义 - 放到 Egg/app/model 下面

  1. 'use strict'
  2. const tableName = 'interviewee'
  3. module.exports = app => {
  4. const { STRING, INTEGER } = app.Sequelize
  5. const Interviewee = app.model.define(tableName, {
  6. id: {
  7. type: INTEGER,
  8. primaryKey: true,
  9. autoIncrement: true,
  10. comment: '主键,自增'
  11. },
  12. name: {
  13. type: STRING,
  14. allowNull: false,
  15. comment: '候选人姓名'
  16. },
  17. phone: {
  18. type: STRING,
  19. allowNull: false,
  20. unique: 'intervieweePhone',
  21. is: /^1[3456789]\d{9}$/,
  22. comment: '候选人电话号码'
  23. },
  24. email: {
  25. type: STRING,
  26. allowNull: true,
  27. comment: '候选人电子邮箱',
  28. isEmail: true
  29. },
  30. address: {
  31. type: STRING,
  32. allowNull: false,
  33. comment: '候选人地址'
  34. },
  35. education: {
  36. type: STRING,
  37. allowNull: false,
  38. comment: '候选人学历,一般为 高中,大专,本科,研究生,博士,其他'
  39. },
  40. type: {
  41. type: INTEGER(1),
  42. comment: '招聘类型,0 社招,1校招,默认 0',
  43. defaultValue: 0
  44. },
  45. is_internship: {
  46. type: INTEGER(1),
  47. comment: '是否实习 0 试用 1 实习,默认 0',
  48. defaultValue: 0
  49. },
  50. job_id: {
  51. type: INTEGER,
  52. allowNull: true,
  53. comment: '推荐岗位外键'
  54. },
  55. reason: {
  56. type: STRING,
  57. allowNull: false,
  58. comment: '推荐理由'
  59. },
  60. resume_path: {
  61. type: STRING,
  62. allowNull: false,
  63. comment: '简历路径'
  64. },
  65. note: {
  66. type: STRING,
  67. allowNull: true,
  68. comment: '推荐备注'
  69. },
  70. channel: {
  71. type: INTEGER,
  72. allowNull: false,
  73. comment: '招聘渠道 0 外部 1 内部,默认 0',
  74. defaultValue: 1
  75. },
  76. status: {
  77. type: STRING,
  78. allowNull: false,
  79. comment: '面试状态,一面,二面,三面,四面,五面,六面,发放 offer,正式入职'
  80. },
  81. state: {
  82. type: INTEGER(1),
  83. defaultValue: 1,
  84. comment: '数据是否有效,0 无效,1 有效,默认 1'
  85. },
  86. is_success: {
  87. type: INTEGER(1),
  88. defaultValue: 0,
  89. comment: '是否成功 0 不成功 1 成功'
  90. },
  91. viewer_id: {
  92. type: INTEGER,
  93. allowNull: true,
  94. comment: '面试官外键'
  95. },
  96. recommender_id: {
  97. type: INTEGER,
  98. allowNull: false,
  99. comment: '内推人外键'
  100. }
  101. }, {
  102. paranoid: true,
  103. deletedAt: true,
  104. underscored: true,
  105. freezeTableName: true,
  106. tableName: tableName,
  107. version: true,
  108. comment: '候选人表',
  109. hooks: {
  110. }
  111. })
  112. Interviewee.associate = () => {
  113. Interviewee.belongsTo(app.model.Job, {
  114. as: 'job',
  115. foreignKey: 'job_id',
  116. constraints: false
  117. })
  118. Interviewee.belongsTo(app.model.Users, {
  119. as: 'viewr',
  120. foreignKey: 'viewer_id',
  121. constraints: false
  122. })
  123. Interviewee.belongsTo(app.model.Users, {
  124. as: 'recommender',
  125. foreignKey: 'recommender_id',
  126. constraints: false
  127. })
  128. }
  129. Interviewee.sync({ alter: true })
  130. return Interviewee
  131. }

接口能力实现

controller 能力扩展说明 —- 放到 Egg/app/core/controller.js 里

  1. 'use strict'
  2. const Controller = require('egg').Controller
  3. class BaseController extends Controller {
  4. validate () {
  5. const { ctx } = this
  6. ctx.validate(this.getResRule(ctx.url))
  7. }
  8. // 校验用户权限
  9. checkUseAuth (type) {
  10. return this.ctx.checkUseAuth(type)
  11. }
  12. // set success data
  13. success (data) {
  14. this.ctx.success(data)
  15. }
  16. successMsg (msg) {
  17. this.ctx.successMsg(msg)
  18. }
  19. // set failed msg
  20. fail (error) {
  21. this.ctx.fail(error)
  22. }
  23. failMsg (msg) {
  24. this.ctx.failMsg(msg)
  25. }
  26. // get params
  27. getParams (key) {
  28. return this.ctx.getParams(key)
  29. }
  30. // get request client ip
  31. get ip () {
  32. return this.ctx.ip
  33. }
  34. // get cookies
  35. get cookies () {
  36. return this.ctx.cookies
  37. }
  38. // get session
  39. get session () {
  40. return this.ctx.session
  41. }
  42. validatePk () {
  43. this.ctx.validate({
  44. id: { type: 'int', convertType: 'number', required: true }
  45. }, this.ctx.params)
  46. }
  47. validatePageParam (type = 'query') {
  48. const checkPath = type === 'body'? 'body' : 'query'
  49. this.ctx.validate(this.ctx.rule.getPageRequest,this.ctx.request[checkPath])
  50. }
  51. formatOptions (obj) {
  52. const badParamList = [undefined,'', null];
  53. for (const name in obj) {
  54. badParamList.includes(obj[name]) && delete obj[name]
  55. }
  56. return obj
  57. }
  58. async find (id) {
  59. const table = await this.service[this.serviceName].findByPk(id, {}, true)
  60. if (!table) {
  61. this.failMsg(`${this.modelName} 表内无标识为 ${id} 的记录`)
  62. }
  63. return table
  64. }
  65. async destroy (needTip = true) {
  66. const ctx = this.ctx
  67. const id = ctx.params.id
  68. const table = await this.find(id)
  69. if (table) {
  70. await table.destroy()
  71. needTip && this.successMsg(`删除${id}成功`)
  72. return true
  73. }
  74. }
  75. // 封装统一的调用检查函数,可以在查询、创建和更新等 Service 中复用
  76. checkSuccess (result) {
  77. const DEAULT_REQ_STATUS_ATTR = this.ctx.getAttr('DEAULT_REQ_STATUS_ATTR')
  78. const DEAULT_REQ_MSG_ATTR = this.ctx.getAttr('DEAULT_REQ_MSG_ATTR')
  79. if (!result[DEAULT_REQ_STATUS_ATTR]) {
  80. this.ctx.failMsg(result[DEAULT_REQ_MSG_ATTR])
  81. } else {
  82. this.ctx.successMsg(result[DEAULT_REQ_MSG_ATTR])
  83. }
  84. }
  85. checkFindData (table, errMsg = '记录未找到') {
  86. if (!table) {
  87. this.failMsg(errMsg)
  88. }
  89. this.success(table)
  90. }
  91. }
  92. module.exports = BaseController

controller 能力消费 —- 放到 Egg/app/controller 里

  1. 'use strict'
  2. const BaseController = require('../core/controller')
  3. const currServiceName = 'sysDicItems'
  4. /**
  5. * @Controller
  6. */
  7. class SysDicItemsController extends BaseController {
  8. get modelName() {
  9. return 'SysDicItems'
  10. }
  11. get serviceName() {
  12. return currServiceName
  13. }
  14. /**
  15. * @Summary 获取指定分组的所有字典项
  16. * @Description 获取指定分组的所有字典项
  17. * @Router get /sysDicItems/group/{groupName}
  18. * @Request path string groupName 分组名
  19. * @Response 200 SysDicItems 获取指定分组的所有字典项
  20. */
  21. async group() {
  22. const { ctx } = this
  23. const { params } = ctx
  24. const { groupName } = params
  25. ctx.validate(ctx.rule.getSysDicGroupRequest)
  26. const res = await this.service[currServiceName].group(groupName)
  27. res && this.success(res)
  28. }
  29. /**
  30. * @Summary 获取所有字典项,不分页
  31. * @Description 获取所有字典项
  32. * @Router get /sysDicItems
  33. * @Response 200 SysDicItems 获取所有字典项成功
  34. */
  35. async index() {
  36. this.success(await this.service[currServiceName].list())
  37. }
  38. /**
  39. * @Summary 分页获取字典项
  40. * @Description 分页获取字典项
  41. * @Router post /sysDicItems/page/
  42. * @Request query integer pageNo 页面数
  43. * @Request query integer pageSize 页面条数
  44. * @Response 200 SysDicItems 获取所有字典项成功
  45. */
  46. async page() {
  47. this.validatePageParam()
  48. const { ctx } = this
  49. const {
  50. // 默认第一页
  51. pageNo = 1,
  52. // 默认每页10条记录
  53. pageSize = 10,
  54. ...restParams
  55. } = ctx.request.body
  56. const where = this.formatOptions({
  57. ...restParams,
  58. })
  59. ctx.validate(ctx.rule.updateSysDicRequest)
  60. const config = { pageNo, pageSize, where }
  61. const res = await this.service[currServiceName].pagination(config)
  62. if (res.count) {
  63. this.success(res)
  64. } else {
  65. this.failMsg(`未找到第 ${pageNo} 页的字典项信息`)
  66. }
  67. }
  68. /**
  69. * @Summary 新增字典项
  70. * @Description 新增字典项信息
  71. * @Router post /sysDicItems
  72. * @Request body createSysDicRequest *body
  73. * @Response 200 defaultBaseResponse 新增字典项成功
  74. */
  75. async create() {
  76. const ctx = this.ctx
  77. ctx.validate(ctx.rule.createSysDicRequest)
  78. const { name, value, note, groupName, sortNum, enable } = ctx.request.body
  79. const res = await this.service[currServiceName].create({ name, value, note, groupName, sortNum, enable })
  80. this.success({ id: res.id })
  81. }
  82. /**
  83. * @Summary 获取所有字典项
  84. * @Description 根据 id 获取字典项信息
  85. * @Router get /sysDicItems/{id}
  86. * @Request path integer id 标识
  87. * @Response 200 SysDicItems 获取指定字典项信息成功
  88. */
  89. async show() {
  90. const ctx = this.ctx
  91. this.validatePk()
  92. const res = await this.find(ctx.params.id)
  93. res && this.success(res)
  94. }
  95. /**
  96. * @Summary 查找字典项
  97. * @Description 查找字典项信息
  98. * @Router post /sysDicItems/find
  99. * @Request body findSysDicRequest *body
  100. * @Response 200 SysDicItems 查找字典项成功
  101. */
  102. async findByParam() {
  103. const { ctx } = this
  104. const { name, value, note, groupName, sortNum, enable } = ctx.request.body
  105. // 参数校验
  106. ctx.validate(ctx.rule.findSysDicRequest, ctx.request.body)
  107. const where = this.formatOptions({ name, value, note, groupName, sortNum, enable })
  108. // 查找字典项数据
  109. const res = await ctx.service[currServiceName].list({ where })
  110. return res ? this.success(res) : this.failMsg(`已存在字典项 ${name}`)
  111. }
  112. /**
  113. * @Summary 修改指定字典项
  114. * @Description 根据 id 修改字典项信息
  115. * @Router put /sysDicItems/{id}
  116. * @Request path integer id 标识
  117. * @Request body createSysDicRequest *body
  118. * @Response 200 defaultBaseResponse 修改指定字典项成功
  119. */
  120. async update() {
  121. const ctx = this.ctx
  122. this.validatePk()
  123. const id = ctx.params.id
  124. const SysDicItems = await this.find(id)
  125. if (SysDicItems) {
  126. ctx.validate(ctx.rule.createSysDicRequest)
  127. const { name, value, note, groupName, sortNum, enable } = ctx.request.body
  128. await this.service[currServiceName].modelUpdateData({
  129. model: SysDicItems, data: { name, value, note, groupName, sortNum, enable }
  130. })
  131. this.success({ id })
  132. }
  133. }
  134. /**
  135. * @Summary 删除指定字典项
  136. * @Description 根据 id 删除字典项信息
  137. * @Router delete /sysDicItems/{id}
  138. * @Request path integer id 标识,可以用逗号隔开
  139. * @Response 200 defaultResponse 删除指定字典项成功
  140. */
  141. async destroy() {
  142. const id = this.ctx.params.id
  143. const state = await this.service[currServiceName].destroy({ where: { id: id.split(',') } })
  144. state && this.successMsg(`删除字典项${this.ctx.params.id}成功`)
  145. }
  146. }
  147. module.exports = SysDicItemsController

获取数据服务

services 能力扩展说明 —- 放到services.js 里

  1. const Service = require('egg').Service
  2. const dayjs = require('dayjs')
  3. const humps = require('humps')
  4. const _symbolJudge = (key, convert) => {
  5. if (typeof key === 'symbol') {
  6. return key
  7. } else {
  8. return convert(key)
  9. }
  10. }
  11. function toInt(str) {
  12. if (typeof str === 'number') return str
  13. if (!str) return str
  14. return parseInt(str, 10) || 0
  15. }
  16. class BaseService extends Service {
  17. get DEAULT_STATUS_LIST() {
  18. return this.ctx.getAttr('DEAULT_STATUS_LIST')
  19. }
  20. get DEAULT_REQ_STATUS_ATTR() {
  21. return this.ctx.getAttr('DEAULT_REQ_STATUS_ATTR')
  22. }
  23. get DEAULT_REQ_MSG_ATTR() {
  24. return this.ctx.getAttr('DEAULT_REQ_MSG_ATTR')
  25. }
  26. get DEAULT_REQ_DATA_ATTR() {
  27. return this.ctx.getAttr('DEAULT_REQ_DATA_ATTR')
  28. }
  29. get model() {
  30. return this.ctx.model
  31. }
  32. get _user() {
  33. return this.ctx.user
  34. }
  35. // 格式化数据库返回的列表数据,将之从下划线转为驼峰
  36. formatDBDataToCamelize(data) {
  37. if (data && data.length > 0) {
  38. return data.map((item) => humps.camelizeKeys(item.dataValues))
  39. } else if (data && data.dataValues) {
  40. return humps.camelizeKeys(data.dataValues)
  41. }
  42. return data
  43. }
  44. // 格式化下划线为驼峰
  45. formatDataToCamelize(data) {
  46. if (data && data.length > 0) {
  47. return data.map((item) => humps.camelizeKeys(item))
  48. } else if (data) {
  49. return humps.camelizeKeys(data)
  50. }
  51. return data
  52. }
  53. // 格式化驼峰参数为下划线参数,便于查询
  54. foramtToUnderscoredParams(data) {
  55. if (data && data.length > 0) {
  56. return data.map((item) => humps.decamelizeKeys(item, _symbolJudge))
  57. } else if (data && data) {
  58. return humps.decamelizeKeys(data, _symbolJudge)
  59. }
  60. return data
  61. }
  62. formatDate(time) {
  63. return dayjs(time).format('YYYY-MM-DD HH:mm:ss')
  64. }
  65. formatDay(time) {
  66. return dayjs(time).format('YYYY-MM-DD')
  67. }
  68. formatTimestampsField(table, callBackFn = (item) => item) {
  69. if (!table) {
  70. return table
  71. }
  72. const CREATE_ATTR = 'createdAt'
  73. const UPDATE_ATTR = 'updatedAt'
  74. const DELETE_ATTR = 'deletedAt'
  75. table[CREATE_ATTR] && (table[CREATE_ATTR] = this.formatDate(table[CREATE_ATTR]))
  76. table[UPDATE_ATTR] && (table[UPDATE_ATTR] = this.formatDate(table[UPDATE_ATTR]))
  77. DELETE_ATTR in table && (table[DELETE_ATTR] = undefined)
  78. return callBackFn(table)
  79. }
  80. dealWithTableField(table, callBackFn) {
  81. return table.dataValues && this.formatTimestampsField(this.formatDataToCamelize(table.dataValues), callBackFn)
  82. }
  83. async dealDBFn(callback, errHandler) {
  84. try {
  85. return await callback()
  86. } catch (e) {
  87. errHandler && await errHandler()
  88. this.ctx.throw(e.errors ? e.errors[0]?.message : e.message)
  89. }
  90. }
  91. async updateData({ modelName, data, configOption }) {
  92. const _t = this
  93. return await _t.dealDBFn(async () => {
  94. const entity = _t.foramtToUnderscoredParams(data)
  95. return _t.dealWithTableField(await _t.model[modelName].update(entity, configOption))
  96. })
  97. }
  98. async modelUpdateData({ model, data, configOption }) {
  99. const _t = this
  100. return await _t.dealDBFn(async () => {
  101. const entity = _t.foramtToUnderscoredParams(data)
  102. return _t.dealWithTableField(await model.update(entity, configOption))
  103. })
  104. }
  105. async createData({ modelName, data, configOption, callBackFn }) {
  106. const _t = this
  107. return await _t.dealDBFn(async () => {
  108. const entity = _t.foramtToUnderscoredParams(data)
  109. const res = await _t.model[modelName].create(entity, configOption)
  110. return res ? _t.dealWithTableField(res, callBackFn) : res
  111. })
  112. }
  113. async findAll({ modelName = '', configOption = {}, callBackFn }) {
  114. const _t = this
  115. return await _t.dealDBFn(async () => {
  116. const res = await _t.model[modelName].findAll(configOption)
  117. return res ? res.map(item => _t.dealWithTableField(item, callBackFn)) : res
  118. })
  119. }
  120. async findByPk({ modelName = '', id = '', configOption = {}, needModal = false, callBackFn }) {
  121. const _t = this
  122. return await _t.dealDBFn(async () => {
  123. const res = await _t.model[modelName].findByPk(toInt(id), configOption)
  124. res && (res.dataValues = _t.dealWithTableField(res, callBackFn))
  125. return needModal ? res : res.dataValues
  126. })
  127. }
  128. async findOne({ modelName = '', configOption = {}, needModal = false, callBackFn }) {
  129. const _t = this
  130. return await _t.dealDBFn(async () => {
  131. const res = await _t.model[modelName].findOne(configOption)
  132. res && (res.dataValues = _t.dealWithTableField(res, callBackFn))
  133. return needModal ? res : res && res.dataValues
  134. })
  135. }
  136. async findOrCreate({ modelName = '', configOption = {}, callBackFn }) {
  137. const _t = this
  138. return await _t.dealDBFn(async () => {
  139. const res = await _t.model[modelName].findOrCreate(configOption)
  140. return res ? _t.dealWithTableField(res, callBackFn) : res
  141. })
  142. }
  143. async destroyData({ modelName = '', configOption = {} }) {
  144. const _t = this
  145. return await _t.dealDBFn(async () => {
  146. await _t.model[modelName].destroy(configOption)
  147. return true
  148. })
  149. }
  150. async pagination({ modelName = '', configOption = {}, needTranUnder = true ,callBackFn }) {
  151. const _t = this
  152. const { attributes, include, order } = configOption
  153. let { pageNo, pageSize, where } = configOption
  154. pageNo = toInt(pageNo)
  155. pageSize = toInt(pageSize)
  156. where && needTranUnder && (where = _t.foramtToUnderscoredParams(where))
  157. return await _t.dealDBFn(async () => {
  158. const options = {
  159. attributes,
  160. include,
  161. where,
  162. order: order || [
  163. ['updated_at', 'DESC']
  164. ],
  165. offset: (pageNo - 1) * pageSize,
  166. limit: toInt(pageSize)
  167. }
  168. const res = await _t.model[modelName].findAndCountAll(options)
  169. const data = res.count > 0 ? res.rows.map(item => _t.dealWithTableField(item, callBackFn)) : []
  170. return { data, count: res.count }
  171. })
  172. }
  173. async findAndCountAll({ modelName = '', configOption = {}, callBackFn }) {
  174. const _t = this
  175. return await _t.dealDBFn(async () => {
  176. const res = await _t.model[modelName].findAndCountAll(configOption)
  177. return res ? res.map(item => _t.dealWithTableField(item, callBackFn)) : res
  178. })
  179. }
  180. // set success data
  181. success(msg, data) {
  182. return this.ctx.successMsgResponse(msg, data)
  183. }
  184. // set failed msg
  185. fail(error) {
  186. return this.ctx.errorMsgResponse(error)
  187. }
  188. }
  189. module.exports = BaseService

services 能力消费举例 —- 放到 Egg/app/service/ 里

  1. const Service = require('egg').BaseService
  2. const fs = require('fs')
  3. const path = require('path')
  4. const currModelName = 'Interviewee'
  5. /**
  6. * 候选人表 Service
  7. */
  8. class Interviewee extends Service {
  9. get configOption() {
  10. return {
  11. include: [
  12. {
  13. as: 'recommender',
  14. model: this.model.Users,
  15. attributes: ['name']
  16. },
  17. {
  18. as: 'job',
  19. model: this.model.Job,
  20. attributes: ['name']
  21. }
  22. ],
  23. where: this.getAuthWhere()
  24. }
  25. }
  26. get callBackFn() {
  27. return (item) => {
  28. item.jobName = item.job && item.job.dataValues.name
  29. item.job = undefined
  30. item.recommenderName = item.recommender && item.recommender.dataValues.name
  31. item.recommender = undefined
  32. return item
  33. }
  34. }
  35. getAuthWhere() {
  36. return this.ctx.checkUseAuth(0, false) ? {} : { recommender_id: this._user.id }
  37. }
  38. async list() {
  39. return await super.findAll({ modelName: currModelName, configOption: this.configOption, callBackFn: this.callBackFn })
  40. }
  41. async findByPk(id, configOption, needModal) {
  42. configOption = { ...this.configOption, ...configOption }
  43. configOption.where.id = id
  44. this.ctx.checkUseAuth(1) && delete configOption.where.recommender_id
  45. return await super.findOne({ modelName: currModelName, configOption, needModal, callBackFn: this.callBackFn })
  46. }
  47. async create(intervieweeInfo, file) {
  48. return await this.dealDBFn(async () => {
  49. const filename = `${intervieweeInfo.name}-${intervieweeInfo.phone}`
  50. const uploadRes = await this.uploadPdf(this.ctx.origin, file, filename)
  51. intervieweeInfo.resumePath = uploadRes.resumePath
  52. // 新增候选人记录
  53. const currUser = this._user || {}
  54. intervieweeInfo.recommenderId = currUser.id || 1
  55. return await this.createData({ modelName: currModelName, data: intervieweeInfo })
  56. })
  57. }
  58. async update(data, configOption, needJoin = true) {
  59. needJoin && (configOption = { ...this.configOption, ...configOption })
  60. let callBackFn = ''
  61. needJoin && (callBackFn = this.callBackFn)
  62. return await this.updateData({ modelName: currModelName, data, configOption, callBackFn })
  63. }
  64. async modelUpdate(model, data, configOption = {}) {
  65. let transaction
  66. return await this.dealDBFn(async () => {
  67. transaction = await this.model.transaction();
  68. configOption.transaction = transaction;
  69. const res = await super.modelUpdateData({ model, data, configOption })
  70. // 维护推荐记录
  71. recommendRecordInfo.id = res.recommenderId
  72. recommendRecordInfo.intervieweeName = res.name
  73. recommendRecordInfo.intervieweeTel = res.phone
  74. recommendRecordInfo.intervieweeType = res.type
  75. await this.service.recommendRecord.update(recommendRecordInfo, { transaction }, false)
  76. await transaction.commit()
  77. return res
  78. }, () => transaction.rollback())
  79. }
  80. async destroy() {
  81. let transaction
  82. return await this.dealDBFn(async () => {
  83. transaction = await this.model.transaction()
  84. const id = this.ctx.params.id
  85. let recommenderId = ''
  86. this.ctx.checkUseAuth(0) && (recommenderId = this._user.id)
  87. const configOption = {
  88. where: { id, recommender_id: recommenderId },
  89. transaction
  90. }
  91. // 查找候选人记录
  92. let res = await this.exist(configOption.where)
  93. if (!res) {
  94. return this.ctx.failMsg(`${currModelName} 表内无标识为 ${id} 的记录`)
  95. }
  96. // 删除候选人记录
  97. res = await this.destroyData({ modelName: currModelName, configOption })
  98. // 删除推荐记录
  99. const recommendRecordWhere = {
  100. interviewee_id: id,
  101. recommender_user_id: recommenderId
  102. }
  103. await this.service.recommendRecord.destroy({ transaction, where: recommendRecordWhere })
  104. await transaction.commit()
  105. return res
  106. }, () => transaction.rollback())
  107. }
  108. /**
  109. * 上传简历
  110. * @returns {string} res 上传路径
  111. */
  112. async uploadPdf(origin, stream, filename) {
  113. const resumeFolderPath = `public/${this.formatDay()}`
  114. const saveFolder = `app/${resumeFolderPath}`
  115. const resumePath = `${resumeFolderPath}/${filename}.pdf`
  116. // 如果文件夹不存在,则直接创建
  117. if (!fs.existsSync(saveFolder)) {
  118. fs.mkdirSync(saveFolder)
  119. }
  120. const writerStream = fs.createWriteStream(path.join(this.config.baseDir, `${saveFolder}/${filename}.pdf`))
  121. await stream.pipe(writerStream)
  122. return { resumePath, uploadPath: `${origin}${resumePath}` }
  123. }
  124. async pagination({ where, ...otherConfig }) {
  125. const _t = this
  126. const { Op } = this.app.Sequelize
  127. const { createdAt, } = where;
  128. where = _t.foramtToUnderscoredParams({ ...where, ..._t.configOption.where })
  129. createdAt && (delete where.created_at) && createdAt.length && (where.created_at = {
  130. [Op.lt]: new Date(createdAt[1]),
  131. [Op.gt]: new Date(createdAt[0]),
  132. });
  133. // [Op.like]: '%hat',
  134. return await super.pagination({ modelName: currModelName, configOption: { ...otherConfig, ..._t.configOption, where }, needTranUnder: false, callBackFn: this.callBackFn })
  135. }
  136. async findAllByCondtion (configOption, needJoin = true) {
  137. needJoin && (configOption = { ...this.configOption, ...configOption })
  138. let callBackFn = ''
  139. needJoin && (callBackFn = this.callBackFn)
  140. return await this.findAll({ modelName: currModelName, configOption, callBackFn })
  141. }
  142. /**
  143. * 判断简历是否存在
  144. * @return true 表示已存在,false 不存在
  145. */
  146. async exist(where) {
  147. // 判断是否已存在
  148. const configOption = {
  149. attributes: ['id'],
  150. where: this.foramtToUnderscoredParams(where)
  151. }
  152. return !!await super.findOne({ modelName: currModelName, configOption })
  153. }
  154. }
  155. module.exports = Interviewee

路由文件配置

路由约束说明 —- 放到 Egg/app/router.js 配置

  1. const { readdirSync } = require('fs')
  2. const { extname, resolve } = require('path')
  3. module.exports = app => {
  4. const checkFileType = (file) => extname(file) === '.js'
  5. const BASE_ROUTER = resolve(__dirname, './routers')
  6. readdirSync(BASE_ROUTER, 'utf-8')
  7. .filter(checkFileType)
  8. .map(file => {
  9. console.log(`[${app.config.projectName}] insert router ${file}`)
  10. require(resolve(BASE_ROUTER, file))(app)
  11. })
  12. }

路由能力注册举例说明 —- 放到 Egg/app/routers/ 下面

  1. module.exports = app => {
  2. const { router, controller } = app
  3. router.post('/interviewee/upload', controller.interviewee.upload)
  4. router.post('/interviewee/page/:param?', controller.interviewee.page)
  5. router.resources('Interviewee', '/interviewee', controller.interviewee)
  6. // 获取所有职能(id, name)列表,不分页
  7. // GET /interviewee app.controllers.interviewee.index
  8. // GET /interviewee/:id app.controllers.interviewee.show
  9. // PUT /interviewee/:id app.controllers.interviewee.update
  10. // DELETE /interviewee/:id app.controllers.interviewee.destroy
  11. }