laravel-actions
Laravel Actionsパッケージを利用して、アクションクラスの作成、既存コードのアクション化、実行方法、テスト、認証設定など、laravel-actionsに関する様々な疑問に対応し、コードの作成、説明、リファクタリングを支援するSkill。
📜 元の英語説明(参考)
Write, scaffold, explain, and refactor code using the lorisleiva/laravel-actions package. Use this skill whenever the user is working with Laravel Actions, wants to create an action class, convert a controller/job/listener/command into an action, asks how to use AsAction trait, dispatch an action as a job, use an action as a controller or listener, set up validation/authorization in an action, test or mock actions, or asks anything about the laravel-actions package pattern.
🇯🇵 日本人クリエイター向け解説
Laravel Actionsパッケージを利用して、アクションクラスの作成、既存コードのアクション化、実行方法、テスト、認証設定など、laravel-actionsに関する様々な疑問に対応し、コードの作成、説明、リファクタリングを支援するSkill。
※ jpskill.com 編集部が日本のビジネス現場向けに補足した解説です。Skill本体の挙動とは独立した参考情報です。
下記のコマンドをコピーしてターミナル(Mac/Linux)または PowerShell(Windows)に貼り付けてください。 ダウンロード → 解凍 → 配置まで全自動。
mkdir -p ~/.claude/skills && cd ~/.claude/skills && curl -L -o laravel-actions.zip https://jpskill.com/download/8679.zip && unzip -o laravel-actions.zip && rm laravel-actions.zip
$d = "$env:USERPROFILE\.claude\skills"; ni -Force -ItemType Directory $d | Out-Null; iwr https://jpskill.com/download/8679.zip -OutFile "$d\laravel-actions.zip"; Expand-Archive "$d\laravel-actions.zip" -DestinationPath $d -Force; ri "$d\laravel-actions.zip"
完了後、Claude Code を再起動 → 普通に「動画プロンプト作って」のように話しかけるだけで自動発動します。
💾 手動でダウンロードしたい(コマンドが難しい人向け)
- 1. 下の青いボタンを押して
laravel-actions.zipをダウンロード - 2. ZIPファイルをダブルクリックで解凍 →
laravel-actionsフォルダができる - 3. そのフォルダを
C:\Users\あなたの名前\.claude\skills\(Win)または~/.claude/skills/(Mac)へ移動 - 4. Claude Code を再起動
⚠️ ダウンロード・利用は自己責任でお願いします。当サイトは内容・動作・安全性について責任を負いません。
🎯 このSkillでできること
下記の説明文を読むと、このSkillがあなたに何をしてくれるかが分かります。Claudeにこの分野の依頼をすると、自動で発動します。
📦 インストール方法 (3ステップ)
- 1. 上の「ダウンロード」ボタンを押して .skill ファイルを取得
- 2. ファイル名の拡張子を .skill から .zip に変えて展開(macは自動展開可)
- 3. 展開してできたフォルダを、ホームフォルダの
.claude/skills/に置く- · macOS / Linux:
~/.claude/skills/ - · Windows:
%USERPROFILE%\.claude\skills\
- · macOS / Linux:
Claude Code を再起動すれば完了。「このSkillを使って…」と話しかけなくても、関連する依頼で自動的に呼び出されます。
詳しい使い方ガイドを見る →- 最終更新
- 2026-05-18
- 取得日時
- 2026-05-18
- 同梱ファイル
- 1
📖 Skill本文(日本語訳)
※ 原文(英語/中国語)を Gemini で日本語化したものです。Claude 自身は原文を読みます。誤訳がある場合は原文をご確認ください。
Laravel Actions
lorisleiva/laravel-actions を使用すると、特定のタスクを1つ 処理する単一の PHP クラスを作成し、それを オブジェクト、コントローラ、ジョブ、リスナー、または コマンド として実行できます。状況に応じて適切なものを選択できます。
インストール: composer require lorisleiva/laravel-actions
作成: php artisan make:action MyAction
コア構造
すべてのアクションは、AsAction トレイトと handle メソッドを持つプレーンな PHP クラスです。
use Lorisleiva\Actions\Concerns\AsAction;
class PublishNewArticle
{
use AsAction;
public function handle(User $author, string $title, string $body): Article
{
return $author->articles()->create(compact('title', 'body'));
}
}
- アクションはトピックごとにグループ化して
app/Actions/に配置します (例:app/Actions/Articles/) - アクション名には、短い動詞から始まる文を使用します:
SendWelcomeEmail、CreateInvoice、SyncContacts - 依存性の注入にはコンストラクタインジェクションを使用します。アクションは常にコンテナから解決されます。
オブジェクトとして
// 解決して実行
PublishNewArticle::run($author, 'Title', 'Body');
// 解決のみ
$action = PublishNewArticle::make();
// 条件付き実行
PublishNewArticle::runIf($condition, $author, 'Title', 'Body');
PublishNewArticle::runUnless($condition, $author, 'Title', 'Body');
コントローラとして
呼び出し可能なコントローラと同様に、ルートに登録します。
Route::post('/articles', PublishNewArticle::class)->middleware('auth');
リクエストデータを handle の引数にマッピングするには、asController を実装します。
public function asController(Request $request): ArticleResource
{
$article = $this->handle(
$request->user(),
$request->input('title'),
$request->input('body'),
);
return new ArticleResource($article);
}
asController が省略された場合、handle が呼び出し可能として直接使用されます。
アクション自体のミドルウェア:
public function getControllerMiddleware(): array
{
return ['auth', 'verified'];
}
JSON と HTML で異なるレスポンス:
public function jsonResponse(Article $article, Request $request): ArticleResource
{
return new ArticleResource($article);
}
public function htmlResponse(Article $article, Request $request): RedirectResponse
{
return redirect()->route('articles.show', $article);
}
インラインでルートを登録 (オプション):
public static function routes(Router $router): void
{
$router->post('/articles', static::class);
}
次に、サービスプロバイダで Actions::registerRoutes(['app/Actions']) を呼び出します。
複数エンドポイントアクションの明示的なルートメソッド:
Route::get('/articles/create', [PublishNewArticle::class, 'showForm']);
Route::post('/articles', PublishNewArticle::class);
バリデーションと認可 (コントローラとして)
ActionRequest を注入して、アクション自体で定義されたバリデーション/認可をトリガーします。
use Lorisleiva\Actions\ActionRequest;
public function asController(ActionRequest $request): ArticleResource
{
$article = $this->handle(
$request->user(),
$request->validated('title'),
$request->validated('body'),
);
return new ArticleResource($article);
}
public function authorize(ActionRequest $request): bool
{
return $request->user()->can('create', Article::class);
}
public function rules(): array
{
return [
'title' => ['required', 'string', 'max:255'],
'body' => ['required', 'string'],
];
}
追加のバリデーションフック:
public function prepareForValidation(ActionRequest $request): void { /* mutate input */ }
public function withValidator(Validator $validator): void { /* add callbacks */ }
public function afterValidator(Validator $validator): void { /* after hook */ }
public function getValidator(): Validator { /* full control */ }
public function getValidationData(): array { return $this->all(); }
public function getValidationMessages(): array { return []; }
public function getValidationAttributes(): array { return []; }
public function getValidationRedirect(Request $request): string { return url()->previous(); }
public function getValidationErrorBag(): string { return 'default'; }
public function getValidationFailure(): void { throw new ValidationException(...); }
public function getAuthorizationFailure(): void { throw new AuthorizationException(...); }
ジョブとして
// 非同期ディスパッチ
PublishNewArticle::dispatch($author, 'Title', 'Body');
// 条件付きディスパッチ
PublishNewArticle::dispatchIf($cond, $author, 'Title', 'Body');
PublishNewArticle::dispatchUnless($cond, $author, 'Title', 'Body');
// 同期ディスパッチ
PublishNewArticle::dispatchSync($author, 'Title', 'Body');
// レスポンス送信後
PublishNewArticle::dispatchAfterResponse($author, 'Title', 'Body');
ジョブ固有の動作が handle と異なる場合にのみ、asJob を実装します。
public function asJob(Team $team): void
{
$this->handle($team, fullReport: true);
}
ジョブのデフォルト設定:
public string $queue = 'emails';
public int $tries = 3;
public int $timeout = 60;
public int $maxExceptions = 2;
public function configureJob(JobDecorator $job): void
{
$job->onQueue('high')->delay(now()->addMinutes(5));
}
public function getJobBackoff(): array { return [10, 30, 60]; }
public function getJobRetryUntil(): DateTime { return now()->addHour(); }
public function getJobMiddleware(): array { return [new WithoutOverlapping($this->team->id)]; }
ユニークジョブ:
use Illuminate\Contracts\Queue\ShouldBeUnique;
class SendTeamReport implements ShouldBeUnique
{
use AsAction;
public function getJobUniqueId(Team $team): int { return $team->id; }
public function getJobUniqueFor(): int { return 3600; }
}
ジョブチェイニング:
SendWelcomeEmail::withChain([
VerifyEmailAddress::makeJob($user),
AssignDefaultRole::makeJob($user),
])->dispatch($user);
バッチ処理:
use Illuminate\Support\Facades\Bus;
Bus::batch([
ProcessInvoice::makeJob($invoiceA),
ProcessInvoice::ma 📜 原文 SKILL.md(Claudeが読む英語/中国語)を展開
Laravel Actions
lorisleiva/laravel-actions lets you write a single PHP class that handles one specific task and run it as an object, controller, job, listener, or command — whichever is appropriate.
Install: composer require lorisleiva/laravel-actions
Create: php artisan make:action MyAction
Core structure
Every action is a plain PHP class with the AsAction trait and a handle method:
use Lorisleiva\Actions\Concerns\AsAction;
class PublishNewArticle
{
use AsAction;
public function handle(User $author, string $title, string $body): Article
{
return $author->articles()->create(compact('title', 'body'));
}
}
- Place actions in
app/Actions/grouped by topic (e.g.app/Actions/Articles/) - Name them as short verb-first sentences:
SendWelcomeEmail,CreateInvoice,SyncContacts - Use constructor injection for dependencies — actions are always resolved from the container
As an Object
// Resolve and run
PublishNewArticle::run($author, 'Title', 'Body');
// Resolve only
$action = PublishNewArticle::make();
// Conditional execution
PublishNewArticle::runIf($condition, $author, 'Title', 'Body');
PublishNewArticle::runUnless($condition, $author, 'Title', 'Body');
As a Controller
Register in routes just like an invokable controller:
Route::post('/articles', PublishNewArticle::class)->middleware('auth');
Implement asController to map request data to handle args:
public function asController(Request $request): ArticleResource
{
$article = $this->handle(
$request->user(),
$request->input('title'),
$request->input('body'),
);
return new ArticleResource($article);
}
If asController is omitted, handle is used directly as the invokable.
Middleware on the action itself:
public function getControllerMiddleware(): array
{
return ['auth', 'verified'];
}
Different responses for JSON vs HTML:
public function jsonResponse(Article $article, Request $request): ArticleResource
{
return new ArticleResource($article);
}
public function htmlResponse(Article $article, Request $request): RedirectResponse
{
return redirect()->route('articles.show', $article);
}
Register routes inline (optional):
public static function routes(Router $router): void
{
$router->post('/articles', static::class);
}
Then call Actions::registerRoutes(['app/Actions']) in a service provider.
Explicit route methods for multi-endpoint actions:
Route::get('/articles/create', [PublishNewArticle::class, 'showForm']);
Route::post('/articles', PublishNewArticle::class);
Validation & Authorization (as Controller)
Inject ActionRequest to trigger validation/authorization defined on the action itself:
use Lorisleiva\Actions\ActionRequest;
public function asController(ActionRequest $request): ArticleResource
{
$article = $this->handle(
$request->user(),
$request->validated('title'),
$request->validated('body'),
);
return new ArticleResource($article);
}
public function authorize(ActionRequest $request): bool
{
return $request->user()->can('create', Article::class);
}
public function rules(): array
{
return [
'title' => ['required', 'string', 'max:255'],
'body' => ['required', 'string'],
];
}
Additional validation hooks:
public function prepareForValidation(ActionRequest $request): void { /* mutate input */ }
public function withValidator(Validator $validator): void { /* add callbacks */ }
public function afterValidator(Validator $validator): void { /* after hook */ }
public function getValidator(): Validator { /* full control */ }
public function getValidationData(): array { return $this->all(); }
public function getValidationMessages(): array { return []; }
public function getValidationAttributes(): array { return []; }
public function getValidationRedirect(Request $request): string { return url()->previous(); }
public function getValidationErrorBag(): string { return 'default'; }
public function getValidationFailure(): void { throw new ValidationException(...); }
public function getAuthorizationFailure(): void { throw new AuthorizationException(...); }
As a Job
// Async dispatch
PublishNewArticle::dispatch($author, 'Title', 'Body');
// Conditional dispatch
PublishNewArticle::dispatchIf($cond, $author, 'Title', 'Body');
PublishNewArticle::dispatchUnless($cond, $author, 'Title', 'Body');
// Sync dispatch
PublishNewArticle::dispatchSync($author, 'Title', 'Body');
// After response is sent
PublishNewArticle::dispatchAfterResponse($author, 'Title', 'Body');
Implement asJob only when the job-specific behaviour differs from handle:
public function asJob(Team $team): void
{
$this->handle($team, fullReport: true);
}
Configure job defaults:
public string $queue = 'emails';
public int $tries = 3;
public int $timeout = 60;
public int $maxExceptions = 2;
public function configureJob(JobDecorator $job): void
{
$job->onQueue('high')->delay(now()->addMinutes(5));
}
public function getJobBackoff(): array { return [10, 30, 60]; }
public function getJobRetryUntil(): DateTime { return now()->addHour(); }
public function getJobMiddleware(): array { return [new WithoutOverlapping($this->team->id)]; }
Unique jobs:
use Illuminate\Contracts\Queue\ShouldBeUnique;
class SendTeamReport implements ShouldBeUnique
{
use AsAction;
public function getJobUniqueId(Team $team): int { return $team->id; }
public function getJobUniqueFor(): int { return 3600; }
}
Job chaining:
SendWelcomeEmail::withChain([
VerifyEmailAddress::makeJob($user),
AssignDefaultRole::makeJob($user),
])->dispatch($user);
Batching:
use Illuminate\Support\Facades\Bus;
Bus::batch([
ProcessInvoice::makeJob($invoiceA),
ProcessInvoice::makeJob($invoiceB),
])->dispatch();
Horizon tags & display name:
public function getJobTags(Team $team): array { return ["team:{$team->id}"]; }
public function getJobDisplayName(): string { return 'Send Team Report'; }
As a Listener
Register in EventServiceProvider:
protected $listen = [
UserRegistered::class => [SendWelcomeEmail::class],
];
Or with the Event facade:
Event::listen(UserRegistered::class, SendWelcomeEmail::class);
For a queueable listener, add implements ShouldQueue to the action.
Use asListener to map event data to handle args:
public function asListener(UserRegistered $event): void
{
$this->handle($event->user);
}
As a Command
Register in Kernel::$commands or auto-register:
Actions::registerCommands(['app/Actions']);
use Illuminate\Console\Command;
class SendTeamReport
{
use AsAction;
public string $commandSignature = 'teams:report {team_id}';
public string $commandDescription = 'Send the weekly report to a team.';
public function asCommand(Command $command): void
{
$team = Team::findOrFail($command->argument('team_id'));
$this->handle($team);
$command->info('Report sent!');
}
// Dynamic signature/description/help:
public function getCommandSignature(): string { return '...'; }
public function getCommandDescription(): string { return '...'; }
public function getCommandHelp(): string { return '...'; }
public function isCommandHidden(): bool { return false; }
}
Testing & Mocking
// Mock — set expectations before running
PublishNewArticle::mock()
->shouldReceive('handle')
->once()
->andReturn($fakeArticle);
// Shorthand
PublishNewArticle::mock()->shouldRun()->once()->andReturn($fakeArticle);
PublishNewArticle::mock()->shouldNotRun();
// Partial mock (only mocked methods get expectations)
PublishNewArticle::partialMock()->shouldReceive('fetch')->andReturn([...]);
// Spy — run first, assert after
PublishNewArticle::spy()->shouldHaveReceived('handle')->once();
PublishNewArticle::spy()->allowToRun();
// Lifecycle helpers
PublishNewArticle::isFake(); // bool — is currently mocked?
PublishNewArticle::clearFake(); // reset to real implementation
Assert jobs were dispatched:
Queue::fake();
// ...trigger code...
PublishNewArticle::assertPushed();
PublishNewArticle::assertPushed(2); // dispatched exactly N times
PublishNewArticle::assertPushed(fn ($action, $args) => $args[0]->is($team));
PublishNewArticle::assertNotPushed();
PublishNewArticle::assertPushedOn('high', fn ($action, $args) => true);
WithAttributes (optional, v2.1+)
For actions that benefit from validated, unified attribute bags (useful when porting v1 code or when the same validation should apply across object and controller usage):
use Lorisleiva\Actions\Concerns\AsAction;
use Lorisleiva\Actions\Concerns\WithAttributes;
class PublishNewArticle
{
use AsAction;
use WithAttributes;
public function handle(User $author, array $data = []): Article
{
$this->fill($data);
$this->validateAttributes(); // triggers authorize + rules
return $author->articles()->create($this->validated());
}
public function asController(ActionRequest $request): Article
{
$this->fillFromRequest($request);
return $this->handle($request->user());
}
}
WithAttributes methods: fill, set, get, has, all, only, except, fillFromRequest, validateAttributes.
Note: when WithAttributes is used, the ActionRequest will not auto-validate — call $request->validate() manually if needed.
More granular traits
Instead of AsAction you can cherry-pick:
AsObject—run,make,runIf,runUnlessAsController— controller decorator supportAsJob— job decorator supportAsListener— listener decorator supportAsCommand— command decorator supportAsFake— mock/spy support
Reference docs
For full API details, see:
- references/api.md — complete method list per trait