跳到主要内容

基于服务账号生成鉴权令牌

只有使用应用设备状态检测服务时才需要配置此章节。

概述

服务账号(Service Account)是一种可实现服务器与服务器之间接口鉴权的账号,在华为开发者联盟的API Console上创建服务账号,您可根据返回的公私钥在业务应用中生成鉴权令牌,调用支持此类鉴权的华为公开API。

服务账号令牌为JWT(JSON Web Token)格式字符串,JWT数据格式包括三个部分:

  • Header(头部)
  • Payload(负载)
  • Signature(签名)

这三个部分通过“.”进行连接,其中Signature为通过SHA256withRSA算法对Header与Payload拼接的字符串签名生成的字符串。

示例

eyJra*****JjNjBjMXXX.
eyJhd*****JodHRXXX.
BRNss*****7az5oU7-Zp5g9X2WJVXXX

更多JWT的相关知识请参见Introduction to JSON Web Tokens

开发步骤

  1. 创建服务账号密钥文件。

    开发者需要在华为开发者联盟的API Console上创建并下载服务账号的密钥文件,相关创建步骤请参见API Console操作指南-服务账号密钥

    您在开发者联盟需要申请开发者级的服务账号凭证。

    服务账号密钥样例文件形式可参考:

    {
    "project_id": "*****",
    "key_id": "*****",
    "private_key": "-----BEGIN PRIVATE KEY-----\nMIIJQgIBADANBgkqhkiG9w0BAQEFAASCCSwwggkoAgEAAoICAQCKw6kJKtCh7qmMvp1u1dI27z2TKZrPOzHbQaXO/Eez0AWZ2EN+ouF496R3pfo7fQXC1XOT/YTbVC4DNZwWSMA54fu3/AOCY9Zzyi46OK*****==\n-----END PRIVATE KEY-----\n",
    "sub_account": "*****",
    "auth_uri": "https://oauth-login.cloud.huawei.com/oauth2/v3/authorize",
    "token_uri": "https://oauth-login.cloud.huawei.com/oauth2/v3/token",
    "auth_provider_cert_uri": "https://oauth-login.cloud.huawei.com/oauth2/v3/certs",
    "client_cert_uri": "https://oauth-login.cloud.huawei.com/oauth2/v3/x509?client_id=*****"
    }
  2. 生成JWT Header数据。

    根据服务账号密钥文件中的key_id字段拼接以下JSON体,对JSON体进行BASE64编码。

    {
    "kid": "*****",
    "typ": "JWT",
    "alg": "PS256"
    }
    字段名描述
    kid服务账号密钥文件中key_id字段。
    typ数据类型,固定为:JWT。
    alg算法类型,固定为:PS256。
  3. 生成JWT Payload数据。

    根据服务账号密钥文件中的sub_account字段拼接以下JSON体,对JSON体进行BASE64编码。

    {
    "aud": "https://oauth-login.cloud.huawei.com/oauth2/v3/token",
    "iss": "*****",
    "exp": 1581410664,
    "iat": 1581407064
    }
    字段名描述
    aud固定为:https://oauth-login.cloud.huawei.com/oauth2/v3/token
    iss服务账号密钥文件中sub_account字段,标识数据生成者。
    expJWT到期UTC时间戳,建议比“iat”晚3600秒。
    iatJWT签发UTC时间戳,为自UTC时间1970年1月1日00:00:00的秒数(应用服务器时间需要校准为标准时间)。
  4. 生成JWT Token数据。

    将完成BASE64编码后的Header字符串与Payload字符串,通过“.”进行连接,并在开发者的应用中,通过服务账号密钥文件中的private_key(华为不进行存储,请您妥善保管),使用SHA256withRSA/PSS算法对拼接的字符串签名,最后将Header,Payload以及字符串签名通过“.”进行连接,即可得到Token数据。

示例代码

为了方便您生成服务账号鉴权令牌,我们提供了Java语言的示例代码,请按照说明替换参数运行。

如果您使用其他开发语言,请选择对应的JWT开源组件进行开发。

import com.alibaba.fastjson.JSONObject;
import org.apache.commons.codec.binary.Base64;
import org.bouncycastle.jce.provider.BouncyCastleProvider;

import java.nio.charset.StandardCharsets;
import java.security.*;
import java.security.spec.InvalidKeySpecException;
import java.security.spec.PKCS8EncodedKeySpec;
import java.util.Objects;

public class JWTGenerateDemo_test {
// please replace it with the private_key in your json file
// this is the plain text in this demo, please encrypt the private key in your code, only get the string between
// '-----BEGIN PRIVATE KEY-----\n' and '\n-----END PRIVATE KEY-----\n'
private static final String PRIVATE_KEY = "xxxx";

// please replace it with the sub_account in your json file
private static final String ISS = "xxx";

// please replace it with the key_id in your json file
private static final String KID = "xxx";

private static final String AUD = "https://oauth-login.cloud.huawei.com/oauth2/v3/token";

private static final String ALG_PS256 = "PS256";

private static final String DOT = ".";

private static PrivateKey getPrivateKey(String key) throws NoSuchAlgorithmException, InvalidKeySpecException {
PKCS8EncodedKeySpec keySpec = new PKCS8EncodedKeySpec(decodeBase64(key));
KeyFactory keyFactory = KeyFactory.getInstance("RSA");
return keyFactory.generatePrivate(keySpec);
}

private static byte[] decodeBase64(String Base64Str) {
return Base64.decodeBase64(Base64Str.getBytes(StandardCharsets.UTF_8));
}

private String createJwt()
throws NoSuchAlgorithmException, InvalidKeySpecException, InvalidKeyException, SignatureException {
long iat = System.currentTimeMillis() / 1000;
long exp = iat + 3600;//JWT有效期3600s

// jwt header
JSONObject header = new JSONObject();
header.put("alg", ALG_PS256);
header.put("kid", KID);
header.put("typ", "JWT");

// jwt payload
JSONObject payload = new JSONObject();
payload.put("aud", AUD);
payload.put("iss", ISS);
payload.put("exp", exp);
payload.put("iat", iat);

// jwt signature
byte[] encodeHeaderBytes = Base64.encodeBase64URLSafe(header.toString().getBytes(StandardCharsets.UTF_8));
byte[] encodePayloadBytes = Base64.encodeBase64URLSafe(payload.toString().getBytes(StandardCharsets.UTF_8));
String encodeHeader = new String(encodeHeaderBytes, StandardCharsets.UTF_8);
String encodePayload = new String(encodePayloadBytes, StandardCharsets.UTF_8);
String jwtHeaderAndPayload = encodeHeader + DOT + encodePayload;
Signature signatureInstance = Signature.getInstance("SHA256withRSA/PSS", new BouncyCastleProvider());
signatureInstance.initSign(getPrivateKey(PRIVATE_KEY));
signatureInstance.update(jwtHeaderAndPayload.getBytes(StandardCharsets.UTF_8));
String signature =
new String(Objects.requireNonNull(Base64.encodeBase64URLSafe(signatureInstance.sign())), StandardCharsets.UTF_8);

return jwtHeaderAndPayload + DOT + signature;
}

public static void main(String args[])
throws InvalidKeySpecException, NoSuchAlgorithmException, SignatureException, InvalidKeyException {
JWTGenerateDemo_test JWTGenerateDemo = new JWTGenerateDemo_test();
JWTGenerateDemo.createJwt();//获取鉴权令牌
}
}

执行main()函数即可。

推荐的java版本为java8,maven依赖如下:

<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.83</version>
</dependency>
<dependency>
<groupId>org.bouncycastle</groupId>
<artifactId>bcprov-jdk18on</artifactId>
<version>1.74</version>
</dependency>
<dependency>
<groupId>commons-codec</groupId>
<artifactId>commons-codec</artifactId>
<version>1.15</version>
</dependency>