Parcourir la source

:new: #2052 【公众号】增加微信电子发票报销方相关接口

1、查询报销发票信息
2、批量查询报销发票信息
3、报销方更新发票状态
4、报销方批量更新发票状态
mrxiao il y a 4 ans
Parent
commit
5155c8c4b1

+ 48 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/api/WxMpReimburseInvoiceService.java

@@ -0,0 +1,48 @@
+package me.chanjar.weixin.mp.api;
+
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.bean.invoice.reimburse.*;
+
+import java.util.List;
+
+/**
+ * 电子发票报销方相关接口
+ * 接口文档: https://developers.weixin.qq.com/doc/offiaccount/WeChat_Invoice/E_Invoice/Reimburser_API_List.html
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+public interface WxMpReimburseInvoiceService {
+
+  /**
+   * 查询报销发票信息
+   * @param request {@link InvoiceInfoRequest} 查询报销发票信息参数
+   * @return {@link InvoiceInfoResponse} 查询结果
+   * @throws WxErrorException 查询失败时
+   */
+  InvoiceInfoResponse getInvoiceInfo(InvoiceInfoRequest request) throws WxErrorException;
+
+
+  /**
+   * 批量查询报销发票信息
+   * @param request {@link InvoiceBatchRequest} 批量查询报销发票信息参数对象
+   * @return {@link InvoiceInfoResponse} 查询结果列表
+   * @throws WxErrorException 查询失败时
+   */
+  List<InvoiceInfoResponse> getInvoiceBatch(InvoiceBatchRequest request) throws WxErrorException;
+
+
+  /**
+   * 更新发票状态
+   * @param request {@link UpdateInvoiceStatusRequest} 更新发票状态参数
+   * @throws WxErrorException 更新失败时
+   */
+  void updateInvoiceStatus(UpdateInvoiceStatusRequest request) throws WxErrorException;
+
+
+  /**
+   * 批量更新发票状态
+   * @param request {@link UpdateStatusBatchRequest} 批量更新发票状态参数
+   * @throws WxErrorException 更新失败时
+   */
+  void updateStatusBatch(UpdateStatusBatchRequest request) throws WxErrorException;
+}

+ 12 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/api/WxMpService.java

@@ -534,6 +534,18 @@ public interface WxMpService extends WxService {
   WxImgProcService getImgProcService();
 
   /**
+   * 返回电子发票报销方相关接口
+   * @return WxMpReimburseInvoiceService
+   */
+  WxMpReimburseInvoiceService getReimburseInvoiceService();
+
+  /**
+   * .
+   * @param reimburseInvoiceService .
+   */
+  void setReimburseInvoiceService(WxMpReimburseInvoiceService reimburseInvoiceService);
+
+  /**
    * .
    *
    * @param kefuService .

+ 4 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/api/impl/BaseWxMpServiceImpl.java

@@ -129,6 +129,10 @@ public abstract class BaseWxMpServiceImpl<H, P> implements WxMpService, RequestH
   @Setter
   private WxOAuth2Service oAuth2Service = new WxMpOAuth2ServiceImpl(this);
 
+  @Getter
+  @Setter
+  private WxMpReimburseInvoiceService reimburseInvoiceService = new WxMpReimburseInvoiceServiceImpl(this);
+
   private Map<String, WxMpConfigStorage> configStorageMap;
 
   private int retrySleepMillis = 1000;

+ 43 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/api/impl/WxMpReimburseInvoiceServiceImpl.java

@@ -0,0 +1,43 @@
+package me.chanjar.weixin.mp.api.impl;
+
+import lombok.AllArgsConstructor;
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.api.WxMpReimburseInvoiceService;
+import me.chanjar.weixin.mp.api.WxMpService;
+import me.chanjar.weixin.mp.bean.invoice.reimburse.*;
+
+import java.util.List;
+
+import static me.chanjar.weixin.mp.enums.WxMpApiUrl.Invoice.*;
+
+/**
+ * 电子发票报销方相关接口实现
+ * 接口文档: https://developers.weixin.qq.com/doc/offiaccount/WeChat_Invoice/E_Invoice/Reimburser_API_List.html
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@AllArgsConstructor
+public class WxMpReimburseInvoiceServiceImpl implements WxMpReimburseInvoiceService {
+
+  private final WxMpService wxMpService;
+
+  @Override
+  public InvoiceInfoResponse getInvoiceInfo(InvoiceInfoRequest request) throws WxErrorException {
+    return InvoiceInfoResponse.fromJson(this.wxMpService.post(GET_INVOICE_INFO,request.toJson()));
+  }
+
+  @Override
+  public List<InvoiceInfoResponse> getInvoiceBatch(InvoiceBatchRequest request) throws WxErrorException {
+    return InvoiceInfoResponse.toList(this.wxMpService.post(GET_INVOICE_BATCH,request.toJson()));
+  }
+
+  @Override
+  public void updateInvoiceStatus(UpdateInvoiceStatusRequest request) throws WxErrorException {
+    this.wxMpService.post(UPDATE_INVOICE_STATUS,request.toJson());
+  }
+
+  @Override
+  public void updateStatusBatch(UpdateStatusBatchRequest request) throws WxErrorException {
+    this.wxMpService.post(UPDATE_STATUS_BATCH,request.toJson());
+  }
+}

+ 36 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/InvoiceBatchRequest.java

@@ -0,0 +1,36 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import com.google.gson.annotations.SerializedName;
+import lombok.Builder;
+import lombok.Data;
+import me.chanjar.weixin.mp.util.json.WxMpGsonBuilder;
+
+import java.io.Serializable;
+import java.util.List;
+
+/**
+ * <pre>
+ * 批量查询报销发票信息参数对象
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+@Builder
+public class InvoiceBatchRequest implements Serializable {
+
+  private static final long serialVersionUID = -9121443117105107231L;
+
+  /**
+   * 发票卡券的card_id
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("item_list")
+  private List<InvoiceInfoRequest> itemList;
+
+  public String toJson() {
+    return WxMpGsonBuilder.create().toJson(this);
+  }
+}

+ 34 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/InvoiceCommodityInfo.java

@@ -0,0 +1,34 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import lombok.Data;
+
+/**
+ * <pre>
+ * 发票商品信息
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+public class InvoiceCommodityInfo {
+
+  /**
+   * 项目(商品)名称
+   */
+  private String name;
+
+  /**
+   * 项目数量
+   */
+  private Integer num;
+
+  /**
+   * 项目单位
+   */
+  private String unit;
+
+  /**
+   * 单价,以分为单位
+   */
+  private Integer price;
+}

+ 48 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/InvoiceInfoRequest.java

@@ -0,0 +1,48 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+
+import com.google.gson.annotations.SerializedName;
+import lombok.Builder;
+import lombok.Data;
+import me.chanjar.weixin.mp.util.json.WxMpGsonBuilder;
+
+import java.io.Serializable;
+
+/**
+ * <pre>
+ * 查询报销发票信息参数对象
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+@Builder
+public class InvoiceInfoRequest implements Serializable {
+
+  private static final long serialVersionUID = 7854633127026139444L;
+
+
+  /**
+  * 发票卡券的card_id
+  * <pre>
+  * 是否必填: 是
+  * </pre>
+  */
+  @SerializedName("card_id")
+  private String cardId;
+
+
+  /**
+  * 发票卡券的加密code,和card_id共同构成一张发票卡券的唯一标识
+  * <pre>
+  * 是否必填: 是
+  * </pre>
+  */
+  @SerializedName("encrypt_code")
+  private String encryptCode;
+
+
+  public String toJson() {
+    return WxMpGsonBuilder.create().toJson(this);
+  }
+}

+ 79 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/InvoiceInfoResponse.java

@@ -0,0 +1,79 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import com.google.gson.JsonObject;
+import com.google.gson.annotations.SerializedName;
+import com.google.gson.reflect.TypeToken;
+import lombok.Data;
+import me.chanjar.weixin.common.util.json.GsonParser;
+import me.chanjar.weixin.mp.util.json.WxMpGsonBuilder;
+
+import java.util.List;
+
+/**
+ * <pre>
+ * 查询报销发票信息响应对象
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+public class InvoiceInfoResponse {
+
+  /**
+   * 发票ID
+   */
+  @SerializedName("card_id")
+  private String cardId;
+
+
+  /**
+   * 发票的有效期起始时间
+   */
+  @SerializedName("begin_time")
+  private Integer beginTime;
+
+  /**
+   * 发票的有效期截止时间
+   */
+  @SerializedName("end_time")
+  private Integer endTime;
+
+  /**
+   * 用户标识
+   */
+  private String openid;
+
+  /**
+   * 发票的类型
+   */
+  private String type;
+
+  /**
+   * 发票的收款方
+   */
+  private String payee;
+
+  /**
+   * 发票详情
+   */
+  private String detail;
+
+  /**
+   * 用户可在发票票面看到的主要信息
+   */
+  @SerializedName("user_info")
+  private InvoiceUserInfo userInfo;
+
+
+  public static InvoiceInfoResponse fromJson(String json) {
+    return WxMpGsonBuilder.create().fromJson(json, InvoiceInfoResponse.class);
+  }
+
+
+  public static List<InvoiceInfoResponse> toList(String json) {
+    JsonObject jsonObject = GsonParser.parse(json);
+    return WxMpGsonBuilder.create().fromJson(jsonObject.get("item_list").toString(),
+      new TypeToken<List<InvoiceInfoResponse>>() {
+      }.getType());
+  }
+}

+ 137 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/InvoiceUserInfo.java

@@ -0,0 +1,137 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import com.google.gson.annotations.SerializedName;
+import lombok.Data;
+
+import java.util.List;
+
+/**
+ * <pre>
+ * 用户可在发票票面看到的主要信息
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+public class InvoiceUserInfo {
+
+  /**
+   * 发票加税合计金额,以分为单位
+   */
+  private Integer fee;
+
+  /**
+   * 发票的抬头
+   */
+  private String title;
+
+  /**
+   * 开票时间
+   */
+  @SerializedName("billing_time")
+  private Integer billingTime;
+
+  /**
+   * 发票代码
+   */
+  @SerializedName("billing_no")
+  private String billingNo;
+
+  /**
+   * 发票号码
+   */
+  @SerializedName("billing_code")
+  private String billingCode;
+
+  /**
+   * 不含税金额,以分为单位
+   */
+  @SerializedName("fee_without_tax")
+  private Integer feeWithoutTax;
+
+  /**
+   * 税额,以分为单位
+   */
+  private Integer tax;
+
+  /**
+   * 发票对应的PDF_URL
+   */
+  @SerializedName("pdf_url")
+  private String pdfUrl;
+
+  /**
+   * 其它消费凭证附件对应的URL
+   */
+  @SerializedName("trip_pdf_url")
+  private String tripPdfUrl;
+
+  /**
+   * 发票报销状态
+   */
+  @SerializedName("reimburse_status")
+  private String reimburseStatus;
+
+  /**
+   * 校验码
+   */
+  @SerializedName("check_code")
+  private String checkCode;
+
+  /**
+   * 购买方纳税人识别号
+   */
+  @SerializedName("buyer_number")
+  private String buyerNumber;
+
+  /**
+   * 购买方地址、电话
+   */
+  @SerializedName("buyer_address_and_phone")
+  private String buyerAddressAndPhone;
+
+  /**
+   * 购买方开户行及账号
+   */
+  @SerializedName("buyer_bank_account")
+  private String buyerBankAccount;
+
+  /**
+   * 销售方纳税人识别号
+   */
+  @SerializedName("seller_number")
+  private String sellerNumber;
+
+  /**
+   * 销售方地址、电话
+   */
+  @SerializedName("seller_address_and_phone")
+  private String sellerAddressAndPhone;
+
+  /**
+   * 销售方开户行及账号
+   */
+  @SerializedName("seller_bank_account")
+  private String sellerBankAccount;
+
+  /**
+   * 备注
+   */
+  private String remarks;
+
+  /**
+   * 收款人
+   */
+  private String cashier;
+
+  /**
+   * 开票人
+   */
+  private String maker;
+
+  /**
+   * 商品信息结构
+   */
+  private List<InvoiceCommodityInfo> info;
+}
+

+ 56 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/UpdateInvoiceStatusRequest.java

@@ -0,0 +1,56 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import com.google.gson.annotations.SerializedName;
+import lombok.Builder;
+import lombok.Data;
+import me.chanjar.weixin.mp.util.json.WxMpGsonBuilder;
+
+import java.io.Serializable;
+
+/**
+ * <pre>
+ * 更新发票状态参数对象
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+@Builder
+public class UpdateInvoiceStatusRequest implements Serializable {
+
+  private static final long serialVersionUID = -4122242332481909977L;
+
+
+  /**
+   * 发票卡券的card_id
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("card_id")
+  private String cardId;
+
+
+  /**
+   * 发票卡券的加密code,和card_id共同构成一张发票卡券的唯一标识
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("encrypt_code")
+  private String encryptCode;
+
+
+  /**
+   * 发票报销状态
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("reimburse_status")
+  private String reimburseStatus;
+
+  public String toJson() {
+    return WxMpGsonBuilder.create().toJson(this);
+  }
+}

+ 53 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/bean/invoice/reimburse/UpdateStatusBatchRequest.java

@@ -0,0 +1,53 @@
+package me.chanjar.weixin.mp.bean.invoice.reimburse;
+
+import com.google.gson.annotations.SerializedName;
+import lombok.Builder;
+import lombok.Data;
+import me.chanjar.weixin.mp.util.json.WxMpGsonBuilder;
+
+import java.io.Serializable;
+import java.util.List;
+
+/**
+ * <pre>
+ * 批量更新发票状态参数对象
+ * </pre>
+ * @author <a href="https://github.com/mr-xiaoyu">xiaoyu</a>
+ * @since 2021-03-23
+ */
+@Data
+@Builder
+public class UpdateStatusBatchRequest implements Serializable {
+
+  private static final long serialVersionUID = 7016357689566912199L;
+  /**
+   * 用户openid
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  private String openid;
+
+  /**
+   * 发票报销状态
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("reimburse_status")
+  private String reimburseStatus;
+
+  /**
+   * 发票列表
+   * <pre>
+   * 是否必填: 是
+   * </pre>
+   */
+  @SerializedName("invoice_list")
+  private List<InvoiceInfoRequest> invoiceList;
+
+  public String toJson() {
+    return WxMpGsonBuilder.create().toJson(this);
+  }
+
+}

+ 20 - 0
weixin-java-mp/src/main/java/me/chanjar/weixin/mp/enums/WxMpApiUrl.java

@@ -1129,6 +1129,26 @@ public interface WxMpApiUrl {
      * 获取关联商户
      */
     GET_PAY_MCH_SET_BIZ_ATTR(API_DEFAULT_HOST_URL, "/card/invoice/setbizattr?action=get_pay_mch"),
+
+    /**
+     * 报销方查询报销发票信息
+     */
+    GET_INVOICE_INFO(API_DEFAULT_HOST_URL,"/card/invoice/reimburse/getinvoiceinfo"),
+
+    /**
+     * 报销方批量查询报销发票信息
+     */
+    GET_INVOICE_BATCH(API_DEFAULT_HOST_URL,"/card/invoice/reimburse/getinvoicebatch"),
+
+    /**
+     * 报销方更新发票状态
+     */
+    UPDATE_INVOICE_STATUS(API_DEFAULT_HOST_URL,"/card/invoice/reimburse/updateinvoicestatus"),
+
+    /**
+     * 报销方批量更新发票状态
+     */
+    UPDATE_STATUS_BATCH(API_DEFAULT_HOST_URL,"/card/invoice/reimburse/updatestatusbatch"),
     ;
     private final String prefix;
     private final String path;

+ 84 - 0
weixin-java-mp/src/test/java/me/chanjar/weixin/mp/api/impl/WxMpReimburseInvoiceServiceImplTest.java

@@ -0,0 +1,84 @@
+package me.chanjar.weixin.mp.api.impl;
+
+import com.google.gson.GsonBuilder;
+import com.google.inject.Inject;
+import lombok.extern.slf4j.Slf4j;
+import me.chanjar.weixin.common.error.WxErrorException;
+import me.chanjar.weixin.mp.api.WxMpService;
+import me.chanjar.weixin.mp.api.test.ApiTestModule;
+import me.chanjar.weixin.mp.bean.invoice.reimburse.*;
+import org.testng.annotations.Guice;
+import org.testng.annotations.Test;
+
+import java.util.ArrayList;
+import java.util.List;
+
+@Slf4j
+@Test(groups = "invoiceAPI")
+@Guice(modules = ApiTestModule.class)
+public class WxMpReimburseInvoiceServiceImplTest {
+
+  @Inject
+  protected WxMpService wxService;
+
+
+  public void getInvoiceInfo() throws WxErrorException {
+    InvoiceInfoRequest request = InvoiceInfoRequest.builder()
+      .cardId("pnAsy0sHQukID3E8d2IUdh2DbzZ4")
+      .encryptCode("O/mPnGTpBu22a1szmK2ogzhFPBh9eYzv2p70L8yzyynlTOEE9fSC4PXvOGuLIWfqZQXA0yBPVcbELCLySWjiLH0RYjMqE4S2bekki6Z2VUjWHGp+shbOkYZ4y9zR4SpGVT6Dyha0ezDMVw6dFMatoA==")
+      .build();
+
+    InvoiceInfoResponse response = this.wxService.getReimburseInvoiceService().getInvoiceInfo(request);
+
+    log.info("response: {}", new GsonBuilder().create().toJson(response));
+  }
+
+
+  public void getInvoiceBatch() throws WxErrorException {
+    List<InvoiceInfoRequest> invoices = new ArrayList<>();
+    InvoiceInfoRequest r = InvoiceInfoRequest.builder()
+      .cardId("pnAsy0sHQukID3E8d2IUdh2DbzZ4")
+      .encryptCode("O/mPnGTpBu22a1szmK2ogzhFPBh9eYzv2p70L8yzyynlTOEE9fSC4PXvOGuLIWfqZQXA0yBPVcbELCLySWjiLH0RYjMqE4S2bekki6Z2VUjWHGp+shbOkYZ4y9zR4SpGVT6Dyha0ezDMVw6dFMatoA==")
+      .build();
+    invoices.add(r);
+    r = InvoiceInfoRequest.builder()
+      .cardId("pnAsy0sHQukID3E8d2IUdh2DbzZ4")
+      .encryptCode("O/mPnGTpBu22a1szmK2ogzhFPBh9eYzv2p70L8yzyynlTOEE9fSC4PXvOGuLIWfqd+8BRcn/yz1GmRwW4LAccaL/dRsSc9RWXektgTHKnoHWHGp+shbOkYZ4y9zR4SpGVT6Dyha0ezDMVw6dFMatoA==")
+      .build();
+    invoices.add(r);
+
+    InvoiceBatchRequest request = InvoiceBatchRequest.builder().itemList(invoices).build();
+
+    List<InvoiceInfoResponse> responses = this.wxService.getReimburseInvoiceService().getInvoiceBatch(request);
+    log.info("responses: {}",new GsonBuilder().create().toJson(responses));
+  }
+
+
+  public void updateInvoiceStatus() throws WxErrorException {
+    UpdateInvoiceStatusRequest request = UpdateInvoiceStatusRequest.builder()
+      .cardId("**************")
+      .encryptCode("**************")
+      .reimburseStatus("INVOICE_REIMBURSE_INIT")
+      .build();
+
+    this.wxService.getReimburseInvoiceService().updateInvoiceStatus(request);
+  }
+
+  public void updateStatusBatch() throws WxErrorException {
+    List<InvoiceInfoRequest> invoices = new ArrayList<>();
+    InvoiceInfoRequest r = InvoiceInfoRequest.builder()
+      .cardId("**************")
+      .encryptCode("**************")
+      .build();
+    invoices.add(r);
+
+    UpdateStatusBatchRequest request = UpdateStatusBatchRequest.builder()
+      .invoiceList(invoices)
+      .openid("**************")
+      .reimburseStatus("INVOICE_REIMBURSE_INIT")
+      .build();
+
+    this.wxService.getReimburseInvoiceService().updateStatusBatch(request);
+  }
+
+}