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=https://zakazat-onlayn-translyaciyu.ru/]https://zakazat-onlayn-translyaciyu.ru/[/url] .
zakazat onlain translyaciu _jzka
15 Oct 25 at 8:50 pm
— Растворы с глюкозой и витаминами для коррекции энергетического обмена. — Минеральные комплексы для нормализации электролитов. — Антиоксиданты и гепатопротекторы для защиты печени. — Спазмолитики для снятия болевого синдрома. — Противорвотные препараты при тошноте.
Подробнее тут – [url=https://narkologicheskaya-klinika-arkhangelsk0.ru/]наркологическая клиника цены[/url]
Craigfek
15 Oct 25 at 8:50 pm
1win ios yukle [url=1win5005.com]1win5005.com[/url]
1win_dzml
15 Oct 25 at 8:51 pm
Ich bin total begeistert von Lowen Play Casino, es ist ein Online-Casino, das wie ein Lowe brullt. Die Casino-Optionen sind vielfaltig und kraftvoll, mit modernen Casino-Slots, die einen in ihren Bann ziehen. Der Casino-Support ist rund um die Uhr verfugbar, ist per Chat oder E-Mail erreichbar. Der Casino-Prozess ist klar und ohne Fallen, aber mehr Casino-Belohnungen waren ein koniglicher Gewinn. Alles in allem ist Lowen Play Casino ein Casino, das man nicht verpassen darf fur Fans moderner Casino-Slots! Nebenbei die Casino-Seite ist ein grafisches Meisterwerk, was jede Casino-Session noch wilder macht.
lГ¶wen play pforzheim|
zappysquirrel3zef
15 Oct 25 at 8:52 pm
Acho simplesmente insano JabiBet Casino, tem uma vibe de jogo que e puro tsunami. A gama do cassino e simplesmente um maremoto, com caca-niqueis de cassino modernos e envolventes. O atendimento ao cliente do cassino e uma mare de qualidade, dando solucoes na hora e com precisao. Os ganhos do cassino chegam voando como uma onda, as vezes mais bonus regulares no cassino seria top. Em resumo, JabiBet Casino garante uma diversao de cassino que e uma mare cheia para os aventureiros do cassino! De bonus a interface do cassino e fluida e cheia de energia oceanica, aumenta a imersao no cassino como uma onda gigante.
jabibet|
zippyoctopus4zef
15 Oct 25 at 8:53 pm
кухни на заказ производство спб [url=http://kuhni-spb-2.ru]http://kuhni-spb-2.ru[/url] .
kyhni spb_iqmn
15 Oct 25 at 8:55 pm
https://slovarikslov.ru/terem/xaus/
https://slovarikslov.ru/terem/xaus/
15 Oct 25 at 8:55 pm
онлайн трансляция на мероприятии [url=https://zakazat-onlayn-translyaciyu1.ru/]онлайн трансляция на мероприятии[/url] .
zakazat onlain translyaciu _wvmt
15 Oct 25 at 8:57 pm
Цікавить будівництво і ремонт? Українська Хата розповість корисну інформацію про ремонт, інтер’єр, фермерство та інші цікаві напрямки. На сайті https://xata.od.ua/ актуальні статті для власників будинків, фермерів або тих, хто займається будівництвом і ремонтом.
fetyxieLok
15 Oct 25 at 8:59 pm
кухня по индивидуальному проекту [url=https://kuhni-spb-1.ru/]https://kuhni-spb-1.ru/[/url] .
kyhni spb_iwmi
15 Oct 25 at 8:59 pm
студия подкастов спб [url=https://studiya-podkastov-spb.ru]студия подкастов спб[/url] .
stydiya podkastov spb_bkka
15 Oct 25 at 9:00 pm
celebrex without prescription
celebrex without prescription
15 Oct 25 at 9:00 pm
https://issuu.com/xbonusblitz
Donaldbox
15 Oct 25 at 9:01 pm
Приобрести диплом ВУЗа поспособствуем. Купить диплом доктора наук – [url=http://diplomybox.com/diplom-doktora-nauk/]diplomybox.com/diplom-doktora-nauk[/url]
Cazrard
15 Oct 25 at 9:01 pm
отзывы потолочкин натяжные потолки [url=https://www.stretch-ceilings-nizhniy-novgorod.ru]https://www.stretch-ceilings-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_mzPl
15 Oct 25 at 9:03 pm
потолочкин натяжные потолки отзывы клиентов нижний новгород [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru/]natyazhnye-potolki-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_dima
15 Oct 25 at 9:05 pm
https://www.claimajob.com/profiles/7270354-1xbet-app-promo-code
CoreyBup
15 Oct 25 at 9:05 pm
современные кухни на заказ в спб [url=www.kuhni-spb-2.ru/]www.kuhni-spb-2.ru/[/url] .
kyhni spb_mnmn
15 Oct 25 at 9:06 pm
Link hsy
nmfoyidin
15 Oct 25 at 9:06 pm
https://t.me/s/rating_online/9
EverettGuemn
15 Oct 25 at 9:07 pm
https://t.me/rating_online/8
EverettGuemn
15 Oct 25 at 9:08 pm
Современные клиники предоставляют широкий спектр услуг, начиная с первичной консультации и заканчивая долгосрочной поддержкой после прохождения курса лечения. Ключевым этапом является детоксикация — процедура, направленная на выведение токсинов и стабилизацию физического состояния пациента.
Исследовать вопрос подробнее – [url=https://narkologicheskaya-pomoshh-arkhangelsk0.ru/]анонимная наркологическая помощь[/url]
MiquelCavop
15 Oct 25 at 9:10 pm
zanaflex online
zanaflex online
15 Oct 25 at 9:11 pm
Every weekend i used to visit this web page, because i want enjoyment,
for the reason that this this web site conations truly good funny data too.
facer.io
15 Oct 25 at 9:12 pm
https://t.me/s/rating_online
EverettGuemn
15 Oct 25 at 9:13 pm
Hi everyone, it’s my first pay a visit at this website,
and article is truly fruitful designed for me, keep up posting these articles.
https://888p-app.com/
15 Oct 25 at 9:14 pm
https://t.me/s/rating_online/13
EverettGuemn
15 Oct 25 at 9:14 pm
запись подкастов студия [url=http://www.studiya-podkastov-spb.ru]запись подкастов студия[/url] .
stydiya podkastov spb_uxka
15 Oct 25 at 9:15 pm
тканевый натяжной потолок нижний новгород [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru]https://natyazhnye-potolki-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_bgma
15 Oct 25 at 9:15 pm
1win bonus [url=1win5005.com]1win bonus[/url]
1win_kpml
15 Oct 25 at 9:16 pm
Ich bin verblufft von NV Casino, es ist ein Abenteuer, das pulsiert wie ein Herzschlag. Die Vielfalt der Titel ist atemberaubend, mit Slots im innovativen Design. Der Kundensupport ist hervorragend, mit praziser Unterstutzung. Die Zahlungen sind sicher und flussig, ab und zu regelma?igere Promos waren super. Zusammengefasst, NV Casino garantiert Top-Unterhaltung fur Spieler auf der Suche nach Action ! Daruber hinaus das Design ist modern und ansprechend, gibt Lust auf mehr.
playnvcasino.de|
GigabitE6zef
15 Oct 25 at 9:16 pm
заказать трансляцию [url=http://www.zakazat-onlayn-translyaciyu.ru]заказать трансляцию[/url] .
zakazat onlain translyaciu _fkka
15 Oct 25 at 9:18 pm
мфо займ онлайн [url=https://www.zaimy-28.ru]мфо займ онлайн[/url] .
zaimi_bpKa
15 Oct 25 at 9:19 pm
стоимость онлайн трансляции на мероприятии [url=https://zakazat-onlayn-translyaciyu.ru]https://zakazat-onlayn-translyaciyu.ru[/url] .
zakazat onlain translyaciu _ldka
15 Oct 25 at 9:22 pm
организация онлайн трансляции мероприятия [url=https://www.zakazat-onlayn-translyaciyu1.ru]организация онлайн трансляции мероприятия[/url] .
zakazat onlain translyaciu _mcmt
15 Oct 25 at 9:22 pm
натяжные потолки официальный сайт [url=https://natyazhnye-potolki-nizhniy-novgorod.ru/]https://natyazhnye-potolki-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_tmOt
15 Oct 25 at 9:24 pm
Процесс терапии в клинике построен по поэтапной схеме, включающей:
Получить больше информации – http://lechenie-alkogolizma-arkhangelsk0.ru/lechenie-narkomanii-i-alkogolizma-arkhangelsk/https://lechenie-alkogolizma-arkhangelsk0.ru
GeorgeCat
15 Oct 25 at 9:25 pm
кухня на заказ спб [url=https://kuhni-spb-1.ru/]kuhni-spb-1.ru[/url] .
kyhni spb_dimi
15 Oct 25 at 9:25 pm
параметры трансляции [url=https://www.zakazat-onlayn-translyaciyu1.ru]https://www.zakazat-onlayn-translyaciyu1.ru[/url] .
zakazat onlain translyaciu _fymt
15 Oct 25 at 9:26 pm
потолки натяжные [url=www.natyazhnye-potolki-nizhniy-novgorod.ru/]www.natyazhnye-potolki-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_bvOt
15 Oct 25 at 9:27 pm
Its such as you learn my mind! You seem to grasp a lot approximately this, such as you wrote the guide in it or something.
I believe that you could do with a few percent to drive the message house a little bit, but instead
of that, that is magnificent blog. An excellent read.
I will certainly be back.
and coupons
15 Oct 25 at 9:27 pm
потолочник потолки [url=https://stretch-ceilings-nizhniy-novgorod.ru/]stretch-ceilings-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_jbPl
15 Oct 25 at 9:28 pm
1win qeydiyyat pulsuz [url=https://1win5004.com/]https://1win5004.com/[/url]
1win_xpoi
15 Oct 25 at 9:28 pm
кухни от производителя спб [url=http://www.kuhni-spb-1.ru]http://www.kuhni-spb-1.ru[/url] .
kyhni spb_hlmi
15 Oct 25 at 9:28 pm
Hey there! I know this is kinda off topic however I’d figured I’d ask.
Would you be interested in trading links or maybe guest authoring a blog article
or vice-versa? My blog goes over a lot of the same topics as yours and I think we could greatly
benefit from each other. If you are interested feel free to
send me an email. I look forward to hearing from you!
Wonderful blog by the way!
boobs
15 Oct 25 at 9:31 pm
https://medicosur.com/# mexico pharmacy
MervinWoorE
15 Oct 25 at 9:32 pm
mobile betting
PHP hook, building hooks in your application – Sjoerd Maessen blog at Sjoerd Maessen blog
mobile betting
15 Oct 25 at 9:33 pm
Wow, this piece of writing is fastidious, my sister is analyzing these things, thus I am going to inform
her.
FIFA World Cup 2026 Tickets
15 Oct 25 at 9:33 pm
аренда экскаватора в москве цена [url=https://arenda-mini-ekskavatora-v-moskve.ru/]аренда экскаватора в москве цена[/url] .
arenda mini ekskavatora v moskve_xjEn
15 Oct 25 at 9:35 pm
потолочкин натяжные [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru/]natyazhnye-potolki-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_nzma
15 Oct 25 at 9:36 pm