- 框架初始化
 - 安装插件
 - 修复PHP8.4报错
This commit is contained in:
2025-04-19 17:21:20 +08:00
commit c6a4e1f5f6
5306 changed files with 967782 additions and 0 deletions

View File

@@ -0,0 +1,192 @@
<?php
/*
* This file is part of the overtrue/socialite.
*
* (c) overtrue <i@overtrue.me>
*
* This source file is subject to the MIT license that is bundled
* with this source code in the file LICENSE.
*/
namespace Overtrue\Socialite\Providers;
use Overtrue\Socialite\AccessToken;
use Overtrue\Socialite\AccessTokenInterface;
use Overtrue\Socialite\AuthorizeFailedException;
use Overtrue\Socialite\InvalidStateException;
use Overtrue\Socialite\ProviderInterface;
use Overtrue\Socialite\User;
/**
* Class FeiShuProvider.
*
* @author qijian.song@show.world
*
* @see https://open.feishu.cn/
*/
class FeiShuProvider extends AbstractProvider implements ProviderInterface
{
/**
* 飞书接口域名.
*
* @var string
*/
protected $baseUrl = 'https://open.feishu.cn';
/**
* 应用授权作用域.
*
* @var array
*/
protected $scopes = ['user_info'];
/**
* 获取登录页面地址.
*
* {@inheritdoc}
*/
protected function getAuthUrl($state)
{
return $this->buildAuthUrlFromBase($this->baseUrl.'/open-apis/authen/v1/index', $state);
}
/**
* 获取授权码接口参数.
*
* @param string|null $state
*
* @return array
*/
protected function getCodeFields($state = null)
{
$fields = [
'redirect_uri' => $this->redirectUrl,
'app_id' => $this->getConfig()->get('client_id'),
];
if ($this->usesState()) {
$fields['state'] = $state;
}
return $fields;
}
/**
* 获取 app_access_token 地址.
*
* {@inheritdoc}
*/
protected function getTokenUrl()
{
return $this->baseUrl.'/open-apis/auth/v3/app_access_token/internal';
}
/**
* 获取 app_access_token.
*
* @return \Overtrue\Socialite\AccessToken
*/
public function getAccessToken($code = '')
{
$response = $this->getHttpClient()->post($this->getTokenUrl(), [
'headers' => ['Content-Type' => 'application/json'],
'json' => $this->getTokenFields($code),
]);
return $this->parseAccessToken($response->getBody()->getContents());
}
/**
* 获取 app_access_token 接口参数.
*
* @return array
*/
protected function getTokenFields($code)
{
return [
'app_id' => $this->getConfig()->get('client_id'),
'app_secret' => $this->getConfig()->get('client_secret'),
];
}
/**
* 格式化 token.
*
* @param \Psr\Http\Message\StreamInterface|array $body
*
* @return \Overtrue\Socialite\AccessTokenInterface
*/
protected function parseAccessToken($body)
{
if (!is_array($body)) {
$body = json_decode($body, true);
}
if (empty($body['app_access_token'])) {
throw new AuthorizeFailedException('Authorize Failed: '.json_encode($body, JSON_UNESCAPED_UNICODE), $body);
}
$data['access_token'] = $body['app_access_token'];
return new AccessToken($data);
}
/**
* 获取用户信息.
*
* @return array|mixed
*/
public function user(AccessTokenInterface $token = null)
{
if (is_null($token) && $this->hasInvalidState()) {
throw new InvalidStateException();
}
$token = $token ?: $this->getAccessToken();
$user = $this->getUserByToken($token, $this->getCode());
$user = $this->mapUserToObject($user)->merge(['original' => $user]);
return $user->setToken($token)->setProviderName($this->getName());
}
/**
* 通过 token 获取用户信息.
*
* @return array|mixed
*/
protected function getUserByToken(AccessTokenInterface $token)
{
$userUrl = $this->baseUrl.'/open-apis/authen/v1/access_token';
$response = $this->getHttpClient()->post(
$userUrl,
[
'json' => [
'app_access_token' => $token->getToken(),
'code' => $this->getCode(),
'grant_type' => 'authorization_code',
],
]
);
$result = json_decode($response->getBody(), true);
return $result['data'];
}
/**
* 格式化用户信息.
*
* @return User
*/
protected function mapUserToObject(?array $user)
{
return new User([
'id' => $this->arrayItem($user, 'open_id'),
'username' => $this->arrayItem($user, 'name'),
'nickname' => $this->arrayItem($user, 'name'),
'avatar' => $this->arrayItem($user, 'avatar_url'),
]);
}
}