askill
acc-create-state

acc-create-stateSafety 95Repository

Generates State pattern for PHP 8.5. Creates state machines with context, state interface, and concrete states for behavior changes. Includes unit tests.

0 stars
1.2k downloads
Updated 2/5/2026

Package Files

Loading files...
SKILL.md

State Pattern Generator

Creates State pattern infrastructure for objects that change behavior based on internal state.

When to Use

ScenarioExample
Object behavior varies by stateOrder (pending, paid, shipped)
Many conditionals based on stateDocument workflow
State-specific transitionsSubscription lifecycle
Finite state machinesPayment processing

Component Characteristics

StateInterface

  • Defines available actions
  • Returns new state after transition
  • Encapsulates state-specific behavior

Context

  • Holds current state
  • Delegates actions to state
  • Manages state transitions

Concrete States

  • Implement behavior for each state
  • Handle valid/invalid transitions
  • Return appropriate next state

Generation Process

Step 1: Analyze State Machine

Determine:

  • All possible states
  • Actions/transitions between states
  • Which actions are valid per state
  • Terminal states (no outgoing transitions)

Step 2: Generate State Components

Path: src/Domain/{BoundedContext}/State/

  1. {Name}StateInterface.php — State contract with all actions
  2. Abstract{Name}State.php — Base class with default (invalid) implementations
  3. {StateName}State.php — Concrete state for each state (PendingState, PaidState, etc.)
  4. {Name}StateFactory.php — Factory for reconstitution from storage

Path: src/Domain/{BoundedContext}/Exception/

  1. InvalidStateTransitionException.php — Exception for invalid transitions

Step 3: Update Entity

Path: src/Domain/{BoundedContext}/Entity/

Update entity to use state pattern with delegation methods.

Step 4: Generate Tests

Path: tests/Unit/Domain/{BoundedContext}/State/

  1. {StateName}StateTest.php — Test each concrete state
  2. {Entity}StateTransitionTest.php — Test full state machine

File Placement

ComponentPath
State Interfacesrc/Domain/{BoundedContext}/State/
Concrete Statessrc/Domain/{BoundedContext}/State/
State Factorysrc/Domain/{BoundedContext}/State/
Entity with Statesrc/Domain/{BoundedContext}/Entity/
Exceptionsrc/Domain/{BoundedContext}/Exception/
Unit Teststests/Unit/Domain/{BoundedContext}/State/

Naming Conventions

ComponentPatternExample
Interface{Name}StateInterfaceOrderStateInterface
Abstract StateAbstract{Name}StateAbstractOrderState
Concrete State{StateName}StatePendingState, PaidState
Factory{Name}StateFactoryOrderStateFactory
ExceptionInvalidStateTransitionExceptionInvalidStateTransitionException
Test{ClassName}TestPendingStateTest

Quick Template Reference

State Interface

interface {Name}StateInterface
{
    public function getName(): string;
    public function {action1}({Context} $context): self;
    public function {action2}({Context} $context): self;
    public function canTransitionTo(self $state): bool;
    /** @return array<string> */
    public function allowedTransitions(): array;
}

Abstract State

abstract readonly class Abstract{Name}State implements {Name}StateInterface
{
    public function {action1}({Context} $context): {Name}StateInterface
    {
        throw InvalidStateTransitionException::actionNotAllowed('{action1}', $this->getName());
    }

    public function canTransitionTo({Name}StateInterface $state): bool
    {
        return in_array($state->getName(), $this->allowedTransitions(), true);
    }
}

Concrete State

final readonly class {StateName}State extends Abstract{Name}State
{
    public function getName(): string
    {
        return '{state_name}';
    }

    public function {action1}({Context} $context): {Name}StateInterface
    {
        $context->recordEvent(new {Event}($context->id()));
        return new {NextState}State();
    }

    public function allowedTransitions(): array
    {
        return ['{next_state_1}', '{next_state_2}'];
    }
}

State Factory

final class {Name}StateFactory
{
    public static function fromName(string $name): {Name}StateInterface
    {
        return match ($name) {
            'pending' => new PendingState(),
            'confirmed' => new ConfirmedState(),
            default => throw new \InvalidArgumentException("Unknown state: $name"),
        };
    }
}

Usage Example

// Entity with state
$order = new Order($id, $customerId, $items);
$order->confirm();  // Pending -> Confirmed
$order->pay();      // Confirmed -> Paid
$order->ship();     // Paid -> Shipped
$order->deliver();  // Shipped -> Delivered

// Check state
if ($order->isInState('delivered')) {
    // ...
}

// Reconstitute from storage
$state = OrderStateFactory::fromName($row['state']);
$order = new Order($id, $customerId, $items, $state);

State Diagram Template

┌──────────┐  action1   ┌──────────┐  action2   ┌──────────┐
│  State1  │───────────▶│  State2  │───────────▶│  State3  │
└────┬─────┘            └────┬─────┘            └──────────┘
     │                       │
     │ cancel                │ cancel
     │                       │
     ▼                       ▼
┌──────────┐            ┌──────────┐
│Cancelled │            │Cancelled │
└──────────┘            └──────────┘

Anti-patterns to Avoid

Anti-patternProblemSolution
Mutable StatesShared state pollutionMake states readonly
God StateOne state handles allSplit into specific states
Missing TransitionsSilent failuresThrow on invalid action
State in EntityMixed concernsExtract to State classes
No FactoryHard to reconstituteAdd StateFactory

References

For complete PHP templates and examples, see:

  • references/templates.md — State interface, abstract, concrete templates
  • references/examples.md — Order state machine example and tests

Install

Download ZIP
Requires askill CLI v1.0+

AI Quality Score

95/100Analyzed 2/11/2026

Comprehensive and highly actionable guide for implementing the State pattern in PHP 8.5, featuring clear steps, templates, and best practices.

95
100
90
98
95

Metadata

Licenseunknown
Version-
Updated2/5/2026
Publishermajiayu000

Tags

github-actionstesting