* * This source file is subject to the MIT license that is bundled * with this source code in the file LICENSE. */ namespace EasyWeChat\BasicService\Jssdk; use EasyWeChat\Kernel\BaseClient; use EasyWeChat\Kernel\Exceptions\RuntimeException; use EasyWeChat\Kernel\Support; use EasyWeChat\Kernel\Traits\InteractsWithCache; /** * Class Client. * * @author overtrue */ class Client extends BaseClient { use InteractsWithCache; /** * @var string */ protected $ticketEndpoint = 'https://api.weixin.qq.com/cgi-bin/ticket/getticket'; /** * Current URI. * * @var string */ protected $url; /** * Get config json for jsapi. * * @param array $jsApiList * @param bool $debug * @param bool $beta * @param bool $json * * @return array|string * * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException * @throws \Psr\SimpleCache\InvalidArgumentException * @throws \EasyWeChat\Kernel\Exceptions\RuntimeException */ public function buildConfig(array $jsApiList, bool $debug = false, bool $beta = false, bool $json = true) { $config = array_merge(compact('debug', 'beta', 'jsApiList'), $this->configSignature()); return $json ? json_encode($config) : $config; } /** * Return jsapi config as a PHP array. * * @param array $apis * @param bool $debug * @param bool $beta * * @return array * * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException * @throws \Psr\SimpleCache\InvalidArgumentException * @throws \EasyWeChat\Kernel\Exceptions\RuntimeException */ public function getConfigArray(array $apis, bool $debug = false, bool $beta = false) { return $this->buildConfig($apis, $debug, $beta, false); } /** * Get js ticket. * * @param bool $refresh * @param string $type * * @return array * * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException * @throws \EasyWeChat\Kernel\Exceptions\RuntimeException * @throws \GuzzleHttp\Exception\GuzzleException * @throws \Psr\SimpleCache\InvalidArgumentException */ public function getTicket(bool $refresh = false, string $type = 'jsapi'): array { $cacheKey = sprintf('easywechat.basic_service.jssdk.ticket.%s.%s', $type, $this->getAppId()); if (!$refresh && $this->getCache()->has($cacheKey)) { return $this->getCache()->get($cacheKey); } /** @var array $result */ $result = $this->castResponseToType( $this->requestRaw($this->ticketEndpoint, 'GET', ['query' => ['type' => $type]]), 'array' ); $this->getCache()->set($cacheKey, $result, $result['expires_in'] - 500); if (!$this->getCache()->has($cacheKey)) { throw new RuntimeException('Failed to cache jssdk ticket.'); } return $result; } /** * Build signature. * * @param string|null $url * @param string|null $nonce * @param int|null $timestamp * * @return array * * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException * @throws \EasyWeChat\Kernel\Exceptions\RuntimeException * @throws \Psr\SimpleCache\InvalidArgumentException */ protected function configSignature(string $url = null, string $nonce = null, $timestamp = null): array { $url = $url ?: $this->getUrl(); $nonce = $nonce ?: Support\Str::quickRandom(10); $timestamp = $timestamp ?: time(); return [ 'appId' => $this->getAppId(), 'nonceStr' => $nonce, 'timestamp' => $timestamp, 'url' => $url, 'signature' => $this->getTicketSignature($this->getTicket()['ticket'], $nonce, $timestamp, $url), ]; } /** * Sign the params. * * @param string $ticket * @param string $nonce * @param int $timestamp * @param string $url * * @return string */ public function getTicketSignature($ticket, $nonce, $timestamp, $url): string { return sha1(sprintf('jsapi_ticket=%s&noncestr=%s×tamp=%s&url=%s', $ticket, $nonce, $timestamp, $url)); } /** * @return string */ public function dictionaryOrderSignature() { $params = func_get_args(); sort($params, SORT_STRING); return sha1(implode('', $params)); } /** * Set current url. * * @param string $url * * @return $this */ public function setUrl(string $url) { $this->url = $url; return $this; } /** * Get current url. * * @return string */ public function getUrl(): string { if ($this->url) { return $this->url; } return Support\current_url(); } /** * @return string */ protected function getAppId() { return $this->app['config']->get('app_id'); } }