reg.js 8.99 KB
/**
 * 注册控制器
 */
'use strict';
const _ = require('lodash');
const Promise = require('bluebird');
const passportHelper = require('../models/passport-helper');
const regService = require('../models/reg-service');
const userService = require('../models/user-service');
const loginService = require('../models/login-service');
const config = require('../../../config/common');
const simpleHeaderModel = require('../../../doraemon/models/simple-header');
let helpers = global.yoho.helpers;
let cache = global.yoho.cache;
let cookie = global.yoho.cookie;

/**
 * 检查图形验证码
 */
let checkCode = (req, res, next) => {
    let verifyCode = _.trim(req.body.verifyCode);

    if (verifyCode !== req.session.captcha) {
        return res.json({
            code: 400,
            message: '验证码错误'
        });
    }
    next();
};

/**
 * 检查手机格式
 */
let checkMobileMiddleware = (req, res, next) => {
    let mobile = req.body.mobile;
    let area = req.body.area;

    if (!(_.toNumber(mobile) && _.toNumber(area))) {
        return res.json({
            code: 400,
            message: '手机号码格式不正确'
        });
    }
    next();
};

/**
 * 检查密码格式
 */
let checkPassword = (req, res, next) => {
    /* 判断密码是否符合规则 */
    let password = req.body.password;

    if (!helpers.verifyPassword(password)) {
        return res.json({
            code: 400,
            message: '密码不正确'
        });
    }
    next();
};

/**
 * 注册首页
 */
let index = (req, res, next) => {
    // 设置注册有效时间30分钟, 防机器刷
    req.session.REG_EXPIRE_ = Date.now() + 1800000;

    let refer = req.query.refer;
    let mobile = req.query.mobile;
    let area = req.query.area;

    refer && res.cookie('refer', encodeURI(refer), {
        domain: config.cookieDomain
    });

    regService.getRegData().then((result) => {
        res.render('reg/index', {
            simpleHeader: simpleHeaderModel.setSimpleHeaderData(),
            title: '新用户注册',
            passport: {
                region: passportHelper.getCountry(),
                location: area || '+86',
                defaultMobile: mobile,
                captchaUrl: helpers.urlFormat('/passport/imagesNode', {t: Date.now()}),
                itemUrl: helpers.urlFormat('/help/detail', {id: 103, contId: 197}),
                referUrl: refer,
                loginUrl: helpers.urlFormat('/signin.html', {refer: refer}),
                coverHref: result.url,
                coverImg: result.img,
                regBtnText: '立即注册'
            },
            module: 'passport',
            page: 'reg'
        });
    }).catch(next);
};

/**
 * 检查手机号码是否注册过
 * @Deprecated 因为安全原因,请不要使用该接口
 */
let checkMobile = (req, res, next) => {
    let data = {
        code: 400
    };
    let mobile = req.body.mobile;
    let area = req.body.area || '86';

    // 判断手机号是否检查超过指定次数
    let regCheckKey = `regCheckMobileNum_${area}-${mobile}`;

    cache.get(regCheckKey).then((checkNum) => {
        checkNum = +(checkNum || 0);

        return cache.set(regCheckKey, ++checkNum).then(() => {
            return checkNum;
        });
    }).then((checkNum) => {
        if (checkNum > 500) {
            data.message = '检查次数太多';
            return res.json(data);
        }

        // 判断用户是否存在
        return userService.findByMobileAsync(area, mobile).then((user) => {
            if (!_.isEmpty(user)) {
                data.message = '手机号码已经存在';
                return res.json(data);
            }

            data.code = 200;
            return res.json(data);
        });
    }).catch(next);
};

/**
 * checkMobile的伪装,始终成功。
 */
let fakeCheckMobile = (req, res) => {
    return res.json({
        code: 200
    });
};

/**
 * 图形验证码校验
 */
let picCaptcha = (req, res) => {
    // 图形验证码已经在中间件校验,如果走到这儿,说明通过校验
    return res.json({
        code: 200,
        message: '验证码正确'
    });
};

/**
 * 发送验证码
 */
let sendBindMsg = (req, res, next) => {
    Promise.coroutine(function*() {
        let data = {
            code: 400,
            message: '',
            data: ''
        };

        let mobile = req.body.mobile;
        let area = req.body.area;

        // 校验是否发送过多
        let sendCodeKey = `send_code_${area}_${mobile}`;
        let sendCodeTimes = yield cache.get(sendCodeKey);

        sendCodeTimes = +(sendCodeTimes || 0);
        if (sendCodeTimes >= 5) {
            data.message = sendCodeTimes >= 10 ? '您已多次提交验证码,请尽快联系客服解决' : '您收到的验证码短信已超过本日限定最多次数,请您耐心等待';
            return res.json(data);
        }

        /* 向手机发送注册验证码 */
        let result = yield regService.sendCodeToMobile(area, mobile);

        req.session.type = 'register-step1';

        return cache.set(sendCodeKey, sendCodeTimes + 1, 3600).then(() => {
            if (result.code) {
                return res.json(result);
            } else {
                data.message = '发送失败';
                return res.json(data);
            }
        });
    })().catch(next);
};

/**
 * 短信验证码校验
 */
let msgCaptcha = (req, res, next) => {
    let data = {
        code: 400,
        message: '',
        data: ''
    };
    let area = req.body.area;
    let mobile = req.body.mobile;
    let code = req.body.code; // 短信验证码

    regService.validMobileCode(area, mobile, code).then((result) => {
        if (result.code) {
            return res.json(result);
        } else {
            data.message = '验证码错误';
            return res.json(data);
        }
    }).catch(next);
};

/**
 * 注册接口
 */
let mobileRegister = (req, res, next) => {
    Promise.coroutine(function*() {
        let data = {
            code: 400,
            message: '',
            data: ''
        };

        if (req.session.type !== 'register-step1') {
            return res.json(data);
        }

        /* 判断是否是有效的注册方式,防注册机刷 */
        let regExpireTime = req.session.REG_EXPIRE_;

        if (!regExpireTime || regExpireTime < Date.now()) {
            data.message = '注册超时';
            return res.json(data);
        }

        /* IP仅允许点击注册500次/时 */
        let ipKey = 'ip_register_' + req.ip;
        let ipTimes = yield cache.get(ipKey);

        if (!ipTimes) {
            ipTimes = 0;
        } else {
            ipTimes = +ipTimes;
        }

        if (ipTimes >= 500) {
            data.message = '由于你IP受限无法注册';
            return res.json(data);
        }

        yield cache.set(ipKey, ipTimes + 1, 3600);

        let area = req.body.area;
        let mobile = req.body.mobile;
        let code = req.body.code; // 短信验证码
        let password = req.body.password;
        let inviteCode = req.body.inviteCode;
        let clientIp = req.yoho.clientIp;
        let result = yield regService.validMobileCode(area, mobile, code); // 验证注册的标识码是否有效

        if (!result.code || result.code !== 200) {
            data.message = '验证码错误';
            return res.json(data);
        }

        /* 手机注册: 调用注册接口*/
        let regResult = yield regService.regMobileAes(
            area, mobile, password, code, cookie.getShoppingKey(req), inviteCode, clientIp
        );

        if (!regResult.code || regResult.code !== 200) {
            data.message = '注册失败';
            return res.json(data);
        }

        // 清除 session type
        req.session.type = '';

        return loginService.syncUserSession(regResult.data, req, res).then(() => {
            return res.json({
                code: 200,
                message: '注册成功',
                data: {
                    href: helpers.urlFormat('/passport/reg/success', {
                        next: cookie.getRefer(req, '/?go=1'),
                        goShoppingUrl: config.siteUrl
                    })
                }
            });
        });
    })().catch(next);
};

let success = (req, res, next) => {
    let goUrl = req.query.next || config.siteUrl;
    let goShoppingUrl = req.query.goShoppingUrl || config.siteUrl;

    regService.getRegData().then((result) => {
        res.render('reg/success', {
            title: '注册成功',
            module: 'passport',
            page: 'reg-success',
            simpleHeader: simpleHeaderModel.setSimpleHeaderData(),
            passport: {
                goUrl: goUrl,
                goShoppong: goShoppingUrl,
                coverHref: result.url,
                coverImg: result.img
            }
        });
    }).catch(next);
};

module.exports = {
    checkCode,
    checkMobileMiddleware,
    checkPassword,
    index,
    success,
    checkMobile,
    picCaptcha,
    sendBindMsg,
    msgCaptcha,
    mobileRegister,
    fakeCheckMobile
};