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://nuruzmarketing.com/melbet-mobilnaya-versiya-2025-obzor/
Michaelcop
27 Oct 25 at 7:10 am
купить диплом о высшем образовании с занесением в реестр в кемерово [url=https://frei-diplom4.ru/]купить диплом о высшем образовании с занесением в реестр в кемерово[/url] .
Diplomi_pqOl
27 Oct 25 at 7:11 am
https://mbg.puree.maison/casino-melbet-obzor-2025/
MichaelSeemn
27 Oct 25 at 7:11 am
купить диплом в когалыме [url=https://rudik-diplom10.ru]https://rudik-diplom10.ru[/url] .
Diplomi_fySa
27 Oct 25 at 7:11 am
https://radiodelprogresopucallpafm.com/melbet-2025-obzor-bk/
Williamsoamn
27 Oct 25 at 7:12 am
кракен онлайн
кракен даркнет
Henryamerb
27 Oct 25 at 7:12 am
медицинские приборы [url=https://medicinskoe–oborudovanie.ru/]medicinskoe–oborudovanie.ru[/url] .
medicinskoe oborydovanie_qcei
27 Oct 25 at 7:12 am
Very rapidly this web page will be famous amid all blogging users, due
to it’s nice posts
jasa buzzer
27 Oct 25 at 7:13 am
Эта публикация завернет вас в вихрь увлекательного контента, сбрасывая стереотипы и открывая двери к новым идеям. Каждый абзац станет для вас открытием, полным ярких примеров и впечатляющих достижений. Подготовьтесь быть вовлеченными и удивленными каждый раз, когда продолжите читать.
Секреты успеха внутри – https://tourpassion.com/ancient-thera-unraveling-the-mysteries-of-a-timeless-destination
DanielKiz
27 Oct 25 at 7:13 am
Viagra generico con pagamento sicuro: Viagra generico online Italia – miglior sito per acquistare Sildenafil online
RandySkync
27 Oct 25 at 7:14 am
купить диплом в уфе [url=https://rudik-diplom2.ru]купить диплом в уфе[/url] .
Diplomi_gipi
27 Oct 25 at 7:15 am
It is appropriate time to make some plans for the future and it’s
time to be happy. I have read this post and if I
could I wish to suggest you few interesting things or suggestions.
Perhaps you can write next articles referring to this article.
I wish to read more things about it!
AU 88
27 Oct 25 at 7:15 am
https://pos303.co/melbet-vhod-zerkalo-2025/
MichaelSeemn
27 Oct 25 at 7:16 am
What a data of un-ambiguity and preserveness of valuable experience regarding
unexpected feelings.
HM88
27 Oct 25 at 7:16 am
kraken ios
кракен vk3
Henryamerb
27 Oct 25 at 7:18 am
https://visualdisplays.in/skachat-besplatno-melbet-na-android-2025/
TerryDob
27 Oct 25 at 7:19 am
купить диплом о среднем образовании с занесением в реестр [url=https://frei-diplom4.ru]купить диплом о среднем образовании с занесением в реестр[/url] .
Diplomi_mpOl
27 Oct 25 at 7:19 am
I like the helpful info you provide on your articles. I
will bookmark your blog and check again here regularly.
I am rather sure I’ll learn a lot of new stuff proper here!
Good luck for the next!
https://uzairnaqash.zoom.us/
27 Oct 25 at 7:21 am
купить диплом с проводкой одной [url=www.frei-diplom5.ru/]купить диплом с проводкой одной[/url] .
Diplomi_tsPa
27 Oct 25 at 7:21 am
купить диплом в георгиевске [url=https://rudik-diplom10.ru]купить диплом в георгиевске[/url] .
Diplomi_yuSa
27 Oct 25 at 7:22 am
купить диплом в бийске [url=https://www.rudik-diplom5.ru]купить диплом в бийске[/url] .
Diplomi_boma
27 Oct 25 at 7:22 am
Ich bin ein gro?er Fan von Cat Spins Casino, es ist ein Ort, der begeistert. Die Auswahl ist einfach unschlagbar, mit aufregenden Live-Casino-Erlebnissen. Mit blitzschnellen Einzahlungen. Der Support ist effizient und professionell. Transaktionen sind immer sicher, dennoch mehr Bonusvarianten waren ein Hit. In Summe, Cat Spins Casino bietet ein unvergleichliches Erlebnis. Au?erdem die Navigation ist klar und flussig, was jede Session spannender macht. Ein starkes Feature ist das VIP-Programm mit exklusiven Stufen, das die Motivation steigert.
Seite entdecken|
Legendbearik1zef
27 Oct 25 at 7:22 am
как купить диплом в колледже [url=http://www.frei-diplom7.ru]как купить диплом в колледже[/url] .
Diplomi_agei
27 Oct 25 at 7:23 am
наркологическая клиника [url=https://narkologicheskaya-klinika-24.ru/]наркологическая клиника[/url] .
narkologicheskaya klinika_pjSr
27 Oct 25 at 7:23 am
https://gap77slot.com/melbet-kazino-sloti-skachat/
Williamsoamn
27 Oct 25 at 7:24 am
диплом техникума казахстана купить [url=www.educ-ua7.ru/]www.educ-ua7.ru/[/url] .
Diplomi_bvea
27 Oct 25 at 7:24 am
Как купить Марки лсд в Калманке?Вот нашел https://gadjetforyou.ru
– цены привлекательные, есть оперативная доставка. Кто-то заказывал у них? Насколько хорошее качество товар?
Stevenref
27 Oct 25 at 7:24 am
купить диплом университета [url=www.rudik-diplom2.ru]купить диплом университета[/url] .
Diplomi_ldpi
27 Oct 25 at 7:25 am
https://betalingsmetodercasino.com/
1-gocasino
27 Oct 25 at 7:25 am
кракен vk6
kraken vpn
Henryamerb
27 Oct 25 at 7:26 am
платная наркологическая клиника в москве [url=www.narkologicheskaya-klinika-24.ru]www.narkologicheskaya-klinika-24.ru[/url] .
narkologicheskaya klinika_haSr
27 Oct 25 at 7:27 am
кракен клиент
кракен vk3
Henryamerb
27 Oct 25 at 7:27 am
аппараты медицинские [url=http://medicinskoe–oborudovanie.ru]http://medicinskoe–oborudovanie.ru[/url] .
medicinskoe oborydovanie_rxei
27 Oct 25 at 7:27 am
https://magme.madeinitalyslc.it/2025/10/11/melbet-sajt-2025-obzor-bk/
MichaelSeemn
27 Oct 25 at 7:28 am
https://smsuaemarketing.com/skachat-prilozhenie-melbet-2025
Michaelcop
27 Oct 25 at 7:29 am
купить диплом в мурманске с занесением в реестр [url=www.frei-diplom5.ru/]купить диплом в мурманске с занесением в реестр[/url] .
Diplomi_hfPa
27 Oct 25 at 7:30 am
купить диплом преподавателя [url=https://rudik-diplom5.ru]купить диплом преподавателя[/url] .
Diplomi_lwma
27 Oct 25 at 7:31 am
kraken vk3
kraken marketplace
Henryamerb
27 Oct 25 at 7:32 am
старые дипломы купить [url=https://www.rudik-diplom4.ru]старые дипломы купить[/url] .
Diplomi_usOr
27 Oct 25 at 7:32 am
купить диплом в нижнем новгороде [url=www.rudik-diplom15.ru/]купить диплом в нижнем новгороде[/url] .
Diplomi_iqPi
27 Oct 25 at 7:32 am
modernvaluecorner – Feels like they really care about quality, everything arrives perfectly packed.
Noah Bostock
27 Oct 25 at 7:32 am
Ich bin begeistert von der Welt bei Cat Spins Casino, es ladt zu unvergesslichen Momenten ein. Das Angebot ist ein Traum fur Spieler, inklusive dynamischer Sportwetten. 100 % bis zu 500 € plus Freispiele. Die Mitarbeiter sind schnell und kompetent. Der Prozess ist einfach und transparent, in manchen Fallen mehr Promo-Vielfalt ware toll. Abschlie?end, Cat Spins Casino sorgt fur kontinuierlichen Spa?. Zudem die Plattform ist optisch ein Highlight, eine vollstandige Immersion ermoglicht. Ein gro?es Plus sind die zuverlassigen Krypto-Zahlungen, schnelle Zahlungen garantieren.
Jetzt Г¶ffnen|
NightRiseos9zef
27 Oct 25 at 7:36 am
купить диплом мастера маникюра и педикюра [url=https://rudik-diplom12.ru/]купить диплом мастера маникюра и педикюра[/url] .
Diplomi_dnPi
27 Oct 25 at 7:36 am
екатеринбург купить диплом в реестр [url=http://www.frei-diplom4.ru]екатеринбург купить диплом в реестр[/url] .
Diplomi_mpOl
27 Oct 25 at 7:36 am
I have read so many content on the topic of the blogger lovers however
this article is truly a fastidious article, keep it up.
레비트라 구입
27 Oct 25 at 7:38 am
kraken ios
kraken vk2
Henryamerb
27 Oct 25 at 7:38 am
https://big789.net/kak-skachat-melbet-kazino-2025/
TerryDob
27 Oct 25 at 7:38 am
купить диплом в гуково [url=www.rudik-diplom11.ru/]купить диплом в гуково[/url] .
Diplomi_qzMi
27 Oct 25 at 7:39 am
бонги для курения
бонги для курения
27 Oct 25 at 7:39 am
диплом об окончании техникума купить [url=https://educ-ua7.ru]https://educ-ua7.ru[/url] .
Diplomi_rwea
27 Oct 25 at 7:39 am