设为首页 收藏本站
查看: 1009|回复: 0

[经验分享] 公司项目NODEJS实践0.3[ mongo / session ...]

[复制链接]

尚未签到

发表于 2016-12-2 09:42:31 | 显示全部楼层 |阅读模式
 
 
  http://www.upopen.cn
一、前言
        书接上回,我们搭建了WEB服务端路由、模板等功能,完成了register 通过ajax与后端的通信,今天主要完成数据与mongodb的存取,实现注册 / 登录 / 退出功能
        DEMO GIT https://github.com/xiaolulu/mynodejs.git
 
二、db操作
        上一节我们已经安装过了mongo,本节主要是对其操作
 
        1、mongoose
        nodejs对 mongo的操作,我们使用 mongoose库
        在package.json添加mongoose,并npm install
        使用参考http://www.upopen.cn/article/info?id=559688a7f0e6e0665b000004
        
        2、/root/web目录下创建 db/sql.js,用于对mongoose的操作,添加如下代码
        ***********************************************************************************
        var mongoose = require( 'mongoose'  ); //引用模块
        mongoose.connect( 'mongodb://127.0.0.1/myDB', function( err ){
            //连接mongoose,连接本地127.0.0.1,mongo的默认端口是 27017
            if( !err ){
                console.log( 'DB == connect to mongodb' );
            } else {
                throw err;
            }
        } );
        
        var Schema = mongoose.Schema;
        
        var UserSchema = new Schema({ //创建User表模型,数据可据需求增减
            username: String,
            password: String,
            email: String,
            disabled: Boolean, //后面加注册后的邮件验证功能
            date: Date,
            power: Number  //后面会用到权限功能  
        });
        
        var UserModel = mongoose.model( 'User', UserSchema, 'User' );
        
        function initData( data, db ){ //对参数做预处理,以防出现不合要求的参数,后面这块会做扩展
            var query = {};
            for( var key in data ){
                if( db.tree[ key ] ){
                    query[ key ] = data[ key ];
                }
            }
            return query;
        }
        
        function addUser( data, cb ){ //增加用户
            data = initData( data, UserSchema );
            ( new UserModel( data )).save( function( err, doc  ){
                cb( err, doc );
            })
        }
        function findUser( data, cb ){ //查找用户
            data = initData( data, UserSchema );
            UserModel.findOne( data ).exec( function( err, doc ){
                cb( err, doc );
            })
        }
        
        module.exports = {
            addUser: addUser,
            findUser: findUser
        }
        ##########################################################################
        
        3、在 /root/web下新建controls/user.js,用于处理路由与数据存储的中间逻辑,添加代码如下
        ****************************************************************************
        var db = require( '../db/sql' ); //添加前面定义的db操作模块
        
        function addUser( req, res ){ //增加用户
            var data = req.body; //post过来的数据在req.body里,get过来的数据在req.query里
            data.date = new Date(); //数据里增加时间
            db.addUser( data, function( err, doc ){
            if( !err ){
                res.send( { code: 0, msg: 'add User Success', data: doc } );
                //对查询结果返回,返回格式统一为 {code: 返回码, msg: 返回描述, data: 返回值}
                }
            })
        }
        
        function findUser( req, res ){ //查找用户
            var data = req.body;
            db.addUser( data, function( err, doc ){
                if( !err ){
                    res.send(  { code: 0, msg: 'find User Success', data: doc }  );
                }
            })
        }
        
        module.exports = {
            addUser: addUser,
            findUser: findUser
        }
        #########################################################################
        
        4、修改上节在/root/web/routes/issue.js定义的register函数改为
        **********************************************************************************
        function registerUser( req, res ){
            user.addUser( req, res );
        }
        #########################################################
        并增加 /web/controls/user.js的引用
        再用node-dev启动项目,访问register,提交表单,可以看到返回成功,至此我们注册用户成功
 
        5、使用shell mongo
        打开shell,执行mongo,打开mongo终端
        执行use myDB //切换到myDB数据库
        执行db.User.find().pretty() //可以看到刚才我们新增的数据
        
        6、增加登录查询
        在/root/web/views/issue下新建login.ejs,添加登录form。
        在/root/static/module/issue 下新建 login的 js/css/img 静态文件,添加登录请求,如注册。
        我们在db操作/db.sql.js里及业务处理/controls/user.js已经增加查询方法,只需在/routes/index.js 及 issue里增加 登录查询即可,这里不在列出,参考 register 流程即可。
 
        7、session
        上一步,我们走通了注册和登录查询功能,然后登录的目的是为了根据用户登录与否判断是否具备访问某些页面的权限。
        这里简单说下session(后面再单独详解):网站保存信息或状态,页面端常用的是cookie,而对应服务端是session,登录状态需要保存服务端以防伪造页面端。而http(后面再单独详解)是无状态的,为使页面端与服务端关联,生成session时,同时会在cookie里写入一个对应的id值,如 session_sid,每次页面与服务器的交互都会自动带上cookie,服务器端会据这个id查找是否有对应的session保存,从而形成状态保存。
        这里我们也使用第三方库 express-session,安装同 mongoose
        在 /root/web/routes/index.js里引入  express-session,并新增app.use如下
        *********************************************************
        var issue = require( './issue' ),
              session = require( 'express-session' ); //添加 express-session引用
        exports.all = function( app ){
            app.use( session({ //配置session
                resave: false,
                saveUninitialized: false,
                secret: 'upopen'
            }))
            app.use( function( req, res, next){
                if( req.path != '/login' && !req.session.status ){ //判断session状态是否是true
                    res.redirect('/login'); //不是则跳转到登录页
                } else {
                    next(); //为true,则继续执行其请求
                }
            })
            app.get( '/', function( req, res ){
                issue.index( req, res );
            });
        …
        
        上面的代码是对所有的页面做了限制,都必须是登录的状态才能访问,所以前端要先有注册成功的账号,这样的权限设置当然是不对的,我们只是做下测试。
        打开/root/web/controls/user.js,在findUser函数下新增如下代码
        ********************************************************************
        …
        function findUser( req, res ){
        var data = req.body;
            db.findUser( data, function( err, doc ){
                if( !err ){
                    if( doc ){ //如果登录有查找结果
                        req.session.status = true;        //则session里记录状态为true
                    }
                    res.send(  { code: 0, msg: 'find User Success', data: doc }  );
                }
            })
        }
        …
        ####################################################
        
        再打开站点测试,发现,无论是访问index 还是 register都是自动跳转login,登录信息成功后,index和register都可以访问了,查看cookies里的信息,会看到 自动生成的connect.sid(名称可能不同),就是保存关联session的。手动删除connect.sid,所有页面又会都跳转到 login。
        执行退出命令,只要设置req.session.status = false,即可。
        
        8、权限设置
        上一步,增加了页面请求对权限登录的验证,但是验证只是针对某些页面的,我们把需要验证的路径罗列下来。
        在/web/config下新增 privilege.js,用来罗列权限表,我们新增几个用户管理页面,用来表示权限需要
        *****************************************************************
        module.exports = {
            '/user/center' : 1,
            '/user/info': 1,
            '/user/blog': 1
        }
        ##################################################        
        修改 /root/web/routes/index.js,引入/web/config/privilege.js,修改验证是否登录的app.use
        *****************************************
        …
        var privilege = require( '../config/privilege' );
        …
        app.use( function( req, res, next){
            if( privilege[ req.path ] && req.path != '/login' && !req.session.status ){
                //privilege[ req.path ] 判断该路径是否需要登录权限
                if( req.method == 'GET' ){ //如果是get请求
                    res.redirect('/login'); 则执行跳转
                } else { //其它请求,基本都是POST,是不能直接redirect
                    res.send( { code: 1001, msg: 'need you to log in'}); //则返回错误码,提示需要登录
                }
            } else {
                next();
            }
        })
        …
        ##########################################################################
 
        在web/routes 、web/views、status/module,新增对应的用户页面,user/blog,user/info,user/center,添加时注意文件夹的命名及细分。
        清除cookie,再访问 index 、register都是可以的,而user下的三个页面都需要登录。
        
        9、页面登录状态显示 及 退出
        在/root/static/public/js 下新建 all.js,用于所有页面都要执行js
        将页面据cookies判断登录状态的js写入,以便页面导航上 显示 登录 或 退出,通过requirejs,在每个页面引入。
        退出,即给退出链接加一个get请求,/logout,在/routes/index.js里,添加logout
        **************************************************************************************
         ...
        app.get('/logout', function( req, res ){
            req.session.status = false; //设置session状态为未登录
            res.setHeader("Set-Cookie","username=null;" ); //清除cookie
            res.redirect( '/' ); //跳转到首页
        })
        ...
        ##################################################################
 
三、至此我们完成了简单完成了注册及登录功能流程。
        本节我们主要完成:
        1、通过mongoose来操作mongo,完成数据增加和查询
        2、通过session保存登录状态
        3、完成注册 / 登录 / 退出
        4、增加权限判断
 
        本节我们虽然使用了session来记录登录状态,但实际使用时还是会有些问题,session是保存在本项目里的,如果上线后web服务端需要用多台计算机来负载,则状态不能共享。可以采用搭建验证服务器,即单独配置一个服务器来执行验证功能,也可以使用redis来保存登录状态。下节我们将使用redis来保存登录状态。
 
        下节主要实现:
        1、注册时的邮件验证
        2、redis保存登录状态
        3、nodejs异常处理,同步 and 异步
        4、git操作

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-308547-1-1.html 上篇帖子: mongodb 学习笔记 (四) Mongo的内存管理 下篇帖子: Mongo的ORM框架的学习Morphia(annotations)
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表