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!
lucky jet краш [url=https://1win5518.ru]https://1win5518.ru[/url]
1win_kg_ghkl
24 Oct 25 at 10:19 am
1 xbet giri? [url=http://1xbet-4.com/]http://1xbet-4.com/[/url] .
1xbet_xsol
24 Oct 25 at 10:19 am
1xbet [url=http://1xbet-giris-8.com/]1xbet[/url] .
1xbet giris_lfPn
24 Oct 25 at 10:19 am
1xbet mobil giri? [url=www.1xbet-giris-10.com/]1xbet mobil giri?[/url] .
1xbet giris_fjka
24 Oct 25 at 10:21 am
1xbet g?ncel adres [url=http://1xbet-giris-7.com]1xbet g?ncel adres[/url] .
1xbet giris_taKn
24 Oct 25 at 10:22 am
регистрация лаки джет [url=https://www.1win5518.ru]https://www.1win5518.ru[/url]
1win_kg_egkl
24 Oct 25 at 10:22 am
1xbet giri? yapam?yorum [url=www.1xbet-giris-9.com/]1xbet giri? yapam?yorum[/url] .
1xbet giris_snon
24 Oct 25 at 10:23 am
esc buchmacher quoten
Visit my web page; die besten wett tipps (Brook)
Brook
24 Oct 25 at 10:23 am
Публикация приглашает вас исследовать неизведанное — от древних тайн до современных достижений науки. Вы узнаете, как случайные находки превращались в революции, а смелые мысли — в новые эры человеческого прогресса.
Смотри, что ещё есть – https://altec.ch/business1
LloydNal
24 Oct 25 at 10:23 am
Dragon Money – онлайн-казино с слотами,
рулеткой и лайв-дилерами. Удобный интерфейс, безопасные
платежи и бонусы для игроков.
промокод драгон мани
WalterDemia
24 Oct 25 at 10:25 am
1xbet resmi giri? [url=https://1xbet-giris-10.com/]1xbet resmi giri?[/url] .
1xbet giris_mqka
24 Oct 25 at 10:27 am
OMT’s mix of online and on-site options provides
versatility, mɑking mathematics easily accessible аnd charming,
whіle inspiring Singapore trainees f᧐r exam success.
Dive into ѕеlf-paced mathematics proficiency ԝith OMT’s 12-month e-learning courses, tоtal witһ practice worksheets annd recorded sessions fߋr comprehensive revision.
Singapore’ѕ focus on important thinking through mathematics highlights tһе significance of math tuition, ᴡhich helps trainees develop tһe analytical skills demanded Ƅy
the country’ѕ forward-thinking syllabus.
primary tuition іs very important for PSLE
as it uѕes restorative support for subjects liкe entire numbers and measurements,
making surе no fundamental weaknesses persist.
Ᏼy offering considerable method ѡith pɑst OLevel papers, tuition furnishes trainees
with familiarity аnd tһe capability to anticipate inquiry patterns.
Junior college math tuition promotes іmportant believing abilities required tо address non-routine issues tһat frequently
show uр іn Ꭺ Level mathematics assessments.
By integrating proprietary ɑpproaches wіtһ thhe MOE syllabus, OMT սses a distinctive approach tһat
highlights clearness аnd deepness in mathematical
thinking.
Visual aids ⅼike layouts hеlp picture problems lor,
enhancing understanding and examination performance.
Singapore’s concentrate ߋn all natural education аnd learning is enhanced ƅy math tuition tһat develops logical thinking fоr lifelong examination advantages.
Нere is my webpage :: engineering maths tuition
engineering maths tuition
24 Oct 25 at 10:27 am
Ремонт двигателя Опель [url=http://www.dzen.ru/a/aPkGhHYePCku9uSe]http://www.dzen.ru/a/aPkGhHYePCku9uSe[/url] .
Kapitalnii remont benzinovih dvigatelei v Moskve_itEn
24 Oct 25 at 10:29 am
1 x bet [url=https://1xbet-giris-9.com/]1 x bet[/url] .
1xbet giris_noon
24 Oct 25 at 10:31 am
1xbet spor bahislerinin adresi [url=http://1xbet-giris-10.com/]1xbet spor bahislerinin adresi[/url] .
1xbet giris_vtka
24 Oct 25 at 10:31 am
1xbet giri?i [url=https://www.1xbet-giris-8.com]1xbet giri?i[/url] .
1xbet giris_nuPn
24 Oct 25 at 10:33 am
1 x bet giri? [url=https://1xbet-9.com/]1xbet-9.com[/url] .
1xbet_jaSn
24 Oct 25 at 10:33 am
1xbet yeni giri? [url=https://1xbet-giris-9.com]1xbet yeni giri?[/url] .
1xbet giris_whon
24 Oct 25 at 10:33 am
Этот медицинский обзор сосредоточен на последних достижениях, которые оказывают влияние на пациентов и медицинскую практику. Мы разбираем инновационные методы лечения и исследований, акцентируя внимание на их значимости для общественного здоровья. Читатели узнают о свежих данных и их возможном применении.
Изучить вопрос глубже – https://womanjour.ru/vygody-lecheniya-na-domu.html
TammyAbews
24 Oct 25 at 10:34 am
Dragon Money – онлайн-казино с слотами,
рулеткой и лайв-дилерами. Удобный интерфейс, безопасные
платежи и бонусы для игроков.
как внести депозит dragon money
WalterDemia
24 Oct 25 at 10:34 am
купить диплом в норильске [url=https://www.rudik-diplom15.ru]купить диплом в норильске[/url] .
Diplomi_etPi
24 Oct 25 at 10:34 am
1xbet tr [url=http://1xbet-giris-7.com/]1xbet tr[/url] .
1xbet giris_ebKn
24 Oct 25 at 10:35 am
Публикация предлагает читателю не просто информацию, а инструменты для анализа и саморазвития. Мы стимулируем критическое мышление, предлагая различные точки зрения и призывая к самостоятельному поиску решений.
Что скрывают от вас? – http://redcube.support/hello-world
Danielcog
24 Oct 25 at 10:35 am
1хБет код для регистрации на приветственный подарок в 2026 — активируйте промокод и оформите приветственный подарок в размере 100% до 100$. Данный бонусный код позволяет активировать приветственный бонус от букмекера 1xBet при регистрации.Актуальный промокод доступен на сайте 1xBet — http://www.medtronik.ru/images/pages/1hbet_2021_promokod_pri_registracii.html.
JesusJuize
24 Oct 25 at 10:39 am
1x bet giri? [url=https://1xbet-giris-10.com]1x bet giri?[/url] .
1xbet giris_ivka
24 Oct 25 at 10:41 am
купить диплом в уссурийске [url=http://rudik-diplom14.ru/]http://rudik-diplom14.ru/[/url] .
Diplomi_tkea
24 Oct 25 at 10:42 am
I’m not sure where you are getting your info, but great topic.
I needs to spend some time learning much more or understanding more.
Thanks for great info I was looking for this information for my
mission.
Replica Watch
24 Oct 25 at 10:43 am
https://confiafarmacia.shop/# farmacia online para hombres
Hermanereli
24 Oct 25 at 10:44 am
1xbet turkiye [url=www.1xbet-giris-8.com]www.1xbet-giris-8.com[/url] .
1xbet giris_goPn
24 Oct 25 at 10:45 am
whoah this blog is great i like reading your articles. Stay up the great work!
You realize, lots of individuals are searching round for this info, you could help
them greatly.
danypatches
24 Oct 25 at 10:45 am
farmacia con entrega rápida: Viagra sin prescripción médica – pastillas de potencia masculinas
Jesuskax
24 Oct 25 at 10:45 am
1xbet [url=https://www.1xbet-giris-3.com]1xbet[/url] .
1xbet giris_ejMi
24 Oct 25 at 10:46 am
Отзыв коротенько. Закладка на месте. Обьяснение распологает. Фото имеется. Бот чёткий. бошки хороши. Вес норма! … ну вот и всё Коротко и ладно)
Онлайн магазин – купить мефедрон, кокаин, бошки
Магазин в полном порядке!!
Thomaspet
24 Oct 25 at 10:47 am
1xbet tr giri? [url=www.1xbet-giris-9.com/]www.1xbet-giris-9.com/[/url] .
1xbet giris_bjon
24 Oct 25 at 10:47 am
1xbet giri? g?ncel [url=http://1xbet-4.com]http://1xbet-4.com[/url] .
1xbet_uyol
24 Oct 25 at 10:47 am
1xbet yeni giri? adresi [url=https://1xbet-giris-10.com/]1xbet yeni giri? adresi[/url] .
1xbet giris_thka
24 Oct 25 at 10:49 am
1xbet giri?i [url=www.1xbet-giris-8.com/]1xbet giri?i[/url] .
1xbet giris_qiPn
24 Oct 25 at 10:51 am
You are so interesting! I don’t believe I’ve read through anything like that before.
So good to find someone with a few genuine thoughts on this issue.
Really.. thank you for starting this up. This site is something that is needed on the internet,
someone with some originality!
commercial kitchen exhaust
24 Oct 25 at 10:52 am
https://www.lenotoplenie.ru/ официальный сайт с полезной информацией о бонусах
CollinDwets
24 Oct 25 at 10:53 am
birxbet giri? [url=www.1xbet-9.com/]www.1xbet-9.com/[/url] .
1xbet_qaSn
24 Oct 25 at 10:54 am
E28BET-এ স্বাগতম – এশিয়া
প্যাসিফিকের নং 1 অনলাইন জুয়া সাইট। বোনাস, উত্তেজনাপূর্ণ গেম এবং একটি বিশ্বস্ত অনলাইন
বেটিং অভিজ্ঞতা উপভোগ করুন।
E28BET - এশিয়া প্যাসিফিকের নং 1 অনলাইন জুয়া সাইট
24 Oct 25 at 10:56 am
куплю диплом младшей медсестры [url=www.frei-diplom15.ru]www.frei-diplom15.ru[/url] .
Diplomi_zzoi
24 Oct 25 at 10:56 am
1xbet giri? [url=http://1xbet-giris-10.com/]1xbet giri?[/url] .
1xbet giris_ebka
24 Oct 25 at 10:56 am
кракен Москва
кракен тор
JamesDaync
24 Oct 25 at 10:59 am
bahis siteler 1xbet [url=www.1xbet-giris-9.com]bahis siteler 1xbet[/url] .
1xbet giris_fson
24 Oct 25 at 10:59 am
1xbet g?ncel [url=1xbet-giris-3.com]1xbet g?ncel[/url] .
1xbet giris_iuMi
24 Oct 25 at 10:59 am
1xbet lite [url=www.1xbet-giris-10.com/]1xbet lite[/url] .
1xbet giris_woka
24 Oct 25 at 10:59 am
urbanstylehub – The packaging was neat and the item arrived in perfect condition today.
Gerald Donlyuk
24 Oct 25 at 11:02 am
1xbet giri? [url=http://www.1xbet-giris-8.com]1xbet giri?[/url] .
1xbet giris_omPn
24 Oct 25 at 11:02 am
https://www.noteflight.com/profile/e4d42de8e15dfd9bade4a4b1f7822fccc8dcc5a5
pertkam
24 Oct 25 at 11:03 am