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://elektricheskie-rulonnye-shtory15.ru/]elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_uxEi
15 Sep 25 at 11:04 pm
электрокарнизы для штор [url=https://karniz-s-elektroprivodom-kupit.ru]https://karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_vvEr
15 Sep 25 at 11:05 pm
рольшторы на окна купить [url=http://www.avtomaticheskie-rulonnye-shtory5.ru]рольшторы на окна купить[/url] .
avtomaticheskie rylonnie shtori_cpsr
15 Sep 25 at 11:06 pm
Сайт о ремонте https://e-proficom.ru полезные статьи, пошаговые инструкции и советы экспертов. От выбора материалов до дизайна интерьеров. Всё, что нужно для ремонта квартир и домов.
JacobRig
15 Sep 25 at 11:09 pm
internet apotheke: Blau Kraft De – online apotheke deutschland
Israelpaync
15 Sep 25 at 11:09 pm
автоматический карниз для штор [url=https://karniz-s-elektroprivodom-kupit.ru]https://karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_nfEr
15 Sep 25 at 11:09 pm
Сайт для женщин https://devchenky.ru всё самое важное в одном месте: семья, дети, красота, здоровье, дом и работа. Советы специалистов, лайфхаки и вдохновение на каждый день.
EverettWes
15 Sep 25 at 11:09 pm
рулонные шторы на широкое окно [url=https://elektricheskie-rulonnye-shtory15.ru]https://elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_szEi
15 Sep 25 at 11:10 pm
Do you mind if I quote a couple of your posts as long as I provide
credit and sources back to your weblog? My website is in the very same area of interest as yours and my users
would genuinely benefit from a lot of the information you provide here.
Please let me know if this alright with you.
Thanks!
canadian pharmacy online
15 Sep 25 at 11:11 pm
I like what you guys are usually up too. Such clever work and coverage!
Keep up the fantastic works guys I’ve added you guys to my personal
blogroll.
live draw sgp
15 Sep 25 at 11:11 pm
рулонные шторы на окна цена [url=avtomaticheskie-rulonnye-shtory5.ru]avtomaticheskie-rulonnye-shtory5.ru[/url] .
avtomaticheskie rylonnie shtori_ahsr
15 Sep 25 at 11:12 pm
It’s hard to come by well-informed people on this subject, but you sound like you know what you’re talking
about! Thanks
Mehdi Deneme bonusu veren siteler
15 Sep 25 at 11:13 pm
производители рулонных штор [url=https://www.elektricheskie-rulonnye-shtory15.ru]https://www.elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_cwEi
15 Sep 25 at 11:13 pm
Блог о ремонте https://ivinstrument.ru полезные статьи, пошаговые инструкции и советы экспертов. Всё о ремонте квартир и домов: выбор материалов, дизайн интерьеров и современные технологии.
JamesAllet
15 Sep 25 at 11:13 pm
автоматические гардины для штор [url=https://www.karniz-s-elektroprivodom-kupit.ru]https://www.karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_kmEr
15 Sep 25 at 11:14 pm
Что делаем
Ознакомиться с деталями – [url=https://narkolog-na-dom-zhukovskij7.ru/]нарколог на дом цена[/url]
CharlesNeacy
15 Sep 25 at 11:15 pm
купить рулонные шторы москва [url=http://www.avtomaticheskie-rulonnye-shtory5.ru]купить рулонные шторы москва[/url] .
avtomaticheskie rylonnie shtori_uksr
15 Sep 25 at 11:15 pm
карниз с приводом для штор [url=http://karnizy-s-elektroprivodom-cena.ru]http://karnizy-s-elektroprivodom-cena.ru[/url] .
karnizi s elektroprivodom cena_zjkr
15 Sep 25 at 11:16 pm
рулонные шторы на окно в кухне [url=www.elektricheskie-rulonnye-shtory15.ru]www.elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_veEi
15 Sep 25 at 11:17 pm
I have been surfing online more than 3 hours today,
yet I never found any interesting article like yours. It is pretty
worth enough for me. In my opinion, if all web owners and bloggers made good content as you did, the internet will
be much more useful than ever before.
penis enlargement
15 Sep 25 at 11:17 pm
автоматические гардины для штор [url=www.karnizy-s-elektroprivodom-cena.ru/]www.karnizy-s-elektroprivodom-cena.ru/[/url] .
karnizi s elektroprivodom cena_lfkr
15 Sep 25 at 11:20 pm
электрическая рулонная штора [url=http://avtomaticheskie-rulonnye-shtory5.ru/]http://avtomaticheskie-rulonnye-shtory5.ru/[/url] .
avtomaticheskie rylonnie shtori_bpsr
15 Sep 25 at 11:20 pm
Процедура начинается с осмотра и сбора анамнеза. После этого специалист проводит экстренную детоксикацию, снимает симптомы абстинентного синдрома, назначает поддерживающую терапию и даёт рекомендации по дальнейшим шагам. По желанию родственников или самого пациента помощь может быть оказана и в условиях стационара клиники.
Выяснить больше – http://narkologicheskaya-pomoshch-domodedovo6.ru/kruglosutochnaya-narkologicheskaya-pomoshch-v-domodedovo/
Davidcratt
15 Sep 25 at 11:20 pm
электрокарнизы для штор [url=http://karnizy-s-elektroprivodom-cena.ru]электрокарнизы для штор[/url] .
karnizi s elektroprivodom cena_vkkr
15 Sep 25 at 11:22 pm
Thanks for sharing your thoughts on FintrexPrime 2.9.
Regards
FintrexPrime 2.9
15 Sep 25 at 11:24 pm
I am sure this piece of writing has touched all the internet
users, its really really fastidious post on building up new blog.
best online gambling sites
15 Sep 25 at 11:25 pm
I’m extremely inspired together with your writing abilities and also with the layout to your blog.
Is that this a paid topic or did you customize it yourself?
Either way stay up the nice high quality writing,
it is uncommon to peer a nice weblog like this one nowadays..
casino utan spelpaus
15 Sep 25 at 11:26 pm
электрокарниз недорого [url=karniz-s-elektroprivodom-kupit.ru]karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_ztEr
15 Sep 25 at 11:27 pm
Generika Potenzmittel rezeptfrei online kaufen [url=https://intimgesund.com/#]kamagra kaufen ohne rezept online[/url] kamagra erfahrungen deutschland
StevenTilia
15 Sep 25 at 11:27 pm
Близкий человек в запое? Не ждите ухудшения. Обратитесь в клинику — здесь проведут профессиональный вывод из запоя с последующим восстановлением организма.
Получить больше информации – [url=https://vyvod-iz-zapoya-krasnodar12.ru/]запой нарколог на дом город краснодар[/url]
Jimmyhed
15 Sep 25 at 11:29 pm
электрокарниз [url=www.karniz-s-elektroprivodom-kupit.ru/]www.karniz-s-elektroprivodom-kupit.ru/[/url] .
karniz s elektroprivodom kypit_nrEr
15 Sep 25 at 11:30 pm
купить электрические рулонные шторы [url=www.elektricheskie-rulonnye-shtory15.ru/]www.elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_zfEi
15 Sep 25 at 11:31 pm
электрические гардины для штор [url=http://karniz-s-elektroprivodom-kupit.ru/]http://karniz-s-elektroprivodom-kupit.ru/[/url] .
karniz s elektroprivodom kypit_koEr
15 Sep 25 at 11:32 pm
constantly i used to read smaller articles that as well clear their motive, and that is also happening with
this piece of writing which I am reading here.
آدرس دانشگاه امیرکبیر (پلی تکنیک تهران)
15 Sep 25 at 11:33 pm
рулонные шторы на заказ цена [url=http://www.elektricheskie-rulonnye-shtory15.ru]http://www.elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_ecEi
15 Sep 25 at 11:34 pm
электрические рулонные жалюзи [url=https://avtomaticheskie-rulonnye-shtory5.ru/]https://avtomaticheskie-rulonnye-shtory5.ru/[/url] .
avtomaticheskie rylonnie shtori_dvsr
15 Sep 25 at 11:34 pm
Greate article. Keep writing such kind of information on your blog.
Im really impressed by your blog.
Hi there, You have done an incredible job. I’ll definitely digg it
and in my view recommend to my friends. I’m confident they’ll be benefited from this
web site.
Vodka зеркало
15 Sep 25 at 11:36 pm
рулонные шторы электрические [url=https://elektricheskie-rulonnye-shtory15.ru]https://elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_wxEi
15 Sep 25 at 11:36 pm
Hello to all, how is everything, I think every one
is getting more from this website, and your views are nice in support of new users.
eSEOspace
15 Sep 25 at 11:36 pm
автоматические рулонные шторы с электроприводом на окна [url=https://avtomaticheskie-rulonnye-shtory5.ru/]автоматические рулонные шторы с электроприводом на окна[/url] .
avtomaticheskie rylonnie shtori_czsr
15 Sep 25 at 11:37 pm
Мега даркнет Мега сайт Мега онион Мега ссылка Mega даркнет Mega сайт Mega онион Mega ссылка Mega darknet Mega onion
RichardPep
15 Sep 25 at 11:39 pm
квартира на сутки Лида https://www.dattainc.com.mx/2025/09/10/snjat-2-komnatnuju-kvartiru-na-sutki-lida-ul/
https://www.dattainc.com.mx/2025/09/10/snjat-2-komnatnuju-kvartiru-na-sutki-lida-ul/
15 Sep 25 at 11:39 pm
рольшторы заказать [url=https://www.avtomaticheskie-rulonnye-shtory5.ru]рольшторы заказать[/url] .
avtomaticheskie rylonnie shtori_rlsr
15 Sep 25 at 11:40 pm
gdfplay 15 euro bonus ohne einzahlung Megapari Casino
RonaldDuh
15 Sep 25 at 11:41 pm
https://vgarderobe.ru/dlinnyi-vyazanyi-zhaket-melrose-g-25938-2.html
StanleyToumb
15 Sep 25 at 11:44 pm
I’m really enjoying the theme/design of your website. Do you ever run into any web browser
compatibility issues? A couple of my blog readers have complained about my
site not working correctly in Explorer but looks great in Firefox.
Do you have any advice to help fix this issue?
jilbab hot
15 Sep 25 at 11:44 pm
Сайт для женщин https://devchenky.ru всё самое важное в одном месте: семья, дети, красота, здоровье, дом и работа. Советы специалистов, лайфхаки и вдохновение на каждый день.
EverettWes
15 Sep 25 at 11:47 pm
Сайт о ремонте https://e-proficom.ru полезные статьи, пошаговые инструкции и советы экспертов. От выбора материалов до дизайна интерьеров. Всё, что нужно для ремонта квартир и домов.
JacobRig
15 Sep 25 at 11:47 pm
электрические гардины [url=http://www.karnizy-s-elektroprivodom-cena.ru]электрические гардины[/url] .
karnizi s elektroprivodom cena_hgkr
15 Sep 25 at 11:49 pm
квартира на сутки Лида https://9mail.new2new.com/?p=258
https://9mail.new2new.com/?p=258
15 Sep 25 at 11:49 pm