config = $config; // 初始化路由 } /** * 执行默认数据类型的通知方法 * @param string $raw 源数据 * @param NotifyHandler $handler 通知处理器 * @return string 处理结果 * @throws ApiException */ public function notifyRunDefault(string $raw, NotifyHandler $handler): string { return $this->notifyRun($raw, "application/json", "application/json", $handler); } /** * 执行通知方法 * @param string $raw 源数据 * @param string $contentType 服务端发送的数据类型 * @param string $accept 服务端需要接收的数据类型 * @param NotifyHandler $handler 通知处理器 * @return string 处理结果 * @throws ApiException */ public function notifyRun(string $raw, string $contentType, string $accept, NotifyHandler $handler): string { if (empty(trim($raw))) { throw new ApiException("内容为空"); } // 解析请求数据 $notifyRequest = null; if ($contentType === 'application/json') { $content = json_decode($raw); $notifyRequest = ObjectSerializer::deserialize($content, '\OpenAPI\Client\Model\AsyncNotifyRequest', []); } elseif ($contentType === 'application/xml') { // XML格式暂不支持 throw new ApiException("暂不支持的格式: " . $contentType); } else { throw new ApiException("不支持的数据格式: " . $contentType); } if ($notifyRequest instanceof AsyncNotifyRequest) { // 检查签名类型 if ($notifyRequest->getSignType() !== 'RSA') { throw new ApiException("不支持的签名类型: " . $notifyRequest->getSignType()); } // 验证通知 if (!$this->verifyNotify($notifyRequest)) { throw new ApiException("签名错误: " . json_encode($notifyRequest)); } $resp = new AsyncNotifyResponse(); // 调用对应通知处理方法 $module = $notifyRequest->getModule(); // 检查路径是否在路由表中 if (array_key_exists($module, $this->notifyRouter)) { // 调用对应的函数 $function = $this->notifyRouter[$module]; try { $answer = $function($handler, $notifyRequest->getPayload()); $resp->setPayload($answer); $resp->setResult("ok"); } catch (Exception $e) { echo sprintf("调用通知回调处理方法失败:在 %s 的第 %d 行: [%d] %s", $e->getFile(), $e->getLine(), $e->getCode(), $e->getMessage()); $resp->setResult("fail"); } $resp->setAnswerId($notifyRequest->getNotifyId()); $resp->setAppId($this->config->getAppid()); $resp->setTime(date($this->config->getDataFormat(), time())); $resp->setSignType($this->config->getSignType()); $resp->setCharset($this->config->getCharset()); $resp->setSign($this->answerSign($resp)); //处理完成 if ($accept === "application/json") { return json_encode($resp); } else if ($accept === "application/xml") { //返回XML格式 throw new ApiException("暂不支持的数据接收格式:" . $accept); } else { //未知的格式 throw new ApiException("未知的数据接收格式:" . $accept); } } else { // 如果路由不存在,抛出异常或进行相应的错误处理 throw new ApiException("路由未定义: " . $module); } } else { throw new ApiException("对象格式错误"); } } /** * @throws ApiException */ function requestSign($params): string { // 获取payload和其他参数 $payload = $params['payload']; $content = sprintf( "%s&%s&%s&%s&%s&%s&%s", $params['path'], $params['app_id'], $params['access_token'], $params['sign_type'], $params['charset'], $params['time'], $this->object2LinkStr($payload) ); // 计算SHA-256摘要 $bytesContent = mb_convert_encoding($content, $params['charset']); $digestData = hash('sha256', $bytesContent, true); // 加载私钥 $privateKeyContent = $this->config->getPrivateKey(); $privateKey = openssl_pkey_get_private($privateKeyContent); if ($privateKey === false) { while ($errMsg = openssl_error_string()) { error_log($errMsg); } throw new ApiException('Invalid private key'); } // 使用SHA256withRSA签名 $signature = ''; $result = openssl_sign($digestData, $signature, $privateKey, OPENSSL_ALGO_SHA256); if ($result === false) { throw new ApiException('Failed to sign data'); } // 检查PHP版本,只有在PHP 7.x版本中才释放私钥资源 if (PHP_VERSION_ID < 80000) { openssl_pkey_free($privateKey); } // 返回十六进制编码的签名 return bin2hex($signature); } /** * @throws ApiException */ private function answerSign(AsyncNotifyResponse $resp): string { // 获取payload和其他参数 $content = sprintf( "%s&%s&%s&%s&%s&%s&%s", $resp->getAnswerId(), $resp->getAppId(), $resp->getResult(), $resp->getSignType(), $resp->getCharset(), $resp->getTime(), $this->object2LinkStr($resp->getPayload()) ); // 计算SHA-256摘要 $bytesContent = mb_convert_encoding($content, $resp->getCharset()); $digestData = hash('sha256', $bytesContent, true); // 加载私钥 $privateKeyContent = $this->config->getPrivateKey(); $privateKey = openssl_pkey_get_private($privateKeyContent); if ($privateKey === false) { while ($errMsg = openssl_error_string()) { error_log($errMsg); } throw new ApiException('Invalid private key'); } // 使用SHA256withRSA签名 $signature = ''; $result = openssl_sign($digestData, $signature, $privateKey, OPENSSL_ALGO_SHA256); if ($result === false) { throw new ApiException('Failed to sign data'); } // 检查PHP版本,只有在PHP 7.x版本中才释放私钥资源 if (PHP_VERSION_ID < 80000) { openssl_pkey_free($privateKey); } // 返回十六进制编码的签名 return bin2hex($signature); } function object2LinkStr($o): string { if ($o === null) { return ""; } // 处理简单值类型 if (is_numeric($o) || is_string($o)) { return strval($o); } if (is_bool($o)) { return $o ? 'true' : 'false'; } // 处理数组 if (is_array($o)) { // 检查数组是否是关联数组 if (array_keys($o) !== range(0, count($o) - 1)) { // 对关联数组按键进行字典排序 ksort($o); $result = []; foreach ($o as $key => $value) { // 递归处理数组内部的每个元素 $result[] = $key . "=" . $this->object2LinkStr($value); } return implode("&", $result); } else { // 非关联数组,按元素顺序处理 return implode(",", array_map([$this, 'object2LinkStr'], $o)); } } // 其他类型数据处理 return strval($o); } /** * @throws ApiException */ function verifySign($data): bool { // 检查签名类型 $signType = $data['sign_type'] ?? null; if ($signType !== 'RSA') { throw new ApiException("不支持的签名类型: {$signType}"); } $waitStr = $this->object2LinkStr($data['payload']); $formatStr = sprintf( "%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s", $data['response_id'] ?? '', $data['err_code'] ?? '', $data['err_msg'] ?? '', $data['sub_err'] ?? '', $data['sub_msg'] ?? '', $data['time'] ?? '', $data['open_id'] ?? '', $data['sign_type'] ?? '', $data['charset'] ?? 'UTF-8', $data['description'] ?? '', $waitStr ); $bodyCharset = $data['charset'] ?? 'UTF-8'; $bytesContent = mb_convert_encoding($formatStr, $bodyCharset); $digestData = hash('sha256', $bytesContent, true); $pubKey = openssl_pkey_get_public($this->config->getApiPublicKey()); if ($pubKey === false) { while ($errMsg = openssl_error_string()) { error_log($errMsg); } return false; } $signature = openssl_verify($digestData, hex2bin($data['sign']), $pubKey, OPENSSL_ALGO_SHA256); // 检查PHP版本,只有在PHP 7.x版本中才释放公钥资源 if (PHP_VERSION_ID < 80000) { openssl_free_key($pubKey); } return $signature === 1; } /** * @throws ApiException */ function verifyNotify(AsyncNotifyRequest $request): bool { // 检查签名类型 $signType = $request->getSignType() ?? null; if ($signType !== 'RSA') { throw new ApiException("不支持的签名类型: {$signType}"); } // 检查payload $waitStr = $this->object2LinkStr($request->getPayload()); $formatStr = sprintf( "%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s&%s", $request->getNotifyId() ?? '', $request->getSourceId() ?? '', $request->getAppId() ?? '', $request->getErrCode() ?? '', $request->getErrMsg() ?? '', $request->getSubErr() ?? '', $request->getSubMsg() ?? '', $request->getTime() ?? '', $request->getOpenId() ?? '', $request->getSignType() ?? '', $request->getCharset() ?? 'UTF-8', $request->getDescription() ?? '', $waitStr ); $bodyCharset = $request->getCharset() ?? 'UTF-8'; $bytesContent = mb_convert_encoding($formatStr, $bodyCharset); $digestData = hash('sha256', $bytesContent, true); $pubKey = openssl_pkey_get_public($this->config->getApiPublicKey()); if ($pubKey === false) { while ($errMsg = openssl_error_string()) { error_log($errMsg); } return false; } $signature = openssl_verify($digestData, hex2bin($request->getSign()), $pubKey, OPENSSL_ALGO_SHA256); // 检查PHP版本,只有在PHP 7.x版本中才释放公钥资源 if (PHP_VERSION_ID < 80000) { openssl_free_key($pubKey); } return $signature === 1; } }