composer-rest-server认证实现

composer-rest-server认证实现

在composer里面关于用户有两种模式。

  1. 多用户模式,用户为参与者。和composer-rest-server并无太大关系,仅作为智能合约里面的参与者。用户通过认证,可以独立执行composer-rest-server,以实现权限隔离。
  2. 认证模式。多个用户(不是参与者)访问同一个restful接口,需要做一定的认证操作,避免对区块链的过度访问。这种模式实在compsoser-rest-server层面发生的,认证过程不会和区块链有任何互动。

这篇文章所讲的就是第二种方式。

composer-rest-server是通过loopback构建的一个项目,所以权限管理大致和loopback相同。

这里提供两种方式供选择
  1. 直接使用loopback用户认证(推荐)
    loopback提供了简单的用户模块,在composer的实现中,用户被隐藏起来了,所以只需要开放登录、登出接口即可。

    需要解决的问题:

    1. composer实现只是在内存保存了数据。如果需要管理用户,需要使用数据库
    2. 开放user数据模块之后,开放的接口太多,需要隐藏一部分

    解决方式

    1. 修改 /server/datasources.json 为
    {
      "db": {
        "name": "db",
        "connector": "memory"
      },
      //添加mysql依赖。注意:使用mysql需要在package.json添加 "loopback-connector-mysql":"5.2.0" 依赖
      "mysqlDs": {
        "name": "mysqlDs",
        "connector": "mysql",
        "host": "localhost",
        "port": 3306,
        "database": "db_rest_server", //需新建数据库和user表。
        "username": "root",
        "password": "123456"
       }
    }
    
    1. 修改 /server/model-config.json
    ...
      "user": {
        "dataSource": "mysqlDs", //修改数据源 memory 为 mysql
        "public": true //开发
        },
    ...
    
    1. 新增脚本 /server/root/create-models.js

    loopback里面,server/boot 下面的脚本会在程序启动的时候执行

    'use strict';
    
        module.exports = function(app) {
            //FIXME only run this script AT first TIME . For create table.
            // app.dataSources.mysqlDs.automigrate('user', function(err) {
            //     if (err) throw err;
            //
            //     app.models.user.create([{
            //         username: 'test1',
            //         email:'test1@email.com',
            //         password: 'test1',
            //     }], function(err, coffeeShops) {
            //         if (err) throw err;
            //         console.log('Models created: \n', coffeeShops);
            //     });
            // });
        
            app.models.user.disableRemoteMethodByName("upsert");                               // disables PATCH /app.models.users
            app.models.user.disableRemoteMethodByName("find");                                 // disables GET /app.models.users
            app.models.user.disableRemoteMethodByName("replaceOrCreate");                      // disables PUT /app.models.users
            app.models.user.disableRemoteMethodByName("create");                               // disables POST /app.models.users
        
            app.models.user.disableRemoteMethodByName("prototype.updateAttributes");           // disables PATCH /app.models.users/{id}
            app.models.user.disableRemoteMethodByName("findById");                             // disables GET /app.models.users/{id}
            app.models.user.disableRemoteMethodByName("exists");                               // disables HEAD /app.models.users/{id}
            app.models.user.disableRemoteMethodByName("replaceById");                          // disables PUT /app.models.users/{id}
            app.models.user.disableRemoteMethodByName("deleteById");                           // disables DELETE /app.models.users/{id}
        
            app.models.user.disableRemoteMethodByName('prototype.__get__accessTokens');        // disable GET /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__create__accessTokens');     // disable POST /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__delete__accessTokens');     // disable DELETE /app.models.users/{id}/accessTokens
        
            app.models.user.disableRemoteMethodByName('prototype.__get__credentials');        // disable GET /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__create__credentials');     // disable POST /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__delete__credentials');     // disable DELETE /app.models.users/{id}/accessTokens
        
            app.models.user.disableRemoteMethodByName('prototype.__get__identities');        // disable GET /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__create__identities');     // disable POST /app.models.users/{id}/accessTokens
            app.models.user.disableRemoteMethodByName('prototype.__delete__identities');     // disable DELETE /app.models.users/{id}/accessTokens
        
            app.models.user.disableRemoteMethodByName('prototype.__findById__accessTokens');   // disable GET /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__updateById__accessTokens'); // disable PUT /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__destroyById__accessTokens');// disable DELETE /app.models.users/{id}/accessTokens/{fk}
        
            app.models.user.disableRemoteMethodByName('prototype.__findById__identities');   // disable GET /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__updateById__identities'); // disable PUT /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__destroyById__identities');// disable DELETE /app.models.users/{id}/accessTokens/{fk}
        
            app.models.user.disableRemoteMethodByName('prototype.__findById__credentials');   // disable GET /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__updateById__credentials'); // disable PUT /app.models.users/{id}/accessTokens/{fk}
            app.models.user.disableRemoteMethodByName('prototype.__destroyById__credentials');// disable DELETE /app.models.users/{id}/accessTokens/{fk}
        
            app.models.user.disableRemoteMethodByName('prototype.__count__accessTokens');      // disable  GET /app.models.users/{id}/accessTokens/count
            app.models.user.disableRemoteMethodByName('prototype.__count__credentials');      // disable  GET /app.models.users/{id}/accessTokens/count
            app.models.user.disableRemoteMethodByName('prototype.__count__identities');      // disable  GET /app.models.users/{id}/accessTokens/count
        
            app.models.user.disableRemoteMethodByName("prototype.verify");                     // disable POST /app.models.users/{id}/verify
            app.models.user.disableRemoteMethodByName("changePassword");                       // disable POST /app.models.users/change-password
            app.models.user.disableRemoteMethodByName("createChangeStream");                   // disable GET and POST /app.models.users/change-stream
        
            app.models.user.disableRemoteMethodByName("confirm");                              // disables GET /app.models.users/confirm
            app.models.user.disableRemoteMethodByName("count");                                // disables GET /app.models.users/count
            app.models.user.disableRemoteMethodByName("findOne");                              // disables GET /app.models.users/findOne
        
        //app.models.user.disableRemoteMethodByName("login");                                // disables POST /app.models.users/login
        //app.models.user.disableRemoteMethodByName("logout");                               // disables POST /app.models.users/logout
        
            app.models.user.disableRemoteMethodByName("resetPassword");                        // disables POST /app.models.users/reset
            app.models.user.disableRemoteMethodByName("setPassword");                          // disables POST /app.models.users/reset-password
            app.models.user.disableRemoteMethodByName("update");                               // disables POST /app.models.users/update
            app.models.user.disableRemoteMethodByName("upsertWithWhere");                      // disables POST /app.models.users/upsertWithWhere
        
        };
    
    
    1. composer-rest-server-local -c cardName -a true [-n never] 启动服务
    在user表新建一条数据。使用 /user/login ,即可获取accessToken。在之后的请求里面,添加请求头:X-Access-Token: 对应token ,即可获取认证通过。
  1. passport实现认证

    passport是一个用来专门管理用户登录的一个库。包含n多第三方登录,已经local登录。适用于任何express扩展出来的项目。

    但是这个的针对auth、auth2等做的个库,所以如果是想提供直接提供api以获得token有些麻烦。

    使用方式:

    修改 /server/providers.json

    {
    "local": { //名称
        "provider": "local",
        "module": "passport-local", //这里表示使用本地用户。还有:github、google等授权库
        "usernameField": "username", //必须字段
        "passwordField": "password", 
        "authPath": "/auth/local", //认证跳转地址
        "successRedirect": "/", //成功回调
        "failureRedirect": "/" //失败回调
        }
    }
    

    直接访问设置的认证地址,在回调地址里面的session就能获取到accessToken。

    遇到的问题:
    本来是计划直接接入passport-local,不暴露user模块的,但是发现获取token方式太复杂,涉及到重定向,针对调用不友好,所以放弃了。

    为了避免修改loopback-componet-passport包内容,所以建议直接使用方式1.

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

推荐阅读更多精彩内容