私はの付着していますアクションクラスの使用の代わりに、アプローチコントローラを。説明は非常に簡単です。コントローラに多くのアクションが含まれていることがよくあります。依存性注入の原則に従って、必要なすべての依存関係をコンストラクタに渡す必要があります。これにより、コントローラに非常に多くの依存関係があるが、特定の瞬間に状況が発生します。(例:リクエスト)一部の依存関係のみを使用します。そのスパゲッティコードを保守およびテストするのは困難です。
明確にするために、私はすでにZend Framework 2でそのアプローチを使用してきましたが、そこにはMiddlewareという名前が付いています。API-Platformでも、ControllerではなくActionクラスを使用する同様のものが見つかりましたが、問題は、それを調理する方法がわからないことです。
UPD:次のアクションクラスを取得して標準のコントローラーを置き換えるにはどうすればよいですか?また、通常のSymfonyプロジェクトにどの構成を追加する必要がありますか?
<?php
declare(strict_types=1);
namespace App\Action\Product;
use App\Entity\Product;
use Doctrine\ORM\EntityManager;
use Sensio\Bundle\FrameworkExtraBundle\Configuration\Method;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\HttpFoundation\Response;
use Symfony\Component\Routing\Annotation\Route;
class SoftDeleteAction
{
/**
* @var EntityManager
*/
private $entityManager;
/**
* @param EntityManager $entityManager
*/
public function __construct(EntityManager $entityManager)
{
$this->entityManager = $entityManager;
}
/**
* @Route(
* name="app_product_delete",
* path="products/{id}/delete"
* )
*
* @Method("DELETE")
*
* @param Product $product
*
* @return Response
*/
public function __invoke(Request $request, $id): Response
{
$product = $this->entityManager->find(Product::class, $id);
$product->delete();
$this->entityManager->flush();
return new Response('', 204);
}
}
質問は少し興味深いですが、stackoverflowについては少しあいまいです。だからここにいくつかの設定の詳細があります。
すぐに使えるS4スケルトンプロジェクトから始めます。
symfony new --version=lts s4api
cd s4api
bin/console --version # 4.4.11
composer require orm-pack
SoftDeleteActionを追加します。
namespace App\Action\Product;
class SoftDeleteAction
{
private $entityManager;
public function __construct(EntityManagerInterface $entityManager)
{
$this->entityManager = $entityManager;
}
public function __invoke(Request $request, int $id) : Response
{
return new Response('Product ' . $id);
}
}
そしてルートを定義します:
# config/routes.yaml
app_product_delete:
path: /products/{id}/delete
controller: App\Action\Product\SoftDeleteAction
この時点で、配線はほぼ完了しています。あなたが得るURLに行くと:
The controller for URI "/products/42/delete" is not callable:
その理由は、サービスはデフォルトでプライベートであるためです。通常は、サービスを公開する処理を行うAbstractControllerから拡張しますが、この場合の最も速い方法は、アクションをコントローラーとしてタグ付けすることです。
# config/services.yaml
App\Action\Product\SoftDeleteAction:
tags: ['controller.service_arguments']
この時点で、ワイヤードアクションが機能しているはずです。
もちろん、多くのバリエーションといくつかの詳細があります。ルートをPOSTまたは偽のDELETEに制限する必要があります。
また、空のControllerServiceArgumentsInterfaceを追加してから、services instanceof機能を使用してコントローラータグを適用することを検討すると、コントローラーサービスを手動で定義する必要がなくなります。
しかし、これはあなたが始めるのに十分なはずです。
この記事はインターネットから収集されたものであり、転載の際にはソースを示してください。
侵害の場合は、連絡してください[email protected]
コメントを追加