-
Notifications
You must be signed in to change notification settings - Fork 20
/
Copy pathAlipay.php
227 lines (201 loc) · 5.79 KB
/
Alipay.php
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
<?php
/**
* A concise library for single transactions in Alipay.
* @author bitmash
* @version 2.0.0
* https://github.com/bitmash/alipay-api-php
*/
class AlipayException extends Exception {}
class Alipay {
/**
* For testing purposes
*/
const ENVIRONMENT = "development";
/**
* The application Partner ID
*
* @var string $_partner_id
*/
private $_partner_id = "";
/**
* The application Private Key
*
* @var string $_key
*/
private $_key = "";
/**
* The Alipay API endpoint
*
* @var string $_endpoint
*/
private $_endpoint = "";
/**
* Instantiates connection vars
*
* @return void
*/
public function __construct()
{
if (self::ENVIRONMENT == "development")
{
$this->_partner_id = "2088101122136241";
$this->_key = "760bdzec6y9goq7ctyx96ezkz78287de";
$this->_endpoint = "https://openapi.alipaydev.com/gateway.do";
}
else
{
$this->_partner_id = ""; // replace with your production partner ID
$this->_key = ""; // replace with your production private key
$this->_endpoint = "https://mapi.alipay.com/gateway.do";
}
}
/**
* We create a transaction URL for Alipay. There are two types of response
* handlers. In some cases a transaction is delayed from being completed
* while it's being verified by Alipay; these ping notify_url later.
*
* return_url
* Alipay sends the buyer back to this URL synchronously, along with a GET response.
*
* notify_url
* A POST response is sent asynchronously once the payment is verified. A
* 'notify_id' will be in the response that needs to be verfied by calling
* verifyPayment. The 'notify_id' expires in 1 minute.
*
* @param string(64) $sale_id Your internal transaction ID for reference
* @param decimal(8,2) $amount The amount charged to the user
* @param string(3) $currency The currency for the amount
* @param string(256) $description A description of the goods being sold
* @param string(200) $return_url Your URL to return to after payment
* @param string(200) $notify_url Your URL Alipay will ping after payment
* @param boolean $is_mobile Whether or not the user is on a mobile device
* @return string
**/
public function createPayment($sale_id = "", $amount = 0, $currency = "USD", $description = "", $return_url = "", $notify_url = "", $is_mobile = false)
{
$data = [
'body' => $description,
'service' => $is_mobile ? 'create_forex_trade_wap' : 'create_forex_trade',
'out_trade_no' => $sale_id,
'currency' => $currency,
'total_fee' => $amount,
'subject' => $description,
'return_url' => $return_url,
'notify_url' => $notify_url,
'partner' => $this->_partner_id,
'_input_charset' => "utf-8"
];
return $this->_endpoint . "?" . $this->_prepData($data);
}
/**
* Compares the signed response data from Alipay with our own key
* using the response parameters. We also verify the transaction by using
* the 'notify_id' and pinging Alipay again.
*
* Possible Trade Status:
* WAIT_BUYER_PAY
* TRADE_CLOSED
* TRADE_FINISHED
*
* @throws AlipayException
* @param array $data The response parameters from Alipay
* @return boolean
**/
public function verifyPayment($data = array())
{
$sign = $data['sign'];
unset($data['sign'], $data['sign_type']);
$new_sign = $this->_sign($data);
if ($sign != $new_sign)
{
throw new AlipayException("Hashes do not match: $sign - $new_sign (Transaction #{$data['out_trade_no']}).");
}
$request = [
'service' => 'notify_verify',
'partner' => $this->_partner_id,
'notify_id' => $data['notify_id']
];
$response = $this->_send(http_build_query($request), "GET");
if (preg_match("/true$/i", $response))
{
if ($data['trade_status'] == "TRADE_FINISHED")
{
return true;
}
}
else
{
throw new AlipayException("Alipay transaction is invalid (Transaction #{$data['out_trade_no']}).");
}
return false;
}
/**
* Send a request
*
* @throws Exception
* @param array $data The payload
* @param string $method The request method: POST, GET
* @return string
*/
private function _send($data = array(), $method = "GET")
{
$method = strtoupper($method);
if ($method == "GET")
{
$curl = curl_init($this->_endpoint . "?$data");
curl_setopt($curl, CURLOPT_POST, false);
}
else
{
$curl = curl_init($this->_endpoint . "?_input_charset=utf-8");
curl_setopt($curl, CURLOPT_POST, true);
}
curl_setopt($curl, CURLOPT_HEADER, 0);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, 1);
curl_setopt($curl, CURLOPT_SSLVERSION, 3);
curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);
curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);
curl_setopt($curl, CURLOPT_CAINFO, "./alipay_ca.pem");
$response = curl_exec($curl);
$error = curl_error($curl);
curl_close($curl);
if ($error)
{
throw new Exception($error);
}
return $response;
}
/**
* Prepares a request for delivery by building and encoding the query.
*
* @param array $data Associative array of request parameters
* @return array
*/
private function _prepData($data = array())
{
$data['sign'] = $this->_sign($data);
$data['sign_type'] = "MD5";
ksort($data);
return http_build_query($data);
}
/**
* Sorts the parameters alphabetically and creates a "secure" hash with the
* secret key appended. When Alipay receives the request, they perform a
* similar procedure to verify the data has not been tampered with.
*
* @param array $data Associative array of request parameters
* @return string
*/
private function _sign($data = array())
{
ksort($data);
$query = "";
foreach ($data as $k => $v)
{
if ($v == "")
continue;
$query .= "$k=$v&";
}
return md5(substr($query, 0, -1) . $this->_key);
}
}