跳到主要内容

直连-本地支付

网关地址

POST

https://sandbox.j-pay.net/pay_index

测试账号

Merchant ID:10172

ApiKey: prx5d81g2ytb1mmokl9mohjsmsb09do3

请求参数

参数名称类型是否必填参与签名参数说明
pay_memberidString平台分配商户号
pay_orderidString上送订单号唯一, 字符长度 20
pay_applydateString提交时间, 时间格式:2016-12-26 18:18:18
pay_bankcodeString银行编码, 参考后续说明:901
pay_notifyurlString服务端返回地址.(POST 返回数据)
pay_callbackurlString页面跳转返回地址(POST 返回数据)
pay_amountString商品金额
pay_md5signString请看 MD5 签名字段格式
pay_currencyString币种代码,大写字母 (USD)
pay_urlString网站URL, 需提交审核、带http(s)://
pay_typeString本地支付固定值:apm
pay_methodString支付方式列表,不传则跳转收银台 如 alipay
pay_app_schemeStringAPP url scheme, 不为空时优先同步返回, 如: jpayapm://payment/result
+pay_productnameString商品信息,参数由下列一组数据,通过JSON格式组成,注意参数名大小写敏感。本字段以JSON数组字符串来填值。示例:[{"sku":"123456789","productName":"MacBook Pro","productImage":"产品图片地址","attributes":"Size:US8 Color:blue","price":"11000.00","quantity":"1"},{"sku":"9876543231","productName":"IPhone 12","productImage":"产品图片地址","attributes":"Size:US8 Color:blue","price":"11000.00","quantity":"1"}]
「skuString商品编号 (唯一)
「productNameString产品名称
「productImageString产品图片URL
「attributesString产品属性
「priceString产品价格
「quantityString产品数量
pay_firstnameString账单名字
pay_lastnameString账单姓氏
pay_street_address1String账单地址1
pay_street_address2String账单地址2
pay_cityString账单城市
pay_postcodeString账单邮编
pay_stateString账单州/省
pay_country_iso_code_2String账单国家
pay_email_addressString账单邮箱
pay_telephoneString账单电话
shipping_firstnameString收件名字
shipping_lastnameString收件姓氏
shipping_street_address1String收件地址1
shipping_street_address2String收件地址2
shipping_cityString收件城市
shipping_stateString收件州/省
shipping_postcodeString收件邮编
shipping_country_iso_code_2String收件国家
shipping_telephoneString收件电话
pay_ipString用户IP地址
pay_useragentString用户浏览器信息
pay_languageString语言
systemString网站系统, 如:Shopyy/Shopline

请求支付返回参数

参数名称类型参数说明
statusIntstatus = 0:SUCCESS
status = 1:3d支付跳转
status = 2:failed
msgString下单成功 or 下单失败
urlString3d支付跳转地址

返回示例

Transaction successful : {
"status": 0,
"msg": "transaction success"
}
Transaction failed : {
"status": 2,
"msg": "transaction failed"
}
Redirect/3ds payment successful : {
"status": 1,
"url": "redirectUrl"
}

3DS验证

跳转支付和直接信用卡支付需要3DS验证

同步返回:(application/x-www-form-urlencoded)

参数名称类型是否必填参数说明
paymentStatusStringSucceeded: 成功
Fail:失败
Processing:处理中
orderIDString订单号

示例

同步返回:

https://www.xxxx.com/callback.html?paymentStatus=succeeded&orderID=1001

异步返回:(application/x-www-form-urlencoded)

参数名称类型是否必填参数说明
memberidString商户 ID
orderidString商户订单号
amountString提交的订单金额
true_amountString买家实际付款的金额
currencyString币种
transaction_idString平台订单号
returncodeString“00” 为成功
"2":为失败
datetimeString交易时间,格式:20220419000114
signString请看验证签名字段格式
attachString
Array{
["memberid"]=>
string(5) "10012"
["orderid"]=>
string(7) "7758292"
["transaction_id"]=>
string(20) "20230117104641499957"
["amount"]=>
string(5) "85.95"
["true_amount"]=>
string(5) "85.95"
["currency"]=>
string(5) "USD"
["datetime"]=>
string(14) "20230117111809"
["returncode"]=>
string(2) "00"
["sign"]=>
string(32) "48CF5D99A2ADE25DCEF73BC1B1136B9E"
["attach"]=>
string(0) ""
}

Demo

import { md5 } from "js-md5";
import axios from "axios";

const KEY = "prx5d81g2ytb1mmokl9mohjsmsb09do3";
const MEMBER_ID = "10172";
let query = {
pay_memberid: MEMBER_ID,
pay_orderid: "O" + Date.now().valueOf(),
pay_applydate: "2024-12-02 16:48:42",
pay_bankcode: "901",
pay_notifyurl: "https://www.google.com",
pay_callbackurl: "https://www.google.com",
pay_amount: "10.00",
};
let signData = [];
Object.keys(query)
.sort()
.forEach((key) => signData.push(`${key}=${query[key]}`));

signData.push(`key=${KEY}`);
Object.assign(query, {
pay_md5sign: md5(signData.join("&")).toUpperCase(),
pay_url: "https://www.google.com",
pay_currency: "USD",
pay_firstname: "Rodrigo",
pay_lastname: "lopez",
pay_country_iso_code_2: "CL",
pay_email_address: "[email protected]",
pay_telephone: "920508435",
pay_ip: "186.156.221.158",
pay_useragent: "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/111.0.0.0 Safari/537.36 Edg/111.0.1661.43",
pay_language: "en",
system: "shopyy",
});

axios
.request({
url: "https://sandbox.j-pay.net/pay_index",
method: "post",
headers: {
"Content-Type": "application/x-www-form-urlencoded",
},
data: query,
})
.then(({ data }) => {
console.log("success", data);
})
.catch((error) => {
console.log("error", error);
});

App Demo

接口新增参数示例:

pay_app_scheme: jpayapm://payment/result

Demo下载:

Kotlin demo

添加(或更新)构建依赖项

dependencies {
implementation "androidx.browser:browser:1.8.0"
}

JPayAPM.kt

package com.jpay.apm

import android.app.Activity
import android.content.Context
import android.net.Uri
import android.os.Build
import android.os.Bundle
import android.os.Handler
import android.os.Looper
import android.util.Log
import androidx.browser.customtabs.*
import org.json.JSONObject
import java.lang.ref.WeakReference

/**
* JPay APM
*/
class JPayAPM {

companion object {
const val PAY_STATUS_UNPAID = 0
const val PAY_STATUS_SUCCESS = 1
const val PAY_STATUS_FAILED = 2
const val PAY_STATUS_CANCEL = 3
private const val TAG = "JPayAPM"
}

private var mActivityRef: WeakReference<Activity>? = null
private var mContextRef: WeakReference<Context>? = null
private var payStatus: Int = PAY_STATUS_UNPAID
private var payCallback: WeakReference<((JSONObject) -> Unit)?> = WeakReference(null)

constructor(activity: Activity, context: Context) {
mActivityRef = WeakReference(activity)
mContextRef = WeakReference(context)
}

/**
* Execute payment
* @param payMethod Payment method (only support googlepay)
* @param payUrl Payment URL
* @param callback Result callback
*/
fun exec(payMethod: String, payUrl: String, callback: ((JSONObject) -> Unit)?) {
Log.d(TAG, "Execute payment: method=$payMethod, url=$payUrl")
payStatus = PAY_STATUS_UNPAID
payCallback = WeakReference(callback)

payByUrl(payUrl)
}

/**
* Handle payment result (main thread callback)
*/
private fun handlePayResult(status: Int, message: String, orderId: String = "") {
payStatus = status
Log.d(TAG, "Payment result: status=$status, msg=$message, orderId=$orderId")

// Build JSON result
val result = JSONObject().apply {
put("status", status)
put("message", message)
put("orderId", orderId)
}

Handler(Looper.getMainLooper()).post {
payCallback.get()?.invoke(result)
payCallback.clear()
}
}

/**
* Open payment URL
*/
private fun payByUrl(payUrl: String) {
val activity = mActivityRef?.get()
val context = mContextRef?.get()
if (context == null) {
handlePayResult(PAY_STATUS_FAILED, "Context is null", "")
return
}
if (activity == null || activity.isFinishing ||
(Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1 && activity.isDestroyed)) {
handlePayResult(PAY_STATUS_FAILED, "Activity is null or destroyed", "")
return
}

val uri: Uri
try {
uri = Uri.parse(payUrl)
if (uri.scheme.isNullOrEmpty() || uri.host.isNullOrEmpty()) {
handlePayResult(PAY_STATUS_FAILED, "Invalid URL format: $payUrl", "")
return
}
} catch (e: Exception) {
handlePayResult(PAY_STATUS_FAILED, "Failed to parse URL: ${e.message}", "")
return
}

try {
val intent = CustomTabsIntent.Builder()
.setShowTitle(true)
.build()

intent.launchUrl(activity, uri)
} catch (e: Exception) {
Log.e(TAG, "Failed to launch payment URL: $payUrl", e)
handlePayResult(PAY_STATUS_FAILED, "Failed to launch payment URL: ${e.message}", "")
}
}

/**
* Trigger payment result (for Scheme callback)
*/
fun triggerPayResult(status: Int, message: String, orderId: String = "") {
handlePayResult(status, message, orderId)
}

/**
* Release resources
*/
fun release() {
mActivityRef?.clear()
mContextRef?.clear()
payCallback.clear()
payStatus = PAY_STATUS_UNPAID
}
}