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!
https://herengezondheid.shop/# Viagra online kopen Nederland
JamesSlilk
26 Oct 25 at 9:05 pm
https://t.me/KudaSaintPetersburg Продолжить знакомство с городом можно, прогуливаясь по Невскому проспекту – главной артерии Петербурга. Здесь, среди исторических зданий и бутиков, кипит жизнь. Загляните в Дом книги, полюбуйтесь архитектурой Казанского собора, сделайте фото на фоне Елисеевского магазина – места, пропитанного духом старого Петербурга.
MichaelMit
26 Oct 25 at 9:07 pm
kraken darknet
kraken vk6
Henryamerb
26 Oct 25 at 9:07 pm
Нужна дезинфекция для автомобиля, посоветуйте.
обработка от педикулеза
KennethceM
26 Oct 25 at 9:08 pm
Результат после дезинсекция потрясающий!
уничтожение клопов
KennethceM
26 Oct 25 at 9:10 pm
поставщик медицинского оборудования [url=https://medoborudovanie-postavka.ru/]поставщик медицинского оборудования[/url] .
postavka medicinskogo oborydovaniya_xosn
26 Oct 25 at 9:10 pm
1xbet yeni giri? adresi [url=https://1xbet-14.com/]1xbet yeni giri? adresi[/url] .
1xbet_kdet
26 Oct 25 at 9:11 pm
1xbet mobi [url=1xbet-17.com]1xbet mobi[/url] .
1xbet_nnpl
26 Oct 25 at 9:12 pm
kraken market
kraken qr code
Henryamerb
26 Oct 25 at 9:13 pm
http://build-info.biz.ua/wr_board/tools.php?event=profile&pname=itifuq
Jameslef
26 Oct 25 at 9:17 pm
лечение зависимостей [url=https://narkologicheskaya-klinika-23.ru]https://narkologicheskaya-klinika-23.ru[/url] .
narkologicheskaya klinika_deet
26 Oct 25 at 9:17 pm
поставка медоборудования [url=http://www.medoborudovanie-postavka.ru]http://www.medoborudovanie-postavka.ru[/url] .
postavka medicinskogo oborydovaniya_ahsn
26 Oct 25 at 9:18 pm
медицинская техника [url=http://medicinskaya-tehnika.ru/]медицинская техника[/url] .
medicinskaya tehnika_hwEi
26 Oct 25 at 9:19 pm
экстренное вытрезвление [url=http://narkologicheskaya-klinika-23.ru/]http://narkologicheskaya-klinika-23.ru/[/url] .
narkologicheskaya klinika_gcet
26 Oct 25 at 9:20 pm
libracryptocurrency.pw – Navigation felt smooth, found everything quickly without any confusing steps.
Val Wolter
26 Oct 25 at 9:20 pm
кракен ios
kraken ссылка
Henryamerb
26 Oct 25 at 9:21 pm
кракен онион
kraken qr code
Henryamerb
26 Oct 25 at 9:21 pm
Где вызвать обработка от педикулеза круглосуточно? Время позднее.
уничтожение постельных клопов
KennethceM
26 Oct 25 at 9:22 pm
Thanks for your personal marvelous posting!
I certainly enjoyed reading it, you could be
a great author. I will make certain to bookmark your blog
and will eventually come back from now on. I want to encourage you to continue your great work, have a nice evening!
ankara kürtaj
26 Oct 25 at 9:23 pm
поставщик медоборудования [url=https://medoborudovanie-postavka.ru/]medoborudovanie-postavka.ru[/url] .
postavka medicinskogo oborydovaniya_rzsn
26 Oct 25 at 9:25 pm
1xbet g?ncel adres [url=https://1xbet-14.com/]1xbet g?ncel adres[/url] .
1xbet_rdet
26 Oct 25 at 9:26 pm
kraken market
kraken vk6
Henryamerb
26 Oct 25 at 9:26 pm
makelifebetter – Great deals here, always find exactly what I need quickly.
Roseline Keliipaakaua
26 Oct 25 at 9:27 pm
кракен Москва
kraken darknet
JamesDaync
26 Oct 25 at 9:29 pm
наркологические услуги [url=https://narkologicheskaya-klinika-23.ru/]https://narkologicheskaya-klinika-23.ru/[/url] .
narkologicheskaya klinika_woet
26 Oct 25 at 9:31 pm
медтехника [url=https://medicinskaya-tehnika.ru]https://medicinskaya-tehnika.ru[/url] .
medicinskaya tehnika_tcEi
26 Oct 25 at 9:32 pm
кракен vk2
kraken qr code
Henryamerb
26 Oct 25 at 9:32 pm
1xbet giri? linki [url=1xbet-17.com]1xbet giri? linki[/url] .
1xbet_lfpl
26 Oct 25 at 9:34 pm
FindYourPerfectDeal – Deals are updated frequently, site is reliable and trustworthy overall.
Bret Tusler
26 Oct 25 at 9:34 pm
Hello it’s me, I am also visiting this site daily,
this web page is really fastidious and the viewers are actually sharing nice thoughts.
Giết người hiếp dâm
26 Oct 25 at 9:35 pm
Have you ever thought about adding a little bit more than just your articles?
I mean, what you say is important and everything.
Nevertheless imagine if you added some great photos or
videos to give your posts more, “pop”! Your content is
excellent but with images and video clips, this website could definitely be
one of the very best in its field. Terrific blog!
mgmarket7 at
26 Oct 25 at 9:37 pm
1xbet turkiye [url=https://1xbet-14.com/]1xbet-14.com[/url] .
1xbet_laet
26 Oct 25 at 9:37 pm
TheBestPlaceToShop – I keep returning here, always find interesting products quickly and easily.
Loan Hoit
26 Oct 25 at 9:38 pm
Нужна обработка от клопов с гарантией для автомобиля, посоветуйте.
уничтожение тараканов
KennethceM
26 Oct 25 at 9:38 pm
kraken СПб
кракен ссылка
Henryamerb
26 Oct 25 at 9:39 pm
поставка медоборудования [url=www.medoborudovanie-postavka.ru]www.medoborudovanie-postavka.ru[/url] .
postavka medicinskogo oborydovaniya_gbsn
26 Oct 25 at 9:40 pm
кракен android
кракен qr код
Henryamerb
26 Oct 25 at 9:40 pm
http://arskland.ru/people/user/25891/
Jameslef
26 Oct 25 at 9:40 pm
Greetings! Very helpful advice in this particular article!
It’s the little changes that will make the most important changes.
Thanks for sharing!
seo
26 Oct 25 at 9:42 pm
https://halloweencrate.shop/2025/09/28/the-best-online-qr-code-creator-tool/
Jameslef
26 Oct 25 at 9:42 pm
kraken вход
kraken ios
JamesDaync
26 Oct 25 at 9:43 pm
changeyourworld – Customer support was helpful, answered all my questions promptly and politely.
Maranda Eisenhart
26 Oct 25 at 9:43 pm
TrendyFindsHub – Love the variety of products, always find what I need quickly.
Lolita Weishaupt
26 Oct 25 at 9:44 pm
ShopAndSmileAlways – Fast shipping, items arrived in perfect condition as expected online.
Paige Butron
26 Oct 25 at 9:44 pm
startsomethinggreat – Fast shipping, products arrived on time and in perfect condition today.
Chung Zollicoffer
26 Oct 25 at 9:46 pm
1 x bet giri? [url=https://1xbet-14.com/]https://1xbet-14.com/[/url] .
1xbet_yget
26 Oct 25 at 9:48 pm
медицинская техника [url=http://medicinskaya-tehnika.ru/]медицинская техника[/url] .
medicinskaya tehnika_awEi
26 Oct 25 at 9:49 pm
learnshareconnect – Great platform for connecting and learning from diverse perspectives.
Jackie Cutchember
26 Oct 25 at 9:50 pm
номер наркологии [url=http://narkologicheskaya-klinika-23.ru]http://narkologicheskaya-klinika-23.ru[/url] .
narkologicheskaya klinika_ymet
26 Oct 25 at 9:50 pm
worldcomputerexpo.com – Color palette felt calming, nothing distracting, just focused, thoughtful design.
Louella Crutchfield
26 Oct 25 at 9:50 pm