手写海康OpenApi签名规范,实现手动调用api(sdk:artemis-http-client)

1. 前言:

artemis-http-client sdk 中提供获取门禁事件图片的方法,但实际图片访问地址为该响应的重定向地址
问题来了:虽然他提供了 sdk ,但没有办法通过 sdk 获取重定向的地址于是产生了本文,自己通过hutools的 httpUtil调用

2. 签名工具类

HikSignUtil 记得先引入一下 artemis-http-client

java 复制代码
package com.lxsy.util;

import cn.hutool.core.codec.Base64;
import cn.hutool.crypto.digest.DigestUtil;
import cn.hutool.crypto.digest.HMac;
import cn.hutool.crypto.digest.HmacAlgorithm;

import java.nio.charset.StandardCharsets;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.time.format.DateTimeFormatter;
import java.util.HashMap;
import java.util.Map;

/**
 * 海康 OpenAPI 签名工具类
 * 签名算法:HmacSHA256 + Base64
 */
public class HikSignUtil {

    private static final DateTimeFormatter RFC_1123_FORMATTER =
            DateTimeFormatter.RFC_1123_DATE_TIME.withZone(ZoneOffset.UTC);

    /**
     * 生成请求头(包含签名)
     *
     * @param appKey    应用Key
     * @param appSecret 应用Secret
     * @param url       请求的相对路径(如 /artemis/api/acs/v1/event/pictures)
     * @param body      请求体(JSON字符串)
     * @return 包含签名的请求头Map
     */
    public static Map<String, String> buildHeaders(String appKey, String appSecret, String url, String body) {
        // Content-MD5:请求体的MD5值,Base64编码
        String contentMd5 = Base64.encode(DigestUtil.md5(body));

        // Date 头(RFC1123)
        String date = RFC_1123_FORMATTER.format(ZonedDateTime.now(ZoneOffset.UTC));

        // 签名字符串拼接
        // httpHeaders = HTTP METHOD + "\n" + Accept + "\n" + Content-MD5 + "\n" + Content-Type + "\n" + Date + "\n"
        // customHeaders = "x-ca-key" + ":" + appKey + "\n"
        // signString = httpHeaders + customHeaders + url
        StringBuilder signBuilder = new StringBuilder();
        signBuilder.append("POST").append("\n");           // HTTP METHOD
        signBuilder.append("*/*").append("\n");            // Accept
        signBuilder.append(contentMd5).append("\n");       // Content-MD5
        signBuilder.append("application/json").append("\n"); // Content-Type
        signBuilder.append(date).append("\n");             // Date
        signBuilder.append("x-ca-key:").append(appKey).append("\n"); // customHeaders
        signBuilder.append(url);                           // url

        String signString = signBuilder.toString();

        // 使用 HmacSHA256 算法计算签名
        HMac hmac = new HMac(HmacAlgorithm.HmacSHA256, appSecret.getBytes(StandardCharsets.UTF_8));
        String signature = Base64.encode(hmac.digest(signString));

        // 构建请求头
        Map<String, String> headers = new HashMap<>();
        headers.put("Accept", "*/*");
        headers.put("Content-Type", "application/json");
        headers.put("Content-MD5", contentMd5);
        headers.put("Date", date);
        headers.put("x-ca-key", appKey);
        headers.put("x-ca-signature", signature);
        headers.put("x-ca-signature-headers", "x-ca-key");

        return headers;
    }
}

3. 使用

示例1

java 复制代码
protected static final ArtemisConfig artemisConfig = new ArtemisConfig("192.XX.13.XX:443", "123", "123");


    /**
     * 使用自建 HttpClient 调用海康接口,便于排查网络/签名问题
     */
    public static PageResponseResult<EventInfoDto> getEventListWithHttpClient(EventsRequest eventsRequest) throws Exception {
        if (eventsRequest == null) {
            eventsRequest = new EventsRequest();
        }

        String url = "/artemis/api/acs/v2/door/events";
        String fullUrl = "https://" + artemisConfig.getHost() + url;
        String body = JSON.toJSONString(eventsRequest);

        Map<String, String> headers = HikSignUtil.buildHeaders(
                artemisConfig.getAppKey(),
                artemisConfig.getAppSecret(),
                url,
                body
        );

        try (CloseableHttpClient httpClient = createHttpClient(false)) {
            HttpPost httpPost = new HttpPost(fullUrl);
            headers.forEach(httpPost::setHeader);
            httpPost.setEntity(new StringEntity(body, ContentType.APPLICATION_JSON));

            try (CloseableHttpResponse response = httpClient.execute(httpPost)) {
                int status = response.getStatusLine().getStatusCode();
                String respBody = response.getEntity() == null ? null : EntityUtils.toString(response.getEntity());
                if (status == HttpStatus.SC_OK) {
                    return checkResp(respBody, new TypeReference<PageResponseResult<EventInfoDto>>() {
                    });
                }
                throw new Exception("获取事件列表失败,状态码: " + status + ", 响应: " + respBody);
            }
        }
    }

示例2

java 复制代码
protected static final ArtemisConfig artemisConfig = new ArtemisConfig("192.XX.13.XX:443", "123", "123");
    
public static String pictures(String svrIndexCode, String picUri) throws Exception {
        String url = "/artemis/api/acs/v1/event/pictures";
        String fullUrl = "https://" + artemisConfig.getHost() + url;

        // 构建请求体
        PicturesRequest request = new PicturesRequest(svrIndexCode, picUri);
        String body = JSON.toJSONString(request);

        // 使用签名工具生成请求头
        Map<String, String> headers = HikSignUtil.buildHeaders(
                artemisConfig.getAppKey(),
                artemisConfig.getAppSecret(),
                url,
                body
        );

        try (CloseableHttpClient httpClient = createHttpClient(false)) {
            HttpPost httpPost = new HttpPost(fullUrl);
            headers.forEach(httpPost::setHeader);
            httpPost.setEntity(new StringEntity(body, ContentType.APPLICATION_JSON));

            try (CloseableHttpResponse response = httpClient.execute(httpPost)) {
                int status = response.getStatusLine().getStatusCode();
                if (status == HttpStatus.SC_MOVED_TEMPORARILY
                        || status == HttpStatus.SC_MOVED_PERMANENTLY
                        || status == HttpStatus.SC_SEE_OTHER
                        || status == HttpStatus.SC_TEMPORARY_REDIRECT
                        || status == HTTP_STATUS_PERMANENT_REDIRECT) {
                    Header locationHeader = response.getFirstHeader("Location");
                    if (locationHeader != null && StrUtil.isNotBlank(locationHeader.getValue())) {
                        return locationHeader.getValue();
                    }
                    throw new Exception("302重定向但未获取到Location头");
                }

                String respBody = response.getEntity() == null ? null : EntityUtils.toString(response.getEntity());
                throw new Exception("获取图片失败,状态码: " + status + ", 响应: " + respBody);
            }
        }
    }
相关推荐
SmartRadio5 分钟前
ESP32-S3 双模式切换实现:兼顾手机_路由器连接与WiFi长距离通信
开发语言·网络·智能手机·esp32·长距离wifi
_.Switch24 分钟前
东方财富股票数据JS逆向:secids字段和AES加密实战
开发语言·前端·javascript·网络·爬虫·python·ecmascript
金色光环1 小时前
FreeModbus释放底层的 TCP 监听端口
服务器·网络·tcp/ip
数智化精益手记局1 小时前
拆解物料管理erp系统的核心功能,看物料管理erp系统如何解决库存积压与缺料难题
大数据·网络·人工智能·安全·信息可视化·精益工程
发光小北3 小时前
Modbus TCP 转 Profibus DP 网关如何应用?
网络协议
灰子学技术3 小时前
Envoy HTTP 过滤器处理技术文档
网络·网络协议·http
Olivia051405145 小时前
Voohu:音频变压器的屏蔽接地技术对50Hz工频噪声抑制的影响
网络·机器人·信息与通信
byoass5 小时前
智巢AI知识库深度解析:企业文档管理从大海捞针到精准狙击的进化之路
开发语言·网络·人工智能·安全·c#·云计算
zhihuishuxia__6 小时前
Multiplex通讯(多路复用通讯)
网络·图像处理·数码相机·计算机视觉·自动化
勤劳的进取家6 小时前
数据链路层基础
网络·学习·算法