1. 程式人生 > >25、【支付模組開發】——將配置好的支付寶沙箱環境整合到我們的專案中以及支付介面的編寫

25、【支付模組開發】——將配置好的支付寶沙箱環境整合到我們的專案中以及支付介面的編寫

####1、將支付寶Demo中的相關檔案複製到我們的專案中:
首先,我們將Demo中src中的包及裡面的檔案複製到我們專案中
image.png
同樣,我們也要講zhifubao.properties這個配置檔案方法我們專案中的 resources目錄下:

image.png
加下來就是我們的jar包了~
首先我們先在webapp目錄下面新建lib資料夾:然後將Demo中如圖所示的四個jar包複製到lib目錄下。
image.png

至於剩下的jar我們只用Maven配置即可

<!-- alipay -->
  <dependencies>
    <dependency>
      <
groupId>
commons-codec</groupId> <artifactId>commons-codec</artifactId> <version>1.10</version> </dependency> <dependency> <groupId>commons-configuration</groupId> <artifactId>commons-configuration</artifactId>
<version>1.10</version> </dependency> <dependency> <groupId>commons-lang</groupId> <artifactId>commons-lang</artifactId> <version>2.6</version> </dependency> <dependency> <groupId>commons-
logging</groupId> <artifactId>commons-logging</artifactId> <version>1.1.1</version> </dependency> <dependency> <groupId>com.google.zxing</groupId> <artifactId>core</artifactId> <version>2.1</version> </dependency> <dependency> <groupId>com.google.code.gson</groupId> <artifactId>gson</artifactId> <version>2.3.1</version> </dependency> <dependency> <groupId>org.hamcrest</groupId> <artifactId>hamcrest-core</artifactId> <version>1.3</version> </dependency> </dependencies>

上面是jar包的配置,接下來我們在pom.xml裡面配置一個外掛,使專案編譯需要的jar在專案部署的時候自動配置到上面的lib資料夾下:

<build>
      <!-- geelynote maven的核心外掛之-complier外掛預設只支援編譯Java 1.4,因此需要加上支援高版本jre的配置,在pom.xml裡面加上 增加編譯外掛 -->
      <plugin>
        <groupId>org.apache.maven.plugins</groupId>
        <artifactId>maven-compiler-plugin</artifactId>
        <configuration>
          <source>1.8</source>
          <target>1.8</target>
          <encoding>UTF-8</encoding>
          <compilerArguments>
            <extdirs>${project.basedir}/src/main/webapp/WEB-INF/lib</extdirs>
          </compilerArguments>
        </configuration>
      </plugin>
    </plugins>
  </build>

繼而我們執行一下複製過來的Main.class,發現Demo是可以跑起來的~
image.png
####2、編寫我們的支付介面
#####1、在portal下新建OrderController類:
image.png
OrderController類上新增這兩行註解:並且新增後面要用到的日誌配置。

@Controller
@RequestMapping("/order/")
public class OrderController {
  private static final Logger logger = LoggerFactory.getLogger(OrderController.class);
 }

*Controller:

    //支付寶介面
    @RequestMapping("pay.do")
    @ResponseBody
    public ServerResponse pay(HttpSession session, Long orderNo, HttpServletRequest request) {
        User user = (User) session.getAttribute(Const.CURRENT_USER);
        if (user == null) {
            return ServerResponse.createByErrorCodeMessage(ResponseCode.NEED_LOGIN.getCode(), ResponseCode.NEED_LOGIN.getDesc());
        }
        //這是一個檔案目錄,使用者存放支付寶中生成的二維碼存放目錄,對應的目錄路徑是tomcat 伺服器目錄
        String path = request.getSession().getServletContext().getRealPath("upload");

        return iOrderService.pay(orderNo, user.getId(), path);
    } 

service包下新建IOrderService介面類
*Service:

public interface IOrderService {
    //支付介面的實現
    ServerResponse pay(Long orderNo, Integer userId, String path);
}

同理在Impl包下新建對應的實現類:
*ServiceImpl:

//支付介面的實現
    public ServerResponse pay(Long orderNo,Integer userId,String path){
        Map<String, String> resultMap= Maps.newHashMap();
        Order order = orderMapper.selectByUserIdAndOrderNo(userId,orderNo);

        if(order == null){
            return ServerResponse.createByErrorMessage("使用者沒有該訂單");
        }

        resultMap.put("orderNo",String.valueOf(order.getOrderNo()));


            // (必填) 商戶網站訂單系統中唯一訂單號,64個字元以內,只能包含字母、數字、下劃線,
            // 需保證商戶系統端不能重複,建議通過資料庫sequence生成,
            String outTradeNo = order.getOrderNo().toString();

            // (必填) 訂單標題,粗略描述使用者的支付目的。如“xxx品牌xxx門店當面付掃碼消費”
            String subject = new StringBuilder().append("happymmall掃碼支付").append(outTradeNo).toString();

            // (必填) 訂單總金額,單位為元,不能超過1億元
            // 如果同時傳入了【打折金額】,【不可打折金額】,【訂單總金額】三者,則必須滿足如下條件:【訂單總金額】=【打折金額】+【不可打折金額】
            String totalAmount = order.getPayment().toString();

            // (可選) 訂單不可打折金額,可以配合商家平臺配置折扣活動,如果酒水不參與打折,則將對應金額填寫至此欄位
            // 如果該值未傳入,但傳入了【訂單總金額】,【打折金額】,則該值預設為【訂單總金額】-【打折金額】
            String undiscountableAmount = "0";

            // 賣家支付寶賬號ID,用於支援一個簽約賬號下支援打款到不同的收款賬號,(打款到sellerId對應的支付寶賬號)
            // 如果該欄位為空,則預設為與支付寶簽約的商戶的PID,也就是appid對應的PID
            String sellerId = "";

            // 訂單描述,可以對交易或商品進行一個詳細地描述,比如填寫"購買商品2件共15.00元"
            String body = new StringBuilder().append("訂單").append(outTradeNo).append("購買商品共").append(totalAmount).append("元").toString();

            // 商戶操作員編號,新增此引數可以為商戶操作員做銷售統計
            String operatorId = "test_operator_id";

            // (必填) 商戶門店編號,通過門店號和商家後臺可以配置精準到門店的折扣資訊,詳詢支付寶技術支援
            String storeId = "test_store_id";

            // 業務擴充套件引數,目前可新增由支付寶分配的系統商編號(通過setSysServiceProviderId方法),詳情請諮詢支付寶技術支援
            ExtendParams extendParams = new ExtendParams();
            extendParams.setSysServiceProviderId("2088100200300400500");

            // 支付超時,定義為120分鐘
            String timeoutExpress = "120m";

            // 商品明細列表,需填寫購買商品詳細資訊,
            List<GoodsDetail> goodsDetailList = new ArrayList<GoodsDetail>();

            List<OrderItem> orderItemList= orderItemMapper.getByOrderNoUserId(orderNo,userId);
            for (OrderItem orderItem : orderItemList){
                // 建立一個商品資訊,引數含義分別為商品id(使用國標)、名稱、單價(單位為分)、數量,如果需要新增商品類別,詳見GoodsDetail
                GoodsDetail goods = GoodsDetail.newInstance(orderItem.getProductId().toString(), orderItem.getProductName(),
                        BigDecimalUtil.mul(orderItem.getCurrentUnitPrice().doubleValue(),
                        new Double(100).doubleValue()).longValue(), orderItem.getQuantity());
                // 建立好一個商品後新增至商品明細列表
                goodsDetailList.add(goods);
            }



//            // 建立一個商品資訊,引數含義分別為商品id(使用國標)、名稱、單價(單位為分)、數量,如果需要新增商品類別,詳見GoodsDetail
//            GoodsDetail goods1 = GoodsDetail.newInstance("goods_id001", "xxx小麵包", 1000, 1);
//            // 建立好一個商品後新增至商品明細列表
//            goodsDetailList.add(goods1);


            // 繼續建立並新增第一條商品資訊,使用者購買的產品為“黑人牙刷”,單價為5.00元,購買了兩件
//            GoodsDetail goods2 = GoodsDetail.newInstance("goods_id002", "xxx牙刷", 500, 2);
//            goodsDetailList.add(goods2);

            // 建立掃碼支付請求builder,設定請求引數
            AlipayTradePrecreateRequestBuilder builder = new AlipayTradePrecreateRequestBuilder()
                    .setSubject(subject).setTotalAmount(totalAmount).setOutTradeNo(outTradeNo)
                    .setUndiscountableAmount(undiscountableAmount).setSellerId(sellerId).setBody(body)
                    .setOperatorId(operatorId).setStoreId(storeId).setExtendParams(extendParams)
                    .setTimeoutExpress(timeoutExpress)
                    .setNotifyUrl(PropertiesUtil.getProperty("alipay.callback.url"))//支付寶伺服器主動通知商戶伺服器裡指定的頁面http路徑,根據需要設定
                    //.setNotifyUrl("http://kxkc7h.natappfree.cc/order/alipay_callback.do")
                    .setGoodsDetailList(goodsDetailList);


        /** 一定要在建立AlipayTradeService之前呼叫Configs.init()設定預設引數
         *  Configs會讀取classpath下的zfbinfo.properties檔案配置資訊,如果找不到該檔案則確認該檔案是否在classpath目錄
         */
        Configs.init("zfbinfo.properties");

        /** 使用Configs提供的預設引數
         *  AlipayTradeService可以使用單例或者為靜態成員物件,不需要反覆new
         */
        AlipayTradeService tradeService = new AlipayTradeServiceImpl.ClientBuilder().build();

        AlipayF2FPrecreateResult result = tradeService.tradePrecreate(builder);

        switch (result.getTradeStatus()) {
                case SUCCESS:
                    logger.info("支付寶預下單成功: )");

                    AlipayTradePrecreateResponse response = result.getResponse();
                    dumpResponse(response);

                    //檢查是否存在上傳資料夾,如果不存在就賦予建立資料夾許可權
                    File folder = new File(path);
                    if(!folder.exists()){
                        folder.setWritable(true);
                        folder.mkdirs();
                    }

                    // 需要修改為執行機器上的路徑  C:\  /Users/sudo/Desktop
                    String qrpath = String.format(path+"/qr-%s.png",response.getOutTradeNo());

                    String qrFileName = String.format("/qr-%s.png",response.getOutTradeNo());

                    ZxingUtils.getQRCodeImge(response.getQrCode(),256,qrpath);

                    File targetFile = new File(path,qrFileName);

                    try {
                        //將生成的二維碼上傳到FTP伺服器上
                        FTPUtil.uploadFile(Lists.newArrayList(targetFile));
                    } catch (IOException e) {
                       logger.info("上傳二維碼異常",e);
                    }


                    logger.info("qrpath:" + qrpath);
                    String qrUrl = PropertiesUtil.getProperty("ftp.server.http.prefix")+targetFile.getName();
                    resultMap.put("qrUrl",qrUrl);
                    return ServerResponse.createBySuccess(resultMap);

                case FAILED:
                    logger.error("支付寶預下單失敗!!!");
                    return ServerResponse.createByErrorMessage("支付寶預下單失敗!!!");

                case UNKNOWN:
                    logger.error("系統異常,預下單狀態未知!!!");
                    return ServerResponse.createByErrorMessage("系統異常,預下單狀態未知!!!");

                default:
                    logger.error("不支援的交易狀態,交易返回異常!!!");
                    return ServerResponse.createByErrorMessage("不支援的交易狀態,交易返回異常!!!");
            }

    }
    // 簡單列印應答
    private void dumpResponse(AlipayResponse response) {
        if (response != null) {
            logger.info(String.format("code:%s, msg:%s", response.getCode(), response.getMsg()));
            if (StringUtils.isNotEmpty(response.getSubCode())) {
                logger.info(String.format("subCode:%s, subMsg:%s", response.getSubCode(),
                        response.getSubMsg()));
            }
            logger.info("body:" + response.getBody());
        }
    }

我們會發現,上面的大多數是直接從支付寶Demo中的Main類中複製過來的,
我們只是更改了對應的引數
上面第四行程式碼我們用到了自己封裝了一個根據訂單號和使用者Id來查詢訂單的方法:
*Mapper

//根據訂單號和使用者的Id來查詢訂單
    Order selectByUserIdAndOrderNo(@Param("userId") Integer userId,@Param("orderNo") Long orderNo);

*Mapper.xml

<select id="selectByUserIdAndOrderNo" parameterType="map" resultMap="BaseResultMap">
    select
    <include refid="Base_Column_List"/>
    from mmall_order
    where order_no = #{orderNo}
    and user_id = #{userId}
  </select>

####2、支付介面測試:
image.png
用沙箱版支付寶掃描支付我們生成的二維碼,發現可以啦~
image.png