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!
Hello! I’ve been reading your site for a long time now and finally got the courage to go ahead and give you a shout out from Kingwood Tx!
Just wanted to tell you keep up the excellent job!
XX88
17 Sep 25 at 11:59 am
купить диплом техникума недорого [url=www.educ-ua4.ru/]www.educ-ua4.ru/[/url] .
Diplomi_zxPl
17 Sep 25 at 12:00 pm
купить диплом о среднем [url=http://educ-ua17.ru]купить диплом о среднем[/url] .
Diplomi_bfSl
17 Sep 25 at 12:00 pm
Магазин тут! kokain mefedron gash alfa-pvp amf
всем добрый день брали 200гр тут всё на высшем уровне подробное описание 5 фоток + геоданный клад был поднят в касание место укромное тихое качество продукта вышка! советую всем брать у Чема !
AnthonygYday
17 Sep 25 at 12:03 pm
Burning Classics слот
Derekjency
17 Sep 25 at 12:03 pm
mostbet sikachat [url=mostbet12015.ru]mostbet12015.ru[/url]
mostbet_uqSr
17 Sep 25 at 12:03 pm
купить аттестат за 11 классов кемерово [url=www.arus-diplom25.ru/]www.arus-diplom25.ru/[/url] .
Diplomi_vlot
17 Sep 25 at 12:03 pm
What a material of un-ambiguity and preserveness of valuable familiarity
concerning unpredicted feelings.
suspended revoked drivers license
17 Sep 25 at 12:04 pm
купить диплом недорого [url=https://educ-ua5.ru/]купить диплом недорого[/url] .
Diplomi_lkKl
17 Sep 25 at 12:05 pm
Мы предлагаем документы институтов, расположенных в любом регионе России. Купить диплом любого ВУЗа:
[url=http://epicit.ru/poluchite-diplom-bez-ekzamenov/]купить аттестат за 11[/url]
Diplomi_fkPn
17 Sep 25 at 12:05 pm
https://mannerkraft.com/# online apotheke
Williamves
17 Sep 25 at 12:07 pm
http://gesunddirekt24.com/# medikamente rezeptfrei
EnriqueVox
17 Sep 25 at 12:08 pm
If some one needs to be updated with newest technologies
therefore he must be go to see this web site and be up to
date all the time.
تفاوت کانون وکلا و مرکز وکلا
17 Sep 25 at 12:08 pm
купить свидетельство о разводе [url=https://educ-ua4.ru/]https://educ-ua4.ru/[/url] .
Diplomi_abPl
17 Sep 25 at 12:08 pm
I’ll immediately seize your rss feed as I can not to find your e-mail subscription link or
newsletter service. Do you’ve any? Please permit me realize in order that I may
subscribe. Thanks.
best australian online pokies
17 Sep 25 at 12:08 pm
техникум какое образование украина [url=https://www.educ-ua6.ru]техникум какое образование украина[/url] .
Diplomi_vvMl
17 Sep 25 at 12:10 pm
купить диплом о высшем образовании бакалавр [url=https://www.educ-ua17.ru]купить диплом о высшем образовании бакалавр[/url] .
Diplomi_vySl
17 Sep 25 at 12:11 pm
login mostbet [url=https://mostbet12014.ru]https://mostbet12014.ru[/url]
mostbet_puKl
17 Sep 25 at 12:11 pm
I just like the helpful information you provide to your articles.
I’ll bookmark your weblog and take a look at once more right here regularly.
I am rather sure I’ll learn plenty of new stuff proper here!
Good luck for the next!
brainrot evolution script mobile
17 Sep 25 at 12:11 pm
купить аттестат 11 класса мурманск [url=https://arus-diplom25.ru/]купить аттестат 11 класса мурманск[/url] .
Diplomi_nlot
17 Sep 25 at 12:12 pm
I think this is among the most important info for me.
And i’m glad reading your article. But want to remark
on some general things, The website style is great, the
articles is really great : D. Good job, cheers
xoilac.co.uk lừa đảo công an truy quét cấm người chơi tham g
17 Sep 25 at 12:12 pm
bs2best at, bs2web at и bs2 market: глубокий анализ технологий 2025 года
blsp at
bs2best.at blacksprut marketplace Official
CharlesNarry
17 Sep 25 at 12:12 pm
купить диплом цена [url=http://www.educ-ua18.ru]купить диплом цена[/url] .
Diplomi_qdPi
17 Sep 25 at 12:13 pm
диплом бакалавра купить украина [url=https://www.educ-ua5.ru]диплом бакалавра купить украина[/url] .
Diplomi_pjKl
17 Sep 25 at 12:14 pm
купить диплом колледжа с занесением в реестр [url=https://arus-diplom33.ru]купить диплом колледжа с занесением в реестр[/url] .
Diplomi_glSa
17 Sep 25 at 12:14 pm
Your means of describing all in this paragraph is in fact good, every one
be able to without difficulty know it, Thanks a lot.
turkey visa for australian
17 Sep 25 at 12:14 pm
высшее образование купить диплом с занесением [url=http://educ-ua2.ru]высшее образование купить диплом с занесением[/url] .
Diplomi_uzOt
17 Sep 25 at 12:16 pm
It’s genuinely very complex in this active life to listen news on Television, thus I only use world wide web for that purpose, and obtain the most recent information.
https://gscpa.com/sorting-levels-assurance-financial-reporting/
17 Sep 25 at 12:16 pm
Today, I went to the beach with my children. I found a sea
shell and gave it to my 4 year old daughter and said “You can hear the ocean if you put this to your ear.” She placed the shell to her ear and screamed.
There was a hermit crab inside and it pinched her ear.
She never wants to go back! LoL I know this is totally off topic but I had to tell someone!
water damage restoration company
17 Sep 25 at 12:17 pm
Мы можем предложить документы университетов, расположенных в любом регионе России. Заказать диплом университета:
[url=http://ozmt.getbb.ru/ucp.php?mode=login/]купить аттестат 11 классов с занесением в реестр[/url]
Diplomi_ewPn
17 Sep 25 at 12:18 pm
купить аттестат [url=www.educ-ua20.ru/]купить аттестат[/url] .
Diplomi_myEn
17 Sep 25 at 12:19 pm
купить диплом занесением реестр киев [url=http://www.itbestsellers.ru/forum/user/111644]http://www.itbestsellers.ru/forum/user/111644[/url] .
Bistro i prosto priobresti diplom o visshem obrazovanii!_smkt
17 Sep 25 at 12:19 pm
купить диплом техникума в спб [url=educ-ua7.ru]купить диплом техникума в спб[/url] .
Diplomi_gmEr
17 Sep 25 at 12:19 pm
Discover ways to make a paper napkin holder on the next web
page. Continue reading to the following web page and discover ways to make a
mini message board.
cialis what does it do
17 Sep 25 at 12:21 pm
купить диплом о среднем образовании цена [url=www.educ-ua5.ru]купить диплом о среднем образовании цена[/url] .
Diplomi_juKl
17 Sep 25 at 12:21 pm
купить диплом в реестре [url=http://educ-ua13.ru]купить диплом в реестре[/url] .
Diplomi_rfpn
17 Sep 25 at 12:21 pm
I was recommended this web site via my cousin. I am now
not certain whether this publish is written by him as no one
else recognize such precise approximately my difficulty.
You are incredible! Thank you!
آدرس دانشگاه پیام نور مرکز تهران غرب
17 Sep 25 at 12:22 pm
мостбет онлайн [url=www.mostbet12015.ru]www.mostbet12015.ru[/url]
mostbet_fdSr
17 Sep 25 at 12:22 pm
купить диплом магистра недорого [url=educ-ua18.ru]educ-ua18.ru[/url] .
Diplomi_mbPi
17 Sep 25 at 12:24 pm
как купить диплом с проведением [url=www.arus-diplom33.ru/]как купить диплом с проведением[/url] .
Diplomi_nsSa
17 Sep 25 at 12:24 pm
купить диплом техникума недорого [url=http://educ-ua7.ru]купить диплом техникума недорого[/url] .
Diplomi_pyEr
17 Sep 25 at 12:26 pm
купить диплом о техническом образовании [url=https://educ-ua2.ru]купить диплом о техническом образовании[/url] .
Diplomi_mpOt
17 Sep 25 at 12:26 pm
скачать мостбет казино на андроид [url=mostbet12014.ru]mostbet12014.ru[/url]
mostbet_kkKl
17 Sep 25 at 12:26 pm
Новички в фотографии обычно стремятся к доступным и действенным рекомендациям для оперативного совершенствования кадров, а в 2025 году специалисты акцентируют базовые приемы: от освоения правила третей в композиции до проб с натуральным освещением, как советует Photography Life в материале от 28 сентября 2024 года с 25 ценными подсказками. Эти рекомендации помогают новичкам избежать распространенных ошибок, таких как неправильная экспозиция или отсутствие фокуса на объекте, и перейти от случайных кадров к осмысленным изображениям, полным эмоций и истории. Если вы только осваиваете камеру, обязательно загляните на https://fotoblog.online, где собраны статьи о знаменитых мастерах вроде Марио Тестино и практические руководства по съемке повседневных объектов. В продолжение, не забывайте о роли тренировки: фотографируйте каждый день, разбирайте кадры в приложениях типа Lightroom, и вскоре увидите улучшения, мотивирующие на свежие опыты в направлениях от ландшафтов до портретов, превращая увлечение в подлинное творчество.
jatoxilib
17 Sep 25 at 12:27 pm
http://blaukraftde.com/# medikamente rezeptfrei
Williamves
17 Sep 25 at 12:28 pm
диплом медсестры с занесением в реестр купить [url=http://mynissanleaf.ru/profile.php?id=24424]диплом медсестры с занесением в реестр купить[/url] .
Kypit diplom ob obrazovanii!_gdkt
17 Sep 25 at 12:28 pm
Купить онлайн кокаин, мефедрон, амф, альфа-пвп
новый запрет вступает через 7 дней. читай список бро,там это вещество есть)
Mathewfus
17 Sep 25 at 12:29 pm
купить старый диплом техникума [url=https://educ-ua4.ru/]купить старый диплом техникума[/url] .
Diplomi_lnPl
17 Sep 25 at 12:30 pm
купить диплом в реестре [url=https://educ-ua13.ru]купить диплом в реестре[/url] .
Diplomi_vypn
17 Sep 25 at 12:31 pm
турецкие сериалы на русском языке [url=https://kinogo-15.top/]турецкие сериалы на русском языке[/url] .
kinogo_tmsa
17 Sep 25 at 12:31 pm