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!
Spedra prezzo basso Italia: Spedra prezzo basso Italia – farmacia viva
ClydeExamp
29 Oct 25 at 4:56 pm
Je suis bluffe par Ruby Slots Casino, il procure une sensation de frisson. Le catalogue est un tresor de divertissements, offrant des tables live interactives. 100% jusqu’a 500 € plus des tours gratuits. Le support est pro et accueillant. Les paiements sont surs et efficaces, en revanche quelques tours gratuits supplementaires seraient cool. Au final, Ruby Slots Casino est un endroit qui electrise. Par ailleurs le site est rapide et engageant, ce qui rend chaque session plus palpitante. Un plus les competitions regulieres pour plus de fun, assure des transactions fiables.
Entrer maintenant|
urbanforceix3zef
29 Oct 25 at 4:56 pm
купить диплом в назрани [url=https://www.rudik-diplom14.ru]купить диплом в назрани[/url] .
Diplomi_psea
29 Oct 25 at 4:58 pm
В этом что-то есть. Раньше я думал иначе, большое спасибо за помощь в этом вопросе.
Depozyty: wiekszosc metod pozwala natychmiast doladuj konto, [url=https://www.demasyemek.com/jogue-agora-no-twin-casino]https://www.demasyemek.com/jogue-agora-no-twin-casino[/url] co pozwala tak szybko, jak to mozliwe grac.
Devonswaps
29 Oct 25 at 4:59 pm
FarmaciaViva: differenza tra Spedra e Viagra – comprare medicinali online legali
RichardImmon
29 Oct 25 at 5:00 pm
online casino with live dealer
top online casinos real money
high limit online slots
online casino real money
29 Oct 25 at 5:05 pm
I have read so many content regarding the blogger lovers except this piece of writing is in fact a
good post, keep it up.
Visit Website
29 Oct 25 at 5:05 pm
kamagra: Kamagra oral jelly France – Kamagra oral jelly France
RobertJuike
29 Oct 25 at 5:06 pm
billig Viagra Norge: viagra reseptfri – Sildenafil tabletter pris
RichardImmon
29 Oct 25 at 5:07 pm
купить диплом маляра [url=https://www.rudik-diplom14.ru]купить диплом маляра[/url] .
Diplomi_hxea
29 Oct 25 at 5:10 pm
купить диплом в грозном [url=http://rudik-diplom13.ru/]купить диплом в грозном[/url] .
Diplomi_naon
29 Oct 25 at 5:10 pm
купить диплом о образовании недорого [url=https://www.educ-ua7.ru]https://www.educ-ua7.ru[/url] .
Diplomi_miea
29 Oct 25 at 5:10 pm
Thankfulness to my father who informed me regarding this
blog, this website is really awesome.
استخدام شرکت نفت تهران
29 Oct 25 at 5:12 pm
купить диплом в волжском [url=rudik-diplom6.ru]купить диплом в волжском[/url] .
Diplomi_kfKr
29 Oct 25 at 5:13 pm
купить диплом штукатура [url=www.rudik-diplom5.ru]www.rudik-diplom5.ru[/url] .
Diplomi_rlma
29 Oct 25 at 5:14 pm
купить диплом в феодосии [url=http://rudik-diplom11.ru/]http://rudik-diplom11.ru/[/url] .
Diplomi_gdMi
29 Oct 25 at 5:15 pm
1000 рублей за регистрацию вывод сразу без вложений на киви
Mosesfrord
29 Oct 25 at 5:16 pm
кто нибудь работает медсестрой по купленному диплому [url=www.frei-diplom13.ru/]www.frei-diplom13.ru/[/url] .
Diplomi_xykt
29 Oct 25 at 5:16 pm
Wow, superb blog layout! How long have you been running a blog for?
you made running a blog look easy. The total look of your web site is fantastic, let alone the content material!
snapinsta
29 Oct 25 at 5:18 pm
вирт номер купить навсегда
вирт номер купить навсегда
29 Oct 25 at 5:19 pm
купить диплом о высшем образовании с занесением в реестр в москве [url=http://frei-diplom1.ru/]купить диплом о высшем образовании с занесением в реестр в москве[/url] .
Diplomi_mgOi
29 Oct 25 at 5:19 pm
Nikmati pengalaman permainan terbaik di RelixPlay, dengan game togel, slot,
kasino live, sportbook untuk mendapatkan jackpot!!!
RELIXPLAY
29 Oct 25 at 5:20 pm
купить диплом в саратове [url=www.rudik-diplom4.ru/]купить диплом в саратове[/url] .
Diplomi_xmOr
29 Oct 25 at 5:20 pm
купить диплом в краснодаре [url=rudik-diplom5.ru]купить диплом в краснодаре[/url] .
Diplomi_pqma
29 Oct 25 at 5:20 pm
куплю диплом высшего образования [url=www.rudik-diplom7.ru]куплю диплом высшего образования[/url] .
Diplomi_voPl
29 Oct 25 at 5:22 pm
testflight linebet
partner linebet
29 Oct 25 at 5:22 pm
купить диплом в петропавловске-камчатском [url=http://rudik-diplom11.ru/]купить диплом в петропавловске-камчатском[/url] .
Diplomi_tsMi
29 Oct 25 at 5:22 pm
BL555 app
nhà cái BL555
29 Oct 25 at 5:24 pm
куплю диплом младшей медсестры [url=https://frei-diplom13.ru/]https://frei-diplom13.ru/[/url] .
Diplomi_ywkt
29 Oct 25 at 5:26 pm
купить диплом в назрани [url=www.rudik-diplom1.ru/]купить диплом в назрани[/url] .
Diplomi_qzer
29 Oct 25 at 5:26 pm
Мы обеспечиваем полную поддержку на всех этапах лечения в Ростове-на-Дону, включая реабилитацию и профилактику рецидивов.
Получить дополнительную информацию – [url=https://vyvod-iz-zapoya-rostov112.ru/]вывод из запоя недорого[/url]
Jamiered
29 Oct 25 at 5:26 pm
купить диплом техникума украина [url=www.frei-diplom9.ru]купить диплом техникума украина[/url] .
Diplomi_peea
29 Oct 25 at 5:27 pm
купить старый диплом техникума [url=www.educ-ua7.ru]www.educ-ua7.ru[/url] .
Diplomi_vvea
29 Oct 25 at 5:27 pm
todesk
PHP hook, building hooks in your application – Sjoerd Maessen blog at Sjoerd Maessen blog
todesk
29 Oct 25 at 5:28 pm
seo бесплатно [url=https://kursy-seo-11.ru/]seo бесплатно[/url] .
kyrsi seo_iqEl
29 Oct 25 at 5:28 pm
купить диплом с проводкой моего [url=http://frei-diplom5.ru/]купить диплом с проводкой моего[/url] .
Diplomi_tvPa
29 Oct 25 at 5:29 pm
купить аттестат школы [url=http://www.rudik-diplom14.ru]купить аттестат школы[/url] .
Diplomi_juea
29 Oct 25 at 5:29 pm
Мы понимаем, как важно быстро и эффективно выйти из запоя, поэтому в Ростове-на-Дону предлагаем оперативную помощь в любое время.
Углубиться в тему – https://vyvod-iz-zapoya-rostov234.ru/
DerrickAroub
29 Oct 25 at 5:30 pm
Hello, i read your blog occasionally and i own a similar one and i was just curious if
you get a lot of spam remarks? If so how do you stop it, any plugin or anything
you can recommend? I get so much lately it’s driving me crazy so any assistance
is very much appreciated.
real money online slots
29 Oct 25 at 5:31 pm
It’s truly a nice and useful piece of information. I am happy that
you simply shared this helpful information with us.
Please keep us informed like this. Thank you for sharing.
website
29 Oct 25 at 5:32 pm
купить диплом в калининграде [url=http://www.rudik-diplom4.ru]купить диплом в калининграде[/url] .
Diplomi_ffOr
29 Oct 25 at 5:32 pm
диплом проведенный купить [url=https://frei-diplom2.ru/]диплом проведенный купить[/url] .
Diplomi_zqEa
29 Oct 25 at 5:32 pm
seo онлайн [url=http://www.kursy-seo-11.ru]seo онлайн[/url] .
kyrsi seo_upEl
29 Oct 25 at 5:32 pm
купить диплом инженера [url=https://www.rudik-diplom13.ru]купить диплом инженера[/url] .
Diplomi_nuon
29 Oct 25 at 5:33 pm
Мы обеспечиваем быстрое и безопасное восстановление после длительного употребления алкоголя.
Узнать больше – [url=https://vyvod-iz-zapoya-nizhnij-novgorod12.ru/]нарколог на дом вывод из запоя в нижний новгороде[/url]
JuniorLom
29 Oct 25 at 5:33 pm
купить диплом геодезиста [url=https://www.rudik-diplom8.ru]купить диплом геодезиста[/url] .
Diplomi_jfMt
29 Oct 25 at 5:33 pm
Мы понимаем, как важно быстро и эффективно выйти из запоя, поэтому в Ростове-на-Дону предлагаем оперативную помощь в любое время.
Изучить вопрос глубже – [url=https://vyvod-iz-zapoya-rostov236.ru/]вывод из запоя[/url]
Danielabema
29 Oct 25 at 5:34 pm
купить диплом с занесением в реестр челябинск [url=https://frei-diplom4.ru/]купить диплом с занесением в реестр челябинск[/url] .
Diplomi_lpOl
29 Oct 25 at 5:34 pm
купить диплом юриста [url=rudik-diplom5.ru]купить диплом юриста[/url] .
Diplomi_ypma
29 Oct 25 at 5:35 pm
купить диплом техникума с занесением в реестр [url=https://frei-diplom1.ru]купить диплом техникума с занесением в реестр[/url] .
Diplomi_cwOi
29 Oct 25 at 5:35 pm