2017 © Pedro Peláez
 

library zf2-command-module

Command object pattern implementation

image

thefrozenfire/zf2-command-module

Command object pattern implementation

  • Thursday, April 30, 2015
  • by TheFrozenFire
  • Repository
  • 0 Watchers
  • 0 Stars
  • 63 Installations
  • PHP
  • 0 Dependents
  • 0 Suggesters
  • 0 Forks
  • 0 Open issues
  • 4 Versions
  • 3 % Grown

The README.md

Command Objects

This module provides a basic implementation of the command object pattern in ZF2, as well as aggregate command objects. Command objects are typically used in place of stringing together the results from various unrelated service methods in an application service, though nothing prevents you from calling application services from command objects as well. In an architecture which isn't service-oriented, you might instead think of command objects as an encapsulation of the logic in your controller actions, in a portable contract., (*1)

The primary purpose of command objects is to encapsulate the dependencies and parameters of a discrete "command" which can be invoked at the application level., (*2)

Basic Usage

Command Object

<?php
$addToFieldOnFoo = $serviceLocator->get('Application\Command\AddToFieldOnFoo');
$addToFieldOnFoo->setFooId(1);
$addToFieldOnFoo->setFieldToAddTo('bar')
    ->setAmountToAdd(5);

$changedFoo = $addToFieldOnFoo->execute();
<?php
namespace Foo;

use Command\AbstractCommand;
use Application\Service;

class AddToFieldOnFoo extends AbstractCommand
{
    protected $fooService;

    protected $fooId;

    protected $fieldToAddTo;

    protected $amountToAdd;

    public function __construct(Service\FooService $fooService)
    {
        $this->setFooService($fooService);
    }

    public function execute()
    {
        $fooService = $this->getFooService();

        $fooId = $this->getFooId();
        $fieldToAddTo = $this->getFieldToAddTo();
        $amountToAdd = $this->getAmountToAdd();

        $foo = $fooService->findById($fooId);

        $fieldValue = $foo->getField($fieldToAddTo);
        $fieldValue += $amountToAdd;

        $foo->setField($fieldToAddTo, $fieldValue);

        $fooService->persist($foo);

        return $foo;
    }

    public function getFooService()
    {
        return $this->fooService;
    }

    public function setFooService(Service\FooService $fooService)
    {
        $this->fooService = $fooService;
        return $this;
    }

    public function getFooId()
    {
        return $this->fooId;
    }

    public function setFooId($fooId)
    {
        $this->fooId = $fooId;
        return $this;
    }

    public function getFieldToAddTo()
    {
        return $this->fieldToAddTo;
    }

    public function setFieldToAddTo($fieldToAddTo)
    {
        $this->fieldToAddTo = $fieldToAddTo;
        return $this;
    }

    public function getAmountToAdd()
    {
        return $this->amountToAdd;
    }

    public function setAmountToAdd($amountToAdd)
    {
        $this->amountToAdd = $amountToAdd;
        return $this;
    }

}

Aggregate Command Object

$addToFieldOnFoo = $serviceLocator->get('Application\Command\AddToFieldOnFoo');
$addToFieldOnFoo->getFindFoo()
    ->setFooId(1);
$addToFieldOnFoo->getAddValueToFooField()
    ->setFieldToAddTo('bar')
    ->setAmountToAdd(5);

$addFieldOnFoo->getEventManager()->attach($addFieldOnFoo::guessEventName($addFieldOnFoo->getFindFoo()),
    function($event) {
        echo "Found Foo with name {$event->getResult()->getName()}".PHP_EOL;
    }, 100);

$addFieldOnFoo->getEventManager()->attach($addFieldOnFoo::guessEventName($addFieldOnFoo->getAddValueToFooField()),
    function($event) {
        $name = $event->getResult()->getName();
        $fieldToAddTo = $event->getTarget()->getFieldToAddTo();
        $newValue = $event->getResult()->getField($fieldToAddTo);
        echo "Foo with name {$name} now has '{$fieldToAddTo}' with value {$newValue}".PHP_EOL;
    }, 100);

$addFieldOnFoo->getEventManager()->attach($addFieldOnFoo::guessEventName($addFieldOnFoo->getPersistFoo()),
    function($event) {
        $name = $event->getResult()->getName();
        echo "Foo with name {$name} has been persisted".PHP_EOL;
    }, 100);

$changedFoo = $addToFieldOnFoo->execute();
<?php
namespace Foo;

use Command\AbstractAggregateCommand;
use Command\CommandEvent;
use Application\Command;

class AddToFieldOnFoo extends AbstractAggregateCommand
{
    protected $findFoo;

    protected $addValueToFooField;

    protected $saveFoo;

    public function __construct(Command\FindFoo $findFoo, Command\AddValueToFooField $addValueToFooField, Command\PersistFoo $persistFoo)
    {
        $this->setFindFoo($findFoo);
        $this->setAddValueToFooField($addValueToFooField);
        $this->setPersistFoo($persistFoo);
    }

    public function execute()
    {
        $findFoo = $this->getFindFoo();
        $addValueToFooField = $this->getAddValueToFooField();
        $persistFoo = $this->getPersistFoo();

        $foo = $this->executeChild($findFoo);

        $addValueToFooField->setFoo($foo);
        $foo = $this->executeChild($addValueToFooField);

        $persistFoo->setFoo($foo);
        $foo = $this->executeChild($persistFoo);

        return $foo;
    }

    public function getFindFoo()
    {
        return $this->findFoo;
    }

    public function setFindFoo($findFoo)
    {
        $this->findFoo = $findFoo;
        return $this;
    }

    public function getAddValueToFooField()
    {
        return $this->addValueToFooField;
    }

    public function setAddValueToFooField($addValueToFooField)
    {
        $this->addValueToFooField = $addValueToFooField;
        return $this;
    }

    public function getPeristFoo()
    {
        return $this->peristFoo;
    }

    public function setPeristFoo($peristFoo)
    {
        $this->peristFoo = $peristFoo;
        return $this;
    }

}

Aggregate Command Object with Event Stringing

<?php
namespace Foo;

use Command\AbstractAggregateCommand;
use Command\CommandEvent;
use Application\Command;

class AddToFieldOnFoo extends AbstractAggregateCommand
{
    protected $findFoo;

    protected $addValueToFooField;

    protected $saveFoo;

    public function __construct(Command\FindFoo $findFoo, Command\AddValueToFooField $addValueToFooField, Command\PersistFoo $persistFoo)
    {
        $this->setFindFoo($findFoo);
        $this->setAddValueToFooField($addValueToFooField);
        $this->setPersistFoo($persistFoo);
    }

    public function execute()
    {
        $command = $this;
        $result = null;

        $findFoo = $this->getFindFoo();
        $addValueToFooField = $this->getAddValueToFooField();
        $persistFoo = $this->getPersistFoo();

        $command->getEventManager()->attach(static::guessEventName($findFoo), function(CommandEvent $event) use ($command, $addValueToFooField) {
            $foo = $event->getResult();
            $addValueToFooField->setFoo($foo);

            $command->executeChild($addValueToFooField);
        }, 10);

        $command->getEventManager()->attach(static::guessEventName($addValueToFooField), function(CommandEvent $event) use ($command, $persistFoo) {
            $foo = $event->getResult();
            $persistFoo->setFoo($foo);

            $command->executeChild($persistFoo);
        }, 10);

        $command->getEventManager()->attach(static::guessEventName($persistFoo), function(CommandEvent $event) use ($command, &$result) {
            $result = $event->getResult();
        }, 10);

        $this->executeChild($findFoo);

        return $foo;
    }

    public function getFindFoo()
    {
        return $this->findFoo;
    }

    public function setFindFoo($findFoo)
    {
        $this->findFoo = $findFoo;
        return $this;
    }

    public function getAddValueToFooField()
    {
        return $this->addValueToFooField;
    }

    public function setAddValueToFooField($addValueToFooField)
    {
        $this->addValueToFooField = $addValueToFooField;
        return $this;
    }

    public function getPersistFoo()
    {
        return $this->persistFoo;
    }

    public function setPersistFoo($persistFoo)
    {
        $this->persistFoo = $persistFoo;
        return $this;
    }

}

Refactoring Controller Methods

Without Command Object, (*3)

<?php
namespace Foo;

use Zend\Mvc\Controller\AbstractActionController;
use Application\Service;

class FooController extends AbstractActionController
{
    protected $fooService;

    public function __construct(Service\FooService $fooService) {
        $this->setFooService($fooService);
    }

    public function addValueToFieldOnFoo()
    {
        $fooService = $this->getFooService();

        $fooId = $this->params('foo_id');
        $fieldToAddTo = $this->params()->fromPost('field_to_add_to');
        $amountToAdd = $this->params()->fromPost('amount_to_add');

        $foo = $fooService->findById($fooId);

        $fieldValue = $foo->getField($fieldToAddTo);
        $fieldValue += $amountToAdd;

        $foo->setField($fieldToAddTo, $fieldValue);

        $fooService->persist($foo);
    }

    public function getFooService()
    {
        return $this->fooService;
    }

    public function setFooService($fooService)
    {
        $this->fooService = $fooService;
        return $this;
    }

}

With Command Object, (*4)

<?php
namespace Foo;

use Zend\Mvc\Controller\AbstractActionController;
use Application\Command;

class FooController extends AbstractActionController
{
    protected $addToFieldOnFoo;

    public function __construct(Command\AddToFieldOnFoo $addToFieldOnFoo) {
        $this->setAddToFieldOnFoo($addToFieldOnFoo);
    }

    public function addValueToFieldOnFoo()
    {
        $addToFieldOnFoo = $this->getAddToFieldOnFoo();

        $fooId = $this->params('foo_id');
        $fieldToAddTo = $this->params()->fromPost('field_to_add_to');
        $amountToAdd = $this->params()->fromPost('amount_to_add');

        $addToFieldOnFoo->getFindFoo()->getFooId($fooId);
        $addToFieldOnFoo->getAddValueToFooField()
            ->setFieldToAddTo('bar')
            ->setAmountToAdd(5);

        $addToFieldOnFoo->execute();
    }

    public function getAddToFieldOnFoo()
    {
        return $this->addToFieldOnFoo;
    }

    public function setAddToFieldOnFoo($addToFieldOnFoo)
    {
        $this->addToFieldOnFoo = $addToFieldOnFoo;
        return $this;
    }

}

The Versions

30/04 2015

dev-master

9999999-dev

Command object pattern implementation

  Sources   Download

MIT

The Development Requires

30/04 2015

1.0.2

1.0.2.0

Command object pattern implementation

  Sources   Download

MIT

The Development Requires

16/04 2015

1.0.1

1.0.1.0

Command object pattern implementation

  Sources   Download

MIT

The Development Requires

24/03 2015

1.0.0

1.0.0.0

Command object pattern implementation

  Sources   Download

MIT