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!
https://t.me/s/Official_mellstroy_casino/42
Calvindreli
28 Oct 25 at 11:36 am
купить проведенный диплом моих [url=https://frei-diplom5.ru]купить проведенный диплом моих[/url] .
Diplomi_vuPa
28 Oct 25 at 11:37 am
наркологический диспансер москва [url=http://narkologicheskaya-klinika-27.ru]наркологический диспансер москва[/url] .
narkologicheskaya klinika_qdpl
28 Oct 25 at 11:38 am
Operation Game Canada: A classic, fun-filled board game where players test their precision by removing ailments from the patient without triggering the buzzer: Operation game rules and pieces
GabrielLyday
28 Oct 25 at 11:39 am
купить диплом матроса [url=https://rudik-diplom4.ru]купить диплом матроса[/url] .
Diplomi_amOr
28 Oct 25 at 11:39 am
https://t.me/s/Official_mellstroy_casino/13
Calvindreli
28 Oct 25 at 11:39 am
обмазочная гидроизоляция цена за работу м2 [url=gidroizolyaciya-cena-7.ru]gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_rnSi
28 Oct 25 at 11:39 am
кракен android
кракен зеркало
Henryamerb
28 Oct 25 at 11:40 am
By somefishermen his approach is regarded as premonitory of the advance of thegreat sperm whale.CHAPTER II.オナホ フィギュア
ラブドール
28 Oct 25 at 11:40 am
купить диплом в новороссийске [url=http://www.rudik-diplom3.ru]купить диплом в новороссийске[/url] .
Diplomi_acei
28 Oct 25 at 11:41 am
купить диплом в дзержинске [url=https://rudik-diplom13.ru]купить диплом в дзержинске[/url] .
Diplomi_wson
28 Oct 25 at 11:42 am
купить диплом с занесением в реестр новокузнецке [url=https://www.frei-diplom6.ru]купить диплом с занесением в реестр новокузнецке[/url] .
Diplomi_weOl
28 Oct 25 at 11:43 am
внутренняя гидроизоляция подвала [url=http://gidroizolyaciya-cena-7.ru/]внутренняя гидроизоляция подвала[/url] .
gidroizolyaciya cena_kvSi
28 Oct 25 at 11:43 am
где можно купить диплом медсестры [url=http://frei-diplom14.ru/]где можно купить диплом медсестры[/url] .
Diplomi_hioi
28 Oct 25 at 11:43 am
цена ремонта подвала [url=https://gidroizolyaciya-cena-7.ru/]https://gidroizolyaciya-cena-7.ru/[/url] .
gidroizolyaciya cena_ljSi
28 Oct 25 at 11:45 am
Great beat ! I wish to apprentice even as you amend your web site, how can i subscribe for a weblog
website? The account helped me a applicable deal. I had been a little bit acquainted of this your broadcast provided bright transparent idea
bk8th
28 Oct 25 at 11:45 am
whoah this weblog is magnificent i like studying your posts.
Stay up the good work! You understand, a lot of individuals are hunting round for this
information, you can aid them greatly.
InvestHub 3.0
28 Oct 25 at 11:45 am
купить диплом инженера по охране труда [url=rudik-diplom4.ru]купить диплом инженера по охране труда[/url] .
Diplomi_rzOr
28 Oct 25 at 11:45 am
наркологическая больница [url=https://narkologicheskaya-klinika-27.ru/]наркологическая больница[/url] .
narkologicheskaya klinika_lapl
28 Oct 25 at 11:46 am
kraken marketplace
кракен vk3
Henryamerb
28 Oct 25 at 11:46 am
kamagra: acheter Kamagra en ligne – acheter Kamagra en ligne
RichardImmon
28 Oct 25 at 11:47 am
Ideally, you spoke with your loved one before they
passed away and they told you where they kept the will.
Sergio
28 Oct 25 at 11:48 am
купить диплом в северске [url=http://rudik-diplom5.ru]http://rudik-diplom5.ru[/url] .
Diplomi_bxma
28 Oct 25 at 11:49 am
top rated clock radio [url=https://www.alarm-radio-clocks.com]https://www.alarm-radio-clocks.com[/url] .
Cd Player Radio Alarm Clocks_znOa
28 Oct 25 at 11:49 am
как купить легальный диплом о среднем образовании [url=www.frei-diplom6.ru/]www.frei-diplom6.ru/[/url] .
Diplomi_hkOl
28 Oct 25 at 11:49 am
мелбет официальный [url=https://melbetofficialsite.ru/]мелбет официальный[/url] .
bk melbet_bfEa
28 Oct 25 at 11:49 am
купить диплом в коврове [url=www.rudik-diplom10.ru/]купить диплом в коврове[/url] .
Diplomi_sySa
28 Oct 25 at 11:50 am
купить диплом в стерлитамаке [url=www.rudik-diplom4.ru]купить диплом в стерлитамаке[/url] .
Diplomi_yiOr
28 Oct 25 at 11:50 am
Greetings from Ohio! I’m bored to death at work so I
decided to check out your site on my iphone during lunch break.
I really like the info you present here and can’t wait to take a look when I get home.
I’m shocked at how fast your blog loaded on my mobile ..
I’m not even using WIFI, just 3G .. Anyways, very good blog!
active indoor cat tips
28 Oct 25 at 11:50 am
В этом информативном тексте представлены захватывающие события и факты, которые заставят вас задуматься. Мы обращаем внимание на важные моменты, которые часто остаются незамеченными, и предлагаем новые перспективы на привычные вещи. Подготовьтесь к тому, чтобы быть поглощенным увлекательными рассказами!
Посмотреть всё – https://www.stordsogelag.no/logo-ss-tiff
Kennethmof
28 Oct 25 at 11:51 am
анонимный наркологический центр [url=https://narkologicheskaya-klinika-27.ru]анонимный наркологический центр[/url] .
narkologicheskaya klinika_cnpl
28 Oct 25 at 11:52 am
https://t.me/Official_mellstroy_casino/55
Calvindreli
28 Oct 25 at 11:52 am
Wah lao, ɡood institutions inclᥙde tech in lessons, preparing youngsters ᴡith online
abilities f᧐r durable professions.
Listen սp, composed pom pi pi hor, gⲟod establishment ⲣrovides tech ցroups, readying fߋr tech-savvy prospective roles.
Οh no, primary mathematics teaches practical implementations ѕuch aѕ budgeting,
tһerefore mаke ѕure your youngster masters this properly starting young age.
Folks, fear tһe gap hor,math foundation is essential at primary school in grasping figures, crucial
fоr modern digital economy.
Ꭺрart to establishment facilities, focus ѡith arithmetic tο
prevent typical mistakes including careless blunders аt assessments.
Alas, primary mathematics teaches real-ԝorld applications ⅼike financial planning, tһus make
sure your youngster masters іt correctly starting yоung age.
Օh, mathematics іs tһe foundation pillar ߋf primary education, helping youngsters
іn dimensional analysis in architecture paths.
Greendale Primary School ߋffers аn engaging atmosphere tһat encourages curiosity ɑnd achievement.
Caring personnel assistance tһorough development ɑnd excellence.
Fengshan Primary School ᥙses dynamic programs in ɑ helpful environment.
Thе school develops ѕelf-confidence through engaging activities.
Parents ѵalue its concentrate on student welⅼ-beіng.
My hοmepage … North Vista Secondary School [Zac]
Zac
28 Oct 25 at 11:52 am
One of one of the most typical inquiries concerning
CoolSculpting is whether the treated fat will ultimately return.
Kristopher
28 Oct 25 at 11:52 am
подвал дома ремонт [url=https://gidroizolyaciya-cena-7.ru/]gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_ntSi
28 Oct 25 at 11:52 am
https://t.me/s/Official_mellstroy_casino/39
Calvindreli
28 Oct 25 at 11:53 am
купить диплом с проводкой моего [url=frei-diplom5.ru]купить диплом с проводкой моего[/url] .
Diplomi_vgPa
28 Oct 25 at 11:54 am
Maintain the excellent job and producing in the crowd!
https://www.ta-praca.pl
https://www.ta-praca.pl
28 Oct 25 at 11:54 am
кракен vk2
кракен qr код
Henryamerb
28 Oct 25 at 11:55 am
kraken darknet
кракен vk3
Henryamerb
28 Oct 25 at 11:55 am
купить диплом в бугульме [url=http://rudik-diplom3.ru/]купить диплом в бугульме[/url] .
Diplomi_wbei
28 Oct 25 at 11:56 am
купить диплом бакалавра [url=http://rudik-diplom10.ru]купить диплом бакалавра[/url] .
Diplomi_kjSa
28 Oct 25 at 11:57 am
купить диплом техникума кемерово [url=http://frei-diplom9.ru]купить диплом техникума кемерово[/url] .
Diplomi_pfea
28 Oct 25 at 11:57 am
高級 ダッチワイフtillwe blinded ourselves to the vulgar fact that we were,all of u peopleof very moderate means.
ラブドール
28 Oct 25 at 11:57 am
сырость в подвале многоквартирного дома [url=https://www.gidroizolyaciya-cena-7.ru]https://www.gidroizolyaciya-cena-7.ru[/url] .
gidroizolyaciya cena_fsSi
28 Oct 25 at 11:59 am
наркологические клиники [url=narkologicheskaya-klinika-27.ru]наркологические клиники[/url] .
narkologicheskaya klinika_lcpl
28 Oct 25 at 12:00 pm
Блестящая фраза
4. Бакара. Это еще одна популярная игра, [url=https://1win-3mv3x.top/]сайт 1вин[/url] привлекающая клиентов азартного клуба. в казино есть казино 1вин приложение для отдыха на портативных девайсах айфонах.
RobertFek
28 Oct 25 at 12:00 pm
купить диплом украины с занесением в реестр [url=www.frei-diplom5.ru]www.frei-diplom5.ru[/url] .
Diplomi_fyPa
28 Oct 25 at 12:00 pm
Itsticks horribly and the pattern just enjoys it! All those strangledheads and bulbous eyes and waddling fungus growths just shriek withderision!I am getting angry enough to do something desperate.To jump out of thewindow would be admirable exercise,高級 ダッチワイフ
ラブドール
28 Oct 25 at 12:00 pm
кракен vk5
кракен даркнет
Henryamerb
28 Oct 25 at 12:01 pm