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_1xbet_1xbet/1784
Josephadvem
18 Oct 25 at 2:54 am
mexican pharmacy: mexican pharmacy – mexico drug store online
AndrewPal
18 Oct 25 at 2:58 am
https://clocks-top.com/
RandallHen
18 Oct 25 at 2:58 am
https://clocks-top.com/
RandallHen
18 Oct 25 at 2:59 am
https://t.me/Official_1xbet_1xbet/1664
Josephadvem
18 Oct 25 at 2:59 am
согласование проекта перепланировки квартиры [url=www.soglasovanie-pereplanirovki-kvartiry11.ru]www.soglasovanie-pereplanirovki-kvartiry11.ru[/url] .
soglasovanie pereplanirovki kvartiri _uiMi
18 Oct 25 at 3:00 am
Hi just wanted to give you a brief heads up and let you know
a few of the pictures aren’t loading properly.
I’m not sure why but I think its a linking issue.
I’ve tried it in two different web browsers and both show the same outcome.
kra41 cc
18 Oct 25 at 3:01 am
согласование проекта перепланировки квартиры [url=https://www.soglasovanie-pereplanirovki-kvartiry14.ru]https://www.soglasovanie-pereplanirovki-kvartiry14.ru[/url] .
soglasovanie pereplanirovki kvartiri _crEl
18 Oct 25 at 3:02 am
Капельница от запоя на дому в Нижнем Новгороде — удобное решение для тех, кто не может посетить клинику. Наши специалисты приедут к вам домой и проведут необходимую процедуру.
Узнать больше – [url=https://vyvod-iz-zapoya-nizhnij-novgorod11.ru/]вывод из запоя цена нижний новгород[/url]
TerrellOwelf
18 Oct 25 at 3:05 am
В Воронеже клиника помогает стабилизировать состояние и восстановить здоровье после запоя.
Подробнее тут – [url=https://vyvod-iz-zapoya-v-stacionare-voronezh22.ru/]вывод из запоя в стационаре клиника в воронеже[/url]
RichardJuids
18 Oct 25 at 3:06 am
фрибет melbet [url=http://www.melbetbonusy.ru]фрибет melbet[/url] .
melbet_jnOi
18 Oct 25 at 3:08 am
Magnificent goods from you, man. I’ve understand your
stuff previous to and you’re just extremely
excellent. I actually like what you have acquired
here, certainly like what you’re saying and the way in which you say it.
You make it entertaining and you still care
for to keep it sensible. I cant wait to read far more
from you. This is actually a wonderful site.
article source
18 Oct 25 at 3:11 am
Hi! Do you use Twitter? I’d like to follow you if that would be ok.
I’m undoubtedly enjoying your blog and look forward to new posts.
our website
18 Oct 25 at 3:12 am
сколько стоит узаконить перепланировку [url=https://stoimost-soglasovaniya-pereplanirovki-kvartiry.ru]https://stoimost-soglasovaniya-pereplanirovki-kvartiry.ru[/url] .
stoimost soglasovaniya pereplanirovki kvartiri_fxPt
18 Oct 25 at 3:13 am
куплю диплом медсестры в москве [url=www.frei-diplom14.ru]куплю диплом медсестры в москве[/url] .
Diplomi_osoi
18 Oct 25 at 3:14 am
mostbet uz online [url=https://www.mostbet4185.ru]https://www.mostbet4185.ru[/url]
mostbet_uz_fier
18 Oct 25 at 3:16 am
Amazing! This blog looks exactly like my old one!
It’s on a completely different subject but it has pretty much the same page layout and design. Excellent choice of colors!
building designers near me
18 Oct 25 at 3:19 am
virtual experience
MichaelSig
18 Oct 25 at 3:20 am
В Сочи клиника «Детокс» предлагает лечение запоя в стационаре с круглосуточным медицинским контролем. Здесь пациент получает безопасное и эффективное восстановление.
Ознакомиться с деталями – [url=https://vyvod-iz-zapoya-sochi22.ru/]врач вывод из запоя[/url]
BillyWoult
18 Oct 25 at 3:20 am
https://t.me/s/Official_1xbet_1xbet/1625
Josephadvem
18 Oct 25 at 3:20 am
В Краснодаре клиника «Детокс» предоставляет услугу вызова нарколога на дом. Специалисты приедут к вам в течение 1–2 часов, проведут осмотр и назначат необходимое лечение. Все процедуры проводятся анонимно и с соблюдением конфиденциальности.
Подробнее можно узнать тут – [url=https://narkolog-na-dom-krasnodar25.ru/]нарколог капельница на дом[/url]
JamieOvedy
18 Oct 25 at 3:21 am
Если самостоятельные попытки выйти из запоя не даются, в стационаре «Частного Медика 24» в Воронеже оказывают помощь профессионально: круглосуточное наблюдение, дозированные капельницы, поддержка сердечно-сосудистой системы, нормализация сна и работы внутренних органов.
Получить больше информации – [url=https://vyvod-iz-zapoya-v-stacionare-voronezh22.ru/]вывод из запоя в стационаре клиника[/url]
RichardJuids
18 Oct 25 at 3:24 am
https://t.me/Official_1xbet_1xbet/1742
Josephadvem
18 Oct 25 at 3:26 am
https://t.me/Official_1xbet_1xbet/1788
Josephadvem
18 Oct 25 at 3:27 am
Hola! I’ve been following your web site for some time now and finally got the courage to go ahead and give you a shout
out from New Caney Texas! Just wanted to say keep up
the good work!
нужна команда разработчиков
18 Oct 25 at 3:27 am
linebet senegal apk iphone
telecharger linebet senegal
18 Oct 25 at 3:29 am
мостбет уз кириш [url=www.mostbet4185.ru]www.mostbet4185.ru[/url]
mostbet_uz_gzer
18 Oct 25 at 3:29 am
Слив курсов ЕГЭ география https://courses-ege.ru
courses-ege-378
18 Oct 25 at 3:31 am
Hello it’s me, I am also visiting this web page on a regular basis,
this site is really good and the people are in fact sharing pleasant thoughts.
Click here
18 Oct 25 at 3:31 am
That is a very good tip particularly to those fresh to the blogosphere.
Simple but very accurate info… Thank you for sharing this one.
A must read post!
Visit
18 Oct 25 at 3:35 am
This is my first time pay a quick visit at here and
i am really impressed to read everthing at single place.
roofing service St Louis MO
18 Oct 25 at 3:35 am
При обращении на горячую линию пациент может получить бесплатную телефонную консультацию. Врач уточняет состояние, собирает предварительный анамнез и согласовывает время визита.
Узнать больше – http://narkologicheskaya-klinika-rostov13.ru/
JosephNoirl
18 Oct 25 at 3:37 am
Вызов нарколога на дом в Челябинске обеспечивает возможность проведения полного обследования и первичной терапии в условиях, привычных для пациента. Специалисты оказывают услуги круглосуточно, что позволяет своевременно купировать острые состояния, связанные с алкогольной интоксикацией, абстинентным синдромом или осложнениями наркозависимости.
Выяснить больше – [url=https://narkolog-na-dom-chelyabinsk13.ru/]частный нарколог на дом[/url]
Charlesnak
18 Oct 25 at 3:38 am
Если вы ищете надежную клинику для вывода из запоя в Сочи, обратитесь в «Детокс». Здесь опытные специалисты окажут необходимую помощь в стационаре. Услуга доступна круглосуточно, анонимно и начинается от 2000 ?.
Получить дополнительные сведения – http://vyvod-iz-zapoya-sochi23.ru
Gordontrive
18 Oct 25 at 3:41 am
ИТ-специалисты — новые инженеры мира кракен онион кракен onion сайт kra ссылка kraken сайт
RichardPep
18 Oct 25 at 3:42 am
экстренный вывод из запоя
narkolog-krasnodar019.ru
экстренный вывод из запоя краснодар
vivodzapojkrasnodarNeT
18 Oct 25 at 3:46 am
This is my first time pay a quick visit at here and i am in fact impressed to read everthing at single place.
quirky car designs Sweatshirts
18 Oct 25 at 3:46 am
Highly descriptive blog, I loved that a lot. Will there be a part 2?
NexisFin Avis
18 Oct 25 at 3:47 am
https://pslk.net/5offgtif
Sammyjunny
18 Oct 25 at 3:47 am
услуги по согласованию перепланировки квартиры [url=soglasovanie-pereplanirovki-kvartiry3.ru]soglasovanie-pereplanirovki-kvartiry3.ru[/url] .
soglasovanie pereplanirovki kvartiri _nrPi
18 Oct 25 at 3:47 am
куплю диплом младшей медсестры [url=http://www.frei-diplom14.ru]http://www.frei-diplom14.ru[/url] .
Diplomi_lfoi
18 Oct 25 at 3:48 am
https://t.me/Official_1xbet_1xbet/1632
Josephadvem
18 Oct 25 at 3:49 am
Сливы курсов ЕГЭ английский https://courses-ege.ru
courses-ege-229
18 Oct 25 at 3:52 am
купить диплом в нижневартовске [url=http://rudik-diplom12.ru/]http://rudik-diplom12.ru/[/url] .
Diplomi_qoPi
18 Oct 25 at 3:53 am
https://t.me/Official_1xbet_1xbet/1777
Josephadvem
18 Oct 25 at 3:55 am
mostbet ro‘yxatdan o‘tish [url=mostbet4185.ru]mostbet4185.ru[/url]
mostbet_uz_ceer
18 Oct 25 at 3:57 am
https://wiki.fuzokudb.com/fdb/%E5%88%A9%E7%94%A8%E8%80%85:ElbaPerryman8
https://wiki.fuzokudb.com/fdb/:ElbaPerryman8
18 Oct 25 at 3:59 am
Hello there! Do you know if they make any plugins to safeguard against hackers?
I’m kinda paranoid about losing everything I’ve worked hard
on. Any suggestions?
Visit
18 Oct 25 at 4:00 am
Салон «Флорион» собирает из свежих цветов изящные композиции в шляпных коробках, сумочках и конвертах — от романтических «сердец из цветов» до новогодних ансамблей с еловыми ветками. На витрине цены прозрачны, есть фильтры по цвету и повoду, а доставка по Москве оперативна. Выберите настроение подарка на https://www.florion.ru/catalog/cvety-v-korobke — «Отблеск солнечного света», «Ягодный щербет» и другие бестселлеры уже ждут. Дарите эмоции в удобном формате, который всегда уместен.
myxiqPlaib
18 Oct 25 at 4:01 am
mostbet uz kirish скачать [url=http://mostbet4185.ru/]http://mostbet4185.ru/[/url]
mostbet_uz_yxer
18 Oct 25 at 4:03 am