在微信中,使用支付宝在线支付时,需要跳转至系统浏览器中,用户登录状态的丢失的解决方案
1、现有的实现,一个接口,是在微信中获取支付宝的支付链接(需要登录),还有一个回调接口(需要验签),还有一个查询接口(需要登录)。现在由于支付在线支付只能够在系统浏览器中进行,那么查询接口需要设置为游客可访问。为了保证安全性,决定做一个签名认证
2、在获取支付链接的接口中,如下实现
<?php namespace api\helpers; class AlipayQuerySignerHelper { /** * 获取签名 * * @param array $data 待签名数据(键值对) * @param string $secret 签名密钥 * @return string 返回签名字符串 */ public static function sign(array $data, string $secret): string { // 确保按 key 升序排列,签名一致 ksort($data); // 拼接成 key=value&key2=value2 的格式 $queryString = http_build_query($data, '', '&', PHP_QUERY_RFC3986); // 生成签名 return hash_hmac('sha256', $queryString, $secret); } /** * 验证签名 * * @param array $data 待验证的数据(必须包含 query_sign 字段) * @param string $secret 签名密钥 * @param int $timeout 签名超时时间(默认 600 秒) * @return bool */ public static function verify(array $data, string $secret, int $timeout = 6000): bool { if (!isset($data['query_sign'], $data['query_timestamp'])) { return false; } // 判断时间是否超出容忍范围(默认 ±5 分钟) if (abs(time() - $data['query_timestamp']) > $timeout) { return false; } $providedSign = $data['query_sign']; unset($data['query_sign']); $expectedSign = self::sign($data, $secret); return hash_equals($expectedSign, $providedSign); } }
$payload = [ 'query_order_id' => $order->id, 'query_user_id' => $order->user_id, 'query_timestamp' => time(), ]; $secret = Yii::$app->params['alipayQuerySignatureKey']; $sign = AlipayQuerySignerHelper::sign($payload, $secret); $query = array_merge($payload, ['query_sign' => $sign]); $queryString = http_build_query($query); $returnUrl = Yii::$app->params['frontendDomainHttps'] . '/convention/#/pages/ticket_payment_result/ticket_payment_result?convention_id=' . $ticketOrder->convention_id . '&ticket_id=' . $ticketOrder->id . '&_version=2&' . $queryString;
3、在查询接口中如下实现,接口响应,如图1
$params = Yii::$app->request->post(); $secret = Yii::$app->params['alipayQuerySignatureKey']; if (!AlipayQuerySignerHelper::verify($params, $secret)) { return [ 'code' => 10010, 'message' => '签名验证失败', ]; } // if (strcmp($order->user_id, Yii::$app->user->id) != 0) { // return [ // 'code' => 10008, // 'message' => '您没权限进行该操作', // ]; // }
近期评论