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!
1xbet resmi giri? [url=www.1xbet-giris-6.com]www.1xbet-giris-6.com[/url] .
1xbet giris_mrsl
24 Oct 25 at 2:27 am
Hello i am kavin, its my first occasion to commenting anyplace, when i
read this paragraph i thought i could also create comment
due to this sensible post.
Olonixis
24 Oct 25 at 2:28 am
1xbet mobil giri? [url=http://1xbet-giris-5.com/]http://1xbet-giris-5.com/[/url] .
1xbet giris_nrSa
24 Oct 25 at 2:28 am
I’m gone to tell my little brother, that he should
also go to see this weblog on regular basis to obtain updated from
most up-to-date reports.
local Home remodeling
24 Oct 25 at 2:29 am
https://brandies.bizlisting.cloud/melbet-promo-code-2026-no-deposit-bonus/
Raymondtaf
24 Oct 25 at 2:30 am
birxbet giri? [url=www.1xbet-giris-9.com/]www.1xbet-giris-9.com/[/url] .
1xbet giris_jyon
24 Oct 25 at 2:30 am
купить диплом в балаково [url=http://rudik-diplom9.ru]купить диплом в балаково[/url] .
Diplomi_nwei
24 Oct 25 at 2:32 am
1xbwt giri? [url=www.1xbet-giris-9.com]www.1xbet-giris-9.com[/url] .
1xbet giris_muon
24 Oct 25 at 2:32 am
birxbet [url=1xbet-giris-2.com]1xbet-giris-2.com[/url] .
1xbet giris_ttPt
24 Oct 25 at 2:34 am
1xbet yeni giri? [url=https://1xbet-giris-8.com/]1xbet yeni giri?[/url] .
1xbet giris_unPn
24 Oct 25 at 2:34 am
Эти данные являются основой для составления индивидуального плана лечения, позволяющего оперативно скорректировать терапевтические меры.
Ознакомиться с деталями – https://narcolog-na-dom-ryazan00.ru/narkolog-na-dom-kruglosutochno-ryazan
FrankSwils
24 Oct 25 at 2:36 am
1xbet resmi sitesi [url=http://1xbet-giris-2.com]http://1xbet-giris-2.com[/url] .
1xbet giris_bfPt
24 Oct 25 at 2:36 am
wonderful issues altogether, you simply received a logo new reader.
What may you recommend in regards to your put up that you simply made a
few days in the past? Any sure?
Quantserax
24 Oct 25 at 2:36 am
pastillas de potencia masculinas [url=https://confiafarmacia.shop/#]farmacia online para hombres[/url] farmacia online para hombres
Davidduese
24 Oct 25 at 2:36 am
кракен vk5
кракен даркнет
JamesDaync
24 Oct 25 at 2:38 am
1xbet yeni giri? adresi [url=www.1xbet-giris-6.com/]1xbet yeni giri? adresi[/url] .
1xbet giris_gbsl
24 Oct 25 at 2:39 am
1xbet giri? linki [url=www.1xbet-giris-4.com/]www.1xbet-giris-4.com/[/url] .
1xbet giris_xrSa
24 Oct 25 at 2:39 am
1вин верификация [url=1win5518.ru]1win5518.ru[/url]
1win_kg_jikl
24 Oct 25 at 2:40 am
кракен сайт
кракен vpn
кракен vpn
24 Oct 25 at 2:40 am
1xbet guncel [url=https://www.1xbet-giris-5.com]1xbet guncel[/url] .
1xbet giris_rgSa
24 Oct 25 at 2:40 am
1xbet tr giri? [url=https://1xbet-giris-1.com/]1xbet-giris-1.com[/url] .
1xbet giris_olkt
24 Oct 25 at 2:40 am
можно ли купить диплом медсестры [url=www.frei-diplom14.ru/]можно ли купить диплом медсестры[/url] .
Diplomi_smoi
24 Oct 25 at 2:41 am
кракен
кракен vk3
кракен маркетплейс
24 Oct 25 at 2:41 am
trustedpartnergroup – Overall positive first impression; I’ll revisit and track how their content evolves.
Marvin Emmerling
24 Oct 25 at 2:42 am
ultimateprofitplan – Found a useful tip this morning that I plan to experiment with this week.
Elouise Drewing
24 Oct 25 at 2:45 am
1x lite [url=http://www.1xbet-giris-6.com]http://www.1xbet-giris-6.com[/url] .
1xbet giris_eosl
24 Oct 25 at 2:46 am
xbet giri? [url=http://1xbet-giris-9.com/]xbet giri?[/url] .
1xbet giris_vnon
24 Oct 25 at 2:46 am
pastillas de potencia masculinas: comprar Sildenafilo sin receta – farmacia confiable en España
Jesuskax
24 Oct 25 at 2:47 am
Полезное руководство по получению бонусов и тому, где искать официальную информацию: в пункте о регистрации мы вставляем ссылку на https://bergkompressor.ru/news/artcles/?1xbet_promokod_pri_registracii_bonus_5.html, дабы показать пример записи в регистрационном поле. Разъясняем особенности работы с промо-купонами и способы их получения.
Petercet
24 Oct 25 at 2:47 am
1xbet g?ncel [url=1xbet-giris-5.com]1xbet g?ncel[/url] .
1xbet giris_bpSa
24 Oct 25 at 2:49 am
Подход клиники строится на принципе минимально достаточной фармакотерапии: мы исключаем полипрагмазию, не «усиливаем схему» за счёт случайных препаратов и всегда сверяем назначения с вашими постоянными лекарствами (антигипертензивные, антиаритмические, сахароснижающие и др.). Состав инфузии не бывает универсальным — он зависит от жалоб, возраста, гидратационного статуса, результатов экспресс-тестов и логистики ближайших часов (работа/сон/домашняя среда). Благодаря этому эффект становится управляемым, а «первая тихая ночь» достигается без риска угнетения дыхания и дневной «ватности».
Изучить вопрос глубже – [url=https://narcolog-na-dom-ekaterinburg00.ru/]врач нарколог на дом екатеринбург[/url]
RichardIsope
24 Oct 25 at 2:49 am
1xbet g?ncel [url=http://1xbet-giris-2.com]http://1xbet-giris-2.com[/url] .
1xbet giris_plPt
24 Oct 25 at 2:50 am
kraken
kraken marketplace
кракен vk5
24 Oct 25 at 2:50 am
где купить дипломы медсестры [url=http://www.frei-diplom14.ru]где купить дипломы медсестры[/url] .
Diplomi_upoi
24 Oct 25 at 2:50 am
Купить диплом о высшем образовании поспособствуем. Купить диплом техникума, колледжа в Улан-Удэ – [url=http://diplomybox.com/kupit-diplom-tekhnikuma-kolledzha-v-ulan-ude/]diplomybox.com/kupit-diplom-tekhnikuma-kolledzha-v-ulan-ude[/url]
Cazrqod
24 Oct 25 at 2:50 am
Hmm it appears like your website ate my first comment (it was extremely
long) so I guess I’ll just sum it up what I wrote and say, I’m thoroughly enjoying your
blog. I as well am an aspiring blog blogger but I’m still new to
everything. Do you have any tips for first-time blog writers?
I’d really appreciate it.
มหาลัยอันดับ 1
24 Oct 25 at 2:51 am
kraken tor
kraken vk5
JamesDaync
24 Oct 25 at 2:52 am
Регулярная чистка раковины предотвращает засоры, появление неприятного запаха, плесени,
а также способствует поддержанию
общей чистоты и порядка на кухне.
клининг помещений спб
24 Oct 25 at 2:53 am
businesssuccesshub – The community feedback section also adds helpful real-world perspective.
Cami Tototzintle
24 Oct 25 at 2:54 am
1x lite [url=https://1xbet-giris-1.com/]1xbet-giris-1.com[/url] .
1xbet giris_jjkt
24 Oct 25 at 2:55 am
Такая систематизация показывает, что врач действует целенаправленно, устраняя не только симптомы, но и их физиологические последствия.
Подробнее тут – [url=https://narcolog-na-dom-ryazan0.ru/]частный нарколог на дом[/url]
Richardshado
24 Oct 25 at 2:56 am
1xbet yeni giri? adresi [url=https://1xbet-giris-6.com/]1xbet yeni giri? adresi[/url] .
1xbet giris_zusl
24 Oct 25 at 2:57 am
I’m extremely impressed together with your writing skills
as neatly as with the layout in your weblog. Is this a paid subject or did you modify it your self?
Either way stay up the excellent quality writing, it is rare to peer a great blog like
this one these days..
m bs2web at
24 Oct 25 at 2:58 am
бк 1 вин официальный сайт [url=https://1win5518.ru]бк 1 вин официальный сайт[/url]
1win_kg_wlkl
24 Oct 25 at 2:58 am
1xbet tr giri? [url=https://1xbet-giris-5.com/]https://1xbet-giris-5.com/[/url] .
1xbet giris_itSa
24 Oct 25 at 2:59 am
1win vk [url=https://1win5518.ru]1win vk[/url]
1win_kg_rokl
24 Oct 25 at 3:00 am
Good way of telling, and good article to get information on the topic of my presentation focus, which i am going
to present in school.
Finxor GPT Ervaringen
24 Oct 25 at 3:01 am
1win букмекерская скачать приложение [url=https://1win5518.ru/]1win букмекерская скачать приложение[/url]
1win_kg_tlkl
24 Oct 25 at 3:01 am
1xbetgiri? [url=https://www.1xbet-giris-2.com]https://www.1xbet-giris-2.com[/url] .
1xbet giris_kgPt
24 Oct 25 at 3:01 am
кракен клиент
kraken vk3
кракен vk6
24 Oct 25 at 3:01 am