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://frei-diplom2.ru]купить диплом внесенный в реестр[/url] .
Diplomi_ejEa
16 Oct 25 at 5:21 am
потолки натяжные [url=www.stretch-ceilings-nizhniy-novgorod-1.ru/]потолки натяжные[/url] .
natyajnie potolki nijnii novgorod_rlOn
16 Oct 25 at 5:22 am
Definitely believe that which you said. Your favorite justification appeared to
be on the net the simplest thing to be aware of.
I say to you, I definitely get annoyed while people
consider worries that they just do not know about.
You managed to hit the nail upon the top and also defined out the whole thing without
having side-effects , people could take a signal.
Will likely be back to get more. Thanks
homepage
16 Oct 25 at 5:22 am
trusted online pharmacy for ED meds: tadalafil tablets without prescription – trusted online pharmacy for ED meds
AndrewPal
16 Oct 25 at 5:24 am
диплом о высшем образовании купить с занесением в реестр [url=frei-diplom4.ru]диплом о высшем образовании купить с занесением в реестр[/url] .
Diplomi_baOl
16 Oct 25 at 5:24 am
Мы — частная наркологическая клиника, где скорость запуска помощи сочетается с деликатностью и прозрачностью. Команда дежурит круглосуточно, выезжает на дом по Подольску и ближайшим районам, принимает в стационаре и амбулаторно. Все решения принимаются после очного осмотра и оценки рисков, без «универсальных» схем и навязанных услуг.
Исследовать вопрос подробнее – [url=https://narkologicheskaya-klinika-podolsk0.ru/]narkologicheskaya-klinika-vrachi[/url]
Jasonmes
16 Oct 25 at 5:25 am
купить диплом во всеволожске [url=rudik-diplom8.ru]rudik-diplom8.ru[/url] .
Diplomi_uiMt
16 Oct 25 at 5:25 am
Следующий этап — согласование индивидуального плана. Состав капельницы и схема симптоматической терапии подбираются с учётом возраста, веса, сопутствующих болезней и текущих жалоб. Мы подробно объясняем, зачем назначается каждый препарат, как он действует и чего ожидать в ближайшие часы.
Получить больше информации – http://narkolog-na-dom-krasnogorsk6.ru/narkolog-na-dom-anonimno-v-krasnogorske/
JosephVem
16 Oct 25 at 5:25 am
потолочник отзывы [url=www.natyazhnye-potolki-nizhniy-novgorod-1.ru/]www.natyazhnye-potolki-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_frma
16 Oct 25 at 5:26 am
купить диплом в старом осколе [url=rudik-diplom14.ru]купить диплом в старом осколе[/url] .
Diplomi_lvea
16 Oct 25 at 5:26 am
потолочкин натяжные потолки отзывы клиентов нижний новгород [url=http://stretch-ceilings-nizhniy-novgorod.ru/]http://stretch-ceilings-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_zkPl
16 Oct 25 at 5:27 am
You have made some good points there. I
looked on the internet for more info about the
issue and found most individuals will go along with your views on this website.
agen bokep situs bokep bokep online terpercaya bandar sampah terbesar situs scam
16 Oct 25 at 5:27 am
натяжной потолок цена нижний новгород [url=stretch-ceilings-nizhniy-novgorod-1.ru]stretch-ceilings-nizhniy-novgorod-1.ru[/url] .
natyajnie potolki nijnii novgorod_pyOn
16 Oct 25 at 5:28 am
натяжные потолки нижний новгород цены [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru]натяжные потолки нижний новгород цены[/url] .
natyajnie potolki nijnii novgorod_lmma
16 Oct 25 at 5:28 am
натяжные потолки нижний новгород цены [url=https://natyazhnye-potolki-nizhniy-novgorod.ru/]https://natyazhnye-potolki-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_rtOt
16 Oct 25 at 5:28 am
купить диплом в химках [url=https://www.rudik-diplom3.ru]купить диплом в химках[/url] .
Diplomi_klei
16 Oct 25 at 5:29 am
как купить диплом с проводкой [url=frei-diplom3.ru]как купить диплом с проводкой[/url] .
Diplomi_ppKt
16 Oct 25 at 5:30 am
мостбет казино играть
мостбет кз официальный сайт
16 Oct 25 at 5:31 am
купить диплом института [url=https://rudik-diplom5.ru/]купить диплом института[/url] .
Diplomi_uwma
16 Oct 25 at 5:31 am
диплом техникума купить в казахстане [url=www.frei-diplom7.ru]диплом техникума купить в казахстане[/url] .
Diplomi_fgei
16 Oct 25 at 5:32 am
1win azerbaycan rəsmisi [url=http://1win5005.com]http://1win5005.com[/url]
1win_ykml
16 Oct 25 at 5:33 am
Cialis online USA [url=https://tadalifepharmacy.shop/#]TadaLife Pharmacy[/url] generic Cialis online pharmacy
CareyMag
16 Oct 25 at 5:34 am
https://zencaremeds.shop/# ZenCare Meds com
MervinWoorE
16 Oct 25 at 5:34 am
купить диплом техникума ссср в кемерово [url=http://frei-diplom10.ru/]купить диплом техникума ссср в кемерово[/url] .
Diplomi_wdEa
16 Oct 25 at 5:35 am
компания потолочник [url=www.natyazhnye-potolki-nizhniy-novgorod.ru]www.natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_nvOt
16 Oct 25 at 5:35 am
купить диплом с занесением в реестр челябинск [url=https://frei-diplom4.ru]купить диплом с занесением в реестр челябинск[/url] .
Diplomi_zpOl
16 Oct 25 at 5:36 am
Woah! I’m really digging the template/theme of this blog.
It’s simple, yet effective. A lot of times it’s hard to get that “perfect balance” between superb usability and visual appeal.
I must say you’ve done a superb job with this. Also, the blog
loads very quick for me on Firefox. Superb Blog!
facer.io
16 Oct 25 at 5:37 am
https://zencaremeds.shop/# online pharmacy
Hermandug
16 Oct 25 at 5:39 am
купить диплом в калуге [url=rudik-diplom3.ru]купить диплом в калуге[/url] .
Diplomi_xbei
16 Oct 25 at 5:40 am
потолочник натяжные потолки отзывы [url=www.natyazhnye-potolki-nizhniy-novgorod-1.ru/]потолочник натяжные потолки отзывы[/url] .
natyajnie potolki nijnii novgorod_xmma
16 Oct 25 at 5:42 am
потолки нижний новгород [url=https://natyazhnye-potolki-nizhniy-novgorod.ru/]https://natyazhnye-potolki-nizhniy-novgorod.ru/[/url] .
natyajnie potolki nijnii novgorod_ohOt
16 Oct 25 at 5:43 am
где купить диплом техникума было [url=frei-diplom7.ru]где купить диплом техникума было[/url] .
Diplomi_pbei
16 Oct 25 at 5:43 am
Great delivery. Sound arguments. Keep up the good spirit.
pabipemkabpolewalimandar.org
16 Oct 25 at 5:43 am
студия для съемки подкастов [url=http://studiya-podkastov-spb1.ru]http://studiya-podkastov-spb1.ru[/url] .
stydiya podkastov spb_nrKn
16 Oct 25 at 5:43 am
купить диплом в старом осколе [url=https://rudik-diplom15.ru/]купить диплом в старом осколе[/url] .
Diplomi_yjPi
16 Oct 25 at 5:44 am
купить диплом в белгороде [url=https://rudik-diplom11.ru/]купить диплом в белгороде[/url] .
Diplomi_ilMi
16 Oct 25 at 5:45 am
Great post. I used to be checking constantly this weblog and I am impressed!
Extremely helpful information particularly the closing part
🙂 I maintain such info much. I used to be looking for this certain info for a long time.
Thank you and good luck.
crack windows anhthoit
16 Oct 25 at 5:47 am
купить диплом с занесением в реестр новосибирск [url=https://frei-diplom6.ru/]купить диплом с занесением в реестр новосибирск[/url] .
Diplomi_ppOl
16 Oct 25 at 5:47 am
студия подкастов в санкт-петербурге [url=www.studiya-podkastov-spb1.ru/]www.studiya-podkastov-spb1.ru/[/url] .
stydiya podkastov spb_daKn
16 Oct 25 at 5:48 am
где купить диплом колледжа в смоленске [url=https://www.frei-diplom11.ru]https://www.frei-diplom11.ru[/url] .
Diplomi_jhsa
16 Oct 25 at 5:49 am
купить диплом товароведа [url=http://rudik-diplom8.ru]купить диплом товароведа[/url] .
Diplomi_dpMt
16 Oct 25 at 5:51 am
купить диплом в вольске [url=www.rudik-diplom5.ru/]www.rudik-diplom5.ru/[/url] .
Diplomi_wtma
16 Oct 25 at 5:51 am
натяжные потолки ру [url=http://www.stretch-ceilings-nizhniy-novgorod-1.ru]натяжные потолки ру[/url] .
natyajnie potolki nijnii novgorod_wkOn
16 Oct 25 at 5:53 am
купить диплом техникума союзной республики [url=www.frei-diplom10.ru]купить диплом техникума союзной республики[/url] .
Diplomi_nvEa
16 Oct 25 at 5:53 am
потолочкин потолки натяжные [url=www.natyazhnye-potolki-nizhniy-novgorod.ru]www.natyazhnye-potolki-nizhniy-novgorod.ru[/url] .
natyajnie potolki nijnii novgorod_qeOt
16 Oct 25 at 5:53 am
диплом техникум купить [url=https://educ-ua7.ru]https://educ-ua7.ru[/url] .
Diplomi_pnea
16 Oct 25 at 5:55 am
купить диплом строителя [url=rudik-diplom15.ru]купить диплом строителя[/url] .
Diplomi_zlPi
16 Oct 25 at 5:55 am
купить диплом во владикавказе [url=https://rudik-diplom4.ru/]https://rudik-diplom4.ru/[/url] .
Diplomi_agOr
16 Oct 25 at 5:56 am
Whats up very cool web site!! Man .. Excellent ..
Wonderful .. I’ll bookmark your blog and take the
feeds also? I’m satisfied to find a lot of useful info right here within the put up, we’d like develop extra techniques in this
regard, thank you for sharing. . . . . .
my digi
16 Oct 25 at 5:56 am
купить диплом с занесением в реестр в нижнем тагиле [url=frei-diplom5.ru]купить диплом с занесением в реестр в нижнем тагиле[/url] .
Diplomi_pbPa
16 Oct 25 at 5:58 am