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://www.frei-diplom2.ru]как купить диплом проведенный[/url] .
Diplomi_xlEa
28 Oct 25 at 10:53 am
диплом автодорожного техникума купить [url=https://frei-diplom9.ru/]диплом автодорожного техникума купить[/url] .
Diplomi_yfea
28 Oct 25 at 10:54 am
платная наркологическая клиника [url=http://narkologicheskaya-klinika-27.ru/]платная наркологическая клиника[/url] .
narkologicheskaya klinika_eipl
28 Oct 25 at 10:55 am
Удобство и стиль в вашем доме с [url=https://rulonnye-shtory-s-avtomatikoy.ru/]автоматические рулонные шторы с электроприводом Prokarniz[/url], которые легко управляются одним нажатием кнопки.
Автоматические рулонные шторы набирают популярность среди современных интерьеров. Они не только привносят эстетику в помещение, но и обеспечивают комфорт .
Удобное дистанционное управление — одно из главных достоинств автоматических рулонных штор. Настроить освещение в комнате возможно, оставаясь на месте.
Кроме того, автоматические рулонные шторы могут быть оборудованы различными датчиками . Это позволяет им реагировать на изменения освещенности и температуры .
Наконец, стоит упомянуть о том, что автоматические рулонные шторы доступны в различных дизайнах и цветах . Выбор подходит под любой стиль интерьера .
автоматические рулонные шторы цена Prokarniz
28 Oct 25 at 10:55 am
купить диплом бурильщика [url=www.rudik-diplom3.ru/]www.rudik-diplom3.ru/[/url] .
Diplomi_ofei
28 Oct 25 at 10:56 am
купить диплом с занесением в реестр чита [url=https://frei-diplom4.ru]https://frei-diplom4.ru[/url] .
Diplomi_qvOl
28 Oct 25 at 10:56 am
ремонт подвального помещения [url=gidroizolyaciya-podvala-cena.ru]ремонт подвального помещения[/url] .
gidroizolyaciya podvala cena_icKt
28 Oct 25 at 10:56 am
This is the perfect website for anybody who wishes to find out about this topic.
You realize so much its almost tough to argue with you (not that I personally will need to…HaHa).
You definitely put a fresh spin on a subject that has been discussed for many years.
Wonderful stuff, just wonderful!
advantages of b2b e commerce
28 Oct 25 at 10:57 am
кракен 2025
kraken darknet
Henryamerb
28 Oct 25 at 10:57 am
в кемерово откроетесь?? https://psp-ekb.ru когда появится новинки??
JasonBoomi
28 Oct 25 at 10:57 am
оснащение конференц залов
Bryanfloky
28 Oct 25 at 10:57 am
гидроизоляция подвала снаружи цена [url=www.gidroizolyaciya-podvala-cena.ru/]www.gidroizolyaciya-podvala-cena.ru/[/url] .
gidroizolyaciya podvala cena_atKt
28 Oct 25 at 10:58 am
наркологические клиники в москве [url=www.narkologicheskaya-klinika-25.ru]наркологические клиники в москве[/url] .
narkologicheskaya klinika_uyPl
28 Oct 25 at 10:58 am
раскрутка и продвижение сайта [url=https://optimizaciya-i-seo-prodvizhenie-sajtov-moskva-1.ru]раскрутка и продвижение сайта[/url] .
optimizaciya i seo prodvijenie saitov moskva_jnPi
28 Oct 25 at 10:58 am
гидроизоляция цена за рулон [url=gidroizolyaciya-cena-7.ru]гидроизоляция цена за рулон[/url] .
gidroizolyaciya cena_gnSi
28 Oct 25 at 10:58 am
https://odessaforum.getbb.ru/viewtopic.php?f=5&t=26616
MichaelKib
28 Oct 25 at 11:00 am
купить диплом в старом осколе [url=https://www.rudik-diplom10.ru]купить диплом в старом осколе[/url] .
Diplomi_paSa
28 Oct 25 at 11:00 am
стоимость гидроизоляции подвала [url=https://gidroizolyaciya-cena-7.ru]стоимость гидроизоляции подвала[/url] .
gidroizolyaciya cena_adSi
28 Oct 25 at 11:00 am
купить диплом колледжа с занесением в реестр [url=http://frei-diplom2.ru]купить диплом колледжа с занесением в реестр[/url] .
Diplomi_dkEa
28 Oct 25 at 11:00 am
Цифровые решения экономят время кракен онион зеркало kraken зеркало рабочее актуальные зеркала kraken kraken сайт зеркала
RichardPep
28 Oct 25 at 11:00 am
купить диплом вуза с реестром [url=https://frei-diplom6.ru]купить диплом вуза с реестром[/url] .
Diplomi_vsOl
28 Oct 25 at 11:01 am
купить свидетельство о разводе [url=http://www.rudik-diplom4.ru]купить свидетельство о разводе[/url] .
Diplomi_lbOr
28 Oct 25 at 11:01 am
наркологическая клиника [url=www.narkologicheskaya-klinika-27.ru]наркологическая клиника[/url] .
narkologicheskaya klinika_ufpl
28 Oct 25 at 11:02 am
купить диплом в череповце [url=https://rudik-diplom9.ru/]https://rudik-diplom9.ru/[/url] .
Diplomi_qbei
28 Oct 25 at 11:02 am
Hi there, I enjoy reading through your article. I like to write a little comment
to support you.
Finozanetix TEST
28 Oct 25 at 11:03 am
кракен vk2
kraken vk3
Henryamerb
28 Oct 25 at 11:03 am
Заказать диплом о высшем образовании можем помочь. Купить диплом Ростов-на-Дону – [url=http://diplomybox.com/kupit-diplom-rostov-na-donu/]diplomybox.com/kupit-diplom-rostov-na-donu[/url]
Cazrwqt
28 Oct 25 at 11:04 am
я купил проведенный диплом [url=www.frei-diplom2.ru]я купил проведенный диплом[/url] .
Diplomi_gyEa
28 Oct 25 at 11:04 am
Главное отличие квартала — его масштаб,
который позволяет создать уникальный оазис в центре Москвы, где проживают люди одного уровня, достатка и социального статуса.
https://dusharossii.borda.ru/?1-3-0-00001250-000-0-0
28 Oct 25 at 11:04 am
купить диплом техникума в магнитогорске [url=http://frei-diplom8.ru/]купить диплом техникума в магнитогорске[/url] .
Diplomi_yjsr
28 Oct 25 at 11:05 am
подвал дома ремонт [url=http://gidroizolyaciya-podvala-cena.ru]http://gidroizolyaciya-podvala-cena.ru[/url] .
gidroizolyaciya podvala cena_jkKt
28 Oct 25 at 11:05 am
Все версии нейросети понимают
русский язык и доступны абсолютно бесплатно.
https://zdorovie.forum24.ru/?1-8-0-00000043-000-0-0
28 Oct 25 at 11:06 am
https://invasovet.myqip.ru/?1-2-0-00002275-000-0-0-1758302920
MichaelKib
28 Oct 25 at 11:06 am
купить диплом украины с занесением в реестр [url=www.frei-diplom6.ru/]www.frei-diplom6.ru/[/url] .
Diplomi_rxOl
28 Oct 25 at 11:07 am
ремонт в подвале [url=www.gidroizolyaciya-cena-7.ru]www.gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_mbSi
28 Oct 25 at 11:07 am
купить диплом физика [url=www.rudik-diplom10.ru]купить диплом физика[/url] .
Diplomi_bvSa
28 Oct 25 at 11:07 am
торкрет бетон цена [url=https://torkretirovanie-1.ru/]торкрет бетон цена[/url] .
torkretirovanie_hfen
28 Oct 25 at 11:08 am
Oh my goodness! Amazing article dude! Thank you, However I am going through troubles with your RSS.
I don’t know the reason why I am unable to join it. Is there anyone else getting identical RSS issues?
Anybody who knows the solution will you kindly respond?
Thanx!!
Fuentoro Ai
28 Oct 25 at 11:08 am
ремонт стекла Ремонт стекол – это комплекс процедур, направленных на восстановление целостности и прозрачности автомобильных стекол. Он включает в себя ремонт сколов, трещин, а также полировку для устранения царапин и потертостей. Своевременный ремонт позволяет избежать дорогостоящей замены стекла и обеспечивает безопасность вождения.
JosephNic
28 Oct 25 at 11:09 am
Купить диплом колледжа в Одесса [url=https://www.educ-ua7.ru]https://www.educ-ua7.ru[/url] .
Diplomi_xtea
28 Oct 25 at 11:09 am
лед экран
Marvinreoky
28 Oct 25 at 11:09 am
центр наркологии москва [url=www.narkologicheskaya-klinika-27.ru]центр наркологии москва[/url] .
narkologicheskaya klinika_pqpl
28 Oct 25 at 11:10 am
Эта информационная статья содержит полезные факты, советы и рекомендации, которые помогут вам быть в курсе последних тенденций и изменений в выбранной области. Материал составлен так, чтобы быть полезным и понятным каждому.
Это ещё не всё… – https://level-twentynine.com/2023/05/10/hello-world
Martinkem
28 Oct 25 at 11:11 am
купить диплом с внесением в реестр [url=http://www.rudik-diplom11.ru]купить диплом с внесением в реестр[/url] .
Diplomi_ljMi
28 Oct 25 at 11:11 am
купить аттестаты за 9 [url=https://rudik-diplom7.ru]купить аттестаты за 9[/url] .
Diplomi_gsPl
28 Oct 25 at 11:11 am
купить диплом с проводкой меня [url=www.frei-diplom4.ru]купить диплом с проводкой меня[/url] .
Diplomi_jwOl
28 Oct 25 at 11:11 am
как купить диплом о высшем образовании с занесением в реестр [url=http://www.frei-diplom6.ru]как купить диплом о высшем образовании с занесением в реестр[/url] .
Diplomi_loOl
28 Oct 25 at 11:11 am
купить диплом агронома [url=http://www.rudik-diplom3.ru]купить диплом агронома[/url] .
Diplomi_oyei
28 Oct 25 at 11:11 am
kraken vk2
кракен vpn
Henryamerb
28 Oct 25 at 11:12 am
купить диплом в новоуральске [url=https://rudik-diplom10.ru/]https://rudik-diplom10.ru/[/url] .
Diplomi_jnSa
28 Oct 25 at 11:13 am