一、财务配置
登录到苹果哦itunsconnect后台后,可以到协议、税务和银行业务
这里交给财务配置就行
二、itunes后台商品配置
进入到我们的iTunes后台后,在App Store
下面 有一个子栏目App 内购买项目
这里点击管理
我们就能看到我们进行商品配置的入口了
我们可以添加的商品类型
- Consumable 消耗品: 可以多次购买,适合游戏内货币
- Non-Consumable 非消耗品: 购买一次,永久有效,适合解锁永久功能;
- Non-Renewing Subscription 非续订订阅: 在固定时间段内可用的内容,如vip
- Auto-Renewing Subscription 自动续费订阅:到期会自动扣款的订阅,适用按月的vip;
详细内容请查看:https://developer.apple.com/support/app-store-connect/#//apple_ref/doc/uid/TP40013727-CH3-SW1
三、客户端集成(integration)
1、支付流程
2、iOS- swift代码
import StoreKit
class InAppPurchaseManager: NSObject,SKPaymentTransactionObserver, SKProductsRequestDelegate {
//登录的时候调用
func addIAPObserver() -> Void {
SKPaymentQueue.default().remove(self)
SKPaymentQueue.default().add(self)
}
//登出的时候调用, 防止发货到未知用户
func removeIAPObserver() -> Void {
SKPaymentQueue.default().remove(self)
}
func addPayment(_ productId:String) {
let product = NSSet(array: [productId] as [AnyObject])
let request = SKProductsRequest(productIdentifiers: product as! Set)
request.delegate = self
request.start()
}
func productsRequest(_ request: SKProductsRequest, didReceive response: SKProductsResponse) {
let avaliableProducts = response.products
guard let product = avaliableProducts.first else {
return
}
let mutablePayment = SKMutablePayment.init(product: product)
//发起购买请求
SKPaymentQueue.default().add(mutablePayment)
}
func paymentQueue(_ queue: SKPaymentQueue, updatedTransactions transactions: [SKPaymentTransaction]) {
for transaction in transactions {
switch transaction.transactionState {
case .purchased:
self.complete(transaction)
break
case .restored:
self.restored(transaction)
break
case .failed:
self.fail(transaction)
break
default:
//其他情况
break
}
}
}
//交易完成
func complete(_ transaction: SKPaymentTransaction) -> Void {
guard let receiptUrl = Bundle.main.appStoreReceiptURL, let receiveData:NSData = NSData(contentsOf: receiptUrl) else {
return
}
let receiptString = receiveData.base64EncodedString(options: .endLineWithLineFeed)
let transactionIdentifier = transaction.transactionIdentifier
//将transactionIdentifier & receiptString发送给服务器,去验证票据的正确性
//todo
}
//交易失败
func fail(_ transaction: SKPaymentTransaction) {
//todo
}
//交易restored
func restored(_ transaction:SKPaymentTransaction) {
//todo
}
//交易完成记得finish掉,不然会出现卡单,无法进行下次支付
func finishTransactionWith(transactionId transctionId:String) {
let transactions = SKPaymentQueue.default().transactions
for transaction in transactions {
if transaction.transactionIdentifier == transctionId {
SKPaymentQueue.default().finishTransaction(transaction)
break
}
}
}
}
四、简单的服务器验票逻辑(nodejs)
整个服务逻辑牵扯太多,我用nodejs整理出单纯的验票部分,简单的校验和借鉴是没有问题的。
const https = require("https")
var postData = "客户端传过来receiptString";
var IAPVerifier = function(){};
IAPVerifier.verifyWithRetry = function(receipt, isBase64, cb) {
var encoded = null, receiptData = {};
if (isBase64) {
encoded = receipt;
} else {
encoded = new Buffer(receipt).toString('base64');
}
receiptData['receipt-data'] = encoded;
var options = this.requestOptions();
return this.verify(receiptData, options, (function(_this) {
return function(error, data) {
if (error) return cb(error);
if ((21007 === (data != null ? data.status : void 0)) && (_this.productionHost == _this.host)) {
// 指向沙盒测试环境再次验证
options.host = 'sandbox.itunes.apple.com';
return _this.verify(receiptData, options, function(err, data) {
return cb(err, data);
});
} else {
return cb(error, data);
}
};
})(this));
};
/*
verify the receipt data
*/
IAPVerifier.verify = function(data, options, cb) {
var post_data = JSON.stringify(data);
var request = https.request(options, (function(_this) {
return function(response) {
var response_chunk = [];
response.on('data', function(data) {
if (response.statusCode !== 200) {
return cb(new Error("response.statusCode != 200"));
}
response_chunk.push(data);
});
return response.on('end', function() {
var responseData, totalData;
totalData = response_chunk.join('');
try {
responseData = JSON.parse(totalData);
} catch (_error) {
return cb(_error);
}
return cb(null, responseData);
});
};
})(this));
request.write(post_data);
request.end();
request.on('error', function (exp) {
console.log('problem with request: ' + exp.message);
});
};
IAPVerifier.requestOptions = function() {
return options = {
host: 'buy.itunes.apple.com',
port: 443,
path: '/verifyReceipt',
method: "POST",
rejectUnauthorized: false/*不加:返回证书不受信任CERT_UNTRUSTED*/
};
};
IAPVerifier.verifyWithRetry(postData, true, function (e, responseData) {
console.log("responseData:",JSON.stringify(responseData,null,'\t'));
//todo 验证客户端传过来的transactionIdentifier,是否存在于"in_app"中
});
参考链接:
https://www.jianshu.com/p/2f98b7937b6f