Help us improve
Share bugs, ideas, or general feedback.
From acc
Generates PHP 8.4 Policy pattern for encapsulating authorization, validation, and business rules. Includes interfaces, implementations, exceptions, and unit tests for domain constraints.
npx claudepluginhub dykyi-roman/awesome-claude-code --plugin accHow this skill is triggered — by the user, by Claude, or both
Slash command
/acc:create-policyThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Creates Policy pattern infrastructure for encapsulating business rules and authorization logic.
Extracts validation rules, guards, business constraints, authorization rules, and invariants from Symfony/Laravel PHP domain code. Maps technical implementations to business terminology for stakeholders.
Guides selection and implementation of authorization models including RBAC, ABAC, ACL, ReBAC, and policy-as-code for permission systems and access control design.
Implements Clean Architecture, Hexagonal (Ports & Adapters), and Domain-Driven Design patterns in PHP 8.3+ with Symfony 7.x. For enterprise app architecture, legacy refactoring, DDD, and testable backends.
Share bugs, ideas, or general feedback.
Creates Policy pattern infrastructure for encapsulating business rules and authorization logic.
| Scenario | Example |
|---|---|
| Authorization checks | Can user cancel order? |
| Business rule validation | Is discount applicable? |
| Complex conditions | Multiple rules combined |
| Auditable decisions | Log why access denied |
Path: src/Domain/Shared/Policy/
PolicyResult.php — Result value object with and/or compositionCompositionMode.php — Enum for AllMustPass/AnyMustPassPath: src/Domain/{BoundedContext}/Policy/
{Name}PolicyInterface.php — Policy contractPath: src/Domain/{BoundedContext}/Policy/
{Rule1}Policy.php — First rule implementation{Rule2}Policy.php — Second rule implementation{Name}Policy.php — Composite policy combining rulesPath: src/Domain/Shared/Exception/
PolicyViolationException.php — Exception with policy context{Rule}PolicyTest.php — Individual rule tests{Name}PolicyTest.php — Composite policy testsPolicyResultTest.php — Result composition tests| Component | Path |
|---|---|
| Policy Interface | src/Domain/{BoundedContext}/Policy/ |
| Policy Implementation | src/Domain/{BoundedContext}/Policy/ |
| PolicyResult | src/Domain/Shared/Policy/ |
| Exception | src/Domain/Shared/Exception/ |
| Unit Tests | tests/Unit/Domain/{BoundedContext}/Policy/ |
| Component | Pattern | Example |
|---|---|---|
| Interface | {Name}PolicyInterface | OrderCancellationPolicyInterface |
| Implementation | {Rule}Policy | OrderOwnershipPolicy |
| Composite | {Name}Policy | OrderCancellationPolicy |
| Result | PolicyResult | PolicyResult |
| Exception | PolicyViolationException | PolicyViolationException |
| Test | {ClassName}Test | OrderOwnershipPolicyTest |
interface {Name}PolicyInterface
{
public function evaluate({SubjectType} $subject, {ResourceType} $resource): PolicyResult;
public function getRuleName(): string;
}
final readonly class PolicyResult
{
public static function allow(): self;
public static function deny(string $reason, array $metadata = []): self;
public function isAllowed(): bool;
public function isDenied(): bool;
public function and(self $other): self; // Both must pass
public function or(self $other): self; // Either can pass
}
final readonly class {Rule}Policy implements {Name}PolicyInterface
{
public function evaluate({Subject} $subject, {Resource} $resource): PolicyResult
{
if ({condition}) {
return PolicyResult::allow();
}
return PolicyResult::deny('{reason}', ['context' => 'data']);
}
public function getRuleName(): string
{
return '{rule_name}';
}
}
final readonly class {Name}Policy implements {Name}PolicyInterface
{
public function evaluate({Subject} $subject, {Resource} $resource): PolicyResult
{
return $this->rule1Policy->evaluate($subject, $resource)
->and($this->rule2Policy->evaluate($subject, $resource))
->and($this->rule3Policy->evaluate($subject, $resource));
}
}
// In UseCase
$result = $this->cancellationPolicy->evaluate($user, $order);
if ($result->isDenied()) {
throw new PolicyViolationException(
$this->cancellationPolicy->getRuleName(),
$result->getReason(),
$result->metadata
);
}
$order->cancel($reason);
| Anti-pattern | Problem | Solution |
|---|---|---|
| Side Effects | Policy modifies state | Keep evaluation pure |
| Boolean Returns | No denial reason | Use PolicyResult |
| Fat Policies | Too many rules | Split into composable policies |
| Hardcoded Values | Can't configure | Inject thresholds |
| No Logging Context | Can't debug | Include metadata |
For complete PHP templates and examples, see:
references/templates.md — Policy, composite, result templatesreferences/examples.md — Order cancellation policies and tests