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=http://narkologicheskaya-klinika-27.ru/]клиника вывод из запоя[/url] .
narkologicheskaya klinika_ahpl
28 Oct 25 at 2:40 am
https://t.me/s/bs_1Win/1272
Georgerah
28 Oct 25 at 2:43 am
kraken сайт
kraken СПб
Henryamerb
28 Oct 25 at 2:43 am
Hello there! Do you use Twitter? I’d like to follow you if that would be okay.
I’m absolutely enjoying your blog and look forward
to new posts.
My blog – zinnat02
zinnat02
28 Oct 25 at 2:44 am
kraken tor
kraken vpn
Henryamerb
28 Oct 25 at 2:44 am
помощь нарколога [url=http://narkologicheskaya-klinika-28.ru/]помощь нарколога[/url] .
narkologicheskaya klinika_jzMa
28 Oct 25 at 2:47 am
Букмекерские компании стараются не только привлечь новых клиентов приятными подарками, но и удержать «старичков». Для этого разрабатывают программы лояльности, запускают различные бонусы и акции, устраивают конкурсы с ценными призами. Самым популярным является мелбет промокод при регистрации на сегодня, который вводится при регистрации нового аккаунта. Благодаря ему вы получаете увеличенный бонус на первый депозит, система добавляет 30% к максимальной сумме. Стандартный бонус на первое пополнение счета составляет 8000 рублей, после введения промокода – 50 000 рублей.
Georgeduh
28 Oct 25 at 2:48 am
кракен актуальная ссылка
kraken vk5
Henryamerb
28 Oct 25 at 2:49 am
Hello, after reading this remarkable post i am too
happy to share my familiarity here with mates.
With thanks
28 Oct 25 at 2:51 am
купить диплом в таганроге [url=https://rudik-diplom10.ru/]https://rudik-diplom10.ru/[/url] .
Diplomi_hqSa
28 Oct 25 at 2:51 am
Hello just wanted to give you a quick heads up. The text
in your article seem to be running off the screen in Ie.
I’m not sure if this is a formatting issue or something to do with web browser compatibility but
I thought I’d post to let you know. The design look great though!
Hope you get the problem resolved soon. Cheers
deep sea fishing
28 Oct 25 at 2:52 am
kraken client
кракен vpn
Henryamerb
28 Oct 25 at 2:55 am
https://t.me/bs_1Win/1117
Georgerah
28 Oct 25 at 2:55 am
Каждый пациент проходит несколько последовательных этапов, направленных на очищение организма, стабилизацию эмоционального состояния и формирование стойкой мотивации к трезвой жизни. Такой подход позволяет не только устранить физическую зависимость, но и укрепить психическую устойчивость.
Углубиться в тему – [url=https://narkologicheskaya-klinika-v-samare16.ru/]narkologicheskaya-klinika-v-samare16.ru/[/url]
RobertDab
28 Oct 25 at 2:56 am
https://t.me/s/bs_1Win/756
Georgerah
28 Oct 25 at 2:56 am
Seobomba.ru/shop — магазин «вечных» ссылок и SEO-услуг: статейные и форумные размещения, линкбилдинг по Ahrefs, усиление Tier2, аудиты и соцсигналы. Ассортимент закрывает задачи продвижения под Яндекс и Google, акцент на ручной работе, низком спаме и трастовых площадках. Удобные тарифы «Эконом», «Бизнес», «Премиум» и разовые услуги под кампанию. Откройте витрину на https://seobomba.ru/shop/ — прозрачные цены, быстрый заказ и дополнительные инструменты вроде анализатора текстов и семантики помогут ускорить рост позиций и конверсий
qobyzensok
28 Oct 25 at 2:57 am
lithestore.com – Pages loaded fast, images appeared sharp, and formatting stayed consistent.
Patrick Castrillo
28 Oct 25 at 2:58 am
клиника наркологическая москва [url=http://narkologicheskaya-klinika-28.ru]http://narkologicheskaya-klinika-28.ru[/url] .
narkologicheskaya klinika_aoMa
28 Oct 25 at 3:00 am
Melbet промокод на сегодня стоит ввести для повышенного бонуса на первый депозит. Большинство букмекерских контор используют такую практику привлечения новых игроков и выдают значительные бонусы. Актуальный промокод Мелбет на 2026 год – RS777. Ввести его могут исключительно новые игроки, у которых еще нет аккаунта в Melbet. Не рекомендуем идти на хитрость и заново проходить процесс регистрации, так как служба безопасности тщательно отслеживает мультиаккаунты. Максимальная сумма бонуса при использовании промокода Мелбет – 50 000 рублей. Это значит, что при вводе средств на баланс после регистрации новому игроку будет зачислено еще 130% от первого депозита в качестве поощрения от букмекера.
Georgeduh
28 Oct 25 at 3:00 am
ремонт подвального помещения [url=http://gidroizolyaciya-cena-8.ru]ремонт подвального помещения[/url] .
gidroizolyaciya cena_oaKn
28 Oct 25 at 3:01 am
платная наркологическая клиника [url=https://narkologicheskaya-klinika-27.ru/]платная наркологическая клиника[/url] .
narkologicheskaya klinika_efpl
28 Oct 25 at 3:02 am
кракен даркнет маркет
кракен Москва
Henryamerb
28 Oct 25 at 3:03 am
kraken ios
kraken ios
Henryamerb
28 Oct 25 at 3:04 am
I was suggested this blog by way of my cousin. I’m not positive whether or not this put up is written by means of him as nobody else recognize such detailed about my difficulty.
You are wonderful! Thanks!
Altrix SpotPro
28 Oct 25 at 3:04 am
можно ли купить легальный диплом [url=www.frei-diplom2.ru/]www.frei-diplom2.ru/[/url] .
Diplomi_onEa
28 Oct 25 at 3:04 am
купил диплом легально [url=www.frei-diplom5.ru]купил диплом легально[/url] .
Diplomi_bhPa
28 Oct 25 at 3:06 am
купить диплом в соликамске [url=https://rudik-diplom5.ru/]купить диплом в соликамске[/url] .
Diplomi_vkma
28 Oct 25 at 3:07 am
купить диплом в бийске [url=http://rudik-diplom13.ru]купить диплом в бийске[/url] .
Diplomi_boon
28 Oct 25 at 3:07 am
купить диплом в благовещенске [url=www.rudik-diplom9.ru/]купить диплом в благовещенске[/url] .
Diplomi_ihei
28 Oct 25 at 3:08 am
kraken ios
kraken vk5
Henryamerb
28 Oct 25 at 3:09 am
What’s up colleagues, how is the whole thing, and what you wish for to say on the topic of this paragraph, in my view its actually remarkable in support of
me.
https://alvarezconsultores.com/melbet-vhod-registratsiya-2025/
28 Oct 25 at 3:09 am
Je suis completement seduit par Sugar Casino, on ressent une ambiance de fete. La selection de jeux est impressionnante, incluant des paris sur des evenements sportifs. 100% jusqu’a 500 € + tours gratuits. Le support est rapide et professionnel. Les transactions sont toujours securisees, quelquefois des recompenses additionnelles seraient ideales. Dans l’ensemble, Sugar Casino est un endroit qui electrise. A mentionner le site est fluide et attractif, ajoute une touche de dynamisme. Un bonus les nombreuses options de paris sportifs, offre des recompenses regulieres.
Apprendre comment|
skyfireos5zef
28 Oct 25 at 3:10 am
экстренное вытрезвление в москве [url=http://narkologicheskaya-klinika-28.ru/]экстренное вытрезвление в москве[/url] .
narkologicheskaya klinika_pyMa
28 Oct 25 at 3:12 am
наркология в москве [url=www.narkologicheskaya-klinika-27.ru]наркология в москве[/url] .
narkologicheskaya klinika_aopl
28 Oct 25 at 3:12 am
Je ne me lasse pas de Ruby Slots Casino, on y trouve une vibe envoutante. Le catalogue est un tresor de divertissements, incluant des paris sportifs en direct. Il donne un avantage immediat. Le suivi est impeccable. Le processus est transparent et rapide, a l’occasion plus de promotions variees ajouteraient du fun. En conclusion, Ruby Slots Casino est un incontournable pour les joueurs. Notons egalement le site est rapide et engageant, booste le fun du jeu. Un element fort les tournois frequents pour l’adrenaline, assure des transactions fluides.
Parcourir maintenant|
Nightspiner8zef
28 Oct 25 at 3:13 am
kraken vk4
кракен vpn
Henryamerb
28 Oct 25 at 3:15 am
купить диплом в черкесске [url=http://rudik-diplom9.ru]купить диплом в черкесске[/url] .
Diplomi_xnei
28 Oct 25 at 3:21 am
Каждый пациент проходит последовательный процесс лечения, где основное внимание уделяется безопасности, контролю состояния и постепенной адаптации. При поступлении проводится диагностика с оценкой степени зависимости, сопутствующих патологий и психологических факторов, влияющих на развитие заболевания. На основании полученных данных формируется персонализированный план терапии, включающий медикаментозные и немедикаментозные методы.
Детальнее – http://narkologicheskaya-klinika-v-spb16.ru/narkologicheskie-dispansery-sankt-peterburga/
Brianvat
28 Oct 25 at 3:21 am
купить диплом логопеда [url=www.rudik-diplom4.ru]купить диплом логопеда[/url] .
Diplomi_epOr
28 Oct 25 at 3:22 am
наркология москва [url=http://narkologicheskaya-klinika-27.ru/]наркология москва[/url] .
narkologicheskaya klinika_rvpl
28 Oct 25 at 3:22 am
кракен vpn
кракен Москва
Henryamerb
28 Oct 25 at 3:23 am
Instead, spinrise [url=https://dmbikecomf565e.zapwp.com/m:0/a:https://slaot.org/]https://dmbikecomf565e.zapwp.com/m:0/a:https://slaot.org/[/url] decided to focus on providing a full-fledged mobile web service that perfectly functions on any platforms without the need something download or install.
Jessemuh
28 Oct 25 at 3:24 am
кракен зеркало
кракен vpn
Henryamerb
28 Oct 25 at 3:24 am
Поэтапное лечение помогает врачам контролировать процесс восстановления и своевременно корректировать терапию в зависимости от состояния пациента.
Подробнее – [url=https://narkologicheskaya-klinika-v-samare16.ru/]наркологические клиники алкоголизм в самаре[/url]
RobertDab
28 Oct 25 at 3:25 am
купить диплом в черкесске [url=https://rudik-diplom9.ru/]купить диплом в черкесске[/url] .
Diplomi_otei
28 Oct 25 at 3:28 am
купить диплом нового образца [url=https://rudik-diplom7.ru/]купить диплом нового образца[/url] .
Diplomi_scPl
28 Oct 25 at 3:28 am
kraken официальный
кракен vk5
Henryamerb
28 Oct 25 at 3:29 am
Автоматические выключатели
JulioGer
28 Oct 25 at 3:30 am
You have fantastic knowlwdge these.
https://iotwock.info
https://iotwock.info
28 Oct 25 at 3:35 am
kraken СПб
kraken ios
Henryamerb
28 Oct 25 at 3:36 am