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!
кинешемский педагогический колледж диплом 1998 года купить [url=https://www.frei-diplom7.ru]https://www.frei-diplom7.ru[/url] .
Diplomi_fpei
16 Oct 25 at 10:54 am
http://buy.mitolyn-sales.us
Mitolyn discount
16 Oct 25 at 10:54 am
купить диплом в ростове-на-дону [url=https://www.rudik-diplom1.ru]купить диплом в ростове-на-дону[/url] .
Diplomi_oyer
16 Oct 25 at 10:55 am
Асфальтирование – это сложный многоэтапный процесс, превращающий грубую основу в гладкое и долговечное дорожное покрытие. Это не просто укладка черной массы, а кропотливая работа, требующая опыта, знаний и применения специализированной техники.
От качества асфальтирования напрямую зависит комфорт и безопасность движения, долговечность дороги и экономическая эффективность ее эксплуатации. Где заказать [url=http://www.bisound.com/forum/showthread.php?p=2865678#post2865678]асфальтирование в спб[/url]
GeorgeDoomb
16 Oct 25 at 10:55 am
аренда экскаватора-погрузчика [url=http://arenda-ekskavatora-pogruzchika-cena.ru/]http://arenda-ekskavatora-pogruzchika-cena.ru/[/url] .
arenda ekskavatora pogryzchika cena_tmSr
16 Oct 25 at 10:57 am
Публикация предлагает уникальную подборку информации, которая будет интересна как специалистам, так и широкому кругу читателей. Здесь вы найдете ответы на часто задаваемые вопросы и полезные инсайты для дальнейшего применения.
Более того — здесь – https://perfecta-travel.com/product/package-name-25
Rodneybon
16 Oct 25 at 10:57 am
потолочник натяжные потолки отзывы [url=https://natyazhnye-potolki-nizhniy-novgorod-1.ru/]потолочник натяжные потолки отзывы[/url] .
natyajnie potolki nijnii novgorod_axma
16 Oct 25 at 10:59 am
как купить легально диплом о высшем образовании [url=http://www.frei-diplom5.ru]как купить легально диплом о высшем образовании[/url] .
Diplomi_bnPa
16 Oct 25 at 11:02 am
Hi every one, here every person is sharing such experience, so it’s nice
to read this website, and I used to visit this blog all the time.
Interior emulsion
16 Oct 25 at 11:03 am
купить диплом с занесением в реестр в калуге [url=frei-diplom4.ru]купить диплом с занесением в реестр в калуге[/url] .
Diplomi_hkOl
16 Oct 25 at 11:03 am
потолочкин потолки [url=https://stretch-ceilings-nizhniy-novgorod-1.ru/]https://stretch-ceilings-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_ceOn
16 Oct 25 at 11:03 am
купить диплом техникума ссср в астрахани [url=http://frei-diplom7.ru]купить диплом техникума ссср в астрахани[/url] .
Diplomi_wxei
16 Oct 25 at 11:03 am
mostbet войти
mostbet kazakhstan
16 Oct 25 at 11:04 am
купить диплом ссср [url=https://rudik-diplom11.ru/]купить диплом ссср[/url] .
Diplomi_giMi
16 Oct 25 at 11:05 am
купить диплом в чайковском [url=http://rudik-diplom3.ru]купить диплом в чайковском[/url] .
Diplomi_feei
16 Oct 25 at 11:06 am
https://t.me/Online_1_xbet/3519
CharlesCic
16 Oct 25 at 11:11 am
купить диплом моториста [url=https://www.rudik-diplom4.ru]купить диплом моториста[/url] .
Diplomi_jnOr
16 Oct 25 at 11:11 am
https://t.me/Online_1_xbet/2514
CharlesCic
16 Oct 25 at 11:12 am
Купить диплом техникума в Хмельницкий [url=educ-ua7.ru]educ-ua7.ru[/url] .
Diplomi_igea
16 Oct 25 at 11:13 am
Эта публикация погружает вас в мир увлекательных фактов и удивительных открытий. Мы расскажем о ключевых событиях, которые изменили ход истории, и приоткроем завесу над научными достижениями, которые вдохновили миллионы. Узнайте, чему может научить нас прошлое и как применить эти знания в будущем.
Узнать больше > – https://gtelectricservice.com/porttitor-porttitor-mollis-vitae-placerat-2
LarryBoogY
16 Oct 25 at 11:13 am
What’s up to every one, since I am in fact keen of reading this blog’s post to be updated
regularly. It contains nice material.
Lys Finthera
16 Oct 25 at 11:13 am
mostbet android
мостбет скачать приложение
16 Oct 25 at 11:14 am
купить диплом в каменске-уральском [url=https://www.rudik-diplom11.ru]купить диплом в каменске-уральском[/url] .
Diplomi_rsMi
16 Oct 25 at 11:14 am
Эта познавательная публикация погружает вас в море интересного контента, который быстро захватит ваше внимание. Мы рассмотрим важные аспекты темы и предоставим вам уникальные Insights и полезные сведения для дальнейшего изучения.
ТОП-5 причин узнать больше – https://microgamingslotsgame.online/gameplay-features-that-elevate-the-belfry-bliss-slots
JosephArils
16 Oct 25 at 11:14 am
купить диплом в мытищах [url=rudik-diplom3.ru]купить диплом в мытищах[/url] .
Diplomi_rrei
16 Oct 25 at 11:15 am
Приобрести диплом любого университета мы поможем. Купить диплом Рязань – [url=http://diplomybox.com/kupit-diplom-ryazan/]diplomybox.com/kupit-diplom-ryazan[/url]
Cazrqfs
16 Oct 25 at 11:15 am
Этот информационный материал собраны данные, которые помогут лучше понять текущие тенденции и процессы в различных сферах жизни. Мы предоставляем четкий анализ, графики и примеры, чтобы информация была не только понятной, но и практичной для принятия решений.
Всё, что нужно знать – http://www.jkrcreative.com/underconstruction
Peterbip
16 Oct 25 at 11:16 am
https://t.me/Online_1_xbet/3387
CharlesCic
16 Oct 25 at 11:16 am
https://t.me/Online_1_xbet/2224
CharlesCic
16 Oct 25 at 11:17 am
Very good post! We will be linking to this great content on our website.
Keep up the great writing.
tenerife keliones
16 Oct 25 at 11:17 am
В этом информативном тексте представлены захватывающие события и факты, которые заставят вас задуматься. Мы обращаем внимание на важные моменты, которые часто остаются незамеченными, и предлагаем новые перспективы на привычные вещи. Подготовьтесь к тому, чтобы быть поглощенным увлекательными рассказами!
Смотри, что ещё есть – http://ltemachinery.ie/btme-harrowgate-roundup
Eugenetaulp
16 Oct 25 at 11:18 am
потолочников натяжные потолки [url=http://natyazhnye-potolki-nizhniy-novgorod-1.ru]потолочников натяжные потолки[/url] .
natyajnie potolki nijnii novgorod_xvma
16 Oct 25 at 11:18 am
Hey there! I’m at work surfing around your blog from my new iphone!
Just wanted to say I love reading your blog and look forward to all your posts!
Carry on the excellent work!
AC repair service
16 Oct 25 at 11:18 am
купить диплом массажиста [url=http://rudik-diplom11.ru]купить диплом массажиста[/url] .
Diplomi_mvMi
16 Oct 25 at 11:20 am
Does yoga help you lose weight
PHP hook, building hooks in your application – Sjoerd Maessen blog at Sjoerd Maessen blog
Does yoga help you lose weight
16 Oct 25 at 11:20 am
диплом купить проведенный [url=http://frei-diplom5.ru]диплом купить проведенный[/url] .
Diplomi_qwPa
16 Oct 25 at 11:21 am
экскаватор погрузчик аренда москва [url=https://arenda-ekskavatora-pogruzchika-cena.ru]экскаватор погрузчик аренда москва[/url] .
arenda ekskavatora pogryzchika cena_dcSr
16 Oct 25 at 11:21 am
купить диплом менеджера [url=http://rudik-diplom4.ru/]купить диплом менеджера[/url] .
Diplomi_ptOr
16 Oct 25 at 11:23 am
купить диплом в энгельсе [url=http://rudik-diplom5.ru/]http://rudik-diplom5.ru/[/url] .
Diplomi_wlma
16 Oct 25 at 11:23 am
купить диплом с реестром вуза [url=http://frei-diplom4.ru]купить диплом с реестром вуза[/url] .
Diplomi_ucOl
16 Oct 25 at 11:24 am
купить диплом техникума в пензе [url=https://frei-diplom10.ru]купить диплом техникума в пензе[/url] .
Diplomi_ndEa
16 Oct 25 at 11:24 am
купить диплом швеи [url=rudik-diplom3.ru]купить диплом швеи[/url] .
Diplomi_gpei
16 Oct 25 at 11:25 am
аренда экскаватора погрузчика цена [url=http://arenda-ekskavatora-pogruzchika-cena.ru/]аренда экскаватора погрузчика цена[/url] .
arenda ekskavatora pogryzchika cena_vrSr
16 Oct 25 at 11:25 am
купить диплом учителя физической культуры [url=rudik-diplom1.ru]купить диплом учителя физической культуры[/url] .
Diplomi_wyer
16 Oct 25 at 11:26 am
Эта информационная заметка содержит увлекательные сведения, которые могут вас удивить! Мы собрали интересные факты, которые сделают вашу жизнь ярче и полнее. Узнайте нечто новое о привычных аспектах повседневности и откройте для себя удивительный мир информации.
Смотри, что ещё есть – https://meet2share.com/the-ultimate-guide-to-academic-question-and-answer-platforms
Joshuanug
16 Oct 25 at 11:26 am
натяжные потолки нижний новгород официальный сайт [url=https://stretch-ceilings-nizhniy-novgorod-1.ru/]https://stretch-ceilings-nizhniy-novgorod-1.ru/[/url] .
natyajnie potolki nijnii novgorod_gqOn
16 Oct 25 at 11:27 am
größte wettanbieter in wette deutschland england
wette deutschland england
16 Oct 25 at 11:30 am
купить диплом слесаря [url=www.rudik-diplom4.ru/]купить диплом слесаря[/url] .
Diplomi_usOr
16 Oct 25 at 11:31 am
creativevision – Each artwork invites the viewer to see the world differently.
Emogene Alescio
16 Oct 25 at 11:31 am
как купить легально диплом о высшем образовании [url=http://www.frei-diplom4.ru]как купить легально диплом о высшем образовании[/url] .
Diplomi_ddOl
16 Oct 25 at 11:32 am