# 自动检测登录

# 准备工作

  1. 注册 Authing 开发者账号

# 原理介绍

自动检测同一设备上关联应用的登录态,本质上就是建立起一个 deviceId (设备 ID) 与 Authing 服务器之间的 session 连接。

当某个用户在某个应用上登录后,调用 Authing 接口创建一个 deviceId 与 Authing 服务器之间的 session,这样用户在同一设备的其他应用登录的时候,就能检测到此 session 的存在,从而跳过登录步骤,实现自动登录。

假设你有三个 App: App 1、App2 和 App3,只要有其中一个 App 和 Authing 服务器建立了 session 关系,就可以检测到 session。

# 开始接入

# 获取设备 ID

请在测试时务必验证自己在不同的 App 内获取到的 deviceId 是一致的!

# iOS

iOS 设备的设备 ID 可以通过 identifierForVendor (opens new window) 获取,同一个 vendor 的应用获取到的设备 ID 是一样的。

什么情况下属于同一个 vendor?

  1. 从 App Store 下载的应用,基于在 App Store 登记的应用信息判断。
  2. 非 App Store 下载的应用
    1. iOS 6 及之前,bundle id 的前两部分一致的应用属于同一 vendor,如 com.example.app1和 com.example.app2 为同一 vendor。com.example.app1.xxx 和 com.example.app2.xxx 也属于同一 vendor。
    2. iOS 7 及之后,bundle id 的除了最后一部分的其余部分一致的应用属于同一 vendor,如 com.example.app1和 com.example.app2 为同一 vendor。但是 com.example.app1.xxx 和 com.example.app2.xxx 不属于同一 vendor。

如果你的应用不属于同一 vendor,建议使用 ASIdentifierManager (opens new window)

Swift 5 代码示例:

let deviceId = UIDevice.current.identifierForVendor!.uuidString

OC 代码示例:

UIDevice *currentDevice = [UIDevice currentDevice];
NSString *deviceId = [[currentDevice identifierForVendor] UUIDString];

# Android

Android 设备可通过 ANDROID_ID (opens new window) 获取:

Java 代码示例:

import android.provider.Settings.Secure;
private String android_id = Secure.getString(getContext().getContentResolver(),
                                                        Secure.ANDROID_ID); 

Kotlin 代码示例:

val deviceID = Settings.Secure.getString(contentResolver,
Settings.Secure.ANDROID_ID)
POST
https://core.authing.cn/oauth/sso/mobile/createSession

创建 session

此接口用于在一个移动应用客户端内创建一个 session ,且用户需处于登录状态,在请求头中加上 authorization 请求头携带用户 token。

Headers
authorization
REQUIRED
string

登录用户的 token

content-type
REQUIRED
string

application/json

Body Paramter
deviceId
REQUIRED
string

设备 ID

userPoolId
REQUIRED
string

用户池 ID

200: OK
{
    code: 200,
    message: "创建 session 成功!",
    data: {
        sessionId: "xxxxxx", // session ID
    }
}

Swift 代码示例:

func createSession(userPoolId: String, token: String){
    // 移动端 SSO: createSession
    struct MobileSSO: Encodable {
        let userPoolId: String
        let deviceId: String
    }
    let body = MobileSSO(
        userPoolId: UserPoolId, 
        deviceId: UIDevice.current.identifierForVendor!.uuidString,
    )
    let headers: HTTPHeaders = [
        "Authorization": token ,
        "Accept": "application/json"
    ]
    let api = "https://core.authing.cn/oauth/sso/mobile/createSession"
    AF.request(api, method: .post, parameters: body, encoder: JSONParameterEncoder.default, headers: headers).response { response in
         debugPrint(response)
    }
}

GET
https://core.authing.cn/oauth/sso/mobile/trackSession

查询 session

此接口用于在移动应用客户端内查询 session,不需要用户处于登录态。

Headers
content-type
REQUIRED
string

application/json

Body Paramter
deviceId
REQUIRED
string

设备 ID

userPoolId
REQUIRED
string

用户池 ID

200: OK

分两种情况:直接返回用户信息返回 ticket

// 直接返回用户信息
{
    code: 200,
    message: '获取 session 用户信息成功',
    data: {
      "_id":"5e05bbf2d51b3761d5c71070",
      "email":"983132@qq.com",
      "emailVerified":false,
      "oauth":"",
      "registerMethod":"default:username-password",
      "username":"983132@qq.com",
      "nickname":"",
      "company":"",
      "photo":"https://usercontents.authing.co/authing-avatar.png",
      "token":"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJkYXRhIjp7ImVtYWlsIjoiOTgzMTMyQHFxLmNvbSIsImlxxxxxxxxx",
      "phone":"",
      "tokenExpiredAt":"2020-01-11T08:08:18.000Z",
      "loginsCount":1,
      "lastIP":"::1",
      "signedUp":"2019-12-27T08:08:18.115Z",
      "blocked":false,
      "isDeleted":false
    }
}

// 返回 ticket
{
    code: 200,
    message: '获取 session 用户信息成功',
    data: {
      ticket: "xxxxdjdkxxxxx",
      nickname: "xxxx",
      photo: "https://usercontents.authing.co/authing-avatar.png"
    }
}

如果查询到 session,Authing trackSession 将返回用户的昵称头像(用于展示目的)以及用于换取用户信息的 ticket:

你可以在前端展示用户昵称和头像,如下图所示:

POST
https://core.authing.cn/oauth/sso/mobile/exchangeUserInfoWithTicket

使用 ticket 换取用户信息

使用 ticket 换取用户信息,此接口需要用户池密钥,请在后端调用

Headers
content-type
REQUIRED
string

application/json

Body Paramter
ticket
REQUIRED
string

trackSession 获取的 ticket

secret
REQUIRED
string

用户池密钥

userPoolId
REQUIRED
string

用户池 ID

200: OK
{ 
   "code":200,
   "message":"换取用户信息成功",
   "data":{ 
      "_id":"5e05bbf2d51b3761d5c71070",
      "email":"983132@qq.com",
      "emailVerified":false,
      "oauth":"",
      "registerMethod":"default:username-password",
      "username":"983132@qq.com",
      "nickname":"",
      "company":"",
      "photo":"https://usercontents.authing.co/authing-avatar.png",
      "token":"eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJkYXRhIjp7ImVtYWlsIjoiOTgzMTMyQHFxLmNvbSIsImlxxxxxxxxx",
      "phone":"",
      "tokenExpiredAt":"2020-01-11T08:08:18.000Z",
      "loginsCount":1,
      "lastIP":"::1",
      "signedUp":"2019-12-27T08:08:18.115Z",
      "blocked":false,
      "isDeleted":false
   }
POST
https://core.authing.cn/oauth/sso/mobile/destorySession

销毁 session

此接口用于在一个移动应用客户端内销毁一个 session ,且用户需处于登录状态,在请求头中加上 authorization 请求头携带用户 token。由于存在多个应用,所以默认情况下只会销毁制定 App 的 session(trackSession 只要还有一个 App 有 session 就会查询到 session),如果你想清除所有 App 的 session,可以设置 destoryAll 为 true。

Headers
authorization
REQUIRED
string

登录用户的 token

content-type
REQUIRED
string

application/json

Body Paramter
deviceId
REQUIRED
string

设备 ID

userPoolId
REQUIRED
string

用户池 ID

200: OK
{
    code: 200,
    message: "销毁 session 成功!"
}

你应该在每次用户退出登录以及删除 App 的时候调用此接口。