mirror of
http://124.126.16.154:8888/singularity/HyperfDevelopmentKitCore.git
synced 2026-01-15 02:15:06 +08:00
163 lines
5.2 KiB
PHP
163 lines
5.2 KiB
PHP
<?php
|
||
/**
|
||
* SessionMiddleware.php@HDK-Core
|
||
*
|
||
* @author 李东云 <Dongyun.Li@LuxCreo.Ai>
|
||
* Powered by PhpStorm
|
||
* Created on 2023/3/8
|
||
*/
|
||
|
||
namespace Singularity\HDK\Core\Middleware;
|
||
|
||
use Carbon\Carbon;
|
||
use Hyperf\Contract\ConfigInterface;
|
||
use Hyperf\Contract\SessionInterface;
|
||
use Hyperf\HttpMessage\Cookie\Cookie;
|
||
use Hyperf\HttpMessage\Server\Response;
|
||
use Hyperf\Session\SessionManager;
|
||
use Psr\Http\Message\RequestInterface;
|
||
use Psr\Http\Message\ResponseInterface;
|
||
use Psr\Http\Message\ServerRequestInterface;
|
||
use Psr\Http\Server\MiddlewareInterface;
|
||
use Psr\Http\Server\RequestHandlerInterface;
|
||
|
||
/**
|
||
* Singularity\HDK\Core\Middleware\SessionMiddleware@HDK-Core
|
||
*
|
||
* @author 李东云 <Dongyun.Li@LuxCreo.Ai>
|
||
* Powered by PhpStorm
|
||
* Created on 2023/3/8
|
||
*
|
||
* @deprecated since 0.2.21, use \Singularity\HDK\Auth\Middleware\SessionMiddleware instead.
|
||
*/
|
||
class SessionMiddleware implements MiddlewareInterface
|
||
{
|
||
public function __construct(private SessionManager $sessionManager, private ConfigInterface $config)
|
||
{
|
||
}
|
||
|
||
/**
|
||
* Process an incoming server request.
|
||
* Processes an incoming server request in order to produce a response.
|
||
* If unable to produce the response itself, it may delegate to the provided
|
||
* request handler to do so.
|
||
*/
|
||
public function process(ServerRequestInterface $request, RequestHandlerInterface $handler): ResponseInterface
|
||
{
|
||
if (!$this->isSessionAvailable()) {
|
||
return $handler->handle($request);
|
||
}
|
||
|
||
$session = $this->sessionManager->start($request);
|
||
|
||
try {
|
||
$response = $handler->handle($request);
|
||
} finally {
|
||
$this->storeCurrentUrl($request, $session);
|
||
$session = $this->sessionManager->getSession();
|
||
|
||
/*
|
||
* 现在的机制,但凡写入 Redis 的数据,
|
||
* 过期时间就是固定的 gc_maxlifetime
|
||
*
|
||
* 而 RedisHandler 只会被依赖注入时加载一次,
|
||
* 后续无法临时修改 gc_maxlifetime
|
||
*
|
||
* 又因为现在没有其他情况用到 session
|
||
* 只有登录之后才会记下用户信息
|
||
*
|
||
* 所以一个妥协的方案,就是只有登录了再写入 Redis
|
||
*
|
||
* (除非重写整套 RedisHandler/RedisHandlerFactory/...)
|
||
*/
|
||
if (!$this->auth($session)) {
|
||
$this->sessionManager->end($session);
|
||
}
|
||
}
|
||
|
||
return $this->addCookieToResponse($request, $response, $session);
|
||
}
|
||
|
||
private function isSessionAvailable(): bool
|
||
{
|
||
return $this->config->has('session.handler');
|
||
}
|
||
|
||
/**
|
||
* Store the current URL for the request if necessary.
|
||
*/
|
||
private function storeCurrentUrl(RequestInterface $request, SessionInterface $session): void
|
||
{
|
||
if ($request->getMethod() === 'GET') {
|
||
$session->setPreviousUrl($this->fullUrl($request));
|
||
}
|
||
}
|
||
|
||
/**
|
||
* Add the session cookie to the response·.
|
||
*/
|
||
private function addCookieToResponse(
|
||
ServerRequestInterface $request,
|
||
ResponseInterface $response,
|
||
SessionInterface $session
|
||
): ResponseInterface {
|
||
$cookie = new Cookie(
|
||
name: $session->getName(),
|
||
value: $session->getId(),
|
||
expire: $this->getCookieExpirationDate($session),
|
||
path: $this->config->get('session.options.path', '/'),
|
||
domain: $this->config->get('session.options.domain', $request->getUri()->getHost()),
|
||
secure: $this->config->get(
|
||
'session.options.secure',
|
||
strtolower($request->getUri()->getScheme()) === 'https'
|
||
),
|
||
httpOnly: true,
|
||
sameSite: $this->config->get('session.options.samesite', Cookie::SAMESITE_LAX)
|
||
);
|
||
if (!method_exists($response, 'withCookie')) {
|
||
return $response->withHeader('Set-Cookie', (string)$cookie);
|
||
}
|
||
/* @var Response $response */
|
||
return $response->withCookie($cookie);
|
||
}
|
||
|
||
/**
|
||
* Get the full URL for the request.
|
||
*/
|
||
private function fullUrl(RequestInterface $request): string
|
||
{
|
||
$uri = $request->getUri();
|
||
$query = $uri->getQuery();
|
||
$question = $uri->getHost() . $uri->getPath() == '/' ? '/?' : '?';
|
||
return $query ? $this->url($request) . $question . $query : $this->url($request);
|
||
}
|
||
|
||
/**
|
||
* Get the session lifetime in seconds.
|
||
*/
|
||
private function getCookieExpirationDate(SessionInterface $session): int
|
||
{
|
||
// if ($this->config->get('session.options.expire_on_close')) {
|
||
if (!$this->auth($session)) {
|
||
$expirationDate = 0;
|
||
} else {
|
||
$expireSeconds = $this->config->get('session.options.cookie_lifetime', 5 * 60 * 60);
|
||
$expirationDate = Carbon::now()->addSeconds($expireSeconds)->getTimestamp();
|
||
}
|
||
return $expirationDate;
|
||
}
|
||
|
||
/**
|
||
* Get the URL (no query string) for the request.
|
||
*/
|
||
private function url(RequestInterface $request): string
|
||
{
|
||
return rtrim(preg_replace('/\?.*/', '', (string)$request->getUri()));
|
||
}
|
||
|
||
private function auth(SessionInterface $session): bool
|
||
{
|
||
return $session->has('userInfo');
|
||
}
|
||
}
|