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.razdvizhnoj-elektrokarniz.ru]http://www.razdvizhnoj-elektrokarniz.ru[/url] .
razdvijnoi elektrokarniz_wpei
18 Sep 25 at 6:15 am
раздвижные карнизы для штор купить [url=https://www.razdvizhnoj-elektrokarniz.ru]https://www.razdvizhnoj-elektrokarniz.ru[/url] .
razdvijnoi elektrokarniz_qwei
18 Sep 25 at 6:18 am
Receptfritt potensmedel – Letar du efter receptfria potensmedel? Upptäck effektiva produkter som kan hjälpa dig utan krångliga recept, med snabb leverans och hög kvalitet direkt hem.
Cialis bästa pris
18 Sep 25 at 6:18 am
https://www.wildberries.ru/catalog/171056105/detail.aspx
Curtissab
18 Sep 25 at 6:19 am
Hello i am kavin, its my first time to commenting anywhere, when i read this piece of writing i thought i could also make
comment due to this brilliant post.
سایت کارا آموزش پرورش
18 Sep 25 at 6:21 am
Kaizenaire.com curates Singapore’ѕ a lot оf іnteresting
promotions ɑnd deals, making it thе top website fօr neighborhood shopping fanatics.
Аlways on sharp for bargains, Singaporeans symbolize Singapore’ѕ spirit ɑѕ a shopping place.
Singaporeans օften exercise tai сhі in parks fоr morning health
routines, ɑnd remember to stay updated on Singapore’smost гecent
promotions and shopping deals.
Tiger Beer, а renowned neighborhood brew, prօvides refreshing ales tһat Singaporeans enjoy for tһeir crisp taste tһroughout social gatherings ɑnd celebrations.
Samsung supplies electronics ⅼike mobile phones аnd TVs lah,
enjoyed by tech enthusiasts іn Singapore fօr thеir sophisticated features
аnd toughness lor.
Pepper Lunch sizzles DIY steaks аnd rice օn warm plates,
preferred fοr interactive cooking аnd delicious, inexpensive dishes.
Eh, Singaporeans, book marking lah, check regularly
fⲟr promotions mah.
Alsօ visit mу web site promo singapore
promo singapore
18 Sep 25 at 6:22 am
https://wanderlog.com/view/vxfutevhhk/купить-кокаин-шарм-эль-шейх/
Timothyces
18 Sep 25 at 6:23 am
It’s genuinely very complex in this full of activity
life to listen news on TV, thus I just use web for that reason, and take
the hottest news.
mm99
18 Sep 25 at 6:23 am
Sou louco pela chama de DonaldBet Casino, tem um ritmo de jogo que danca como um acrobata. A colecao e um espetaculo de entretenimento. com slots tematicos de espetaculos vibrantes. O servico e confiavel como uma corda bamba. com ajuda que ilumina como um refletor. As transacoes sao faceis como um estalo. em alguns momentos mais giros gratis seriam vibrantes. No geral, DonaldBet Casino e um picadeiro de emocoes para quem curte apostar com estilo circense! E mais a navegacao e facil como um malabarismo. fazendo o cassino reluzir como um holofote.
donaldbet afiliados|
flickerzippyowl5zef
18 Sep 25 at 6:23 am
Рапорт командиру на выплату боевых — образец с формулировками, командир сразу понял. денежные выплаты военным
Brentagila
18 Sep 25 at 6:24 am
http://blaukraftde.com/# online apotheke gГјnstig
EnriqueVox
18 Sep 25 at 6:25 am
Thanks for the auspicious writeup. It in truth used to
be a leisure account it. Glance advanced to more added agreeable from you!
However, how can we keep up a correspondence?
Lueur Flowdex
18 Sep 25 at 6:26 am
https://intimgesund.com/# Intim Gesund
Williamves
18 Sep 25 at 6:27 am
Hmm is anyone else having problems with the pictures on this blog
loading? I’m trying to figure out if its a problem on my
end or if it’s the blog. Any feedback would be greatly appreciated.
AV ซับไทย
18 Sep 25 at 6:28 am
список займов онлайн на карту [url=https://zaimy-15.ru/]https://zaimy-15.ru/[/url] .
zaimi_ekpn
18 Sep 25 at 6:29 am
https://xn--krken23-bn4c.com
Howardreomo
18 Sep 25 at 6:29 am
you’re in reality a excellent webmaster. The website loading velocity is incredible.
It kind of feels that you’re doing any unique trick.
Also, The contents are masterwork. you’ve performed a fantastic task on this subject!
شرایط تدریس در دانشگاه علمی کاربردی
18 Sep 25 at 6:30 am
карниз раздвижной купить [url=https://www.razdvizhnoj-elektrokarniz.ru]https://www.razdvizhnoj-elektrokarniz.ru[/url] .
razdvijnoi elektrokarniz_toei
18 Sep 25 at 6:30 am
Распознать необходимость лечения просто: достаточно внимательно отнестись к тревожным признакам. Вот лишь некоторые ситуации, когда обращение в наркологическую клинику становится жизненно важным:
Разобраться лучше – http://narkologicheskaya-klinika-balashiha5.ru/chastnaya-narkologicheskaya-klinika-v-balashihe/
RichardPab
18 Sep 25 at 6:31 am
все микрозаймы [url=https://www.zaimy-14.ru]все микрозаймы[/url] .
zaimi_elSr
18 Sep 25 at 6:32 am
официальные займы онлайн на карту бесплатно [url=https://zaimy-15.ru/]https://zaimy-15.ru/[/url] .
zaimi_plpn
18 Sep 25 at 6:32 am
Чтобы успешно «открутить» его, новичкам не рекомендуется брать за раз больше 5000
рублей.
7к казино
18 Sep 25 at 6:34 am
https://xn--krken21-bn4c.com
Howardreomo
18 Sep 25 at 6:35 am
список займов онлайн [url=www.zaimy-15.ru]список займов онлайн[/url] .
zaimi_ajpn
18 Sep 25 at 6:35 am
Your style is very unique compared to other people I’ve read stuff from.
I appreciate you for posting when you’ve got the opportunity, Guess I will just book mark this web site.
как похудеть на 10 кг за три недели
18 Sep 25 at 6:36 am
If some one desires to be updated with newest technologies afterward he
must be visit this web site and be up to date everyday.
88fc
18 Sep 25 at 6:36 am
все онлайн займы [url=http://www.zaimy-14.ru]http://www.zaimy-14.ru[/url] .
zaimi_vtSr
18 Sep 25 at 6:38 am
электрокарниз двухрядный [url=razdvizhnoj-elektrokarniz.ru]razdvizhnoj-elektrokarniz.ru[/url] .
razdvijnoi elektrokarniz_ldei
18 Sep 25 at 6:38 am
Hey there I am so delighted I found your blog, I really found you by mistake, while I was researching on Google
for something else, Anyhow I am here now and would just
like to say thanks for a remarkable post and a all round
exciting blog (I also love the theme/design), I don’t have time to go through it all at the
moment but I have bookmarked it and also included your RSS feeds, so when I have time I will be back to read more, Please do keep up the excellent
work.
kra33
18 Sep 25 at 6:41 am
раздвижные карнизы для штор купить [url=www.razdvizhnoj-elektrokarniz.ru/]www.razdvizhnoj-elektrokarniz.ru/[/url] .
razdvijnoi elektrokarniz_vhei
18 Sep 25 at 6:43 am
Hi to every one, because I am genuinely eager of reading this
website’s post to be updated daily. It contains fastidious data.
สล็อตเว็บตรง
18 Sep 25 at 6:43 am
все займы рф [url=https://zaimy-14.ru]https://zaimy-14.ru[/url] .
zaimi_gpSr
18 Sep 25 at 6:43 am
https://xn--krken21-bn4c.com
Howardreomo
18 Sep 25 at 6:44 am
купить диплом колледжа недорого [url=https://educ-ua9.ru]купить диплом колледжа недорого[/url] .
Diplomi_bspr
18 Sep 25 at 6:44 am
https://www.brownbook.net/business/54273723/мончегорск-купить-кокаин-альфа-пвп-мефедрон/
Timothyces
18 Sep 25 at 6:45 am
купить диплом техникума об окончании [url=educ-ua8.ru]купить диплом техникума об окончании[/url] .
Diplomi_sept
18 Sep 25 at 6:45 am
все займы онлайн [url=https://zaimy-15.ru]все займы онлайн[/url] .
zaimi_wrpn
18 Sep 25 at 6:45 am
раздвижной карниз [url=https://www.razdvizhnoj-elektrokarniz.ru]https://www.razdvizhnoj-elektrokarniz.ru[/url] .
razdvijnoi elektrokarniz_pbei
18 Sep 25 at 6:45 am
https://intimgesund.com/# generisches sildenafil alternative
Williamves
18 Sep 25 at 6:48 am