我在考虑在后台进行集成,除了支付页面之外没有任何网站作为c ++桌面应用程序的一部分。 是否可以遵循以下方案: 1.生成发票/销售,并通过REST API获取某种事务的唯一ID。 2.使用唯一ID将Paypal网站重定向到临时付款页面。 3.在后台,如果付款,则通过REST API每隔几分钟检查一次。
答案 0 :(得分:1)
我们终于找到了一种方法,这就是为什么我发布这个答案以及我们开发的一些代码(POC)。这是内置支付处理引擎的POC,允许您接受来自任何信用卡持卡人的付款(无论是否为PayPal客户),并支付解锁软件产品或特定功能的费用。
要处理付款,您需要申请PayPal developer并获取自己的PayPal凭据。然后,您将收到2组凭据。一个用于测试(“沙箱”),另一个用于现实生活。
首先,您可以使用Sandbox来测试API
Void InitPayPal(BOOL Sandbox, LPTSTR User, LPTSTR password, LPTSTR signature, LPTSTR successUrl, LPTSTR failedURL)
Sandbox - 表示您是使用PayPal的Sandbox帐户测试集成还是上线。
用户 - 您的PayPal用户名
密码 - 您的PayPal密码
签名 - 您的PayPal签名
successUrl - 指向您希望在付款成功后显示的网页的网址。
failedURL - 一个网址,指向您希望在失败/可以付款后显示的网页。
这个功能很简单:
void InitPayPal(BOOL Sandbox, LPTSTR User, LPTSTR password, LPTSTR signature, LPTSTR successUrl, LPTSTR failedURL, LPWSTR ProductName)
{
m_sandbox = Sandbox;
m_user = User;
m_password = password;
m_signature = signature;
m_SuccessURL = successUrl;
m_FailureURL = failedURL;
m_ProductName = ProductName;
CUR_CHAR = L"$";
SYSTEMTIME st;
GetSystemTime(&st);
g_tPayStart = CTime(st);
InitilizedPaypal = TRUE;
}
发起付款 如果您希望从程序中启动付款,请调用我编写的以下函数,该函数通常构建一个字符串(ExpChkoutStr)并使用以下PayPal API调用:
// Send string to PayPal server
WinHttpClient WinClient1(ExpChkoutStr.GetBuffer());
WinClient1.SetRequireValidSslCertificates(false);
WinClient1.SendHttpRequest(L"GET");
httpResponseContent1 = WinClient1.GetResponseContent();
CString strTransactionRet = UrlDecode(httpResponseContent1.c_str());
Express Checkout String(ExpChkoutStr)由另一个函数生成,该函数将成员变量的值和事务详细信息用于单个字符串:
CString result;
result = (m_sandbox) ? PAYPAL_SANDBOX_HTTPS : PAYPAL_REAL_HTTPS;
result += Q_USER;
result += m_user;
result += AND_PASSWORD;
result += m_password;
result += AND_SIGNATURE;
result += m_signature;
result += AND_PAYMENTAMOUNT;
result += strAmount;
result += L"&METHOD=SetExpressCheckout";
result += AND_RETURN_URL;
result += m_SuccessURL;
result += AND_CANCEL_URL;
result += m_FailureURL;
result += AND_VERSION;
result += L"&NOSHIPPING=1";
result += L"&ADDROVERRIDE=0&BRANDNAME=Secured Globe, Inc.";
result += L"&PAYMENTREQUEST_0_DESC=";
result += L"Item name: " + strUnits + L"(" + UnitName + L") ";
result += L"Price: " + strAmount;
result += L"&NOTETOBUYER=Here you can add a note to the buyer";
PayPal服务器的结果是一个“令牌”,用于找出必须打开的一次性网页(LinkToOpen),以便最终用户确认购买:
// Extract token from response
CString sToken = ExtractElement(strTransactionRet, L"TOKEN");
if (sToken == L"")
{
wprintf(L"Internal error: (Paypal): no token was generated (%s)", strTransactionRet);
MessageBox(NULL, L"Internal payment processing error", L"", MB_OK);
return FALSE;
}
CString LinkToOpen = (m_sandbox) ? SANDBOX_PAYPAL_CHECKOUT : REAL_PAYPAL_CHECKOUT;
LinkToOpen += L"&token=";
LinkToOpen += sToken;
然后,我们使用默认的Web浏览器以编程方式打开此一次性网页:
STARTUPINFO si;
PROCESS_INFORMATION pi;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
ZeroMemory(&pi, sizeof(pi));
CString command_line;
command_line.Format(L"cmd.exe /c start \"link\" \"%s\" ", LinkToOpen);
// LinkToOpen
if (!CreateProcess(NULL, // No module name (use command line)
command_line.GetBuffer(),
NULL, // Process handle not inheritable
NULL, // Thread handle not inhberitable
FALSE, // Set handle inheritance to FALSE
NORMAL_PRIORITY_CLASS | CREATE_NO_WINDOW, // No creation flags
NULL, // Use parent's environment block
NULL, // Use parent's starting directory
&si, // Pointer to STARTUPINFO structure
&pi) // Pointer to PROCESS_INFORMATION structure
)
{
wprintf(L"CreateProcess failed (%d).\n", GetLastError());
// At this stage you would want to mark this transaction as "failed"
return FALSE;
}
然后剩下的就是维护一个包含所有待处理事务的小型数据库,并跟进每个事务,直到它成功,失败,取消或超时已经过去。
为了从PayPal服务器响应中提取元素,我们编写了这个小函数:
CString ExtractElement(CString EntireString, CString ElementName)
{
CString result = L"";
CString WhatToFind = ElementName + L"=";
int foundToken = EntireString.Find(WhatToFind);
if (foundToken > -1)
{
int EndToken = EntireString.Find(L"&", foundToken);
if (EndToken != -1)
{
result = EntireString.Mid(foundToken + ElementName.GetLength()+1, EndToken - foundToken - ElementName.GetLength()-1);
}
}
return result;
}