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!
verapamil for sale
verapamil for sale
15 Oct 25 at 10:22 pm
сколько стоит аренда мини экскаватора [url=https://arenda-mini-ekskavatora-v-moskve.ru/]arenda-mini-ekskavatora-v-moskve.ru[/url] .
arenda mini ekskavatora v moskve_thEn
15 Oct 25 at 10:22 pm
потолочник натяжные потолки нижний новгород [url=http://stretch-ceilings-nizhniy-novgorod-1.ru/]http://stretch-ceilings-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_ztOn
15 Oct 25 at 10:24 pm
Мы работаем с триггерами, режимом сна/питания, энергобалансом и типичными стрессовыми ситуациями. Короткие и регулярные контрольные контакты снижают риск рецидива и помогают уверенно вернуться к работе и бытовым задачам.
Выяснить больше – http://narkologicheskaya-klinika-mytishchi0.ru/narkologicheskaya-klinika-stacionar-v-mytishchah/https://narkologicheskaya-klinika-mytishchi0.ru
WilliamVox
15 Oct 25 at 10:24 pm
натяжные потолки нижний новгород официальный сайт [url=www.natyazhnye-potolki-nizhniy-novgorod.ru]www.natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_ohOt
15 Oct 25 at 10:26 pm
Hello everyone, it’s my first visit at this website, and paragraph is in fact fruitful for me, keep up posting these content.
82200219
15 Oct 25 at 10:27 pm
I visited multiple blogs however the audio feature for audio songs present at this website is genuinely wonderful.
facer.io
15 Oct 25 at 10:27 pm
запись подкастов студия [url=http://www.studiya-podkastov-spb.ru]запись подкастов студия[/url] .
stydiya podkastov spb_txka
15 Oct 25 at 10:27 pm
натяжные потолки потолочкин [url=http://stretch-ceilings-nizhniy-novgorod.ru]http://stretch-ceilings-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_hlPl
15 Oct 25 at 10:28 pm
Ich liebe die Atmosphare von NV Casino, es erzeugt eine Energie, die suchtig macht. Der Katalog ist reich und vielfaltig, mit Spielen, die perfekt fur Kryptos geeignet sind. Der Support ist von herausragender Qualitat, mit praziser Unterstutzung. Die Auszahlungen sind ultraschnell, obwohl zusatzliche Freispiele waren toll. Alles in allem, NV Casino ist ein Muss fur Gamer fur Fans von Online-Wetten ! Zusatzlich die Navigation ist kinderleicht, gibt Lust auf mehr.
playnvcasino.de|
GigabitE6zef
15 Oct 25 at 10:29 pm
натяжной потолок нижний новгород [url=www.natyazhnye-potolki-nizhniy-novgorod.ru/]www.natyazhnye-potolki-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_ssOt
15 Oct 25 at 10:30 pm
натяжные потолки сайт [url=natyazhnye-potolki-nizhniy-novgorod-1.ru]натяжные потолки сайт[/url] .
natyajnie potolki nijnii novgorod_bxma
15 Oct 25 at 10:31 pm
buy cialis online: generic Cialis online pharmacy – buy cialis online
AndrewPal
15 Oct 25 at 10:32 pm
1win az bonusları [url=www.1win5004.com]www.1win5004.com[/url]
1win_zqoi
15 Oct 25 at 10:33 pm
https://t.me/rating_online/9
EverettGuemn
15 Oct 25 at 10:33 pm
микрозаймы онлайн [url=https://www.zaimy-28.ru]микрозаймы онлайн[/url] .
zaimi_ieKa
15 Oct 25 at 10:33 pm
https://t.me/s/rating_online/7
EverettGuemn
15 Oct 25 at 10:34 pm
blsp at Неужели ты до сих пор не знаешь, что переворачивает мир даркнета? Blacksprut – это не просто название. Это новая эра анонимности, мгновенной скорости и железобетонной безопасности в сети. Загляни на bs2best.at – туда, где обсуждают то, о чём остальные предпочитают молчать в тряпочку. Тебе откроется доступ к информации, тщательно оберегаемой от посторонних глаз. Только для тех, кто понимает правила игры. Никаких следов, никаких компромиссов, только абсолютная конфиденциальность – это Blacksprut. Не упусти свой шанс стать первооткрывателем – bs2best.at уже распахнул свои двери для тебя. Хватит ли у тебя смелости взглянуть правде прямо в лицо?
HermanRhype
15 Oct 25 at 10:36 pm
No, you can download videos of any length and select
your preferred quality, including HD, if available.
Https://socialplug.to/
15 Oct 25 at 10:36 pm
взо [url=https://zaimy-28.ru/]взо[/url] .
zaimi_ncKa
15 Oct 25 at 10:37 pm
как купить диплом техникума с занесением в реестр цена в [url=https://frei-diplom7.ru]как купить диплом техникума с занесением в реестр цена в[/url] .
Diplomi_unei
15 Oct 25 at 10:38 pm
Great post but I was wanting to know if you could write
a litte more on this subject? I’d be very grateful if you could elaborate a
little bit more. Thanks!
ako nastaviť SEO tak
15 Oct 25 at 10:39 pm
Thank you for the good writeup. It in fact was a amusement account it.
Look advanced to far added agreeable from you! However,
how can we communicate?
https://66b-app.com/
15 Oct 25 at 10:40 pm
сайт натяжных потолков [url=http://natyazhnye-potolki-nizhniy-novgorod.ru]http://natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_mxOt
15 Oct 25 at 10:40 pm
https://slovarikslov.ru/terem/xaus/
https://slovarikslov.ru/terem/xaus/
15 Oct 25 at 10:42 pm
Bullish on $MTAUR coin’s 9% sector growth tie-in. Token conversions boost in-game edge. Early perks unbeatable.
mtaur token
WilliamPargy
15 Oct 25 at 10:43 pm
тканевый натяжной потолок нижний новгород [url=www.stretch-ceilings-nizhniy-novgorod-1.ru]www.stretch-ceilings-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_nsOn
15 Oct 25 at 10:44 pm
аренда мини экскаватора в московской области цены [url=http://arenda-mini-ekskavatora-v-moskve.ru/]http://arenda-mini-ekskavatora-v-moskve.ru/[/url] .
arenda mini ekskavatora v moskve_urEn
15 Oct 25 at 10:46 pm
Its such as you read my mind! You seem to understand a lot about this, such as you wrote the e-book in it or something.
I think that you simply could do with a few % to pressure the message house
a little bit, but instead of that, that is excellent blog.
A fantastic read. I’ll definitely be back.
boobs
15 Oct 25 at 10:48 pm
натяжные потолки нижний новгород дешево [url=www.natyazhnye-potolki-nizhniy-novgorod.ru]www.natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_baOt
15 Oct 25 at 10:48 pm
аренда миниэкскаватора [url=https://arenda-mini-ekskavatora-v-moskve.ru/]аренда миниэкскаватора[/url] .
arenda mini ekskavatora v moskve_tbEn
15 Oct 25 at 10:49 pm
ZenCareMeds: buy amoxil – affordable online pharmacy for Americans
Andresstold
15 Oct 25 at 10:50 pm
why are some players looking for other options slot machines without the need for registration lure gamblers not for one reasons [url=https://desertadventurevip.com/best-non-gamstop-casinos-in-the-uk-a-comprehensive-9/]https://desertadventurevip.com/best-non-gamstop-casinos-in-the-uk-a-comprehensive-9/[/url] does not need registration to avoid self-exclusion.
TomZip
15 Oct 25 at 10:52 pm
I really like it when individuals come together and share thoughts.
Great site, continue the good work!
Centerton roofers
15 Oct 25 at 10:52 pm
Bullish on $MTAUR coin’s 9% sector growth tie-in. Token conversions boost in-game edge. Early perks unbeatable.
minotaurus ico
WilliamPargy
15 Oct 25 at 10:57 pm
1win poker otağı [url=https://1win5005.com]https://1win5005.com[/url]
1win_meml
15 Oct 25 at 10:57 pm
First off I want to say excellent blog! I had a quick question which I’d
like to ask if you do not mind. I was interested to find out how you center yourself and clear your thoughts before writing.
I have had trouble clearing my thoughts in getting my thoughts
out. I truly do take pleasure in writing but it just seems like
the first 10 to 15 minutes are generally wasted simply just trying to figure out
how to begin. Any ideas or tips? Appreciate it!
facer.io
15 Oct 25 at 10:57 pm
потолочкин ру нижний новгород [url=www.natyazhnye-potolki-nizhniy-novgorod-1.ru/]www.natyazhnye-potolki-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_jtma
15 Oct 25 at 10:58 pm
https://t.me/rating_online/7
EverettGuemn
15 Oct 25 at 10:59 pm
https://aishasicx979372.activoblog.com/44759433/mysteries-of-the-sandswept-tombs
WarrenBug
15 Oct 25 at 11:00 pm
потолочки [url=http://www.natyazhnye-potolki-nizhniy-novgorod.ru]http://www.natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_tjOt
15 Oct 25 at 11:01 pm
Great info. Lucky me I recently found your blog by
chance (stumbleupon). I’ve bookmarked it for later!
which states is rainbet legal
15 Oct 25 at 11:01 pm
все займы онлайн [url=https://zaimy-28.ru]все займы онлайн[/url] .
zaimi_wtKa
15 Oct 25 at 11:03 pm
https://t.me/rating_online
EverettGuemn
15 Oct 25 at 11:04 pm
What’s up, every time i used to check blog posts here early in the break of day, for the reason that i enjoy to gain knowledge of more
and more.
poolfolie
15 Oct 25 at 11:05 pm
Touche. Great arguments. Keep up the amazing effort.
warna mingguan sydney
15 Oct 25 at 11:06 pm
Портал о строительстве https://pamel-stroy.ru и ремонте. Пошаговые инструкции, идеи, технологии, новости и советы экспертов. Всё, что нужно, чтобы строить и ремонтировать грамотно.
Andrewdeefs
15 Oct 25 at 11:07 pm
With $MTAUR coin, collecting in-game currency while running mazes is addictive. Presale bonuses for vesting make holding appealing. Team’s track record impresses me.
minotaurus coin
WilliamPargy
15 Oct 25 at 11:07 pm
натяжные потолки нижний новгород с установкой [url=www.stretch-ceilings-nizhniy-novgorod.ru]www.stretch-ceilings-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_adPl
15 Oct 25 at 11:08 pm
Hey! I just wanted to ask if you ever have any trouble with
hackers? My last blog (wordpress) was hacked and I ended
up losing many months of hard work due to no back up.
Do you have any solutions to protect against hackers?
Flod Fundline Anmeldelse
15 Oct 25 at 11:08 pm