anasa / response-strategy
该包的最新版本(v1.0.2)没有提供许可证信息。
将根据源(API/Web)隔离控制器返回类型的逻辑的包
v1.0.2
2024-09-02 12:40 UTC
Requires
- php: ^8.0
README
根据源(API/Web)隔离控制器返回类型的逻辑的包。
这是Laravel中处理CRUD将实现的输出类型的可选解决方案。请确保遵守命名规范,否则您将需要修改AdditionalDataRequest的(setMethod, setView, setRoute)输入。
工作原理
1- 从中间件或类似逻辑中设置Additional Data Request并识别所需的Response Strategy
<?php namespace App\Http\Middleware; use Closure; use Illuminate\Http\Request; use SebastianBergmann\Type\Exception; use Symfony\Component\HttpFoundation\Response; use Anasa\ResponseStrategy\{AdditionalDataRequest,ResponseStrategyFactory,ResponseContextInterface}; class ApiOrWebMiddleware { public function __construct(protected ResponseContextInterface $responseContext) { } /** * Handle an incoming request. * * @param \Closure(\Illuminate\Http\Request): (\Symfony\Component\HttpFoundation\Response) $next */ public function handle(Request $request, Closure $next): Response { /** * Set additional data request: * this will add the controller, method, view and resource. It's * some dinamic data to be used in the strategy to identify and build * the response. A facade will be used. */ $service = AdditionalDataRequest::getInstance(); $this->setAdditionalDataRequest($request, $service); $this->defineResponseStrategy($service); return $next($request); } private function setAdditionalDataRequest(Request $request, $service): void { $action = $request->route()->getAction(); $controller = class_basename($action['controller']); [, $methodName] = explode('@', $controller); $service->setMethod($request->expectsJson() || $request->is('api/*') ? 'API' : $methodName); $service->setView($request->route()->getName()); $service->setRoute($request->route()->getName()); } public function defineResponseStrategy() { try { $strategy = ResponseStrategyFactory::createStrategy($service->getMethod()); } catch (Exception $e) { throw new Exception('Unknown method'); } $this->responseContext->setStrategy($strategy); } }
注意
- setMethod将所有输入json输出设置为API。
- 如果您的项目使用自定义前缀来处理API输入,请确保添加Accept: application/json Header以标识json输出。
$service->setMethod($request->expectsJson() || $request->is('api/*') ? 'API' : $methodName);
2- 设置Response Service提供者
<?php namespace App\Providers; use Illuminate\Support\ServiceProvider; use Anasa\ResponseStrategy\{ResponseContext,ResponseContextInterface}; use Anasa\ResponseStrategy\Output\{ApiResponseStrategy, ViewResponseStrategy, RedirectResponseStrategy}; use Anasa\ResponseStrategy\OutputDataFormat\{StrategyData,StrategyDataInterface}; class ResponseServiceProvider extends ServiceProvider { public function register() { $this->app->bind(ApiResponseStrategy::class, function ($app) { return new ApiResponseStrategy(); }); $this->app->bind(ViewResponseStrategy::class, function ($app) { return new ViewResponseStrategy(); }); $this->app->bind(RedirectResponseStrategy::class, function ($app) { return new RedirectResponseStrategy(); }); $this->app->bind(StrategyDataInterface::class, function ($app) { return new StrategyData(); }); $this->app->singleton(ResponseContextInterface::class, function ($app) { return new ResponseContext(); }); } }
3- 在控制器中使用它。 没有检查或条件,只有请求和输出的输入和输出方式。
<?php namespace App\Http\Controllers; use Illuminate\Http\JsonResponse; use Illuminate\Contracts\View\View; use App\Http\Controllers\Controller; use App\Repositories\YourRepository; use Anasa\ResponseStrategy\ResponseContextInterface; use Anasa\ResponseStrategy\OutputDataFormat\StrategyDataInterface; class YourController extends Controller { public function __construct(protected YourRepository $repository, protected ResponseContextInterface $responseContext, protected StrategyDataInterface $strategyData) { } public function index(): View|JsonResponse { $data = $this->repository->all(); $strategy = $this->strategyData->setStrategyData(YourResource::collection($data)); return $this->responseContext->executeStrategy($strategy); } /** * No strategy needed */ public function create(): View { return View('yourResource.create'); } public function store(YourRequest $request): JsonResponse|RedirectResponse { $data = $this->repository->create($request->validated()); $strategy = $this->strategyData->setStrategyData(new YourResource($data), 'YourResource created successfully', Response::HTTP_CREATED); return $this->responseContext->executeStrategy($strategy); } public function show($id): JsonResponse|View { $data = $this->repository->find($id); //it uses findOrFail from the repository $strategy = $this->strategyData->setStrategyData(new YourResource($data)); return $this->responseContext->executeStrategy($strategy); } public function edit(string $id): View { $data = $this->repository->find($id); //it uses findOrFail from the repository return View('gateway.edit', ['YourData' => $data]); } public function update(YourRequest $request, string $id): JsonResponse|RedirectResponse { $updated_data = $this->repository->update($id, $request->validated()); //it uses findOrFail $strategy = $this->strategyData->setStrategyData(new YourResource($updated_data), 'YourResource updated successfully', Response::HTTP_OK); return $this->responseContext->executeStrategy($strategy); } public function destroy($id): JsonResponse|RedirectResponse { $this->repository->delete($id); //it uses findOrFail from the repository return $this->responseContext->executeStrategy($this->strategyData->setStrategyData([], 'YourResource deleted successfully', Response::HTTP_OK)); }
4- 对于测试,您可以添加:$service->setMethod('API');
namespace Tests\Feature; use Tests\TestCase; use Anasa\ResponseStrategy\AdditionalDataRequest; class GatewayTest extends TestCase { protected function setUp(): void { parent::setUp(); $service = AdditionalDataRequest::getInstance(); $service->setMethod('API'); }