├── public ├── images │ ├── b2.jpeg │ ├── bg1.jpeg │ ├── 404.png │ ├── arrow.png │ ├── logo.png │ ├── step1.png │ ├── step2.png │ ├── step3.png │ ├── step4.png │ ├── wxpay.png │ ├── favicon.ico │ ├── success1.png │ ├── success2.png │ ├── zhifubao.png │ ├── renrenpay.ico │ ├── partner_wxpay.jpg │ └── partner_alipay.jpg ├── dist │ ├── static │ │ ├── css │ │ │ ├── chunk-f779.79af9e6e.css │ │ │ ├── chunk-5743.8cad7bcd.css │ │ │ ├── chunk-b69e.de0b32b4.css │ │ │ ├── chunk-27db.6a1661a5.css │ │ │ ├── chunk-0224.f01dfcab.css │ │ │ ├── chunk-26f3.a8b3c6c5.css │ │ │ ├── chunk-fc3b.b79580c9.css │ │ │ ├── chunk-3200.6e36a35c.css │ │ │ ├── chunk-7e70.92d696e8.css │ │ │ ├── chunk-280b.7469e992.css │ │ │ ├── chunk-libs.e73a8678.css │ │ │ ├── chunk-6929.486b8768.css │ │ │ └── app.cab2c720.css │ │ ├── img │ │ │ ├── 404.a57b6f3.png │ │ │ └── bg.cf0887a.jpg │ │ ├── fonts │ │ │ └── element-icons.6f0a763.ttf │ │ └── js │ │ │ ├── eej3.42f74184.js │ │ │ ├── EwA2.cf227d21.js │ │ │ ├── chunk-27db.8a51fb19.js │ │ │ ├── chunk-f779.74ec5683.js │ │ │ ├── chunk-54db.2bca911d.js │ │ │ ├── chunk-0224.e4d32421.js │ │ │ ├── chunk-5743.e739dad9.js │ │ │ ├── chunk-3200.30d05b08.js │ │ │ ├── chunk-7e70.17b6490e.js │ │ │ ├── chunk-280b.82d6890b.js │ │ │ ├── chunk-6929.2461bbf3.js │ │ │ └── chunk-fc3b.827b4c52.js │ ├── favicon.ico │ └── index.html ├── stylesheets │ └── style.css ├── css │ └── normalize.css └── font │ └── iconfont.js ├── dump.rdb ├── playground.js ├── utils ├── md5.js ├── fileIO.js ├── logger.js ├── jwt.js ├── sendSms.js ├── sendEmail.js ├── callback.js ├── verifyParams.js ├── errorCode.js ├── verifyToken.js ├── websocket.js ├── redisClient.js ├── errorHandler.js ├── timeformatter.js ├── returnJson.js └── statistic.js ├── keys ├── master-public.pem └── master-privatekey.pem ├── test.js ├── .gitignore ├── README.md ├── db ├── index.js ├── models │ ├── withdraws.js │ ├── logs.js │ ├── statistic.js │ ├── payCode.js │ ├── trades.js │ ├── deposits.js │ ├── paychecks.js │ ├── orders.js │ ├── apps.js │ └── users.js ├── schema │ ├── logs.js │ ├── paycode.js │ ├── withdraws.js │ ├── trades.js │ ├── deposits.js │ ├── paychecks.js │ ├── apps.js │ ├── orders.js │ ├── statistic.js │ └── users.js └── sqlTable.js ├── conf └── default.js ├── views └── 404.ejs ├── package.json ├── app.js ├── bin └── www ├── init └── index.js └── controllers ├── withdrawsController.js ├── ordersController.js ├── index.js ├── paychecksController.js └── tradeController.js /public/images/b2.jpeg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /public/images/bg1.jpeg: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-f779.79af9e6e.css: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /dump.rdb: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/dump.rdb -------------------------------------------------------------------------------- /public/dist/static/css/chunk-5743.8cad7bcd.css: -------------------------------------------------------------------------------- 1 | .orders-index-page[data-v-07355ba6]{padding:32px} -------------------------------------------------------------------------------- /public/images/404.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/404.png -------------------------------------------------------------------------------- /public/dist/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/dist/favicon.ico -------------------------------------------------------------------------------- /public/images/arrow.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/arrow.png -------------------------------------------------------------------------------- /public/images/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/logo.png -------------------------------------------------------------------------------- /public/images/step1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/step1.png -------------------------------------------------------------------------------- /public/images/step2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/step2.png -------------------------------------------------------------------------------- /public/images/step3.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/step3.png -------------------------------------------------------------------------------- /public/images/step4.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/step4.png -------------------------------------------------------------------------------- /public/images/wxpay.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/wxpay.png -------------------------------------------------------------------------------- /playground.js: -------------------------------------------------------------------------------- 1 | 2 | let a = "pay" + 100 + "wechat" + 2 + Math.random().toString().slice(-8); 3 | console.log(a) -------------------------------------------------------------------------------- /public/images/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/favicon.ico -------------------------------------------------------------------------------- /public/images/success1.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/success1.png -------------------------------------------------------------------------------- /public/images/success2.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/success2.png -------------------------------------------------------------------------------- /public/images/zhifubao.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/zhifubao.png -------------------------------------------------------------------------------- /public/images/renrenpay.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/renrenpay.ico -------------------------------------------------------------------------------- /public/images/partner_wxpay.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/partner_wxpay.jpg -------------------------------------------------------------------------------- /public/images/partner_alipay.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/images/partner_alipay.jpg -------------------------------------------------------------------------------- /public/dist/static/img/404.a57b6f3.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/dist/static/img/404.a57b6f3.png -------------------------------------------------------------------------------- /public/dist/static/img/bg.cf0887a.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/dist/static/img/bg.cf0887a.jpg -------------------------------------------------------------------------------- /public/dist/static/fonts/element-icons.6f0a763.ttf: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/verylove/renrenpay-server/HEAD/public/dist/static/fonts/element-icons.6f0a763.ttf -------------------------------------------------------------------------------- /public/stylesheets/style.css: -------------------------------------------------------------------------------- 1 | body { 2 | padding: 50px; 3 | font: 14px "Lucida Grande", Helvetica, Arial, sans-serif; 4 | } 5 | 6 | a { 7 | color: #00B7FF; 8 | } 9 | -------------------------------------------------------------------------------- /utils/md5.js: -------------------------------------------------------------------------------- 1 | const crypto = require('crypto') 2 | 3 | const MD5 = data => { 4 | const md5 = crypto.createHash('md5') 5 | md5.update(data) 6 | return md5.digest('hex') 7 | } 8 | 9 | module.exports = MD5 10 | -------------------------------------------------------------------------------- /keys/master-public.pem: -------------------------------------------------------------------------------- 1 | -----BEGIN PUBLIC KEY----- 2 | MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDKx57/DKGMVaPrCm9dewigq/uV 3 | 3/LvmD8gA9OkcG3Tfd/V4yrLqqBH3eH0Qkfi3SbYo2mo6sFYhvZI1Cg0jlPUwT1m 4 | NedGDNJsEugXujeNMDAEc2xpjwCvuAO7XGWUmyEynGbS+t4APX1vR0eqZ1XtVDFc 5 | WD+Si5EoLcJ+9zcSbQIDAQAB 6 | -----END PUBLIC KEY----- 7 | -------------------------------------------------------------------------------- /test.js: -------------------------------------------------------------------------------- 1 | let crypto = require('crypto'); 2 | const md5 = info => { 3 | const md5 = crypto.createHash('md5') 4 | md5.update(info) 5 | return md5.digest('hex') 6 | } 7 | 8 | // console.log(md5("qwerty")) 9 | let dbInfo = { 10 | }; 11 | const pwd = md5(md5(dbInfo.password) + dbInfo.salt); 12 | console.log(pwd) -------------------------------------------------------------------------------- /public/dist/static/js/eej3.42f74184.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["eej3"],{eej3:function(e,n,t){"use strict";t.r(n);var i=t("KHd+"),s=Object(i.a)({},function(){var e=this.$createElement;return(this._self._c||e)("div",{staticClass:"api-index-page"},[this._v("\n API 文档\n")])},[],!1,null,null,null);s.options.__file="Index.vue";n.default=s.exports}}]); -------------------------------------------------------------------------------- /public/dist/static/js/EwA2.cf227d21.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["EwA2"],{EwA2:function(n,e,t){"use strict";t.r(e);var s=t("KHd+"),i=Object(s.a)({},function(){var n=this.$createElement;return(this._self._c||n)("div",{staticClass:"download-index-page"},[this._v("\n 软件下载页面\n")])},[],!1,null,null,null);i.options.__file="Index.vue";e.default=i.exports}}]); -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.db 2 | dapps/ 3 | .DS_Store 4 | .ed25519-node 5 | .idea/ 6 | *.log 7 | __MACOSX/ 8 | nodejs/ 9 | node_modules 10 | logs 11 | .project 12 | release 13 | sqlite/ 14 | ssl/ 15 | stacktrace* 16 | tmp 17 | *.pid 18 | package-lock.json 19 | .vs/ 20 | .vscode 21 | public/bower_components 22 | public/npm-debug.log 23 | public/.sass-cache 24 | public/.tmp 25 | public/.gulp 26 | public/*.css 27 | public/node_modules 28 | *.save 29 | accountKey.json 30 | 31 | -------------------------------------------------------------------------------- /utils/fileIO.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs') 2 | const path = require('path') 3 | const { 4 | loggerHttp 5 | } = require('./logger.js') 6 | 7 | const fileIO = { 8 | // write into a file 9 | async writeFile(filename, data) { 10 | try { 11 | fs.writeFileSync(path.join(__dirname, filename), data) 12 | return true 13 | } catch (err) { 14 | loggerHttp.error(`writeFile -- ${err}`) 15 | } 16 | } 17 | } 18 | 19 | module.exports = fileIO 20 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # 利用xposed框架监听支付宝微信的收付款服务端 2 | 安卓端地址:https://github.com/cxyxxx0924/renrenpay-android.git 3 | ## 说明 4 | 服务端和客户端需要配合使用,不然客户端都无法打开 5 | 6 | ## 安装步骤 7 | ``` 8 | 下载node环境 http://nodejs.cn/ (版本最好新一点,我使用的是10.14.2) 9 | 10 | git clone https://github.com/cxyxxx0924/renrenpay-server.git 11 | cd renrenpay-server 12 | npm i 13 | 下载redis 14 | 下载mysql 15 | 修改./conf/default.js下的user和password为mysql的用户名和密码 16 | 添加 数据库:db_renrenpay 17 | 运行 bin/www 既可以运行项目 18 | 打开浏览器 127.0.0.1:3000 既可以看到项目运行 19 | 确保运行后在打开安卓客户端 20 | ``` 21 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-27db.8a51fb19.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-27db"],{mS28:function(e,n,t){"use strict";var i=t("z+hT");t.n(i).a},r2dA:function(e,n,t){"use strict";t.r(n);var i={components:{Membership:t("I8ep").a}},s=(t("mS28"),t("KHd+")),c=Object(s.a)(i,function(){var e=this.$createElement,n=this._self._c||e;return n("div",{staticClass:"admin-member-index"},[n("Membership",{attrs:{mode:"edit"}})],1)},[],!1,null,"51c447f2",null);c.options.__file="Index.vue";n.default=c.exports},"z+hT":function(e,n,t){}}]); -------------------------------------------------------------------------------- /public/dist/static/css/chunk-b69e.de0b32b4.css: -------------------------------------------------------------------------------- 1 | .pay-comp[data-v-549a193a]{text-align:center}.pay-comp.wepay .deposit-title[data-v-549a193a]{color:#09bb07}.pay-comp.alipay .deposit-title[data-v-549a193a]{color:#00aae8}.pay-comp .deposit-title[data-v-549a193a]{font-size:24px}.pay-comp .deposit-num[data-v-549a193a]{font-size:40px;font-weight:700;margin-top:30px;color:#606266}.app-index-page[data-v-00346527]{padding:32px}.app-index-page .input-hint[data-v-00346527]{font-size:12px;line-height:1.7;color:#909399;margin-top:5px}.app-index-page .input-hint a[data-v-00346527]{text-decoration:underline} -------------------------------------------------------------------------------- /db/index.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const SqlTable = require('./sqlTable.js') 3 | const config = require('../conf/default.js') 4 | 5 | let sequelize, sqlTable 6 | 7 | async function init() { 8 | sequelize = new Sequelize(config.db.db, config.db.user, config.db.password, { 9 | host: config.db.host, 10 | port: config.db.port, 11 | dialect: config.db.dialect, 12 | pool: { 13 | max: 100, 14 | min: 0, 15 | acquire: 30000, 16 | idle: 10000, 17 | }, 18 | timezone: '+08:00' 19 | }) 20 | // init 21 | sqlTable = new SqlTable(Sequelize, sequelize) 22 | } 23 | 24 | init() 25 | module.exports = {Sequelize, sequelize, sqlTable} 26 | -------------------------------------------------------------------------------- /db/models/withdraws.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const withdraws = { 7 | async addWithdraw(data) { 8 | return sqlTable.withdraws.create({ 9 | userid: data.userid, 10 | uuid: data.uuid, 11 | type: data.type || '', 12 | amount: data.amount || 0, 13 | memo: data.memo || '', 14 | status: 1 15 | }) 16 | }, 17 | 18 | async updateWithdrawInfo(data) { 19 | return sqlTable.withdraws.update({ 20 | status: data.status || 0 21 | }, { 22 | where: { 23 | id: data.userid || 0 24 | } 25 | }) 26 | }, 27 | } 28 | 29 | module.exports = withdraws 30 | -------------------------------------------------------------------------------- /utils/logger.js: -------------------------------------------------------------------------------- 1 | const log4js = require('log4js') 2 | 3 | log4js.configure({ 4 | appenders: { 5 | out: { type: 'stdout' }, 6 | httpError: { type: 'dateFile', filename: `${__dirname}/../logs/http/httpError.log`, alwaysIncludePattern: true }, 7 | httpInfo: { type: 'dateFile', filename: `${__dirname}/../logs/httpInfo/httpInfo.log`, alwaysIncludePattern: true } 8 | }, 9 | categories: { 10 | default: { appenders: ['out'], level: 'error' }, 11 | httpError: { appenders: ['httpError'], level: 'error' }, 12 | httpInfo: { appenders: ['httpInfo'], level: 'info' } 13 | } 14 | }) 15 | 16 | const loggerHttp = log4js.getLogger('httpError') 17 | const loggerHttpInfo = log4js.getLogger('httpInfo') 18 | 19 | module.exports = { loggerHttp, loggerHttpInfo } 20 | -------------------------------------------------------------------------------- /db/models/logs.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const logs = { 7 | async setLog(data){ 8 | return await sqlTable.logs.create({ 9 | userid:data.userid, 10 | content:data.content, 11 | timestamps:data.timestamps 12 | }) 13 | }, 14 | async setLogs(data){ 15 | return await sqlTable.logs.bulkCreate(data.list) 16 | }, 17 | async queryLog(data){ 18 | let limit = data.page_size*1; 19 | let offset = data.page_size * (data.page - 1); 20 | return sqlTable.deposits.findAndCountAll({ 21 | limit, 22 | offset, 23 | where: { 24 | userid:data.userid 25 | } 26 | }) 27 | } 28 | } 29 | 30 | module.exports=logs -------------------------------------------------------------------------------- /public/dist/static/css/chunk-27db.6a1661a5.css: -------------------------------------------------------------------------------- 1 | .admin-member-index .combo-list[data-v-51c447f2]{font-size:0;margin-left:-20px}.admin-member-index .combo-list .combo-item[data-v-51c447f2]{display:inline-block;width:150px;height:150px;padding:0 10px;border:1px solid #e4e7ed;border-radius:4px;text-align:center;overflow:hidden;margin-left:20px;cursor:pointer}.admin-member-index .combo-list .combo-item.active[data-v-51c447f2]{border:1px solid #f56c6c}.admin-member-index .combo-list .item-title[data-v-51c447f2]{height:50px;line-height:50px;border-bottom:1px dashed #dcdfe6;font-size:18px;color:#333}.admin-member-index .combo-list .item-num[data-v-51c447f2]{font-size:18px;margin-top:15px}.admin-member-index .combo-list .primary[data-v-51c447f2]{font-size:30px;color:#f56c6c;font-weight:700;margin-right:5px}.admin-member-index .combo-list .item-desc[data-v-51c447f2]{font-size:14px;margin-top:15px} -------------------------------------------------------------------------------- /db/schema/logs.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // columns 3 | static logs(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | userid: { 12 | type: Sequelize.STRING, 13 | allowNull: false, 14 | }, 15 | content: { 16 | type: Sequelize.STRING 17 | }, 18 | timestamps: { 19 | type: Sequelize.BIGINT, 20 | }, 21 | } 22 | } 23 | 24 | static publicSet() { 25 | return { 26 | freezeTableName: true, 27 | paranoid: true, 28 | timestamps: true, 29 | } 30 | } 31 | } 32 | 33 | module.exports = sqlTable 34 | -------------------------------------------------------------------------------- /keys/master-privatekey.pem: -------------------------------------------------------------------------------- 1 | -----BEGIN RSA PRIVATE KEY----- 2 | MIICXAIBAAKBgQDKx57/DKGMVaPrCm9dewigq/uV3/LvmD8gA9OkcG3Tfd/V4yrL 3 | qqBH3eH0Qkfi3SbYo2mo6sFYhvZI1Cg0jlPUwT1mNedGDNJsEugXujeNMDAEc2xp 4 | jwCvuAO7XGWUmyEynGbS+t4APX1vR0eqZ1XtVDFcWD+Si5EoLcJ+9zcSbQIDAQAB 5 | AoGAecgfLe+pCb3ZHux6a3rbeJsxEhaYMNqPdqu1rHk1DCH5jtvKNNy5O8/+xpIT 6 | 2CSXEt45M5AhJAiR2rVHnCCqKPdjCK0vNef4dI3Q1ngkn7f9pFd/n7QklIUj5HXC 7 | 1I+ucylCVGEHkb6MAV4lya2KWORAgBIm5PYeDVr2bxV+Z4UCQQDlw8EvJZZikI37 8 | 3b8HCMUuHJRSUM4VNCITW67OKUin1GfUy6qdQq5Zqrh/PEM6lUfjrRbraGuMPQba 9 | uSoVfZXrAkEA4e8SVBy0r2ZYqr4QIeJWhK9eRjigvT1TaaJmQ6L6FAx3XxEgk+eE 10 | tpZ6lxy5lM6Z/PkVxwWV1fzXH77Wm+yrBwJAKm2ZkOE6Af17mpT6H4xpDPjOEt1F 11 | EPlNiZGWmA9bkVJR0iPB38ReecRiyWwRZhIG8m4WG2Zu5binOWCT+t/z/QJAWdfk 12 | 1PfZTgqH5dXQuniz6O2HsqrgGICjg9Ulj23mvkQ6z94Ss14UYzsxope0oJ0LlRJD 13 | +7GX5NE3hpMv6i9RzwJBAIZdi6bw53IsMH63KfmIBhfoXSrdma1TrLet6nwDD5b1 14 | 1nC39Y30zzIY9BXKy7Vpc/tPyQRQXfUtPfQAdSVwYWU= 15 | -----END RSA PRIVATE KEY----- 16 | -------------------------------------------------------------------------------- /db/models/statistic.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const Op = Sequelize.Op 3 | const { 4 | sqlTable,sequelize 5 | } = require('../index.js') 6 | 7 | const statistic={ 8 | async set_statistic(data){ 9 | return sqlTable.statistic.bulkCreate(data.list) 10 | }, 11 | 12 | async getToday(data) { 13 | return sequelize.query(`select w_count,p_count,w_total1,p_total1,w_total2,p_total2 from statistic where userid=? and date=?`,{ 14 | type: Sequelize.QueryTypes.SELECT, 15 | replacements:[data.userid,data.day] 16 | }) 17 | }, 18 | 19 | async getWeek(data) { 20 | return sequelize.query(`select w_count,p_count,w_total1,p_total1,w_total2,p_total2,date from statistic where userid=? and date>=? and date<=? order by date`,{ 21 | type: Sequelize.QueryTypes.SELECT, 22 | replacements:[data.userid,data.startDate,data.endDate] 23 | }) 24 | }, 25 | 26 | } 27 | 28 | module.exports = statistic -------------------------------------------------------------------------------- /utils/jwt.js: -------------------------------------------------------------------------------- 1 | const fs = require('fs') 2 | const path = require('path') 3 | const jwt = require('jsonwebtoken') 4 | 5 | class Jwt { 6 | constructor(data) { 7 | this.data = data 8 | } 9 | 10 | generateToken() { 11 | let data = this.data 12 | let now = Math.floor(Date.now() / 1000) 13 | let cert = fs.readFileSync(path.join(__dirname, '../keys/master-privatekey.pem')) 14 | let token = jwt.sign({ 15 | data, 16 | exp: now + 60 * 60 / 2, 17 | }, cert, { algorithm: 'RS256' }) 18 | 19 | return token 20 | } 21 | 22 | verifyToken() { 23 | let token = this.data 24 | let cert = fs.readFileSync(path.join(__dirname, '../keys/master-public.pem')) 25 | let res 26 | try { 27 | let result = jwt.verify(token, cert, { algorithm: ['RS256'] }) || {} 28 | let {exp = 0} = result, current = Math.floor(Date.now() / 1000) 29 | if (current <= exp) { 30 | res = result.data || {} 31 | } 32 | } catch (err) { 33 | res = 'err' 34 | } 35 | return res 36 | } 37 | } 38 | 39 | module.exports = Jwt 40 | -------------------------------------------------------------------------------- /utils/sendSms.js: -------------------------------------------------------------------------------- 1 | const SMS = require('@alicloud/sms-sdk') 2 | 3 | /** 4 | * 短信相关 5 | * */ 6 | const accessKeyId = 'accessKeyId' 7 | const secretAccessKey = 'secretAccessKey' 8 | const sign_name = 'sign_name' 9 | 10 | const tp1 = 'tp1' // 注册模板 11 | 12 | const sendSms = async (mobile_phone, code, type) => { 13 | let cli = new SMS({accessKeyId, secretAccessKey}) 14 | 15 | try { 16 | cli.sendSMS({ 17 | PhoneNumbers: mobile_phone, //必填:待发送手机号。支持以逗号分隔的形式进行批量调用,批量上限为1000个手机号码,批量调用相对于单条调用及时性稍有延迟,验证码类型的短信推荐使用单条调用的方式;发送国际/港澳台消息时,接收号码格式为00+国际区号+号码,如“0085200000000” 18 | SignName: sign_name, //必填:短信签名-可在短信控制台中找到 19 | TemplateCode: tp1, //必填:短信模板-可在短信控制台中找到,发送国际/港澳台消息时,请使用国际/港澳台短信模版 20 | TemplateParam: `{"code": "${code}"}` //可选:模板中的变量替换JSON串,如模板内容为"亲爱的${name},您的验证码为${code}"时。 21 | }).then(res => { 22 | let {Code} = res 23 | if (Code === 'OK') { 24 | return true 25 | } else { 26 | return false 27 | } 28 | }) 29 | return true 30 | } catch (err) { 31 | console.info(err,'error'); 32 | } 33 | } 34 | 35 | module.exports = sendSms 36 | 37 | -------------------------------------------------------------------------------- /utils/sendEmail.js: -------------------------------------------------------------------------------- 1 | const Mailer = require('nodemailer') 2 | 3 | // 邮箱服务 4 | const host = ''; 5 | const port = 80; 6 | const user = ''; 7 | const pass = ''; 8 | 9 | const sendEmail = async (email, code, type) => { 10 | const transport = Mailer.createTransport({ 11 | host:host, // 服务 12 | port:port, // smtp端口 13 | secureConnection: true, // 使用ssl 14 | auth: { 15 | user, 16 | pass 17 | } 18 | }) 19 | 20 | try { 21 | let html1 = `

欢迎使用人人收付平台,您本次注册的验证码为

您的验证码是:${code}

请将以上验证码输入注册页面中的验证码输入框内以完成注册;

如非本人操作,请忽略

`;//注册 22 | 23 | transport.sendMail({ 24 | from: user, // 发件邮箱 25 | to: email, // 收件列表 26 | subject: '来自人人收付平台的邮件', // 标题 27 | // text:'ass', 28 | html: html1 // html 内容 29 | }, function (err, data) { 30 | if (err) { 31 | console.info(err,'error'); 32 | } 33 | transport.close(); // 如果没用,关闭连接池 34 | }) 35 | 36 | return true 37 | } catch (err) { 38 | console.info(err,'error'); 39 | } 40 | } 41 | 42 | module.exports = sendEmail 43 | 44 | -------------------------------------------------------------------------------- /utils/callback.js: -------------------------------------------------------------------------------- 1 | let request = require('request') 2 | 3 | let callback = (callurl,data)=>{ 4 | return new Promise((resolve,reject)=>{ 5 | request.get({url: callurl+`?orderid=${data.orderid}&appid=${data.appid}&amount=${data.amount}&sign=${data.sign}`},(err, _, body)=>{ 6 | if(err){ 7 | reject(); 8 | }else{ 9 | resolve() 10 | } 11 | }) 12 | }) 13 | } 14 | let callPost = (callurl,data)=>{ 15 | var options = { 16 | method: 'POST', 17 | url: callurl, 18 | form: { 19 | sign: data.sign, 20 | appid:data.appid, 21 | orderid:data.orderid, 22 | payway:data.payway, 23 | amount:data.amount 24 | }, 25 | headers: { 26 | 'Content-Type': 'text/plain; charset=utf-8' 27 | }, 28 | timeout: 50000 29 | }; 30 | return new Promise((resolve,reject)=>{ 31 | request.post(options,(err, response, body)=>{ 32 | if(err){ 33 | reject(); 34 | }else{ 35 | resolve() 36 | } 37 | }) 38 | }) 39 | } 40 | 41 | module.exports={ 42 | callPost, 43 | callback 44 | } -------------------------------------------------------------------------------- /db/models/payCode.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const Op = Sequelize.Op 3 | const { 4 | sqlTable, 5 | } = require('../index.js') 6 | 7 | const paycodes = { 8 | async addPayCode(data) { 9 | return sqlTable.paycodes.create({ 10 | userid: data.userid, 11 | qrcode: data.qrcode, 12 | pay_code: data.pay_code, 13 | amount: data.amount, 14 | use_num: 0 15 | }) 16 | }, 17 | async getPayCode(data) { 18 | return sqlTable.paycodes.findOne({ 19 | where: { 20 | userid: data.userid, 21 | amount: data.amount, 22 | pay_code:{$notIn:data.qrcode} 23 | }, 24 | order: ["use_num"], 25 | }, Op.notIn[{qrcode: data.qrcode}]) 26 | }, 27 | async addPayCodeNum(data) { 28 | return sqlTable.paycodes.update({ 29 | num: data.num + 1 30 | }, { 31 | where: { 32 | id: data.id 33 | } 34 | }) 35 | }, 36 | async deletePayCode(data){ 37 | return sqlTable.paycodes.destroy( { 38 | where: { 39 | userid: data.userid 40 | } 41 | }) 42 | } 43 | }; 44 | 45 | module.exports = paycodes 46 | -------------------------------------------------------------------------------- /db/schema/paycode.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | static paycodes(Sequelize) { 3 | return { 4 | id: { 5 | type: Sequelize.BIGINT, 6 | primaryKey: true, 7 | autoIncrement: true, 8 | allowNull: false 9 | }, 10 | // 用户id 11 | userid: { 12 | type: Sequelize.INTEGER, 13 | allowNull: false 14 | }, 15 | // 付款码 16 | qrcode: { 17 | type: Sequelize.STRING, 18 | allowNull: false 19 | }, 20 | // 付款信息备注 21 | pay_code: { 22 | type: Sequelize.STRING, 23 | allowNull: false 24 | }, 25 | // 金额 26 | amount: { 27 | type: Sequelize.INTEGER, 28 | allowNull: false 29 | }, 30 | // 使用次数 31 | use_num: { 32 | type: Sequelize.INTEGER, 33 | defaultValue: 0 34 | } 35 | } 36 | } 37 | 38 | static publicSet() { 39 | return { 40 | freezeTableName: true, 41 | paranoid: true, 42 | timestamps: true, 43 | } 44 | } 45 | } 46 | 47 | 48 | module.exports = sqlTable -------------------------------------------------------------------------------- /db/schema/withdraws.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // columns 3 | static withdraws(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | userid: { 12 | type: Sequelize.STRING, 13 | allowNull: false, 14 | }, 15 | uuid: { 16 | type: Sequelize.STRING, 17 | allowNull: false, 18 | unique: true 19 | }, 20 | type: { 21 | type: Sequelize.STRING, 22 | allowNull: false, 23 | unique: true 24 | }, 25 | amount: { 26 | type: Sequelize.FLOAT 27 | }, 28 | memo: { 29 | type: Sequelize.STRING 30 | }, 31 | status: { 32 | type: Sequelize.TINYINT(1), 33 | defaultValue: 1 34 | } 35 | } 36 | } 37 | 38 | static publicSet() { 39 | return { 40 | freezeTableName: true, 41 | paranoid: true, 42 | timestamps: true, 43 | } 44 | } 45 | } 46 | 47 | module.exports = sqlTable 48 | -------------------------------------------------------------------------------- /db/schema/trades.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // 系统商品表 3 | static trades(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | name: { // 产品名称 12 | type: Sequelize.STRING, 13 | allowNull: false, 14 | }, 15 | amount: { // 产品金额 16 | type: Sequelize.INTEGER, 17 | allowNull: false 18 | }, 19 | create_account_id: { 20 | type: Sequelize.INTEGER, 21 | allowNull: false 22 | }, 23 | desc: { // 商品描述 24 | type: Sequelize.STRING, 25 | allowNull: false 26 | }, 27 | day: { //充值天数 28 | type: Sequelize.INTEGER, 29 | defaultValue: 0 30 | }, 31 | lv: { //充值等级 32 | type: Sequelize.INTEGER, 33 | defaultValue: 0 34 | } 35 | } 36 | }; 37 | 38 | static publicSet() { 39 | return { 40 | freezeTableName: true, 41 | paranoid: true, 42 | timestamps: true, 43 | }; 44 | } 45 | } 46 | 47 | module.exports = sqlTable; -------------------------------------------------------------------------------- /conf/default.js: -------------------------------------------------------------------------------- 1 | const _ = require('lodash') 2 | 3 | let config = { 4 | port: 3000, 5 | db: { 6 | host: process.env.host || '127.0.0.1', 7 | port: process.env.port || 3306, 8 | db: process.env.db ||'db_renrenpay', 9 | dialect: 'mysql', 10 | user: process.env.username || 'root', 11 | password: process.env.password || '123456' 12 | }, 13 | redis: { 14 | host: '127.0.0.1', 15 | port: 6379 16 | }, 17 | websocket: { 18 | port: 7234 19 | }, 20 | appid: process.env.appid || '15c821df1e52', 21 | adminPwd: process.env.adminpwd || '123456', 22 | userid: '1', 23 | pay_userid: process.env.pay_userid ||'2088112172418889' 24 | } 25 | 26 | const init = () => { 27 | if (process.env.NODE_ENV === 'development') { 28 | const localConfig = { 29 | db: _.extend(config.db, { 30 | user: config.user, 31 | password: db.password 32 | }) 33 | } 34 | config = _.extend(config, localConfig) 35 | } 36 | 37 | if (process.env.NODE_ENV === 'production') { 38 | const localConfig = { 39 | db: _.extend(config.db, { 40 | user: '', 41 | password: '' 42 | }) 43 | } 44 | config = _.extend(config, localConfig) 45 | } 46 | 47 | return config 48 | } 49 | 50 | module.exports = init() 51 | -------------------------------------------------------------------------------- /views/404.ejs: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 404 6 | 49 | 50 | 51 | 52 |
53 |


54 |
55 | 56 |
57 |
58 | 59 | 60 | 61 | 62 | 63 | -------------------------------------------------------------------------------- /db/models/trades.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const Op = Sequelize.Op 3 | const { 4 | sqlTable, 5 | } = require('../index.js') 6 | const trades = { 7 | //添加充值订单 8 | async addTrades(data) { 9 | return sqlTable.trades.create({ 10 | name: data.name, 11 | amount: data.amount, 12 | create_account_id: data.userid, 13 | desc: data.desc, 14 | day: data.day, 15 | lv: data.lv 16 | }); 17 | }, 18 | // 修改商品 19 | async modifyTrades(data) { 20 | return sqlTable.trades.update(data, { 21 | where: { 22 | id: data.tradeid 23 | } 24 | }) 25 | }, 26 | /** 27 | * deleteTrades 28 | * 删除商品 29 | * */ 30 | async deleteTrades(data) { 31 | return sqlTable.trades.destroy({ 32 | where: { 33 | id: data.tradeid 34 | } 35 | }) 36 | }, 37 | /** 38 | * 查询订单列表 39 | * start 开始条数 40 | * size 查询页面大小 41 | */ 42 | async getTrades(data) { 43 | return sqlTable.trades.findAndCountAll({ 44 | attributes:[['id', 'tradeid'], 'name', 'desc', 'amount', 'day', 'lv'], 45 | limit: data.page_size * 1, 46 | offset: data.start 47 | }) 48 | }, 49 | /** 50 | * 查找指定的商品 51 | * */ 52 | async findTradeById(trade_id) { 53 | return sqlTable.trades.findById(trade_id); 54 | } 55 | }; 56 | 57 | module.exports = trades; -------------------------------------------------------------------------------- /db/schema/deposits.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // columns 3 | static deposits(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | userid: { 12 | type: Sequelize.STRING, 13 | allowNull: false, 14 | }, 15 | orderid: { 16 | type: Sequelize.STRING, 17 | allowNull: false, 18 | unique: true 19 | }, 20 | tradeid: { 21 | type: Sequelize.INTEGER, 22 | allowNull: false, 23 | defaultValue: 0 24 | }, 25 | //1支付宝充值,2微信充值,3购买会员,4平台消费 26 | type: { 27 | type: Sequelize.TINYINT(6), 28 | allowNull: false, 29 | }, 30 | amount: {// 31 | type: Sequelize.INTEGER 32 | }, 33 | memo: { 34 | type: Sequelize.STRING 35 | }, 36 | status: {//0生成订单 1订单成功 37 | type: Sequelize.TINYINT(1), 38 | defaultValue: 1 39 | } 40 | } 41 | } 42 | 43 | static publicSet() { 44 | return { 45 | freezeTableName: true, 46 | paranoid: true, 47 | timestamps: true, 48 | } 49 | } 50 | } 51 | 52 | module.exports = sqlTable 53 | -------------------------------------------------------------------------------- /db/schema/paychecks.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | static paychecks(Sequelize){ 3 | return { 4 | id: { 5 | type: Sequelize.BIGINT, 6 | primaryKey: true, 7 | autoIncrement: true, 8 | allowNull: false 9 | }, 10 | userid: {//用户id 11 | type: Sequelize.STRING, 12 | allowNull: false, 13 | }, 14 | appid: {// 15 | type: Sequelize.STRING, 16 | }, 17 | orderid: {//订单的id 18 | type:Sequelize.STRING, 19 | unique: true 20 | }, 21 | transid:{//支付宝,微信平台获取的交易id 22 | type: Sequelize.STRING, 23 | unique: true 24 | }, 25 | amount: {//金额 26 | type: Sequelize.FLOAT, 27 | defaultValue: 0.0 28 | }, 29 | payway:{//收款方式 30 | type: Sequelize.TINYINT(1), 31 | defaultValue: 1 32 | }, 33 | memo:{//备注 34 | type:Sequelize.STRING 35 | }, 36 | status: {//订单状态 37 | type: Sequelize.TINYINT(1), 38 | defaultValue: 1, 39 | } 40 | } 41 | } 42 | static publicSet() { 43 | return { 44 | freezeTableName: true, 45 | paranoid: true, 46 | timestamps: true, 47 | } 48 | } 49 | } 50 | 51 | module.exports = sqlTable -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "renrenpay", 3 | "version": "0.1.0", 4 | "private": true, 5 | "scripts": { 6 | "start": "node bin/www", 7 | "dev": "./node_modules/.bin/nodemon bin/www", 8 | "prd": "pm2 start bin/www", 9 | "test": "echo \"Error: no test specified\" && exit 1" 10 | }, 11 | "dependencies": { 12 | "@alicloud/sms-sdk": "^1.1.6", 13 | "async": "^2.6.2", 14 | "debug": "^2.6.3", 15 | "decimal.js": "^10.1.1", 16 | "ejs": "~2.3.3", 17 | "ioredis": "^4.9.0", 18 | "jsonwebtoken": "^8.5.0", 19 | "koa": "^2.2.0", 20 | "koa-bodyparser": "^3.2.0", 21 | "koa-convert": "^1.2.0", 22 | "koa-cors": "0.0.16", 23 | "koa-json": "^2.0.2", 24 | "koa-logger": "^2.0.1", 25 | "koa-onerror": "^1.2.1", 26 | "koa-router": "^7.1.1", 27 | "koa-session": "^5.10.1", 28 | "koa-simple-router": "^0.2.0", 29 | "koa-static": "^3.0.0", 30 | "koa-views": "^5.2.1", 31 | "lodash": "^4.17.11", 32 | "log4js": "^4.0.2", 33 | "mysql2": "^1.6.5", 34 | "nanoid": "^2.0.1", 35 | "node-fetch": "^2.3.0", 36 | "node-schedule": "^1.3.2", 37 | "nodemailer": "^5.1.1", 38 | "redis": "^2.8.0", 39 | "request": "^2.88.0", 40 | "sequelize": "^4.43.0", 41 | "shelljs": "^0.8.3", 42 | "uuid": "^3.3.2", 43 | "websocket": "^1.0.28" 44 | }, 45 | "devDependencies": { 46 | "babel-core": "^6.26.3", 47 | "babel-preset-es2015": "^6.24.1", 48 | "cross-env": "^5.2.0", 49 | "gulp": "^4.0.0", 50 | "gulp-babel": "^8.0.0", 51 | "nodemon": "^1.8.1" 52 | } 53 | } 54 | -------------------------------------------------------------------------------- /db/schema/apps.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // columns 3 | static apps(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | userid: { 12 | type: Sequelize.STRING, 13 | allowNull: false, 14 | }, 15 | name: { 16 | type: Sequelize.STRING, 17 | allowNull: false 18 | // unique: true 19 | }, 20 | callback_url: { 21 | type: Sequelize.STRING 22 | }, 23 | whitelist: { 24 | type: Sequelize.STRING 25 | }, 26 | appid: { 27 | type: Sequelize.STRING, 28 | allowNull: false, 29 | unique: true 30 | }, 31 | appsecret: { 32 | type: Sequelize.STRING, 33 | allowNull: false, 34 | }, 35 | pay_userid:{ 36 | type: Sequelize.STRING, 37 | // allowNull: false 38 | }, 39 | status: { 40 | type: Sequelize.TINYINT(1), 41 | defaultValue: 1 42 | } 43 | } 44 | } 45 | 46 | static publicSet() { 47 | return { 48 | freezeTableName: true, 49 | paranoid: true, 50 | timestamps: true, 51 | } 52 | } 53 | } 54 | 55 | module.exports = sqlTable 56 | -------------------------------------------------------------------------------- /db/schema/orders.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | static orders(Sequelize){ 3 | return { 4 | id: { 5 | type: Sequelize.BIGINT, 6 | primaryKey: true, 7 | autoIncrement: true, 8 | allowNull: false 9 | }, 10 | userid: { 11 | type: Sequelize.STRING, 12 | }, 13 | appid: {//用户应用的id 14 | type: Sequelize.STRING, 15 | }, 16 | orderid: {//订单的id 17 | type:Sequelize.STRING, 18 | unique: true 19 | }, 20 | amount: {//金额 21 | type: Sequelize.BIGINT, 22 | defaultValue: 0.0 23 | }, 24 | payway:{//1支付宝,2微信 25 | type: Sequelize.INTEGER, 26 | defaultValue: 1 27 | }, 28 | qrcode:{ 29 | type:Sequelize.STRING, 30 | }, 31 | memo:{//备注 32 | type:Sequelize.STRING 33 | }, 34 | state:{//0生成订单,订单成功 35 | type: Sequelize.TINYINT(1), 36 | defaultValue: 0 37 | }, 38 | //付款码备注 39 | pay_code: { 40 | type:Sequelize.STRING, 41 | allowNull: false 42 | } 43 | } 44 | } 45 | static publicSet() { 46 | return { 47 | freezeTableName: true, 48 | paranoid: true, 49 | timestamps: true, 50 | } 51 | } 52 | } 53 | 54 | module.exports = sqlTable -------------------------------------------------------------------------------- /public/dist/static/css/chunk-0224.f01dfcab.css: -------------------------------------------------------------------------------- 1 | .login-container .el-input{display:inline-block;height:47px;width:85%}.login-container .el-input input{background:transparent;border:0;-webkit-appearance:none;border-radius:0;padding:12px 5px 12px 15px;color:#eee;height:47px}.login-container .el-input input:-webkit-autofill{-webkit-box-shadow:0 0 0 1000px #2d3a4b inset!important;-webkit-text-fill-color:#fff!important}.login-container .el-form-item{border:1px solid hsla(0,0%,100%,.1);background:rgba(0,0,0,.1);border-radius:5px;color:#454545}.login-container[data-v-b6901faa]{position:fixed;height:100%;width:100%;background-color:#2d3a4b;background:url(/static/img/bg.cf0887a.jpg) no-repeat;background-size:100% 100%}.login-container .login-form[data-v-b6901faa]{position:absolute;top:20%;left:50%;right:0;width:520px;max-width:100%;-webkit-transform:translate(-50%);transform:translate(-50%);padding:35px 35px 15px;background:rgba(0,0,0,.3);border-radius:8px}.login-container .tips[data-v-b6901faa]{font-size:14px;color:#fff;margin-bottom:10px}.login-container .tips a[data-v-b6901faa]{color:#409eff}.login-container .tips-line[data-v-b6901faa]{color:hsla(0,0%,100%,.8);margin:0 10px}.login-container .svg-container[data-v-b6901faa]{padding:6px 5px 6px 15px;color:#889aa4;vertical-align:middle;width:30px;display:inline-block}.login-container .title[data-v-b6901faa]{font-size:26px;font-weight:400;color:#eee;margin:0 auto 40px;text-align:center;font-weight:700}.login-container .show-pwd[data-v-b6901faa]{position:absolute;right:10px;top:7px;font-size:16px;color:#889aa4;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none} -------------------------------------------------------------------------------- /app.js: -------------------------------------------------------------------------------- 1 | const Koa = require('koa') 2 | const app = new Koa() 3 | const router = require('koa-simple-router') 4 | const views = require('koa-views') 5 | const json = require('koa-json') 6 | const onerror = require('koa-onerror') 7 | const bodyparser = require('koa-bodyparser') 8 | const logger = require('koa-logger') 9 | const cors = require('koa-cors') 10 | const errorHandler = require('./utils/errorHandler.js') 11 | const controllers = require('./controllers/index.js'); 12 | let websocket = require('./utils/websocket.js'); 13 | const events = require('events'); 14 | let event = new events.EventEmitter(); 15 | require('./utils/statistic.js').scheduleCronstyle() 16 | let socket = new websocket(); 17 | 18 | require('./db/index.js') 19 | require('./init'); 20 | // error handler 21 | onerror(app) 22 | 23 | // middlewares 24 | app.use(cors()) 25 | app.use(bodyparser({ 26 | enableTypes:['json', 'form', 'text'] 27 | })) 28 | app.use(json()) 29 | app.use(logger()) 30 | app.use(require('koa-static')(__dirname + '/public/dist')) 31 | 32 | app.use(views(__dirname + '/views', { 33 | extension: 'ejs' 34 | })) 35 | 36 | // logger 37 | app.use(async (ctx, next) => { 38 | const start = new Date() 39 | await next() 40 | const ms = new Date() - start 41 | 42 | if ((/\/api\/*/).test(ctx.url)){ 43 | console.log(`${ctx.method} ${ctx.url} - ${ms}ms`) 44 | // await next() 45 | // return 46 | } 47 | 48 | }) 49 | 50 | 51 | // error handler 52 | errorHandler.error(app) 53 | // routers 54 | controllers.getAllRouters(app, router,socket,event) 55 | 56 | // error-handling 57 | app.on('error', (err, ctx) => { 58 | console.error('server error', err, ctx) 59 | }); 60 | 61 | module.exports = app 62 | -------------------------------------------------------------------------------- /utils/verifyParams.js: -------------------------------------------------------------------------------- 1 | const nonEmpty = data => { 2 | if (typeof data == 'number') { 3 | return true 4 | } 5 | 6 | if (typeof data == 'boolean') { 7 | return true 8 | } 9 | 10 | if (data && data.length > 0) { 11 | return true 12 | } 13 | 14 | return false 15 | } 16 | 17 | const verifyParams = async (data, ...params) => { 18 | let result = true 19 | 20 | params.forEach(item => { 21 | if (item === 'mobile_phone_or_email') { 22 | result = nonEmpty(data['mobile_phone']) || nonEmpty(data['email']) 23 | } else if (item === 'lv') { 24 | if (nonEmpty(data['lv'])) { 25 | if (parseInt(data['lv']) === 1 || 26 | parseInt(data['lv']) === 2) { 27 | result = true 28 | } else { 29 | result = false 30 | } 31 | } else { 32 | result = false 33 | } 34 | } else if (item === 'membership_expired') { 35 | if (nonEmpty(data['membership_expired'])) { 36 | if (parseInt(data['membership_expired']) === 30 || 37 | parseInt(data['membership_expired']) === 90 || 38 | parseInt(data['membership_expired']) === 180 || 39 | parseInt(data['membership_expired']) === 365) { 40 | result = true 41 | } else { 42 | result = false 43 | } 44 | } else { 45 | result = false 46 | } 47 | } else if (!nonEmpty(data[item])) { 48 | result = false 49 | } 50 | }) 51 | 52 | return result 53 | } 54 | 55 | module.exports = verifyParams 56 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-26f3.a8b3c6c5.css: -------------------------------------------------------------------------------- 1 | .membership-comp.edit .combo-item:hover .item-del[data-v-1fb68bb0]{-webkit-transform:translate(0);transform:translate(0)}.membership-comp .dead-line[data-v-1fb68bb0]{margin:15px 0}.membership-comp .dead-line .dead-line_member[data-v-1fb68bb0]{color:#e6a23c;margin-left:10px}.membership-comp .dead-line .dead-line_primary[data-v-1fb68bb0]{color:#333;margin-left:10px}.membership-comp .dead-line .dead-line_desc[data-v-1fb68bb0]{margin-left:10px;color:#909399;font-size:12px}.membership-comp .combo-list[data-v-1fb68bb0]{font-size:0;margin-left:-20px}.membership-comp .combo-item[data-v-1fb68bb0]{display:inline-block;width:150px;height:150px;padding:0 10px;border:1px solid #e4e7ed;border-radius:4px;text-align:center;overflow:hidden;margin-left:20px;cursor:pointer;position:relative}.membership-comp .combo-item.active[data-v-1fb68bb0]{border:1px solid #f56c6c}.membership-comp .combo-plus[data-v-1fb68bb0]{font-size:40px;line-height:150px;color:#e4e7ed}.membership-comp .item-title[data-v-1fb68bb0]{height:50px;line-height:50px;border-bottom:1px dashed #dcdfe6;font-size:18px;color:#333}.membership-comp .item-num[data-v-1fb68bb0]{font-size:18px;margin-top:15px}.membership-comp .primary[data-v-1fb68bb0]{font-size:30px;color:#f56c6c;font-weight:700;margin-right:5px}.membership-comp .item-desc[data-v-1fb68bb0]{font-size:14px;margin-top:15px}.membership-comp .item-del[data-v-1fb68bb0]{position:absolute;top:0;left:0;right:0;bottom:0;z-index:9;background:rgba(0,0,0,.8);text-align:center;font-size:30px;color:hsla(0,0%,100%,.7);padding:0 15px;-webkit-transform:translateY(100%);transform:translateY(100%);-webkit-transition:all .3s;transition:all .3s}.membership-comp .item-del div[data-v-1fb68bb0]{height:75px;line-height:75px;cursor:pointer}.membership-comp .item-del div[data-v-1fb68bb0]:first-child{border-bottom:1px dashed hsla(0,0%,100%,.3)} -------------------------------------------------------------------------------- /db/sqlTable.js: -------------------------------------------------------------------------------- 1 | const appsTable = require('./schema/apps.js') 2 | const depositsTable = require('./schema/deposits.js') 3 | const usersTable = require('./schema/users.js') 4 | const withdrawsTable = require('./schema/withdraws.js') 5 | const ordersTable = require('./schema/orders.js') 6 | const paychecksTable = require('./schema/paychecks.js') 7 | const paycodeTable = require('./schema/paycode.js') 8 | const tradesTable = require('./schema/trades.js') 9 | const logsTable = require('./schema/logs.js') 10 | const statisticTable = require('./schema/statistic.js') 11 | 12 | class sqlTable { 13 | constructor(Sequelize, sequelize) { 14 | this.apps = sequelize.define('apps', appsTable.apps(Sequelize), appsTable.publicSet()) 15 | this.deposits = sequelize.define('deposits', depositsTable.deposits(Sequelize), depositsTable.publicSet()) 16 | this.users = sequelize.define('users', usersTable.users(Sequelize), usersTable.publicSet()) 17 | this.withdraws = sequelize.define('withdraws', withdrawsTable.withdraws(Sequelize), withdrawsTable.publicSet()) 18 | this.orders = sequelize.define('orders', ordersTable.orders(Sequelize), ordersTable.publicSet()) 19 | this.paychecks = sequelize.define('paychecks', paychecksTable.paychecks(Sequelize), paychecksTable.publicSet()) 20 | this.paycodes = sequelize.define('paycodes', paycodeTable.paycodes(Sequelize), paycodeTable.publicSet()) 21 | this.trades = sequelize.define('trades', tradesTable.trades(Sequelize), tradesTable.publicSet()) 22 | this.logs = sequelize.define('logs', logsTable.logs(Sequelize), logsTable.publicSet()) 23 | this.statistic = sequelize.define('statistic', statisticTable.statistic(Sequelize),statisticTable.publicSet()) 24 | sequelize.sync({ 25 | logging: false 26 | }) 27 | } 28 | } 29 | 30 | module.exports = sqlTable 31 | -------------------------------------------------------------------------------- /db/models/deposits.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const deposits = { 7 | async addDeposit(data) { 8 | return sqlTable.deposits.create({ 9 | userid: data.userid, 10 | orderid: data.orderid, 11 | type: data.type || 0, 12 | amount: data.amount || 0, 13 | memo: data.memo || data.pay_code || data.orderid || '', 14 | status:data.status || 0 15 | }) 16 | }, 17 | 18 | async updateDepositInfo(data) { 19 | return sqlTable.deposits.update({ 20 | status: data.status || 0 21 | }, { 22 | where: { 23 | orderid: data.orderid || '0', 24 | } 25 | }) 26 | }, 27 | 28 | async getDepositList(data) { 29 | let limit = data.page_size*1; 30 | let offset = data.page_size * (data.page - 1); 31 | let w = {}; 32 | if(data.state == 1){//充值的记录 33 | w = { 34 | userid: data.userid, 35 | status:1, 36 | $or:[{type:1},{type:2}] 37 | } 38 | }else if(data.state == 2){ //购买会员的记录 39 | w = { 40 | userid: data.userid, 41 | status:1, 42 | type:3 43 | } 44 | }else if(data.state == 3){//手续费记录 45 | w = { 46 | userid: data.userid, 47 | status:1, 48 | type:4 49 | } 50 | }else{ 51 | w = { 52 | userid: data.userid, 53 | status:1 54 | } 55 | } 56 | return sqlTable.deposits.findAndCountAll({ 57 | limit, 58 | offset, 59 | where: w 60 | }) 61 | }, 62 | 63 | } 64 | 65 | module.exports = deposits 66 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-fc3b.b79580c9.css: -------------------------------------------------------------------------------- 1 | .login-container .el-input{height:47px;width:85%}.login-container .el-input input{background:transparent;border:0;-webkit-appearance:none;border-radius:0;padding:12px 5px 12px 15px;color:#eee;height:47px}.login-container .el-input input:-webkit-autofill{-webkit-box-shadow:0 0 0 1000px #2d3a4b inset!important;-webkit-text-fill-color:#fff!important}.login-container .el-form-item{border:1px solid hsla(0,0%,100%,.1);background:rgba(0,0,0,.1);border-radius:5px;color:#454545}.login-container .el-input-group__append{background:transparent;color:#eee;border:none}.login-container[data-v-b89682e8]{position:fixed;height:100%;width:100%;background-color:#2d3a4b;background:url(/static/img/bg.cf0887a.jpg) no-repeat;background-size:100% 100%}.login-container .login-form[data-v-b89682e8]{position:absolute;top:20%;left:50%;right:0;-webkit-transform:translate(-50%);transform:translate(-50%);width:520px;max-width:100%;padding:35px 35px 15px;background:rgba(0,0,0,.3);border-radius:8px}.login-container .verify-btn[data-v-b89682e8]{position:absolute;z-index:9;right:20px;top:50%;color:#fff;-webkit-transform:translateY(-50%);transform:translateY(-50%);cursor:pointer;text-align:right}.login-container .tips[data-v-b89682e8]{font-size:14px;color:#fff;margin-bottom:10px}.login-container .tips a[data-v-b89682e8]{color:#409eff}.login-container .tips-line[data-v-b89682e8]{color:hsla(0,0%,100%,.5);margin:0 10px}.login-container .svg-container[data-v-b89682e8]{padding:6px 5px 6px 15px;color:#889aa4;vertical-align:middle;width:30px;display:inline-block}.login-container .title[data-v-b89682e8]{font-size:26px;font-weight:400;color:#eee;margin:0 auto 40px;text-align:center;font-weight:700}.login-container .show-pwd[data-v-b89682e8]{position:absolute;right:10px;top:7px;font-size:16px;color:#889aa4;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none} -------------------------------------------------------------------------------- /db/models/paychecks.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable,sequelize 4 | } = require('../index.js') 5 | 6 | const paychecks = { 7 | async addPaychecks(data){ 8 | return sqlTable.paychecks.create({ 9 | userid:data.userid || '', 10 | appid:data.appid || '', 11 | orderid:data.orderid || null, 12 | amount:data.amount || 0, 13 | payway:data.payway || 1, 14 | memeo:data.memeo || '', 15 | transid:data.transid || null 16 | }) 17 | }, 18 | 19 | async getList(data){ 20 | let limit = data.page_size*1; 21 | let offset = data.page_size * (data.page - 1); 22 | let w = {} 23 | if(data.state*1 === 1){//订单支付成功来源 24 | w = { 25 | userid: data.userid, 26 | orderid:{$ne:null}, 27 | } 28 | }else if(data.state*1 === 0){ //其他来源 29 | w = { 30 | userid: data.userid, 31 | orderid:{$eq:null} 32 | } 33 | }else{ 34 | w = { 35 | userid: data.userid 36 | } 37 | } 38 | return sqlTable.paychecks.findAndCountAll({ 39 | limit, 40 | offset, 41 | order: [['createdAt', 'DESC']] , 42 | where: w 43 | }) 44 | }, 45 | 46 | async totalAmount(data){ 47 | // return sequelize.query(`select amount,payway,appid from paychecks where userid=? and dayofyear(createdAt)=?`,{ 48 | return sequelize.query(`SELECT amount,payway,appid FROM paychecks WHERE userid=? and DATE_FORMAT( createdAt,'%Y-%m-%d') = DATE_FORMAT(CURDATE(),'%Y-%m-%d') ORDER BY createdAt DESC;`,{ 49 | type:Sequelize.QueryTypes.SELECT, 50 | replacements:[data.userid], 51 | }) 52 | }, 53 | } 54 | module.exports = paychecks -------------------------------------------------------------------------------- /public/dist/static/css/chunk-3200.6e36a35c.css: -------------------------------------------------------------------------------- 1 | .forget-container .el-input{height:47px;width:85%}.forget-container .el-input input{background:transparent;border:0;-webkit-appearance:none;border-radius:0;padding:12px 5px 12px 15px;color:#eee;height:47px}.forget-container .el-input input:-webkit-autofill{-webkit-box-shadow:0 0 0 1000px #2d3a4b inset!important;-webkit-text-fill-color:#fff!important}.forget-container .el-form-item{border:1px solid hsla(0,0%,100%,.1);background:rgba(0,0,0,.1);border-radius:5px;color:#454545}.forget-container .el-input-group__append{background:transparent;color:#eee;border:none}.forget-container[data-v-6227c914]{position:fixed;height:100%;width:100%;background-color:#2d3a4b;background:url(/static/img/bg.cf0887a.jpg) no-repeat;background-size:100% 100%}.forget-container .login-form[data-v-6227c914]{position:absolute;top:20%;left:50%;right:0;-webkit-transform:translate(-50%);transform:translate(-50%);width:520px;max-width:100%;padding:35px 35px 15px;background:rgba(0,0,0,.3);border-radius:8px}.forget-container .verify-btn[data-v-6227c914]{position:absolute;z-index:9;right:20px;top:50%;color:#fff;-webkit-transform:translateY(-50%);transform:translateY(-50%);cursor:pointer;text-align:right}.forget-container .tips[data-v-6227c914]{font-size:14px;color:#fff;margin-bottom:10px}.forget-container .tips a[data-v-6227c914]{color:#409eff}.forget-container .tips-line[data-v-6227c914]{color:hsla(0,0%,100%,.5);margin:0 10px}.forget-container .svg-container[data-v-6227c914]{padding:6px 5px 6px 15px;color:#889aa4;vertical-align:middle;width:30px;display:inline-block}.forget-container .title[data-v-6227c914]{font-size:26px;font-weight:400;color:#eee;margin:0 auto 40px;text-align:center;font-weight:700}.forget-container .show-pwd[data-v-6227c914]{position:absolute;right:10px;top:7px;font-size:16px;color:#889aa4;cursor:pointer;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none} -------------------------------------------------------------------------------- /db/schema/statistic.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | static statistic(Sequelize) { 3 | return { 4 | id: { 5 | type: Sequelize.BIGINT, 6 | primaryKey: true, 7 | autoIncrement: true, 8 | allowNull: false 9 | }, 10 | userid: { 11 | type: Sequelize.STRING, 12 | allowNull: false, 13 | }, 14 | appid:{ 15 | type: Sequelize.STRING, 16 | // allowNull: false, 17 | }, 18 | w_count:{//微信订单数量 19 | type: Sequelize.BIGINT, 20 | defaultValue: 0, 21 | }, 22 | p_count:{//支付宝订单数量 23 | type: Sequelize.BIGINT, 24 | defaultValue: 0, 25 | }, 26 | w_total1:{//微信已知来源总额 27 | type: Sequelize.BIGINT, 28 | defaultValue: 0, 29 | }, 30 | p_total1:{//支付宝已知来源总额 31 | type: Sequelize.BIGINT, 32 | defaultValue: 0, 33 | }, 34 | w_total2:{//微信未知来源总额 35 | type: Sequelize.BIGINT, 36 | defaultValue: 0, 37 | }, 38 | p_total2:{//支付宝未知来源总额 39 | type: Sequelize.BIGINT, 40 | defaultValue: 0, 41 | }, 42 | // payway:{ 43 | // type: Sequelize.TINYINT(1), 44 | // // defaultValue: 1 45 | // }, 46 | // status:{ 47 | // type: Sequelize.TINYINT(1), 48 | // defaultValue: 1 49 | // }, 50 | date:{ 51 | type: Sequelize.STRING, 52 | } 53 | } 54 | } 55 | static publicSet() { 56 | return { 57 | freezeTableName: true, 58 | paranoid: true, 59 | timestamps: true, 60 | } 61 | } 62 | } 63 | 64 | module.exports = sqlTable -------------------------------------------------------------------------------- /bin/www: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | 3 | /** 4 | * Module dependencies. 5 | */ 6 | 7 | var app = require('../app'); 8 | var debug = require('debug')('demo:server'); 9 | var http = require('http'); 10 | var config = require('../conf/default.js') 11 | 12 | /** 13 | * Get port from environment and store in Express. 14 | */ 15 | 16 | var port = normalizePort(process.env.PORT || config.port); 17 | // app.set('port', port); 18 | 19 | /** 20 | * Create HTTP server. 21 | */ 22 | 23 | var server = http.createServer(app.callback()); 24 | 25 | /** 26 | * Listen on provided port, on all network interfaces. 27 | */ 28 | 29 | server.listen(port); 30 | server.on('error', onError); 31 | server.on('listening', onListening); 32 | 33 | /** 34 | * Normalize a port into a number, string, or false. 35 | */ 36 | 37 | function normalizePort(val) { 38 | var port = parseInt(val, 10); 39 | 40 | if (isNaN(port)) { 41 | // named pipe 42 | return val; 43 | } 44 | 45 | if (port >= 0) { 46 | // port number 47 | return port; 48 | } 49 | 50 | return false; 51 | } 52 | 53 | /** 54 | * Event listener for HTTP server "error" event. 55 | */ 56 | 57 | function onError(error) { 58 | if (error.syscall !== 'listen') { 59 | throw error; 60 | } 61 | 62 | var bind = typeof port === 'string' 63 | ? 'Pipe ' + port 64 | : 'Port ' + port; 65 | 66 | // handle specific listen errors with friendly messages 67 | switch (error.code) { 68 | case 'EACCES': 69 | console.error(bind + ' requires elevated privileges'); 70 | process.exit(1); 71 | break; 72 | case 'EADDRINUSE': 73 | console.error(bind + ' is already in use'); 74 | process.exit(1); 75 | break; 76 | default: 77 | throw error; 78 | } 79 | } 80 | 81 | /** 82 | * Event listener for HTTP server "listening" event. 83 | */ 84 | 85 | function onListening() { 86 | var addr = server.address(); 87 | var bind = typeof addr === 'string' 88 | ? 'pipe ' + addr 89 | : 'port ' + addr.port; 90 | console.log('Listening on ' + bind); 91 | } 92 | -------------------------------------------------------------------------------- /db/models/orders.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const orders = { 7 | async addOrder(data) { 8 | return sqlTable.orders.create({ 9 | userid: data.userid, 10 | appid: data.appid, 11 | orderid: data.orderid || '', 12 | amount: data.amount || '', 13 | payway: data.payway || '', 14 | memo:data.memo || '', 15 | qrcode:data.qrcode || '', 16 | pay_code: data.pay_code || '' 17 | }) 18 | }, 19 | 20 | async findOne(data){ 21 | return sqlTable.orders.findOne({ 22 | where:{orderid:data.orderid} 23 | }) 24 | }, 25 | 26 | async findOne2(data){ 27 | return sqlTable.orders.findOne({ 28 | where:{memo:data.memo,userid:data.userid} 29 | }) 30 | }, 31 | 32 | async update(data){ 33 | return sqlTable.orders.update({ 34 | qrcode:data.qrcode 35 | },{ 36 | where:{ 37 | orderid:data.orderid 38 | } 39 | }) 40 | }, 41 | 42 | async update2(data){ 43 | return sqlTable.orders.update({ 44 | state:1 45 | },{ 46 | where:{ 47 | orderid:data.orderid 48 | } 49 | }) 50 | }, 51 | 52 | async getList(data){ 53 | let limit = data.page_size*1; 54 | let offset = data.page_size * (data.page - 1); 55 | let w = {} 56 | if(data.state === 0){ 57 | w = { 58 | userid: data.userid, 59 | status:0 60 | } 61 | }else if(data.state === 1){ 62 | w = { 63 | userid: data.userid, 64 | status:1 65 | } 66 | }else{ 67 | w = { 68 | userid: data.userid 69 | } 70 | } 71 | return sqlTable.deposits.findAndCountAll({ 72 | limit, 73 | offset, 74 | where: w 75 | }) 76 | } 77 | } 78 | 79 | module.exports=orders -------------------------------------------------------------------------------- /db/schema/users.js: -------------------------------------------------------------------------------- 1 | class sqlTable { 2 | // columns 3 | static users(Sequelize) { 4 | return { 5 | id: { 6 | type: Sequelize.BIGINT, 7 | primaryKey: true, 8 | autoIncrement: true, 9 | allowNull: false 10 | }, 11 | mobile_phone: { 12 | type: Sequelize.STRING, 13 | unique: true 14 | }, 15 | email: { 16 | type: Sequelize.STRING, 17 | unique: true 18 | }, 19 | password: { 20 | type: Sequelize.STRING, 21 | allowNull: false, 22 | }, 23 | salt: { 24 | type: Sequelize.STRING, 25 | defaultValue: '' 26 | }, 27 | balance: { 28 | type: Sequelize.BIGINT, 29 | defaultValue: 100 30 | }, 31 | lv: { 32 | type: Sequelize.INTEGER, 33 | allowNull: false, 34 | defaultValue: 0 35 | }, 36 | membership_expired: { 37 | type: Sequelize.BIGINT(13), 38 | defaultValue: 0 39 | }, 40 | status: { 41 | type: Sequelize.TINYINT(1), 42 | defaultValue: 1 43 | }, 44 | nickname: { 45 | type: Sequelize.STRING, 46 | defaultValue: '' 47 | }, 48 | weixin: { 49 | type: Sequelize.STRING, 50 | defaultValue: '' 51 | }, 52 | qq: { 53 | type: Sequelize.STRING, 54 | defaultValue: '' 55 | }, 56 | address: { 57 | type: Sequelize.STRING, 58 | defaultValue: '' 59 | } 60 | } 61 | } 62 | 63 | static publicSet() { 64 | return { 65 | freezeTableName: true, 66 | paranoid: true, 67 | timestamps: true, 68 | } 69 | } 70 | } 71 | 72 | module.exports = sqlTable 73 | -------------------------------------------------------------------------------- /utils/errorCode.js: -------------------------------------------------------------------------------- 1 | const ErrorCode = { 2 | 3 | // 系统相关 4 | ERRORCODE_SERVER_INSIDE_ERROR: 1001, // 服务器错误 5 | ERRORCODE_DATABASE: 1002, // 数据库操作异常 6 | ERRORCODE_MISSING_PARAMS: 1003, // 缺少参数 7 | ERRORCODE_INVALID_PARAMS: 1004, // 参数验证失败 8 | ERRORCODE_DATA_STORAGE: 1005, // 数据保存失败 9 | ERRORCODE_SEND_SMS_FAIED: 1006, // 发送短信失败 10 | ERRORCODE_VERIFY_SMS_FAIED: 1007, // 验证短信失败 11 | ERRORCODE_SEND_EMAILCODE_FAIED: 1008, // 发送邮箱验证码失败 12 | ERRORCODE_VERIFY_EMAILCODE_FAIED: 1009, // 验证邮箱验证码失败 13 | 14 | // 应用相关 15 | ERRORCODE_APP_NOT_EXIST: 2001, // 应用不存在 16 | ERRORCODE_APP_ALREADY_EXIST: 2002, // 应用已存在 17 | ERRORCODE_APP_CREATE_FAIED: 2003, // 应用创建失败 18 | ERRORCODE_APP_OVER_LIMIT: 2004, // 应用超过上限 19 | ERRORCODE_APP_NO_CALLBACK: 2005, // 应用没有回调函数 20 | ERRORCODE_APP_GET_APPINFO_FAIED: 2006, // 应用列表获取失败 21 | ERRORCODE_APP_UPDATE_APPINFO_FAIED: 2007, // 应用信息更新失败 22 | ERRORCODE_APP_NOT_ME:2008,//应用不是本人 23 | 24 | // 用户相关 25 | ERRORCODE_USER_NOT_EXIST: 3001, // 用户不存在 26 | ERRORCODE_USER_ALREADY_EXIST: 3002, // 用户已存在 27 | ERRORCODE_USER_REGISTER_FAIED: 3003, // 用户注册失败 28 | ERRORCODE_USER_GET_SALT_FAIED: 3004, // 用户取盐失败 29 | ERRORCODE_USER_LOGIN_FAIED: 3005, // 用户登录失败 30 | ERRORCODE_USER_LOGOUT_FAIED: 3006, // 用户退出失败 31 | ERRORCODE_USER_NOT_ENOUGH_BALANCE: 3007, // 用户余额不足 32 | ERRORCODE_USER_INVALID_USERNAME_OR_PASSWORD: 3008, // 用户名/邮箱/手机号码或密码不正确 33 | ERRORCODE_USER_INVALID_TOKEN: 3009, // 用户Token错误 34 | ERRORCODE_USER_PERMISSION_DENIED: 3010, // 用户权限不足 35 | ERRORCODE_USER_INVALID_SIGNATURE: 3011, // 用户签名错误 36 | ERRORCODE_USER_GET_USERINFO_FAIED: 3012, // 用户信息获取失败 37 | ERRORCODE_USER_DEPOSIT_FAILED: 3013, // 用户充值不足 38 | ERRORCODE_USER_WITHDRAW_FAILED: 3014, // 用户提现不足 39 | ERRORCODE_USER_UPDATE_USERINFO_FAIED: 3015, // 用户信息更新失败 40 | ERRORCODE_USER_RECHARGE_MEMBERSHIP_FAILED: 3016, // 用户购买会员失败 41 | ERRORCODE_USER_BALANCE_NOT_ENOUGH: 3017, // 用户余额不足 42 | ERRORCODE_USER_PASSWORD_ERR: 3018, // 密码正确 43 | 44 | ERRORCODE_SIGN_ERROR:4001,//签名错误 45 | ERRORCODE_APPID_NOT_EXIST:4002,//appid 不存在 46 | NOT_FIND_PHONE:4003,//设备不存在 47 | 48 | ERRORCODE_PERMISSIONS_DENIED: 5001, //权限不足 49 | ERRORCODE_OVER_LOAD_LIMIT: 5002, //会员等级异常 50 | 51 | 52 | } 53 | 54 | module.exports = ErrorCode 55 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-7e70.92d696e8.css: -------------------------------------------------------------------------------- 1 | .dashboard-container[data-v-604c748c]{padding:32px}.dashboard-container .panel-group[data-v-604c748c]{margin-top:18px}.dashboard-container .panel-group .card-panel-col[data-v-604c748c]{margin-bottom:32px}.dashboard-container .panel-group .card-panel[data-v-604c748c]{display:block;height:108px;cursor:pointer;font-size:12px;position:relative;overflow:hidden;color:#666;background:#fff;-webkit-box-shadow:4px 4px 40px rgba(0,0,0,.05);box-shadow:4px 4px 40px rgba(0,0,0,.05);border-color:rgba(0,0,0,.05)}.dashboard-container .panel-group .card-panel:hover .card-panel-icon-wrapper[data-v-604c748c]{color:#fff}.dashboard-container .panel-group .card-panel:hover .icon-people[data-v-604c748c]{background:#40c9c6}.dashboard-container .panel-group .card-panel:hover .icon-message[data-v-604c748c]{background:#36a3f7}.dashboard-container .panel-group .card-panel:hover .icon-money[data-v-604c748c]{background:#f4516c}.dashboard-container .panel-group .card-panel:hover .icon-shopping[data-v-604c748c]{background:#34bfa3}.dashboard-container .panel-group .card-panel .icon-people[data-v-604c748c]{color:#40c9c6}.dashboard-container .panel-group .card-panel .icon-message[data-v-604c748c]{color:#36a3f7}.dashboard-container .panel-group .card-panel .icon-money[data-v-604c748c]{color:#f4516c}.dashboard-container .panel-group .card-panel .icon-shopping[data-v-604c748c]{color:#34bfa3}.dashboard-container .panel-group .card-panel .card-panel-icon-wrapper[data-v-604c748c]{float:left;margin:14px 0 0 14px;padding:16px;-webkit-transition:all .38s ease-out;transition:all .38s ease-out;border-radius:6px}.dashboard-container .panel-group .card-panel .card-panel-icon[data-v-604c748c]{float:left;font-size:48px}.dashboard-container .panel-group .card-panel .card-panel-description[data-v-604c748c]{float:right;font-weight:700;padding:0 30px;margin:26px 0;text-align:right}.dashboard-container .panel-group .card-panel .card-panel-description.primary[data-v-604c748c]{border-right:1px solid #ccc}.dashboard-container .panel-group .card-panel .card-panel-description .card-panel-text[data-v-604c748c]{line-height:18px;color:rgba(0,0,0,.45);font-size:14px;margin-bottom:12px;font-weight:400}.dashboard-container .panel-group .card-panel .card-panel-description .card-panel-num[data-v-604c748c]{font-size:20px}.dashboard-container .chart-title[data-v-604c748c]{text-align:center;font-size:20px;color:#36a3f7;margin:30px 0 20px} -------------------------------------------------------------------------------- /init/index.js: -------------------------------------------------------------------------------- 1 | const users = require("../db/models/users"); 2 | const apps = require("../db/models/apps"); 3 | const crypto = require('crypto'); 4 | const MD5 = require('../utils/md5.js'); 5 | const config = require('../conf/default') 6 | const initController = { 7 | // 初始化用户信息 8 | async initAdmin() { 9 | try { 10 | let sum = 1; 11 | const pwd = MD5(config.adminPwd); 12 | let data = { 13 | mobile_phone: 'admin', 14 | password: pwd, 15 | lv: 99, 16 | // nick_name: 'Rick Chen', 17 | amount: 1000000000, 18 | pay_userid:'2088112172418889', 19 | membership_expired:'9999999999999' 20 | // e_mail: '573099498@qq.com', 21 | // wechart: 'cxyxxx0924', 22 | }; 23 | // 添加用户 24 | const admin = await users.getUserById({userid: 1}); 25 | if(!admin) 26 | await users.addUser(data); 27 | } catch (err) { 28 | console.log(err) 29 | console.error(`initAdmin--${err}`); 30 | } 31 | }, 32 | async initApps() { 33 | const uuidv1 = require('uuid/v1') 34 | // const appid = uuidv1().substring(24, 36) 35 | const appid = config.appid; 36 | console.info(appid); 37 | const appsecret = MD5(appid + Date.now()).toUpperCase() 38 | console.info(appsecret, '--'); 39 | try { 40 | let data = { 41 | userid: config.userid, 42 | name: 'renrenpay', 43 | callback_url: 'http://127.0.0.1:3000/api/deposits/verbCall', 44 | whitelist: '', 45 | appid: appid, 46 | appsecret: appsecret, 47 | pay_userid: config.pay_userid 48 | }; 49 | await apps.chkAppExist(data).then(result => { 50 | if (result.count > 0) { 51 | throw { 52 | code: 3000 53 | } 54 | } 55 | }) 56 | await apps.addApplication(data); 57 | } catch (err) { 58 | console.log(err); 59 | console.error(`initApps--${err}`); 60 | } 61 | } 62 | } 63 | setTimeout(item => { 64 | initController.initAdmin(); 65 | initController.initApps(); 66 | }, 3000) 67 | -------------------------------------------------------------------------------- /utils/verifyToken.js: -------------------------------------------------------------------------------- 1 | const errorCode = require('./errorCode.js') 2 | const redisClient = require('./redisClient.js') 3 | const returnCodesAndMessages = require('./returnJson.js') 4 | const Jwt = require('./jwt.js') 5 | const { 6 | loggerHttp 7 | } = require('./logger.js') 8 | 9 | const verifyToken = async (ctx, rank = 0) => { 10 | try { 11 | const token = ctx.request.header.token 12 | 13 | if (token) { 14 | // 验证Token 15 | let jwt = new Jwt(token) 16 | let result = jwt.verifyToken() 17 | if (!result || result === 'err') { 18 | throw { 19 | code: errorCode.ERRORCODE_USER_INVALID_TOKEN 20 | } 21 | } 22 | ctx.state.user = result 23 | // 判断用户等级 24 | if (rank <= result.lv) { 25 | return result 26 | } else { 27 | throw { 28 | code: errorCode.ERRORCODE_USER_PERMISSION_DENIED 29 | } 30 | } 31 | } else { 32 | throw { 33 | code: errorCode.ERRORCODE_USER_INVALID_TOKEN 34 | } 35 | } 36 | } catch (err) { 37 | if (err.code) { 38 | returnCodesAndMessages.err(ctx, err.code, err.message) 39 | } else { 40 | returnCodesAndMessages.err(ctx, errorCode.ERRORCODE_SERVER_INSIDE_ERROR, '') 41 | loggerHttp.error(`verifyToken -- ${err}`) 42 | } 43 | } 44 | } 45 | 46 | const verifyToken2 = async (ctx, rank = 0)=>{ 47 | try { 48 | const token = ctx.request.header.token 49 | 50 | if (token) { 51 | let result = await redisClient.get('3', token); 52 | if(result){ 53 | ctx.state.user = JSON.parse(result); 54 | return JSON.parse(result); 55 | }else { 56 | throw { 57 | code: errorCode.ERRORCODE_USER_PERMISSION_DENIED 58 | } 59 | } 60 | } else { 61 | throw { 62 | code: errorCode.ERRORCODE_USER_INVALID_TOKEN 63 | } 64 | } 65 | } catch (err) { 66 | if (err.code) { 67 | returnCodesAndMessages.err(ctx, err.code, err.message) 68 | } else { 69 | returnCodesAndMessages.err(ctx, errorCode.ERRORCODE_SERVER_INSIDE_ERROR, '') 70 | loggerHttp.error(`verifyToken -- ${err}`) 71 | } 72 | } 73 | } 74 | 75 | module.exports = { 76 | verifyToken, 77 | verifyToken2 78 | } 79 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-f779.74ec5683.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-f779"],{DmCF:function(e,r,s){"use strict";var t=s("gEW1");s.n(t).a},gEW1:function(e,r,s){},nIie:function(e,r,s){"use strict";s.r(r);var t=s("14Xm"),a=s.n(t),o=s("D3Ub"),n=s.n(o),i=s("wk8/"),l=s("gjeX"),p=s.n(l),d=s("QqA1"),u=s.n(d),c={data:function(){var e=this;return{active:"first",loading:!1,form:{old_password:"",new_password:"",new_password_confirm:""},rules:{old_password:[{required:!0,message:"请输入旧密码",trigger:"blur"}],new_password:[{required:!0,message:"请输入新密码",trigger:"blur"},{min:6,message:"密码长度不低于6位",trigger:"blur"}],new_password_confirm:[{required:!0,message:"请再次输入密码",trigger:"blur"},{validator:function(r,s,t){s!==e.form.new_password?t(new Error("两次输入密码不一致!")):t()},trigger:"blur"}]}}},methods:{submit:function(){var e=this;this.$refs.form.validate(function(){var r=n()(a.a.mark(function r(s){var t;return a.a.wrap(function(r){for(;;)switch(r.prev=r.next){case 0:if(!s){r.next=12;break}return e.loading=!0,(t=u()({},e.form)).old_password=p()(t.old_password),t.new_password=p()(t.new_password),r.next=7,Object(i.i)(t);case 7:if(null!==r.sent){r.next=10;break}return r.abrupt("return");case 10:e.loading=!1,e.$message({type:"success",message:"修改密码成功"});case 12:case"end":return r.stop()}},r,e)}));return function(e){return r.apply(this,arguments)}}())}}},m=(s("DmCF"),s("KHd+")),w=Object(m.a)(c,function(){var e=this,r=e.$createElement,s=e._self._c||r;return s("div",{staticClass:"settings-index-page"},[s("el-tabs",{model:{value:e.active,callback:function(r){e.active=r},expression:"active"}},[s("el-tab-pane",{attrs:{label:"修改密码",name:"first"}},[s("el-form",{ref:"form",staticStyle:{"max-width":"600px"},attrs:{model:e.form,rules:e.rules,"label-width":"100px"}},[s("el-form-item",{attrs:{label:"旧密码",prop:"old_password"}},[s("el-input",{attrs:{type:"password"},model:{value:e.form.old_password,callback:function(r){e.$set(e.form,"old_password",r)},expression:"form.old_password"}})],1),e._v(" "),s("el-form-item",{attrs:{label:"新密码",prop:"new_password"}},[s("el-input",{attrs:{type:"password"},model:{value:e.form.new_password,callback:function(r){e.$set(e.form,"new_password",r)},expression:"form.new_password"}})],1),e._v(" "),s("el-form-item",{attrs:{label:"新密码确认",prop:"new_password_confirm"}},[s("el-input",{attrs:{type:"password"},model:{value:e.form.new_password_confirm,callback:function(r){e.$set(e.form,"new_password_confirm",r)},expression:"form.new_password_confirm"}})],1),e._v(" "),s("el-form-item",[s("el-button",{attrs:{loading:e.loading,type:"primary"},on:{click:e.submit}},[e._v("提交")])],1)],1)],1)],1)],1)},[],!1,null,"3815b536",null);w.options.__file="Index.vue";r.default=w.exports}}]); -------------------------------------------------------------------------------- /controllers/withdrawsController.js: -------------------------------------------------------------------------------- 1 | const nanoid = require('nanoid') 2 | const _ = require('lodash') 3 | const { 4 | loggerHttp 5 | } = require('../utils/logger.js') 6 | const ErrorCode = require('../utils/errorCode.js') 7 | const returnJson = require('../utils/returnJson.js') 8 | const redisCli = require('../utils/redisClient.js') 9 | const MD5 = require('../utils/md5.js') 10 | const Jwt = require('../utils/jwt.js') 11 | const verifyParams = require('../utils/verifyParams.js') 12 | const verifyToken = require('../utils/verifyToken.js') 13 | const sendSms = require('../utils/sendSms.js') 14 | const sendEmail = require('../utils/sendEmail.js') 15 | const withdraws = require('../db/models/withdraws.js') 16 | const users = require('../db/models/users.js') 17 | const uuidv1 = require('uuid/v1') 18 | 19 | const withdrawsController = { 20 | withdraw() { 21 | return async (ctx, next) => { 22 | try { 23 | const data = ctx.request.body 24 | data.userid = ctx.state.user.id 25 | data.action = 'withdraw' 26 | data.uuid = uuidv1() 27 | data.memo = '' 28 | 29 | // 判断参数是否缺少 30 | await verifyParams(data, 'type', 'amount', 'memo').then(result => { 31 | if (!result) { 32 | throw { 33 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 34 | message: '或者参数错误' 35 | } 36 | } 37 | }) 38 | 39 | // 添加提现记录 40 | await withdraws.addWithdraw(data).then(result => { 41 | if (!result) { 42 | throw { 43 | code: ErrorCode.ERRORCODE_USER_WITHDRAW_FAILED, 44 | message: '添加提现记录错误' 45 | } 46 | } 47 | }) 48 | 49 | // 更新balance 50 | await users.updateBalance(data).then(result => { 51 | if (result[0]) { 52 | returnJson.success(ctx) 53 | } else { 54 | throw { 55 | code: ErrorCode.ERRORCODE_USER_NOT_EXIST 56 | } 57 | } 58 | }) 59 | } catch (err) { 60 | console.info(`${ctx.url} -- `,err) 61 | if (err.code) { 62 | returnJson.err(ctx, err.code, err.message) 63 | } else { 64 | returnJson.err(ctx, ErrorCode.ERRORCODE_USER_WITHDRAW_FAILED) 65 | } 66 | } 67 | } 68 | }, 69 | } 70 | 71 | module.exports = withdrawsController 72 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-54db.2bca911d.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-54db"],{"4XeK":function(t,e,a){"use strict";a.d(e,"a",function(){return o}),a.d(e,"b",function(){return r});var n=a("t3Un"),o=function(t){var e=t.payway,a=t.amount;return Object(n.a)({url:"/api/deposits/qrcode",method:"post",data:{payway:e,amount:a}})},r=function(t){var e=t.page,a=void 0===e?1:e,o=t.page_size,r=void 0===o?10:o,i=t.state,s=void 0===i?"":i;return Object(n.a)({url:"/api/deposits/list",method:"get",data:{page:a,page_size:r,state:s}})}},E8qc:function(t,e,a){"use strict";a.r(e);var n=a("14Xm"),o=a.n(n),r=a("D3Ub"),i=a.n(r),s=a("4XeK"),c={data:function(){return{form:{page:1,page_size:10},total:0,loading:!1,list:[]}},created:function(){this.fetch()},methods:{fetch:function(){var t=this;return i()(o.a.mark(function e(){var a;return o.a.wrap(function(e){for(;;)switch(e.prev=e.next){case 0:return t.loading=!0,e.next=3,Object(s.b)(t.form);case 3:if(a=e.sent,t.loading=!1,null!==a){e.next=7;break}return e.abrupt("return");case 7:a.rows.forEach(function(t){t._createdAt=new Date(t.createdAt).toLocaleString()}),t.list=a.rows,t.total=a.count;case 10:case"end":return e.stop()}},e,t)}))()}}},l=a("KHd+"),u=Object(l.a)(c,function(){var t=this,e=t.$createElement,a=t._self._c||e;return a("div",{staticClass:"bill-index-page"},[a("el-table",{staticStyle:{width:"100%","margin-top":"20px"},attrs:{data:t.list,loading:t.loading,stripe:""}},[a("el-table-column",{attrs:{prop:"orderid",label:"id","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"_createdAt",label:"时间","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"amount",label:"类型","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[a("div",[1==e.row.type?a("div",{staticStyle:{color:"#00aae8"}},[a("svg-icon",{attrs:{"icon-class":"zhifubao"}}),t._v(" 支付宝充值")],1):t._e(),t._v(" "),2==e.row.type?a("div",{staticStyle:{color:"#09bb07"}},[a("svg-icon",{attrs:{"icon-class":"wepay"}}),t._v(" 微信充值")],1):t._e(),t._v(" "),3==e.row.type?a("el-tag",{attrs:{type:"success"}},[t._v("购买会员")]):t._e(),t._v(" "),4==e.row.type?a("el-tag",{attrs:{type:"success"}},[t._v("手续费扣除")]):t._e()],1)]}}])}),t._v(" "),a("el-table-column",{attrs:{prop:"amount",label:"金额(元)","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[1==e.row.type||2==e.row.type?a("div",[a("b",{staticStyle:{color:"#409EFF","font-size":"16px","font-weight":"400"}},[t._v("+ "+t._s(e.row.amount/100)+" ")])]):a("div",[a("span",{staticStyle:{color:"#909399"}},[t._v("- "+t._s(e.row.amount/100))])])]}}])})],1),t._v(" "),a("el-pagination",{staticStyle:{"text-align":"right","margin-top":"20px"},attrs:{background:"",layout:"prev, pager, next",total:t.total,"current-page":t.form.page},on:{"update:currentPage":function(e){t.$set(t.form,"page",e)},"current-change":t.fetch}})],1)},[],!1,null,null,null);u.options.__file="index.vue";e.default=u.exports}}]); -------------------------------------------------------------------------------- /utils/websocket.js: -------------------------------------------------------------------------------- 1 | const WebsocketServer = require('websocket').server 2 | const http = require('http') 3 | const wsMap = new Map() 4 | const ErrorCode = require('../utils/errorCode.js') 5 | const config = require('../conf/default.js') 6 | 7 | 8 | const httpServer = http.createServer(function (req, res) { 9 | console.log((new Date()) + ' Received request for ' + req.url) 10 | res.writeHead(404) 11 | res.end() 12 | }) 13 | 14 | function websocket() { 15 | httpServer.listen(config.websocket.port, function () { 16 | console.log((new Date()) + ' Server is listening on port ' + config.websocket.port) 17 | }) 18 | 19 | let wsServer = new WebsocketServer({ 20 | httpServer: httpServer, 21 | autoAcceptConnections: false 22 | }) 23 | 24 | wsServer.on('request', async function (req) { 25 | // let connection = req.accept('echo-protocol', req.origin); 26 | console.info(req.resourceURL.query.user) 27 | console.log((new Date()) + ' Origin: ' + req.origin) 28 | if (!originIsAllowed(req.origin)) { 29 | req.reject() 30 | console.log((new Date()) + ' Connection from origin ' + req.origin + ' rejected.') 31 | return 32 | } 33 | let conn; 34 | try { 35 | conn = req.accept('echo-protocol', req.origin); 36 | } catch (error) { 37 | wsMap.delete(req.origin) 38 | return error 39 | } 40 | let key = req.resourceURL.query.user; 41 | if(key) { 42 | console.info('--->',key); 43 | wsMap.set(key.toString(), conn) 44 | } else { 45 | return 46 | } 47 | 48 | console.log((new Date()) + ' Connection from origin ' + req.origin + ' accepted.') 49 | 50 | conn.on('message', async function (msg) { 51 | console.log(req.origin + ' >>>',req.resourceURL.query.user) 52 | console.log('Received message: ' + msg.utf8Data) 53 | }) 54 | 55 | conn.on('close', function (reasonCode, desc) { 56 | console.log(req.origin + ' >>>') 57 | console.log('Received close: ' + reasonCode + ' desc: ' + desc) 58 | 59 | wsMap.delete(req.origin) 60 | console.log((new Date()) + ' Peer ' + conn.remoteAddrress + ' disconnected.') 61 | }) 62 | }) 63 | } 64 | 65 | function originIsAllowed(origin) { 66 | return true 67 | } 68 | 69 | websocket.prototype.sendMsg = function(useId, msg) { 70 | try { 71 | console.info(useId,msg); 72 | const connection = wsMap.get(useId + ""); 73 | if(!connection) 74 | throw {code: ErrorCode.NOT_FIND_PHONE, message: "没有找到设备"} 75 | connection.sendUTF(msg); 76 | } catch (error) { 77 | console.log(error) 78 | throw {code: ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR, message: "sendMsg error"} 79 | } 80 | 81 | }; 82 | 83 | module.exports = websocket 84 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-0224.e4d32421.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-0224"],{B30W:function(t,n,e){},"Yq+K":function(t,n,e){"use strict";var s=e("B30W");e.n(s).a},c11S:function(t,n,e){"use strict";var s=e("gTgX");e.n(s).a},gTgX:function(t,n,e){},ntYl:function(t,n,e){"use strict";e.r(n);var s=e("14Xm"),o=e.n(s),a=e("D3Ub"),i=e.n(a),r=(e("gjeX"),{name:"Login",data:function(){return{account:"",loginForm:{password:"",mobile_phone:"",email:""},loading:!1,pwdType:"password",redirect:void 0}},watch:{$route:{handler:function(t){this.redirect=t.query&&t.query.redirect},immediate:!0}},methods:{showPwd:function(){"password"===this.pwdType?this.pwdType="":this.pwdType="password"},handleLogin:function(){var t=this;return i()(o.a.mark(function n(){var e;return o.a.wrap(function(n){for(;;)switch(n.prev=n.next){case 0:return t.account||t.$message({type:"warning",message:"请输入邮箱或者手机号"}),/\d{11}/.test(t.account)||"admin"==t.account?t.loginForm.mobile_phone=t.account:t.loginForm.email=t.account,t.loading=!0,n.next=5,t.$store.dispatch("Login",t.loginForm);case 5:e=n.sent,t.loading=!1,null!==e&&(t.$message({type:"success",message:"登录成功"}),t.$router.push({path:"/"}));case 8:case"end":return n.stop()}},n,t)}))()}}}),c=(e("c11S"),e("Yq+K"),e("KHd+")),l=Object(c.a)(r,function(){var t=this,n=t.$createElement,e=t._self._c||n;return e("div",{staticClass:"login-container"},[e("el-form",{ref:"loginForm",staticClass:"login-form",attrs:{model:t.loginForm,"auto-complete":"on","label-position":"left"}},[e("h3",{staticClass:"title"},[t._v("人人收付")]),t._v(" "),e("el-form-item",[e("span",{staticClass:"svg-container"},[e("svg-icon",{attrs:{"icon-class":"phone"}})],1),t._v(" "),e("el-input",{attrs:{name:"phone",type:"text","auto-complete":"on",placeholder:"邮箱/手机号"},model:{value:t.account,callback:function(n){t.account=n},expression:"account"}})],1),t._v(" "),e("el-form-item",{attrs:{prop:"password"}},[e("span",{staticClass:"svg-container"},[e("svg-icon",{attrs:{"icon-class":"password"}})],1),t._v(" "),e("el-input",{attrs:{type:t.pwdType,name:"password","auto-complete":"on",placeholder:"请输入密码"},nativeOn:{keyup:function(n){return"button"in n||!t._k(n.keyCode,"enter",13,n.key,"Enter")?t.handleLogin(n):null}},model:{value:t.loginForm.password,callback:function(n){t.$set(t.loginForm,"password",n)},expression:"loginForm.password"}}),t._v(" "),e("span",{staticClass:"show-pwd",on:{click:t.showPwd}},[e("svg-icon",{attrs:{"icon-class":"password"===t.pwdType?"eye":"eye-open"}})],1)],1),t._v(" "),e("el-form-item",[e("el-button",{staticStyle:{width:"100%"},attrs:{loading:t.loading,type:"primary"},on:{click:t.handleLogin}},[t._v("\n 登录\n ")])],1),t._v(" "),e("div",{staticClass:"tips"},[e("router-link",{staticClass:"tips-link",attrs:{to:{path:"forget"}}},[t._v("忘记密码?")]),t._v(" "),e("span",{staticClass:"tips-line"},[t._v("|")]),t._v(" "),e("router-link",{staticClass:"tips-link",attrs:{to:{path:"register"}}},[t._v(" 注册")])],1)],1)],1)},[],!1,null,"b6901faa",null);l.options.__file="index.vue";n.default=l.exports}}]); -------------------------------------------------------------------------------- /db/models/apps.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const apps = { 7 | async addApplication(data) { 8 | return sqlTable.apps.create({ 9 | userid: data.userid, 10 | name: data.name, 11 | callback_url: data.callback_url || '', 12 | whitelist: data.whitelist || '', 13 | appid: data.appid, 14 | appsecret: data.appsecret, 15 | pay_userid:data.pay_userid, 16 | status: 1 17 | }) 18 | }, 19 | 20 | async chkAppExist(data) { 21 | return sqlTable.apps.findAndCountAll({ 22 | where: { 23 | userid: data.userid, 24 | name: data.name 25 | } 26 | }) 27 | }, 28 | async chkAppExist2(data) { 29 | return sqlTable.apps.findAndCountAll({ 30 | where: { 31 | userid: data.userid, 32 | name: data.name, 33 | id:{$ne:data.id} 34 | } 35 | }) 36 | }, 37 | async getAppCount(data) { 38 | return sqlTable.apps.findAndCountAll({ 39 | where: { 40 | userid: data.userid, 41 | } 42 | }) 43 | }, 44 | 45 | async getApps(data) { 46 | let limit = data.page_size*1; 47 | let offset = data.page_size * (data.page - 1); 48 | return sqlTable.apps.findAndCount({ 49 | limit, 50 | offset, 51 | where: { 52 | userid: data.userid, 53 | } 54 | }) 55 | }, 56 | 57 | async queryApps(data) { 58 | return sqlTable.apps.findAll({ 59 | where: { 60 | appid: data.appid, 61 | } 62 | }) 63 | }, 64 | 65 | async updateAppInfo(data) { 66 | return sqlTable.apps.update({ 67 | callback_url: data.callback_url || '', 68 | whitelist: data.whitelist || '', 69 | name: data.name, 70 | pay_userid:data.pay_userid 71 | }, { 72 | where: { 73 | id: data.id || 0, 74 | userid: data.userid 75 | } 76 | }) 77 | }, 78 | 79 | async deleteApp(data) { 80 | return sqlTable.apps.destroy({ 81 | where: { 82 | id: data.id, 83 | userid: data.userid 84 | } 85 | }) 86 | }, 87 | 88 | async allApps(data) { 89 | return sqlTable.apps.findAll({ 90 | where: { 91 | userid: data.userid, 92 | } 93 | }) 94 | }, 95 | 96 | async isMeApp(data) { 97 | return sqlTable.apps.findAll({ 98 | where: { 99 | userid: data.userid, 100 | appid: data.appid 101 | } 102 | }) 103 | } 104 | } 105 | 106 | module.exports = apps 107 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-280b.7469e992.css: -------------------------------------------------------------------------------- 1 | .pay-comp[data-v-549a193a]{text-align:center}.pay-comp.wepay .deposit-title[data-v-549a193a]{color:#09bb07}.pay-comp.alipay .deposit-title[data-v-549a193a]{color:#00aae8}.pay-comp .deposit-title[data-v-549a193a]{font-size:24px}.pay-comp .deposit-num[data-v-549a193a]{font-size:40px;font-weight:700;margin-top:30px;color:#606266}.deposit-comp[data-v-3258e044]{padding:0 20px 20px;-webkit-box-shadow:0 2px 4px 0 rgba(0,0,0,.12),0 0 6px 0 rgba(0,0,0,.04);box-shadow:0 2px 4px 0 rgba(0,0,0,.12),0 0 6px 0 rgba(0,0,0,.04);border:1px solid #dcdfe6;color:#606266;font-size:14px}.deposit-comp .balance-cell[data-v-3258e044]{display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;height:60px;border-bottom:1px solid #dcdfe6}.deposit-comp .balance-cell .cell-bd[data-v-3258e044]{margin-left:10px}.deposit-comp .balance-cell .primary[data-v-3258e044]{color:#f56c6c}.deposit-comp .balance-cell .cell-ft[data-v-3258e044]{margin-left:30px}.deposit-comp .amount-list[data-v-3258e044]{margin-left:-15px}.deposit-comp .amount-item[data-v-3258e044]{display:inline-block;width:100px;height:100px;line-height:100px;vertical-align:middle;border:1px solid #e4e7ed;border-radius:4px;text-align:center;cursor:pointer;margin-left:15px;margin-bottom:20px;font-size:20px;color:#606266;font-weight:400}.deposit-comp .amount-item[data-v-3258e044]:hover{-webkit-box-shadow:0 2px 12px 0 rgba(0,0,0,.1);box-shadow:0 2px 12px 0 rgba(0,0,0,.1)}.deposit-comp .amount-item.hover[data-v-3258e044]{border:1px solid #409eff;color:#409eff}.deposit-comp .amount-item .desc[data-v-3258e044]{font-size:16px}.deposit-comp .num-widget[data-v-3258e044]{font-size:24px;vertical-align:super}.deposit-comp .combo-list[data-v-3258e044]{font-size:0;margin-left:-20px}.deposit-comp .combo-list .combo-item[data-v-3258e044]{display:inline-block;width:150px;height:150px;padding:0 10px;border:1px solid #e4e7ed;border-radius:4px;text-align:center;overflow:hidden;margin-left:20px;cursor:pointer}.deposit-comp .combo-list .combo-item.active[data-v-3258e044]{border:1px solid #f56c6c}.deposit-comp .combo-list .item-title[data-v-3258e044]{height:50px;line-height:50px;border-bottom:1px dashed #dcdfe6;font-size:18px;color:#333}.deposit-comp .combo-list .item-num[data-v-3258e044]{font-size:18px;margin-top:15px}.deposit-comp .combo-list .primary[data-v-3258e044]{font-size:30px;color:#f56c6c;font-weight:700;margin-right:5px}.deposit-comp .combo-list .item-desc[data-v-3258e044]{font-size:14px;margin-top:15px}.intro-index-page .sec[data-v-63a081ac]{padding:0 20px 20px;-webkit-box-shadow:0 2px 4px 0 rgba(0,0,0,.12),0 0 6px 0 rgba(0,0,0,.04);box-shadow:0 2px 4px 0 rgba(0,0,0,.12),0 0 6px 0 rgba(0,0,0,.04);border:1px solid #dcdfe6;color:#606266;font-size:14px;margin-top:20px}.intro-index-page .sec-title[data-v-63a081ac]{height:60px;line-height:60px;font-size:16px;border-bottom:1px solid #dcdfe6;color:#303133}.intro-index-page .title[data-v-63a081ac]{font-size:15px;margin:20px 0}.intro-index-page .text[data-v-63a081ac]{font-size:14px;margin:10px 0;line-height:1.8} -------------------------------------------------------------------------------- /controllers/ordersController.js: -------------------------------------------------------------------------------- 1 | const nanoid = require('nanoid') 2 | const _ = require('lodash') 3 | const { 4 | loggerHttp 5 | } = require('../utils/logger.js') 6 | const ErrorCode = require('../utils/errorCode.js') 7 | const returnJson = require('../utils/returnJson.js') 8 | const redisCli = require('../utils/redisClient.js') 9 | const MD5 = require('../utils/md5.js') 10 | const Jwt = require('../utils/jwt.js') 11 | const verifyParams = require('../utils/verifyParams.js') 12 | const verifyToken = require('../utils/verifyToken.js') 13 | const sendSms = require('../utils/sendSms.js') 14 | const sendEmail = require('../utils/sendEmail.js') 15 | const deposits = require('../db/models/deposits.js') 16 | const users = require('../db/models/users.js') 17 | const uuidv1 = require('uuid/v1') 18 | const orders = require('../db/models/orders.js') 19 | 20 | const ordersController={ 21 | getList(){//查询订单列表 22 | return async (ctx, next)=>{ 23 | try { 24 | const data = ctx.query 25 | data.userid = ctx.state.user.id 26 | data.page = data.page || 1 27 | data.page_size = data.page_size || 10 28 | 29 | await verifyParams(data, 'userid').then(result => {// 30 | if (!result) { 31 | throw { 32 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 33 | message: '或者参数错误' 34 | } 35 | } 36 | }) 37 | await orders.getList(data).then(result=>{ 38 | returnJson.success(ctx, result); 39 | }) 40 | } catch (err) { 41 | console.info(`${ctx.url} -- `, err) 42 | if(err.code) { 43 | returnJson.err(ctx, err.code, err.message); 44 | } else { 45 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR) 46 | } 47 | } 48 | 49 | 50 | } 51 | }, 52 | 53 | getDetail(){//查询订单详情 54 | return async(ctx,next)=>{ 55 | try { 56 | const data = ctx.query 57 | data.userid = ctx.state.user.id 58 | await verifyParams(data, 'orderid').then(result => { 59 | if (!result) { 60 | throw { 61 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 62 | message: '或者参数错误' 63 | } 64 | } 65 | }) 66 | 67 | await orders.findOne(data).then(result=>{ 68 | returnJson.success(ctx, result); 69 | }) 70 | } catch (err) { 71 | console.info(`${ctx.url} -- `, err) 72 | if(err.code) { 73 | returnJson.err(ctx, err.code, err.message); 74 | } else { 75 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR) 76 | } 77 | } 78 | } 79 | } 80 | } 81 | 82 | module.exports = ordersController; -------------------------------------------------------------------------------- /utils/redisClient.js: -------------------------------------------------------------------------------- 1 | // const redis = require('redis') 2 | // const bluebird = require('bluebird') 3 | // const config = require('../conf/default.js') 4 | // const { 5 | // loggerHttp 6 | // } = require('./logger.js') 7 | 8 | // const redis_client = redis.createClient(config.redis.port, config.redis.host) 9 | 10 | // bluebird.promisifyAll(redis.RedisClient.prototype) 11 | // bluebird.promisifyAll(redis.Multi.prototype) 12 | 13 | // redis_client.on('error', err => { 14 | // loggerHttp.error(`redis -- ${err}`) 15 | // }) 16 | 17 | // const redis_cli = { 18 | 19 | // async set (db, key, value, expired) { 20 | 21 | // await redis_client.selectAsync(db) 22 | 23 | // await redis_client.setAsync(key, value) 24 | 25 | // return redis_client.expireAsync(key, expired) 26 | // }, 27 | 28 | // async get (db, key) { 29 | // await redis_client.selectAsync(db) 30 | 31 | // return await redis_client.getAsync(key) 32 | // }, 33 | 34 | // async del(db, key) { 35 | // await redis_client.selectAsync(db) 36 | 37 | // return await redis_client.delAsync(key) 38 | // }, 39 | // async push(key, value, expired) { 40 | // await redis_client.selectAsync(7) 41 | // await redis_client.lpush(key, value); 42 | // return await redis_client.expireAsync(key, expired); 43 | // }, 44 | // async getList(key) { 45 | // await redis_client.selectAsync(7) 46 | // return await redis_client.lrange(key, 0, 100); 47 | // } 48 | // } 49 | 50 | 51 | 52 | const Redis = require('ioredis') 53 | const bluebird = require('bluebird') 54 | const config = require('../conf/default.js') 55 | 56 | const redis_client = new Redis({ 57 | port: config.redis.port, 58 | host: config.redis.host, 59 | family: 4, 60 | // password:settings.ioredis.password, 61 | db: 0 62 | }); 63 | redis_client.on('connect', function () { 64 | // console.info('Reids connect success') 65 | }); 66 | redis_client.on('end', function () { 67 | console.error('Redis connect error') 68 | }); 69 | 70 | const redis_cli = { 71 | 72 | async set (db, key, value, expired) { 73 | await redis_client.select(db) 74 | // await redis_client.set(key, value,'EX',expired) 75 | await redis_client.set(key, value) 76 | return redis_client.expire(key, expired) 77 | }, 78 | 79 | async get (db, key) { 80 | await redis_client.select(db) 81 | return await redis_client.get(key) 82 | }, 83 | 84 | async del(db, key) { 85 | await redis_client.select(db) 86 | return await redis_client.del(key) 87 | }, 88 | 89 | async push(key, value, expired) { 90 | await redis_client.select(4) 91 | await redis_client.lpush(key, value); 92 | return await redis_client.expire(key, expired); 93 | }, 94 | async getList(key) { 95 | await redis_client.select(4) 96 | return await redis_client.lrange(key, 0, 100); 97 | }, 98 | async remove(key,value){ 99 | await redis_client.select(4) 100 | await redis_client.lrem(key,0,value) 101 | } 102 | 103 | } 104 | 105 | module.exports = redis_cli -------------------------------------------------------------------------------- /utils/errorHandler.js: -------------------------------------------------------------------------------- 1 | const { 2 | loggerHttp, 3 | loggerHttpInfo 4 | } = require('./logger.js') 5 | const verify = require('./verifyToken.js') 6 | const ErrorCode = require('../utils/errorCode.js') 7 | const returnJson = require('../utils/returnJson.js') 8 | const errorHandler = { 9 | error(app) { 10 | // 判断Token 11 | app.use(async (ctx, next) => { 12 | try { 13 | if (ctx.url === '/') { 14 | await next() 15 | return 16 | } 17 | if ((/\/api\/users\/sms\?mobile_phone=.*/ ).test(ctx.url)) { 18 | await next() 19 | return 20 | } 21 | if ((/\/api\/users\/email\?email=.*/ ).test(ctx.url)) { 22 | await next() 23 | return 24 | } 25 | if ((/\/api3\/*/).test(ctx.url)){ 26 | await next() 27 | return 28 | } 29 | if (ctx.url === '/api/users/register') { 30 | await next() 31 | return 32 | } 33 | if (ctx.url === '/api/users/forget') { 34 | await next() 35 | return 36 | } 37 | if ((/\/api\/users\/salt\?mobile_phone=.*/).test(ctx.url)) { 38 | await next() 39 | return 40 | } 41 | if ((/\/api\/get\/salt\?email=.*/).test(ctx.url)) { 42 | await next() 43 | return 44 | } 45 | if (ctx.url === '/api/users/login' || ctx.url === '/api2/phone/login') { 46 | await next() 47 | return 48 | } 49 | if ((/\/api\/get\/userinfo\?mobile_phone=.*/).test(ctx.url)) { 50 | await next() 51 | return 52 | } 53 | if ((/\/api\/get\/userinfo\?email=.*/).test(ctx.url)) { 54 | await next() 55 | return 56 | } 57 | if ((/\/api2\/.*/).test(ctx.url)) { 58 | const result = await verify.verifyToken2(ctx) 59 | if(result){ 60 | await next() 61 | return 62 | }else{ 63 | return 64 | } 65 | } 66 | const result = await verify.verifyToken(ctx) 67 | if (!result) { 68 | return 69 | } 70 | await next() 71 | } catch (err) { 72 | console.info(`${ctx.url} -- `,err) 73 | if (err.code) { 74 | returnJson.err(ctx, err.code, err.message) 75 | } else { 76 | returnJson.err(ctx, ErrorCode.ERRORCODE_USER_INVALID_TOKEN) 77 | } 78 | } 79 | }) 80 | // default 81 | app.use(async (ctx, next) => { 82 | try { 83 | await next() 84 | } catch (err) { 85 | loggerHttp.error(err) 86 | ctx.status = err.status || 500 87 | ctx.body = 'server inside error' 88 | } 89 | }) 90 | } 91 | } 92 | 93 | module.exports = errorHandler 94 | -------------------------------------------------------------------------------- /public/dist/static/css/chunk-libs.e73a8678.css: -------------------------------------------------------------------------------- 1 | /*! normalize.css v7.0.0 | MIT License | github.com/necolas/normalize.css */html{line-height:1.15;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,footer,header,nav,section{display:block}h1{font-size:2em;margin:.67em 0}figcaption,figure,main{display:block}figure{margin:1em 40px}hr{-webkit-box-sizing:content-box;box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a{background-color:transparent;-webkit-text-decoration-skip:objects}abbr[title]{border-bottom:none;text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}b,strong{font-weight:inherit;font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}dfn{font-style:italic}mark{background-color:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}audio,video{display:inline-block}audio:not([controls]){display:none;height:0}img{border-style:none}svg:not(:root){overflow:hidden}button,input,optgroup,select,textarea{font-family:sans-serif;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=reset],[type=submit],button,html [type=button]{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:1px dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{-webkit-box-sizing:border-box;box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{display:inline-block;vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{-webkit-box-sizing:border-box;box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-cancel-button,[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}details,menu{display:block}summary{display:list-item}canvas{display:inline-block}[hidden],template{display:none}#nprogress{pointer-events:none}#nprogress .bar{background:#29d;position:fixed;z-index:1031;top:0;left:0;width:100%;height:2px}#nprogress .peg{display:block;position:absolute;right:0;width:100px;height:100%;-webkit-box-shadow:0 0 10px #29d,0 0 5px #29d;box-shadow:0 0 10px #29d,0 0 5px #29d;opacity:1;-webkit-transform:rotate(3deg) translateY(-4px);transform:rotate(3deg) translateY(-4px)}#nprogress .spinner{display:block;position:fixed;z-index:1031;top:15px;right:15px}#nprogress .spinner-icon{width:18px;height:18px;-webkit-box-sizing:border-box;box-sizing:border-box;border-color:#29d transparent transparent #29d;border-style:solid;border-width:2px;border-radius:50%;-webkit-animation:nprogress-spinner .4s linear infinite;animation:nprogress-spinner .4s linear infinite}.nprogress-custom-parent{overflow:hidden;position:relative}.nprogress-custom-parent #nprogress .bar,.nprogress-custom-parent #nprogress .spinner{position:absolute}@-webkit-keyframes nprogress-spinner{0%{-webkit-transform:rotate(0deg)}to{-webkit-transform:rotate(1turn)}}@keyframes nprogress-spinner{0%{-webkit-transform:rotate(0deg);transform:rotate(0deg)}to{-webkit-transform:rotate(1turn);transform:rotate(1turn)}} -------------------------------------------------------------------------------- /utils/timeformatter.js: -------------------------------------------------------------------------------- 1 | var TimeFormatter = {}; 2 | 3 | TimeFormatter.getTimeStamp = function(str) { 4 | if (str == undefined) { 5 | var now = new Date() 6 | var nowstr = now.getTime(); 7 | return nowstr; 8 | } else if (str == 'formatdate') { 9 | var now = new Date(); 10 | var nowstr = now.format("yyyy-MM-dd hh:mm:ss"); 11 | return nowstr; 12 | } else if (str == 'formatdateforshort') { 13 | var now = new Date(); 14 | var nowstr = now.format("yyyyMMdd"); 15 | nowstr = nowstr.substring(0, 11).replace("-", "").replace("-", "").replace("/", "").replace("/", "").replace(",", ""); 16 | return nowstr; 17 | } else { 18 | return TimeStampUtilities.getDate(str); 19 | } 20 | } 21 | 22 | TimeFormatter.getDate = function(str) { 23 | 24 | this.width = 1; 25 | this.height = 0; 26 | this.depth = 8; 27 | this.dispNumber = "2"; 28 | this.widthAverage = parseInt(this.width/this.dispNumber.length); 29 | 30 | var p = null; 31 | 32 | for (var numSection=0;numSection=myself.numMask.length?0:font); 38 | //var random_x_offs = 0, random_y_offs = 0; 39 | var random_x_offs = parseInt(Math.random()*(this.widthAverage - myself.numMask[font][dispNum][0].length)); 40 | var random_y_offs = parseInt(Math.random()*(this.height - myself.numMask[font][dispNum].length)); 41 | random_x_offs = (random_x_offs<0?0:random_x_offs); 42 | random_y_offs = (random_y_offs<0?0:random_y_offs); 43 | 44 | for (var i=0;(i { 13 | // 应用相关 14 | _.post('/api/apps/create', appsController.createApp()) 15 | _.get('/api/apps/list', appsController.getApps()) //查询app 16 | _.post('/api/apps/update_appinfo', appsController.updateAppInfo()) //修改应用 17 | _.post('/api/apps/delete_app', appsController.deleteApp()) //删除应用 18 | _.post('/api/apps/test', appsController.testQrcode(socket, event))//应用 19 | 20 | 21 | // 充值相关 22 | _.post('/api/deposits/deposit', depositsController.deposit()) 23 | _.post('/api/deposits/qrcode', depositsController.qrCode(socket, event)) //获取二维码 24 | _.get('/api/deposits/list', depositsController.getList()) 25 | 26 | // 用户相关 27 | _.post('/api/users/register', usersController.register()) //注册 28 | _.get('/api/users/salt', usersController.getSalt()) //获取盐 29 | _.post('/api/users/login', usersController.login()) //登陆 30 | _.get('/api/users/logout', usersController.logout()) //登出 31 | _.get('/api/users/userinfo', usersController.getUserInfo()) //获取用信息 32 | _.post('/api/users/update_userinfo', usersController.updateUserInfo()) //修改用户信息 33 | _.post('/api/users/recharge_membership', usersController.rechargeMembership()) //购买会员 34 | _.get('/api/users/sms', usersController.sendSms1()) 35 | _.get('/api/users/email', usersController.sendEmail1()) 36 | _.post('/api/users/update_pwd', usersController.updatePwd()) 37 | _.post('/api/users/forget', usersController.forgetPassword()) 38 | 39 | // 会员应用相关 40 | _.post('/api/trade/create', tradeController.addTrade()); //添加会员策略 41 | _.post('/api/trade/modify', tradeController.modifyTrade()); // 修改会员策略 42 | _.get('/api/trade/delete', tradeController.deleteTrade()); // 删除会员策略 43 | _.get('/api/trade/list', tradeController.tradeList()); //购买会员列表 44 | _.get('/api/trade/info', tradeController.tradeInfo()); //查询制定id的会员策略 45 | 46 | // 提现相关 47 | // _.post('/api/withdraws/withdraw', withdrawsController.withdraw()) 48 | // 订单相关 49 | _.get('/api/orders/list', ordersController.getList());//订单查询 50 | _.get('/api/orders/detail', ordersController.getDetail());//订单详情 51 | _.get('/api/paychecks/list', paychecksController.getList());//收钱来源 52 | _.get('/api/paychecks/statistic', paychecksController.statistic()); 53 | // 支付记录相关 54 | 55 | //手机端相关 56 | _.post('/api2/phone/login', phoneController.login2());//app端登陆 57 | _.post('/api2/phone/get_qrcode', phoneController.sendQrcode(socket, event));//告诉我生成的二维码 58 | _.post('/api2/phone/query_app', phoneController.queryApp());//获取app列表 59 | _.post('/api2/phone/pay_success', phoneController.paySuccess(socket, event));//通知我支付成功 60 | _.get('/api2/phone/paychecks_list', phoneController.paychecks_list()); 61 | _.post('/api2/phone/clear_wx', phoneController.clear_wx()); 62 | _.post('/api2/phone/set_logs', phoneController.setLogs()) 63 | _.post('/api2/phone/query_logs', phoneController.queryLogs()) 64 | 65 | //商户相关 66 | _.post('/api3/needQrcode', merchantController.needQrcode(socket, event));//通知我需要生成二维码 67 | _.post('/api3/test', merchantController.test(socket, event));//通知我需要生成二维码 68 | _.post('/api3/receive', merchantController.receive(event)); //商户收到支付成功通知 69 | 70 | _.get('/', merchantController.offical()); 71 | })) 72 | }, 73 | } 74 | 75 | module.exports = init 76 | -------------------------------------------------------------------------------- /public/dist/index.html: -------------------------------------------------------------------------------- 1 | 人人收付--管理后台
-------------------------------------------------------------------------------- /public/dist/static/js/chunk-5743.e739dad9.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-5743"],{XKNw:function(t,e,a){},bQYc:function(t,e,a){"use strict";var n=a("XKNw");a.n(n).a},sw8x:function(t,e,a){"use strict";a.d(e,"a",function(){return r}),a.d(e,"b",function(){return o});var n=a("t3Un"),r=function(t){var e=t.page,a=void 0===e?1:e,r=t.page_size,o=void 0===r?10:r,i=t.state;return Object(n.a)({url:"/api/paychecks/list",method:"get",params:{page:a,page_size:o,state:i}})},o=function(t){var e=t.startDate,a=t.endDate;return Object(n.a)({url:"/api/paychecks/statistic",method:"get",params:{startDate:e,endDate:a}})}},v5wW:function(t,e,a){"use strict";a.r(e);var n=a("14Xm"),r=a.n(n),o=a("D3Ub"),i=a.n(o),l=a("sw8x"),c={data:function(){return{form:{page:1,page_size:10,state:"1"},list:[],total:0,loading:!1,unknowForm:{page:1,page_size:10,state:"1"},unknowList:[],unknowTotal:0,unknowLoading:!1}},created:function(){this.fetch()},methods:{fetch:function(){var t=this;return i()(r.a.mark(function e(){var a;return r.a.wrap(function(e){for(;;)switch(e.prev=e.next){case 0:return t.loading=!0,e.next=3,Object(l.a)(t.form);case 3:if(a=e.sent,t.loading=!1,null!==a){e.next=7;break}return e.abrupt("return");case 7:a.rows.forEach(function(t){t._createdAt=new Date(t.createdAt).toLocaleString()}),t.list=a.rows,t.total=a.count;case 10:case"end":return e.stop()}},e,t)}))()},fetchUnknow:function(){var t=this;return i()(r.a.mark(function e(){var a;return r.a.wrap(function(e){for(;;)switch(e.prev=e.next){case 0:return t.unknowLoading=!0,e.next=3,Object(l.a)(t.form);case 3:if(a=e.sent,t.unknowLoading=!1,null!==a){e.next=7;break}return e.abrupt("return");case 7:a.rows.forEach(function(t){t._createdAt=new Date(t.createdAt).toLocaleString()}),t.unknowList=a.rows,t.unknowTotal=a.count;case 10:case"end":return e.stop()}},e,t)}))()}}},s=(a("bQYc"),a("KHd+")),u=Object(s.a)(c,function(){var t=this,e=t.$createElement,a=t._self._c||e;return a("div",{staticClass:"orders-index-page"},[a("el-tabs",{model:{value:t.form.state,callback:function(e){t.$set(t.form,"state",e)},expression:"form.state"}},[a("el-tab-pane",{attrs:{label:"订单来源收款",name:"1"}},[a("el-table",{staticStyle:{width:"100%","margin-top":"20px"},attrs:{data:t.list,loading:t.loading,stripe:""}},[a("el-table-column",{attrs:{prop:"orderid",label:"id","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"_createdAt",label:"时间","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"appid",label:"app","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{label:"支付渠道","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[1==e.row.payway?a("div",{staticStyle:{color:"#00aae8"}},[a("svg-icon",{attrs:{"icon-class":"zhifubap"}}),t._v(" 支付宝")],1):t._e(),t._v(" "),2==e.row.payway?a("div",{staticStyle:{color:"#09bb07"}},[a("svg-icon",{attrs:{"icon-class":"wepay"}}),t._v(" 微信支付")],1):t._e()]}}])}),t._v(" "),a("el-table-column",{attrs:{label:"金额(元)","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[a("div",[t._v(t._s(e.row.amount/100))])]}}])})],1),t._v(" "),a("el-pagination",{staticStyle:{"text-align":"right","margin-top":"20px"},attrs:{background:"",layout:"prev, pager, next",total:t.total,"current-page":t.form.page},on:{"update:currentPage":function(e){t.$set(t.form,"page",e)},"current-change":t.fetch}})],1),t._v(" "),a("el-tab-pane",{attrs:{label:"未知来源收款",name:"0"}},[a("el-table",{staticStyle:{width:"100%","margin-top":"20px"},attrs:{data:t.unknowList,loading:t.unknowLoading,stripe:""}},[a("el-table-column",{attrs:{prop:"orderid",label:"id","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"_createdAt",label:"时间","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{prop:"appid",label:"app","min-width":"180"}}),t._v(" "),a("el-table-column",{attrs:{label:"支付渠道","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[1==e.row.payway?a("div",{staticStyle:{color:"#00aae8"}},[a("svg-icon",{attrs:{"icon-class":"zhifubap"}}),t._v(" 支付宝")],1):t._e(),t._v(" "),2==e.row.payway?a("div",{staticStyle:{color:"#09bb07"}},[a("svg-icon",{attrs:{"icon-class":"wepay"}}),t._v(" 微信支付")],1):t._e()]}}])}),t._v(" "),a("el-table-column",{attrs:{label:"金额(元)","min-width":"180"},scopedSlots:t._u([{key:"default",fn:function(e){return[a("div",[t._v(t._s(e.row.amount/100))])]}}])})],1),t._v(" "),a("el-pagination",{staticStyle:{"text-align":"right","margin-top":"20px"},attrs:{background:"",layout:"prev, pager, next",total:t.unknowTotal,"current-page":t.unknowForm.page},on:{"update:currentPage":function(e){t.$set(t.unknowForm,"page",e)},"current-change":t.fetchUnknow}})],1)],1)],1)},[],!1,null,"07355ba6",null);u.options.__file="Index.vue";e.default=u.exports}}]); -------------------------------------------------------------------------------- /public/dist/static/css/chunk-6929.486b8768.css: -------------------------------------------------------------------------------- 1 | .wscn-http404-container[data-v-9b45c3f8]{-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);position:absolute;top:40%;left:50%}.wscn-http404[data-v-9b45c3f8]{position:relative;width:1200px;padding:0 50px;overflow:hidden}.wscn-http404 .pic-404[data-v-9b45c3f8]{position:relative;float:left;width:600px;overflow:hidden}.wscn-http404 .pic-404__parent[data-v-9b45c3f8]{width:100%}.wscn-http404 .pic-404__child[data-v-9b45c3f8]{position:absolute}.wscn-http404 .pic-404__child.left[data-v-9b45c3f8]{width:80px;top:17px;left:220px;opacity:0;-webkit-animation-name:cloudLeft-data-v-9b45c3f8;animation-name:cloudLeft-data-v-9b45c3f8;-webkit-animation-duration:2s;animation-duration:2s;-webkit-animation-timing-function:linear;animation-timing-function:linear;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards;-webkit-animation-delay:1s;animation-delay:1s}.wscn-http404 .pic-404__child.mid[data-v-9b45c3f8]{width:46px;top:10px;left:420px;opacity:0;-webkit-animation-name:cloudMid-data-v-9b45c3f8;animation-name:cloudMid-data-v-9b45c3f8;-webkit-animation-duration:2s;animation-duration:2s;-webkit-animation-timing-function:linear;animation-timing-function:linear;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards;-webkit-animation-delay:1.2s;animation-delay:1.2s}.wscn-http404 .pic-404__child.right[data-v-9b45c3f8]{width:62px;top:100px;left:500px;opacity:0;-webkit-animation-name:cloudRight-data-v-9b45c3f8;animation-name:cloudRight-data-v-9b45c3f8;-webkit-animation-duration:2s;animation-duration:2s;-webkit-animation-timing-function:linear;animation-timing-function:linear;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards;-webkit-animation-delay:1s;animation-delay:1s}@-webkit-keyframes cloudLeft-data-v-9b45c3f8{0%{top:17px;left:220px;opacity:0}20%{top:33px;left:188px;opacity:1}80%{top:81px;left:92px;opacity:1}to{top:97px;left:60px;opacity:0}}@keyframes cloudLeft-data-v-9b45c3f8{0%{top:17px;left:220px;opacity:0}20%{top:33px;left:188px;opacity:1}80%{top:81px;left:92px;opacity:1}to{top:97px;left:60px;opacity:0}}@-webkit-keyframes cloudMid-data-v-9b45c3f8{0%{top:10px;left:420px;opacity:0}20%{top:40px;left:360px;opacity:1}70%{top:130px;left:180px;opacity:1}to{top:160px;left:120px;opacity:0}}@keyframes cloudMid-data-v-9b45c3f8{0%{top:10px;left:420px;opacity:0}20%{top:40px;left:360px;opacity:1}70%{top:130px;left:180px;opacity:1}to{top:160px;left:120px;opacity:0}}@-webkit-keyframes cloudRight-data-v-9b45c3f8{0%{top:100px;left:500px;opacity:0}20%{top:120px;left:460px;opacity:1}80%{top:180px;left:340px;opacity:1}to{top:200px;left:300px;opacity:0}}@keyframes cloudRight-data-v-9b45c3f8{0%{top:100px;left:500px;opacity:0}20%{top:120px;left:460px;opacity:1}80%{top:180px;left:340px;opacity:1}to{top:200px;left:300px;opacity:0}}.wscn-http404 .bullshit[data-v-9b45c3f8]{position:relative;float:left;width:300px;padding:30px 0;overflow:hidden}.wscn-http404 .bullshit__oops[data-v-9b45c3f8]{font-size:32px;line-height:40px;color:#1482f0;margin-bottom:20px;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards}.wscn-http404 .bullshit__headline[data-v-9b45c3f8],.wscn-http404 .bullshit__oops[data-v-9b45c3f8]{font-weight:700;opacity:0;-webkit-animation-name:slideUp-data-v-9b45c3f8;animation-name:slideUp-data-v-9b45c3f8;-webkit-animation-duration:.5s;animation-duration:.5s}.wscn-http404 .bullshit__headline[data-v-9b45c3f8]{font-size:20px;line-height:24px;color:#222;margin-bottom:10px;-webkit-animation-delay:.1s;animation-delay:.1s;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards}.wscn-http404 .bullshit__info[data-v-9b45c3f8]{font-size:13px;line-height:21px;color:grey;margin-bottom:30px;-webkit-animation-delay:.2s;animation-delay:.2s;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards}.wscn-http404 .bullshit__info[data-v-9b45c3f8],.wscn-http404 .bullshit__return-home[data-v-9b45c3f8]{opacity:0;-webkit-animation-name:slideUp-data-v-9b45c3f8;animation-name:slideUp-data-v-9b45c3f8;-webkit-animation-duration:.5s;animation-duration:.5s}.wscn-http404 .bullshit__return-home[data-v-9b45c3f8]{display:block;float:left;width:110px;height:36px;background:#1482f0;border-radius:100px;text-align:center;color:#fff;font-size:14px;line-height:36px;cursor:pointer;-webkit-animation-delay:.3s;animation-delay:.3s;-webkit-animation-fill-mode:forwards;animation-fill-mode:forwards}@-webkit-keyframes slideUp-data-v-9b45c3f8{0%{-webkit-transform:translateY(60px);transform:translateY(60px);opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}}@keyframes slideUp-data-v-9b45c3f8{0%{-webkit-transform:translateY(60px);transform:translateY(60px);opacity:0}to{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}} -------------------------------------------------------------------------------- /utils/returnJson.js: -------------------------------------------------------------------------------- 1 | const errorCode = require('./errorCode.js') 2 | 3 | const codesAndMessages = [ 4 | // 系统相关 5 | { 6 | code: errorCode.ERRORCODE_SERVER_INSIDE_ERROR, 7 | message: '服务器错误' 8 | }, 9 | { 10 | code: errorCode.ERRORCODE_DATABASE, 11 | message: '数据库操作异常' 12 | }, 13 | { 14 | code: errorCode.ERRORCODE_MISSING_PARAMS, 15 | message: '缺少参数' 16 | }, 17 | { 18 | code: errorCode.ERRORCODE_INVALID_PARAMS, 19 | message: '参数验证失败' 20 | }, 21 | { 22 | code: errorCode.ERRORCODE_DATA_STORAGE, 23 | message: '数据保存失败' 24 | }, 25 | { 26 | code: errorCode.ERRORCODE_SEND_SMS_FAIED, 27 | message: '发送短信失败' 28 | }, 29 | { 30 | code: errorCode.ERRORCODE_VERIFY_SMS_FAIED, 31 | message: '验证短信失败' 32 | }, 33 | { 34 | code: errorCode.ERRORCODE_SEND_EMAILCODE_FAIED, 35 | message: '发送邮箱验证码失败' 36 | }, 37 | { 38 | code: errorCode.ERRORCODE_VERIFY_EMAILCODE_FAIED, 39 | message: '验证邮箱验证码失败' 40 | }, 41 | // 应用相关 42 | { 43 | code: errorCode.ERRORCODE_APP_NOT_EXIST, 44 | message: '应用不存在' 45 | }, 46 | { 47 | code: errorCode.ERRORCODE_APP_ALREADY_EXIST, 48 | message: '应用已存在' 49 | }, 50 | { 51 | code: errorCode.ERRORCODE_APP_CREATE_FAIED, 52 | message: '应用创建失败' 53 | }, 54 | { 55 | code: errorCode.ERRORCODE_APP_OVER_LIMIT, 56 | message: '应用超过上限' 57 | }, 58 | { 59 | code: errorCode.ERRORCODE_APP_NO_CALLBACK, 60 | message: '应用没有回调函数' 61 | }, 62 | { 63 | code: errorCode.ERRORCODE_APP_GET_APPINFO_FAIED, 64 | message: '应用信息获取失败' 65 | }, 66 | { 67 | code: errorCode.ERRORCODE_APP_UPDATE_APPINFO_FAIED, 68 | message: '应用信息更新失败' 69 | }, 70 | // 用户相关 71 | { 72 | code: errorCode.ERRORCODE_USER_NOT_EXIST, 73 | message: '用户不存在' 74 | }, 75 | { 76 | code: errorCode.ERRORCODE_USER_ALREADY_EXIST, 77 | message: '用户已存在' 78 | }, 79 | { 80 | code: errorCode.ERRORCODE_USER_REGISTER_FAIED, 81 | message: '用户注册失败' 82 | }, 83 | { 84 | code: errorCode.ERRORCODE_USER_GET_SALT_FAIED, 85 | message: '用户取盐失败' 86 | }, 87 | { 88 | code: errorCode.ERRORCODE_USER_LOGIN_FAIED, 89 | message: '用户登录失败' 90 | }, 91 | { 92 | code: errorCode.ERRORCODE_USER_LOGOUT_FAIED, 93 | message: '用户退出失败' 94 | }, 95 | { 96 | code: errorCode.ERRORCODE_USER_NOT_ENOUGH_BALANCE, 97 | message: '用户余额不足' 98 | }, 99 | { 100 | code: errorCode.ERRORCODE_USER_INVALID_USERNAME_OR_PASSWORD, 101 | message: '用户名/邮箱/手机号码或密码不正确' 102 | }, 103 | { 104 | code: errorCode.ERRORCODE_USER_INVALID_TOKEN, 105 | message: '用户Token错误' 106 | }, 107 | { 108 | code: errorCode.ERRORCODE_USER_PERMISSION_DENIED, 109 | message: '用户权限不足' 110 | }, 111 | { 112 | code: errorCode.ERRORCODE_USER_INVALID_SIGNATURE, 113 | message: '用户签名错误' 114 | }, 115 | { 116 | code: errorCode.ERRORCODE_USER_GET_USERINFO_FAIED, 117 | message: '用户信息获取失败' 118 | }, 119 | { 120 | code: errorCode.ERRORCODE_USER_UPDATE_USERINFO_FAIED, 121 | message: '用户信息更新失败' 122 | }, 123 | { 124 | code: errorCode.ERRORCODE_USER_DEPOSIT_FAILED, 125 | message: '用户充值不足' 126 | }, 127 | { 128 | code: errorCode.ERRORCODE_USER_WITHDRAW_FAILED, 129 | message: '用户提现不足' 130 | }, 131 | { 132 | code: errorCode.ERRORCODE_USER_RECHARGE_MEMBERSHIP_FAILED, 133 | message: '用户购买会员失败' 134 | }, 135 | { 136 | code:errorCode.ERRORCODE_SIGN_ERROR, 137 | message:'签名错误' 138 | }, 139 | { 140 | code:errorCode.ERRORCODE_APPID_NOT_EXIST, 141 | message:'appid 不存在' 142 | }, 143 | { 144 | code:errorCode.NOT_FIND_PHONE, 145 | message:'没有发现设备' 146 | }, 147 | { 148 | code:errorCode.ERRORCODE_USER_PASSWORD_ERR, 149 | message:'密码错误' 150 | }, 151 | ] 152 | 153 | const returnJson = { 154 | async success(ctx, data) { 155 | // console.info(`${ctx.url} -- `,data); 156 | ctx.body = { 157 | code: 200, 158 | message: '操作成功', 159 | data: data, 160 | } 161 | }, 162 | err(ctx, code, extra_message, data) { 163 | let message 164 | codesAndMessages.filter(item => { 165 | if (item.code == code) { 166 | message = item.message 167 | } 168 | }) 169 | if (extra_message) { 170 | message = message + ' (' + extra_message + ')' 171 | } 172 | ctx.body = { 173 | code: code, 174 | message: message || '', 175 | data: data || {}, 176 | } 177 | } 178 | } 179 | 180 | module.exports = returnJson 181 | -------------------------------------------------------------------------------- /db/models/users.js: -------------------------------------------------------------------------------- 1 | const Sequelize = require('sequelize') 2 | const { 3 | sqlTable, 4 | } = require('../index.js') 5 | 6 | const users = { 7 | async addUser(data) { 8 | if (data.mobile_phone !== undefined && data.mobile_phone !== null) { 9 | return sqlTable.users.create({ 10 | mobile_phone: data.mobile_phone, 11 | // email: '', 12 | password: data.password, 13 | lv: data.lv || 0, 14 | membership_expired: data.membership_expired || 0, 15 | amount: data.amount || 0 16 | }) 17 | } else { 18 | return sqlTable.users.create({ 19 | // mobile_phone: '', 20 | lv: data.lv || 0, 21 | amount: data.amount || 0, 22 | email: data.email, 23 | membership_expired: data.membership_expired || 0, 24 | password: data.password, 25 | }) 26 | } 27 | }, 28 | 29 | async chkUserExist(data) { 30 | return sqlTable.users.findAndCountAll({ 31 | attributes: ['id', 'mobile_phone', 'email', 'password', 'salt', 'balance', 'lv', 'membership_expired', 'status', 'nickname', 'weixin', 'qq', 'address'], 32 | where: { 33 | [Sequelize.Op.or]: [{mobile_phone: data.mobile_phone || 'NA'}, {email: data.email || 'NA'}] 34 | } 35 | }) 36 | }, 37 | 38 | async getUserById(data) { 39 | // return sqlTable.users.findById({}) 40 | return sqlTable.users.findOne({ 41 | attributes: [['id', 'userid'], 'mobile_phone', 'email', 'lv', 'membership_expired', 'nickname', 'balance', 'password'], 42 | where: { 43 | id: data.userid || 0 44 | } 45 | }) 46 | }, 47 | 48 | async getUser(data) { 49 | return sqlTable.users.findOne({ 50 | attributes: ['id', 'mobile_phone', 'email', 'lv', 'membership_expired', 'nickname', 'balance'], 51 | where: { 52 | [Sequelize.Op.or]: [{mobile_phone: data.mobile_phone || 'NA'}, {email: data.email || 'NA'}] 53 | } 54 | }) 55 | }, 56 | 57 | async getUser2(data) { 58 | return sqlTable.users.findOne({ 59 | attributes: ['mobile_phone', 'email', 'lv', 'membership_expired', 'nickname', 'balance'], 60 | where: {id: data.userid} 61 | }) 62 | }, 63 | 64 | async getBalance(data) { 65 | return sqlTable.users.findOne({ 66 | attributes: ['balance', 'membership_expired'], 67 | where: { 68 | id: data.userid || 0 69 | } 70 | }) 71 | }, 72 | 73 | async getMembershipExpired(data) { 74 | return sqlTable.users.findOne({ 75 | attributes: ['membership_expired'], 76 | where: { 77 | id: data.userid || 0 78 | } 79 | }) 80 | }, 81 | 82 | async updateSalt(data) { 83 | if (data.mobile_phone !== undefined && data.mobile_phone !== null && data.mobile_phone) { 84 | return sqlTable.users.update({ 85 | salt: data.salt 86 | }, { 87 | where: { 88 | mobile_phone: data.mobile_phone || 'NA' 89 | } 90 | }) 91 | } else { 92 | return sqlTable.users.update({ 93 | salt: data.salt 94 | }, { 95 | where: { 96 | email: data.email || 'NA' 97 | } 98 | }) 99 | } 100 | }, 101 | 102 | async updateBalance(data) { 103 | let balance = 0 104 | await this.getBalance(data).then(result => { 105 | balance = result.balance 106 | }) 107 | balance += data.action === 'deposit' ? parseInt(data.amount) : (-1) * parseInt(data.amount) 108 | return sqlTable.users.update({ 109 | balance: balance 110 | }, { 111 | where: { 112 | id: data.userid || 0 113 | } 114 | }) 115 | }, 116 | 117 | async updateMembership(data) { 118 | return sqlTable.users.update({ 119 | lv: data.lv, 120 | membership_expired: data.membership_expired, 121 | balance: data.balance 122 | }, { 123 | where: { 124 | id: data.userid || 0 125 | } 126 | }) 127 | }, 128 | 129 | async updateMem(data) { 130 | return sqlTable.users.update({ 131 | lv: data.lv, 132 | membership_expired: membership_expired 133 | }, { 134 | where: { 135 | id: data.userid || 0 136 | } 137 | }) 138 | }, 139 | 140 | async updateStatus(data) { 141 | return sqlTable.users.update({ 142 | status: data.status 143 | }, { 144 | where: { 145 | id: data.userid || 0 146 | } 147 | }) 148 | }, 149 | 150 | async updateUserBasicInfo(data) { 151 | return sqlTable.users.update({ 152 | nickname: data.nickname || '', 153 | weixin: data.weixin || '', 154 | qq: data.qq || '', 155 | address: data.address || '' 156 | }, { 157 | where: { 158 | id: data.userid || 0 159 | } 160 | }) 161 | }, 162 | 163 | async updatePwd(data) { 164 | return sqlTable.users.update({ 165 | password: data.new_password 166 | }, { 167 | where: { 168 | id: data.userid 169 | } 170 | }) 171 | 172 | } 173 | } 174 | 175 | module.exports = users 176 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-3200.30d05b08.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-3200"],{"5Xj4":function(e,t,r){"use strict";var s=r("qqHM");r.n(s).a},RBx4:function(e,t,r){},gRYT:function(e,t,r){"use strict";var s=r("RBx4");r.n(s).a},qqHM:function(e,t,r){},xwdn:function(e,t,r){"use strict";r.r(t);var s=r("14Xm"),n=r.n(s),a=r("D3Ub"),o=r.n(a),i=r("gjeX"),c=r.n(i),u=r("wk8/"),l=r("QqA1"),p=r.n(l),m={data:function(){return{mode:"mobile",smsCount:-1,form:{mobile_phone:"",email:"",newPassword:"",code:""},emailCount:-1,emailLoading:!1,emailForm:{email:"",password:"",code:""},rules:{email:[{type:"email",message:"请输入正确的邮箱地址",trigger:"blur"}],code:[{required:!0,message:"请输入验证码",trigger:"blur"}],password:[{required:!0,message:"请输入密码",trigger:"blur"},{min:6,message:"密码长度不低于6位",trigger:"blur"}]},loading:!1,pwdType:"password",redirect:void 0}},methods:{showPwd:function(){"password"===this.pwdType?this.pwdType="":this.pwdType="password"},submit:function(){var e=this;return o()(n.a.mark(function t(){return n.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(/\d{11}/.test(e.form.mobile_phone)){t.next=3;break}return e.$message({type:"warning",message:"请输入正确的手机号"}),t.abrupt("return");case 3:if(e.form.code){t.next=6;break}return e.$message({type:"warning",message:"请输入验证码"}),t.abrupt("return");case 6:if(e.form.password){t.next=9;break}return e.$message({type:"warning",message:"请输入密码"}),t.abrupt("return");case 9:return t.next=11,Object(u.f)({mobile_phone:e.form.mobile_phone,code:e.form.code,password:c()(e.form.password)});case 11:if(null!==t.sent){t.next=14;break}return t.abrupt("return");case 14:e.$message({type:"success",message:"注册成功,请登录"}),e.$router.push({path:"/login"});case 16:case"end":return t.stop()}},t,e)}))()},emailSubmit:function(){var e=this;return o()(n.a.mark(function t(){return n.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:e.$refs.emailForm.validate(function(){var t=o()(n.a.mark(function t(r){var s,a;return n.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(r){t.next=2;break}return t.abrupt("return");case 2:return e.emailLoading=!0,(s=p()({},e.emailForm)).password=c()(s.password),t.next=7,Object(u.f)(s);case 7:if(a=t.sent,e.emailLoading=!1,null!==a){t.next=11;break}return t.abrupt("return");case 11:e.$message({type:"success",message:"注册成功,请登录"}),e.$router.push({path:"/login"});case 13:case"end":return t.stop()}},t,e)}));return function(e){return t.apply(this,arguments)}}());case 1:case"end":return t.stop()}},t,e)}))()},sendSMS:function(){var e=this;return o()(n.a.mark(function t(){var r;return n.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(!(e.smsCount>=0)){t.next=2;break}return t.abrupt("return");case 2:if(/\d{11}/.test(e.form.mobile_phone)){t.next=5;break}return e.$message({type:"warning",message:"请输入正确的手机号"}),t.abrupt("return");case 5:return t.next=7,Object(u.h)({mobile_phone:e.form.mobile_phone});case 7:if(null!==t.sent){t.next=10;break}return t.abrupt("return");case 10:e.$message({type:"success",message:"短信发送成功"}),e.smsCount=60,r=setInterval(function(){e.smsCount<=0&&clearInterval(r),e.smsCount-=1},1e3);case 13:case"end":return t.stop()}},t,e)}))()},sendMail:function(){var e=this;return o()(n.a.mark(function t(){var r;return n.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(!(e.emailCount>=0)){t.next=2;break}return t.abrupt("return");case 2:return t.next=4,Object(u.g)({email:e.emailForm.email});case 4:if(null!==t.sent){t.next=7;break}return t.abrupt("return");case 7:e.$message({type:"success",message:"发送成功"}),e.emailCount=60,r=setInterval(function(){e.emailCount<=0&&clearInterval(r),e.emailCount-=1},1e3);case 10:case"end":return t.stop()}},t,e)}))()}}},f=(r("5Xj4"),r("gRYT"),r("KHd+")),d=Object(f.a)(m,function(){var e=this,t=e.$createElement,r=e._self._c||t;return r("div",{staticClass:"forget-container"},["mobile"==e.mode?r("el-form",{ref:"form",staticClass:"login-form",attrs:{model:e.form,rules:e.rules,"auto-complete":"on","label-position":"left"}},[r("h3",{staticClass:"title"},[e._v("忘记密码")]),e._v(" "),r("el-form-item",{attrs:{prop:"phone"}},[r("span",{staticClass:"svg-container"},[r("svg-icon",{attrs:{"icon-class":"phone"}})],1),e._v(" "),r("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"邮箱/手机号"},model:{value:e.form.mobile_phone,callback:function(t){e.$set(e.form,"mobile_phone",t)},expression:"form.mobile_phone"}})],1),e._v(" "),r("el-form-item",{attrs:{prop:"phone"}},[r("span",{staticClass:"svg-container"},[r("svg-icon",{attrs:{"icon-class":"verifycode"}})],1),e._v(" "),r("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"验证码"},model:{value:e.form.code,callback:function(t){e.$set(e.form,"code",t)},expression:"form.code"}}),e._v(" "),r("div",{staticClass:"verify-btn",on:{click:e.sendSMS}},[e._v(e._s(e.smsCount>=0?e.smsCount:"获取验证码"))])],1),e._v(" "),r("el-form-item",{attrs:{prop:"password"}},[r("span",{staticClass:"svg-container"},[r("svg-icon",{attrs:{"icon-class":"password"}})],1),e._v(" "),r("el-input",{attrs:{type:e.pwdType,name:"password","auto-complete":"on",placeholder:"密码"},nativeOn:{keyup:function(t){return"button"in t||!e._k(t.keyCode,"enter",13,t.key,"Enter")?e.handleLogin(t):null}},model:{value:e.form.password,callback:function(t){e.$set(e.form,"password",t)},expression:"form.password"}}),e._v(" "),r("span",{staticClass:"show-pwd",on:{click:e.showPwd}},[r("svg-icon",{attrs:{"icon-class":"password"===e.pwdType?"eye":"eye-open"}})],1)],1),e._v(" "),r("el-form-item",[r("el-button",{staticStyle:{width:"100%"},attrs:{loading:e.loading,type:"primary"},on:{click:e.submit}},[e._v("\n 提交\n ")])],1),e._v(" "),r("div",{staticClass:"tips"},[r("router-link",{staticClass:"tips-link",attrs:{to:{path:"/login"},href:""}},[e._v("登录")]),e._v(" "),r("span",{staticClass:"tips-line"},[e._v("|")]),e._v(" "),r("router-link",{staticClass:"tips-link",attrs:{to:{path:"/register"},href:""}},[e._v("注册")])],1)],1):e._e()],1)},[],!1,null,"6227c914",null);d.options.__file="Forget.vue";t.default=d.exports}}]); -------------------------------------------------------------------------------- /public/dist/static/js/chunk-7e70.17b6490e.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-7e70"],{AzDp:function(t,a,e){},HGhp:function(t,a,e){"use strict";var n=e("AzDp");e.n(n).a},"r/pB":function(t,a,e){"use strict";e.r(a);var n=e("14Xm"),r=e.n(n),s=e("D3Ub"),i=e.n(s),o=e("7BsA"),c=e.n(o),l=e("MT78"),d=e.n(l);e("gX0l");var u={props:{className:{type:String,default:"chart"},width:{type:String,default:"100%"},height:{type:String,default:"350px"},autoResize:{type:Boolean,default:!0},chartData:{type:Object,required:!0}},data:function(){return{chart:null,sidebarElm:null}},watch:{chartData:{deep:!0,handler:function(t){this.setOptions(t)}}},mounted:function(){var t=this;this.initChart(),this.autoResize&&(this.__resizeHandler=function(t,a,e){var n=void 0,r=void 0,s=void 0,i=void 0,o=void 0,c=function c(){var l=+new Date-i;l0?n=setTimeout(c,a-l):(n=null,e||(o=t.apply(s,r),n||(s=r=null)))};return function(){for(var r=arguments.length,l=Array(r),d=0;d0&&void 0!==arguments[0]?arguments[0]:{},a=t.order,e=t.money;this.chart.setOption({xAxis:{data:["周一","周二","周三","周四","周五","周六","周日"],boundaryGap:!1,axisTick:{show:!1}},grid:{left:10,right:10,bottom:20,top:30,containLabel:!0},tooltip:{trigger:"axis",axisPointer:{type:"cross"},padding:[5,10]},yAxis:{axisTick:{show:!1}},legend:{data:["订单","金额"]},series:[{name:"订单",itemStyle:{normal:{color:"#FF005A",lineStyle:{color:"#FF005A",width:2}}},smooth:!0,type:"line",data:a,animationDuration:1500,animationEasing:"cubicInOut"},{name:"金额",smooth:!0,type:"line",itemStyle:{normal:{color:"#3888fa",lineStyle:{color:"#3888fa",width:2},areaStyle:{color:"#f3f8ff"}}},data:e,animationDuration:1500,animationEasing:"quadraticOut"}]})},initChart:function(){this.chart=d.a.init(this.$el,"macarons"),this.setOptions(this.chartData)}}},h=e("KHd+"),p=Object(h.a)(u,function(){var t=this.$createElement;return(this._self._c||t)("div",{class:this.className,style:{height:this.height,width:this.width}})},[],!1,null,null,null);p.options.__file="chart.vue";var f=p.exports,v=e("sw8x"),m={name:"Dashboard",data:function(){return{lineChartData:{order:[],money:[]},todayOrders:0,todayMoney:0}},components:{CountTo:c.a,Chart:f},created:function(){this.fetchTodayPayStat(),this.fetchWeekPayStat()},methods:{formatDate:function(t){var a=t.getFullYear(),e=t.getMonth()+1,n=t.getDate();return a+"-"+(e<10?"0"+e:e)+"-"+(n<10?"0"+n:n)},fetchWeekPayStat:function(){var t=this;return i()(r.a.mark(function a(){var e,n,s,i,o;return r.a.wrap(function(a){for(;;)switch(a.prev=a.next){case 0:return e=new Date,n=e.getDay(),s=new Date(e.setDate(e.getDate()-n+1)),i=new Date(e.setDate(e.getDate()+7-n)),a.next=6,Object(v.b)({startDate:t.formatDate(s),endDate:t.formatDate(i)});case 6:if(null!==(o=a.sent)){a.next=9;break}return a.abrupt("return");case 9:o.week_list.forEach(function(a){t.lineChartData.order.push(a.p_count+a.w_count),t.lineChartData.money.push((a.p_total1+a.w_total1)/100)});case 10:case"end":return a.stop()}},a,t)}))()},fetchTodayPayStat:function(){var t=this;return i()(r.a.mark(function a(){var e,n;return r.a.wrap(function(a){for(;;)switch(a.prev=a.next){case 0:return e=new Date,a.next=3,Object(v.b)({startDate:t.formatDate(e),endDate:t.formatDate(e)});case 3:if(null!==(n=a.sent)){a.next=6;break}return a.abrupt("return");case 6:t.todayOrders=n.total_count,t.todayMoney=n.total_amount/100;case 8:case"end":return a.stop()}},a,t)}))()}}},y=(e("HGhp"),Object(h.a)(m,function(){var t=this,a=t.$createElement,e=t._self._c||a;return e("div",{staticClass:"dashboard-container"},[e("el-row",{staticClass:"panel-group",attrs:{gutter:40}},[e("el-col",{staticClass:"card-panel-col",attrs:{xs:12,sm:12,lg:12}},[e("router-link",{staticClass:"card-panel",attrs:{to:{path:"/receipt/orders"}}},[e("div",{staticClass:"card-panel-icon-wrapper icon-people"},[e("svg-icon",{attrs:{"icon-class":"list","class-name":"card-panel-icon"}})],1),t._v(" "),e("div",{staticClass:"card-panel-description"},[e("div",{staticClass:"card-panel-text"},[t._v("今日订单")]),t._v(" "),e("count-to",{staticClass:"card-panel-num",attrs:{"start-val":0,"end-val":t.todayOrders,duration:2600}})],1)])],1),t._v(" "),e("el-col",{staticClass:"card-panel-col",attrs:{xs:12,sm:12,lg:12}},[e("router-link",{staticClass:"card-panel",attrs:{to:{path:"/receipt/orders"}}},[e("div",{staticClass:"card-panel-icon-wrapper icon-message"},[e("svg-icon",{attrs:{"icon-class":"money","class-name":"card-panel-icon"}})],1),t._v(" "),e("div",{staticClass:"card-panel-description"},[e("div",{staticClass:"card-panel-text"},[t._v("今日金额(元)")]),t._v(" "),e("count-to",{staticClass:"card-panel-num",attrs:{"start-val":0,"end-val":t.todayMoney,decimals:2,duration:3e3}})],1)])],1)],1),t._v(" "),e("div",{staticClass:"chart-title"},[t._v("本周收款")]),t._v(" "),e("Chart",{attrs:{"chart-data":t.lineChartData}})],1)},[],!1,null,"604c748c",null));y.options.__file="Index.vue";a.default=y.exports},sw8x:function(t,a,e){"use strict";e.d(a,"a",function(){return r}),e.d(a,"b",function(){return s});var n=e("t3Un"),r=function(t){var a=t.page,e=void 0===a?1:a,r=t.page_size,s=void 0===r?10:r,i=t.state;return Object(n.a)({url:"/api/paychecks/list",method:"get",params:{page:e,page_size:s,state:i}})},s=function(t){var a=t.startDate,e=t.endDate;return Object(n.a)({url:"/api/paychecks/statistic",method:"get",params:{startDate:a,endDate:e}})}}}]); -------------------------------------------------------------------------------- /controllers/paychecksController.js: -------------------------------------------------------------------------------- 1 | const nanoid = require('nanoid') 2 | const _ = require('lodash') 3 | const { 4 | loggerHttp 5 | } = require('../utils/logger.js') 6 | const ErrorCode = require('../utils/errorCode.js') 7 | const returnJson = require('../utils/returnJson.js') 8 | const redisCli = require('../utils/redisClient.js') 9 | const MD5 = require('../utils/md5.js') 10 | const Jwt = require('../utils/jwt.js') 11 | const verifyParams = require('../utils/verifyParams.js') 12 | const verifyToken = require('../utils/verifyToken.js') 13 | const sendSms = require('../utils/sendSms.js') 14 | const sendEmail = require('../utils/sendEmail.js') 15 | const deposits = require('../db/models/deposits.js') 16 | const users = require('../db/models/users.js') 17 | const uuidv1 = require('uuid/v1') 18 | const paychecks = require('../db/models/paychecks.js') 19 | const statistic = require('../db/models/statistic.js') 20 | 21 | 22 | const paychecksController = { 23 | getList() {//查询收钱列表 24 | return async (ctx, next) => { 25 | try { 26 | const data = ctx.query 27 | data.userid = ctx.state.user.id 28 | data.page = data.page || 1 29 | data.page_size = data.page_size || 10 30 | 31 | await verifyParams(data, 'userid').then(result => {// 32 | if (!result) { 33 | throw { 34 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 35 | message: '或者参数错误' 36 | } 37 | } 38 | }) 39 | 40 | await paychecks.getList(data).then(result => { 41 | returnJson.success(ctx, result); 42 | }) 43 | } catch (err) { 44 | console.info(`${ctx.url} -- `, err) 45 | if (err.code) { 46 | returnJson.err(ctx, err.code, err.message); 47 | } else { 48 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR) 49 | } 50 | } 51 | } 52 | }, 53 | 54 | statistic() { 55 | return async (ctx, next) => { 56 | try { 57 | const data = ctx.query 58 | data.userid = ctx.state.user.id 59 | await verifyParams(data, 'userid', 'startDate', 'endDate').then(result => { 60 | if (!result) { 61 | throw { 62 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 63 | message: '或者参数错误' 64 | } 65 | } 66 | }) 67 | 68 | let now = new Date(); 69 | data.day = parseInt((now - new Date(now.getFullYear().toString())) / (24 * 60 * 60 * 1000)) + 1; 70 | data.now = now.format("yyyy-MM-dd"); 71 | let row = { 72 | total_count: 0, 73 | total_amount: 0, 74 | week_list: [] 75 | } 76 | if (data.endDate >= data.now) { 77 | let results = await paychecks.totalAmount(data); 78 | let result = { 79 | date: data.now, 80 | w_count: 0,//微信订单数量 81 | p_count: 0,//支付宝订单数量 82 | w_total1: 0,//微信已知来源金额总量 83 | p_total1: 0,//支付宝已知来源金额总量 84 | w_total2: 0,//微信未知来源金额总量 85 | p_total2: 0//支付宝未知来源金额总量 86 | } 87 | results.forEach(item => { 88 | if (item.appid && item.payway == '2') { 89 | result.w_count++; 90 | result.w_total1 += item.amount; 91 | } 92 | if (item.appid && item.payway == '1') { 93 | result.p_count++; 94 | result.p_total1 += item.amount; 95 | } 96 | if (!item.appid && item.payway == '2') { 97 | result.w_total2 += item.amount; 98 | } 99 | if (!item.appid && item.payway == '1') { 100 | result.p_total2 += item.amount; 101 | } 102 | }); 103 | row.total_count = result.w_count + result.p_count; 104 | row.total_amount = result.w_total1 + result.p_total1 + result.w_total2 + result.p_total2; 105 | if (data.endDate == data.startDate) { 106 | row.week_list.push(result); 107 | returnJson.success(ctx, row) 108 | } else { 109 | let week_list = await statistic.getWeek(data); 110 | week_list.forEach(item => {//w_total1,p_total1,w_total2,p_total2 111 | row.total_count = row.total_count + item.w_count + item.p_count 112 | row.total_amount = row.total_amount + item.w_total1 + item.p_total1 + item.w_total2 + item.p_total2 113 | row.week_list.push(item) 114 | }) 115 | row.week_list.push(result); 116 | returnJson.success(ctx, row) 117 | } 118 | } else { 119 | let week_list = await statistic.getWeek(data); 120 | week_list.forEach(item => {//w_total1,p_total1,w_total2,p_total2 121 | row.total_count = row.total_count + item.w_count + item.p_count 122 | row.total_amount = row.total_amount + item.w_total1 + item.p_total1 + item.w_total2 + item.p_total2 123 | row.week_list.push(item) 124 | }) 125 | returnJson.success(ctx, row) 126 | } 127 | } catch (err) { 128 | console.info(`${ctx.url} -- `, err) 129 | if (err.code) { 130 | returnJson.err(ctx, err.code, err.message); 131 | } else { 132 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR) 133 | } 134 | } 135 | } 136 | } 137 | } 138 | 139 | module.exports = paychecksController; -------------------------------------------------------------------------------- /controllers/tradeController.js: -------------------------------------------------------------------------------- 1 | const ErrorCode = require('../utils/errorCode.js') 2 | const returnJson = require('../utils/returnJson.js') 3 | const trades = require('../db/models/trades'); 4 | const verifyParams = require('../utils/verifyParams.js') 5 | 6 | const tradeController = { 7 | // 添加系统商品(会员相关) 8 | addTrade() { 9 | return async (ctx, next) => { 10 | try { 11 | const data = ctx.query; 12 | const user = ctx.state.user; 13 | data.userid = user.id; 14 | if (user.lv < 10) { 15 | throw {code: ErrorCode.ERRORCODE_PERMISSIONS_DENIED, message: "权限不足"} 16 | } 17 | await verifyParams(data, 'amount', 'day', 'lv', 'userid').then(result => { 18 | if (!result) { 19 | throw { 20 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 21 | message: '或者参数错误' 22 | } 23 | } 24 | }); 25 | if (data.lv > 3) { 26 | throw {code: ErrorCode.ERRORCODE_OVER_LOAD_LIMIT, message: "级别错误"} 27 | } 28 | await trades.addTrades(data).then(result => { 29 | returnJson.success(ctx, { 30 | name: result.name, 31 | amount: result.amount, 32 | desc: result.desc, 33 | day: result.day, 34 | tradeid: result.id, 35 | lv: result.lv 36 | }); 37 | }) 38 | } catch (err) { 39 | if (err.code) { 40 | console.log(err); 41 | returnJson.err(ctx, err.code, err); 42 | } else { 43 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR); 44 | } 45 | } 46 | } 47 | }, 48 | // 修改系统商品(会员相关) 49 | modifyTrade() { 50 | return async (ctx, next) => { 51 | try { 52 | const data = ctx.query; 53 | const user = ctx.state.user; 54 | if (user.lv < 10) { 55 | throw {code: ErrorCode.ERRORCODE_PERMISSIONS_DENIED, message: "权限不足"} 56 | } 57 | await verifyParams(data, 'amount', 'day', 'lv', 'tradeid').then(result => { 58 | if (!result) { 59 | throw { 60 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 61 | message: '或者参数错误' 62 | } 63 | } 64 | }); 65 | if (data.lv > 3) { 66 | throw {code: ErrorCode.ERRORCODE_OVER_LOAD_LIMIT, message: "级别错误"} 67 | } 68 | await trades.modifyTrades(data).then(result => { 69 | returnJson.success(ctx, result); 70 | }) 71 | } catch (err) { 72 | if (err.code) { 73 | console.log(err); 74 | returnJson.err(ctx, err.code, err); 75 | } else { 76 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR); 77 | } 78 | } 79 | } 80 | }, 81 | // 删除会员商品 82 | deleteTrade() { 83 | return async (ctx, next) => { 84 | try { 85 | const data = ctx.query; 86 | const user = ctx.state.user; 87 | if (user.lv < 10) { 88 | throw {code: ErrorCode.ERRORCODE_PERMISSIONS_DENIED, message: "权限不足"} 89 | } 90 | await verifyParams(data, 'tradeid').then(result => { 91 | if (!result) { 92 | throw { 93 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 94 | message: '或者参数错误' 95 | } 96 | } 97 | }); 98 | await trades.deleteTrades(data).then(result => { 99 | returnJson.success(ctx, result); 100 | }) 101 | } catch (err) { 102 | if (err.code) { 103 | console.log(err); 104 | returnJson.err(ctx, err.code, err); 105 | } else { 106 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR); 107 | } 108 | } 109 | } 110 | }, 111 | // 查询会员商品列表 112 | tradeList() { 113 | return async (ctx, next) => { 114 | try { 115 | const data = ctx.query; 116 | data.page = data.page ? data.page : 1; 117 | data.page_size = data.page_size ? data.page_size : 10; 118 | data.start = (data.page - 1) * data.page_size; 119 | await trades.getTrades(data).then(result => { 120 | return returnJson.success(ctx, result); 121 | }) 122 | } catch (err) { 123 | console.log(err) 124 | returnJson.err(ctx, ErrorCode.ERRORCODE_SERVER_INSIDE_ERROR); 125 | } 126 | } 127 | }, 128 | // 制定的会员策略 129 | tradeInfo() { 130 | return async (ctx, next) => { 131 | try { 132 | const data = ctx.query; 133 | await verifyParams(data, 'tradeid').then(result => { 134 | if (!result) { 135 | throw { 136 | code: ErrorCode.ERRORCODE_MISSING_PARAMS, 137 | message: '或者参数错误' 138 | } 139 | } 140 | }); 141 | await trades.getTrades(data).then(result => { 142 | returnJson.success(ctx, result); 143 | }) 144 | } catch (err) { 145 | if (err.code) { 146 | returnJson.err(ctx, err.code, err.message); 147 | } else { 148 | returnJson.err(ctx, err.ERRORCODE_SERVER_INSIDE_ERROR) 149 | } 150 | } 151 | } 152 | } 153 | }; 154 | 155 | const md5 = info => { 156 | const md5 = crypto.createHash('md5') 157 | md5.update(info) 158 | return md5.digest('hex') 159 | } 160 | 161 | module.exports = tradeController; 162 | -------------------------------------------------------------------------------- /public/dist/static/js/chunk-280b.82d6890b.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-280b"],{"1jgq":function(t,s,a){},"3NST":function(t,s,a){"use strict";var e=a("HKbb");a.n(e).a},"4XeK":function(t,s,a){"use strict";a.d(s,"a",function(){return n}),a.d(s,"b",function(){return i});var e=a("t3Un"),n=function(t){var s=t.payway,a=t.amount;return Object(e.a)({url:"/api/deposits/qrcode",method:"post",data:{payway:s,amount:a}})},i=function(t){var s=t.page,a=void 0===s?1:s,n=t.page_size,i=void 0===n?10:n,o=t.state,c=void 0===o?"":o;return Object(e.a)({url:"/api/deposits/list",method:"get",data:{page:a,page_size:i,state:c}})}},"4trN":function(t,s,a){"use strict";var e=a("vF8W");a.n(e).a},C5dt:function(t,s,a){"use strict";var e=a("QbLZ"),n=a.n(e),i=a("0FX9"),o=a.n(i),c=a("L2JU"),r={props:{qrcode:String,payway:String,amount:Number},ws:null,computed:n()({},Object(c.c)(["user"])),mounted:function(){var t=this;this.ws=new WebSocket("wss://ws2.renrenpay.info?user=_"+this.user.info.id,"echo-protocol"),this.ws.onopen=function(s){o.a.toCanvas(t.$el.querySelector(".qrcode-canvas"),t.qrcode,{width:300})},this.ws.onmessage=function(s){try{var a=JSON.parse(s.data);1==a.paySuccess?t.$emit("success",a):t.$emit("fail")}catch(t){alert(t)}}},methods:{close:function(){this.ws.close()}}},l=(a("3NST"),a("KHd+")),u=Object(l.a)(r,function(){var t=this,s=t.$createElement,a=t._self._c||s;return a("div",{staticClass:"pay-comp",class:{wepay:2==t.payway,alipay:1==t.payway}},[1==t.payway?a("div",{staticClass:"deposit-title"},[a("svg-icon",{attrs:{"icon-class":"zhifubao"}}),t._v(" 支付宝\n ")],1):t._e(),t._v(" "),2==t.payway?a("div",{staticClass:"deposit-title"},[a("svg-icon",{attrs:{"icon-class":"wepay"}}),t._v(" 微信支付\n ")],1):t._e(),t._v(" "),a("div",{staticClass:"deposit-num"},[a("span",{staticClass:"num-widget"},[t._v("¥")]),t._v("\n "+t._s(t.amount/100)+"\n ")]),t._v(" "),a("canvas",{staticClass:"qrcode-canvas"})])},[],!1,null,"549a193a",null);u.options.__file="Pay.vue";s.a=u.exports},GgHD:function(t,s,a){"use strict";a.r(s);var e=a("14Xm"),n=a.n(e),i=a("D3Ub"),o=a.n(i),c=a("QbLZ"),r=a.n(c),l=a("4XeK"),u=(a("wk8/"),a("L2JU")),p=(a("0FX9"),a("I8ep")),v=a("C5dt"),d={data:function(){return{dialogVisible:!1,amountList:[1,1e3,2e3,3e3,5e3,1e4,3e4,1e5],btnLoading:!1,form:{payway:"2",amount:""},depositStep:1,qrcode:""}},watch:{"form.payway":function(){this.form.amount=""},dialogVisible:function(t){t&&(this.depositStep=1)}},computed:r()({},Object(u.c)({user:function(t){return t.user}})),components:{Membership:p.a,Pay:v.a},created:function(){var t=this;return o()(n.a.mark(function s(){return n.a.wrap(function(s){for(;;)switch(s.prev=s.next){case 0:t.$store.dispatch("FetchUser");case 1:case"end":return s.stop()}},s,t)}))()},methods:{changeMember:function(t){this.member=t},paySuccess:function(){this.$message({type:"success",message:"充值成功"}),this.dialogVisible=!1,this.$store.dispatch("FetchUser")},fetchQrcodeSubmit:function(){var t=this;return o()(n.a.mark(function s(){var a;return n.a.wrap(function(s){for(;;)switch(s.prev=s.next){case 0:if(t.form.amount){s.next=3;break}return t.$message({type:"warning",message:"请选择充值金额"}),s.abrupt("return");case 3:return t.btnLoading=!0,s.next=6,Object(l.a)(t.form);case 6:if(a=s.sent,t.btnLoading=!1,null!==a){s.next=10;break}return s.abrupt("return");case 10:t.qrcode=a.qrcode,t.depositStep=2;case 12:case"end":return s.stop()}},s,t)}))()}}},m=(a("rOdO"),a("KHd+")),f=Object(m.a)(d,function(){var t=this,s=t.$createElement,a=t._self._c||s;return a("div",{staticClass:"deposit-comp"},[a("div",{staticClass:"balance-cell"},[a("div",{staticClass:"cell-hd"},[t._v("账户余额 : ")]),t._v(" "),a("div",{staticClass:"cell-bd"},[a("b",{staticClass:"primary"},[t._v(t._s(t.user.info.balance/100||0))]),t._v("\n 元\n ")]),t._v(" "),a("div",{staticClass:"cell-ft"},[a("el-button",{attrs:{size:"small",type:"primary"},on:{click:function(s){t.dialogVisible=!0}}},[t._v("充值")])],1)]),t._v(" "),a("Membership",{on:{change:t.changeMember}}),t._v(" "),a("el-dialog",{attrs:{title:"充值",visible:t.dialogVisible,width:"40%"},on:{"update:visible":function(s){t.dialogVisible=s}}},[1==t.depositStep?[a("el-tabs",{model:{value:t.form.payway,callback:function(s){t.$set(t.form,"payway",s)},expression:"form.payway"}},[a("el-tab-pane",{attrs:{name:"2"}},[a("span",{attrs:{slot:"label"},slot:"label"},[a("svg-icon",{attrs:{"icon-class":"wepay"}}),t._v(" 微信支付\n ")],1),t._v(" "),a("div",{staticClass:"amount-list"},t._l(t.amountList,function(s,e){return a("div",{key:e,staticClass:"amount-item",class:{hover:t.form.amount==s},on:{click:function(a){t.form.amount=s}}},[t._v("\n "+t._s(s/100)+" \n "),a("span",{staticClass:"desc"},[t._v("元")])])}))]),t._v(" "),a("el-tab-pane",{attrs:{name:"1"}},[a("span",{attrs:{slot:"label"},slot:"label"},[a("svg-icon",{attrs:{"icon-class":"zhifubao"}}),t._v(" 支付宝\n ")],1),t._v(" "),a("div",{staticClass:"amount-list"},t._l(t.amountList,function(s,e){return a("div",{key:e,staticClass:"amount-item",class:{hover:t.form.amount==s},on:{click:function(a){t.form.amount=s}}},[t._v("\n "+t._s(s/100)+" \n "),a("span",{staticClass:"desc"},[t._v("元")])])}))])],1),t._v(" "),a("span",{staticClass:"dialog-footer",attrs:{slot:"footer"},slot:"footer"},[a("el-button",{on:{click:function(s){t.dialogVisible=!1}}},[t._v("取 消")]),t._v(" "),a("el-button",{attrs:{type:"primary",loading:t.btnLoading},on:{click:t.fetchQrcodeSubmit}},[t._v("确 定")])],1)]:t._e(),t._v(" "),2==t.depositStep?a("pay",{attrs:{amount:t.form.amount,payway:t.form.payway,qrcode:t.qrcode},on:{success:t.paySuccess}}):t._e()],2)],1)},[],!1,null,"3258e044",null);f.options.__file="Deposit.vue";var _={components:{Deposit:f.exports}},b=(a("4trN"),Object(m.a)(_,function(){var t=this.$createElement,s=this._self._c||t;return s("div",{staticClass:"intro-index-page"},[s("deposit"),this._v(" "),this._m(0)],1)},[function(){var t=this,s=t.$createElement,a=t._self._c||s;return a("div",{staticClass:"sec"},[a("div",{staticClass:"sec-title"},[a("b",[t._v("使用教程")])]),t._v(" "),a("div",{staticClass:"title"},[a("b",[t._v("第一步: 创建应用")])]),t._v(" "),a("div",{staticClass:"text"},[t._v("\n 点击左侧菜单栏 ”应用管理“,进入界面,然后点击里面的”创建应用“按钮,开始创建您的第一个应用。 "),a("br"),t._v("创建成功后,就会为你生成AppID和AppSecret,这两个字段后面用于手机上的app配置\n ")]),t._v(" "),a("div",{staticClass:"title"},[a("b",[t._v("第二步: 下载App")])]),t._v(" "),a("div",{staticClass:"text"},[t._v("\n 点击左侧菜单栏 ”软件下载“,进入界面,按照要求依次下载里面的软件。如果手机内预先安装了其他版本的微信、支付宝,请卸载后再安装\n ")]),t._v(" "),a("div",{staticClass:"title"},[a("b",[t._v("第三步: 配置App")])]),t._v(" "),a("div",{staticClass:"text"},[t._v("\n 点击左侧菜单栏 ”软件下载“,进入界面,按照要求依次下载里面的软件。如果手机内预先安装了其他版本的微信、支付宝,请卸载后再安装\n ")]),t._v(" "),a("div",{staticClass:"title"},[a("b",[t._v("第四步: 测试支付")])]),t._v(" "),a("div",{staticClass:"text"},[t._v("\n 点击左侧菜单栏 ”软件下载“,进入界面,按照要求依次下载里面的软件。如果手机内预先安装了其他版本的微信、支付宝,请卸载后再安装\n ")])])}],!1,null,"63a081ac",null));b.options.__file="Index.vue";s.default=b.exports},HKbb:function(t,s,a){},rOdO:function(t,s,a){"use strict";var e=a("1jgq");a.n(e).a},vF8W:function(t,s,a){}}]); -------------------------------------------------------------------------------- /public/dist/static/css/app.cab2c720.css: -------------------------------------------------------------------------------- 1 | .fade-enter-active,.fade-leave-active{-webkit-transition:opacity .28s;transition:opacity .28s}.fade-enter,.fade-leave-active{opacity:0}.fade-transform-enter-active,.fade-transform-leave-active{-webkit-transition:all .5s;transition:all .5s}.fade-transform-enter{opacity:0;-webkit-transform:translateX(-30px);transform:translateX(-30px)}.fade-transform-leave-to{opacity:0;-webkit-transform:translateX(30px);transform:translateX(30px)}.breadcrumb-enter-active,.breadcrumb-leave-active{-webkit-transition:all .5s;transition:all .5s}.breadcrumb-enter,.breadcrumb-leave-active{opacity:0;-webkit-transform:translateX(20px);transform:translateX(20px)}.breadcrumb-move{-webkit-transition:all .5s;transition:all .5s}.breadcrumb-leave-active{position:absolute}.el-upload input[type=file]{display:none!important}.el-upload__input{display:none}.el-dialog{-webkit-transform:none;transform:none;left:0;position:relative;margin:0 auto}.upload-container .el-upload{width:100%}.upload-container .el-upload .el-upload-dragger{width:100%;height:200px}#app .main-container{min-height:100%;-webkit-transition:margin-left .28s;transition:margin-left .28s;margin-left:180px;position:relative}#app .sidebar-container{-webkit-transition:width .28s;transition:width .28s;width:180px!important;height:100%;position:fixed;font-size:0;top:0;bottom:0;left:0;z-index:1001;overflow:hidden}#app .sidebar-container .horizontal-collapse-transition{-webkit-transition:width 0s ease-in-out,padding-left 0s ease-in-out,padding-right 0s ease-in-out;transition:width 0s ease-in-out,padding-left 0s ease-in-out,padding-right 0s ease-in-out}#app .sidebar-container .scrollbar-wrapper{overflow-x:hidden!important}#app .sidebar-container .scrollbar-wrapper .el-scrollbar__view{height:100%}#app .sidebar-container .el-scrollbar__bar.is-vertical{right:0}#app .sidebar-container .is-horizontal{display:none}#app .sidebar-container a{display:inline-block;width:100%;overflow:hidden}#app .sidebar-container .svg-icon{margin-right:16px}#app .sidebar-container .el-menu{border:none;height:100%;width:100%!important}#app .sidebar-container .el-submenu__title:hover,#app .sidebar-container .submenu-title-noDropdown:hover{background-color:#263445!important}#app .sidebar-container .is-active>.el-submenu__title{color:#f4f4f5!important}#app .sidebar-container .el-submenu .el-menu-item,#app .sidebar-container .nest-menu .el-submenu>.el-submenu__title{min-width:180px!important;background-color:#1f2d3d!important}#app .sidebar-container .el-submenu .el-menu-item:hover,#app .sidebar-container .nest-menu .el-submenu>.el-submenu__title:hover{background-color:#001528!important}#app .hideSidebar .sidebar-container{width:36px!important}#app .hideSidebar .main-container{margin-left:36px}#app .hideSidebar .submenu-title-noDropdown{padding-left:10px!important;position:relative}#app .hideSidebar .submenu-title-noDropdown .el-tooltip{padding:0 10px!important}#app .hideSidebar .el-submenu{overflow:hidden}#app .hideSidebar .el-submenu>.el-submenu__title{padding-left:10px!important}#app .hideSidebar .el-submenu>.el-submenu__title .el-submenu__icon-arrow{display:none}#app .hideSidebar .el-menu--collapse .el-submenu>.el-submenu__title>span{height:0;width:0;overflow:hidden;visibility:hidden;display:inline-block}#app .el-menu--collapse .el-menu .el-submenu{min-width:180px!important}#app .mobile .main-container{margin-left:0}#app .mobile .sidebar-container{-webkit-transition:-webkit-transform .28s;transition:-webkit-transform .28s;transition:transform .28s;transition:transform .28s,-webkit-transform .28s;width:180px!important}#app .mobile.hideSidebar .sidebar-container{pointer-events:none;-webkit-transition-duration:.3s;transition-duration:.3s;-webkit-transform:translate3d(-180px,0,0);transform:translate3d(-180px,0,0)}#app .withoutAnimation .main-container,#app .withoutAnimation .sidebar-container{-webkit-transition:none;transition:none}.el-menu--vertical>.el-menu .svg-icon{margin-right:16px}.el-menu--vertical .el-menu-item:hover,.el-menu--vertical .nest-menu .el-submenu>.el-submenu__title:hover{background-color:#263445!important}.el-menu--vertical>.el-menu--popup{max-height:100vh;overflow-y:auto}.el-menu--vertical>.el-menu--popup::-webkit-scrollbar-track-piece{background:#d3dce6}.el-menu--vertical>.el-menu--popup::-webkit-scrollbar{width:6px}.el-menu--vertical>.el-menu--popup::-webkit-scrollbar-thumb{background:#99a9bf;border-radius:20px}body{height:100%;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;text-rendering:optimizeLegibility;font-family:Helvetica Neue,Helvetica,PingFang SC,Hiragino Sans GB,Microsoft YaHei,Arial,sans-serif}label{font-weight:700}html{-webkit-box-sizing:border-box;box-sizing:border-box}#app,html{height:100%}*,:after,:before{-webkit-box-sizing:inherit;box-sizing:inherit}a,a:active,a:focus,a:hover,div:focus{outline:none}a,a:focus,a:hover{cursor:pointer;color:inherit;text-decoration:none}.clearfix:after{visibility:hidden;display:block;font-size:0;content:" ";clear:both;height:0}.app-main{min-height:100%}.app-container{padding:20px}.cell{display:-webkit-box;display:-ms-flexbox;display:flex}.cell .primary{-webkit-box-flex:1;-ms-flex:1;flex:1}.app-breadcrumb.el-breadcrumb[data-v-6eeb655c]{display:inline-block;font-size:14px;line-height:50px;margin-left:10px}.app-breadcrumb.el-breadcrumb .no-redirect[data-v-6eeb655c]{color:#97a8be;cursor:text}.hamburger[data-v-68efea28]{display:inline-block;cursor:pointer;width:20px;height:20px}.hamburger.is-active[data-v-68efea28]{-webkit-transform:rotate(180deg);transform:rotate(180deg)}.navbar[data-v-9bbe2cc8]{height:50px;line-height:50px;-webkit-box-shadow:0 1px 3px 0 rgba(0,0,0,.12),0 0 3px 0 rgba(0,0,0,.04);box-shadow:0 1px 3px 0 rgba(0,0,0,.12),0 0 3px 0 rgba(0,0,0,.04)}.navbar .hamburger-container[data-v-9bbe2cc8]{line-height:58px;height:50px;float:left;padding:0 10px}.navbar .screenfull[data-v-9bbe2cc8]{position:absolute;right:90px;top:16px;color:red}.navbar .avatar-container[data-v-9bbe2cc8]{height:50px;display:inline-block;position:absolute;right:35px}.navbar .avatar-container .avatar-wrapper[data-v-9bbe2cc8]{cursor:pointer;margin-top:5px;position:relative;line-height:normal;display:-webkit-box;display:-ms-flexbox;display:flex;-webkit-box-align:center;-ms-flex-align:center;align-items:center;height:40px}.navbar .avatar-container .avatar-wrapper .user-avatar[data-v-9bbe2cc8]{width:40px;height:40px;border-radius:50%}.navbar .avatar-container .avatar-wrapper .user-nickname[data-v-9bbe2cc8]{margin:0 10px}.navbar .avatar-container .avatar-wrapper .el-icon-caret-bottom[data-v-9bbe2cc8]{font-size:12px}.contact-us[data-v-5e16de72]{background:#263445;cursor:pointer;font-size:14px;color:#bfcbd9;position:absolute;left:0;right:0;height:70px;line-height:70px;text-align:center;bottom:0}.app-main[data-v-dfcc6c68]{min-height:calc(100vh - 50px);position:relative;overflow:hidden;padding:32px}.app-wrapper[data-v-6ec8cb25]{position:relative;height:100%;width:100%}.app-wrapper[data-v-6ec8cb25]:after{content:"";display:table;clear:both}.app-wrapper.mobile.openSidebar[data-v-6ec8cb25]{position:fixed;top:0}.drawer-bg[data-v-6ec8cb25]{background:#000;opacity:.3;width:100%;top:0;height:100%;position:absolute;z-index:999}.svg-icon[data-v-77a1c50c]{width:1em;height:1em;vertical-align:-.15em;fill:currentColor;overflow:hidden} -------------------------------------------------------------------------------- /utils/statistic.js: -------------------------------------------------------------------------------- 1 | /* 2 | * @Author: hechao 3 | * @Date: 2019-04-01 16:46:36 4 | * @describe: 定时任务,跑统计结果 5 | */ 6 | 7 | const { 8 | sqlTable,sequelize 9 | } = require('../db/index.js') 10 | const Sequelize = require('sequelize') 11 | const schedule = require('node-schedule'); 12 | 13 | Date.prototype.format = function (fmt) { 14 | var o = { 15 | "M+": this.getMonth() + 1, //月份 16 | "d+": this.getDate(), //日 17 | "h+": this.getHours(), //小时 18 | "m+": this.getMinutes(), //分 19 | "s+": this.getSeconds(), //秒 20 | "q+": Math.floor((this.getMonth() + 3) / 3), //季度 21 | "S": this.getMilliseconds() //毫秒 22 | }; 23 | if (/(y+)/.test(fmt)) { 24 | fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length)); 25 | } 26 | for (var k in o) { 27 | if (new RegExp("(" + k + ")").test(fmt)) { 28 | fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length))); 29 | } 30 | } 31 | return fmt; 32 | }; 33 | //今日统计更新 34 | async function day_statistic(){ 35 | // setInterval(()=>{ 36 | try { 37 | let now = new Date(); 38 | let date = now.format("yyyy-MM-dd"); 39 | let day =parseInt(( now - new Date(now.getFullYear().toString()))/(24*60*60*1000))+1; 40 | let result = await sequelize.query(`select * from paychecks where dayofyear(createdAt)=?`,{ 41 | type: Sequelize.QueryTypes.SELECT, 42 | replacements:[day] 43 | }) 44 | let obj={} 45 | result.forEach(item => { 46 | if(obj[item['userid']] || (obj[item['userid']] = [])) { 47 | obj[item['userid']].push(item) 48 | } 49 | }); 50 | console.info(obj) 51 | for(var p in obj){ 52 | let w_count=0,p_count=0,w_total1=0,p_total1=0,w_total2=0,p_total2=0;//微信订单数量,支付宝订单数量,微信已知来源收款总额.. 53 | obj[p].forEach(item=>{ 54 | if(item.appid && item.payway == 1){ 55 | p_count++; 56 | p_total1+=item.amount; 57 | } 58 | if(item.appid && item.payway == 2){ 59 | w_count++; 60 | w_total1+=item.amount; 61 | } 62 | if(!item.appid && item.payway == 1){ 63 | p_total2+=item.amount; 64 | } 65 | if(!item.appid && item.payway == 2){ 66 | w_total2+=item.amount; 67 | } 68 | }) 69 | let statistics = await sequelize.query(`select id from statistic where userid=? and date=?`,{ 70 | type: Sequelize.QueryTypes.SELECT, 71 | replacements:[p,date] 72 | }) 73 | if(statistics.length>0){ 74 | await sequelize.query(`update statistic set w_count=?,p_count=?,w_total1=?,p_total1=?, 75 | w_total2=?,p_total2=? where userid=? and date=?`,{ 76 | type: Sequelize.QueryTypes.UPDATE, 77 | replacements:[w_count,p_count,w_total1,p_total1,w_total2,p_total2,p,date] 78 | }) 79 | }else{ 80 | await sqlTable.statistic.create({ 81 | userid:p, 82 | w_count, 83 | p_count, 84 | w_total1, 85 | p_total1, 86 | w_total2, 87 | p_total2, 88 | date 89 | }) 90 | } 91 | } 92 | } catch (error) { 93 | console.info('error----->',error) 94 | } 95 | 96 | // },10000) 97 | 98 | 99 | } 100 | 101 | //昨天统计 102 | async function yesterday_statistic(){ 103 | try { 104 | let now = new Date(); 105 | // let date = now.format("yyyy-MM-dd"); 106 | let yesterday = new Date(now.getTime()-24*60*60*1000); 107 | let day =parseInt(( yesterday - new Date(yesterday.getFullYear().toString()))/(24*60*60*1000))+1; 108 | let date= yesterday.format("yyyy-MM-dd") 109 | let result = await sequelize.query(`select * from paychecks where dayofyear(createdAt)=?`,{ 110 | type: Sequelize.QueryTypes.SELECT, 111 | replacements:[day] 112 | }) 113 | let users = await sequelize.query(`select id from users`,{ 114 | type: Sequelize.QueryTypes.SELECT, 115 | }) 116 | let obj={} 117 | users.forEach(item=>{ 118 | obj[item['id']]=[] 119 | }) 120 | result.forEach(item => { 121 | if(obj[item['userid']] || (obj[item['userid']] = [])) { 122 | obj[item['userid']].push(item) 123 | } 124 | }); 125 | console.info(obj) 126 | for(var p in obj){ 127 | let w_count=0,p_count=0,w_total1=0,p_total1=0,w_total2=0,p_total2=0;//微信订单数量,支付宝订单数量,微信已知来源收款总额.. 128 | obj[p].forEach(item=>{ 129 | if(item.appid && item.payway == 1){ 130 | p_count++; 131 | p_total1+=item.amount; 132 | } 133 | if(item.appid && item.payway == 2){ 134 | w_count++; 135 | w_total1+=item.amount; 136 | } 137 | if(!item.appid && item.payway == 1){ 138 | p_total2+=item.amount; 139 | } 140 | if(!item.appid && item.payway == 2){ 141 | w_total2+=item.amount; 142 | } 143 | }) 144 | let statistics = await sequelize.query(`select id from statistic where userid=? and date=?`,{ 145 | type: Sequelize.QueryTypes.SELECT, 146 | replacements:[p,date] 147 | }) 148 | if(statistics.length>0){ 149 | await sequelize.query(`update statistic set w_count=?,p_count=?,w_total1=?,p_total1=?, 150 | w_total2=?,p_total2=? where userid=? and date=?`,{ 151 | type: Sequelize.QueryTypes.UPDATE, 152 | replacements:[w_count,p_count,w_total1,p_total1,w_total2,p_total2,p,date] 153 | }) 154 | }else{ 155 | await sqlTable.statistic.create({ 156 | userid:p, 157 | w_count, 158 | p_count, 159 | w_total1, 160 | p_total1, 161 | w_total2, 162 | p_total2, 163 | date 164 | }) 165 | } 166 | } 167 | } catch (error) { 168 | console.info('err---->=',error) 169 | } 170 | } 171 | 172 | 173 | function scheduleCronstyle(){ 174 | // yesterday_statistic(); 175 | // day_statistic() 176 | schedule.scheduleJob('0 0 10 * * *', yesterday_statistic); 177 | // schedule.scheduleJob('0 5 * * * *', day_statistic); 178 | } 179 | module.exports={ 180 | scheduleCronstyle 181 | } -------------------------------------------------------------------------------- /public/dist/static/js/chunk-6929.2461bbf3.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-6929"],{Jvyq:function(s,t){s.exports=""},jNvO:function(s,t,i){"use strict";i.r(t);var n=[function(){var s=this.$createElement,t=this._self._c||s;return t("div",{staticClass:"pic-404"},[t("img",{staticClass:"pic-404__parent",attrs:{src:i("o2sD"),alt:"404"}}),this._v(" "),t("img",{staticClass:"pic-404__child left",attrs:{src:i("Jvyq"),alt:"404"}}),this._v(" "),t("img",{staticClass:"pic-404__child mid",attrs:{src:i("Jvyq"),alt:"404"}}),this._v(" "),t("img",{staticClass:"pic-404__child right",attrs:{src:i("Jvyq"),alt:"404"}})])},function(){var s=this.$createElement,t=this._self._c||s;return t("div",{staticClass:"bullshit__info"},[this._v("版权所有\n "),t("a",{staticClass:"link-type",attrs:{href:"https://wallstreetcn.com",target:"_blank"}},[this._v("华尔街见闻")])])}],r={name:"Page404",computed:{message:function(){return"网管说这个页面你不能进......"}}},l=(i("yGHO"),i("KHd+")),e=Object(l.a)(r,function(){var s=this,t=s.$createElement,i=s._self._c||t;return i("div",{staticClass:"wscn-http404-container"},[i("div",{staticClass:"wscn-http404"},[s._m(0),s._v(" "),i("div",{staticClass:"bullshit"},[i("div",{staticClass:"bullshit__oops"},[s._v("OOPS!")]),s._v(" "),s._m(1),s._v(" "),i("div",{staticClass:"bullshit__headline"},[s._v(s._s(s.message))]),s._v(" "),i("div",{staticClass:"bullshit__info"},[s._v("请检查您输入的网址是否正确,请点击以下按钮返回主页或者发送错误报告")]),s._v(" "),i("a",{staticClass:"bullshit__return-home",attrs:{href:""}},[s._v("返回首页")])])])])},n,!1,null,"9b45c3f8",null);e.options.__file="404.vue";t.default=e.exports},o2sD:function(s,t,i){s.exports=i.p+"static/img/404.a57b6f3.png"},oBSS:function(s,t,i){},yGHO:function(s,t,i){"use strict";var n=i("oBSS");i.n(n).a}}]); -------------------------------------------------------------------------------- /public/dist/static/js/chunk-fc3b.827b4c52.js: -------------------------------------------------------------------------------- 1 | (window.webpackJsonp=window.webpackJsonp||[]).push([["chunk-fc3b"],{"2uYv":function(e,t,s){"use strict";var a=s("UDD5");s.n(a).a},Timz:function(e,t,s){"use strict";var a=s("qvc3");s.n(a).a},UDD5:function(e,t,s){},qeMi:function(e,t,s){"use strict";s.r(t);var a=s("14Xm"),r=s.n(a),n=s("D3Ub"),o=s.n(n),i=s("gjeX"),l=s.n(i),c=s("wk8/"),m=s("QqA1"),p=s.n(m),u={data:function(){return{mode:"mobile",smsCount:-1,mobileForm:{mobile_phone:"",password:"",code:""},emailCount:-1,emailLoading:!1,emailForm:{email:"",password:"",code:""},emailRules:{email:[{required:!0,message:"请输入邮箱",trigger:"blur"},{type:"email",message:"请输入正确的邮箱地址",trigger:"blur"}],code:[{required:!0,message:"请输入验证码",trigger:"blur"}],password:[{required:!0,message:"请输入密码",trigger:"blur"},{min:6,message:"密码长度不低于6位",trigger:"blur"}]},loading:!1,pwdType:"password",redirect:void 0}},methods:{showPwd:function(){"password"===this.pwdType?this.pwdType="":this.pwdType="password"},mobileSubmit:function(){var e=this;return o()(r.a.mark(function t(){return r.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(/\d{11}/.test(e.mobileForm.mobile_phone)){t.next=3;break}return e.$message({type:"warning",message:"请输入正确的手机号"}),t.abrupt("return");case 3:if(e.mobileForm.code){t.next=6;break}return e.$message({type:"warning",message:"请输入验证码"}),t.abrupt("return");case 6:if(e.mobileForm.password){t.next=9;break}return e.$message({type:"warning",message:"请输入密码"}),t.abrupt("return");case 9:return t.next=11,Object(c.f)({mobile_phone:e.mobileForm.mobile_phone,code:e.mobileForm.code,password:l()(e.mobileForm.password)});case 11:if(null!==t.sent){t.next=14;break}return t.abrupt("return");case 14:e.$message({type:"success",message:"注册成功,请登录"}),e.$router.push({path:"/login"});case 16:case"end":return t.stop()}},t,e)}))()},emailSubmit:function(){var e=this;return o()(r.a.mark(function t(){return r.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:e.$refs.emailForm.validate(function(){var t=o()(r.a.mark(function t(s){var a,n;return r.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(s){t.next=2;break}return t.abrupt("return");case 2:return e.emailLoading=!0,(a=p()({},e.emailForm)).password=l()(a.password),t.next=7,Object(c.f)(a);case 7:if(n=t.sent,e.emailLoading=!1,null!==n){t.next=11;break}return t.abrupt("return");case 11:e.$message({type:"success",message:"注册成功,请登录"}),e.$router.push({path:"/login"});case 13:case"end":return t.stop()}},t,e)}));return function(e){return t.apply(this,arguments)}}());case 1:case"end":return t.stop()}},t,e)}))()},sendSMS:function(){var e=this;return o()(r.a.mark(function t(){var s;return r.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(!(e.smsCount>=0)){t.next=2;break}return t.abrupt("return");case 2:if(/\d{11}/.test(e.mobileForm.mobile_phone)){t.next=5;break}return e.$message({type:"warning",message:"请输入正确的手机号"}),t.abrupt("return");case 5:return t.next=7,Object(c.h)({mobile_phone:e.mobileForm.mobile_phone});case 7:if(null!==t.sent){t.next=10;break}return t.abrupt("return");case 10:e.$message({type:"success",message:"短信发送成功"}),e.smsCount=60,s=setInterval(function(){e.smsCount<=0&&clearInterval(s),e.smsCount-=1},1e3);case 13:case"end":return t.stop()}},t,e)}))()},sendMail:function(){var e=this;return o()(r.a.mark(function t(){var s;return r.a.wrap(function(t){for(;;)switch(t.prev=t.next){case 0:if(!(e.emailCount>=0)){t.next=2;break}return t.abrupt("return");case 2:return t.next=4,Object(c.g)({email:e.emailForm.email});case 4:if(null!==t.sent){t.next=7;break}return t.abrupt("return");case 7:e.$message({type:"success",message:"发送成功"}),e.emailCount=60,s=setInterval(function(){e.emailCount<=0&&clearInterval(s),e.emailCount-=1},1e3);case 10:case"end":return t.stop()}},t,e)}))()}}},d=(s("Timz"),s("2uYv"),s("KHd+")),v=Object(d.a)(u,function(){var e=this,t=e.$createElement,s=e._self._c||t;return s("div",{staticClass:"login-container"},["mobile"==e.mode?s("el-form",{ref:"mobileForm",staticClass:"login-form",attrs:{model:e.mobileForm,"auto-complete":"on","label-position":"left"}},[s("h3",{staticClass:"title"},[e._v("手机注册")]),e._v(" "),s("el-form-item",{attrs:{prop:"phone"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"phone"}})],1),e._v(" "),s("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"请输入手机号"},model:{value:e.mobileForm.mobile_phone,callback:function(t){e.$set(e.mobileForm,"mobile_phone",t)},expression:"mobileForm.mobile_phone"}})],1),e._v(" "),s("el-form-item",{attrs:{prop:"phone"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"verifycode"}})],1),e._v(" "),s("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"验证码"},model:{value:e.mobileForm.code,callback:function(t){e.$set(e.mobileForm,"code",t)},expression:"mobileForm.code"}}),e._v(" "),s("div",{staticClass:"verify-btn",on:{click:e.sendSMS}},[e._v(e._s(e.smsCount>=0?e.smsCount:"获取验证码"))])],1),e._v(" "),s("el-form-item",{attrs:{prop:"password"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"password"}})],1),e._v(" "),s("el-input",{attrs:{type:e.pwdType,name:"password","auto-complete":"on",placeholder:"密码"},nativeOn:{keyup:function(t){return"button"in t||!e._k(t.keyCode,"enter",13,t.key,"Enter")?e.handleLogin(t):null}},model:{value:e.mobileForm.password,callback:function(t){e.$set(e.mobileForm,"password",t)},expression:"mobileForm.password"}}),e._v(" "),s("span",{staticClass:"show-pwd",on:{click:e.showPwd}},[s("svg-icon",{attrs:{"icon-class":"password"===e.pwdType?"eye":"eye-open"}})],1)],1),e._v(" "),s("el-form-item",[s("el-button",{staticStyle:{width:"100%"},attrs:{loading:e.loading,type:"primary"},on:{click:e.mobileSubmit}},[e._v("\n 注册\n ")])],1),e._v(" "),s("div",{staticClass:"tips"},[s("a",{on:{click:function(t){e.mode="email"}}},[e._v("邮箱注册")]),e._v(" "),s("span",{staticClass:"tips-line"},[e._v("|")]),e._v(" "),s("router-link",{staticClass:"tips-link",attrs:{to:{path:"/login"},href:""}},[e._v("登录")])],1)],1):s("el-form",{ref:"emailForm",staticClass:"login-form",attrs:{model:e.emailForm,rules:e.emailRules,"auto-complete":"on","label-position":"left"}},[s("h3",{staticClass:"title"},[e._v("邮箱注册")]),e._v(" "),s("el-form-item",{attrs:{prop:"email"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"email"}})],1),e._v(" "),s("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"请输入邮箱"},model:{value:e.emailForm.email,callback:function(t){e.$set(e.emailForm,"email",t)},expression:"emailForm.email"}})],1),e._v(" "),s("el-form-item",{attrs:{prop:"code"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"verifycode"}})],1),e._v(" "),s("el-input",{attrs:{type:"text","auto-complete":"on",placeholder:"验证码"},model:{value:e.emailForm.code,callback:function(t){e.$set(e.emailForm,"code",t)},expression:"emailForm.code"}}),e._v(" "),s("div",{staticClass:"verify-btn",on:{click:e.sendMail}},[e._v(e._s(e.emailCount>=0?e.emailCount:"获取验证码"))])],1),e._v(" "),s("el-form-item",{attrs:{prop:"password"}},[s("span",{staticClass:"svg-container"},[s("svg-icon",{attrs:{"icon-class":"password"}})],1),e._v(" "),s("el-input",{attrs:{type:e.pwdType,name:"password","auto-complete":"on",placeholder:"密码"},nativeOn:{keyup:function(t){return"button"in t||!e._k(t.keyCode,"enter",13,t.key,"Enter")?e.emailSubmit(t):null}},model:{value:e.emailForm.password,callback:function(t){e.$set(e.emailForm,"password",t)},expression:"emailForm.password"}}),e._v(" "),s("span",{staticClass:"show-pwd",on:{click:e.showPwd}},[s("svg-icon",{attrs:{"icon-class":"password"===e.pwdType?"eye":"eye-open"}})],1)],1),e._v(" "),s("el-form-item",[s("el-button",{staticStyle:{width:"100%"},attrs:{loading:e.emailLoading,type:"primary"},on:{click:e.emailSubmit}},[e._v("\n 注册\n ")])],1),e._v(" "),s("div",{staticClass:"tips"},[s("a",{on:{click:function(t){e.mode="mobile"}}},[e._v("手机注册")]),e._v(" "),s("span",{staticClass:"tips-line"},[e._v("|")]),e._v(" "),s("router-link",{staticClass:"tips-link",attrs:{to:{path:"/login"},href:""}},[e._v("登录")])],1)],1)],1)},[],!1,null,"b89682e8",null);v.options.__file="Register.vue";t.default=v.exports},qvc3:function(e,t,s){}}]); -------------------------------------------------------------------------------- /public/css/normalize.css: -------------------------------------------------------------------------------- 1 | /*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */ 2 | 3 | /* Document 4 | ========================================================================== */ 5 | 6 | /** 7 | * 1. Correct the line height in all browsers. 8 | * 2. Prevent adjustments of font size after orientation changes in iOS. 9 | */ 10 | 11 | html { 12 | line-height: 1.15; /* 1 */ 13 | -webkit-text-size-adjust: 100%; /* 2 */ 14 | } 15 | 16 | /* Sections 17 | ========================================================================== */ 18 | 19 | /** 20 | * Remove the margin in all browsers. 21 | */ 22 | 23 | body { 24 | margin: 0; 25 | } 26 | 27 | /** 28 | * Render the `main` element consistently in IE. 29 | */ 30 | 31 | main { 32 | display: block; 33 | } 34 | 35 | /** 36 | * Correct the font size and margin on `h1` elements within `section` and 37 | * `article` contexts in Chrome, Firefox, and Safari. 38 | */ 39 | 40 | h1 { 41 | font-size: 2em; 42 | margin: 0.67em 0; 43 | } 44 | 45 | /* Grouping content 46 | ========================================================================== */ 47 | 48 | /** 49 | * 1. Add the correct box sizing in Firefox. 50 | * 2. Show the overflow in Edge and IE. 51 | */ 52 | 53 | hr { 54 | box-sizing: content-box; /* 1 */ 55 | height: 0; /* 1 */ 56 | overflow: visible; /* 2 */ 57 | } 58 | 59 | /** 60 | * 1. Correct the inheritance and scaling of font size in all browsers. 61 | * 2. Correct the odd `em` font sizing in all browsers. 62 | */ 63 | 64 | pre { 65 | font-family: monospace, monospace; /* 1 */ 66 | font-size: 1em; /* 2 */ 67 | } 68 | 69 | /* Text-level semantics 70 | ========================================================================== */ 71 | 72 | /** 73 | * Remove the gray background on active links in IE 10. 74 | */ 75 | 76 | a { 77 | background-color: transparent; 78 | } 79 | 80 | /** 81 | * 1. Remove the bottom border in Chrome 57- 82 | * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari. 83 | */ 84 | 85 | abbr[title] { 86 | border-bottom: none; /* 1 */ 87 | text-decoration: underline; /* 2 */ 88 | text-decoration: underline dotted; /* 2 */ 89 | } 90 | 91 | /** 92 | * Add the correct font weight in Chrome, Edge, and Safari. 93 | */ 94 | 95 | b, 96 | strong { 97 | font-weight: bolder; 98 | } 99 | 100 | /** 101 | * 1. Correct the inheritance and scaling of font size in all browsers. 102 | * 2. Correct the odd `em` font sizing in all browsers. 103 | */ 104 | 105 | code, 106 | kbd, 107 | samp { 108 | font-family: monospace, monospace; /* 1 */ 109 | font-size: 1em; /* 2 */ 110 | } 111 | 112 | /** 113 | * Add the correct font size in all browsers. 114 | */ 115 | 116 | small { 117 | font-size: 80%; 118 | } 119 | 120 | /** 121 | * Prevent `sub` and `sup` elements from affecting the line height in 122 | * all browsers. 123 | */ 124 | 125 | sub, 126 | sup { 127 | font-size: 75%; 128 | line-height: 0; 129 | position: relative; 130 | vertical-align: baseline; 131 | } 132 | 133 | sub { 134 | bottom: -0.25em; 135 | } 136 | 137 | sup { 138 | top: -0.5em; 139 | } 140 | 141 | /* Embedded content 142 | ========================================================================== */ 143 | 144 | /** 145 | * Remove the border on images inside links in IE 10. 146 | */ 147 | 148 | img { 149 | border-style: none; 150 | } 151 | 152 | /* Forms 153 | ========================================================================== */ 154 | 155 | /** 156 | * 1. Change the font styles in all browsers. 157 | * 2. Remove the margin in Firefox and Safari. 158 | */ 159 | 160 | button, 161 | input, 162 | optgroup, 163 | select, 164 | textarea { 165 | font-family: inherit; /* 1 */ 166 | font-size: 100%; /* 1 */ 167 | line-height: 1.15; /* 1 */ 168 | margin: 0; /* 2 */ 169 | } 170 | 171 | /** 172 | * Show the overflow in IE. 173 | * 1. Show the overflow in Edge. 174 | */ 175 | 176 | button, 177 | input { /* 1 */ 178 | overflow: visible; 179 | } 180 | 181 | /** 182 | * Remove the inheritance of text transform in Edge, Firefox, and IE. 183 | * 1. Remove the inheritance of text transform in Firefox. 184 | */ 185 | 186 | button, 187 | select { /* 1 */ 188 | text-transform: none; 189 | } 190 | 191 | /** 192 | * Correct the inability to style clickable types in iOS and Safari. 193 | */ 194 | 195 | button, 196 | [type="button"], 197 | [type="reset"], 198 | [type="submit"] { 199 | -webkit-appearance: button; 200 | } 201 | 202 | /** 203 | * Remove the inner border and padding in Firefox. 204 | */ 205 | 206 | button::-moz-focus-inner, 207 | [type="button"]::-moz-focus-inner, 208 | [type="reset"]::-moz-focus-inner, 209 | [type="submit"]::-moz-focus-inner { 210 | border-style: none; 211 | padding: 0; 212 | } 213 | 214 | /** 215 | * Restore the focus styles unset by the previous rule. 216 | */ 217 | 218 | button:-moz-focusring, 219 | [type="button"]:-moz-focusring, 220 | [type="reset"]:-moz-focusring, 221 | [type="submit"]:-moz-focusring { 222 | outline: 1px dotted ButtonText; 223 | } 224 | 225 | /** 226 | * Correct the padding in Firefox. 227 | */ 228 | 229 | fieldset { 230 | padding: 0.35em 0.75em 0.625em; 231 | } 232 | 233 | /** 234 | * 1. Correct the text wrapping in Edge and IE. 235 | * 2. Correct the color inheritance from `fieldset` elements in IE. 236 | * 3. Remove the padding so developers are not caught out when they zero out 237 | * `fieldset` elements in all browsers. 238 | */ 239 | 240 | legend { 241 | box-sizing: border-box; /* 1 */ 242 | color: inherit; /* 2 */ 243 | display: table; /* 1 */ 244 | max-width: 100%; /* 1 */ 245 | padding: 0; /* 3 */ 246 | white-space: normal; /* 1 */ 247 | } 248 | 249 | /** 250 | * Add the correct vertical alignment in Chrome, Firefox, and Opera. 251 | */ 252 | 253 | progress { 254 | vertical-align: baseline; 255 | } 256 | 257 | /** 258 | * Remove the default vertical scrollbar in IE 10+. 259 | */ 260 | 261 | textarea { 262 | overflow: auto; 263 | } 264 | 265 | /** 266 | * 1. Add the correct box sizing in IE 10. 267 | * 2. Remove the padding in IE 10. 268 | */ 269 | 270 | [type="checkbox"], 271 | [type="radio"] { 272 | box-sizing: border-box; /* 1 */ 273 | padding: 0; /* 2 */ 274 | } 275 | 276 | /** 277 | * Correct the cursor style of increment and decrement buttons in Chrome. 278 | */ 279 | 280 | [type="number"]::-webkit-inner-spin-button, 281 | [type="number"]::-webkit-outer-spin-button { 282 | height: auto; 283 | } 284 | 285 | /** 286 | * 1. Correct the odd appearance in Chrome and Safari. 287 | * 2. Correct the outline style in Safari. 288 | */ 289 | 290 | [type="search"] { 291 | -webkit-appearance: textfield; /* 1 */ 292 | outline-offset: -2px; /* 2 */ 293 | } 294 | 295 | /** 296 | * Remove the inner padding in Chrome and Safari on macOS. 297 | */ 298 | 299 | [type="search"]::-webkit-search-decoration { 300 | -webkit-appearance: none; 301 | } 302 | 303 | /** 304 | * 1. Correct the inability to style clickable types in iOS and Safari. 305 | * 2. Change font properties to `inherit` in Safari. 306 | */ 307 | 308 | ::-webkit-file-upload-button { 309 | -webkit-appearance: button; /* 1 */ 310 | font: inherit; /* 2 */ 311 | } 312 | 313 | /* Interactive 314 | ========================================================================== */ 315 | 316 | /* 317 | * Add the correct display in Edge, IE 10+, and Firefox. 318 | */ 319 | 320 | details { 321 | display: block; 322 | } 323 | 324 | /* 325 | * Add the correct display in all browsers. 326 | */ 327 | 328 | summary { 329 | display: list-item; 330 | } 331 | 332 | /* Misc 333 | ========================================================================== */ 334 | 335 | /** 336 | * Add the correct display in IE 10+. 337 | */ 338 | 339 | template { 340 | display: none; 341 | } 342 | 343 | /** 344 | * Add the correct display in IE 10. 345 | */ 346 | 347 | [hidden] { 348 | display: none; 349 | } -------------------------------------------------------------------------------- /public/font/iconfont.js: -------------------------------------------------------------------------------- 1 | !function(l){var t,o='',e=(t=document.getElementsByTagName("script"))[t.length-1].getAttribute("data-injectcss");if(e&&!l.__iconfont__svg__cssinject__){l.__iconfont__svg__cssinject__=!0;try{document.write("")}catch(t){console&&console.log(t)}}!function(t){if(document.addEventListener)if(~["complete","loaded","interactive"].indexOf(document.readyState))setTimeout(t,0);else{var e=function(){document.removeEventListener("DOMContentLoaded",e,!1),t()};document.addEventListener("DOMContentLoaded",e,!1)}else document.attachEvent&&(o=t,n=l.document,i=!1,c=function(){i||(i=!0,o())},(s=function(){try{n.documentElement.doScroll("left")}catch(t){return void setTimeout(s,50)}c()})(),n.onreadystatechange=function(){"complete"==n.readyState&&(n.onreadystatechange=null,c())});var o,n,i,c,s}(function(){var t,e;(t=document.createElement("div")).innerHTML=o,o=null,(e=t.getElementsByTagName("svg")[0])&&(e.setAttribute("aria-hidden","true"),e.style.position="absolute",e.style.width=0,e.style.height=0,e.style.overflow="hidden",function(t,e){e.firstChild?function(t,e){e.parentNode.insertBefore(t,e)}(t,e.firstChild):e.appendChild(t)}(e,document.body))})}(window); --------------------------------------------------------------------------------