基于服务账号生成鉴权令牌
只有使用应用设备状态检测服务时才需要配置此章节。
概述
服务账号(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。
开发步骤
-
创建服务账号密钥文件。
开发者需要在华为开发者联盟的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=*****"} -
生成JWT Header数据。
根据服务账号密钥文件中的key_id字段拼接以下JSON体,对JSON体进行BASE64编码。
{"kid": "*****","typ": "JWT","alg": "PS256"}字段名 描述 kid 服务账号密钥文件中key_id字段。 typ 数据类型,固定为:JWT。 alg 算法类型,固定为:PS256。 -
生成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字段,标识数据生成者。 exp JWT到期UTC时间戳,建议比“iat”晚3600秒。 iat JWT签发UTC时间戳,为自UTC时间1970年1月1日00:00:00的秒数(应用服务器时间需要校准为标准时间)。 -
生成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>