mirror of
				https://gitee.com/hhyykk/ipms-sjy.git
				synced 2025-11-04 12:18:42 +08:00 
			
		
		
		
	快递100 快递查询实现
This commit is contained in:
		@@ -55,10 +55,20 @@ public class TradeExpressQueryProperties {
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递100 配置项 TODO
 | 
			
		||||
     * 快递100 配置项
 | 
			
		||||
     */
 | 
			
		||||
    @Data
 | 
			
		||||
    public static class Kd100Config {
 | 
			
		||||
 | 
			
		||||
        /**
 | 
			
		||||
         * 快递 100 授权码
 | 
			
		||||
         */
 | 
			
		||||
        @NotEmpty(message = "快递 100 授权码配置项不能为空")
 | 
			
		||||
        private String customer;
 | 
			
		||||
        /**
 | 
			
		||||
         * 快递 100 授权 key
 | 
			
		||||
         */
 | 
			
		||||
        @NotEmpty(message = "快递 100 授权 Key 配置项不能为空")
 | 
			
		||||
        private String key;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -2,6 +2,8 @@ package cn.iocoder.yudao.module.trade.framework.delivery.core.convert;
 | 
			
		||||
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryRespDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100.Kd100ExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100.Kd100ExpressQueryRespDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kdniao.KdNiaoExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kdniao.KdNiaoExpressQueryRespDTO;
 | 
			
		||||
import org.mapstruct.Mapper;
 | 
			
		||||
@@ -16,5 +18,9 @@ public interface ExpressQueryConvert {
 | 
			
		||||
 | 
			
		||||
    List<ExpressQueryRespDTO> convertList(List<KdNiaoExpressQueryRespDTO.ExpressTrack> expressTrackList);
 | 
			
		||||
 | 
			
		||||
    List<ExpressQueryRespDTO> convertList2(List<Kd100ExpressQueryRespDTO.ExpressTrack> expressTrackList);
 | 
			
		||||
 | 
			
		||||
    KdNiaoExpressQueryReqDTO convert(ExpressQueryReqDTO dto);
 | 
			
		||||
 | 
			
		||||
    Kd100ExpressQueryReqDTO convert2(ExpressQueryReqDTO dto);
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,47 @@
 | 
			
		||||
package cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100;
 | 
			
		||||
 | 
			
		||||
import com.fasterxml.jackson.annotation.JsonInclude;
 | 
			
		||||
import com.fasterxml.jackson.annotation.JsonProperty;
 | 
			
		||||
import lombok.Data;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 快递 100 快递查询 Req DTO
 | 
			
		||||
 *
 | 
			
		||||
 * @author jason
 | 
			
		||||
 */
 | 
			
		||||
@Data
 | 
			
		||||
@JsonInclude(JsonInclude.Include.NON_NULL)
 | 
			
		||||
public class Kd100ExpressQueryReqDTO {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递公司编码
 | 
			
		||||
     */
 | 
			
		||||
    @JsonProperty("com")
 | 
			
		||||
    private String expressCompanyCode;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递单号
 | 
			
		||||
     */
 | 
			
		||||
    @JsonProperty("num")
 | 
			
		||||
    private String logisticsNo;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 收、寄件人的电话号码
 | 
			
		||||
     */
 | 
			
		||||
    private String phone;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 出发地城市
 | 
			
		||||
     */
 | 
			
		||||
    private String from;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 目的地城市,到达目的地后会加大监控频率
 | 
			
		||||
     */
 | 
			
		||||
    private String to;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 返回结果排序:desc降序(默认),asc 升序
 | 
			
		||||
     */
 | 
			
		||||
    private String order;
 | 
			
		||||
}
 | 
			
		||||
@@ -0,0 +1,59 @@
 | 
			
		||||
package cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100;
 | 
			
		||||
 | 
			
		||||
import com.fasterxml.jackson.annotation.JsonProperty;
 | 
			
		||||
import lombok.Data;
 | 
			
		||||
 | 
			
		||||
import java.util.List;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 快递 100 实时快递查询 Resp DTO 参见  <a href="https://api.kuaidi100.com/document/5f0ffb5ebc8da837cbd8aefc">快递 100 文档</a>
 | 
			
		||||
 *
 | 
			
		||||
 * @author jason
 | 
			
		||||
 */
 | 
			
		||||
@Data
 | 
			
		||||
public class Kd100ExpressQueryRespDTO {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递公司编码
 | 
			
		||||
     */
 | 
			
		||||
    @JsonProperty("com")
 | 
			
		||||
    private String expressCompanyCode;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递单号
 | 
			
		||||
     */
 | 
			
		||||
    @JsonProperty("nu")
 | 
			
		||||
    private String logisticsNo;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递单当前状态
 | 
			
		||||
     */
 | 
			
		||||
    private String state;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 查询结果, 失败返回 "false"
 | 
			
		||||
     */
 | 
			
		||||
    private String result;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 查询结果失败时的错误信息
 | 
			
		||||
     */
 | 
			
		||||
    private String message;
 | 
			
		||||
 | 
			
		||||
    @JsonProperty("data")
 | 
			
		||||
    private List<ExpressTrack> tracks;
 | 
			
		||||
 | 
			
		||||
    @Data
 | 
			
		||||
    public static class ExpressTrack {
 | 
			
		||||
        /**
 | 
			
		||||
         * 轨迹发生时间
 | 
			
		||||
         */
 | 
			
		||||
        @JsonProperty("time")
 | 
			
		||||
        private String time;
 | 
			
		||||
        /**
 | 
			
		||||
         * 轨迹描述
 | 
			
		||||
         */
 | 
			
		||||
        @JsonProperty("context")
 | 
			
		||||
        private String state;
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
@@ -1,19 +1,39 @@
 | 
			
		||||
package cn.iocoder.yudao.module.trade.framework.delivery.core.impl;
 | 
			
		||||
 | 
			
		||||
import cn.hutool.core.collection.CollUtil;
 | 
			
		||||
import cn.hutool.core.util.HexUtil;
 | 
			
		||||
import cn.hutool.crypto.digest.DigestUtil;
 | 
			
		||||
import cn.iocoder.yudao.framework.common.util.json.JsonUtils;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.config.TradeExpressQueryProperties;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.ExpressQueryProvider;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryRespDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100.Kd100ExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kd100.Kd100ExpressQueryRespDTO;
 | 
			
		||||
import lombok.extern.slf4j.Slf4j;
 | 
			
		||||
import org.springframework.http.*;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
import org.springframework.util.MultiValueMap;
 | 
			
		||||
import org.springframework.web.client.RestTemplate;
 | 
			
		||||
 | 
			
		||||
import java.util.Collections;
 | 
			
		||||
import java.util.List;
 | 
			
		||||
import java.util.Objects;
 | 
			
		||||
 | 
			
		||||
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.enums.ErrorCodeConstants.EXPRESS_API_QUERY_FAILED;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.enums.ErrorCodeConstants.EXPRESS_API_QUERY_ERROR;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.framework.delivery.core.convert.ExpressQueryConvert.INSTANCE;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 快递 100 服务商 TODO
 | 
			
		||||
 * 快递 100 服务商
 | 
			
		||||
 *
 | 
			
		||||
 * @author jason
 | 
			
		||||
 */
 | 
			
		||||
@Slf4j
 | 
			
		||||
public class Kd100ExpressQueryProvider implements ExpressQueryProvider {
 | 
			
		||||
 | 
			
		||||
    private static final String REAL_TIME_QUERY_URL = "https://poll.kuaidi100.com/poll/query.do";
 | 
			
		||||
    private final RestTemplate restTemplate;
 | 
			
		||||
 | 
			
		||||
    private final TradeExpressQueryProperties.Kd100Config config;
 | 
			
		||||
@@ -25,6 +45,63 @@ public class Kd100ExpressQueryProvider implements ExpressQueryProvider {
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public List<ExpressQueryRespDTO> realTimeQueryExpress(ExpressQueryReqDTO reqDTO) {
 | 
			
		||||
        return null;
 | 
			
		||||
        Kd100ExpressQueryReqDTO kd100ReqParam = INSTANCE.convert2(reqDTO);
 | 
			
		||||
        // 快递公司编码需要转成小写
 | 
			
		||||
        kd100ReqParam.setExpressCompanyCode(kd100ReqParam.getExpressCompanyCode().toLowerCase());
 | 
			
		||||
        Kd100ExpressQueryRespDTO respDTO = sendExpressQueryReq(REAL_TIME_QUERY_URL, kd100ReqParam,
 | 
			
		||||
                Kd100ExpressQueryRespDTO.class);
 | 
			
		||||
        log.debug("快递 100 接口 查询接口返回 {}", respDTO);
 | 
			
		||||
        if (Objects.equals("false", respDTO.getResult())) {
 | 
			
		||||
            log.error("快递 100 接口 返回失败 {} ", respDTO.getMessage());
 | 
			
		||||
            throw exception(EXPRESS_API_QUERY_FAILED, respDTO.getMessage());
 | 
			
		||||
        } else {
 | 
			
		||||
            if (CollUtil.isNotEmpty(respDTO.getTracks())) {
 | 
			
		||||
                return INSTANCE.convertList2(respDTO.getTracks());
 | 
			
		||||
            } else {
 | 
			
		||||
                return Collections.emptyList();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 发送快递 100 实时快递查询请求,可以作为通用快递 100 通用请求接口。 目前没有其它场景需要使用。暂时放这里
 | 
			
		||||
     * @param url 请求 url
 | 
			
		||||
     * @param req 对应请求的请求参数
 | 
			
		||||
     * @param respClass 对应请求的响应 class
 | 
			
		||||
     * @param <Req> 每个请求的请求结构 Req DTO
 | 
			
		||||
     * @param <Resp> 每个请求的响应结构 Resp DTO
 | 
			
		||||
     */
 | 
			
		||||
    private <Req, Resp> Resp sendExpressQueryReq(String url, Req req, Class<Resp> respClass) {
 | 
			
		||||
        // 请求头
 | 
			
		||||
        HttpHeaders headers = new HttpHeaders();
 | 
			
		||||
        headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
 | 
			
		||||
        // 生成签名
 | 
			
		||||
        String param = JsonUtils.toJsonString(req);
 | 
			
		||||
        String sign = generateReqSign(param, config.getKey(), config.getCustomer());
 | 
			
		||||
        log.debug("快递 100 快递 接口生成签名的: {}", sign);
 | 
			
		||||
        // 请求体
 | 
			
		||||
        MultiValueMap<String, String> requestBody = new LinkedMultiValueMap<>();
 | 
			
		||||
        requestBody.add("customer", config.getCustomer());
 | 
			
		||||
        requestBody.add("sign", sign);
 | 
			
		||||
        requestBody.add("param", param);
 | 
			
		||||
        log.debug("快递 100 接口的请求参数: {}", requestBody);
 | 
			
		||||
 | 
			
		||||
        HttpEntity<MultiValueMap<String, String>> requestEntity = new HttpEntity<>(requestBody, headers);
 | 
			
		||||
        // 发送请求
 | 
			
		||||
        ResponseEntity<String> responseEntity = restTemplate.exchange(url, HttpMethod.POST, requestEntity, String.class);
 | 
			
		||||
        log.debug("快递 100 接口响应结果 {}", responseEntity);
 | 
			
		||||
        // 处理响应
 | 
			
		||||
        if (responseEntity.getStatusCode().is2xxSuccessful()) {
 | 
			
		||||
            String response = responseEntity.getBody();
 | 
			
		||||
            return JsonUtils.parseObject(response, respClass);
 | 
			
		||||
        } else {
 | 
			
		||||
            throw exception(EXPRESS_API_QUERY_ERROR);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private String generateReqSign(String param, String key, String customer) {
 | 
			
		||||
        String plainText = String.format("%s%s%s", param, key, customer);
 | 
			
		||||
        log.debug("快递 100 接口待签名的数据 {}", plainText);
 | 
			
		||||
        return HexUtil.encodeHexStr(DigestUtil.md5(plainText), false);
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -11,7 +11,6 @@ import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryReq
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.ExpressQueryRespDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kdniao.KdNiaoExpressQueryReqDTO;
 | 
			
		||||
import cn.iocoder.yudao.module.trade.framework.delivery.core.dto.provider.kdniao.KdNiaoExpressQueryRespDTO;
 | 
			
		||||
import com.fasterxml.jackson.core.type.TypeReference;
 | 
			
		||||
import lombok.extern.slf4j.Slf4j;
 | 
			
		||||
import org.springframework.http.*;
 | 
			
		||||
import org.springframework.util.LinkedMultiValueMap;
 | 
			
		||||
@@ -23,6 +22,7 @@ import java.util.List;
 | 
			
		||||
 | 
			
		||||
import static cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil.exception;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.enums.ErrorCodeConstants.EXPRESS_API_QUERY_FAILED;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.enums.ErrorCodeConstants.EXPRESS_API_QUERY_ERROR;
 | 
			
		||||
import static cn.iocoder.yudao.module.trade.framework.delivery.core.convert.ExpressQueryConvert.INSTANCE;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
@@ -57,34 +57,28 @@ public class KdNiaoExpressQueryProvider implements ExpressQueryProvider {
 | 
			
		||||
        KdNiaoExpressQueryRespDTO respDTO = sendKdNiaoApiRequest(REAL_TIME_QUERY_URL, REAL_TIME_FREE_REQ_TYPE,
 | 
			
		||||
                kdNiaoReqData, KdNiaoExpressQueryRespDTO.class);
 | 
			
		||||
        log.debug("快递鸟即时查询接口返回 {}", respDTO);
 | 
			
		||||
        if (respDTO.getSuccess() && CollUtil.isNotEmpty(respDTO.getTracks())) {
 | 
			
		||||
            return INSTANCE.convertList(respDTO.getTracks());
 | 
			
		||||
        if(!respDTO.getSuccess()){
 | 
			
		||||
            throw exception(EXPRESS_API_QUERY_FAILED, respDTO.getReason());
 | 
			
		||||
        }else{
 | 
			
		||||
            return Collections.emptyList();
 | 
			
		||||
            if (CollUtil.isNotEmpty(respDTO.getTracks())) {
 | 
			
		||||
                return INSTANCE.convertList(respDTO.getTracks());
 | 
			
		||||
            }else{
 | 
			
		||||
                return Collections.emptyList();
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    private  <Req, Resp> Resp sendKdNiaoApiRequest(String url, String requestType, Req req,
 | 
			
		||||
                                                   Class<Resp> respClass){
 | 
			
		||||
        return sendKdNiaoApiRequest(url, requestType, req, respClass, null);
 | 
			
		||||
    }
 | 
			
		||||
    private  <Req, Resp> Resp sendKdNiaoApiRequest(String url, String requestType, Req req,
 | 
			
		||||
                                                   TypeReference<Resp> typeRefResp){
 | 
			
		||||
        return sendKdNiaoApiRequest(url, requestType, req, null, typeRefResp);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 快递鸟 通用的 API 请求
 | 
			
		||||
     * 快递鸟 通用的 API 请求, 暂时没有其他应用场景, 暂时放这里
 | 
			
		||||
     * @param url 请求 url
 | 
			
		||||
     * @param requestType 对应的请求指令 (快递鸟的RequestType)
 | 
			
		||||
     * @param req  对应请求的请求参数
 | 
			
		||||
     * @param respClass 对应请求的响应 class
 | 
			
		||||
     * @param typeRefResp 对应请求的响应  TypeReference
 | 
			
		||||
     * @param <Req> 每个请求的请求结构 Req DTO
 | 
			
		||||
     * @param <Resp> 每个请求的响应结构 Resp DTO
 | 
			
		||||
     */
 | 
			
		||||
    private  <Req, Resp> Resp sendKdNiaoApiRequest(String url, String requestType, Req req,
 | 
			
		||||
                                                   Class<Resp> respClass, TypeReference<Resp> typeRefResp){
 | 
			
		||||
                                                   Class<Resp> respClass){
 | 
			
		||||
        // 请求头
 | 
			
		||||
        HttpHeaders headers = new HttpHeaders();
 | 
			
		||||
        headers.setContentType(MediaType.APPLICATION_FORM_URLENCODED);
 | 
			
		||||
@@ -106,13 +100,9 @@ public class KdNiaoExpressQueryProvider implements ExpressQueryProvider {
 | 
			
		||||
        // 处理响应
 | 
			
		||||
        if (responseEntity.getStatusCode().is2xxSuccessful()) {
 | 
			
		||||
            String response = responseEntity.getBody();
 | 
			
		||||
            if (respClass != null && typeRefResp == null) {
 | 
			
		||||
                return JsonUtils.parseObject(response, respClass);
 | 
			
		||||
            } else {
 | 
			
		||||
                return JsonUtils.parseObject(response, typeRefResp);
 | 
			
		||||
            }
 | 
			
		||||
            return JsonUtils.parseObject(response, respClass);
 | 
			
		||||
        } else {
 | 
			
		||||
            throw exception(EXPRESS_API_QUERY_FAILED);
 | 
			
		||||
            throw exception(EXPRESS_API_QUERY_ERROR);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user