Crossbar.io 认证实战

Crossbar.io 提供了一系列的认证方法。

  • WAMP-Anonymouse
  • WAMP-Ticket
  • WAMP-CRA
  • WAMP-Cryptosign
  • WAMP-Cookie
  • WAMP-TLS

项目中目前用到了会话相关的 Anonymous 、Ticket、 CRA 三种方式。

后端使用静态 Ticket ,前端普通用户使用 Anonymous 连接注册用户信息并保存到数据中,然后使用 CRA 动态认证通过查询数据库中的信息进行对比。

思路清晰后,开始一项项进行:

静态 Ticket 实现

修改 crossbar 运行项目的配置文件 .crossbar/config.json 如下,添加 backend 角色,可以进行注册调用,使用 prefix 匹配 uri

{
    ...
    "workers": [
        {
            ...
            "realms": [
                {
                    "name": "realm1",
                    "roles": [
                        {
                            "name": "backend",
                            "permissions": [
                                {
                                    "uri": "",
                                    "match": "prefix",
                                    "allow": {
                                        "call": true,
                                        "register": true,
                                        "publish": false,
                                        "subscribe": false
                                    },
                                    "disclose": {
                                        "caller": false,
                                        "publisher": false
                                    },
                                    "cache": true
                                }
                            ]
                        },
...

ws 属性设置 roleticket , 这样配置文件就搞定了

...
"transports": [
                {
                    "type": "web",
                    "endpoint": {
                        "type": "tcp",
                        "port": 8080
                    },
                    "paths": {
                        "/": {
                            "type": "static",
                            "directory": "../web"
                        },
                        "ws": {
                            "type": "websocket",
                            "serializers": [
                                "json"
                            ],
                            "auth": {
                                "ticket": {
                                    "type": "static",
                                    "principals": {
                                        "backend": {
                                            "ticket": "123456",
                                            "role": "backend"
                                        }
                                    }
                                }
                            }
...

后端使用 Python , 文件中添加 roleticket 信息,在 self.join 中添加对应的 ticketROLE 参数,onChanllenge 中进行类型判断返回 ticket ,认证通过后会执行 onJoin 里面的内容。

...

ROLE = u'backend'
TICKET = u'123456'

class ClientSession(ApplicationSession):

   def onConnect(self):
      self.join(self.config.realm, [u"ticket"], ROLE)

   def onChallenge(self, challenge):
      if challenge.method == u"ticket":
         return TICKET
      else:
         raise Exception("Invalid authmethod {}".format(challenge.method))

   @inlineCallbacks
   def onJoin(self, details):
      def hi():
         return 'hi'

      try:
         reg2 = yield self.register(hi, u'com.example.frontend.hi')
         print("com.example.frontend.hi")
      except Exception as e:
         print("could not register procedure: {}".format(e))


...

后端静态 Ticket 连接搞定了,开始做前端的相关认证,先从匿名注册用户开始。

Anonymous 注册用户

回到 crossbar 的配置文件中 , 添加一个public 角色,只能调用一个匿名注册函数 com.example.anonymous.frontend.register, 使用 exact (完全匹配) 的方式,设置该角色只能执行 call ,后端程序注册好函数后,这样就提供了一个匿名用户只能注册的接口。

后端提供的这个接口连接数据库,把用户信息保存到数据库中,CRA 动态认证的时候使用注册的用户名密码等进行登录认证。

...            
 "realms": [
    {
        "name": "realm1",
        "roles": [
            ...
            {
                "name": "public",
                "permissions": [
                    {
                        "uri": "com.example.anonymous.frontend.register",
                        "match": "exact",
                        "allow": {
                            "call": true,
                            "register": false,
                            "publish": false,
                            "subscribe": false
                        },
                        "disclose": {
                            "caller": false,
                            "publisher": false
                        },
                        "cache": true
                    }
                ]
            }
          ...
WAMP-CRA 动态认证

这一部分的实现思路是单独写一个基于 crossbar 连接的 component ,使用静态 Ticket 实现连接,在认证函数中通过前端登录页面输入的用户信息连接数据库进行对比。

再次操作 crossbar 的配置文件,添加一个 frontend 前端角色,使用 wildcard (通配符匹配)的方式匹配 uri ,前端角色只能执行 call

添加一个 authenticator 认证角色,使用 exact 匹配 uri ,所有 crossbar 动态 CRA 认证连接通过 com.example.authenticate 接口来控制。


...
"realms": [
    {
        "name": "realm1",
        "roles": [
            {
                "name": "frontend",
                "permissions": [
                    {
                        "uri": "com.example.frontend.",
                        "match": "wildcard",
                        "allow": {
                            "call": true,
                            "register": false,
                            "publish": false,
                            "subscribe": false
                        },
                        "disclose": {
                            "caller": false,
                            "publisher": false
                        },
                        "cache": true
                    }
                ]
            },
            {
                "name": "authenticator",
                "permissions": [
                    {
                        "uri": "com.example.authenticate",
                        "match": "exact",
                        "allow": {
                            "call": false,
                            "register": true,
                            "publish": false,
                            "subscribe": false
                        },
                        "disclose": {
                            "caller": false,
                            "publisher": false
                        },
                        "cache": true
                    }
                ]
            }
          ...

使用 Python 实现认证 Component , 认证组件独立运行在服务端,使用静态 Ticket 连接,注册一个 com.example.authenticate 接口进行动态验证,调用接收到的用户信息 authenid 匹配数据库信息,用户名密码正确,认证通过。

...
ROLE = u'backend'
TICKET = u'sg-ai.com'

class ClientSession(ApplicationSession):
    def onConnect(self):
        self.join(self.config.realm, [u"ticket"], ROLE)

    def onChallenge(self, challenge):
        if challenge.method == u"ticket":
            return TICKET
        else:
            raise Exception("Invalid authmethod {}".format(challenge.method))

    async def onJoin(self, details):
        async def authenticate(realm, authid, details):
            try:

                data = {'userName': authid}
                res = await self.call(u'com.db.mongo.backend.select', data, 'user')
                result = {'role': 'frontend', 'secret': res[0]['password']}
                return result

            except Exception as e:
                print("authenticate function error: {0}".format(e))

        try:
            await self.register(authenticate, u'com.example.authenticate')
        except Exception as e:
            print("Failed to register dynamic authenticator: {0}".format(e))

    def onLeave(self, details):
        print("Client session left: {}".format(details))
        self.disconnect()

    def onDisconnect(self):
        print("Client session disconnected.")

...

目前前端实现使用两套连接,第一次使用匿名方式连接,注册成功后连接关闭,然后在登录操作中使用CRA的动态认证连接方式。(不知道还有没有更好的实现方式)

注意

crossbar配置文件中 uri 中不能含有 - 字符

添加混合认证后提供给前端的接口 uri 中不能含有大写字母,不然会认证错误

Error {
  error: 'wamp.error.not_authorized',
  args: 
   [ 'session is not authorized to call procedure \'com.ai.user.frontend.getUserInfo\'' ],
  kwargs: {} }

版本信息: js "autobahn": "^17.5.2"

Crossbar.io COMMUNITY 17.12.1

有任何疑问欢迎留言交流 ^ - ^

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,649评论 18 139
  • 这篇文章介绍了Mobile BI(移动商务智能)使用过程中涉及的各种身份认证的方式,主要目的是对这些方式的原理进行...
    雨_树阅读 2,013评论 1 2
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,801评论 6 342
  • 2017年6月28日 星期三 阴转晴 今晚上班,换了气洗了激发台,打了标样,想着顺便擦洗哈键盘鼠标,擦好键盘,顺便...
    希多妈阅读 255评论 0 0
  • 清风袭来香沁露, 乍泄春光色。 多情笑我痴, 嗔怒无处躲, 却低首敛眉不与说。 娇羞失身花泪落, 轻薄不由我。 云...
    泽霈君阅读 353评论 12 6