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=www.frei-diplom8.ru]где купить диплом техникума хорошую[/url] .
Diplomi_tasr
17 Oct 25 at 8:34 am
keywordfuel.click – Found a few useful articles already, excited to dig deeper.
Lavera Eheler
17 Oct 25 at 8:36 am
как купить диплом с проводкой [url=http://frei-diplom2.ru/]как купить диплом с проводкой[/url] .
Diplomi_yuEa
17 Oct 25 at 8:37 am
Публикация предлагает читателю не просто информацию, а инструменты для анализа и саморазвития. Мы стимулируем критическое мышление, предлагая различные точки зрения и призывая к самостоятельному поиску решений.
Обратиться к источнику – https://pfendustri.com/katodik-koruma-olcumu
HarryBon
17 Oct 25 at 8:37 am
купить диплом с внесением в реестр [url=https://www.frei-diplom1.ru]купить диплом с внесением в реестр[/url] .
Diplomi_juOi
17 Oct 25 at 8:38 am
купить диплом медицинского училища [url=www.rudik-diplom7.ru]купить диплом медицинского училища[/url] .
Diplomi_biPl
17 Oct 25 at 8:39 am
ZenCare Meds: trusted online pharmacy USA – trusted online pharmacy USA
AndrewPal
17 Oct 25 at 8:39 am
netvortex.click – Overall experience is positive, feels professional and reliable.
Cedrick Arvay
17 Oct 25 at 8:40 am
купить диплом колледжа в нижнем тагиле [url=https://www.frei-diplom8.ru]https://www.frei-diplom8.ru[/url] .
Diplomi_umsr
17 Oct 25 at 8:40 am
pagepilot.click – Pages load fast and visuals are crisp, very pleasant.
Reuben Bosson
17 Oct 25 at 8:40 am
купить диплом техникума в мурманске [url=http://frei-diplom12.ru]купить диплом техникума в мурманске[/url] .
Diplomi_qgPt
17 Oct 25 at 8:41 am
The Minotaurus presale offers real incentives like bonus tokens. $MTAUR’s audited ecosystem builds confidence. Game beta can’t come soon enough.
mtaur coin
WilliamPargy
17 Oct 25 at 8:42 am
I feel this is one of the so much important info
for me. And i’m glad reading your article. However wanna commentary on some general issues, The
website taste is wonderful, the articles is really great : D.
Just right activity, cheers
정품 비아그라 구매
17 Oct 25 at 8:42 am
trafficcraft.click – It performs well on mobile, I had no issues at all.
Hank Parido
17 Oct 25 at 8:42 am
linebet testflight
linebet apk download
17 Oct 25 at 8:43 am
программы https://softprogram-free.ru/
Maximodaf
17 Oct 25 at 8:44 am
mostbet apk skachat [url=http://mostbet4185.ru]http://mostbet4185.ru[/url]
mostbet_uz_gcer
17 Oct 25 at 8:44 am
ремонт двигатлей ауди в москве [url=www.vc.ru/id5379722/2274350-remont-dvigateley-acura-audi-bmw-i-drugikh-v-moskve/]www.vc.ru/id5379722/2274350-remont-dvigateley-acura-audi-bmw-i-drugikh-v-moskve/[/url] .
Remont dvigatelei Dvigovichkoff_lyKa
17 Oct 25 at 8:45 am
купить диплом занесением реестр отзывы [url=https://www.frei-diplom3.ru]купить диплом занесением реестр отзывы[/url] .
Diplomi_kgKt
17 Oct 25 at 8:45 am
купить диплом в хабаровске [url=https://www.rudik-diplom10.ru]https://www.rudik-diplom10.ru[/url] .
Diplomi_asSa
17 Oct 25 at 8:45 am
sitepilot.click – The color scheme is subtle and pleasing; doesn’t hurt the eyes.
Hayley Misty
17 Oct 25 at 8:45 am
mostbet uz [url=http://mostbet4185.ru]http://mostbet4185.ru[/url]
mostbet_uz_dger
17 Oct 25 at 8:48 am
ремонт двигателей бмв [url=http://vc.ru/id5379722/2274350-remont-dvigateley-acura-audi-bmw-i-drugikh-v-moskve]http://vc.ru/id5379722/2274350-remont-dvigateley-acura-audi-bmw-i-drugikh-v-moskve[/url] .
Remont dvigatelei Dvigovichkoff_llKa
17 Oct 25 at 8:49 am
купить диплом об окончании техникума [url=www.frei-diplom12.ru]купить диплом об окончании техникума[/url] .
Diplomi_hxPt
17 Oct 25 at 8:50 am
https://t.me/Official_1xbet_1xbet/s/885
AlbertEnark
17 Oct 25 at 8:53 am
купить диплом врача с занесением в реестр [url=frei-diplom1.ru]купить диплом врача с занесением в реестр[/url] .
Diplomi_zhOi
17 Oct 25 at 8:53 am
куплю диплом цена [url=rudik-diplom2.ru]куплю диплом цена[/url] .
Diplomi_mvpi
17 Oct 25 at 8:54 am
successbridge – Always something meaningful to read, not just surface level.
Karisa Leonaggeo
17 Oct 25 at 8:55 am
купить диплом в махачкале [url=http://rudik-diplom10.ru]купить диплом в махачкале[/url] .
Diplomi_pwSa
17 Oct 25 at 8:55 am
Выбор краски для внутренних работ – ответственный шаг. Важно учитывать не только эстетические предпочтения, но и технические характеристики материала. Краски для внутренних работ должны быть экологически безопасными, не выделять вредных веществ и обладать хорошей адгезией к дереву. Также важно выбирать краску, устойчивую к истиранию, влаге и перепадам температур, особенно если речь идет об окрашивании деревянных полов или мебели в ванной комнате.
Существует множество видов красок для дерева, предназначенных для внутренних работ. Акриловые краски, например, отличаются высокой экологичностью, быстро сохнут и образуют прочное покрытие. Масляные краски, в свою очередь, славятся своей долговечностью и насыщенностью цвета, но требуют более длительной сушки. Алкидные краски – компромисс между этими двумя. А какая [url=http://www.bisound.com/forum/showthread.php?p=2862914#post2862914]краска по дереву для внутренних работ устраивает Вас?[/url]
Wilburevets
17 Oct 25 at 8:56 am
With $MTAUR coin, collecting in-game currency while running mazes is addictive. Presale bonuses for vesting make holding appealing. Team’s track record impresses me.
minotaurus token
WilliamPargy
17 Oct 25 at 8:56 am
купить диплом техникума в петрозаводске [url=http://frei-diplom12.ru/]купить диплом техникума в петрозаводске[/url] .
Diplomi_mcPt
17 Oct 25 at 8:56 am
Эта публикация дает возможность задействовать различные источники информации и представить их в удобной форме. Читатели смогут быстро найти нужные данные и получить ответы на интересующие их вопросы. Мы стремимся к четкости и доступности материала для всех!
Почему это важно? – https://mcpakistan.com/education-reference/top-10-must-read-ebooks-finding-for-personal-growth
MarcusCehaw
17 Oct 25 at 8:59 am
https://t.me/Official_1xbet_1xbet/s/1225
AlbertEnark
17 Oct 25 at 8:59 am
Эта статья предлагает живое освещение актуальной темы с множеством интересных фактов. Мы рассмотрим ключевые моменты, которые делают данную тему важной и актуальной. Подготовьтесь к насыщенному путешествию по неизвестным аспектам и узнайте больше о значимых событиях.
Хочешь знать всё? – https://www.bpml.in/bpml-fevicon
Donaldirofe
17 Oct 25 at 9:00 am
https://t.me/Official_1xbet_1xbet/s/506
AlbertEnark
17 Oct 25 at 9:00 am
Публикация предлагает читателю не просто информацию, а инструменты для анализа и саморазвития. Мы стимулируем критическое мышление, предлагая различные точки зрения и призывая к самостоятельному поиску решений.
Смотри, что ещё есть – https://template65.webekspor.com/?p=1
RolandToinc
17 Oct 25 at 9:01 am
купить диплом в димитровграде [url=www.rudik-diplom10.ru]купить диплом в димитровграде[/url] .
Diplomi_gzSa
17 Oct 25 at 9:02 am
Spam aman dan cuan.
slot spam
17 Oct 25 at 9:02 am
купить диплом с занесением в реестр в иркутске [url=http://frei-diplom1.ru/]купить диплом с занесением в реестр в иркутске[/url] .
Diplomi_emOi
17 Oct 25 at 9:03 am
купить диплом в братске [url=http://www.rudik-diplom2.ru]купить диплом в братске[/url] .
Diplomi_bnpi
17 Oct 25 at 9:04 am
https://asian-cars.ru/
Michaelbed
17 Oct 25 at 9:05 am
https://emigranto.ru/
Michaelbed
17 Oct 25 at 9:05 am
медсестра которая купила диплом врача [url=http://www.frei-diplom13.ru]медсестра которая купила диплом врача[/url] .
Diplomi_rhkt
17 Oct 25 at 9:06 am
купить диплом колледжа в москве [url=www.frei-diplom9.ru]www.frei-diplom9.ru[/url] .
Diplomi_esea
17 Oct 25 at 9:06 am
Just extended $MTAUR vesting for extras. ICO’s tokenomics balanced. Play-to-earn future here.
minotaurus ico
WilliamPargy
17 Oct 25 at 9:06 am
Этот информационный материал привлекает внимание множеством интересных деталей и необычных ракурсов. Мы предлагаем уникальные взгляды на привычные вещи и рассматриваем вопросы, которые волнуют общество. Будьте в курсе актуальных тем и расширяйте свои знания!
Откройте для себя больше – https://www.skyhilocksmith.com/locksmith/jeep-locksmith-colorado-springs
RolandJut
17 Oct 25 at 9:06 am
диплом колледжа купить екатеринбург [url=http://frei-diplom8.ru/]http://frei-diplom8.ru/[/url] .
Diplomi_xtsr
17 Oct 25 at 9:07 am
buy Doxycycline [url=https://zencaremeds.shop/#]order medicine discreetly USA[/url] order medicine discreetly USA
CareyMag
17 Oct 25 at 9:08 am
Покупка номера телефона
Ricardopam
17 Oct 25 at 9:09 am