/AlipayF2F

支付宝 nodejs api sdk

Primary LanguageJavaScript

AlipayF2F-NodeJS

支付宝当面付NodeJS API.

如何使用

安装方式:

npm install alipay-ftof

express-example为express的example项目, 您可以将config.js.tpl复制为config.js然后修改好您的当面付配置信息后直接使用.

5分钟快速了解

目前已实现的功能

  • createQRPay: 预创建二维码支付宝订单, 当扫码后才是真创建订单.
  • verifyCallback: 支付宝回调验签.
  • checkInvoiceStatus: 使用商户订单号查询订单状况.
  • checkInvoiceStatusWithAlipayTradeNo: 使用使用支付宝订单号查询订单状况.
  • refund: 使用商户订单号请求退款操作.(guanbo)
  • refundWithAlipayTradeNo: 使用支付宝订单号请求退款操作.(guanbo)
  • createBarCodePay: 使用支付宝用户付款码付款.(CloudnuY)

屁话多!如何使用?

您先需要准备一个Object对象内部存放alipay的配置如下:

// config.js

module.exports = {

	/* 以下信息可以在https://openhome.alipay.com/platform/appManage.htm查到, 不过merchantPrivateKey需要您自己生成 */

	/* 应用AppID */
	"appid": 0,

	/* 通知URL 接受支付宝异步通知需要用到  */
	"notifyUrl": "",

	/* 公钥 和 私钥 的填写方式 */
	"testPrivateKey": "-----BEGIN RSA PRIVATE KEY-----\n" +
		          "公钥或私钥内容..." +
		          "\n-----END RSA PRIVATE KEY-----",

	/* 应用RSA私钥 请勿忘记 -----BEGIN RSA PRIVATE KEY----- 与 -----END RSA PRIVATE KEY-----  */
	"merchantPrivateKey": "",

	/* 支付宝公钥 如果为注释掉会使用沙盒公钥 请勿忘记 -----BEGIN PUBLIC KEY----- 与 -----END PUBLIC KEY----- */
	"alipayPublicKey": "",
	
	/* 支付宝支付网关 如果为注释掉会使用沙盒网关 */
	"gatewayUrl": "",
};

注意!您需要生成RSA2公钥与私钥来配合本项目使用.

预创建订单

new一个alipayf2f对象并将刚刚的config.js传入

var alipay_f2f = new alipayf2f(require("./config.js"));

然后就能使用createQRPay肛出一个二维码来让用户扫了

alipay_f2f.createQRPay({
    tradeNo: "123",      // 必填 商户订单主键, 就是你要生成的
    subject: "女装",      // 必填 商品概要
    totalAmount: 0.5,    // 必填 多少钱
    body: "黑丝吊带小蜡烛", // 可选 订单描述, 可以对交易或商品进行一个详细地描述,比如填写"购买商品2件共15.00元"
    timeExpress: 5       // 可选 支付超时, 默认为5分钟
}).then(result => {
    console.log(result) // 支付宝返回的结果
}).catch(error => console.error(error));

如果一切都是理想情况, 支付宝应该会返回code10000的一段这样JSON:

{
    "code":"10000",
    "msg":"Success",
    "out_trade_no":"123",
    "qr_code":"https://qr.alipay.com/bax09352lmfssonc0gzr0028"
}

这时您就可以直接把qr_code中的字符串进行二维码化然后直接给用户扫就可以了!

用户扫码并支付

当用户付款后支付宝会发一个post请求到您设置的notifyUrl.

你要想确认这个回调请求是不是支付宝的该如何判断? 不要慌有个方法叫verifyCallback就能帮您鉴别是不是支付宝发送的了!

假设我们现在使用的是express, 并且notifyUrlhttp://example.com/callback

router.post("/callback", (req, res) => {
	/* 请勿改动支付宝回调过来的post参数, 否则会导致验签失败 */
	var signStatus = alipay_f2f.verifyCallback(req.body);
	if(signStatus === false) {
		return res.error("回调签名验证未通过");
	}

	/* 商户订单号 */
	var noInvoice = req.body["out_trade_no"];
	/* 订单状态 */
	var invoiceStatus = req.body["trade_status"];

	// 支付宝回调通知有多种状态您可以点击已下链接查看支付宝全部通知状态
	// https://doc.open.alipay.com/docs/doc.htm?spm=a219a.7386797.0.0.aZMdK2&treeId=193&articleId=103296&docType=1#s1
	if(invoiceStatus !== "TRADE_SUCCESS") {
		return res.send("success");
	}

	/* 一切都验证好后就能更新数据库里数据说用户已经付钱啦 */
	req.database.update(noInvoice, { pay: true }).then(result => res.send("success")).catch(err => res.catch(err));
});

需要注意的是当处理完支付宝回调后应当返回success来告诉支付宝我已经搞定了, 否则支付宝会重复通知防止掉单.

服务器网络暴毙? 没收到支付宝回调?

不要慌我给你实现了一个checkInvoiceStatus方法.

这个方法呢可以给你手动查询订单状态. 使用起来也很方便, 只需要传入一个商户订单号就行了:

alipay_f2f.checkInvoiceStatus("2333333").then(result => {
  console.log(result);
}).catch(error => { });

输出

{
    "code":"10000",
    "msg":"Success",
    "buyer_logon_id":"494***@qq.com",
    "buyer_pay_amount":"0.01",
    "buyer_user_id":"0000702210000000",
    "fund_bill_list":[
        {
            "amount":"0.01",
            "fund_channel":"ALIPAYACCOUNT"
        }
    ],
    "invoice_amount":"0.01",
    "open_id":"00001023939817879028820892810000",
    "out_trade_no":"alipayf2f_1481800000000",
    "point_amount":"0.00",
    "receipt_amount":"0.01",
    "send_pay_date":"2016-12-16 00:00:00",
    "total_amount":"0.01",
    "trade_no":"2016121621001000000000000000",
    "trade_status":"TRADE_SUCCESS"
}

使用用户付款码付款

var payment = {
    // 商户订单号
    tradeNo: Date.now().toString() + Math.floor(Math.random() * 9999) + 1,
    // 订单标题
    subject: "女装",
    // 订单总金额
    totalAmount: 100,
    // 支付授权码,25~30开头的长度为16~24位的数字,实际字符串长度以开发者获取的付款码长度为准
    authCode: ""
}

响应

{ 
    code: '10000',
    msg: 'Success',
    buyer_logon_id: 'yqu***@sandbox.com',
    buyer_pay_amount: '100.00',
    buyer_user_id: '2088102168717885',
    buyer_user_type: 'PRIVATE',
    fund_bill_list: [ { amount: '100.00', fund_channel: 'ALIPAYACCOUNT' } ],
    gmt_payment: '2017-11-25 14:32:10',
    invoice_amount: '100.00',
    open_id: '20880033278643371074141272412388',
    out_trade_no: '151159149052644861',
    point_amount: '0.00',
    receipt_amount: '100.00',
    total_amount: '100.00',
    trade_no: '2017112521001004880200394043' 
}

使用商户订单号请求退款

var refund = {
  /* 退款编号 可选 用于分批退款 */
  refundNo: Date.now(),
  /* 退款金额 如果refundNo为空 refundAmount必须为订单全款 */
  refundAmount: payment.totalAmount
}
app.alipay_f2f.refund("123456", refund).then(result => {
  result.should.have.property('code', '10000');
});

响应

{
  code: '10000',
  msg: 'Success',
  buyer_logon_id: 'hya***@sandbox.com',
  buyer_user_id: '2088102170322284',
  fund_change: 'Y',
  gmt_refund_pay: '2017-02-06 17:46:34',
  open_id: '20881013820906275677621172812028',
  out_trade_no: '1486372683551',
  refund_detail_item_list: [{
    amount: '100.00',
    fund_channel: 'ALIPAYACCOUNT'
  }],
  refund_fee: '100.00',
  send_back_fee: '100.00',
  trade_no: '2017020621001004280200150698'
}

测试

make test

测试需要您设置完毕express-example/config.js文件. 测试过程中需要人工介入来 生成二维码 .

直接看看效果?

点这里 然后我也不介意您给我5毛的(: