Let’s check the definition made by the author of the concept:
When a significant process or transformation in the domain is not a natural responsibility of an ENTITY or VALUE OBJECT, add an operation to the model as standalone interface declared as a SERVICE. Define the interface in terms of the language of the model and make sure the operation name is part of the UBIQUITOUS LANGUAGE. Make the SERVICE stateless.
In plain English it means that it’t nothing more then a procedure. Why? Because you take an object, operate upon its data and pass it to another object. It contradicts to basic principles of OOP, where data and behavior reside together. So in my object world everything is some object’s responsibility.Typical example of procedural code that involves a domain service is password hashing.
Every time you need some new behavior, you should consider using a decorator pattern. So that’s how the example above can be re-written:
try {echo (new HashedPassword(new RegexMatchedPassword(new NonEmptyPassword(new Password('asdSd12@A1')),new Regex('/^(?=.*[a-z])(?=.*[A-Z])(?=.*\d)(?=.*[$@$!%*?&])[A-Za-z\d$@$!%*?&]{8,10}$/'))))->string();} catch (Exception $exception) {echo $exception->getMessage();}
All attention to ‘HashedPassword’ class. All hashing functionality resides there. I like this name — hashed password. It’s very declarative, unlike “hash password”, which is more like “how”, than “what”, thus being inherently imperative.
Besides moving domain service’s functionality to DDD’s value object, I employed validation decorators, the concept introduced by Yegor Bugayenko.
So here is the rest of the code:
interface IPassword{public function string();}
class Password implements IPassword{private $password;
**public function** \_\_construct($password)
{
$this->**password** \= $password;
}
**public function** string()
{
**return** $this->**password**;
}
}
class NonEmptyPassword implements IPassword{private $password;
**public function** \_\_construct(IPassword $password)
{
$this->**password** \= $password;
}
**public function** string()
{
**if** (_is\_null_($this->**password**\->string())) {
**throw new** Exception(**'Password can not be empty'**);
}
**return** $this->**password**\->string();
}
}
class RegexMatchedPassword implements IPassword{private $password;private $regex;
**public function** \_\_construct(IPassword $password, IRegex $regex)
{
$this->**password** \= $password;
$this->**regex** \= $regex;
}
**public function** string()
{
_// https://stackoverflow.com/a/21456918/618020_ **if** (!_preg\_match_($this->**regex**\->value(), $this->**password**\->string())) {
**throw new** Exception(
**'Password must have minimum eight and maximum 10 characters, at least one uppercase letter,
one lowercase letter, one number and one special character.'** );
}
**return** $this->**password**\->string();
}
}
interface IRegex{public function value();}
class Regex implements IRegex{private $regex;
**public function** \_\_construct($regex)
{
$this->**regex** \= $regex;
}
**public function** value()
{
**if** (_preg\_match_($this->**regex**, **null**) === **false**) {
**throw new** Exception(**'Your regex is broken.'**);
}
**return** $this->**regex**;
}
}
class HashedPassword implements IPassword{private $password;
**public function** \_\_construct(IPassword $password)
{
$this->**password** \= $password;
}
**public function** string()
{
**return** _sha1_($this->**password**\->string());
}
}
So if objects are designed properly, there are no services. There are just normal objects, representing the real things. So following this heuristics, I claim that Application services are a mistake too.
Stay tuned.