HONGKONG-PAYMENT-QFPAY

QFPay API is a comprehensive payment solution that offers various payment methods to meet the needs of different businesses. This skill provides complete API integration guidelines including environment configuration, request formats, signature generation, payment types, supported currencies, and status codes.

MIT-0 · Free to use, modify, and redistribute. No attribution required.
0 · 835 · 0 current installs · 0 all-time installs
MIT-0
Security Scan
VirusTotalVirusTotal
Benign
View report →
OpenClawOpenClaw
Suspicious
medium confidence
Purpose & Capability
The name and description claim a QFPay API integration guide and the SKILL.md indeed contains request formats, signature rules, endpoints, and env var names. That content is consistent with a payment-integration documentation skill. However the skill metadata lists no required environment variables or a primary credential while the SKILL.md explicitly instructs users to export QFPAY_APPCODE and QFPAY_KEY (and optionally QFPAY_MCHID). This metadata/instruction mismatch is suspicious (likely an omission) and reduces transparency.
Instruction Scope
The SKILL.md stays within the expected scope of a payment API guide: environment endpoints, request/response formats, headers, signature generation, rate limits, and notify_url usage. It does not instruct the agent to read unrelated system files or arbitrary credentials beyond the documented QFPay vars. No broad or vague 'gather whatever context you need' instructions were found.
Install Mechanism
This is an instruction-only skill with no install spec and no code files. That minimizes install-time risk because nothing is downloaded or written to disk by the skill package itself.
!
Credentials
The SKILL.md directs users to set sensitive credentials (QFPAY_APPCODE, QFPAY_KEY, optionally QFPAY_MCHID and QFPAY_ENV). Those are reasonable and proportional for a payment integration, but the skill's declared requirements list zero environment variables and no primary credential. The inconsistency reduces transparency about what secrets the skill expects and how they will be used. Users should treat these env vars as secrets and confirm where/when they're submitted.
Persistence & Privilege
The skill does not request elevated presence: always:false, no install steps, and it does not claim to modify other skills or system-wide config. Autonomous invocation is allowed (platform default) but is not combined with other privilege red flags.
What to consider before installing
This skill looks like a documentation-only guide for QFPay (sandbox/test/prod endpoints, headers, signature rules), which is plausible. However: 1) SKILL.md asks you to export sensitive variables (QFPAY_APPCODE, QFPAY_KEY, QFPAY_MCHID) but the skill metadata declares none — treat that as an omission and a transparency concern. 2) Do not paste production keys into any third-party skill or into conversational inputs unless you trust the skill source. Prefer using sandbox credentials while testing. 3) Verify the skill's origin (homepage/source repository) before using in production; the package lists no homepage. 4) If you will have the agent perform real requests, configure secrets in a secure secret store (not in chat), validate signature generation locally or from trusted SDKs, and confirm notify_url endpoints are secured (validate incoming notifications). 5) If anything in SKILL.md is unclear (truncated snippets, missing code examples claimed in README), ask the publisher for full, canonical docs or use the official QFPay docs at sdk.qfapi.com before providing credentials.

Like a lobster shell, security has layers — review code before you run it.

Current versionv1.0.0
Download zip
latestvk978ve0n3w66415pjqrkrhmp7580zfty

License

MIT-0
Free to use, modify, and redistribute. No attribution required.

SKILL.md

QFPay Payment API Skill

Overview

QFPay API is a comprehensive payment solution that offers various payment methods to meet the needs of different businesses. This skill provides complete API integration guidelines including environment configuration, request formats, signature generation, payment types, supported currencies, and status codes.

Environment Configuration

QFPay API is accessible via three main environments:

EnvironmentBase URLDescription
Sandboxhttps://openapi-int.qfapi.comFor simulating payments without real fund deduction
Testinghttps://test-openapi-hk.qfapi.comReal payment flow but linked to test accounts (no settlement)
Productionhttps://openapi-hk.qfapi.comReal live payments with actual settlement

Important Notes:

  • Transactions in Testing environment using test accounts will NOT be settled
  • Always ensure refunds are triggered immediately for test transactions
  • Mixing credentials or endpoints across environments will result in signature or authorization errors

Environment Variables Setup

Configure these environment variables before using the API:

export QFPAY_APPCODE="your_app_code_here"
export QFPAY_KEY="your_client_key_here"
export QFPAY_MCHID="your_merchant_id"  # Optional, depends on account setup
export QFPAY_ENV="sandbox"  # Options: prod, test, sandbox

API Usage Guidelines

Rate Limiting

To ensure fair usage and optimal performance:

  • Limit: Maximum 100 requests per second (RPS) and 400 requests per minute per merchant
  • Exceeding Limit: API responds with HTTP 429 (Too Many Requests)

Best Practices

  1. Batch Requests: Use batch processing to minimize individual requests
  2. Efficient Queries: Utilize filtering and pagination
  3. Caching: Implement response caching to avoid repeated requests
  4. Monitoring: Track API usage and implement logging for request patterns

Error Handling

When receiving HTTP 429:

  1. Pause further requests for a specified duration
  2. Implement exponential backoff for retries
  3. Log errors for monitoring

Traffic Spike Management

For anticipated traffic spikes (e.g., promotional events), contact:

Request Format

HTTP Request

POST /trade/v1/payment

Public Payment Request Parameters

AttributeRequiredTypeDescription
txamtYesInt(11)Transaction amount in cents (100 = $1). Suggest > 200 to avoid risk control
txcurrcdYesString(3)Transaction currency. See Currencies section for full list
pay_typeYesString(6)Payment type code. See Payment Types section
out_trade_noYesString(128)External transaction number. Must be unique per merchant account
txdtmYesString(20)Transaction time format: YYYY-MM-DD hh:mm:ss
auth_codeYes (CPM only)String(128)Authorization code for scanning barcode/QR. Expires within one day
expired_timeNo (MPM only)String(3)QR expiration in minutes. Default: 30, Min: 5, Max: 120
goods_nameNoString(64)Item description. Max 20 chars, UTF-8 for Chinese. Required for App payments
mchidNoString(16)Merchant ID. Required if assigned, must NOT be included if not assigned
udidNoString(40)Unique device ID for internal tracking
notify_urlNoString(256)URL for asynchronous payment notifications

HTTP Header Requirements

FieldRequiredDescription
X-QF-APPCODEYesApp code assigned to merchant
X-QF-SIGNYesSignature generated per signature algorithm
X-QF-SIGNTYPENoSignature algorithm. Use SHA256 or defaults to MD5

Content Specifications

  • Character Encoding: UTF-8
  • Method: POST/GET (depends on endpoint)
  • Content-Type: application/x-www-form-urlencoded

Response Format

Success Response Structure

{
  "respcd": "0000",
  "respmsg": "success",
  "data": {
    "txamt": "100",
    "out_trade_no": "20231101000001",
    "txcurrcd": "HKD",
    "txstatus": "SUCCESS",
    "qf_trade_no": "9000020231101000001",
    "pay_type": "800101",
    "txdtm": "2023-11-01 10:00:00"
  }
}

Response Fields

FieldTypeDescription
respcdString(4)Return code. "0000" means success
respmsgString(64)Message description of respcd
dataObjectPayment transaction data

Data Object Fields

FieldTypeDescription
txamtStringTransaction amount in cents
out_trade_noStringMerchant's original order number
txcurrcdStringCurrency code (e.g., HKD)
txstatusStringPayment status: SUCCESS, FAILED, PENDING
qf_trade_noStringQFPay's unique transaction number
pay_typeStringPayment method code
txdtmStringPayment time (YYYY-MM-DD HH:mm:ss)

Signature Verification

Response may include X-QF-SIGN and X-QF-SIGNTYPE headers. Verify by:

  1. Extracting data fields in sorted order
  2. Concatenating as key1=value1&key2=value2&...
  3. Appending client key
  4. Generating MD5 hash and comparing

Signature Generation

All API requests must include a digital signature in the HTTP header:

X-QF-SIGN: <your_signature>

Step-by-Step Guide

Step 1: Sort Parameters

Sort all request parameters by parameter name in ASCII ascending order.

Example:

ParameterValue
mchidZaMVg12345
txamt100
txcurrcdHKD

Sorted result:

mchid=ZaMVg12345&txamt=100&txcurrcd=HKD

Step 2: Append Client Key

Append your secret client_key to the string.

If client_key = abcd1234:

mchid=ZaMVg12345&txamt=100&txcurrcd=HKDabcd1234

Step 3: Hash the String

Hash using MD5 or SHA256 (SHA256 recommended):

SHA256("mchid=ZaMVg12345&txamt=100&txcurrcd=HKDabcd1234")

Step 4: Add to Header

X-QF-SIGN: <your_hashed_signature>

Important Notes

  • Do NOT insert line breaks, tabs, or extra spaces
  • Parameter names and values are case-sensitive
  • Double-check parameter order and encoding if signature fails

Code Examples

Python

import os
import hashlib

APPCODE = os.getenv('QFPAY_APPCODE')
KEY = os.getenv('QFPAY_KEY')

def generate_signature(params, key):
    """Generate MD5 signature"""
    keys = list(params.keys())
    keys.sort()
    query = []
    for k in keys:
        if k not in ('sign', 'sign_type') and (params[k] or params[k] == 0):
            query.append(f'{k}={params[k]}')
    
    data = '&'.join(query) + key
    md5 = hashlib.md5()
    md5.update(data.encode('utf-8'))
    return md5.hexdigest().upper()

def generate_signature_sha256(params, key):
    """Generate SHA256 signature"""
    keys = list(params.keys())
    keys.sort()
    query = []
    for k in keys:
        if k not in ('sign', 'sign_type') and (params[k] or params[k] == 0):
            query.append(f'{k}={params[k]}')
    
    data = '&'.join(query) + key
    sha256 = hashlib.sha256()
    sha256.update(data.encode('utf-8'))
    return sha256.hexdigest().upper()

Payment Types

The pay_type parameter specifies which payment method to use. This affects transaction routing and UI requirements.

Note: Not all pay_type values are enabled for every merchant. Contact technical.support@qfpay.com for clarification.

Supported Payment Types

CodeDescription
800008CPM for WeChat, Alipay, UnionPay QuickPass, PayMe
800101Alipay MPM (Overseas Merchants)
800108Alipay CPM (Overseas & HK Merchants)
801101Alipay Web Payment (Overseas)
801107Alipay WAP Payment (Overseas)
801110Alipay In-App Payment (Overseas)
800107Alipay Service Window H5 Payment
801501Alipay MPM (HK Merchants)
801510Alipay In-App (HK Merchants)
801512Alipay WAP (HK Merchants)
801514Alipay Web (HK Merchants)
800201WeChat MPM (Overseas & HK)
800208WeChat CPM (Overseas & HK)
800207WeChat JSAPI Payment (Overseas & HK)
800212WeChat H5 Payment
800210WeChat In-App Payment (Overseas & HK)
800213WeChat Mini-Program Payment
801008WeChat Pay HK CPM (Direct Settlement, HK)
801010WeChat Pay HK In-App (Direct Settlement, HK)
805801PayMe MPM (HK Merchants)
805808PayMe CPM (HK Merchants)
805814PayMe Web Payment (HK Merchants)
805812PayMe WAP Payment (HK Merchants)
800701UnionPay QuickPass MPM
800708UnionPay QuickPass CPM
800712UnionPay WAP Payment (HK)
800714UnionPay PC-Web Payment (HK)
802001FPS MPM (HK Merchants)
803701Octopus MPM (HK Merchants)
803712Octopus WAP Payment (HK)
803714Octopus PC-Web Payment (HK)
802801Visa / Mastercard Online
802808Visa / Mastercard Offline
806527ApplePay Online
806708UnionPay Card Offline
806808American Express Card Offline

Special Remarks

  • 801101: Transaction amount must be greater than 1 HKD
  • 802001: This payment method does not support refunds

Supported Currencies

All codes follow ISO 4217 format (3-letter uppercase):

CodeDescription
AEDArab Emirates Dirham
CNYChinese Yuan
EUREuro
HKDHong Kong Dollar
IDRIndonesian Rupiah
JPYJapanese Yen
MMKMyanmar Kyat
MYRMalaysian Ringgit
SGDSingapore Dollar
THBThai Baht
USDUnited States Dollar
CADCanadian Dollar
AUDAustralian Dollar

Note: Some payment methods may only support HKD. Verify with your integration manager before non-HKD transactions.

Status Codes

Standard respcd values returned by QFPay API:

CodeDescription
0000Transaction successful
1100System under maintenance
1101Reversal error
1102Duplicate request
1103Request format error
1104Request parameter error
1105Device not activated
1106Invalid device
1107Device not allowed
1108Signature error
1125Transaction already refunded
1136Transaction does not exist or not operational
1142Order already closed
1143Order not paid, password being entered
1145Processing, please wait
1147WeChat Pay transaction error
1150T0 billing method does not support cancellation
1155Refund request denied
1181Order expired
1201Insufficient balance
1202Incorrect or expired payment code
1203Merchant account error
1204Bank error
1205Transaction failed, try again later
1212Please use UnionPay overseas payment code
1241Store does not exist or status incorrect
1242Store not configured correctly
1243Store has been disabled
1250Transaction forbidden
1251Store configuration error
1252System error making order request
1254Problem occurred, resolving issue
1260Order already paid
1261Order not paid
1262Order already refunded
1263Order already cancelled
1264Order already closed
1265Transaction cannot be refunded (11:30pm-0:30am)
1266Transaction amount wrong
1267Order information mismatch
1268Order does not exist
1269Insufficient unsettled balance for refund
1270Currency does not support partial refund
1271Transaction does not support partial refund
1272Refund amount exceeds maximum refundable
1294Transaction non-compliant, prohibited by bank
1295Connection slow, waiting for network response
1296Connection slow, try again later
1297Banking system busy
1298Connection slow, do not repeat payment if already paid
2005Customer payment code incorrect or expired
2011Transaction serial number repeats

Usage Examples

Complete Payment Flow (Python)

import os
import requests
import hashlib
import datetime
import time

# Load configuration from environment variables
APPCODE = os.getenv('QFPAY_APPCODE')
KEY = os.getenv('QFPAY_KEY')
MCHID = os.getenv('QFPAY_MCHID')
ENV = os.getenv('QFPAY_ENV', 'test')

# Environment URLs
ENV_URLS = {
    'prod': 'https://openapi-hk.qfapi.com',
    'test': 'https://test-openapi-hk.qfapi.com',
    'sandbox': 'https://openapi-int.qfapi.com'
}

BASE_URL = ENV_URLS.get(ENV, ENV_URLS['test'])

def generate_signature(params, key, sign_type='SHA256'):
    """Generate signature for QFPay API request"""
    keys = list(params.keys())
    keys.sort()
    query = []
    for k in keys:
        if k not in ('sign', 'sign_type') and (params[k] or params[k] == 0):
            query.append(f'{k}={params[k]}')
    
    data = '&'.join(query) + key
    
    if sign_type == 'SHA256':
        sha256 = hashlib.sha256()
        sha256.update(data.encode('utf-8'))
        return sha256.hexdigest().upper()
    else:
        md5 = hashlib.md5()
        md5.update(data.encode('utf-8'))
        return md5.hexdigest().upper()

def create_payment(amount, currency, pay_type, goods_name, notify_url=None):
    """Create a payment request"""
    params = {
        'txamt': str(amount),
        'txcurrcd': currency,
        'pay_type': pay_type,
        'out_trade_no': f"ORDER{int(time.time() * 10000)}",
        'txdtm': datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S'),
        'goods_name': goods_name
    }
    
    if MCHID:
        params['mchid'] = MCHID
    
    if notify_url:
        params['notify_url'] = notify_url
    
    signature = generate_signature(params, KEY)
    
    headers = {
        'X-QF-APPCODE': APPCODE,
        'X-QF-SIGN': signature,
        'Content-Type': 'application/x-www-form-urlencoded'
    }
    
    response = requests.post(
        f'{BASE_URL}/trade/v1/payment',
        data=params,
        headers=headers
    )
    
    return response.json()

def query_transaction(out_trade_no):
    """Query transaction status"""
    params = {
        'out_trade_no': out_trade_no,
        'txdtm': datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
    }
    
    if MCHID:
        params['mchid'] = MCHID
    
    signature = generate_signature(params, KEY)
    
    headers = {
        'X-QF-APPCODE': APPCODE,
        'X-QF-SIGN': signature,
        'Content-Type': 'application/x-www-form-urlencoded'
    }
    
    response = requests.post(
        f'{BASE_URL}/trade/v1/query',
        data=params,
        headers=headers
    )
    
    return response.json()

def refund_transaction(out_trade_no, txamt, qf_trade_no=None):
    """Process a refund"""
    params = {
        'out_trade_no': out_trade_no,
        'txamt': str(txamt),
        'txdtm': datetime.datetime.now().strftime('%Y-%m-%d %H:%M:%S')
    }
    
    if qf_trade_no:
        params['qf_trade_no'] = qf_trade_no
    
    if MCHID:
        params['mchid'] = MCHID
    
    signature = generate_signature(params, KEY)
    
    headers = {
        'X-QF-APPCODE': APPCODE,
        'X-QF-SIGN': signature,
        'Content-Type': 'application/x-www-form-urlencoded'
    }
    
    response = requests.post(
        f'{BASE_URL}/trade/v1/refund',
        data=params,
        headers=headers
    )
    
    return response.json()

# Example usage
if __name__ == '__main__':
    # Create a payment
    result = create_payment(
        amount=100,  # $1.00 HKD
        currency='HKD',
        pay_type='800101',
        goods_name='Test Product'
    )
    print(f"Payment created: {result}")
    
    if result.get('respcd') == '0000':
        out_trade_no = result['data']['out_trade_no']
        
        # Query transaction
        query_result = query_transaction(out_trade_no)
        print(f"Transaction status: {query_result}")

Environment Configuration with Multiple Merchants

import os

# Configuration loader from environment
class QFPayConfig:
    def __init__(self, env='sandbox'):
        self.env = env
        self.appcode = os.getenv(f'QFPAY_{env.upper()}_APPCODE') or os.getenv('QFPAY_APPCODE')
        self.key = os.getenv(f'QFPAY_{env.upper()}_KEY') or os.getenv('QFPAY_KEY')
        self.mchid = os.getenv(f'QFPAY_{env.upper()}_MCHID') or os.getenv('QFPAY_MCHID')
        
    @property
    def base_url(self):
        urls = {
            'prod': 'https://openapi-hk.qfapi.com',
            'test': 'https://test-openapi-hk.qfapi.com',
            'sandbox': 'https://openapi-int.qfapi.com'
        }
        return urls.get(self.env, urls['sandbox'])

# Usage
config = QFPayConfig(env=os.getenv('QFPAY_ENV', 'sandbox'))
print(f"Using base URL: {config.base_url}")

Important Notes

  1. Signature Security: Never expose your client_key in frontend code or client-side applications. Always compute signatures on the server side.

  2. Order Number Uniqueness: out_trade_no must be unique across all payment and refund requests under the same merchant account.

  3. Character Encoding: All requests and responses use UTF-8 encoding.

  4. Timeout Handling: For payment requests that don't return promptly, implement a polling mechanism to query transaction status.

  5. Async Notifications: Configure notify_url to receive asynchronous payment completion notifications and verify notification signatures.

  6. Refund Limitations: FPS (802001) payment type does not support refunds. Confirm business requirements before integration.

  7. Amount Format: Amount is in cents. For example, 100 represents 1 HKD.

  8. Timezone: The txdtm parameter uses the merchant's local timezone.

  9. Environment Variables: Always load sensitive credentials from environment variables, never hardcode them in source files.

Technical Support

For any integration issues, contact:

See Also

Files

2 total
Select a file
Select a file to preview.

Comments

Loading comments…