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!
That is very attention-grabbing, You’re a very professional
blogger. I have joined your feed and look ahead
to seeking extra of your wonderful post. Also, I’ve shared your site in my social networks
trygge norske casino
28 Oct 25 at 12:35 am
гидроизоляция цена за рулон [url=http://gidroizolyaciya-cena-8.ru/]гидроизоляция цена за рулон[/url] .
gidroizolyaciya cena_rvKn
28 Oct 25 at 12:35 am
Автоматические выключатели
JulioGer
28 Oct 25 at 12:36 am
https://t.me/bs_1Win/1205
Georgerah
28 Oct 25 at 12:36 am
https://t.me/bs_1Win/740
Georgerah
28 Oct 25 at 12:37 am
Hey there, I think your website might be having browser compatibility issues.
When I look at your website in Safari, it looks
fine but when opening in Internet Explorer, it has some overlapping.
I just wanted to give you a quick heads up! Other then that, very good blog!
airmatic
28 Oct 25 at 12:38 am
наркологическая больница [url=http://www.narkologicheskaya-klinika-25.ru]наркологическая больница[/url] .
narkologicheskaya klinika_fzPl
28 Oct 25 at 12:39 am
Right here is the perfect blog for everyone who would
like to understand this topic. You know a whole lot its almost hard to argue with you
(not that I personally would want to…HaHa). You definitely put
a fresh spin on a topic which has been discussed for ages.
Wonderful stuff, just wonderful!
Zyberlich Beam AI Legit Or Not
28 Oct 25 at 12:41 am
kraken android
кракен сайт
Henryamerb
28 Oct 25 at 12:42 am
кракен vk3
кракен даркнет маркет
Henryamerb
28 Oct 25 at 12:43 am
гидроизоляция подвала цена [url=https://gidroizolyaciya-cena-8.ru/]гидроизоляция подвала цена[/url] .
gidroizolyaciya cena_rjKn
28 Oct 25 at 12:44 am
If you are going for finest contents like me, only pay a quick visit this site daily for the reason that it provides feature contents, thanks
bokep indonesia sma
28 Oct 25 at 12:45 am
адреса наркологических клиник [url=www.narkologicheskaya-klinika-25.ru/]www.narkologicheskaya-klinika-25.ru/[/url] .
narkologicheskaya klinika_qqPl
28 Oct 25 at 12:46 am
гидроизоляция подвала москва [url=gidroizolyaciya-podvala-cena.ru]гидроизоляция подвала москва[/url] .
gidroizolyaciya podvala cena_dpKt
28 Oct 25 at 12:46 am
внутренняя гидроизоляция подвала [url=www.gidroizolyaciya-cena-7.ru/]внутренняя гидроизоляция подвала[/url] .
gidroizolyaciya cena_hdSi
28 Oct 25 at 12:47 am
кракен Москва
кракен клиент
Henryamerb
28 Oct 25 at 12:47 am
Этот список — памятка на холодильник. Он не раскрывает диагноз, но точно описывает поводы для связи с дежурным врачом. Своевременная эскалация — часть безопасности, а не «подстраховка». Если сомневаетесь — лучше позвонить.
Получить дополнительную информацию – https://kapelnicza-ot-zapoya-v-chelyabinske16.ru/kapelnicza-vyvod-iz-zapoya-chelyabinsk/
Craigtic
28 Oct 25 at 12:48 am
купить диплом в камышине [url=www.rudik-diplom1.ru/]купить диплом в камышине[/url] .
Diplomi_iwer
28 Oct 25 at 12:51 am
явно не 15-ти минутка, почитайте отзывы в соответвующей теме. Качество товара на высоте у нас всегда . https://nebezit.ru согласен с вами ребят
JasonBoomi
28 Oct 25 at 12:53 am
кракен
кракен даркнет маркет
Henryamerb
28 Oct 25 at 12:53 am
sheacentral.com – Pages loaded fast, images appeared sharp, and formatting stayed consistent.
Lupe Bownds
28 Oct 25 at 12:54 am
платный наркологический диспансер москва [url=http://www.narkologicheskaya-klinika-28.ru]http://www.narkologicheskaya-klinika-28.ru[/url] .
narkologicheskaya klinika_viMa
28 Oct 25 at 12:57 am
Получить диплом любого ВУЗа можем помочь. Купить диплом техникума, колледжа в Тольятти – [url=http://diplomybox.com/kupit-diplom-tekhnikuma-kolledzha-v-tolyatti/]diplomybox.com/kupit-diplom-tekhnikuma-kolledzha-v-tolyatti[/url]
Cazrwlh
28 Oct 25 at 12:58 am
клиники наркологические москва [url=https://narkologicheskaya-klinika-25.ru]https://narkologicheskaya-klinika-25.ru[/url] .
narkologicheskaya klinika_cfPl
28 Oct 25 at 12:58 am
наркология лечение [url=http://narkologicheskaya-klinika-28.ru/]наркология лечение[/url] .
narkologicheskaya klinika_knMa
28 Oct 25 at 1:00 am
Le code promo est supprime : entrez-le dans le champ « Code promo » et reclamez un bonus de bienvenue de 100% jusqu’a 130€, pour vos paris sportifs. Inscrivez-vous sur 1xBet ou via l’application mobile. Apres votre premier depot, vous activerez le code bonus. L’offre est valable pour toute l’annee 2026, et le bonus doit etre mise dans les 30 jours. Decouvrez plus d’informations sur le code promo via ce lien — Codes Promo 1xbet. Le code promo 1xBet casino offre des tours gratuits et un bonus de depot 1xBet pour les nouveaux joueurs. Avec le code promotionnel 1xBet pour nouveaux utilisateurs, recevez jusqu’a 130€ de bonus d’inscription 1xBet. Utilisez le code promo 1xBet aujourd’hui pour jouer au casino en ligne 1xBet et profiter de toutes les offres disponibles.
Charlescex
28 Oct 25 at 1:01 am
ремонт подвала в частном доме [url=www.gidroizolyaciya-cena-8.ru/]www.gidroizolyaciya-cena-8.ru/[/url] .
gidroizolyaciya cena_alKn
28 Oct 25 at 1:01 am
Удобные и стильные [url=https://avtomaticheskie-karnizy-s-pultom.ru/]автоматические карнизы для штор +7 (499) 638-25-37[/url] делают управление шторами простым и комфортным.
Также стоит отметить разнообразие дизайнов автоматических карнизов.
купить автоматический карниз Prokarniz
28 Oct 25 at 1:01 am
кракен официальный сайт
kraken сайт
Henryamerb
28 Oct 25 at 1:02 am
kraken vk4
кракен онион
Henryamerb
28 Oct 25 at 1:02 am
купить диплом техникума отзывы [url=www.frei-diplom9.ru]купить диплом техникума отзывы[/url] .
Diplomi_gaea
28 Oct 25 at 1:02 am
http://vitalpharma24.com/# Kamagra 100mg bestellen
Davidjealp
28 Oct 25 at 1:03 am
купить диплом в заречном [url=https://rudik-diplom9.ru]купить диплом в заречном[/url] .
Diplomi_cdei
28 Oct 25 at 1:04 am
Простая подготовка уменьшает стресс, повышает переносимость капельницы и делает оценку состояния более точной. Попросите одного ответственного взрослого быть на связи с врачом — так решения будут приниматься быстрее, а ночной интервал пройдёт спокойнее и предсказуемее.
Получить дополнительные сведения – http://vyvod-iz-zapoya-v-chelyabinske16.ru/vyvod-iz-zapoya-chelyabinsk-anonimno/
MichaelPix
28 Oct 25 at 1:04 am
Wow cuz this is great job! Congrats and keep
it up.
https://www.tvobiektyw.pl
https://www.tvobiektyw.pl
28 Oct 25 at 1:05 am
купить диплом в буйнакске [url=rudik-diplom1.ru]rudik-diplom1.ru[/url] .
Diplomi_hker
28 Oct 25 at 1:07 am
наркология в москве [url=http://narkologicheskaya-klinika-25.ru/]наркология в москве[/url] .
narkologicheskaya klinika_qmPl
28 Oct 25 at 1:07 am
Je suis enthousiasme par Sugar Casino, ca transporte dans un univers de plaisirs. La variete des jeux est epoustouflante, comprenant des jeux optimises pour Bitcoin. Avec des transactions rapides. Le suivi est impeccable. Les transactions sont toujours fiables, mais encore des offres plus consequentes seraient parfaites. En somme, Sugar Casino merite une visite dynamique. En extra la plateforme est visuellement vibrante, apporte une energie supplementaire. A mettre en avant les transactions en crypto fiables, offre des recompenses continues.
DГ©couvrir le web|
skyfireos5zef
28 Oct 25 at 1:07 am
kraken зеркало
kraken обмен
Henryamerb
28 Oct 25 at 1:07 am
экстренное вытрезвление [url=https://narkologicheskaya-klinika-28.ru/]экстренное вытрезвление[/url] .
narkologicheskaya klinika_qpMa
28 Oct 25 at 1:09 am
Je ne me lasse pas de Ruby Slots Casino, ca transporte dans un univers de plaisirs. Le catalogue de titres est vaste, proposant des jeux de cartes elegants. Il booste votre aventure des le depart. Les agents sont rapides et pros. Les transactions sont fiables et efficaces, cependant plus de promos regulieres ajouteraient du peps. En somme, Ruby Slots Casino est un incontournable pour les joueurs. En complement l’interface est fluide comme une soiree, booste le fun du jeu. A mettre en avant les evenements communautaires vibrants, propose des privileges personnalises.
Visiter la plateforme|
Nightspiner8zef
28 Oct 25 at 1:09 am
Автоматические выключатели
JulioGer
28 Oct 25 at 1:10 am
It’s an awesome paragraph designed for all the internet visitors; they
will take benefit from it I am sure.
deutsche online casino
28 Oct 25 at 1:10 am
гидроизоляция цена за м2 [url=http://gidroizolyaciya-cena-8.ru/]гидроизоляция цена за м2[/url] .
gidroizolyaciya cena_qsKn
28 Oct 25 at 1:11 am
кракен vk4
кракен даркнет маркет
Henryamerb
28 Oct 25 at 1:14 am
купить диплом в самаре [url=http://www.rudik-diplom1.ru]купить диплом в самаре[/url] .
Diplomi_azer
28 Oct 25 at 1:14 am
купить диплом в тюмени [url=http://rudik-diplom8.ru/]купить диплом в тюмени[/url] .
Diplomi_gyMt
28 Oct 25 at 1:15 am
wetten tipps vorhersagen (http://www.vlastakruntoradova.cz) und
gewinnen
www.vlastakruntoradova.cz
28 Oct 25 at 1:15 am
можно купить диплом медсестры [url=http://frei-diplom15.ru/]можно купить диплом медсестры[/url] .
Diplomi_zeoi
28 Oct 25 at 1:15 am
Thanks on your marvelous posting! I definitely enjoyed reading it, you’re a
great author.I will make sure to bookmark your blog and definitely
will come back in the foreseeable future. I want to encourage
continue your great posts, have a nice day!
pawn gold price
28 Oct 25 at 1:16 am