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://rudik-diplom8.ru/]купить диплом слесаря[/url] .
Diplomi_seMt
14 Oct 25 at 10:07 am
как купить диплом с занесением в реестр в екатеринбурге [url=http://frei-diplom4.ru]http://frei-diplom4.ru[/url] .
Diplomi_maOl
14 Oct 25 at 10:08 am
диплом о высшем образовании с проводкой купить [url=https://frei-diplom6.ru/]диплом о высшем образовании с проводкой купить[/url] .
Diplomi_juOl
14 Oct 25 at 10:08 am
аренда экскаватора смена [url=www.arenda-ekskavatora-pogruzchika-cena-2.ru]www.arenda-ekskavatora-pogruzchika-cena-2.ru[/url] .
arenda ekskavatora pogryzchika cena_mgst
14 Oct 25 at 10:08 am
Купить диплом техникума в Львов [url=www.educ-ua7.ru/]www.educ-ua7.ru/[/url] .
Diplomi_sqea
14 Oct 25 at 10:09 am
купить диплом в новокузнецке [url=www.rudik-diplom3.ru/]купить диплом в новокузнецке[/url] .
Diplomi_bdei
14 Oct 25 at 10:09 am
купить диплом в симферополе [url=https://rudik-diplom5.ru]https://rudik-diplom5.ru[/url] .
Diplomi_axma
14 Oct 25 at 10:09 am
кожаные жалюзи с электроприводом [url=https://www.zhalyuzi-s-elektroprivodom77.ru]https://www.zhalyuzi-s-elektroprivodom77.ru[/url] .
jaluzi na okna s elektroprivodom_jjpa
14 Oct 25 at 10:09 am
электрические карнизы купить [url=https://karniz-shtor-elektroprivodom.ru]https://karniz-shtor-elektroprivodom.ru[/url] .
karniz dlya shtor s elektroprivodom_rrer
14 Oct 25 at 10:10 am
рулонные шторы на окна москва [url=https://rulonnaya-shtora-s-elektroprivodom.ru/]https://rulonnaya-shtora-s-elektroprivodom.ru/[/url] .
rylonnaya shtora s elektroprivodom_qoKt
14 Oct 25 at 10:11 am
With havin so much content and articles do you ever run into any issues
of plagorism or copyright infringement? My site has a lot
of exclusive content I’ve either written myself or outsourced but it seems a lot of it is popping it up all over the web without my agreement.
Do you know any techniques to help prevent content
from being stolen? I’d truly appreciate it.
shipping
14 Oct 25 at 10:13 am
купить диплом в великих луках [url=rudik-diplom4.ru]купить диплом в великих луках[/url] .
Diplomi_cmOr
14 Oct 25 at 10:14 am
купить диплом повара-кондитера [url=http://www.rudik-diplom1.ru]купить диплом повара-кондитера[/url] .
Diplomi_ijer
14 Oct 25 at 10:16 am
купить диплом в артеме [url=www.rudik-diplom3.ru]www.rudik-diplom3.ru[/url] .
Diplomi_evei
14 Oct 25 at 10:16 am
перепланировка нежилого помещения [url=www.pereplanirovka-nezhilogo-pomeshcheniya10.ru]перепланировка нежилого помещения[/url] .
pereplanirovka nejilogo pomesheniya_lsSr
14 Oct 25 at 10:16 am
Wow, that’s what I was exploring for, what a stuff! existing here at this webpage, thanks admin of this
web site.
AI Arbitrage
14 Oct 25 at 10:17 am
рулонные шторы автоматические купить [url=rulonnaya-shtora-s-elektroprivodom.ru]rulonnaya-shtora-s-elektroprivodom.ru[/url] .
rylonnaya shtora s elektroprivodom_cfKt
14 Oct 25 at 10:17 am
I’ve been following the Minotaurus presale closely, and it’s impressive how they’ve structured the tokenomics for long-term sustainability. The vesting bonuses are a smart incentive that could really reward patient holders. Excited to see $MTAUR launch and disrupt the blockchain gaming space.
minotaurus coin
WilliamPargy
14 Oct 25 at 10:18 am
купить диплом занесением реестр украины [url=https://frei-diplom6.ru]https://frei-diplom6.ru[/url] .
Diplomi_ofOl
14 Oct 25 at 10:19 am
купить диплом стоматолога [url=rudik-diplom11.ru]купить диплом стоматолога[/url] .
Diplomi_lwMi
14 Oct 25 at 10:19 am
аренда миниэкскаватора [url=http://arenda-mini-ekskavatora-v-moskve-2.ru/]аренда миниэкскаватора[/url] .
arenda mini ekskavatora v moskve_zsKt
14 Oct 25 at 10:20 am
старые дипломы купить [url=https://rudik-diplom4.ru]старые дипломы купить[/url] .
Diplomi_ysOr
14 Oct 25 at 10:20 am
купить диплом медсестры [url=http://rudik-diplom3.ru/]купить диплом медсестры[/url] .
Diplomi_qjei
14 Oct 25 at 10:22 am
электрокарниз москва [url=www.karniz-shtor-elektroprivodom.ru]www.karniz-shtor-elektroprivodom.ru[/url] .
karniz dlya shtor s elektroprivodom_fber
14 Oct 25 at 10:22 am
купить диплом с занесением в реестр в красноярске [url=www.frei-diplom5.ru/]www.frei-diplom5.ru/[/url] .
Diplomi_lwPa
14 Oct 25 at 10:22 am
перепланировка офиса согласование [url=pereplanirovka-nezhilogo-pomeshcheniya9.ru]pereplanirovka-nezhilogo-pomeshcheniya9.ru[/url] .
pereplanirovka nejilogo pomesheniya_haKl
14 Oct 25 at 10:22 am
a cool way to improve
PHP hook, building hooks in your application – Sjoerd Maessen blog at Sjoerd Maessen blog
a cool way to improve
14 Oct 25 at 10:24 am
купить диплом ветеринара [url=rudik-diplom8.ru]купить диплом ветеринара[/url] .
Diplomi_erMt
14 Oct 25 at 10:24 am
купить диплом спб занесением реестр [url=https://www.frei-diplom4.ru]купить диплом спб занесением реестр[/url] .
Diplomi_pvOl
14 Oct 25 at 10:25 am
рулонные шторы с электроприводом и дистанционным управлением [url=www.rulonnaya-shtora-s-elektroprivodom.ru/]www.rulonnaya-shtora-s-elektroprivodom.ru/[/url] .
rylonnaya shtora s elektroprivodom_htKt
14 Oct 25 at 10:25 am
купить диплом электромонтажника [url=rudik-diplom1.ru]купить диплом электромонтажника[/url] .
Diplomi_hyer
14 Oct 25 at 10:25 am
перепланировка в нежилом здании [url=https://www.pereplanirovka-nezhilogo-pomeshcheniya9.ru]перепланировка в нежилом здании[/url] .
pereplanirovka nejilogo pomesheniya_qxKl
14 Oct 25 at 10:25 am
купить диплом в сызрани [url=https://www.rudik-diplom5.ru]купить диплом в сызрани[/url] .
Diplomi_sima
14 Oct 25 at 10:25 am
карнизы с электроприводом [url=https://elektrokarnizy797.ru]карнизы с электроприводом[/url] .
elektrokarnizi_soMl
14 Oct 25 at 10:26 am
Приобрести диплом ВУЗа поможем. Купить диплом специалиста в Белгороде – [url=http://diplomybox.com/kupit-diplom-spetsialista-v-belgorode/]diplomybox.com/kupit-diplom-spetsialista-v-belgorode[/url]
Cazrhge
14 Oct 25 at 10:27 am
алюминиевые электрожалюзи [url=www.zhalyuzi-s-elektroprivodom77.ru/]www.zhalyuzi-s-elektroprivodom77.ru/[/url] .
jaluzi na okna s elektroprivodom_xupa
14 Oct 25 at 10:28 am
Эта информационная статья охватывает широкий спектр актуальных тем и вопросов. Мы стремимся осветить ключевые факты и события с ясностью и простотой, чтобы каждый читатель мог извлечь из нее полезные знания и полезные инсайты.
Ознакомиться с полной информацией – https://cis-invest.com/?p=254
MichaelDyday
14 Oct 25 at 10:28 am
Se você é novo no AI Studios ou quer turbinar seu fluxo de trabalho de criação de vídeos,nossa seção de perguntas frequentes ajudará você a aprender mais sobre nossos recursos.
baixar do you tube
14 Oct 25 at 10:28 am
купить диплом высшем образовании занесением реестр [url=https://www.frei-diplom5.ru]купить диплом высшем образовании занесением реестр[/url] .
Diplomi_qqPa
14 Oct 25 at 10:29 am
жалюзи с электроприводом купить [url=http://www.zhalyuzi-s-elektroprivodom77.ru]жалюзи с электроприводом купить[/url] .
jaluzi na okna s elektroprivodom_nbpa
14 Oct 25 at 10:29 am
разрешение на перепланировку нежилого помещения не требуется [url=pereplanirovka-nezhilogo-pomeshcheniya10.ru]pereplanirovka-nezhilogo-pomeshcheniya10.ru[/url] .
pereplanirovka nejilogo pomesheniya_dcSr
14 Oct 25 at 10:30 am
купить диплом о высшем образовании с занесением в реестр цены [url=https://frei-diplom4.ru]купить диплом о высшем образовании с занесением в реестр цены[/url] .
Diplomi_deOl
14 Oct 25 at 10:31 am
карнизы с электроприводом купить [url=https://www.karniz-shtor-elektroprivodom.ru]карнизы с электроприводом купить[/url] .
karniz dlya shtor s elektroprivodom_uger
14 Oct 25 at 10:31 am
купить диплом в клинцах [url=http://rudik-diplom8.ru]купить диплом в клинцах[/url] .
Diplomi_smMt
14 Oct 25 at 10:31 am
MelStroy Casino предлагает более 1000 игр, включая слоты, рулетку и живые столы. Быстрые выплаты, круглосуточная поддержка и щедрые бонусы для новичков и постоянных игроков
меллстрой казино
Walterhek
14 Oct 25 at 10:32 am
потолки натяжные в самаре [url=natyazhnye-potolki-samara-2.ru]natyazhnye-potolki-samara-2.ru[/url] .
natyajnie potolki samara_yvPi
14 Oct 25 at 10:34 am
Whats up very cool site!! Guy .. Beautiful .. Superb ..
I will bookmark your web site and take the feeds also? I’m happy to find a
lot of useful information right here within the post, we want work out more techniques on this regard,
thanks for sharing. . . . . .
비아그라 구매
14 Oct 25 at 10:34 am
рулонные шторки на окна [url=http://rulonnaya-shtora-s-elektroprivodom.ru/]рулонные шторки на окна[/url] .
rylonnaya shtora s elektroprivodom_rxKt
14 Oct 25 at 10:35 am
аренда мини экскаваторов [url=arenda-mini-ekskavatora-v-moskve-2.ru]arenda-mini-ekskavatora-v-moskve-2.ru[/url] .
arenda mini ekskavatora v moskve_gcKt
14 Oct 25 at 10:35 am
кожаные жалюзи с электроприводом [url=http://zhalyuzi-s-elektroprivodom77.ru]http://zhalyuzi-s-elektroprivodom77.ru[/url] .
jaluzi na okna s elektroprivodom_hxpa
14 Oct 25 at 10:36 am