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=https://frei-diplom9.ru]https://frei-diplom9.ru[/url] .
Diplomi_esea
20 Sep 25 at 12:21 am
купить диплом в кургане [url=http://rudik-diplom10.ru/]купить диплом в кургане[/url] .
Diplomi_hgSa
20 Sep 25 at 12:22 am
займы россии [url=http://zaimy-17.ru]займы россии[/url] .
zaimi_yzSa
20 Sep 25 at 12:22 am
I was very pleased to find this web site. I need to to thank
you for ones time for this wonderful read!!
I definitely appreciated every part of it
and I have you saved as a favorite to look at new information in your web site.
online order medicine
20 Sep 25 at 12:23 am
купить диплом химика [url=https://rudik-diplom8.ru/]купить диплом химика[/url] .
Diplomi_uuMt
20 Sep 25 at 12:23 am
Приобрести диплом о высшем образовании можем помочь. Купить диплом специалиста в Астрахани – [url=http://diplomybox.com/kupit-diplom-spetsialista-v-astrakhani/]diplomybox.com/kupit-diplom-spetsialista-v-astrakhani[/url]
Cazrcnq
20 Sep 25 at 12:24 am
If some one wishes to be updated with hottest technologies therefore he must be pay a visit this site and be up to date
all the time.
QuantumWealth Capital
20 Sep 25 at 12:25 am
cialis for sale [url=https://evertrustmeds.com/#]Ever Trust Meds[/url] Tadalafil Tablet
Michealstilm
20 Sep 25 at 12:25 am
Купить диплом техникума в Киев [url=www.educ-ua7.ru]www.educ-ua7.ru[/url] .
Diplomi_leea
20 Sep 25 at 12:25 am
купить диплом о высшем образовании с занесением в реестр в ижевске [url=http://frei-diplom1.ru/]купить диплом о высшем образовании с занесением в реестр в ижевске[/url] .
Diplomi_ryOi
20 Sep 25 at 12:26 am
купить диплом техникума в сочи [url=www.frei-diplom12.ru/]купить диплом техникума в сочи[/url] .
Diplomi_diPt
20 Sep 25 at 12:26 am
https://talk.plesk.com/members/1xbtpromocode.452759/#about
RobertHoN
20 Sep 25 at 12:27 am
Pretty! This has been an incredibly wonderful post.
Thanks for supplying this info.
rutinitas skincare pagi hari yang simpel
20 Sep 25 at 12:28 am
накрутка подписчиков в тг без регистрации
DavidNOb
20 Sep 25 at 12:29 am
где купить диплом техникума была [url=www.frei-diplom8.ru/]где купить диплом техникума была[/url] .
Diplomi_cdsr
20 Sep 25 at 12:29 am
https://xn--krken21-bn4c.com
Howardreomo
20 Sep 25 at 12:29 am
купить диплом с занесением реестра [url=frei-diplom4.ru]купить диплом с занесением реестра[/url] .
Diplomi_koOl
20 Sep 25 at 12:30 am
бот для накрутки подписчиков в телеграм бесплатно
MatthewRow
20 Sep 25 at 12:32 am
купить диплом с занесением в реестр в кемерово [url=www.frei-diplom2.ru/]www.frei-diplom2.ru/[/url] .
Diplomi_upEa
20 Sep 25 at 12:32 am
диплом о среднем образовании купить легально [url=https://frei-diplom3.ru/]https://frei-diplom3.ru/[/url] .
Diplomi_tzKt
20 Sep 25 at 12:32 am
Discover the rise of Farmasi International, a global leader in cosmetics and wellness.
Explore its strong European roots, signature vegan-friendly
products, and worldwide success. Learn why Farmasi is a trusted
brand in beauty, skincare, and health across continents.
Farmasi Europe
20 Sep 25 at 12:33 am
займ всем [url=https://zaimy-25.ru]https://zaimy-25.ru[/url] .
zaimi_xcoa
20 Sep 25 at 12:34 am
купить диплом техникума с занесением пять плюс [url=https://frei-diplom8.ru/]купить диплом техникума с занесением пять плюс[/url] .
Diplomi_kxsr
20 Sep 25 at 12:34 am
диплом высшего образования проведенный купить [url=http://www.frei-diplom5.ru]диплом высшего образования проведенный купить[/url] .
Diplomi_yzPa
20 Sep 25 at 12:34 am
Wonderful post however I was wanting to know if you could
write a litte more on this subject? I’d be very thankful if you could elaborate a little bit more.
Thanks!
Wexley Vaultix Scam
20 Sep 25 at 12:35 am
Футболки с авторскими принтами и футболки печать на заказ в Белгороде. Толстовку белую с капюшоном без рисунка и шапка Puma в Кирове. Надпись для мужа на футболке и футболки джоджо в Петрозаводске. Таблица по уходу за одеждой и хип хоп стиль одежды в Новосибирске. Футболка оптом месси аргентина и футболки турецкие мужские https://futbolki-s-printom.ru/
Gregorysnisp
20 Sep 25 at 12:35 am
Christmas Catch играть в Покердом
BrandonLum
20 Sep 25 at 12:36 am
купить диплом проведенный москва [url=http://frei-diplom6.ru]http://frei-diplom6.ru[/url] .
Diplomi_beOl
20 Sep 25 at 12:36 am
купить диплом о средне специальном образовании с занесением в реестр [url=http://frei-diplom4.ru]купить диплом о средне специальном образовании с занесением в реестр[/url] .
Diplomi_ixOl
20 Sep 25 at 12:38 am
диплом техникума старого образца до 1996 г купить [url=http://www.educ-ua7.ru]http://www.educ-ua7.ru[/url] .
Diplomi_hkea
20 Sep 25 at 12:38 am
займы онлайн все [url=https://zaimy-22.ru]https://zaimy-22.ru[/url] .
zaimi_bsKi
20 Sep 25 at 12:39 am
купить диплом в выборге [url=https://rudik-diplom1.ru/]https://rudik-diplom1.ru/[/url] .
Diplomi_puer
20 Sep 25 at 12:39 am
купить диплом в воронеже [url=rudik-diplom10.ru]купить диплом в воронеже[/url] .
Diplomi_wpSa
20 Sep 25 at 12:40 am
купить диплом с проводкой моих [url=https://frei-diplom2.ru/]купить диплом с проводкой моих[/url] .
Diplomi_msEa
20 Sep 25 at 12:40 am
купить диплом пту с занесением в реестр [url=www.frei-diplom3.ru]купить диплом пту с занесением в реестр[/url] .
Diplomi_qjKt
20 Sep 25 at 12:40 am
все микрозаймы на карту [url=https://www.zaimy-18.ru]все микрозаймы на карту[/url] .
zaimi_bkMl
20 Sep 25 at 12:41 am
купить диплом в рязани [url=http://www.rudik-diplom8.ru]купить диплом в рязани[/url] .
Diplomi_uxMt
20 Sep 25 at 12:41 am
Получить диплом ВУЗа можем помочь. Купить диплом Барнаул – [url=http://diplomybox.com/kupit-diplom-barnaul/]diplomybox.com/kupit-diplom-barnaul[/url]
Cazrpbb
20 Sep 25 at 12:43 am
купить диплом в уфе с реестром [url=www.frei-diplom4.ru/]www.frei-diplom4.ru/[/url] .
Diplomi_yfOl
20 Sep 25 at 12:43 am
купить диплом с занесением в реестр в иркутске [url=http://www.frei-diplom5.ru]купить диплом с занесением в реестр в иркутске[/url] .
Diplomi_gtPa
20 Sep 25 at 12:43 am
диплом колледжа купить с занесением в реестр [url=https://frei-diplom1.ru]диплом колледжа купить с занесением в реестр[/url] .
Diplomi_xrOi
20 Sep 25 at 12:44 am
Thanks for the auspicious writeup. It actually used
to be a amusement account it. Look advanced to far introduced agreeable from you!
By the way, how can we keep in touch?
Trygg Bitrow
20 Sep 25 at 12:44 am
займ всем [url=http://zaimy-19.ru/]займ всем[/url] .
zaimi_ggKl
20 Sep 25 at 12:44 am
купить диплом о среднем специальном образовании с занесением в реестр [url=http://frei-diplom6.ru]купить диплом о среднем специальном образовании с занесением в реестр[/url] .
Diplomi_etOl
20 Sep 25 at 12:45 am
Преимущества вывода из запоя от опытных специалистов в условиях Донецка ДНР многочисленны. Такой формат лечения позволяет:
Подробнее можно узнать тут – http://vyvod-iz-zapoya-donetsk-dnr00.ru/
Michaelduxuh
20 Sep 25 at 12:45 am
взо [url=www.zaimy-22.ru/]www.zaimy-22.ru/[/url] .
zaimi_qsKi
20 Sep 25 at 12:46 am
купить диплом в мурманске с занесением в реестр [url=http://www.frei-diplom2.ru]купить диплом в мурманске с занесением в реестр[/url] .
Diplomi_pzEa
20 Sep 25 at 12:46 am
купить диплом о высшем образовании с занесением в реестр в москве [url=http://frei-diplom3.ru]купить диплом о высшем образовании с занесением в реестр в москве[/url] .
Diplomi_dkKt
20 Sep 25 at 12:47 am
+905516067299 fetoden dolayi ulkeyi terk etti
AHMET ENGİN
20 Sep 25 at 12:47 am
купить диплом машиниста [url=http://rudik-diplom1.ru/]купить диплом машиниста[/url] .
Diplomi_aoer
20 Sep 25 at 12:48 am