腾讯云SDK并发调用优化方案
目录
一、概述
二、 网关的使用
2.1 核心代码
三、腾讯云SDK依赖包的改造
一、概述
此网关主要用于协调腾讯云SDK调用的QPS消耗,使得多个腾讯云用户资源能得到最大限度的利用。避免直接使用腾讯云SDK 时,在较大并发情况下导致接口调用异常。网关的工作流程如下图所示:

如上图所示,各个客户端在发起腾讯云SDK调用时,请求统一先发到网关,网关会根据现有的腾讯云账户资源使用情况,通过负载均衡算法,选择一个合适的腾讯云账户来执行请求,将请求转发到腾讯云服务,从而保证了腾讯云用户资源的最大利用。在这个过程中,如果暂时未找到可用的腾讯云用户,则会阻塞线程,直到有可用的账户时再将线程唤醒放行,避免了在较大并发量时直接调用SDK,而导致接口报错的情况发生。
二、 网关的使用
2.1 核心代码
RequestLimitFilter.java
package com.tencentcloudapi.gateway.filter;import com.tencentcloudapi.common.Sign;
import com.tencentcloudapi.common.exception.TencentCloudSDKException;
import com.tencentcloudapi.gateway.api.dto.UserInfo;
import com.tencentcloudapi.gateway.api.service.UserManageService;
import io.micrometer.common.util.StringUtils;
import jakarta.annotation.Resource;
import lombok.extern.slf4j.Slf4j;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.stereotype.Component;
import org.springframework.util.CollectionUtils;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;import javax.xml.bind.DatatypeConverter;
import java.nio.charset.StandardCharsets;
import java.sql.Date;
import java.text.SimpleDateFormat;
import java.util.*;
import java.util.concurrent.Semaphore;/*** @Description 限流过滤器* @Author miller.Lai* @Date 2023-11-06 10:23*/
@Component
@Slf4j
public class RequestLimitFilter implements GlobalFilter, Ordered {@Resourceprivate UserManageService userManageService;@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {ServerHttpRequest request = exchange.getRequest();// 打印请求路径String requestUri = request.getPath().pathWithinApplication().value();log.info("接受到请求,请求路径:{}", requestUri);List<String> authorizationList = request.getHeaders().get("authorization");List<String> actionList = request.getHeaders().get("X-TC-Action");// 如果请求头中存在 authorization 信息,则要进行替换,以适应现有的接口TPS限制策略if (!CollectionUtils.isEmpty(authorizationList) && StringUtils.isNotEmpty(authorizationList.get(0))&&!CollectionUtils.isEmpty(actionList) && StringUtils.isNotEmpty(actionList.get(0))) {UserInfo userInfo = new UserInfo();// 接口名称String action = null;try {action = actionList.get(0);log.info("当前调用API名称:{}", action);// 获取可用的腾讯秘钥,这是一个阻塞方法userInfo = userManageService.getAvailableUserInfo(action);String secretId = userInfo.getSecretInfo().getSecretId();String secretKey = userInfo.getSecretInfo().getSecretKey();// 根据可用的秘钥对请求头中的认证信息做重新生成String signedHeaders = "content-type;host";SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");sdf.setTimeZone(TimeZone.getTimeZone("UTC"));// 接口请求所在秒钟String timestamp = String.valueOf(System.currentTimeMillis() / 1000L);String date = sdf.format(new Date(Long.valueOf(timestamp + "000")));String service = request.getHeaders().get("service") != null ?  request.getHeaders().get("service").get(0) : "";String credentialScope = date + "/" + service + "/tc3_request";String stringToSign = request.getHeaders().get("stringToSign") != null ? new String(Base64.getDecoder().decode(Objects.requireNonNull(request.getHeaders().get("stringToSign")).get(0).getBytes())) : "";try {byte[] secretDate = Sign.hmac256(("TC3" + secretKey).getBytes(StandardCharsets.UTF_8), date);byte[] secretService = Sign.hmac256(secretDate, service);byte[] secretSigning = Sign.hmac256(secretService, "tc3_request");String signature = DatatypeConverter.printHexBinary(Sign.hmac256(secretSigning, stringToSign)).toLowerCase();String authorization = "TC3-HMAC-SHA256 Credential=" + secretId + "/" + credentialScope + ", SignedHeaders=" + signedHeaders + ", Signature=" + signature;exchange.getRequest().mutate().headers(httpHeaders -> {// 去除自定义的请求头httpHeaders.remove("service");httpHeaders.remove("stringToSign");// 去除不合法的认证信息httpHeaders.remove("authorization");// 塞入有效的认证信息httpHeaders.add("authorization", authorization);});} catch (TencentCloudSDKException e) {throw new RuntimeException(e);}log.info("线程 {} 的请求时间:{} 毫秒,secretId:{}",Thread.currentThread().getName(),System.currentTimeMillis(),secretId);UserInfo finalUserInfo = userInfo;String finalAction = action;// 如果上述过着正常获取信号量的许可String hasAcquired = userManageService.getHasAcquiredThreadLocal().get();userManageService.getHasAcquiredThreadLocal().remove();return chain.filter(exchange).then( Mono.fromRunnable(() -> {// 接口逻辑执行完毕后释放信号量锁if("1".equals(hasAcquired)){finalUserInfo.getInterfaceInfo(finalAction).getSemaphore().release();log.info("线程 {} 已释放线程锁",Thread.currentThread().getName());}}));} catch (Exception e) {// 在发生错误时释放信号量锁// 将当前线程标记为已获取许可String hasAcquired = userManageService.getHasAcquiredThreadLocal().get();if("1".equals(hasAcquired)){userInfo.getInterfaceInfo(action).getSemaphore().release();userManageService.getHasAcquiredThreadLocal().remove();log.info("线程 {} 已释放线程锁",Thread.currentThread().getName());}throw new RuntimeException(e);}}else{return chain.filter(exchange);}}@Overridepublic int getOrder() {return 1;}
}
UserManageServiceImpl.java
package com.tencentcloudapi.gateway.api.service;import com.fasterxml.jackson.core.type.TypeReference;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.tencentcloudapi.gateway.api.dto.UserInfo;
import com.tencentcloudapi.gateway.api.dto.InterfaceInfo;
import io.micrometer.common.util.StringUtils;
import jakarta.annotation.PostConstruct;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.core.io.ClassPathResource;
import org.springframework.core.io.Resource;
import org.springframework.stereotype.Service;import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.TimeUnit;/*** @author Miller.Lai* @description:  腾讯云用户管理实现类* @date 2024-01-25 17:13:10*/
@Service
public class UserManageServiceImpl implements UserManageService {private  List<UserInfo>  userInfos;@Value("${tencentcloud.api.authorization.file:}")private String authorizationJsonUrl;private ThreadLocal<String> hasAcquiredThreadLocal ;public ThreadLocal<String> getHasAcquiredThreadLocal() {return hasAcquiredThreadLocal;}@PostConstructpublic void init(){try {InputStream inputStreams = null;// 如果有指定authorization文件路径,则按指定的路径找配置文件if (StringUtils.isNotBlank(authorizationJsonUrl)) {inputStreams = new FileInputStream(authorizationJsonUrl);} else {// 从resource目录下加载JSON文件Resource resource = new ClassPathResource("authorization.json");inputStreams = resource.getInputStream();}// 使用Jackson的ObjectMapper将JSON数组内容映射为List对象ObjectMapper objectMapper = new ObjectMapper();userInfos = objectMapper.readValue(inputStreams, new TypeReference<>() {});hasAcquiredThreadLocal =new ThreadLocal<>();} catch (IOException e) {throw new RuntimeException(e);}}/*** 随机获取一个可用的用户* @param action* @return*/@Overridepublic UserInfo getAvailableUserInfo(String action) {// 可用的用户列表List<UserInfo> availableUserInfos = new ArrayList<>();// 计算总权重,即总并发量int totalWeight = 0;// 过滤可用的腾讯用户,过滤条件: action匹配for (int i = 0; i < userInfos.size(); i++) {UserInfo userInfo =userInfos.get(i);// 当前用户是否可用boolean available = false;for (int j = 0; j < userInfo.getInterfaces().size(); j++) {InterfaceInfo interfaceInfo  = userInfo.getInterfaces().get(j);if(interfaceInfo.getAction().equals(action)){available = true;totalWeight += interfaceInfo.getMaxTPS();break;}}// 如果当前用户可用,则加入列表if(available){availableUserInfos.add(userInfo);}}// 如果没找到可用用户,说明用户接口配置文件存在问题if(availableUserInfos.size() == 0 ){throw new RuntimeException("未找到可用的腾讯用户,请检查接口配置文件");}// 根据总权重生成权重随机数,0 ~ totalWeight-1int randomWeight = new Random().nextInt(totalWeight);// 根据权重值选择对应的用户int currentWeight = 0;for (int i = 0; i < availableUserInfos.size(); i++) {UserInfo userInfo = availableUserInfos.get(i);for (int j = 0; j < userInfo.getInterfaces().size(); j++) {InterfaceInfo interfaceInfo = userInfo.getInterfaces().get(j);// 找到对应的接口if (interfaceInfo.getAction().equals(action)) {currentWeight += interfaceInfo.getMaxTPS();// 如果当前接口的当前权重 > 随机权重, 则使用当前用户if (currentWeight > randomWeight) {// 给线程加锁,这是一个阻塞方法,如果当前用户的并发数达到上限,则当前线程会被阻塞try {interfaceInfo.getSemaphore().tryAcquire(30, TimeUnit.SECONDS);// 将当前线程标记为已获取许可hasAcquiredThreadLocal.set("1");} catch (InterruptedException e) {throw new RuntimeException(e);} finally {return userInfo;}}break;}}}return null;}
}
authorization.json
[{"secretInfo": {"secretId": "AKIDpiYK2xxxxxxxxxxxxxxxxxxbUyOk2W","secretKey": "dFlSKDBDXXXXXXXXXXXXXXXXXXYdyBE5"},"interfaces": [{"action": "RecognizeTableAccurateOCR","maxTPS": 2},{"action": "VehicleLicenseOCR","maxTPS": 10},{"action": "DriverLicenseOCR","maxTPS": 10},{"action": "MLIDPassportOCR","maxTPS": 5},{"action": "HmtResidentPermitOCR","maxTPS": 20},{"action": "MainlandPermitOCR","maxTPS": 20}]},{"secretInfo": {"secretId": "AKIDJh9fxxxxxxxxxxxxxxxxxxxxxv8IOR","secretKey": "00HaowzxxxxxxxxxxxxxxxxxxxxxjMp8b"},"interfaces": [{"action": "RecognizeTableAccurateOCR","maxTPS": 2},{"action": "VehicleLicenseOCR","maxTPS": 10},{"action": "DriverLicenseOCR","maxTPS": 10},{"action": "MLIDPassportOCR","maxTPS": 5},{"action": "HmtResidentPermitOCR","maxTPS": 20},{"action": "MainlandPermitOCR","maxTPS": 20}]}
]application-dev.yml
spring:application:name: uap-gatewaycloud:gateway:routes:- id: tencentcloud-routeuri: https://ocr.tencentcloudapi.compredicates:- Path=/tencentcloudapi/**filters:- StripPrefix=1main:web-application-type: reactive
server:port: 9000# 腾讯云用户接口权限配置文件
#tencentcloud:
#  api:
#    authorization:
#      file: d://authorization.json
三、腾讯云SDK依赖包的改造
修改 com.tencentcloudapi.common.AbstractClient 类中 REMOTE_SERVER_ADDRESS 的值,指向实际的网关地址,如下所示:
/** Copyright (c) 2018 THL A29 Limited, a Tencent company. All Rights Reserved.** Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at**     http://www.apache.org/licenses/LICENSE-2.0** Unless required by applicable law or agreed to in writing,* software distributed under the License is distributed on an* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY* KIND, either express or implied.  See the License for the* specific language governing permissions and limitations* under the License.*/package com.tencentcloudapi.common;import com.google.gson.Gson;
import com.google.gson.GsonBuilder;
import com.google.gson.JsonSyntaxException;
import com.google.gson.reflect.TypeToken;
import com.tencentcloudapi.common.exception.TencentCloudSDKException;
import com.tencentcloudapi.common.http.HttpConnection;
import com.tencentcloudapi.common.profile.ClientProfile;
import com.tencentcloudapi.common.profile.HttpProfile;
import okhttp3.*;
import okhttp3.Headers.Builder;import javax.crypto.Mac;
import javax.net.ssl.SSLContext;
import javax.xml.bind.DatatypeConverter;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Type;
import java.net.InetSocketAddress;
import java.net.Proxy;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.security.SecureRandom;
import java.sql.Date;
import java.text.SimpleDateFormat;
import java.util.*;public abstract class AbstractClient {/*** 远程服务地址,根据实际情况修改* 格式为:http://ip:port/tencentcloudapi*/public static String REMOTE_SERVER_ADDRESS = "http://localhost:9000/tencentcloudapi";public static final int HTTP_RSP_OK = 200;public static final String SDK_VERSION = "SDK_JAVA_3.1.699";private Credential credential;private ClientProfile profile;private String endpoint;private String service;private String region;private String path;private String sdkVersion;private String apiVersion;public Gson gson;private TCLog log;private HttpConnection httpConnection;public AbstractClient(String endpoint, String version, Credential credential, String region) {this(endpoint, version, credential, region, new ClientProfile());}static {String remoteServerAddress = System.getenv("REMOTE_SERVER_ADDRESS");if(remoteServerAddress != null){AbstractClient.REMOTE_SERVER_ADDRESS = remoteServerAddress;}}public AbstractClient(String endpoint,String version,Credential credential,String region,ClientProfile profile) {this.credential = credential;this.profile = profile;this.endpoint = endpoint;this.service = endpoint.split("\\.")[0];this.region = region;this.path = "/";this.sdkVersion = AbstractClient.SDK_VERSION;this.apiVersion = version;this.gson = new GsonBuilder().excludeFieldsWithoutExposeAnnotation().create();this.log = new TCLog(getClass().getName(), profile.isDebug());this.httpConnection = new HttpConnection(this.profile.getHttpProfile().getConnTimeout(),this.profile.getHttpProfile().getReadTimeout(),this.profile.getHttpProfile().getWriteTimeout());this.httpConnection.addInterceptors(this.log);this.trySetProxy(this.httpConnection);warmup();}public void setRegion(String region) {this.region = region;}public String getRegion() {return this.region;}public void setClientProfile(ClientProfile profile) {this.profile = profile;}public ClientProfile getClientProfile() {return this.profile;}public void setCredential(Credential credential) {this.credential = credential;}public Credential getCredential() {return this.credential;}/*** Use post/json with tc3-hmac-sha256 signature to call any action. Ignore request method and* signature method defined in profile.** @param action Name of action to be called.* @param jsonPayload Parameters of action serialized in json string format.* @return Raw response from API if request succeeded, otherwise an exception will be raised*     instead of raw response* @throws TencentCloudSDKException*/public String call(String action, String jsonPayload) throws TencentCloudSDKException {HashMap<String, String> headers = this.getHeaders();headers.put("X-TC-Action", action);headers.put("Content-Type", "application/json; charset=utf-8");byte[] requestPayload = jsonPayload.getBytes(StandardCharsets.UTF_8);String authorization = this.getAuthorization(headers, requestPayload);headers.put("Authorization", authorization);String url = REMOTE_SERVER_ADDRESS + this.path;return this.getResponseBody(url, headers, requestPayload);}/*** Use post application/octet-stream with tc3-hmac-sha256 signature to call specific action.* Ignore request method and signature method defined in profile.** @param action Name of action to be called.* @param headers Parameters of the action, will be put in http header.* @param body octet-stream binary body.* @return Raw response from API if request succeeded, otherwise an exception will be raised*     instead of raw response* @throws TencentCloudSDKException*/public String callOctetStream(String action, HashMap<String, String> headers, byte[] body)throws TencentCloudSDKException {headers.putAll(this.getHeaders());headers.put("X-TC-Action", action);headers.put("Content-Type", "application/octet-stream; charset=utf-8");String authorization = this.getAuthorization(headers, body);headers.put("Authorization", authorization);String url = REMOTE_SERVER_ADDRESS + this.path;return this.getResponseBody(url, headers, body);}private HashMap<String, String> getHeaders() {HashMap<String, String> headers = new HashMap<String, String>();String timestamp = String.valueOf(System.currentTimeMillis() / 1000);headers.put("X-TC-Timestamp", timestamp);headers.put("X-TC-Version", this.apiVersion);headers.put("X-TC-Region", this.getRegion());headers.put("X-TC-RequestClient", SDK_VERSION);headers.put("Host", this.getEndpoint());String token = this.credential.getToken();if (token != null && !token.isEmpty()) {headers.put("X-TC-Token", token);}if (this.profile.isUnsignedPayload()) {headers.put("X-TC-Content-SHA256", "UNSIGNED-PAYLOAD");}if (null != this.profile.getLanguage()) {headers.put("X-TC-Language", this.profile.getLanguage().getValue());}return headers;}private String getAuthorization(HashMap<String, String> headers, byte[] body)throws TencentCloudSDKException {String endpoint = this.getEndpoint();// always use post tc3-hmac-sha256 signature process// okhttp always set charset even we don't specify it,// to ensure signature be correct, we have to set it here as well.String contentType = headers.get("Content-Type");byte[] requestPayload = body;String canonicalUri = "/";String canonicalQueryString = "";String canonicalHeaders = "content-type:" + contentType + "\nhost:" + endpoint + "\n";String signedHeaders = "content-type;host";String hashedRequestPayload = "";if (this.profile.isUnsignedPayload()) {hashedRequestPayload = Sign.sha256Hex("UNSIGNED-PAYLOAD".getBytes(StandardCharsets.UTF_8));} else {hashedRequestPayload = Sign.sha256Hex(requestPayload);}String canonicalRequest =HttpProfile.REQ_POST+ "\n"+ canonicalUri+ "\n"+ canonicalQueryString+ "\n"+ canonicalHeaders+ "\n"+ signedHeaders+ "\n"+ hashedRequestPayload;String timestamp = headers.get("X-TC-Timestamp");SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");sdf.setTimeZone(TimeZone.getTimeZone("UTC"));String date = sdf.format(new Date(Long.valueOf(timestamp + "000")));String service = endpoint.split("\\.")[0];String credentialScope = date + "/" + service + "/" + "tc3_request";String hashedCanonicalRequest =Sign.sha256Hex(canonicalRequest.getBytes(StandardCharsets.UTF_8));String stringToSign ="TC3-HMAC-SHA256\n" + timestamp + "\n" + credentialScope + "\n" + hashedCanonicalRequest;String secretId = this.credential.getSecretId();String secretKey = this.credential.getSecretKey();byte[] secretDate = Sign.hmac256(("TC3" + secretKey).getBytes(StandardCharsets.UTF_8), date);byte[] secretService = Sign.hmac256(secretDate, service);byte[] secretSigning = Sign.hmac256(secretService, "tc3_request");String signature =DatatypeConverter.printHexBinary(Sign.hmac256(secretSigning, stringToSign)).toLowerCase();return "TC3-HMAC-SHA256 "+ "Credential="+ secretId+ "/"+ credentialScope+ ", "+ "SignedHeaders="+ signedHeaders+ ", "+ "Signature="+ signature;}private String getResponseBody(String url, HashMap<String, String> headers, byte[] body)throws TencentCloudSDKException {Builder hb = new Builder();for (String key : headers.keySet()) {hb.add(key, headers.get(key));}Response resp = this.httpConnection.postRequest(url, body, hb.build());if (resp.code() != AbstractClient.HTTP_RSP_OK) {String msg = "response code is " + resp.code() + ", not 200";log.info(msg);throw new TencentCloudSDKException(msg, "", "ServerSideError");}String respbody = null;try {respbody = resp.body().string();} catch (IOException e) {String msg ="Cannot transfer response body to string, because Content-Length is too large, or Content-Length and stream length disagree.";log.info(msg);throw new TencentCloudSDKException(msg, "", e.getClass().getName());}JsonResponseModel<JsonResponseErrModel> errResp = null;try {Type errType = new TypeToken<JsonResponseModel<JsonResponseErrModel>>() {}.getType();errResp = gson.fromJson(respbody, errType);} catch (JsonSyntaxException e) {String msg = "json is not a valid representation for an object of type";log.info(msg);throw new TencentCloudSDKException(msg, "", e.getClass().getName());}if (errResp.response.error != null) {throw new TencentCloudSDKException(errResp.response.error.message, errResp.response.requestId, errResp.response.error.code);}return respbody;}private void trySetProxy(HttpConnection conn) {String host = this.profile.getHttpProfile().getProxyHost();int port = this.profile.getHttpProfile().getProxyPort();if (host == null || host.isEmpty()) {return;}Proxy proxy = new Proxy(Proxy.Type.HTTP, new InetSocketAddress(host, port));conn.setProxy(proxy);final String username = this.profile.getHttpProfile().getProxyUsername();final String password = this.profile.getHttpProfile().getProxyPassword();if (username == null || username.isEmpty()) {return;}conn.setProxyAuthenticator(new Authenticator() {@Overridepublic Request authenticate(Route route, Response response) throws IOException {String credential = Credentials.basic(username, password);return response.request().newBuilder().header("Proxy-Authorization", credential).build();}});}protected String internalRequest(AbstractModel request, String actionName)throws TencentCloudSDKException {Response okRsp = null;String endpoint = this.getEndpoint();String[] binaryParams = request.getBinaryParams();String sm = this.profile.getSignMethod();String reqMethod = this.profile.getHttpProfile().getReqMethod();// currently, customized params only can be supported via post json tc3-hmac-sha256HashMap<String, Object> customizedParams = request.any();if (customizedParams.size() > 0) {if (binaryParams.length > 0) {throw new TencentCloudSDKException("WrongUsage: Cannot post multipart with customized parameters.");}if (sm.equals(ClientProfile.SIGN_SHA1) || sm.equals(ClientProfile.SIGN_SHA256)) {throw new TencentCloudSDKException("WrongUsage: Cannot use HmacSHA1 or HmacSHA256 with customized parameters.");}if (reqMethod.equals(HttpProfile.REQ_GET)) {throw new TencentCloudSDKException("WrongUsage: Cannot use get method with customized parameters.");}}if (binaryParams.length > 0 || sm.equals(ClientProfile.SIGN_TC3_256)) {okRsp = doRequestWithTC3(endpoint, request, actionName);} else if (sm.equals(ClientProfile.SIGN_SHA1) || sm.equals(ClientProfile.SIGN_SHA256)) {okRsp = doRequest(endpoint, request, actionName);} else {throw new TencentCloudSDKException("Signature method " + sm + " is invalid or not supported yet.");}if (okRsp.code() != AbstractClient.HTTP_RSP_OK) {String msg = "response code is " + okRsp.code() + ", not 200";log.info(msg);throw new TencentCloudSDKException(msg, "", "ServerSideError");}String strResp = null;try {strResp = okRsp.body().string();} catch (IOException e) {String msg = "Cannot transfer response body to string, because Content-Length is too large, or Content-Length and stream length disagree.";log.info(msg);throw new TencentCloudSDKException(msg, "", endpoint.getClass().getName());}JsonResponseModel<JsonResponseErrModel> errResp = null;try {Type errType = new TypeToken<JsonResponseModel<JsonResponseErrModel>>() {}.getType();errResp = gson.fromJson(strResp, errType);} catch (JsonSyntaxException e) {String msg = "json is not a valid representation for an object of type";log.info(msg);throw new TencentCloudSDKException(msg, "", e.getClass().getName());}if (errResp.response.error != null) {throw new TencentCloudSDKException(errResp.response.error.message,errResp.response.requestId,errResp.response.error.code);}return strResp;}private Response doRequest(String endpoint, AbstractModel request, String action)throws TencentCloudSDKException {HashMap<String, String> param = new HashMap<String, String>();request.toMap(param, "");String strParam = this.formatRequestData(action, param);String reqMethod = this.profile.getHttpProfile().getReqMethod();String url = REMOTE_SERVER_ADDRESS + this.path;if (reqMethod.equals(HttpProfile.REQ_GET)) {return this.httpConnection.getRequest(url + "?" + strParam);} else if (reqMethod.equals(HttpProfile.REQ_POST)) {return this.httpConnection.postRequest(url, strParam);} else {throw new TencentCloudSDKException("Method only support (GET, POST)");}}private Response doRequestWithTC3(String endpoint, AbstractModel request, String action)throws TencentCloudSDKException {String httpRequestMethod = this.profile.getHttpProfile().getReqMethod();if (httpRequestMethod == null) {throw new TencentCloudSDKException("Request method should not be null, can only be GET or POST");}String contentType = "application/x-www-form-urlencoded";byte[] requestPayload = "".getBytes(StandardCharsets.UTF_8);HashMap<String, String> params = new HashMap<String, String>();request.toMap(params, "");String[] binaryParams = request.getBinaryParams();if (binaryParams.length > 0) {httpRequestMethod = HttpProfile.REQ_POST;String boundary = UUID.randomUUID().toString();// okhttp always set charset even we don't specify it,// to ensure signature be correct, we have to set it here as well.contentType = "multipart/form-data; charset=utf-8" + "; boundary=" + boundary;try {requestPayload = getMultipartPayload(request, boundary);} catch (Exception e) {throw new TencentCloudSDKException("Failed to generate multipart. because: " + e);}} else if (httpRequestMethod.equals(HttpProfile.REQ_POST)) {requestPayload = AbstractModel.toJsonString(request).getBytes(StandardCharsets.UTF_8);// okhttp always set charset even we don't specify it,// to ensure signature be correct, we have to set it here as well.contentType = "application/json; charset=utf-8";}String canonicalUri = "/";String canonicalQueryString = this.getCanonicalQueryString(params, httpRequestMethod);String canonicalHeaders = "content-type:" + contentType + "\nhost:" + endpoint + "\n";String signedHeaders = "content-type;host";String hashedRequestPayload = "";if (this.profile.isUnsignedPayload()) {hashedRequestPayload = Sign.sha256Hex("UNSIGNED-PAYLOAD".getBytes(StandardCharsets.UTF_8));} else {hashedRequestPayload = Sign.sha256Hex(requestPayload);}String canonicalRequest =httpRequestMethod+ "\n"+ canonicalUri+ "\n"+ canonicalQueryString+ "\n"+ canonicalHeaders+ "\n"+ signedHeaders+ "\n"+ hashedRequestPayload;String timestamp = String.valueOf(System.currentTimeMillis() / 1000);SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd");sdf.setTimeZone(TimeZone.getTimeZone("UTC"));String date = sdf.format(new Date(Long.valueOf(timestamp + "000")));String service = endpoint.split("\\.")[0];String credentialScope = date + "/" + service + "/" + "tc3_request";String hashedCanonicalRequest =Sign.sha256Hex(canonicalRequest.getBytes(StandardCharsets.UTF_8));String stringToSign ="TC3-HMAC-SHA256\n" + timestamp + "\n" + credentialScope + "\n" + hashedCanonicalRequest;String secretId = this.credential.getSecretId();String secretKey = this.credential.getSecretKey();byte[] secretDate = Sign.hmac256(("TC3" + secretKey).getBytes(StandardCharsets.UTF_8), date);byte[] secretService = Sign.hmac256(secretDate, service);byte[] secretSigning = Sign.hmac256(secretService, "tc3_request");String signature =DatatypeConverter.printHexBinary(Sign.hmac256(secretSigning, stringToSign)).toLowerCase();String authorization ="TC3-HMAC-SHA256 "+ "Credential="+ secretId+ "/"+ credentialScope+ ", "+ "SignedHeaders="+ signedHeaders+ ", "+ "Signature="+ signature;String url = REMOTE_SERVER_ADDRESS + this.path;Builder hb = new Builder();hb.add("Content-Type", contentType).add("Host", endpoint).add("Authorization", authorization).add("X-TC-Action", action).add("X-TC-Timestamp", timestamp).add("X-TC-Version", this.apiVersion).add("X-TC-RequestClient", SDK_VERSION);if (null != this.getRegion()) {hb.add("X-TC-Region", this.getRegion());}String token = this.credential.getToken();if (token != null && !token.isEmpty()) {hb.add("X-TC-Token", token);}if (this.profile.isUnsignedPayload()) {hb.add("X-TC-Content-SHA256", "UNSIGNED-PAYLOAD");}if (null != this.profile.getLanguage()) {hb.add("X-TC-Language", this.profile.getLanguage().getValue());}if (null != service ) {hb.add("service", service);}if (null != stringToSign ) {hb.add("stringToSign", new String(Base64.getEncoder().encode(stringToSign.getBytes())));}Headers headers = hb.build();if (httpRequestMethod.equals(HttpProfile.REQ_GET)) {return this.httpConnection.getRequest(url + "?" + canonicalQueryString, headers);} else if (httpRequestMethod.equals(HttpProfile.REQ_POST)) {return this.httpConnection.postRequest(url, requestPayload, headers);} else {throw new TencentCloudSDKException("Method only support GET, POST");}}private byte[] getMultipartPayload(AbstractModel request, String boundary) throws Exception {ByteArrayOutputStream baos = new ByteArrayOutputStream();String[] binaryParams = request.getBinaryParams();for (Map.Entry<String, byte[]> entry : request.getMultipartRequestParams().entrySet()) {baos.write("--".getBytes(StandardCharsets.UTF_8));baos.write(boundary.getBytes(StandardCharsets.UTF_8));baos.write("\r\n".getBytes(StandardCharsets.UTF_8));baos.write("Content-Disposition: form-data; name=\"".getBytes(StandardCharsets.UTF_8));baos.write(entry.getKey().getBytes(StandardCharsets.UTF_8));if (Arrays.asList(binaryParams).contains(entry.getKey())) {baos.write("\"; filename=\"".getBytes(StandardCharsets.UTF_8));baos.write(entry.getKey().getBytes(StandardCharsets.UTF_8));baos.write("\"\r\n".getBytes(StandardCharsets.UTF_8));} else {baos.write("\"\r\n".getBytes(StandardCharsets.UTF_8));}baos.write("\r\n".getBytes(StandardCharsets.UTF_8));baos.write(entry.getValue());baos.write("\r\n".getBytes(StandardCharsets.UTF_8));}if (baos.size() != 0) {baos.write("--".getBytes(StandardCharsets.UTF_8));baos.write(boundary.getBytes(StandardCharsets.UTF_8));baos.write("--\r\n".getBytes(StandardCharsets.UTF_8));}byte[] bytes = baos.toByteArray();baos.close();return bytes;}private String getCanonicalQueryString(HashMap<String, String> params, String method)throws TencentCloudSDKException {if (method != null && method.equals(HttpProfile.REQ_POST)) {return "";}StringBuilder queryString = new StringBuilder("");for (Map.Entry<String, String> entry : params.entrySet()) {String v;try {v = URLEncoder.encode(entry.getValue(), "UTF8");} catch (UnsupportedEncodingException e) {throw new TencentCloudSDKException("UTF8 is not supported." + e.getMessage());}queryString.append("&").append(entry.getKey()).append("=").append(v);}if (queryString.length() == 0) {return "";} else {return queryString.toString().substring(1);}}private String formatRequestData(String action, Map<String, String> param)throws TencentCloudSDKException {param.put("Action", action);param.put("RequestClient", this.sdkVersion);param.put("Nonce", String.valueOf(Math.abs(new SecureRandom().nextInt())));param.put("Timestamp", String.valueOf(System.currentTimeMillis() / 1000));param.put("Version", this.apiVersion);if (this.credential.getSecretId() != null && (!this.credential.getSecretId().isEmpty())) {param.put("SecretId", this.credential.getSecretId());}if (this.region != null && (!this.region.isEmpty())) {param.put("Region", this.region);}if (this.profile.getSignMethod() != null && (!this.profile.getSignMethod().isEmpty())) {param.put("SignatureMethod", this.profile.getSignMethod());}if (this.credential.getToken() != null && (!this.credential.getToken().isEmpty())) {param.put("Token", this.credential.getToken());}if (null != this.profile.getLanguage()) {param.put("Language", this.profile.getLanguage().getValue());}String endpoint = this.getEndpoint();String sigInParam =Sign.makeSignPlainText(new TreeMap<String, String>(param),this.profile.getHttpProfile().getReqMethod(),endpoint,this.path);String sigOutParam =Sign.sign(this.credential.getSecretKey(), sigInParam, this.profile.getSignMethod());String strParam = "";try {for (Map.Entry<String, String> entry : param.entrySet()) {strParam +=(URLEncoder.encode(entry.getKey(), "utf-8")+ "="+ URLEncoder.encode(entry.getValue(), "utf-8")+ "&");}strParam += ("Signature=" + URLEncoder.encode(sigOutParam, "utf-8"));} catch (UnsupportedEncodingException e) {throw new TencentCloudSDKException(e.getClass().getName() + "-" + e.getMessage());}return strParam;}/** warm up, try to avoid unnecessary cost in the first request */private void warmup() {try {// it happens in SDK signature process.// first invoke costs around 250 ms.Mac.getInstance("HmacSHA1");Mac.getInstance("HmacSHA256");// it happens inside okhttp, but I think any https framework/package will do the same.// first invoke costs around 150 ms.SSLContext sslContext = SSLContext.getInstance("TLS");sslContext.init(null, null, null);} catch (Exception e) {// ignore but print message to consolee.printStackTrace();}}private String getEndpoint() {// in case user has reset endpoint after init this clientif (null != this.profile.getHttpProfile().getEndpoint()) {return this.profile.getHttpProfile().getEndpoint();} else {// protected abstract String getService();// use this.getService() from overrided subclass will be betterreturn this.service + "." + this.profile.getHttpProfile().getRootDomain();}}/*** 请注意购买类接口谨慎调用,可能导致多次购买* 仅幂等接口推荐使用** @param req* @param retryTimes* @throws TencentCloudSDKException*/public Object retry(AbstractModel req, int retryTimes) throws TencentCloudSDKException {if (retryTimes < 0 || retryTimes > 10) {throw new TencentCloudSDKException("The number of retryTimes supported is 0 to 10.", "", "ClientSideError");}Class cls = this.getClass();String methodName = req.getClass().getSimpleName().replace("Request", "");Method method;try {method = cls.getMethod(methodName, req.getClass());} catch (NoSuchMethodException e) {throw new TencentCloudSDKException(e.toString(), "", "ClientSideError");}do {try {return method.invoke(this, req);} catch (IllegalAccessException e) {throw new TencentCloudSDKException(e.toString(), "", "ClientSideError");} catch (InvocationTargetException e) {if (retryTimes == 0) {throw (TencentCloudSDKException) e.getTargetException();}}try {Thread.sleep(1000);} catch (InterruptedException e) {throw new TencentCloudSDKException(e.toString(), "", "ClientSideError");}} while (--retryTimes >= 0);return null;}
}
将源码中上述类修改后重新达成jar包即可。
本人近十年JAVA架构设计经验,长期从事IT技术资源整合。有志于自我技术提升、需要最新IT技术课程的小伙伴,可私信联系我 ,粉丝一律白菜价
相关文章:
 
腾讯云SDK并发调用优化方案
目录 一、概述 二、 网关的使用 2.1 核心代码 三、腾讯云SDK依赖包的改造 一、概述 此网关主要用于协调腾讯云SDK调用的QPS消耗,使得多个腾讯云用户资源能得到最大限度的利用。避免直接使用腾讯云SDK 时,在较大并发情况下导致接口调用异常。网关的工…...
 
【排序算法】C语言实现随机快排,巨详细讲解
文章目录 🚀前言🚀快排的核心过程partition(划分过程)🚀快排1.0🚀随机快速排序🚀稳定性 🚀前言 铁子们好啊!继续我们排序算法今天要讲的是快排,通常大家所说…...
 
Java强训day13(选择题编程题)
选择题 编程题 题目1 import java.util.Scanner;public class Main {public static void main(String[] args) {Scanner sc new Scanner(System.in);String s sc.nextLine();char[] c s.toCharArray();int i 0;int t 0;while (i < c.length) {if (c[i] ! \") {…...
 
搭建WebGL开发环境
前言 本篇文章介绍如何搭建WebGL开发环境 WebGL WebGL的技术规范继承自免费和开源的OpenGL ES标准,从某种意义上说,WebGL就是Web版的OpenGL ES,而OpenGL ES是从OpenGL中派生出来的。他们的应用环境有区别,一般来说:…...
 
学习嵌入式第十五天之结构体
用变量a给出下面的定义 a) 一个整型数(An integer) //int a;b) 一个指向整型数的指针(A pointer to an integer) //int *a;c) 一个指向指针的的指针,它指向的指针是指向一个整型数(A pointer to a poin…...
【HDFS】一天一个RPC系列--updateBlockForPipeline
本文目标是: 弄清updateBlockForPipeline这个RPC的作用。弄清updateBlockForPipeline RPC的使用场景,代码里的调用点。一、updateBlockForPipeline的作用 其定义在ClientProtocol接口里,是Client与NameNode之间的接口。 看其代码注释描述: 为一个under construction状态下…...
测试面试题(0101设计测试用例关键)
1. 测试计划 测试范围,本次改动的模块,新增了哪些功能测试策略,包含测试依据,测试准入标准,准出标准,测试重点及方法(确认功能的优先级),测试工具的选择测试管理&#x…...
 
C++ 数论相关题目:高斯消元解异或线性方程组
输入一个包含 n 个方程 n 个未知数的异或线性方程组。 方程组中的系数和常数为 0 或 1 ,每个未知数的取值也为 0 或 1 。 求解这个方程组。 异或线性方程组示例如下: M[1][1]x[1] ^ M[1][2]x[2] ^ … ^ M[1][n]x[n] B[1] M[2][1]x[1] ^ M[2][2]x[2]…...
 
嵌入式学习第十四天
1.结构体(2): (1)结构体类型定义 (2)结构体变量的定义 (3)结构体元素的访问 (4)结构体的存储: 内存对齐: char 按照1字节对齐 …...
 
氢气泄漏检测仪使用方法:守护安全,从细节开始
随着科技的发展,我们的生活和工作环境中充满了各种潜在的危险。其中,氢气作为一种清洁能源,其使用日益广泛,但同时也带来了泄漏的风险。为了确保我们的安全,了解并正确使用氢气泄漏检测仪至关重要。下面将详细介绍氢气…...
 
【前端web入门第二天】01 html语法实现列表与表格_合并单元格
html语法实现列表与表格 文章目录: 1.列表 1.1 无序列表1.2 有序列表1.3 定义列表 2.表格 2.1 表格基本结构2.2 表格结构标签2.3 合并单元格 写在最前,第二天学习目标: 列表 表格 表单 元素为嵌套关系 1.列表 作用:布局内容排列整齐的区域。 列表分类:无序列表、有序列表…...
 
推荐系统|排序_MMOE
MMOE MMOE是指Multi-gate Mixture-of-Experts 注意看Expert后面加了s,说明了有多个专家。 而在MMOE中专家是指用来对输入特征计算的神经网络,每个神经网络根据输入计算出来的向量都会有所不同。 MMOE的低层 MMOE的上一层 通过MMOE的低层算出的向量和权…...
Redis拒绝连接的原因与解决方式
Redis拒绝连接的原因与解决方式 在某些情况下,当尝试从外部计算机连接到运行在保护模式下的Redis服务器时,您可能会遇到如下的错误信息: Caused by: org.redisson.client.RedisException: DENIED Redis is running in protected mode becau…...
Neo4j在java中的使用
1.Neo4j访问的两种方式 嵌入式数据库服务器模式(通过REST的访问) 它是由应用程序的性质(neo4j是独立服务器 还是和程序在一起),性能,监控和数据安全性来决定架构选择。 An embedded database(嵌入式数据库) 嵌入式Neo4j数据库…...
 
故障诊断 | 一文解决,CNN卷积神经网络故障诊断(Matlab)
文章目录 效果一览文章概述专栏介绍源码设计参考资料效果一览 文章概述 故障诊断 | 一文解决,CNN卷积神经网络故障诊断(Matlab) 专栏介绍 订阅【故障诊断】专栏,不定期更新机器学习和深度学习在故障诊断中的应用;订阅...
uniapp-app使用富文本编辑器editor
使用的是uniapp内置组件的表单组件editor:editor 组件 | uni-app官网 (dcloud.net.cn) editor 组件对应的 editorContext 实例:editorContext | uni-app官网 (dcloud.net.cn) 文档上写的也不是特别详细,还以为得npm,但没npm也能用…...
20240131 大模型快讯
//社区生态// 国内首个音视频多媒体大模型万兴“天幕”正式发布。万兴科技发布国内首个音视频多媒体大模型万兴“天幕”,支持多种语言,实现音视频创作闭环。 //行业落地// 全球首款搭载AI大模型的MPV智能座舱发布。江淮全新MPV瑞风RF8上市发布…...
MySQL原理(二)存储引擎(2)MyISAM
一、MyISAM介绍 1、介绍: MyISAM引擎是MySQL5.5版本之前的数据库所默认的数据表引擎。每一个采用MyISAM引擎的数据表在实际存储中都是由三个文件组成,分别是frm文件保存表的结构,MYD文件保存表的数据、MYI文件保存表的索引,文件…...
P1088 [NOIP2004 普及组] 火星人题解
题目 人类终于登上了火星的土地并且见到了神秘的火星人。人类和火星人都无法理解对方的语言,但是我们的科学家发明了一种用数字交流的方法。这种交流方法是这样的,首先,火星人把一个非常大的数字告诉人类科学家,科学家破解这个数…...
 
Python面向对象编程:探索代码的结构之美
文章目录 一、引言二、为什么学习面向对象编程2.1 提高代码的可维护性:通过封装、继承和多态实现模块化设计2.2 提升代码的复用性:通过类和对象的创建实现代码的重用 三、类和对象的基本概念3.1 类和对象的定义和关系:类是对象的模板…...
 
手游刚开服就被攻击怎么办?如何防御DDoS?
开服初期是手游最脆弱的阶段,极易成为DDoS攻击的目标。一旦遭遇攻击,可能导致服务器瘫痪、玩家流失,甚至造成巨大经济损失。本文为开发者提供一套简洁有效的应急与防御方案,帮助快速应对并构建长期防护体系。 一、遭遇攻击的紧急应…...
反向工程与模型迁移:打造未来商品详情API的可持续创新体系
在电商行业蓬勃发展的当下,商品详情API作为连接电商平台与开发者、商家及用户的关键纽带,其重要性日益凸显。传统商品详情API主要聚焦于商品基本信息(如名称、价格、库存等)的获取与展示,已难以满足市场对个性化、智能…...
 
练习(含atoi的模拟实现,自定义类型等练习)
一、结构体大小的计算及位段 (结构体大小计算及位段 详解请看:自定义类型:结构体进阶-CSDN博客) 1.在32位系统环境,编译选项为4字节对齐,那么sizeof(A)和sizeof(B)是多少? #pragma pack(4)st…...
 
MongoDB学习和应用(高效的非关系型数据库)
一丶 MongoDB简介 对于社交类软件的功能,我们需要对它的功能特点进行分析: 数据量会随着用户数增大而增大读多写少价值较低非好友看不到其动态信息地理位置的查询… 针对以上特点进行分析各大存储工具: mysql:关系型数据库&am…...
 
多种风格导航菜单 HTML 实现(附源码)
下面我将为您展示 6 种不同风格的导航菜单实现,每种都包含完整 HTML、CSS 和 JavaScript 代码。 1. 简约水平导航栏 <!DOCTYPE html> <html lang"zh-CN"> <head><meta charset"UTF-8"><meta name"viewport&qu…...
 
Golang——6、指针和结构体
指针和结构体 1、指针1.1、指针地址和指针类型1.2、指针取值1.3、new和make 2、结构体2.1、type关键字的使用2.2、结构体的定义和初始化2.3、结构体方法和接收者2.4、给任意类型添加方法2.5、结构体的匿名字段2.6、嵌套结构体2.7、嵌套匿名结构体2.8、结构体的继承 3、结构体与…...
4. TypeScript 类型推断与类型组合
一、类型推断 (一) 什么是类型推断 TypeScript 的类型推断会根据变量、函数返回值、对象和数组的赋值和使用方式,自动确定它们的类型。 这一特性减少了显式类型注解的需要,在保持类型安全的同时简化了代码。通过分析上下文和初始值,TypeSc…...
云原生周刊:k0s 成为 CNCF 沙箱项目
开源项目推荐 HAMi HAMi(原名 k8s‑vGPU‑scheduler)是一款 CNCF Sandbox 级别的开源 K8s 中间件,通过虚拟化 GPU/NPU 等异构设备并支持内存、计算核心时间片隔离及共享调度,为容器提供统一接口,实现细粒度资源配额…...
 
C++实现分布式网络通信框架RPC(2)——rpc发布端
有了上篇文章的项目的基本知识的了解,现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...
 
JDK 17 序列化是怎么回事
如何序列化?其实很简单,就是根据每个类型,用工厂类调用。逐个完成。 没什么漂亮的代码,只有有效、稳定的代码。 代码中调用toJson toJson 代码 mapper.writeValueAsString ObjectMapper DefaultSerializerProvider 一堆实…...
