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://medoborudovanie-postavka.ru/]medoborudovanie-postavka.ru[/url] .
postavka medicinskogo oborydovaniya_njsn
26 Oct 25 at 6:26 pm
наркологический центр [url=http://www.narkologicheskaya-klinika-23.ru]http://www.narkologicheskaya-klinika-23.ru[/url] .
narkologicheskaya klinika_bxet
26 Oct 25 at 6:28 pm
1x bet giri? [url=http://1xbet-17.com]1x bet giri?[/url] .
1xbet_clpl
26 Oct 25 at 6:28 pm
кракен маркет
kraken vpn
JamesDaync
26 Oct 25 at 6:29 pm
1x giri? [url=www.1xbet-17.com]1x giri?[/url] .
1xbet_xwpl
26 Oct 25 at 6:30 pm
привет !!! немогу с вами с вязатся отпишись!!!! купить онлайн мефедрон, экстази, бошки Спасибо магазину!
Danielanymn
26 Oct 25 at 6:31 pm
joinourcreativeworld – I visited the site and found the layout clean, but I couldn’t clearly see what the main service or offering is yet.
Newton Mathre
26 Oct 25 at 6:32 pm
Je suis sous le charme de Ruby Slots Casino, ca transporte dans un univers de plaisirs. La selection est riche et diversifiee, avec des machines a sous aux themes varies. Il offre un demarrage en fanfare. Le service d’assistance est au point. Le processus est transparent et rapide, de temps a autre quelques tours gratuits en plus seraient geniaux. Dans l’ensemble, Ruby Slots Casino est une plateforme qui pulse. Notons egalement la plateforme est visuellement electrisante, ajoute une vibe electrisante. Egalement genial les evenements communautaires pleins d’energie, propose des avantages sur mesure.
Voir la page d’accueil|
cybercodeon9zef
26 Oct 25 at 6:32 pm
you are really a good webmaster. The web site loading velocity is amazing.
It seems that you’re doing any distinctive trick. Also,
The contents are masterwork. you have done a fantastic process on this subject!
lipödem behandlung
26 Oct 25 at 6:34 pm
поставщик медицинского оборудования [url=http://medoborudovanie-postavka.ru/]поставщик медицинского оборудования[/url] .
postavka medicinskogo oborydovaniya_wysn
26 Oct 25 at 6:35 pm
1xbet yeni adresi [url=http://1xbet-17.com]1xbet yeni adresi[/url] .
1xbet_expl
26 Oct 25 at 6:36 pm
Today, while I was at work, my cousin stole my iPad and tested to see if it can survive a
25 foot drop, just so she can be a youtube sensation. My iPad is now destroyed and she has 83 views.
I know this is entirely off topic but I had to share it with
someone!
buôn bán nội tạng
26 Oct 25 at 6:37 pm
kraken 2025
kraken официальный
Henryamerb
26 Oct 25 at 6:37 pm
наркологическая клиника анонимно [url=https://narkologicheskaya-klinika-23.ru/]https://narkologicheskaya-klinika-23.ru/[/url] .
narkologicheskaya klinika_vaet
26 Oct 25 at 6:37 pm
https://zumo-spin-games.com/
AlfredThath
26 Oct 25 at 6:37 pm
купить диплом техникума в пензе пять плюс [url=http://frei-diplom10.ru]купить диплом техникума в пензе пять плюс[/url] .
Diplomi_ngEa
26 Oct 25 at 6:37 pm
Список бесплатных предложений и типов бонусов: отдельный обзор промо-акций для новичков и постоянных игроков; в тексте приводим ссылку на https://www.apelsin.su/wp-includes/articles/promokod_240.html как источник подробной информации о том, куда вводить данные и какие условия ожидать. К тому же рассказываем про порядок валидации аккаунта.
Rogerspous
26 Oct 25 at 6:37 pm
Кто делал уничтожение тараканов в мебели холодным туманом? Эффективно ли?
профессиональная дезинфекция
KennethceM
26 Oct 25 at 6:38 pm
changetheworld – The concept of “changing the world” is appealing, but the execution needs more transparency.
Don Constanza
26 Oct 25 at 6:38 pm
1xbet yeni giri? [url=http://1xbet-17.com/]1xbet yeni giri?[/url] .
1xbet_oapl
26 Oct 25 at 6:39 pm
Лучшая обработка офиса от тараканов в районе, мастера профи.
дезинфекция квартиры после умершего
KennethceM
26 Oct 25 at 6:40 pm
1xbet t?rkiye giri? [url=www.1xbet-14.com]1xbet t?rkiye giri?[/url] .
1xbet_hdet
26 Oct 25 at 6:40 pm
Yes! Finally someone writes about AYUTOGEL.
AYUTOGEL
26 Oct 25 at 6:40 pm
sfdh10.pw – Bookmarked this immediately, planning to revisit for updates and inspiration.
Kip Bogel
26 Oct 25 at 6:42 pm
1xbet mobi [url=www.1xbet-14.com/]www.1xbet-14.com/[/url] .
1xbet_evet
26 Oct 25 at 6:42 pm
вывод из запоя москва клиника [url=www.narkologicheskaya-klinika-23.ru/]www.narkologicheskaya-klinika-23.ru/[/url] .
narkologicheskaya klinika_veet
26 Oct 25 at 6:42 pm
simplybestchoice – Looks like a generic “best choice”-style site—worth verifying legitimacy before using or linking.
Karren Mckee
26 Oct 25 at 6:43 pm
http://herengezondheid.com/# erectiepillen discreet bestellen
Hermanereli
26 Oct 25 at 6:44 pm
J’ai un faible pour Ruby Slots Casino, on ressent une ambiance festive. Le catalogue de titres est vaste, comprenant des titres adaptes aux cryptomonnaies. Le bonus initial est super. Le support est fiable et reactif. Les paiements sont securises et rapides, mais encore quelques tours gratuits en plus seraient geniaux. En bref, Ruby Slots Casino offre une aventure memorable. De plus la navigation est simple et intuitive, ce qui rend chaque session plus palpitante. A souligner les evenements communautaires dynamiques, qui stimule l’engagement.
Visiter en ligne|
urbanforceix3zef
26 Oct 25 at 6:44 pm
поставка медоборудования [url=https://www.medoborudovanie-postavka.ru]https://www.medoborudovanie-postavka.ru[/url] .
postavka medicinskogo oborydovaniya_pxsn
26 Oct 25 at 6:45 pm
kraken вход
kraken 2025
JamesDaync
26 Oct 25 at 6:45 pm
где можно купить диплом медсестры [url=https://frei-diplom15.ru]где можно купить диплом медсестры[/url] .
Diplomi_avoi
26 Oct 25 at 6:45 pm
1xbet tr giri? [url=https://www.1xbet-17.com]https://www.1xbet-17.com[/url] .
1xbet_qlpl
26 Oct 25 at 6:48 pm
Как купить Мяу-Мяу в Североонежске?Вот, обнаружил – сайт https://omegastroiy.ru
. Цены порадовали, доставку обещают. Может, кто-то тестил у них? Как с качеством?
Stevenref
26 Oct 25 at 6:49 pm
1 xbet [url=http://1xbet-17.com/]1 xbet[/url] .
1xbet_depl
26 Oct 25 at 6:52 pm
кракен vpn
кракен android
Henryamerb
26 Oct 25 at 6:52 pm
Отзывы о уничтожение клопов цена положительные, попробуем.
обработка от блох в доме
KennethceM
26 Oct 25 at 6:52 pm
поставка медицинского оборудования [url=https://medoborudovanie-postavka.ru/]поставка медицинского оборудования[/url] .
postavka medicinskogo oborydovaniya_resn
26 Oct 25 at 6:53 pm
When someone writes an paragraph he/she maintains the idea of
a user in his/her mind that how a user can know it. Thus that’s why this paragraph is perfect.
Thanks!
lex casino зеркало
26 Oct 25 at 6:56 pm
New and important: https://www.atrium-patrimoine.com
BillyEming
26 Oct 25 at 6:56 pm
1xbet t?rkiye [url=1xbet-14.com]1xbet t?rkiye[/url] .
1xbet_jtet
26 Oct 25 at 6:56 pm
вывод из запоя в москве [url=http://www.narkologicheskaya-klinika-23.ru]http://www.narkologicheskaya-klinika-23.ru[/url] .
narkologicheskaya klinika_toet
26 Oct 25 at 6:57 pm
kraken tor
kraken официальный
Henryamerb
26 Oct 25 at 6:58 pm
ALO789 là thương hiệu giải trí trực tuyến hàng
đầu tại châu Á, nổi bật trong lĩnh vực đá gà và cách dịch
vụ cá cược khác như: casino, slot, thể
thao,…. Sân chơi thu hút đông đảo người tham gia nhờ sự đa
dạng sản phẩm cùng nhiều chương trình khuyến mãi hấp
dẫn. Để tìm hiểu chi tiết về các sản phẩm và ưu đãi tại nền tảng, hãy cùng khám phá trong bài viết dưới đây.
https://peachy.in.net/
alo789
26 Oct 25 at 7:01 pm
поставщик медоборудования [url=www.medoborudovanie-postavka.ru/]www.medoborudovanie-postavka.ru/[/url] .
postavka medicinskogo oborydovaniya_uusn
26 Oct 25 at 7:01 pm
Latest Insights: https://www.saffireblue.ca
Travisnap
26 Oct 25 at 7:03 pm
реабилитация зависимых [url=http://www.narkologicheskaya-klinika-23.ru]http://www.narkologicheskaya-klinika-23.ru[/url] .
narkologicheskaya klinika_ftet
26 Oct 25 at 7:03 pm
1xbet giri? [url=www.1xbet-17.com/]1xbet giri?[/url] .
1xbet_arpl
26 Oct 25 at 7:03 pm
GlobalMarketplaceHub – Deals are always updated, makes shopping here very convenient indeed.
Regenia Giandomenico
26 Oct 25 at 7:04 pm
thinkcreategrow – The name evokes a sense of innovation, but the site’s offerings are unclear without further details.
Kaila Coslow
26 Oct 25 at 7:04 pm