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=medicinskaya-tehnika.ru]medicinskaya-tehnika.ru[/url] .
medicinskaya tehnika_grEi
27 Oct 25 at 5:44 am
kraken обмен
kraken darknet market
Henryamerb
27 Oct 25 at 5:45 am
кракен vk2
kraken vk3
Henryamerb
27 Oct 25 at 5:45 am
Ich bin beeindruckt von Cat Spins Casino, es bietet eine dynamische Erfahrung. Es gibt unzahlige packende Spiele, mit traditionellen Tischspielen. Er bietet einen gro?artigen Vorteil. Der Service ist rund um die Uhr verfugbar. Gewinne werden ohne Wartezeit uberwiesen, dennoch mehr Bonusoptionen waren top. Im Gro?en und Ganzen, Cat Spins Casino ist perfekt fur Casino-Liebhaber. Zusatzlich die Seite ist schnell und einladend, das Vergnugen maximiert. Ein tolles Feature die haufigen Turniere fur mehr Spa?, regelma?ige Boni bieten.
Mehr erhalten|
NightRiseos9zef
27 Oct 25 at 5:46 am
Sildenafil utan recept: mannens apotek – onlineapotek för män
Jesuskax
27 Oct 25 at 5:47 am
купить диплом в выборге [url=http://rudik-diplom11.ru/]http://rudik-diplom11.ru/[/url] .
Diplomi_lsMi
27 Oct 25 at 5:47 am
I read this paragraph fully regarding the difference
of newest and preceding technologies, it’s amazing article.
Website mua bán vũ khí
27 Oct 25 at 5:48 am
купить диплом матроса [url=https://rudik-diplom2.ru]купить диплом матроса[/url] .
Diplomi_zkpi
27 Oct 25 at 5:49 am
оборудование медицинское [url=http://medicinskoe–oborudovanie.ru]оборудование медицинское[/url] .
medicinskoe oborydovanie_qnei
27 Oct 25 at 5:50 am
kraken сайт
кракен маркет
Henryamerb
27 Oct 25 at 5:50 am
Backlinks for your site
Effective across all subjects of the platform.
I create inbound links to your page.
Such inbound links draw in web crawlers to the platform, that is very important for positioning, thus it is important to promote a platform lacking issues that will interfere with ranking.
Positioning is harmless for your site!
I avoid filling in feedback boxes, (contact forms negatively impact the resource as there are complaints from administrators).
Posting takes place in authorized locations.
Links are added to updated frequently updated index. Numerous resources in the list.
Backlinks for your site
27 Oct 25 at 5:50 am
The best on one page: https://clinicamaddarena.com.br
RodneyJaick
27 Oct 25 at 5:50 am
protraderacademy – Updates are consistent, and the information always feels relevant and current.
Shenika Rethman
27 Oct 25 at 5:51 am
купить диплом с занесением в реестр в украине [url=www.frei-diplom1.ru]www.frei-diplom1.ru[/url] .
Diplomi_vtOi
27 Oct 25 at 5:51 am
медицинская техника [url=http://medicinskaya-tehnika.ru/]медицинская техника[/url] .
medicinskaya tehnika_keEi
27 Oct 25 at 5:51 am
куплю диплом цена [url=https://www.rudik-diplom8.ru]куплю диплом цена[/url] .
Diplomi_ebMt
27 Oct 25 at 5:52 am
диплом государственного образца купить реестр [url=http://www.frei-diplom4.ru]диплом государственного образца купить реестр[/url] .
Diplomi_ddOl
27 Oct 25 at 5:53 am
купить диплом в каспийске [url=www.rudik-diplom3.ru/]www.rudik-diplom3.ru/[/url] .
Diplomi_rvei
27 Oct 25 at 5:53 am
мед оборудование [url=http://medicinskoe–oborudovanie.ru/]мед оборудование[/url] .
medicinskoe oborydovanie_gmei
27 Oct 25 at 5:53 am
купить диплом в губкине [url=www.rudik-diplom10.ru]www.rudik-diplom10.ru[/url] .
Diplomi_zaSa
27 Oct 25 at 5:54 am
медтехника [url=medicinskaya-tehnika.ru]medicinskaya-tehnika.ru[/url] .
medicinskaya tehnika_caEi
27 Oct 25 at 5:55 am
купить диплом медсестры [url=www.rudik-diplom2.ru/]купить диплом медсестры[/url] .
Diplomi_lypi
27 Oct 25 at 5:55 am
купить диплом в барнауле [url=https://rudik-diplom15.ru]купить диплом в барнауле[/url] .
Diplomi_kyPi
27 Oct 25 at 5:56 am
kraken tor
kraken vk5
Henryamerb
27 Oct 25 at 5:56 am
купить диплом в таганроге [url=http://rudik-diplom11.ru/]http://rudik-diplom11.ru/[/url] .
Diplomi_tfMi
27 Oct 25 at 5:56 am
I do consider all of the concepts you have introduced for your post.
They are really convincing and can certainly work.
Still, the posts are too quick for starters. May just you please prolong them a bit from subsequent time?
Thank you for the post.
https://ontologyschmology.com/
Gizi Anak
27 Oct 25 at 5:57 am
You could certainly see your enthusiasm in the work you write.
The world hopes for more passionate writers like you who aren’t
afraid to mention how they believe. All the time follow your
heart.
Immediate Fastx
27 Oct 25 at 5:57 am
Expanded version here: https://barracamalvin.com.uy
Harleyclomi
27 Oct 25 at 5:58 am
Additional materials: https://baldebranco.com.br
Davidcub
27 Oct 25 at 5:59 am
В данной обзорной статье представлены интригующие факты, которые не оставят вас равнодушными. Мы критикуем и анализируем события, которые изменили наше восприятие мира. Узнайте, что стоит за новыми открытиями и как они могут изменить ваше восприятие реальности.
Обратиться к источнику – https://stage-curacao.nl/curacao-strandjes-playa-kanoa
VernonChoox
27 Oct 25 at 5:59 am
медтехника [url=https://www.medicinskaya-tehnika.ru]https://www.medicinskaya-tehnika.ru[/url] .
medicinskaya tehnika_ylEi
27 Oct 25 at 6:00 am
купить диплом в туле [url=https://rudik-diplom2.ru/]купить диплом в туле[/url] .
Diplomi_lkpi
27 Oct 25 at 6:00 am
купить легальный диплом техникума [url=https://frei-diplom10.ru]купить легальный диплом техникума[/url] .
Diplomi_tqEa
27 Oct 25 at 6:01 am
https://kuljetus.jdmmediagroup.com/2025/10/08/bukmeker-melbet-2025-obzor-prognozy/
Michaelcop
27 Oct 25 at 6:01 am
Эта публикация дает возможность задействовать различные источники информации и представить их в удобной форме. Читатели смогут быстро найти нужные данные и получить ответы на интересующие их вопросы. Мы стремимся к четкости и доступности материала для всех!
Посмотреть подробности – https://hoctienganhtaiphilippines.vn/lien-he
DanielKiz
27 Oct 25 at 6:01 am
https://baroon.pk/melbet-2025-obzor-bk-medicisoft/
Michaelcop
27 Oct 25 at 6:02 am
купить диплом в омске [url=http://www.rudik-diplom15.ru]купить диплом в омске[/url] .
Diplomi_puPi
27 Oct 25 at 6:02 am
экстренное вытрезвление в москве [url=https://www.narkologicheskaya-klinika-24.ru]https://www.narkologicheskaya-klinika-24.ru[/url] .
narkologicheskaya klinika_vaSr
27 Oct 25 at 6:03 am
apotek online utan recept: köpa Viagra online Sverige – Sildenafil utan recept
RandySkync
27 Oct 25 at 6:04 am
kraken СПб
кракен даркнет маркет
Henryamerb
27 Oct 25 at 6:05 am
купить диплом о высшем образовании проведенный [url=https://frei-diplom1.ru/]купить диплом о высшем образовании проведенный[/url] .
Diplomi_qlOi
27 Oct 25 at 6:05 am
медицинское оборудование для больниц [url=https://medicinskoe–oborudovanie.ru/]medicinskoe–oborudovanie.ru[/url] .
medicinskoe oborydovanie_duei
27 Oct 25 at 6:05 am
кракен онион
kraken vk3
Henryamerb
27 Oct 25 at 6:05 am
officiële Sildenafil webshop: Heren Gezondheid – goedkope Viagra tabletten online
Jesuskax
27 Oct 25 at 6:06 am
медицинская техника [url=http://www.medicinskaya-tehnika.ru]медицинская техника[/url] .
medicinskaya tehnika_ybEi
27 Oct 25 at 6:07 am
купить диплом в нефтекамске [url=https://www.rudik-diplom5.ru]купить диплом в нефтекамске[/url] .
Diplomi_eoma
27 Oct 25 at 6:07 am
купить диплом с занесением в реестр в украине [url=http://www.frei-diplom5.ru]http://www.frei-diplom5.ru[/url] .
Diplomi_slPa
27 Oct 25 at 6:07 am
купить диплом с занесением в реестр в иркутске [url=www.frei-diplom6.ru]купить диплом с занесением в реестр в иркутске[/url] .
Diplomi_wqOl
27 Oct 25 at 6:08 am
Great post. I was checking constantly this blog and I’m inspired!
Very helpful information specially the ultimate part :
) I handle such information a lot. I was looking for this
particular info for a long time. Thanks and good luck.
ads pepek
27 Oct 25 at 6:09 am
https://ramongarciamaquinas.com/sloty-melbet-obzor-2025/
TerryDob
27 Oct 25 at 6:09 am