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=www.rudik-diplom11.ru/]купить диплом в нижнем новгороде[/url] .
Diplomi_xcMi
27 Oct 25 at 9:44 am
Публикация предлагает уникальную подборку информации, которая будет интересна как специалистам, так и широкому кругу читателей. Здесь вы найдете ответы на часто задаваемые вопросы и полезные инсайты для дальнейшего применения.
Информация доступна здесь – https://www.joyetjoie.com/screen-shot-2017-07-24-at-10-40-29-am
StevenNok
27 Oct 25 at 9:45 am
What’s up to every body, it’s my first visit of this weblog; this weblog consists of awesome and in fact excellent data in favor of visitors.
https://rtproma777.com/melbet-skachat-prilozhenie-dlya-android-2025/
EarnestAbent
27 Oct 25 at 9:45 am
Hi, I do think this is an excellent site. I stumbledupon it 😉 I’m
going to return once again since i have book-marked
it. Money and freedom is the greatest way to change, may you be rich and continue to guide other people.
Trang chủ au88
27 Oct 25 at 9:45 am
кракен маркетплейс
кракен android
Henryamerb
27 Oct 25 at 9:46 am
kraken vk6
kraken tor
Henryamerb
27 Oct 25 at 9:47 am
купить диплом во владивостоке [url=rudik-diplom2.ru]купить диплом во владивостоке[/url] .
Diplomi_tzpi
27 Oct 25 at 9:48 am
адреса наркологических клиник [url=www.narkologicheskaya-klinika-24.ru]www.narkologicheskaya-klinika-24.ru[/url] .
narkologicheskaya klinika_qrSr
27 Oct 25 at 9:48 am
купить диплом в шадринске [url=https://rudik-diplom15.ru/]купить диплом в шадринске[/url] .
Diplomi_ghPi
27 Oct 25 at 9:48 am
медицинское оборудование [url=www.medicinskoe–oborudovanie.ru]медицинское оборудование[/url] .
medicinskoe oborydovanie_uhei
27 Oct 25 at 9:48 am
медицинская аппаратура [url=https://medicinskoe–oborudovanie.ru/]медицинская аппаратура[/url] .
medicinskoe oborydovanie_ryei
27 Oct 25 at 9:51 am
The gist is here: https://chhapai.com
Charlesdum
27 Oct 25 at 9:51 am
kraken сайт
кракен онлайн
Henryamerb
27 Oct 25 at 9:52 am
Don’t miss the essentials: https://trilhaseaventuras.com.br
Stevenhew
27 Oct 25 at 9:52 am
купить диплом в кургане [url=http://www.rudik-diplom12.ru]купить диплом в кургане[/url] .
Diplomi_qcPi
27 Oct 25 at 9:53 am
4M Dental Implant Center
3918 ᒪong Beach Blvd #200, Long Beach,
СA 90807, United States
15622422075
Bookmarks (https://www.protopage.com)
https://www.protopage.com
27 Oct 25 at 9:53 am
купить диплом нового образца [url=rudik-diplom11.ru]купить диплом нового образца[/url] .
Diplomi_qeMi
27 Oct 25 at 9:54 am
диплом техникума союзных республик купить [url=https://frei-diplom10.ru]диплом техникума союзных республик купить[/url] .
Diplomi_gnEa
27 Oct 25 at 9:55 am
Список бесплатных предложений и типов бонусов: отдельный обзор промо-акций для новичков и постоянных игроков; в середине абзаца приводим ссылку на промокод при регистрации 1xbet как источник подробной информации о том, куда вводить данные и какие условия ожидать. К тому же рассказываем про порядок валидации аккаунта.
FrankWeism
27 Oct 25 at 9:56 am
кракен vk3
кракен тор
Henryamerb
27 Oct 25 at 9:58 am
как купить диплом с занесением в реестр [url=http://frei-diplom1.ru/]как купить диплом с занесением в реестр[/url] .
Diplomi_xyOi
27 Oct 25 at 9:58 am
В данной обзорной статье представлены интригующие факты, которые не оставят вас равнодушными. Мы критикуем и анализируем события, которые изменили наше восприятие мира. Узнайте, что стоит за новыми открытиями и как они могут изменить ваше восприятие реальности.
Обратиться к источнику – https://akademik.upr.ac.id/pengumuman/pengumuman-pendaftaran-wisuda-universitas-palangka-raya-periode-bulan-maret-tahun-2024
Charlieagric
27 Oct 25 at 9:59 am
halbzeit oder endstand wette
Stop by my homepage – Amerikanische Buchmacher
Amerikanische Buchmacher
27 Oct 25 at 9:59 am
купить диплом в геленджике [url=rudik-diplom15.ru]rudik-diplom15.ru[/url] .
Diplomi_cgPi
27 Oct 25 at 10:02 am
медицинские приборы [url=http://medicinskoe–oborudovanie.ru/]http://medicinskoe–oborudovanie.ru/[/url] .
medicinskoe oborydovanie_kxei
27 Oct 25 at 10:04 am
диплом государственного образца купить реестр [url=http://www.frei-diplom1.ru]диплом государственного образца купить реестр[/url] .
Diplomi_dtOi
27 Oct 25 at 10:04 am
kraken vk5
кракен android
Henryamerb
27 Oct 25 at 10:06 am
купить диплом в ленинск-кузнецком [url=www.rudik-diplom2.ru]www.rudik-diplom2.ru[/url] .
Diplomi_gspi
27 Oct 25 at 10:06 am
kraken vk4
kraken market
Henryamerb
27 Oct 25 at 10:07 am
What’s up, I read your blog regularly. Your writing style is awesome, keep up the good work!
more
27 Oct 25 at 10:08 am
клиники наркологические [url=narkologicheskaya-klinika-24.ru]narkologicheskaya-klinika-24.ru[/url] .
narkologicheskaya klinika_jwSr
27 Oct 25 at 10:08 am
купить вкладыш к диплому техникума пять плюс [url=https://www.frei-diplom7.ru]купить вкладыш к диплому техникума пять плюс[/url] .
Diplomi_jmei
27 Oct 25 at 10:11 am
купить диплом о высшем образовании с занесением в реестр в калуге [url=https://frei-diplom1.ru/]купить диплом о высшем образовании с занесением в реестр в калуге[/url] .
Diplomi_lkOi
27 Oct 25 at 10:12 am
kraken vk3
kraken онлайн
Henryamerb
27 Oct 25 at 10:12 am
Note: The best: https://www.cbtrends.com
Josephacubs
27 Oct 25 at 10:14 am
мед оборудование [url=https://medicinskoe–oborudovanie.ru]мед оборудование[/url] .
medicinskoe oborydovanie_kwei
27 Oct 25 at 10:15 am
заказать гриндер для травы
заказать гриндер для травы
27 Oct 25 at 10:16 am
кракен qr код
кракен маркет
Henryamerb
27 Oct 25 at 10:18 am
купить диплом медсестры [url=frei-diplom15.ru]купить диплом медсестры[/url] .
Diplomi_luoi
27 Oct 25 at 10:19 am
купить диплом в железногорске [url=https://rudik-diplom2.ru/]https://rudik-diplom2.ru/[/url] .
Diplomi_orpi
27 Oct 25 at 10:19 am
Fırat Engin bahis siteleri, Casino Siteleri Nargül Engin, Slot Siteleri Hüseyin Engin, Deneme Bonusu Veren Siteler
Ahmet Engin, Mehdi Deneme Bonusu, Mehdi Deneme bonusu veren siteler
Slot Siteleri Hüseyin Engin
27 Oct 25 at 10:20 am
Yes! Finally something about Bitzonex Hub.
Bitzonex Hub Scam
27 Oct 25 at 10:21 am
Где купить Ксанакс в Шимске?Что думаете про https://lemoon89.ru
? Адекватные цены, доставка оперативная. Кто-то покупал? Хочу знать о качестве?
Stevenref
27 Oct 25 at 10:21 am
мед оборудование [url=https://medicinskoe–oborudovanie.ru/]мед оборудование[/url] .
medicinskoe oborydovanie_xhei
27 Oct 25 at 10:21 am
kraken market Kraken сайт – это таинственный портал в мир эксклюзивных возможностей, где анонимность и безопасность возведены в абсолют. Здесь границы стираются, а мечты обретают реальность, но помните: с большой свободой приходит и большая ответственность. Каждый шаг требует осознанности и понимания, ведь за порогом скрываются не только сокровища, но и подводные камни. Подходите к изучению платформы с мудростью и осторожностью, взвешивая каждый свой выбор.
Kevinmoots
27 Oct 25 at 10:24 am
different methods for Games and Games: The 1xBet app supports a necessary range of betting, including Sports match betting, games gambling table, live gambling houses and innovative games with instant bets, listed in aviator, jetx, [url=https://wp.supover.com/discover-1xbet-thailand-your-ultimate-betting/]https://wp.supover.com/discover-1xbet-thailand-your-ultimate-betting/[/url] and aviatrix.
Monicatax
27 Oct 25 at 10:25 am
This info is priceless. When can I find out more?
balondor88
27 Oct 25 at 10:25 am
купить диплом техникума спб в южно сахалинске [url=https://frei-diplom7.ru]купить диплом техникума спб в южно сахалинске[/url] .
Diplomi_zhei
27 Oct 25 at 10:26 am
купить диплом в каменске-шахтинском [url=www.rudik-diplom2.ru/]www.rudik-diplom2.ru/[/url] .
Diplomi_hkpi
27 Oct 25 at 10:27 am
kraken vk6
kraken client
Henryamerb
27 Oct 25 at 10:27 am