本文档适用于uni-id 4.0.0及以上版本,需 HBuilderX 3.5.0 及以上版本。旧版本文档请访问:uni-id 3.x.x 文档

# 需求背景

99%的应用,都要开发用户注册、登录、发送短信验证码、修改密码、密码加密保存、密码防探测、token管理、页面访问权限、注册用户统计等众多功能,从前端到后端都需要。

为什么不能有一个开源的通用项目,避免大家的重复开发呢?

uni-id应需而生。

uni-iduniCloud开发者提供了开源、易用、安全、丰富、可扩展的用户管理框架。

clientDBDB Schemauni-starteruni-admin,这些产品都基于uni-id的账户体系。可以说uni-id是uniCloud不可或缺的基础能力。

# uni-id 的价值

  1. 节省了大量重复劳动
  2. 降低门槛,前端开发者无需研究数据库如何设计、账户安全如何保障
  3. 多系统打通用户和上下游协同

关于第三点,着重强调下。

一个应用,往往需要集成多个功能模块。比如一个电商应用,需要一个基本电商模板,还需要客服聊天模板、统计看板模板。

在插件市场,每类模板插件都能找到,但他们如果不是基于同一套用户体系设计,就很难整合。

所有uniCloud的标准应用,都基于uni-id来做。uni-id-common公共模块自动内置在每个服务空间里的。

有了统一的账户规范,并且围绕这套账户规范,有各种各样插件,那么开发者可以随意整合这些插件,让数据互通。

规范,还可以让上下游充分协同。插件市场会出现各种数据迁移插件,比如把从discuz里把用户迁移到uni-id中的插件,相信围绕这套规范的产业链会非常活跃。

目前插件市场上各种优秀的uniCloud轮子,几乎都是基于uni-id的。

# 功能清单

uni-id已完成的功能:

  • 注册、登录、发送短信验证码、密码加密保存、修改密码、忘记密码、头像管理、token管理、rbac权限角色体系、页面访问权限路由控制、用户邀请裂变、用户签到、日志记录、账户防刷

关于登录方式,目前已实现

  • 账户密码登录
  • 手机号+短信验证码登录 (内置uniCloud短信能力)
  • App手机号一键认证,免验证码(内置uni-app App一键登录能力)
  • 三方登录:App中的微信登录、Apple ID、QQ登录;微信小程序中的微信登录;支付宝小程序中的支付宝账户登录;QQ小程序中的QQ登录

由于三方登录很多,DCloud没有精力全部实现,在uni-id-co中留下了空实现,欢迎开发者自行补充、提交pr或发布扩展插件,共同完善uni-id。。

后续计划:DCloud未来将内置 微信扫码登录和公众号登录、邮箱验证集成、facebook等海外主流社交账户登录、活体检测。

其他方面,各种常见开源项目如discuz、wordPress、ecshop的用户导入插件,不属于uni-id主工程,欢迎开发者单独提交插件到插件市场。

# 组成结构

uni-id贯穿了uni-app前端到uniCloud后端的各个环节。

模块 说明
前端uni-app框架的相关API uniIdRouter页面路由、token管理客户端API
前端页面uni-id-pages 登录、注册、修改密码、忘记密码、个人中心、修改头像等前端页面
网络传输自动管理用户token 自动保存、续期token、网络自动传输token
云端云对象uni-id-co 与uni-id-pages搭配的云对象,相关业务的云端部分
云端配置uni-config-center 在uni-config-center下提供各种配置
云端公共模块uni-id-common 用于云函数或云对象集成该模块验证token身份
云数据库的用户相关数据表 uni-id-users等各种opendb数据表
uni-admin Admin管理后台,包括用户角色权限管理、注册用户统计
  1. 云数据库的uni-id相关表

数据库是一个系统的核心,uni-id首先规范化了十几张用户相关的opendb数据表

其中最为重要的4张opendb表,如下:

主表为uni-id-users表,保存用户的基本信息。扩展字段有很多,如实名认证数据、工作履历数据。由于MongoDB的特性,开发者可以自由扩展字段。

所有uni-id的数据表,不管在HBuilderX中新建 DB Schema 还是在 uniCloud web控制台新建表的界面上,都可以选择模板直接建好。

  1. 云端公共模块uni-id-common

uni-id-common公共模块包含了账户体系服务端的核心权限、token管理,内置在每个uniCloud服务空间里。

如开发者需要在自己的云函数/云对象里校验前端用户token,则需要引用uni-id-common公共模块。

uniCloud众多功能(如DB Schema的权限、uni-id-co)也都依赖 uni-id-common。

详见

  1. 云端uni-config-center下的uni-id配置

uni-id在云端有很多配置,比如密码加密秘钥、短信和微信登录的appsecret等等。在uni-config-center下的uni-id目录下的config.json里存放着这些配置。

见下

  1. 客户端API uni-app框架内置了uni-id的token管理。

uni-app与uniCloud搭配且使用uni-id,登录后自动下发token、网络传输层自动传输token(uni-app 2.7.13+版本)、token临近过期会自动续期(uni-app 3.4.13 +版本),也就是说开发者无需自己管理token了。

uni-app客户端还有一批uni-id相关的内置API:

  • uniIDHasRole:判断当前用户是否拥有某角色。详情
  • uniIDHasPermission:判断当前用户是否拥有某权限。详情
  • uniCloud.getCurrentUserInfo():客户端获取当前用户信息。详情
  1. 云端一体页面模板 uni-id-pages(含uni-id-co)

基于uni-id-common,DCloud还提供了一组完整的前端页面和后端云对象 ,合称uni-id-pages

uni-id-pages的功能包括:用户注册(含用户协议、隐私协议)、退出、修改密码、忘记密码等各种功能,同时适配PC宽屏和各种手机平台(App、H5、小程序)。

此外,DCloud的其他产品也为uni-id提供了众多支持:

  • uni-admin后台管理框架,为uni-id提供了现成的用户、角色、权限的后台管理功能,以及注册用户统计报表。

以上全部是开源的。

历史遗留

在HBuilderX 3.5之前,DCloud提供了一个公共模块uni-id(注意不叫uni-id-common)和一个示例性云函数uni-id-cf(集成在uni-starter和uni-admin中)。

老的公共模块uni-id是一个大而全的账户管理公共模块,体积太大,不适合被其他云函数引用。比如某个业务云函数需要校验用户token,引用的uni-id公共模块还包含了忘记密码的代码,很浪费资源。

在云对象发布之前,DCloud基于云函数方式提供了uni-id-cf。但在HBuilderX 3.5 以后,推荐使用基于云对象的uni-id-pages,代码更简单清晰。

从HBuilder 3.5起,uni-id和uni-id-cf都将被淘汰,不再更新。老的公共模块uni-id被拆开,变成了uni-id-common公共模块和uni-id-co云对象。

uni-id-common很精简,只包括token和权限,适合被所有云函数引用。

uni-id-co则是一个更加比uni-id-cf更完善和规范的用户管理的云对象。

老版升级指南,详见

# 快速上手

uni-id-common的插件市场地址为:uni-id-common插件。但一般不需要单独下载这个插件,但更新uni-id-common公共模块时需要从这里下载更新。

一般推荐直接使用uni-starter项目模板来开始开发,或者在新项目里导入uni-id-pages页面模板来使用。

uni-id云端的配置是依赖uni-config-center公用模块的,在工程目录uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json。(如未安装uni-config-center需安装,如缺少目录需手动创建)

uni-id云端同时依赖了公共模块uni-captcha,这个功能模块负责生成和校验验证码,进行人机验证。

体验uni-id需保证uniCloud服务空间至少有数据表uni-id-usersopendb-verify-codes(验证码表)

使用uni-id,首先需要确定2件事:

  1. 确定注册登录方式

你的应用采用什么方式注册登录?比如用户名密码、手机号+短信验证码、或者微信登录。

很多登录方式涉及三方服务,需要开通短信验证码服务、开通App一键登录、或者向微信等申请登录的appid和appsecret信息。

申请开通相关服务后,需要把配置信息填写在云端配置文件config.json中。

  1. 配置各项Secret

账户如果涉及密码,那么需要配置passwordSecret,账户的密码会根据passwordSecret使用sha1摘要加密算法以不可逆的方式存储在数据库中。

配置tokenSecret是为了防止token被第三方解密,模拟用户身份。

千万不要使用默认的passwordSecret和tokenSecret,会造成系统安全隐患。

云端的config.json还有各种配置,详见下个章节。前端的配置请参考uni-id-pages的文档。

# 云端配置config.json的说明

uni-id的云端配置文件在uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json中。

注意:

  • config.json是一个标准json文件,不支持注释

配置项:

  • passwordSecret为用于加密密码入库的密钥
  • tokenSecret为生成token需要的密钥
  • tokenExpiresIntoken有效期,以秒为单位
  • passwordErrorLimit密码错误重试次数,分ip记录密码错误次数,达到重试次数之后等待passwordErrorRetryTime时间之后才可以重试
  • passwordErrorRetryTime单位为秒
  • 如果使用sendSmsCode接口发送短信需要前往uniCloud控制台开通并充值,配置config.jsonservice字段,字段说明见下方示例
  • 另外可以按照客户端平台进行不同的配置,参考下面示例

下面的配置文件中所有时间的单位都是秒

!!!重要!!! passwordSecret与tokenSecret十分重要,切记妥善保存(不要直接使用下面示例中的passwordSecret与tokenSecret)。修改passwordSecret会导致老用户使用密码无法登录,修改tokenSecret会导致所有已经下发的token失效。如果重新导入uni-id切勿直接覆盖config.json相关配置

// 如果拷贝此内容切记去除注释
{
  "passwordSecret": [
    {
      "type": "hmac-sha256",
      "version": 1
    }
  ], // 数据库中password字段是加密存储的,这里的passwordSecret即为加密密码所用的加密算法,详见[密码安全]
  "passwordStrength": "medium", // 密码强度,新增于 uni-id-pages 1.0.8版本,见下方说明
  "tokenSecret": "", // 生成token所用的密钥,注意修改为自己的,使用一个较长的字符串即可
  "requestAuthSecret": "", // URL化请求鉴权签名密钥
  "tokenExpiresIn": 7200, // 全平台token过期时间,未指定过期时间的平台会使用此值
  "tokenExpiresThreshold": 3600, // 新增于uni-id 1.1.7版本,checkToken时如果token有效期小于此值且在有效期内则自动获取新token,请注意将新token返回给前端保存(云对象会自动保存符合uniCloud响应体规范的响应内的新token),如果不配置此参数则不开启自动获取新token功能
  "maxTokenLength": 10, // 数据库用户记录token数组的最大长度,默认为10。新增于uni-id-common 1.0.16
  "passwordErrorLimit": 6, // 密码错误最大重试次数
  "passwordErrorRetryTime": 3600, // 密码错误重试次数超限之后的冻结时间
  "autoSetInviteCode": false, // 是否在用户注册时自动设置邀请码,默认不自动设置
  "forceInviteCode": false, // 是否强制用户注册时必填邀请码,默认为false
  "idCardCertifyLimit": 1, // 实名认证相关; 限制每个身份证可以绑定几个账号
  "realNameCertifyLimit": 5, // 实名认证相关; 限制用户每日认证次数,防止接口被刷
  "sensitiveInfoEncryptSecret": "", // 敏感信息加密密钥(长度为32位的字符串),如使用实名认证功能需配置此密钥
  "frvNeedAlivePhoto": false, // 实名认证相关;是否获取认证照片
  "userRegisterDefaultRole": [], // 用户注册时的默认角色
  "app": { // 如果你使用旧版本uni-id公共模块而不是uni-id-common这里可能配置的是app-plus,务必注意调整为app
    "tokenExpiresIn": 2592000,
    "tokenExpiresThreshold": 864000,
    "oauth": {
      // App微信登录所用到的appid、appsecret需要在微信开放平台获取,注意:不是公众平台而是开放平台
      "weixin": {
        "appid": "",
        "appsecret": ""
      },
      // App QQ登录所用到的appid、appsecret需要在腾讯开放平台获取,注意:不是公众平台而是开放平台
      "qq": {
        "appid": "",
        "appsecret": ""
      },
      "apple": { // 使用苹果登录时需要
        "bundleId": ""
      },
      "huawei": { // 鸿蒙应用使用华为登录
        "clientId": "",
        "clientSecret": ""
      }
    }
  },
  "web": { // 如果你使用旧版本uni-id公共模块而不是uni-id-common这里可能配置的是h5,务必注意调整为web
    "tokenExpiresIn": 7200,
    "tokenExpiresThreshold": 3600,
    "oauth": {
      "weixin-h5": { // 微信公众号登录配置
        "appid": "",
        "appsecret": ""
      },
      "weixin-web": { // 微信PC页面扫码登录配置
        "appid": "",
        "appsecret": ""
      }
    }
  },
  "mp-weixin": {
    "tokenExpiresIn": 259200,
    "tokenExpiresThreshold": 86400,
    "oauth": {
      // 微信小程序登录所用的appid、appsecret需要在对应的小程序管理控制台获取
      "weixin": {
        "appid": "",
        "appsecret": ""
      }
    }
  },
  "mp-qq": {
    "tokenExpiresIn": 259200,
    "tokenExpiresThreshold": 86400,
    "oauth": {
      // QQ小程序登录所用的appid、appsecret需要在对应的小程序管理控制台获取
      "qq": {
        "appid": "",
        "appsecret": ""
      }
    }
  },
  "mp-alipay": {
    "tokenExpiresIn": 259200,
    "tokenExpiresThreshold": 86400,
    "oauth": {
      // 支付宝小程序登录用到的appid、privateKey请参考支付宝小程序的文档进行设置或者获取,https://opendocs.alipay.com/open/291/105971#LDsXr
      "alipay": {
        "appid": "",
        "privateKey": "", // 私钥
        "keyType": "PKCS8" // 私钥类型,如果私钥类型不是PKCS8,需要填写此字段,否则会出现“error:0D0680A8:asn1 encoding routines:ASN1_CHECK_TLEN:wrong tag”错误
      }
    },
    "mp-harmony": {
        "huawei": { // 鸿蒙元服务使用华为登录
          "clientId": "",
          "clientSecret": ""
        }
    }
  },
  "service": {
    "sms": {
      "name": "", // 应用名称,对应短信模版的name
      "codeExpiresIn": 180, // 验证码过期时间,单位为秒,注意一定要是60的整数倍
      "scene": {
        "bind-mobile-by-sms": { // 对绑定手机号场景的配置,短信验证码场景值参考:https://uniapp.dcloud.net.cn/uniCloud/uni-id/summary.html#sms-scene
          "templateId": "", // 绑定手机号使用的短信验证码模板
          "codeExpiresIn": 240 // 绑定手机号验证码过期时间
        }
      }
    },
    "univerify": {
      "appid": "" // uni-id-co 1.1.17及以上版本无需填写,当前应用的appid,使用云函数URL化,此项必须配置
    }
  }
}

# 密码强度

新增于uni-id-pages 1.0.8

passwordStrength配置项支持以下四种内置规则

{
  // 密码必须包含大小写字母、数字和特殊符号
  super: /^(?=.*[0-9])(?=.*[a-z])(?=.*[A-Z])(?=.*[~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/])[0-9a-zA-Z~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/]{8,16}$/,
  // 密码必须包含字母、数字和特殊符号
  strong: /^(?=.*[0-9])(?=.*[a-zA-Z])(?=.*[~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/])[0-9a-zA-Z~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/]{8,16}$/,
  // 密码必须为字母、数字和特殊符号任意两种的组合
  medium: /^(?![0-9]+$)(?![a-zA-Z]+$)(?![~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/]+$)[0-9a-zA-Z~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/]{8,16}$/,
  // 密码必须包含字母和数字
  weak: /^(?=.*[0-9])(?=.*[a-zA-Z])[0-9a-zA-Z~!@#$%^&*_\-+=`|\\(){}[\]:;"'<>,.?/]{6,16}$/
}

uni-id-co 与 uni-id-pages 内的前端页面均支持这四个内置规则

# 登录方式及配置说明

登录方式 配置及获取方式
用户名、手机号、邮箱+密码 配置passwordSecret即可
手机号+验证码 配置service.sms,在uniCloud控制台短信服务内获取配置信息:短信服务
手机号一键登录 配置service.univerify,在开发者中心一键登录服务内获取:一键登录
微信小程序登录 配置mp-weixin.oauth.weixin,在微信公众平台获取:微信公众平台
微信公众号登录 配置web.oauth.weixin-h5,在微信公众平台获取:微信公众平台
微信PC页面扫码登录 配置web.oauth.weixin-web,在微信开放平台获取:微信开放平台
微信APP端登录 配置app.oauth.weixin,在微信开放平台获取:微信开放平台
QQ 小程序端登录 配置mp-qq.oauth.qq,在QQ开放平台获取:QQ开放平台
QQ APP端登录 配置app.oauth.qq,在QQ互联获取:QQ互联
支付宝小程序端登录 配置mp-alipay.oauth.alipay,在支付宝开放平台获取:支付宝开放平台
Apple APP端登录 配置app.oauth.apple,在Apple开发者中心自行配置:Apple开发者中心

# 用户注册时设置默认角色uni-id-pages 1.1.6+

默认情况下,用户注册后不会关联任何角色,如果需要用户在注册后关联角色,可以通过此配置项开启。

配置项userRegisterDefaultRole的值类型为Array,每个元素为角色ID(role_id), 例如:

{
  "userRegisterDefaultRole": ["user"]
}

# token令牌

注意

如需保持活跃客户端的登录状态,请勿将token有效期设置一个很大的值,具体如何实现请参考:保持客户端登录状态

首先解释下token的概念。token是服务器颁发给客户端的一个令牌。

用户在客户端登录时,云端通过登录接口对用户的用户名+密码,或者手机号+验证码进行校验,校验通过后服务器会给客户端下发一个token(就是根据tokenSecret生成的一串加密字符串),并同时给出有效期。

客户端把这个token保存在storage中,然后每次联网请求服务器时,都带上这个token。服务器解密这个token,通过这个token认定客户端的身份。

这样就避免了客户端每次请求服务器,都需要再传输一次用户名和密码。

这是业内通行的设计。

在传统开发下,客户端和服务器各自需要为了token做很多事情。在uni云端一体下,开发者无需操心,只需要在uni-id云端config.json中配置好token的secret和有效期即可。剩余的工作都被自动处理了。

uni-id云端会在login方法成功后自动返回token,uni-app前端框架会自动识别并保存这个token在storage中(uni_id_token),在前端每次连接uniCloud(不管是clientDB、callfunction、云对象调用),都会自动带上这个token。

云函数和云对象都提供了获取和校验token的方法,在uni-id相关业务中,校验token的代码都已经写好。

包括token快到期时的自动续期,开发者只需在config.json中配置好临近多久自动续期,续期的代码也无需开发者编写,框架已经内置。

注:不同平台的token有效期一般不一样,app有效期较长,web有效期较短。每个平台的有效期都可以单独在config.json里配置。

# 用户角色权限

为什么需要角色权限管理?

  • 企业管理系统,比如uni-admin,除了超级管理员,不同账号通常需根据职位、责任设定不同的系统权限。比如部门管理员、Hr。
  • clientDB允许前端直接操作数据库,但部分字段应该是系统计算或管理员设置的,比如文章的阅读数、收藏数及是否加精置顶,这些字段不允许普通用户在前端通过clientDB直接修改,此时也需要通过权限控制来保证系统的安全稳定。

uni-id基于经典的RBAC模型实现了角色权限系统。

# RBAC模型简介

RBAC:Role-Based Access Control,基于角色的访问控制。

其基本思想:对系统操作的各种权限不是直接授予具体的用户,而是在用户集合与权限集合之间建立一个角色集合。每一种角色对应一组相应的权限。一旦用户被分配了适当的角色后,该用户就拥有此角色的所有权限。

这样做的好处是,增强系统管理的扩展性,对于批量用户的权限变更,仅需变更该批用户角色对应权限即可,而无需对该批每个用户变更权限。

这个模型有三个关键名词:用户、角色、权限:

  • 用户:使用系统的人,一个用户可以同时有多个角色,比如内容审核员、比如部门管理员
  • 角色:权限的集合,一个角色可以有多个权限。比如内容审核员这个角色,有隐藏帖子权限、有帖子加精权限;而部门管理员这个角色,有给本部门新增员工权限、有删除本部门员工的权限。
  • 权限:数据权限或业务权限,例如:删除用户、帖子加精等

用户、角色、权限都存在数据库了,都可以动态创建和修改。当权限对应的代码实现完成后,用户的新入、退出、角色升迁都无需再修改代码,在uni-admin后台的web界面可以由运维人员可视化的给每个用户调整角色、给每个角色调整权限。

# 用户

用户信息存储在uni-id-users表中,然后通过role字段保存该用户所拥有的所有角色ID,角色ID即角色表(uni-id-roles表)中的role_id字段,注意不是_id字段。

// uni-id-users 表
{
  {
    "_id":"5f8428181c229600010389f6",
    "username":"张三",
    "email":"zhangsan@dcloud.io",
    "role":[
      "USER_ADMIN",
      "NOTICE_ADMIN"
    ],
    "created_date":1602495783272
  }
}

Tips:将用户角色设计为用户表的字段,而没有新建用户角色关联表的原因:避免mongodb在跨表查询时的性能开销

# 角色

角色信息存储在uni-id-roles表中

字段 类型 必填 描述
_id Object ID 系统自动生成的Id
role_id String 角色唯一标识
role_name String 角色名,展示用
permission Array 角色拥有的权限列表
comment String 备注
created_date Timestamp 角色创建时间

其中:

  • role_id为角色标志,全局唯一,可用于clientDB中的权限控制,建议按照语义化命名,例如:USER_ADMIN表示人事管理、NOTICE_ADMIN表示公告管理
  • permission为数组类型,存储该角色拥有的所有权限ID,权限ID即权限表(uni-id-permissions表)中的permission_id字段,注意不是_id字段

如下为示例:

{
  {
    "_id":"5f8428181c229600010389f6",
    "role_id":"USER_ADMIN",
    "role_name":"人事管理",
    "permission":[
      "USER_ADD",
      "USER_EDIT",
      "USER_DEL"
    ],
    "created_date":1602495783272
  },
  {
    "_id":"5f842836d8daea0001906785",
    "role_id":"NOTICE_ADMIN",
    "role_name":"公告管理",
    "permission":[
      "NOTICE_ADD",
      "NOTICE_EDIT",
      "NOTICE_DEL"
    ],
    "created_date":1602495784372
  }
}

如下是角色在clientDB中的配置示例:

// uni-id-users.schema.json
{
  "permission": {
    "update":"doc._id == auth.uid || 'USER_ADMIN' in auth.role" //用户自己或人事管理员可执行用户表的.update操作
  }
}

Tips1:uni-id中admin为超级管理员角色,uni-clientDB也基于同样的策略;如果用户角色包含admin,则该用户就拥有所有数据表的全部权限。

Tips2:出厂时可内置常用角色,也可上线后由运营人员动态创建角色。

# 权限

权限信息在uni-id-permissions表中,表结构定义如下:

字段 类型 必填 描述
_id Object ID 系统自动生成的Id
permission_id String 权限唯一标识
permission_name String 权限名,展示用
comment String 备注
created_date Timestamp 权限创建时间

其中,permission_id为权限标志,全局唯一,可用于clientDB中的权限配置,建议按照语义化命名,例如:USER_DELBRANCH_ADD权限总数量不得超过500

如下为示例内容:

{
  {
    "_id":"5f8428181c229600010389f6",
    "permission_id":"USER_EDIT",
    "permission_name":"修改用户",
    "created_date":1602495783272
  },
  {
    "_id":"5f842836d8daea0001906785",
    "permission_id":"USER_DEL",
    "permission_name":"删除用户",
    "created_date":1602495784372
  }
}

如下是权限在clientDB中的配置示例:

// uni-id-users.schema.json
{
  "permission": {
    "update":"doc._id == auth.uid || 'USER_EDIT' in auth.permission" //用户自己或有`USER_EDIT`权限的用户,可执行用户表的.update操作
  }
}

Tips1:建议代码交付时内置所有权限,方便clientDB中的权限配置和调整。

# 其他说明

uni-id将用户的角色权限缓存在token内。详情参考:缓存角色权限

如下是通过token判断权限的简单示例:

// 简单的权限校验示例
function hasPermission(token, permission) {
  const {
	  uid,
	  role,
	  permission
  } = await uniID.checkToken(token)
  return role.includes('admin') || checkTokenRes.permission.includes(permission) // admin用户的permission为空数组,但是拥有所有权限
}

注意: 在uniCloud admin中,封装了可视化的用户、权限、角色的管理,新增删除修改均支持,无需自己维护。详见

# uni-id数据表

uni-id的所有数据表,都在opendb规范中。

在unicloud web控制台 新建数据表时,可以从uni-id的模板分类里找到下面的表,并一键创建这些表。HBuilderX 3.4.11起新建 DB Schema 也有模板可选择。

# 用户表

存放用户基本信息。

表名:uni-id-users

字段 类型 必填 描述
_id Object ID 存储文档 ID(用户 ID),系统自动生成
username String 用户名,不允许重复
password password 密码。加密存储
nickname String 用户昵称
gender int 用户性别:0 未知 1 男性 2 女性
role Array 用户角色列表,由role_id组成的数组
status int 用户状态:0 正常,1 禁用,2 审核中,3 审核拒绝,4 已注销
dcloud_appid Array 允许登录的客户端的appid列表,不同应用同时复用一个user表时适用,比如 司机端和乘客端是2个appid,在登陆时可以隔离,见下
mobile String 手机号码
mobile_confirmed int 手机号验证状态:0 未验证 1 已验证,未验证用户不可登录
email String 邮箱地址
email_confirmed int 邮箱验证状态:0 未验证 1 已验证,未验证用户不可登录
avatar String 头像地址
wx_unionid String 微信unionid
wx_openid Object 微信各个平台openid。子结构详见下文
qq_unionid String QQ unionid
qq_openid Object QQ各个平台openid。子结构详见下文
ali_openid String 支付宝平台openid
apple_openid String 苹果登录openid
comment String 备注
realname_auth Object 实名认证信息。子结构详见下文
register_date Timestamp 注册时间
register_ip String 注册时 IP 地址,uni-id 3.3.14起移至register_env内
last_login_date Timestamp 最后登录时间
last_login_ip String 最后登录时 IP 地址
login_ip_limit Array 登录 IP 限制
inviter_uid Array 邀请人uid,按层级从下往上排列的uid数组,即第一个是直接上级
my_invite_code String 用户自己的邀请码
register_env Object 用户注册时的环境信息,新增于uni-id 3.3.14。子结构详见下文

注意

  • 本表格只列出部分字段,完整字段详见
  • 最后登录时间、IP,并非只有登录操作会修改,token刷新时也会修改最后登录时间、ip。应用启动时若token有效则不会触发登录行为,也不会更新本值。

wx_openid字段定义

opendb中uni-id-users表1.0.0调整为下面的结构,uni-id-co使用此标准。如何处理旧数据请参考:自uni-id升级为uni-id-co+uni-id-common

字段 类型 必填 描述
app String app平台微信openid
mp String 微信小程序平台openid
web String 微信网页应用openid
h5 String 微信公众号应用openid

qq_openid字段定义

opendb中uni-id-users表1.0.0调整为下面的结构,uni-id-co使用此标准。如何处理旧数据请参考:自uni-id升级为uni-id-co+uni-id-common

字段 类型 必填 描述
app String app平台QQ openid
mp String QQ小程序平台openid

realNameAuth 扩展字段定义 该字段存储实名认证信息,子节点说明如下。

字段 类型 必填 描述
type Integer 用户类型:0 个人用户 1 企业用户
auth_status Integer 认证状态:0 未认证 1 等待认证 2 认证通过 3 认证失败
auth_date Timestamp 认证通过时间
real_name String 真实姓名/企业名称
identity String 身份证号码/营业执照号码
id_card_front String 身份证正面照 URL
id_card_back String 身份证反面照 URL
id_card_in_hand String 手持身份证照片 URL
license String 营业执照 URL
contact_person String 联系人姓名
contact_mobile String 联系人手机号码
contact_email String 联系人邮箱

register_env字段定义

注意:该字段是在前端注册用户时记录的前端环境信息。如果是管理员在云端调用uni-id的addUser添加的用户则无此字段

字段 类型 必填 描述
appid String 注册时的客户端appId
uni_platform String 注册时的客户端平台,如h5、app、mp-weixin等
os_name String 注册时的客户端系统名,ios、android、windows、mac、linux
app_name String 注册时的客户端名称
app_version String 注册时的客户版本
app_version_code String 注册时的客户版本号
channel String 注册时的客户端启动场景(小程序)或应用渠道(app)
client_ip String 注册时的客户端IP

用户集合示例:

{
  "_id": "f2a60d815ee1da3900823d45541bb162",
  "username": "姓名"
  "password": "503005d4dd16dd7771b2d0a47aaef927e9dba89e",
  "status":0,//用户状态:0正常 1禁用 2审核中 3审核拒绝
  "mobile":"",
  "mobile_confirmed":0, //手机号是否验证,0为未验证,1为已验证
  "email":"amdin@domain.com",
  "email_confirmed":0, //邮箱是否验证,0为未验证,1为已验证
  "avatar":"https://cdn.domain.com/avatar.png"
  "last_login_ip": "123.120.11.128", //最后登录IP

}

# 用户表索引使用注意

目前 opendb 内提供的 uni-id-users表 包含完整的索引,数据库在索引量多且频繁更新的情况下可能会出现写入缓慢的情况,因此推荐开发者在使用 uni-id-users表 时可以删除没有用到的索引。

例:项目内只使用了微信登录,不使用其他登录方式,可以只保留wx_unionid、wx_openid.mp这些账号相关的索引,删除其他登录方式的索引(比如username、mobile)

不了解索引请参考:索引

# 验证码表

表名:opendb-verify-codes

该表的前缀不是uni-id,意味着该表的设计用途是通用的,不管是uni-id的手机号验证码,或者支付等关键业务需要验证码,都使用此表。

每条验证信息,都记录在本表中。uni-id不会自动删除本表的历史数据,数据保留有效期需要开发者自行管理,可以在云函数中设置一个定时运行来清理过期数据。

字段 类型 必填 描述
_id Object ID 存储文档 ID(验证码 ID),系统自动生成
mobile String 手机号,和邮箱二选一
email String 邮箱,和手机号二选一
code String 验证码
scene String 验证场景
state Integer 验证状态:0 未验证 1 已验证 2 已作废
ip String 请求时 IP 地址
create_date Timestamp 创建时间
expired_date Timestamp 验证码过期时间

# 角色表

表名:uni-id-roles

字段 类型 必填 描述
_id Object ID 系统自动生成的Id
role_id String 角色唯一标识
role_name String 角色名,展示用
permission Array 角色拥有的权限列表
comment String 备注
created_date Timestamp 角色创建时间

# 权限表

表名:uni-id-permissions

字段 类型 必填 描述
_id Object ID 系统自动生成的Id
permission_id String 权限唯一标识
permission_name String 权限名,展示用
comment String 备注
created_date Timestamp 权限创建时间

# 更多表

还有更多uni-id的配套数据表,可以在uniCloud web控制台新建表时选择相应模板。此处不再详述,仅罗列清单:

  • 日志表:uni-id-log
  • 积分表:uni-id-scores
  • 地址信息表:uni-id-address
  • 订单表:uni-id-base-order
  • 设备表:uni-id-device
  • 关注粉丝表:uni-id-followers
  • 任务表:uni-id-task
  • 任务日志表:uni-id-task-log

# 常量

# 用户状态

对应uni-id-users表的status字段

说明
0 正常状态
1 封禁状态
2 审核中
3 审核失败
4 注销状态

# log类型

对应uni-id-log表的type字段

说明
logout 登出
login 登录
register 注册
reset-pwd 重置密码
bind-mobile 绑定手机
bind-weixin 绑定微信账号
bind-qq 绑定QQ账号
bind-apple 绑定苹果账号
bind-alipay 绑定支付宝账号

# 手机、邮箱验证码使用场景

对应opendb-verify-codes表短信、邮箱验证码相关记录的的scene字段

说明
login-by-sms 短信验证码登录
reset-pwd-by-sms 短信验证码重置密码
bind-mobile-by-sms 短信验证码绑定手机号
login-by-email 邮箱验证码登录
reset-pwd-by-email 邮箱验证码重置密码
set-pwd-by-sms 手机验证码设置登录密码

# 图形验证码使用场景

对应opendb-verify-codes表图形验证码相关记录的scene字段

说明
register 用户名密码注册
login-by-pwd 用户名/手机/邮箱+密码登录
login-by-sms 短信验证码登录
reset-pwd-by-sms 短信验证码重置密码
reset-pwd-by-email 邮箱验证码重置密码
send-sms-code 发送短信验证码
send-email-code 发送邮箱验证码
bind-mobile-by-sms 短信验证码绑定手机号
set-pwd-by-sms 手机验证码设置登录密码

# uniIdRouter自动路由

支持的HBuilderX版本

uni-app uni-app x
3.5.0+ 3.99+

uniIdRouter 是一个运行在前端的、对前端页面访问权限路由进行控制的方案。

大多数应用,都会指定某些页面需要登录才能访问。以往开发者需要写不少代码。

现在,只需在项目的pages.json内配置登录页路径、需要登录才能访问的页面等信息,uni-app框架的路由跳转,会自动在需要登录且客户端登录状态过期或未登录时跳转到登录页面。

结合以下代码及注释了解如何使用uniIdRouter

{
	"pages": [
		{
			"path": "pages/index/index",
			"style": {
				"navigationBarTitleText": "uni-app"
			},
			"needLogin": false // 当前页面是否需要登录才可以访问,此配置优先级高于uniIdRouter下的needLogin
		}, {
			"path": "pages/list/list",
			"style": {
				"navigationBarTitleText": "uni-app"
			},
			"needLogin": true
		}, {
			"path": "pages/detail/detail",
			"style": {
				"navigationBarTitleText": "uni-app"
			}
		}
	],
	"globalStyle": {
		"navigationBarTextStyle": "black",
		"navigationBarTitleText": "uni-app",
		"navigationBarBackgroundColor": "#F8F8F8",
		"backgroundColor": "#F8F8F8"
	},
	"uniIdRouter": {
		"loginPage": "pages/index/index", // 登录页面路径
		"needLogin": [
			"pages/detail/.*" // 需要登录才可访问的页面列表,可以使用正则语法
		],
		"resToLogin": true // 自动解析云对象及clientDB的错误码,如果是客户端token不正确或token过期则自动跳转配置的登录页面,配置为false则关闭此行为,默认true
	}
}

以上代码,指定了登录页为首页index,然后将list页面和detail目录下的所有页面,设为需要登录才能访问。那么访问list页面和detail目录下的页面时,如果客户端未登录或登录状态过期(也就是uni_id_token失效),那么会自动跳转到index页面来登录。

与此功能对应的有两个uniCloud客户端api,uniCloud.onNeedLogin()uniCloud.offNeedLogin(),开发者在监听onNeedLogin事件后,框架就不再自动跳转到登录页面,而是由开发者在onNeedLogin事件内自行处理。详情参考:uniCloud.onNeedLogin

自动跳转到登录页面时会携带uniIdRedirectUrl参数,其值为encodeURIComponent(${跳转前的页面(包含路径和参数的完整页面地址)}),如果希望用户登录后跳转回之前的页面,可以使用此参数实现。

以下为登录页面跳转到之前访问页面的简单示例:

pages/login/login.vue

<template>
	<view>
		<button @click="login">login</button>
	</view>
</template>

<script>
	export default {
		data() {
			return {
				uniIdRedirectUrl: ''
			}
		},
		onLoad(options) {
			this.uniIdRedirectUrl = decodeURIComponent(options.uniIdRedirectUrl)
		},
		methods: {
			async login() {
				// ...执行登录操作,在成功回调里跳转页面
				if (this.uniIdRedirectUrl) {
					uni.redirectTo({
						url: this.uniIdRedirectUrl
					})
				}
			}
		}
	}
</script>

# 云对象响应触发needLogin

云对象抛出uni-id token过期或token无效错误码时,会触发客户端自动跳转配置的登录页面,以下代码为一个简单示例

// todo云对象
const uniIdCommon = require('uni-id-common')
module.exports = {
	_before(){
		this.uniIdCommon = uniIdCommon.createInstance({
			clientInfo: this.getClientInfo()
		})
	},
	addTodo(title) {
		const {
			errCode,
			errMsg,
			uid
		} = await this.uniIdCommon.checkToken(this.getUniIdToken())
		if(errCode) { // uni-id-common的checkToken接口可能返回`uni-id-token-expired`、`uni-id-check-token-failed`错误码,二者均会触发客户端跳转登陆页面
			return {
				errCode,
				errMsg
			}
		}
		// ...
	}
}

客户端add-todo.vue

<template>
	<!-- 略 -->
</template>
<script>
	export default {
		data() {
			return {
			}
		},
		onLoad() {},
		methods: {
			async addTodo(title){
				const todo = uniCloud.importObject('todo')
				await todo.addTodo(title) // 调用addTodo时云端checkToken如果返回了token错误、token失效的错误码就会自动跳转到配置的登录页面
			}
		}
	}
</script>
<style>
</style>

注意

  • pages.json内有uniIdRouter节点上述逻辑才会生效,自HBuilderX 3.5.0起创建空项目模板会自动配置空的uniIdRouter节点
  • uniIdRouter底层使用navigateTo、redirectTo、reLaunch、switchTab的拦截器进行页面跳转拦截,不会拦截进入首页,web端和app端会拦截原生tabbar点击,其他端不会拦截原生tabbar点击。 一般tabbar页面都不做自动跳转,而是在页面内再提供登录按钮。比如tabbar上有购物车或个人中心,点击购物车后在购物车页面内部会放一个提示语和按钮,告知用户需要登录。 在页面内判断用户是否登录,使用APIuniCloud.getCurrentUserInfo()

# 云端错误码

错误码errCode 错误信息errMsg 说明
0(数字) 成功 -
uni-id-token-expired 登陆状态失效,token已过期 -
uni-id-check-token-failed token校验未通过 -
uni-id-account-exists 账户已存在 -
uni-id-account-not-exists 账户不存在 -
uni-id-account-not-exists-in-current-app 匹配到的用户不可在当前应用登录
uni-id-account-conflict 用户账号冲突 可能会由开发者手动更新数据库导致,正常情况下不应出现
uni-id-account-banned 此账号已封禁 -
uni-id-account-auditing 此账号正在审核中 -
uni-id-account-audit-failed 此账号审核失败 -
uni-id-account-closed 此账号已注销 -
uni-id-captcha-required 请输入图形验证码 -
uni-id-password-error 用户名或密码错误 -
uni-id-invalid-username 用户名不合法 -
uni-id-invalid-password 密码不合法 -
uni-id-invalid-mobile 手机号码不合法 -
uni-id-invalid-email 邮箱不合法 -
uni-id-invalid-nickname 昵称不合法 -
uni-id-invalid-param 参数错误 -
uni-id-param-required 缺少参数 -
uni-id-get-third-party-account-failed 获取第三方账号失败 -
uni-id-get-third-party-user-info-failed 获取第三方用户信息失败 -
uni-id-mobile-verify-code-error 手机验证码错误或已过期 -
uni-id-email-verify-code-error 邮箱验证码错误或已过期 -
uni-id-admin-exists 超级管理员已存在 -
uni-id-permission-error 权限错误 -
uni-id-system-error 系统错误 -
uni-id-set-invite-code-failed 设置邀请码失败 -
uni-id-invalid-invite-code 邀请码不可用 -
uni-id-change-inviter-forbidden 禁止修改邀请人 -
uni-id-bind-conflict 此账号(微信、QQ、手机号等)已被绑定 -

# 多个应用复用相同uni-id-user表

有些系统由多个子应用组成,且没有各自独立服务空间,而是需要共享一个服务空间。此时就涉及一个问题,多个应用注册的账户都在uni-id-user表中,如何有效隔离。

比如一个打车软件,有乘客端、司机端、管理端,都要注册账户。它们也都有自己的DCloud appID(manifest.json里第一个配置)

uni-id-user表中有一个数组型字段dcloud_appid,可以存贮这个用户有权登陆哪个应用。

比如乘客端的appid是__uni_111111,司机端appid是__uni_222222,那么2个appid都存入dcloud_appid,即表示这个用户有权登录这2个应用。

# 隔离不同应用的用户

uni-id 3.3.0版本起用户注册时会自动在用户表的记录内标记为注册应用对应的用户,如果没有单独授权登录其他应用的话则只能登录这个应用。即在乘客端应用注册的,默认只能在乘客端应用登录。

如何授权登录其他应用请参考:授权、禁止用户在特定客户端应用登录

需要注意的是客户端APPID信息是由端上传上来的,并非完全可信,尽量在入口处进行校验。例:

exports.main = async function(event, context){
  if(context.APPID !== '__UNI__xxx1') {
    throw new Error('应用ID非法')
  }
}

DCloud Appid是一个很重要的配置,如无必要请勿随意更换。

不同端用户数据通过用户表的dcloud_appid字段隔离,同一个手机号、微信号也可以同时注册管理端和用户端,绑定账号同理。

注意

  • uni-id会自动在用户表每条用户记录插入dcloud_appid字段(此字段是一个数组,标识此用户可以在哪些端登录)。
  • 为兼容旧版本产生的旧数据,针对没有dcloud_appid字段的用户,允许登录任意端。
  • 如果用户数据库记录中dcloud_appid字段是一个空数组,表示当前用户不能在任何客户端登录
  • 已有dcloud_appid的用户,如果使用相同的用户标识(用户名、邮箱、手机、微信等)+ 不同的DCloud Appid登录会被判定为不同的用户,如果此时数据库没有对应的记录,会报用户不存在的错误

# 隔离不同应用的配置

uni-id的config.json支持配置为数组,每项都是一个完整的配置,对不同的配置使用dcloudAppid字段进行区分(此字段与项目内的manifest.json里面的DCloud AppId一致), uni-id会自动根据客户端的appid来判断该使用哪套配置。

需要注意的是客户端APPID信息是由端上传上来的,并非完全可信,尽量在入口处进行校验。例:

exports.main = async function(event, context){
  if(context.APPID !== '__UNI__xxx1') {
    throw new Error('应用ID非法')
  }
}

示例

数组每一项都是一个完整的配置文件,全部选项请参考:uni-id 配置

注意:如果允许同一账号在不同端使用相同的账号+密码登录需要将不同端的passwordSecret设置成一样的

[{
  "dcloudAppid": "__UNI__xxxx1", // 务必替换为对应项目manifest.json内的DCloud Appid
  "isDefaultConfig": true, // 默认配置标记,未匹配到dcloudAppid的情况下使用默认配置
  "passwordSecret": "passwordSecret-demo",
  "tokenSecret": "tokenSecret-demo",
  "tokenExpiresIn": 7200,
  "tokenExpiresThreshold": 600
}, {
  "dcloudAppid": "__UNI__xxxx2", // 务必替换为对应项目manifest.json内的DCloud Appid
  "passwordSecret": "passwordSecret-demo",
  "tokenSecret": "tokenSecret-demo",
  "tokenExpiresIn": 7200,
  "tokenExpiresThreshold": 600
}]

# 其他功能

# token更新及保持客户端登录状态

一般来说token的有效期不会无限长,示例配置内web端token有效期为2小时,微信小程序为3天,app端为30天。你可以回忆一下你所用的软件,只要每天都打开就一直不需要重新登录,这样就牵扯到保持客户端的登录状态的问题。

uni-id使用了判断token剩余有效时间小于一定的阈值(配置文件内的tokenExpiresThreshold)但是大于0时自动下发新token的逻辑来保证活跃客户端一直处于登录状态,返回新token的逻辑由checkToken方法实现。具体该将token有效期和token刷新阈值设置为多少,需要根据多数用户软件使用频率来确定。

举个例子,开发者配置的token有效期(tokenExpiresIn)为1天,token刷新阈值(tokenExpiresThreshold)为8小时。用户在0点0分0秒获取了token,如果用户在16点后(token有效期已小于8小时)调用接口时执行了checkToken方法则会返回新token。

# 裂变

config.jsonuniCloud/cloudfuntions/common/uni-config-center/uni-id/config.json,以下config.json均指此文件)内配置了autoSetInviteCode: true则在用户注册时会自动给设置不重复的6位邀请码

config.json内配置了forceInviteCode: true则只有使用邀请码才可以注册。

uni-id-co在会产生注册行为的接口均添加了inviteCode参数,用于传递邀请码使注册用户接受邀请

# 修改passwordSecret

注意:通常情况下设定好passwordSecret之后不需要再进行修改,使用此功能时请务必小心谨慎

说明

在config.json内修改passwordSecret会导致历史用户无法通过密码登录。但是某些情况下有些应用有修改passwordSecret的需求,例如刚开始使用uni-id时没有自定义passwordSecret,后续需要修改,此时可以使用uni-id 2.0.1版本新增的修改passwordSecret功能。(注意:2.0.1版本验证码表名调整为了opendb-verify-codes

如何使用

下面以将passwordSecret从passwordSecret-demo修改为qwertyasdfgh为例介绍如何使用

// 旧config.json
{
  "passwordSecret": "passwordSecret-demo"
}

// 新config.json
{
  "passwordSecret": [{
    "version": 1,
    "value": "passwordSecret-demo"
  },{
    "version": 2,
    "value": "qwertyasdfgh"
  }]
}

如果在上面基础上再修改passwordSecret为1q2w3e4r5t,config.json调整如下

!!!注意只有在数据库内完全没有使用某个版本(password_secret_version字段表示了用户密钥版本)密钥的用户才可以将此密钥从config.json内去除。没有password_secret_version的用户使用的是最旧版本的passwordSecret,如果存在这样的用户对应的passwordSecret也不可去除。

// 新config.json,
{
  "passwordSecret": [{
    "version": 1,
    "value": "passwordSecret-demo"
  },{
    "version": 2,
    "value": "qwertyasdfgh"
  },{
    "version": 3,
    "value": "1q2w3e4r5t"
  }]
}

原理

uni-id-users表内存储的password字段为使用hmac-sha1生成的hash值,此值不可逆向推出用户真实密码。所以直接修改passwordSecret会导致老用户无法使用密码登录。

上述修改通过密钥版本号区分新旧密钥,用户登录时如果密钥版本小于当前最新版本,会为用户更新数据库内存储的password字段,并记录当前使用的密钥版本。

用户对应的数据库记录内没有密钥版本的话会使用最低版本密钥进行密码校验,校验通过后为用户更新为最新版密钥对应的password并记录版本号。

由于是不可逆加密,理论上passwordSecret泄露不会造成用户的真实密码被泄露,自定义passwordSecret只是进一步加强安全性。

# 缓存角色权限

使用uni-id-common时,token内会缓存用户的角色权限。

为什么要在token缓存角色权限?token校验是高频操作,云数据库是按照读写次数来收取费用的,并且读写数据库会拖慢接口响应速度。

比较经济的做法就是在token里缓存角色权限。更好的方案是在redis里缓存角色权限,只是redis需要付费开通。

注意

  • 由于角色权限缓存在token内,可能会存在权限已经更新但是用户token未过期之前依然是旧版角色权限的情况。可以调短一些token过期时间来减少这种情况的影响。或者使用redis来缓存用户权限
  • admin角色token内不包含permission,如需自行判断用户是否有某个权限,要注意admin角色需要额外判断一下,只要角色为admin或permission包含期待值,都视为拥有权限

# 自定义token内容

uni-id-common支持在token内缓存用户的角色权限。但是某些情况下开发者可能还希望缓存一些别的东西,以便在客户端能方便的访问(注意:不可缓存机密信息到token内)。

用法

uni-config-center模块内的uni-id插件内创建custom-token.js内容如下:

module.exports = async (tokenObj) => {
  // tokenObj为原始token信息结构如下
  // {
  //   uid: 'abc', // 用户id
  //   role: [], // 用户角色列表
  //   permission: [] // 用户权限列表,admin角色的用户权限列表为空数组
  // }

  tokenObj.customField = 'hello custom token' // 自定义token字段
  return tokenObj // 注意务必返回修改后的token对象
}

uni-id会自动加载custom-token.js进行处理,在所有生成token的操作(包括:登录、注册、token过期自动刷新、开发者自行调用createToken)执行时自动获取新token信息,并生成token。

注意

  • 不要删除原始token内的字段

# 自定义国际化语言

完整词句列表参考:

用法

cloudfunctions/common/uni-config-center/uni-id/lang/目录下创建index.js,内容示例如下:

module.exports = {
	'zh-hant': { // 语言代码
		'alipay': '支付寶'
	}
}

uni-id会自动进行语言匹配,无需额外配置

# 自动保存用户sessionKey、accessToken等信息

uni-id-co在微信、QQ登录或注册时会自动保存用户的sessionKey、accessToken信息。

uni-id-pages 1.0.8之前,uni-id-co直接将这些信息保存在了用户表(uni-id-users)的third_party字段下,仅按照平台区分没有按照不同应用区分。具体结构如下

{
  "_id": "uid",
  "wx_unionid": "xxx",
  "qq_unionid": "xxx",
  "third_party": {
    "mp_weixin": {
      "session_key": "xxxx"
    },
    "app_weixin": {
      "access_token": "accessToken",
      "access_token_expired": 1111
    },
    "mp_qq": {
      "session_key": "xxxx"
    },
    "app_qq": {
      "access_token": "accessToken",
      "access_token_expired": 1111
    }
  }
}

此结构无法满足多应用同一平台关联同一服务空间且允许用户跨应用登录的场景。因此在uni-id-pages 1.0.8及更高版本对此做出了调整,改为使用uni-open-bridge-common存储用户在三方平台的凭据信息。同时为了兼容旧版本上述third_party字段仍存有这些信息。

目前被uni-id-co保存的三方凭据有以下几种:

  • 微信小程序端用户session_key,通过uni-open-bridge-commonsetSessionKey方法写入
  • 微信公众号页面用户access_token,通过uni-open-bridge-commonsetUserAccessToken方法写入
  • 微信web页面扫码登录时返回的用户access_token,通过uni-open-bridge-commonsetUserAccessToken方法写入
  • 微信APP登录时返回的用户access_token,通过uni-open-bridge-commonsetUserAccessToken方法写入
  • QQ小程序端用户session_key,通过uni-open-bridge-commonsetSessionKey方法写入
  • QQ APP登录时返回的用户access_token,通过uni-open-bridge-commonsetUserAccessToken方法写入

开发者如需获取某用户对应的openid,可以在用户使用相应的登录操作之后自行读取用户记录获取。代码示例如下:

const uid = 'xx'
const appId = '__UNI_xxx'
const wxPlatform = 'mp' // mp:小程序,h5:公众号,web:web页面,app:App微信登录

const db = uniCloud.database()
const getUserRes = await db.collection('uni-id-users').doc(uid).get()

const userRecord = getUserRes.data[0]
if(!userRecord) {
  throw new Error('未匹配到此用户')
}
const wxOpenid = userRecord.wx_openid || {}
const openid = wxOpenid[wxPlatform + '_' + appId]  || wxOpenid[wxPlatform]

# 钩子

新增于 uni-id-pages 1.0.8

uni-id-co是一个完整的云对象,里面注册登录等流程都已完全实现,开发者不方便进行修改。例如要实现注册时为某端用户统一添加一个角色的功能,只能去修改uni-id-co的代码。因此uni-id-co提供了通过钩子干涉内置逻辑的功能

uni-id钩子函数需要在uni-config-center内配置。在uni-config-center/uni-id下创建hooks目录并在其内创建index.js内容如下

module.exports = {
  beforeRegister: function (){
    // 注册前钩子
  }
}

# beforeRegister

beforeRegister在注册用户记录入库前触发。钩子会接收到如下参数,需要返回处理后的用户记录用以入库存储

参数名 类型 说明
userRecord Object 即将入库的用户记录
clientInfo Object 客户端信息,参考:云对象 getClientInfo

以为__UNI_123123这个应用注册的用户添加"teacher"角色为例,beforeRegister钩子示例如下

// 钩子函数示例 hooks/index.js

function beforeRegister({
  userRecord,
  clientInfo
} = {}) {
  if(clientInfo.appId === '__UNI_123123') {
    if(userRecord.role) {
      userRecord.role.push('teacher')
    } else {
      userRecord.role = ['teacher']
    }
  }
  return userRecord // 务必返回处理后的userRecord
}

module.exports = {
  beforeRegister
}

# 密码安全

uni-id 默认使用了 hmac-sha1 加密算法对密码进行加密,自 uni-id-pages@1.0.28 版本起新增了 hmac-sha256 加密算法,开发者可以自己需求选择不同的算法,推荐使用 hmac-sha256算法。

uni-config-center/uni-id/config.json 中配置, uni-id/config.json说明

{
  "passwordSecret": [
    {
      "type": "hmac-sha256", // 必须指定算法类型 默认 hmac-sha1
      "version": 1
    }
  ]
}

修改 passwordSecret 参考

# 升级 hmac-256 加密算法指南

适用于 uni-id-pages@1.0.28 以下版本, 首先确认 uni-config-center/uni-id/config.jsonpasswordSecret 字段类型

passwordSecret 字段可能是string或者array类型,示例如下:

// 1 string
{
  "passwordSecret": "passwordSecret-demo"
}
// 2 array
{
  "passwordSecret": [
    {
      "value": "passwordSecret-demo",
      "version": 1
    }
  ]
}

如果 passwordSecret 是字符串类型,修改为数组类型后,在 passwordSecret 中添加 hmac-256 算法,同时 version 加 1

{
  "passwordSecret": [
    {
      "value": "passwordSecret-demo",
      "version": 1
    },{
      "type": "hmac-256",
      "version": 2
    }
  ]
}

# 自定义加密算法

如果内置的加密算法无法满足业务需求,可以自定义加密规则。

首先在 uni-config-center/uni-id/config.json 中增加自定义密码类型 custom

{
  "passwordSecret": [
    {
      "value": "passwordSecret-demo",
      "version": 1
    },
    {
      "type": "hmac-sha256", // 必须指定算法类型 默认 hmac-sha1
      "version": 2
    },
    {
      "type": "custom", // 固定值 custom,代表使用自定义规则
      "version": 3
    }
  ]
}

uni-config-center/uni-id/custom-password.js文件(没有请手动创建)中创建加密与验证方法即可。

module.exports = {
    /**
     * 密码加密
     * @param {String} password 用户输入的密码
     * @param {Object} clientInfo 客户端信息
     * @param {Object} passwordSecret config.json 匹配到的 passwordSecret
     * @return {{version, passwordHash}}
     */
    encryptPassword: function ({password, clientInfo, passwordSecret}) {
        // 必须按照此格式返回
        return {
            passwordHash: password,
            version: passwordSecret.version
        }
    },
    /**
     * 密码验证
     * @param {String} password 用户输入的密码
     * @param {Object} userRecord 用户信息
     * @param {Object} clientInfo 客户端信息
     * @param {Object} passwordSecret config.json 匹配到的 passwordSecret
     * @return {boolean}
     */
    verifyPassword: function ({password, userRecord, clientInfo, passwordSecret}) {
        return password === userRecord.password
    }
}

# 将用户迁移至 uni-id

如果你想将自己系统内的用户导入至 uni-id,请按照以下步骤操作

# 1. 准备自定义用户密码函数

uni-id 默认使用了 hmac-256 密码加密算法,可能与你的加密算法不同,所以在迁移前需要自定义你的密码加密函数。 当用户第一次在 uni-id 中进行登录时,会先使用自定义验证密码(verifyPassword)函数进行验证,这样用户用之前的密码依旧能够登录,不需要用户重置密码。 在用户第一次登录成功后用户密码的加密算法规则将升级为配置文件中最新的算法规则。

首先在uni-config-center/uni-id/config.json文件中创建自定义类型的paswordSecret,如下:

{
  "passwordSecret": [
    {
      "type": "custom",
      "version": 1
    }
  ]
}

uni-config-center/uni-id/custom-password.js 文件(不存在请手动创建)中创建 verifyPassword 函数验证之前用户密码。

module.exports = {
    /**
     * 密码验证
     * @param {String} password 用户输入的密码
     * @param {Object} userRecord 用户信息
     * @param {Object} clientInfo 客户端信息
     * @param {Object} passwordSecret config.json 匹配到的 passwordSecret
     * @return {boolean}
     */
    verifyPassword: function ({password, userRecord, clientInfo, passwordSecret}) {
        return password === userRecord.password
    }
}

注意

如果配置文件中passwordSecret最新版本的 type 是 custom,那么所有用户的加密及校验都会使用自定义算法规则。

如果仅是为了迁移使用,请另外在 passwordSecret 中添加 hmac-sha256 类型算法,用于其他用户的加密与校验。

这样只有之前的用户才会使用 custom 自定义规则,其他用户或者迁移后的新注册的用户会使用 hmac-sha256 算法规则。

{
  "passwordSecret": [
  {
    "type": "custom",
    "version": 1
  },
  {
    "type": "hmac-sha256",
    "version": 2
  }
  ]
}

# 2. 导出用户数据

导出文件大小最大不超过 50MB,超过 50MB 请导出多个json文件

如果存在表A关联表B的字段的场景需要保证关联字段在A、B内是一致的(特别需要注意的是各种与_id关联的字段)

参考从文件中导入数据

请将用户数据导出为json格式文件,注意json文件不是标准的json文件,请按照以下格式每行是一个json格式的用户记录导出:

{"user_id":0,"nickname":"张三","age":25,"password":"123456"}
{"user_id":1,"nickname":"李四","age":18,"password":"000000"}

# 3. 处理用户数据

在 uni-id 中 userId 是系统自动创建的 _id,如果想保留之前用户的userId,可以将用户的 userId 映射为 _id,如果不保留 userId 建议删除 userId 字段,在数据导入中会创建 _id。

注意password_secret_version字段,字段值需要修改为自定义密码类型的 version

导入到 uni-id 之前,需要处理用户数据与 uni-id 字段的映射关系,见下方 uni-id 字段及说明:

字段 类型 默认值 说明
_id - - 存储文档 ID(用户 ID),系统自动生成
ali_openid string - 支付宝平台openid
apple_openid string - 苹果登录openid
avatar string - 头像地址 (完整路径)
avatar_file file - 用file类型方便使用uni-file-picker组件
comment string - 备注
dcloud_appid array - 允许登录的客户端的appid列表
department_id array - 部门ID
email string - 邮箱地址
email_confirmed int 0 邮箱验证状态:0 未验证 1 已验证
gender int 0 用户性别:0 未知 1 男性 2 女性
invite_time timestamp - 受邀时间
inviter_uid array - 用户全部上级邀请者
last_login_date timestamp - 最后登录时间
last_login_ip string - 最后登录时 IP 地址
mobile string - 手机号码
mobile_confirmed int 0 手机号验证状态:0 未验证 1 已验证
my_invite_code string - 用户自身邀请码
nickname string - 用户昵称
password password - 密码,加密存储
password_secret_version int - 密码使用的passwordSecret版本
realname_auth object - 实名认证信息;见下方 realname_auth 结构
register_date timestamp - 注册时间
register_ip string - 注册时 IP 地址
role array - 用户角色
score int - 用户积分,积分变更记录可参考:uni-id-scores表定义
status int - 用户状态:0 正常 1 禁用 2 审核中 3 审核拒绝
token array - 用户token
username string - 用户名,不允许重复
wx_openid object - 微信各个平台openid;见下方 wx_openid 结构
wx_unionid string - 微信unionid
qq_openid object - QQ各个平台openid;见下方 qq_openid 结构
qq_unionid string - QQ unionid
third_party object - 三方平台凭证

realname_auth 结构

字段 类型 默认值 说明
auth_date timestamp - 认证通过时间
auth_status int 0 认证状态:0 未认证 1 等待认证 2 认证通过 3 认证失败
contact_email string - 联系人邮箱
contact_mobile string - 联系人手机号码
contact_person string - 联系人姓名
id_card_back string - 身份证反面照 URL
id_card_front string - 身份证正面照 URL
identity string - 身份证号码/营业执照号码
in_hand string - 手持身份证照片 URL
license string - 营业执照 URL
real_name string - 真实姓名/企业名称
type int - 用户类型:0 个人用户 1 企业用户

wx_openid 结构

字段 类型 默认值 说明
app string - app平台微信openid
mp string - 微信小程序平台openid
h5 string - 微信公众号登录openid
web string - PC页面扫码登录openid

qq_openid 结构

字段 类型 默认值 说明
app string - app平台QQ openid
mp string - QQ小程序平台openid

# 4. 导入数据

uniCloud 控制台,找到 uni-id 所在的服务空间,在云数据库中选中 uni-id-users 表,点击导入按钮,上传用户数据json文件即可。

从文件中导入数据说明

# 实名认证uni-id-pages 1.1.7+

HBuilderX 版本需大于3.7.6

实人认证相关功能建议或问题,可以加入uni-im交流群进行讨论,点此加入

基于uni实人认证服务实现,可以实现用户刷脸核验真实身份,完成实名认证。

目前仅APP端支持实名认证。

uni-id-pages / uni-id-pages-x 中内置了实名认证页面

  • uni-id-pages: uni-id-pages/pages/userinfo/realname-verify/realname-verify
  • uni-id-pages-x: uni-id-pages-x/pages/userinfo/realnameAuth/realnameAuth

如没有实名认证需求,可以将实名认证相关页面注释:

  1. uni-id-pages/pages/userinfo/userinfo页面中,注释掉实名认证的uni-list-item标签。
  2. pages.json中注释掉实名认证页面

# 开通与使用

  1. 开通uni实人认证,开通指南详见
  2. 开通完成后,在插件市场中导入uni-id-pages至项目中。
  3. 参考配置项说明完成配置设置,详细的uni-id配置文件参考
  4. uni-config-centeruni-id-co模块上传至云端。
  5. manifest.json中找到App模块设置-打包模块设置,勾选”实人认证“。
  6. 运行iOS/Android标准基座即可测试实名认证功能。

# 配置项说明

实名认证相关配置项如下,配置文件路径uniCloud/cloudfunctions/common/uni-config-center/uni-id/config.json,详细的uni-id配置文件参考

字段 类型 默认值 说明
idCardCertifyLimit number 1 限制每个身份证可以绑定几个账号
realNameCertifyLimit number 5 限制用户每日认证次数,防止接口被刷
sensitiveInfoEncryptSecret string 敏感信息加密密钥(长度必须32位); 见下方敏感信息加密
frvNeedAlivePhoto boolean false 是否获取认证照片

注意

# 接口参考

# 敏感信息加密

用户的姓名、身份证号、实人认证照片属于用户隐私信息,为了防止隐私信息泄露,在数据存储上使用了对称加密aes-256-cbc算法对数据进行加密。 在前端页面需要使用时,例如”获取用户实名信息“接口,只会返回脱敏后的数据,减少暴露风险,提高安全性。

由于加密密钥sensitiveInfoEncryptSecret来源于config.json配置文件,强烈建议更换为自定义的字符串,不要使用默认的密钥。 密钥长度需要是32位的字符串。

实人认证照片将会上传至云存储中,阿里云与腾讯云存储路径如下:

  • 阿里云 /{uid}.b64
  • 腾讯云 /user/id-card/{uid}.b64

注意:文件不是图片不可直接下载打开。

# 常见问题

# Invalid uni-id config file

  • 检查uni-id 配置文件是否存在,不存在请创建 uni-id 配置文件,uni-id 配置参考
  • 如果云函数在云端运行,请检查 uni-id 配置文件是否上传至云端
  • uni-id 配置文件为标准的 json 文件,文件内不允许添加 js 注释,请检查是否添加了 js 注释