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!
Excited for $MTAUR’s exchange debut at double presale price. Token utility in outfits personalizes fun. Community engagement strong.
minotaurus coin
WilliamPargy
21 Oct 25 at 11:24 am
топ seo компаний [url=http://www.reiting-seo-agentstv.ru]http://www.reiting-seo-agentstv.ru[/url] .
reiting seo agentstv_vpsa
21 Oct 25 at 11:27 am
купить диплом с проводкой моего [url=http://frei-diplom5.ru/]купить диплом с проводкой моего[/url] .
Diplomi_nzPa
21 Oct 25 at 11:30 am
купить новый диплом [url=www.rudik-diplom11.ru]купить новый диплом[/url] .
Diplomi_kdMi
21 Oct 25 at 11:30 am
When I originally commented I appear to have clicked on the -Notify me when new comments are added- checkbox and now every
time a comment is added I get 4 emails with the exact
same comment. Is there a means you are able to remove me from that service?
Thanks a lot!
check my blog
21 Oct 25 at 11:31 am
пин ап вывод через узкард [url=https://pinup5008.ru]https://pinup5008.ru[/url]
pin_up_uz_iuSt
21 Oct 25 at 11:35 am
Статья знакомит с важнейшими моментами, которые сформировали наше общество. От великих изобретений до культурных переворотов — вы узнаете, как прошлое влияет на наше мышление, технологии и образ жизни.
Прочитать подробнее – https://kartikaroulylawfirm.com/peran-ahli-dalam-proposal-perdamaian-pkpu
DavidVus
21 Oct 25 at 11:35 am
сео продвижение компания [url=http://www.reiting-kompanii-po-prodvizheniyu-sajtov.ru]сео продвижение компания[/url] .
agentstvo poiskovogo prodvijeniya_rsKt
21 Oct 25 at 11:36 am
This is a topic that’s near to my heart… Best wishes!
Exactly where are your contact details though?
Fuentoro Ai Scam
21 Oct 25 at 11:37 am
рейтинг seo [url=https://www.seo-prodvizhenie-reiting.ru]https://www.seo-prodvizhenie-reiting.ru[/url] .
seo prodvijenie reiting_lxEa
21 Oct 25 at 11:40 am
купить диплом электромонтажника [url=https://rudik-diplom11.ru]купить диплом электромонтажника[/url] .
Diplomi_hhMi
21 Oct 25 at 11:42 am
купить диплом в реестр [url=https://frei-diplom5.ru]купить диплом в реестр[/url] .
Diplomi_fqPa
21 Oct 25 at 11:42 am
продвижение сайта в топ с гарантией [url=https://www.reiting-runeta-seo.ru]продвижение сайта в топ с гарантией[/url] .
reiting ryneta seo_akma
21 Oct 25 at 11:43 am
сколько стоит купить диплом медсестры [url=http://www.frei-diplom13.ru]сколько стоит купить диплом медсестры[/url] .
Diplomi_aokt
21 Oct 25 at 11:44 am
купить диплом московского торгово экономического техникума [url=https://www.frei-diplom12.ru]купить диплом московского торгово экономического техникума[/url] .
Diplomi_yhPt
21 Oct 25 at 11:44 am
пин ап демо режим [url=www.pinup5007.ru]www.pinup5007.ru[/url]
pin_up_uz_kusr
21 Oct 25 at 11:46 am
seo продвижение рейтинг [url=seo-prodvizhenie-reiting.ru]seo продвижение рейтинг[/url] .
seo prodvijenie reiting_doEa
21 Oct 25 at 11:47 am
Девушка не просто мастер, а настоящая художница тела. Каждое движение было продумано и наполнено теплом. Эмоции невероятные. Очень советую, эротический массаж заказать Новосиб – https://sibirka.com/. Чисто и уютно, атмосфера расслабляющая.
Bobbyham
21 Oct 25 at 11:48 am
сео продвижение москва [url=https://reiting-seo-agentstv-moskvy.ru/]сео продвижение москва[/url] .
reiting seo agentstv moskvi_dnMl
21 Oct 25 at 11:50 am
That is a very good tip especially to those fresh to the blogosphere.
Short but very precise info… Appreciate your sharing this
one. A must read post!
Finance Legend App
21 Oct 25 at 11:52 am
Эта информационная статья содержит полезные факты, советы и рекомендации, которые помогут вам быть в курсе последних тенденций и изменений в выбранной области. Материал составлен так, чтобы быть полезным и понятным каждому.
Это ещё не всё… – https://germanautohaus.co.uk/slide/homepage-slide-4/german-autohaus-1
EugeneVem
21 Oct 25 at 11:54 am
где купить диплом техникума в томске [url=frei-diplom10.ru]где купить диплом техникума в томске[/url] .
Diplomi_rtEa
21 Oct 25 at 11:59 am
пин ап регистрация по коду [url=http://pinup5008.ru]пин ап регистрация по коду[/url]
pin_up_uz_xzSt
21 Oct 25 at 12:01 pm
пин ап авиатор взлом [url=https://pinup5008.ru]https://pinup5008.ru[/url]
pin_up_uz_yfSt
21 Oct 25 at 12:04 pm
top rated seo [url=www.reiting-seo-agentstv.ru/]www.reiting-seo-agentstv.ru/[/url] .
reiting seo agentstv_zdsa
21 Oct 25 at 12:05 pm
пин ап активация бонуса [url=https://pinup5008.ru]пин ап активация бонуса[/url]
pin_up_uz_tmSt
21 Oct 25 at 12:05 pm
I read this article completely about the difference of latest and preceding technologies, it’s remarkable article.
Bl555
21 Oct 25 at 12:09 pm
seo agentura [url=https://reiting-kompanii-po-prodvizheniyu-sajtov.ru]https://reiting-kompanii-po-prodvizheniyu-sajtov.ru[/url] .
agentstvo poiskovogo prodvijeniya_hzKt
21 Oct 25 at 12:10 pm
где купить дипломы медсестры [url=https://frei-diplom13.ru/]где купить дипломы медсестры[/url] .
Diplomi_yakt
21 Oct 25 at 12:11 pm
Inspiring quest there. What happened after?
Good luck!
Seattle Chimney cap installation
21 Oct 25 at 12:11 pm
купить диплом техникума в смоленске [url=https://frei-diplom12.ru]купить диплом техникума в смоленске[/url] .
Diplomi_nePt
21 Oct 25 at 12:12 pm
I’m not that much of a online reader to be honest but your blogs really nice,
keep it up! I’ll go ahead and bookmark your website to come back later on. Many thanks
비아그라 구매
21 Oct 25 at 12:13 pm
pin up shikoyatlar uz [url=www.pinup5008.ru]pin up shikoyatlar uz[/url]
pin_up_uz_tpSt
21 Oct 25 at 12:13 pm
топ seo продвижение заказать [url=seo-prodvizhenie-reiting.ru]топ seo продвижение заказать[/url] .
seo prodvijenie reiting_emEa
21 Oct 25 at 12:13 pm
Девушки невероятно красивые и внимательные, создают атмосферу доверия и покоя. Массаж плавный, чувственный и расслабляющий. После сеанса ощущение лёгкости и гармонии. Очень советую, эротический массаж вызвать Новосиб: https://sibirka.com/. Девушка превзошла все мои ожидания.
Bobbyham
21 Oct 25 at 12:13 pm
Hello my friend! I want to say that this article is awesome, nice written and
include approximately all significant infos. I’d like
to peer more posts like this .
slot gacor
21 Oct 25 at 12:14 pm
Здесь все ровно братья как и всегда
https://khartsyzkue.ru
Про себя могу сказать что толлерантность ОЧЕНЬ выская
MichaelVag
21 Oct 25 at 12:14 pm
медсестра которая купила диплом врача [url=http://frei-diplom13.ru/]медсестра которая купила диплом врача[/url] .
Diplomi_gokt
21 Oct 25 at 12:18 pm
купить диплом вуза диплом техникума пять плюс [url=https://www.frei-diplom10.ru]купить диплом вуза диплом техникума пять плюс[/url] .
Diplomi_bpEa
21 Oct 25 at 12:18 pm
pin up uz [url=https://www.pinup5007.ru]https://www.pinup5007.ru[/url]
pin_up_uz_jesr
21 Oct 25 at 12:19 pm
купить диплом техникума в нальчике [url=www.frei-diplom12.ru/]купить диплом техникума в нальчике[/url] .
Diplomi_pgPt
21 Oct 25 at 12:19 pm
раскрутка сайта топ 10 [url=reiting-runeta-seo.ru]reiting-runeta-seo.ru[/url] .
reiting ryneta seo_wcma
21 Oct 25 at 12:22 pm
Публикация предлагает читателю не просто информацию, а инструменты для анализа и саморазвития. Мы стимулируем критическое мышление, предлагая различные точки зрения и призывая к самостоятельному поиску решений.
Всё, что нужно знать – https://jamalweb.com/?p=139
RichardHar
21 Oct 25 at 12:22 pm
сео компания [url=https://reiting-seo-agentstv.ru/]сео компания[/url] .
reiting seo agentstv_musa
21 Oct 25 at 12:22 pm
топ сео компаний [url=http://seo-prodvizhenie-reiting.ru]http://seo-prodvizhenie-reiting.ru[/url] .
seo prodvijenie reiting_pjEa
21 Oct 25 at 12:23 pm
купить медицинский диплом медсестры [url=http://frei-diplom13.ru/]купить медицинский диплом медсестры[/url] .
Diplomi_gfkt
21 Oct 25 at 12:27 pm
Very good information. Lucky me I recently found your blog by chance (stumbleupon).
I’ve saved as a favorite for later!
buy ativan online without prescription
21 Oct 25 at 12:27 pm
горный техникум диплом купить [url=http://frei-diplom12.ru/]горный техникум диплом купить[/url] .
Diplomi_dgPt
21 Oct 25 at 12:28 pm
http://inquisnower.phorum.pl/viewtopic.php?p=667173#667173
http://inquisnower.phorum.pl/viewtopic.php?p=667173#667173
21 Oct 25 at 12:28 pm
A person essentially help to make severely posts I would state.
This is the first time I frequented your web page and up to
now? I amazed with the research you made to create this actual put
up incredible. Wonderful job!
تعمیرات لباسشویی بوش
21 Oct 25 at 12:28 pm