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=www.educ-ua13.ru/]диплом купить в реестре[/url] .
Diplomi_jnpn
17 Sep 25 at 5:56 pm
Каждый из методов подбирается индивидуально, что позволяет учитывать медицинские показания и личные особенности пациента.
Ознакомиться с деталями – http://lechenie-alkogolizma-perm0.ru/lechenie-alkogolizma-perm-anonimno/
Timothygep
17 Sep 25 at 5:57 pm
https://xn--krken23-bn4c.com
Howardreomo
17 Sep 25 at 5:57 pm
купить дизельный генератор
Williamcem
17 Sep 25 at 5:58 pm
фильмы онлайн без подписки [url=www.kinogo-12.top/]www.kinogo-12.top/[/url] .
kinogo_rtol
17 Sep 25 at 5:59 pm
купить диплом старого образца [url=www.educ-ua20.ru/]купить диплом старого образца[/url] .
Diplomi_usEn
17 Sep 25 at 6:01 pm
свидетельство о браке купить недорого [url=https://educ-ua4.ru]https://educ-ua4.ru[/url] .
Diplomi_jvPl
17 Sep 25 at 6:01 pm
Ad Networks
Brianfub
17 Sep 25 at 6:02 pm
This is my first time visit at here and i am truly impressed to read
everthing at one place.
Redefined water damage restoration service Chicago
17 Sep 25 at 6:02 pm
https://rcsp-saratov.ru/
Travistug
17 Sep 25 at 6:02 pm
I’m not sure where you’re getting your information, but great topic.
I needs to spend some time learning more or understanding more.
Thanks for magnificent information I was looking for this info for my mission.
online casino bonus
17 Sep 25 at 6:03 pm
купить свидетельство о расторжении брака [url=educ-ua5.ru]educ-ua5.ru[/url] .
Diplomi_ndKl
17 Sep 25 at 6:04 pm
купить аттестат за 11 класс красноярск [url=http://arus-diplom25.ru/]http://arus-diplom25.ru/[/url] .
Diplomi_ddot
17 Sep 25 at 6:04 pm
смотреть сериалы новинки [url=https://www.kinogo-12.top]https://www.kinogo-12.top[/url] .
kinogo_jjol
17 Sep 25 at 6:04 pm
Thank you a lot for sharing this with all folks you actually realize what you are talking approximately!
Bookmarked. Please also visit my web site =).
We may have a hyperlink trade agreement between us
آدرس دانشگاه بینالمللی امام خمینی قزوین
17 Sep 25 at 6:05 pm
купить диплом о среднем специальном образовании с занесением в реестр [url=https://educ-ua13.ru]купить диплом о среднем специальном образовании с занесением в реестр[/url] .
Diplomi_thpn
17 Sep 25 at 6:07 pm
Hi there! This is kind of off topic but I need some guidance from an established blog.
Is it difficult to set up your own blog? I’m not very techincal but I can figure things out pretty fast.
I’m thinking about setting up my own but I’m not sure where
to start. Do you have any tips or suggestions? Appreciate it
Opulatrix
17 Sep 25 at 6:08 pm
Just wish to say your article is as surprising. The clarity in your put up is just excellent and that
i could assume you are knowledgeable on this subject.
Well with your permission allow me to clutch your RSS feed to stay up to date with drawing close post.
Thank you a million and please keep up the gratifying work.
top casino online
17 Sep 25 at 6:09 pm
I’m extremely impressed with your writing skills as well as with
the layout on your weblog. Is this a paid theme or did you modify it yourself?
Either way keep up the excellent quality writing,
it is rare to see a great blog like this one these days.
تحصیلات تکمیلی دانشگاه علم و صنعت
17 Sep 25 at 6:11 pm
исторические фильмы [url=https://kinogo-14.top/]исторические фильмы[/url] .
kinogo_bwEl
17 Sep 25 at 6:12 pm
all the time i used to read smaller articles which also clear their motive, and that is also happening with this post which I am reading at this time.
Cronetrium System
17 Sep 25 at 6:13 pm
bs2best at, bs2web at и bs2 market: глубокий анализ технологий 2025 года
bs2best at
bs2best.at blacksprut marketplace Official
CharlesNarry
17 Sep 25 at 6:13 pm
My family every time say that I am killing my time here
at net, but I know I am getting knowledge everyday by reading thes fastidious articles.
جواب امتحانات شهریور کی میاد نی نی سایت
17 Sep 25 at 6:13 pm
https://push-network-rankings.com/
Brianfub
17 Sep 25 at 6:14 pm
I really like it when people get together and share ideas.
Great site, continue the good work!
kidnap girl
17 Sep 25 at 6:16 pm
аниме смотреть онлайн [url=kinogo-14.top]аниме смотреть онлайн[/url] .
kinogo_xiEl
17 Sep 25 at 6:17 pm
https://push-network-rankings.com/
Brianfub
17 Sep 25 at 6:17 pm
диплом о высшем образовании купить в киеве [url=http://educ-ua18.ru/]http://educ-ua18.ru/[/url] .
Diplomi_snPi
17 Sep 25 at 6:17 pm
Сочетание этих направлений позволяет комплексно воздействовать на проблему зависимости и минимизировать риск рецидива.
Ознакомиться с деталями – http://narkologicheskaya-klinika-v-permi0.ru/
Gregoryabems
17 Sep 25 at 6:17 pm
Каждый этап имеет важное значение и обеспечивает долгосрочный эффект от терапии.
Получить дополнительные сведения – https://lechenie-alkogolizma-tver0.ru/lechenie-ot-alkogolizma-tver/
BrianCaupe
17 Sep 25 at 6:18 pm
купить диплом образование киеве [url=www.educ-ua4.ru/]купить диплом образование киеве[/url] .
Diplomi_whPl
17 Sep 25 at 6:19 pm
купить диплом магистра недорого [url=https://educ-ua20.ru]https://educ-ua20.ru[/url] .
Diplomi_xtEn
17 Sep 25 at 6:19 pm
кино онлайн [url=www.kinogo-12.top]кино онлайн[/url] .
kinogo_fool
17 Sep 25 at 6:20 pm
I have been surfing online more than three hours today, yet I never discovered any fascinating
article like yours. It is beautiful value enough for me.
In my view, if all website owners and bloggers made just right content material as
you did, the net might be a lot more useful than ever before.
montaż auto gaz
17 Sep 25 at 6:20 pm
смотреть мультфильмы онлайн бесплатно [url=http://kinogo-15.top/]смотреть мультфильмы онлайн бесплатно[/url] .
kinogo_qjsa
17 Sep 25 at 6:21 pm
все онлайн займы [url=http://www.zaimy-11.ru]http://www.zaimy-11.ru[/url] .
zaimi_qiPt
17 Sep 25 at 6:21 pm
купить диплом о высшем образовании цены украина [url=www.educ-ua5.ru/]купить диплом о высшем образовании цены украина[/url] .
Diplomi_qpKl
17 Sep 25 at 6:21 pm
сколько стоит аттестат 11 классов купить [url=http://arus-diplom25.ru]http://arus-diplom25.ru[/url] .
Diplomi_jkot
17 Sep 25 at 6:21 pm
https://xn--krken21-bn4c.com
Howardreomo
17 Sep 25 at 6:24 pm
турецкие сериалы на русском языке [url=http://kinogo-12.top]http://kinogo-12.top[/url] .
kinogo_mvol
17 Sep 25 at 6:24 pm
займы онлайн [url=http://www.zaimy-11.ru]http://www.zaimy-11.ru[/url] .
zaimi_vfPt
17 Sep 25 at 6:25 pm
советские фильмы смотреть онлайн бесплатно [url=www.kinogo-15.top/]советские фильмы смотреть онлайн бесплатно[/url] .
kinogo_fgsa
17 Sep 25 at 6:25 pm
купить диплом в киеве недорого [url=http://educ-ua18.ru]http://educ-ua18.ru[/url] .
Diplomi_pyPi
17 Sep 25 at 6:25 pm
Write more, thats all I have to say. Literally, it seems as though
you relied on the video to make your point. You clearly know what
youre talking about, why waste your intelligence on just posting
videos to your site when you could be giving us something informative to read?
اعتراض امتحانات نهایی پایه دوازدهم
17 Sep 25 at 6:27 pm
смотреть фильмы онлайн [url=http://www.kinogo-14.top]смотреть фильмы онлайн[/url] .
kinogo_cxEl
17 Sep 25 at 6:27 pm
weed in prague buy coke in prague
prague-drugs-517
17 Sep 25 at 6:27 pm
фильмы в хорошем качестве [url=http://www.kinogo-12.top]http://www.kinogo-12.top[/url] .
kinogo_mhol
17 Sep 25 at 6:27 pm
Je suis captive par Casinia Casino, ca degage une ambiance de jeu aussi noble qu’un tournoi de chevaliers. est une legende de sensations qui enchante. proposant des slots de casino a theme medieval. Le personnel du casino offre un accompagnement digne d’un ecuyer. resonnant comme une legende parfaite. arrivent comme un tournoi. par moments les offres du casino pourraient etre plus genereuses. Globalement, Casinia Casino est un joyau pour les fans de casino pour ceux qui cherchent l’adrenaline royale du casino! De surcroit la plateforme du casino brille par son style epique. cadence l’aventure avec une rhapsodie epique.
casinia casino logo|
shadowwhirllynx2zef
17 Sep 25 at 6:28 pm
купить диплом бакалавра [url=https://educ-ua20.ru/]купить диплом бакалавра[/url] .
Diplomi_zzEn
17 Sep 25 at 6:28 pm
купить диплом о техническом образовании [url=http://educ-ua4.ru/]купить диплом о техническом образовании[/url] .
Diplomi_zyPl
17 Sep 25 at 6:28 pm