PHP hook, building hooks in your application
Introduction
One of the real challenges in building any type of framework, core or application is making it possible for the developers to hook into the business logic at specific points. Since PHP is not event based, nor it works with interrupts you have to come up an alternative.
The test case
Lets assume we are the main developers of a webshop framework. Programmers can use our framework to build complete webshops. Programmers can manage the orders that are placed on the webshop with the order class. The order class is part of our framework and we don’t want it to be extended by any programmer. However we don’t want to limit to programmers in their possibilities to hook into the orders process.
For example programmers should be able to send an email to the webshopowner if an order changes from one specific delivery status to another. This functionality is not part of the default behavior in our framework and is custom for the progammers webshop implementation.
Like said before, PHP doesn’t provide interrupts or real events so we need to come up with another way to implement hooks into our application. Lets take a look at the observer pattern.
Implementing the Observer pattern
The observer pattern is a design-pattern that describes a way for objects to be notified to specific state-changes in objects of the application.
For the first implementation we can use SPL. The SPL provides in two simple objects:
SPLSubject
- attach (new observer to attach)
- detach (existing observer to detach)
- notify (notify all observers)
SPLObserver
- update (Called from the subject (i.e. when it’s value has changed).
iOrderRef = $iOrderRef;
// Get order information from the database or an other resources
$this->iStatus = Order::STATUS_SHIPPED;
}
/**
* Attach an observer
*
* @param SplObserver $oObserver
* @return void
*/
public function attach(SplObserver $oObserver)
{
$sHash = spl_object_hash($oObserver);
if (isset($this->aObservers[$sHash])) {
throw new Exception('Observer is already attached');
}
$this->aObservers[$sHash] = $oObserver;
}
/**
* Detach observer
*
* @param SplObserver $oObserver
* @return void
*/
public function detach(SplObserver $oObserver)
{
$sHash = spl_object_hash($oObserver);
if (!isset($this->aObservers[$sHash])) {
throw new Exception('Observer not attached');
}
unset($this->aObservers[$sHash]);
}
/**
* Notify the attached observers
*
* @param string $sEvent, name of the event
* @param mixed $mData, optional data that is not directly available for the observers
* @return void
*/
public function notify()
{
foreach ($this->aObservers as $oObserver) {
try {
$oObserver->update($this);
} catch(Exception $e) {
}
}
}
/**
* Add an order
*
* @param array $aOrder
* @return void
*/
public function delete()
{
$this->notify();
}
/**
* Return the order reference number
*
* @return int
*/
public function getRef()
{
return $this->iOrderRef;
}
/**
* Return the current order status
*
* @return int
*/
public function getStatus()
{
return $this->iStatus;
}
/**
* Update the order status
*/
public function updateStatus($iStatus)
{
$this->notify();
// ...
$this->iStatus = $iStatus;
// ...
$this->notify();
}
}
/**
* Order status handler, observer that sends an email to secretary
* if the status of an order changes from shipped to delivered, so the
* secratary can make a phone call to our customer to ask for his opinion about the service
*
* @package Shop
*/
class OrderStatusHandler implements SplObserver
{
/**
* Previous orderstatus
* @var int
*/
protected $iPreviousOrderStatus;
/**
* Current orderstatus
* @var int
*/
protected $iCurrentOrderStatus;
/**
* Update, called by the observable object order
*
* @param Observable_Interface $oSubject
* @param string $sEvent
* @param mixed $mData
* @return void
*/
public function update(SplSubject $oSubject)
{
if(!$oSubject instanceof Order) {
return;
}
if(is_null($this->iPreviousOrderStatus)) {
$this->iPreviousOrderStatus = $oSubject->getStatus();
} else {
$this->iCurrentOrderStatus = $oSubject->getStatus();
if($this->iPreviousOrderStatus === Order::STATUS_SHIPPED && $this->iCurrentOrderStatus === Order::STATUS_DELIVERED) {
$sSubject = sprintf('Order number %d is shipped', $oSubject->getRef());
//mail('secratary@example.com', 'Order number %d is shipped', 'Text');
echo 'Mail sended to the secratary to help her remember to call our customer for a survey.';
}
}
}
}
$oOrder = new Order(26012011);
$oOrder->attach(new OrderStatusHandler());
$oOrder->updateStatus(Order::STATUS_DELIVERED);
$oOrder->delete();
?>
There are several problems with the implementation above. To most important disadvantage is that we have only one update method in our observer. In this update method we don’t know when and why we are getting notified, just that something happened. We should keep track of everything that happens in the subject. (Or use debug_backtrace… just joking, don’t even think about using it that way ever!).
Taking it a step further, events
Lets take a look at the next example, we will extend the Observer implementation with some an additional parameter for the eventname that occured.
Finishing up, optional data
iOrderRef = $iOrderRef;
// Get order information from the database or something else...
$this->iStatus = Order::STATUS_SHIPPED;
}
/**
* Attach an observer
*
* @param Observer_Interface $oObserver
* @return void
*/
public function attachObserver(Observer_Interface $oObserver)
{
$sHash = spl_object_hash($oObserver);
if (isset($this->aObservers[$sHash])) {
throw new Exception('Observer is already attached');
}
$this->aObservers[$sHash] = $oObserver;
}
/**
* Detach observer
*
* @param Observer_Interface $oObserver
* @return void
*/
public function detachObserver(Observer_Interface $oObserver)
{
$sHash = spl_object_hash($oObserver);
if (!isset($this->aObservers[$sHash])) {
throw new Exception('Observer not attached');
}
unset($this->aObservers[$sHash]);
}
/**
* Notify the attached observers
*
* @param string $sEvent, name of the event
* @param mixed $mData, optional data that is not directly available for the observers
* @return void
*/
public function notifyObserver($sEvent, $mData=null)
{
foreach ($this->aObservers as $oObserver) {
try {
$oObserver->update($this, $sEvent, $mData);
} catch(Exception $e) {
}
}
}
/**
* Add an order
*
* @param array $aOrder
* @return void
*/
public function add($aOrder = array())
{
$this->notifyObserver('onAdd');
}
/**
* Return the order reference number
*
* @return int
*/
public function getRef()
{
return $this->iOrderRef;
}
/**
* Return the current order status
*
* @return int
*/
public function getStatus()
{
return $this->iStatus;
}
/**
* Update the order status
*/
public function updateStatus($iStatus)
{
$this->notifyObserver('onBeforeUpdateStatus');
// ...
$this->iStatus = $iStatus;
// ...
$this->notifyObserver('onAfterUpdateStatus');
}
}
/**
* Order status handler, observer that sends an email to secretary
* if the status of an order changes from shipped to delivered, so the
* secratary can make a phone call to our customer to ask for his opinion about the service
*
* @package Shop
*/
class OrderStatusHandler implements Observer_Interface
{
protected $iPreviousOrderStatus;
protected $iCurrentOrderStatus;
/**
* Update, called by the observable object order
*
* @param Observable_Interface $oObservable
* @param string $sEvent
* @param mixed $mData
* @return void
*/
public function update(Observable_Interface $oObservable, $sEvent, $mData=null)
{
if(!$oObservable instanceof Order) {
return;
}
switch($sEvent) {
case 'onBeforeUpdateStatus':
$this->iPreviousOrderStatus = $oObservable->getStatus();
return;
case 'onAfterUpdateStatus':
$this->iCurrentOrderStatus = $oObservable->getStatus();
if($this->iPreviousOrderStatus === Order::STATUS_SHIPPED && $this->iCurrentOrderStatus === Order::STATUS_DELIVERED) {
$sSubject = sprintf('Order number %d is shipped', $oObservable->getRef());
//mail('secratary@example.com', 'Order number %d is shipped', 'Text');
echo 'Mail sended to the secratary to help her remember to call our customer for a survey.';
}
}
}
}
$oOrder = new Order(26012011);
$oOrder->attachObserver(new OrderStatusHandler());
$oOrder->updateStatus(Order::STATUS_DELIVERED);
$oOrder->add();
?>
Now we are able to take action on different events that occur.
Disadvantages
Although this implementation works quite well there are some drawbacks. One of those drawbacks is that we need to dispatch an event in our framework, if we don’t programmers can’t hook into our application. Triggering events everywhere give us a small performance penalty however I do think this way of working gives the programmers a nice way to hook into your application on those spots that you want them to hook in.
Just for the record
Notice that this code is just an example and can still use some improvements, for example: each observer is initialized even it will maybe never be notified, therefore I suggest to make use of lazy in some cases for loading the objects. There are other systems to hook into an application, more to follow!
купить диплом в горно-алтайске [url=rudik-diplom3.ru]rudik-diplom3.ru[/url] .
Diplomi_uiei
19 Oct 25 at 3:54 am
купить диплом педагога [url=rudik-diplom8.ru]купить диплом педагога[/url] .
Diplomi_vqMt
19 Oct 25 at 3:54 am
купить проведенный диплом спб [url=https://frei-diplom3.ru]https://frei-diplom3.ru[/url] .
Diplomi_ctKt
19 Oct 25 at 3:55 am
купить диплом в дербенте [url=www.rudik-diplom5.ru]купить диплом в дербенте[/url] .
Diplomi_xhma
19 Oct 25 at 3:55 am
купить диплом в нефтеюганске [url=https://rudik-diplom1.ru/]купить диплом в нефтеюганске[/url] .
Diplomi_pfer
19 Oct 25 at 3:55 am
точные прогнозы на футбол сегодня бесплатно [url=https://www.prognozy-na-futbol-10.ru]точные прогнозы на футбол сегодня бесплатно[/url] .
prognozi na fytbol_yzOi
19 Oct 25 at 3:56 am
http://medtronik.ru/ сайт, где собраны инструкции по получению фрибетов и бонусов
Aaronawads
19 Oct 25 at 3:58 am
https://bs2site3.io
Hermannalia
19 Oct 25 at 3:59 am
купить диплом в ейске [url=www.rudik-diplom13.ru]www.rudik-diplom13.ru[/url] .
Diplomi_zkon
19 Oct 25 at 3:59 am
купить диплом в каменске-шахтинском [url=rudik-diplom11.ru]rudik-diplom11.ru[/url] .
Diplomi_mjMi
19 Oct 25 at 4:00 am
cuttingthered – Sharp branding, message delivers confidence and determination effectively.
MichaelLar
19 Oct 25 at 4:00 am
Howdy! Do you know if they make any plugins to protect against hackers?
I’m kinda paranoid about losing everything I’ve worked hard on. Any tips?
Locksmith company
19 Oct 25 at 4:00 am
купить диплом электромонтера [url=rudik-diplom4.ru]купить диплом электромонтера[/url] .
Diplomi_iqOr
19 Oct 25 at 4:01 am
согласование перепланировки квартиры в москве цена [url=www.proekt-pereplanirovki-kvartiry17.ru]www.proekt-pereplanirovki-kvartiry17.ru[/url] .
proekt pereplanirovki kvartiri_lmml
19 Oct 25 at 4:01 am
согласованте [url=http://soglasovanie-pereplanirovki-kvartiry3.ru]http://soglasovanie-pereplanirovki-kvartiry3.ru[/url] .
soglasovanie pereplanirovki kvartiri _djPi
19 Oct 25 at 4:03 am
медсестра которая купила диплом врача [url=https://frei-diplom13.ru]медсестра которая купила диплом врача[/url] .
Diplomi_cakt
19 Oct 25 at 4:03 am
купить технический диплом [url=https://www.rudik-diplom1.ru]купить технический диплом[/url] .
Diplomi_tver
19 Oct 25 at 4:03 am
В Краснодаре клиника «Детокс» предоставляет услугу вызова нарколога на дом. Специалисты приедут к вам в течение 1–2 часов, проведут осмотр и назначат необходимое лечение. Все процедуры проводятся анонимно и с соблюдением конфиденциальности.
Ознакомиться с деталями – [url=https://narkolog-na-dom-krasnodar26.ru/]нарколог на дом вывод[/url]
DanielCaupe
19 Oct 25 at 4:04 am
купить диплом в йошкар-оле [url=https://www.rudik-diplom10.ru]купить диплом в йошкар-оле[/url] .
Diplomi_jmSa
19 Oct 25 at 4:04 am
купить проведенный диплом всеми [url=www.frei-diplom6.ru]купить проведенный диплом всеми[/url] .
Diplomi_hcOl
19 Oct 25 at 4:04 am
кракен даркнет маркет
кракен qr код
JamesDaync
19 Oct 25 at 4:05 am
купить диплом в гуково [url=https://rudik-diplom11.ru/]купить диплом в гуково[/url] .
Diplomi_nqMi
19 Oct 25 at 4:07 am
купить аттестат [url=https://rudik-diplom4.ru]купить аттестат[/url] .
Diplomi_fzOr
19 Oct 25 at 4:08 am
перепланировка [url=www.soglasovanie-pereplanirovki-kvartiry4.ru]перепланировка[/url] .
soglasovanie pereplanirovki kvartiri _fdOr
19 Oct 25 at 4:09 am
согласование перепланировки помещений [url=www.soglasovanie-pereplanirovki-kvartiry14.ru/]www.soglasovanie-pereplanirovki-kvartiry14.ru/[/url] .
soglasovanie pereplanirovki kvartiri _deEl
19 Oct 25 at 4:09 am
Thank you for any other great post. Where else
may just anyone get that kind of info in such a perfect way of
writing? I’ve a presentation subsequent week, and I’m on the look for such info.
شركة تنظيم حفلات في الرياض
19 Oct 25 at 4:10 am
бесплатные прогнозы на хоккей [url=https://prognozy-na-khokkej5.ru/]бесплатные прогнозы на хоккей[/url] .
prognozi na hokkei_ilEa
19 Oct 25 at 4:11 am
I’ve been following the Minotaurus presale closely, and it’s impressive how they’ve structured the tokenomics for long-term sustainability. The vesting bonuses are a smart incentive that could really reward patient holders. Excited to see $MTAUR launch and disrupt the blockchain gaming space.
mtaur token
WilliamPargy
19 Oct 25 at 4:11 am
купить диплом учителя физической культуры [url=https://rudik-diplom5.ru]купить диплом учителя физической культуры[/url] .
Diplomi_gima
19 Oct 25 at 4:12 am
купить диплом в череповце [url=rudik-diplom13.ru]rudik-diplom13.ru[/url] .
Diplomi_bmon
19 Oct 25 at 4:12 am
купить диплом в владивостоке [url=rudik-diplom1.ru]купить диплом в владивостоке[/url] .
Diplomi_boer
19 Oct 25 at 4:12 am
купить диплом в смоленске [url=https://rudik-diplom3.ru]https://rudik-diplom3.ru[/url] .
Diplomi_drei
19 Oct 25 at 4:13 am
купить диплом в новом уренгое [url=http://www.rudik-diplom4.ru]http://www.rudik-diplom4.ru[/url] .
Diplomi_rbOr
19 Oct 25 at 4:13 am
купить диплом о техническом образовании с занесением в реестр [url=frei-diplom6.ru]frei-diplom6.ru[/url] .
Diplomi_txOl
19 Oct 25 at 4:15 am
кто нибудь работает медсестрой по купленному диплому [url=https://frei-diplom13.ru/]https://frei-diplom13.ru/[/url] .
Diplomi_cckt
19 Oct 25 at 4:15 am
купить диплом украина с занесением в реестр [url=http://www.frei-diplom5.ru]http://www.frei-diplom5.ru[/url] .
Diplomi_aePa
19 Oct 25 at 4:17 am
купить диплом в рязани [url=https://www.rudik-diplom11.ru]купить диплом в рязани[/url] .
Diplomi_xcMi
19 Oct 25 at 4:18 am
купить диплом в лениногорске [url=https://www.rudik-diplom13.ru]https://www.rudik-diplom13.ru[/url] .
Diplomi_uvon
19 Oct 25 at 4:19 am
перепланировка помещения [url=www.soglasovanie-pereplanirovki-kvartiry3.ru]www.soglasovanie-pereplanirovki-kvartiry3.ru[/url] .
soglasovanie pereplanirovki kvartiri _jsPi
19 Oct 25 at 4:19 am
сколько стоит перепланировка в москве [url=https://stoimost-soglasovaniya-pereplanirovki-kvartiry.ru/]stoimost-soglasovaniya-pereplanirovki-kvartiry.ru[/url] .
stoimost soglasovaniya pereplanirovki kvartiri_cmPt
19 Oct 25 at 4:21 am
купить диплом судоводителя [url=https://www.rudik-diplom10.ru]купить диплом судоводителя[/url] .
Diplomi_znSa
19 Oct 25 at 4:21 am
Unquestionably consider that that you stated.
Your favorite reason seemed to be at the web the simplest thing to consider of.
I say to you, I certainly get irked whilst other folks think
about issues that they plainly do not recognise about.
You controlled to hit the nail upon the top
as neatly as defined out the entire thing without having side
effect , other folks could take a signal. Will likely be again to get more.
Thank you
Also visit my web site; สมัครล็อตโต้วีไอพี
สมัครล็อตโต้วีไอพี
19 Oct 25 at 4:21 am
экспресс на футбол сегодня [url=www.prognozy-na-futbol-10.ru]www.prognozy-na-futbol-10.ru[/url] .
prognozi na fytbol_gpOi
19 Oct 25 at 4:22 am
купить диплом о высшем образовании с занесением в реестр в калуге [url=http://frei-diplom6.ru]купить диплом о высшем образовании с занесением в реестр в калуге[/url] .
Diplomi_lrOl
19 Oct 25 at 4:22 am
купить технический диплом [url=http://www.rudik-diplom8.ru]купить технический диплом[/url] .
Diplomi_taMt
19 Oct 25 at 4:22 am
купить диплом дизайнера [url=http://www.rudik-diplom5.ru]купить диплом дизайнера[/url] .
Diplomi_usma
19 Oct 25 at 4:22 am
согласованте [url=http://soglasovanie-pereplanirovki-kvartiry4.ru]http://soglasovanie-pereplanirovki-kvartiry4.ru[/url] .
soglasovanie pereplanirovki kvartiri _bvOr
19 Oct 25 at 4:23 am
диплом купить с внесением в реестр [url=https://www.frei-diplom5.ru]диплом купить с внесением в реестр[/url] .
Diplomi_dxPa
19 Oct 25 at 4:24 am
бесплатные прогнозы на хоккей [url=https://prognozy-na-khokkej5.ru/]бесплатные прогнозы на хоккей[/url] .
prognozi na hokkei_qrEa
19 Oct 25 at 4:24 am
kraken ссылка
kraken зеркало
JamesDaync
19 Oct 25 at 4:26 am