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!
Online oyun dünyasında, oyuncuların öncelikli olarak istediği detaylardan biri sorunsuz
bağlantı imkanıdır. Casino levant giriş, kolay bağlantı yapısı ile üyelerin istediği
zaman sisteme ulaşmasına olanak tanıyor.
Casino levant kullanıcı dostu yapısıyla fark yaratıyor
Kolay arayüz, profesyonel sistem ile birleşince casinolevant,
her seviyeden üyelerin sorunsuz bir oyun oynamasına garanti ediyor.
Siteye bilgisayardan bağlanmak oldukça pratik.
casinolevant güncel giriş
20 Aug 25 at 7:52 am
**Info Menarik Kompetisi Spin Toto Slot 88 & Pasang Angka Togel 4D Terbaik – TOGELONLINE88**
togel
20 Aug 25 at 7:52 am
Некоторые состояния требуют немедленного вмешательства нарколога, так как отказ от лечения может привести к тяжелым осложнениям и риску для жизни.
Получить дополнительную информацию – [url=https://narcolog-na-dom-novokuznetsk00.ru/]vyzov-narkologa-na-dom novokuznetsk[/url]
AlfonsoBup
20 Aug 25 at 7:54 am
Ukrainian President Volodymyr Zelensky condemned Russian attacks on the Ukrainian regions of Kharkiv, Zaporizhzhia and Sumy on Monday, saying that the Kremlin intends to “humiliate diplomatic efforts” just hours before European leaders visit the White House.
[url=https://kra30-cc.com]kra24 cc[/url]
“The Russian war machine continues to destroy lives despite everything,” Zelensky said in a statement, hours before he’s due to meet US President Donald Trump in the Oval Office. “That is precisely why we are seeking assistance to put an end to the killings. That is why reliable security guarantees are required. That is why Russia should not be rewarded for its participation in this war.”
[url=https://kra–25.cc]kra23 at[/url]
“Everyone seeks dignified peace and true security,” the Ukrainian president said. “And at this very moment, the Russians are attacking Kharkiv, Zaporizhzhia, the Sumy region, and Odesa, destroying residential buildings and our civilian infrastructure.”
At least seven people were killed in Russia’s attack? on Kharkiv and a further three killed in the ballistic missile strike on the city of Zaporizhzhia, with scores more injured, according to Ukrainian authorities.
“This was a demonstrative and cynical Russian strike,” Zelensky added.
kra24
https://kra27cc.net
Rafaelwem
20 Aug 25 at 7:58 am
It’s hard to come by well-informed people in this particular topic, but you sound like you know what you’re talking about!
Thanks
آموزش ترید در تهرا
20 Aug 25 at 7:59 am
Have you ever thought about including a little bit more than just your articles?
I mean, what you say is valuable and everything. Nevertheless imagine
if you added some great visuals or video clips to give your posts more, “pop”!
Your content is excellent but with images and videos, this website
could definitely be one of the best in its field. Amazing blog!
how to clean solar pathway lights
20 Aug 25 at 8:00 am
После поступления звонка специалисты нашей клиники оперативно выезжают по адресу пациента в Новосибирске. Врач начинает работу с детальной диагностики: измеряет пульс, артериальное давление, сатурацию (уровень кислорода в крови), оценивает состояние нервной и сердечно-сосудистой систем, уточняет наличие хронических заболеваний, аллергических реакций, длительность и тяжесть запоя.
Получить больше информации – [url=https://vyvod-iz-zapoya-novosibirsk0.ru/]вывод из запоя на дому в новосибирске[/url]
RickeyEcomo
20 Aug 25 at 8:01 am
http://sildenapeak.com/# viagra 150 mg pills
Danielchumn
20 Aug 25 at 8:02 am
Но через службу поддержки можно узнать об альтернативной — «Машина времени».
pokerdom
20 Aug 25 at 8:05 am
I’ve read a few good stuff here. Definitely worth bookmarking for revisiting.
I wonder how much attempt you put to create this type of great informative
site.
how to charge a solar light without sunlight
20 Aug 25 at 8:07 am
http://sildenapeak.com/# cheap generic viagra
Danielchumn
20 Aug 25 at 8:07 am
https://www.metooo.io/u/68a0ef941d53fa51f9728e09
Jimmybub
20 Aug 25 at 8:07 am
I simply could not depart your website before suggesting that I actually loved the usual info a person provide on your guests?
Is going to be back incessantly in order to check out
new posts
Feel free to visit my web blog; International career coach for career pivots and growth
International career coach for career pivots and growth
20 Aug 25 at 8:09 am
купить аттестаты за 11 класс 2017 [url=https://arus-diplom21.ru]купить аттестаты за 11 класс 2017[/url] .
Diplomi_vgPr
20 Aug 25 at 8:15 am
It’s really a nice and helpful piece of information. I’m glad that you simply shared this helpful information with us.
Please stay us informed like this. Thank you for sharing.
roof replacement company
20 Aug 25 at 8:19 am
My brother suggested I might like this blog. He was totally right.
This submit truly made my day. You can not imagine simply how so much time
I had spent for this information! Thank you!
https://advocacia.arsenontech.com.br/
20 Aug 25 at 8:19 am
He has had more cordial, more productive, meetings with US President Donald Trump since that now-notorious encounter on February 28.
[url=https://kraken5af44k24fwzohe6fvqfgxfsee4lgydb3ayzkfhlzqhuwlo33ad0.com]kraken7jmgt7yhhe2c4iyilthnhcugfylcztsdhh7otrr6jgdw667pqd onion[/url]
But for Ukrainian President Volodymyr Zelensky, today’s meeting at the White House will surely trigger awkward memories of that very public clash with the US President almost six months ago. Navigating the treacherous waters in which he finds himself today will be no easier.
[url=https://kraken4qzqnoi7ogpzpzwrxk7mw53n5i56loydwiyonu4owxsh4g67ydonion.info]kraken5af44k24fwzohe6fvqfgxfsee4lgydb3ayzkfhlzqhuwlo33ad[/url]
Increasingly, it appears likely he will be told to give up land in exchange for some sort of security guarantees.
The land side of that “deal” will be obvious. It can be drawn on a map. Crimea: gone, says Trump. Donetsk: give all of it up, says Putin, apparently with Trump’s blessing.
But the security guarantees? That’s where far more challenging ideas, like credibility, come into play. Could Zelensky rely on the US to deliver on some NATO Article 5-type promise, to defend Ukraine if Russia breaches any peace agreement?
Putin himself might even see an opportunity to further weaken the West, by testing any such guarantees, confident they are a bluff he could call. But all that would be for the future.
For now, it looks like Zelensky will have to weigh up whether he could bring his country with him if he were to cede territory to Russia – some of it still in Ukrainian hands – or whether he and his people could bear the costs of potentially defying Trump a Nobel Peace Prize, and say no.
If he chose the latter, would the US President immediately end all remaining American support for Ukraine, in terms of military aid and intelligence sharing, for instance?
If that happened, to what extent could Zelensky’s European allies really step in and fill in the gaps left by any full US retreat?
It is an almost impossibly hard choice before him.
kraken2trfqodidvlh4aa337cpzfrhdlfldhve5nf7njhumwr7instad
https://kraken5af44k24fwzohe6fvqfgxfsee4lgydb3ayzkfhlzqhuwlo33adonion.info
Jamesbow
20 Aug 25 at 8:20 am
Основные типы бетонных опор
несущая способность забивной сваи 150х150
20 Aug 25 at 8:20 am
Вот, что говорят эксперты по этому поводу:
По теме “obender.ru”, там просто кладезь информации.
Ссылка ниже:
[url=https://obender.ru]https://obender.ru[/url]
Буду следить за обсуждением.
rusPoito
20 Aug 25 at 8:20 am
купить аттестат 11 класс рб [url=arus-diplom22.ru]купить аттестат 11 класс рб[/url] .
Diplomi_sfKt
20 Aug 25 at 8:21 am
This is a really good tip especially to those fresh to the blogosphere.
Simple but very precise info… Appreciate
your sharing this one. A must read article!
nhà cái uy tín
20 Aug 25 at 8:21 am
Этот текст призван помочь читателю расширить кругозор и получить практические знания. Мы используем простой язык, наглядные примеры и структурированное изложение, чтобы сделать обучение максимально эффективным и увлекательным.
Информация доступна здесь – https://www.fourfliesrecords.com/logo_nero
BillyCer
20 Aug 25 at 8:22 am
He has had more cordial, more productive, meetings with US President Donald Trump since that now-notorious encounter on February 28.
[url=https://kraken3yvbvzmhytnrnuhsy772i6dfobofu652e27f5hx6y5cpj7rgydd.com]kraken2trfqodidvlh4aa337cpzfrhdlfldhve5nf7njhumwr7instad[/url]
But for Ukrainian President Volodymyr Zelensky, today’s meeting at the White House will surely trigger awkward memories of that very public clash with the US President almost six months ago. Navigating the treacherous waters in which he finds himself today will be no easier.
[url=https://kraken2trfqodidvlh4aa337cpzfrhdlfldhve5n7instad.com]kraken5af44k24fwzohe6fvqfgxfsee4lgydb3ayzkfhlzqhuwlo33ad onion[/url]
Increasingly, it appears likely he will be told to give up land in exchange for some sort of security guarantees.
The land side of that “deal” will be obvious. It can be drawn on a map. Crimea: gone, says Trump. Donetsk: give all of it up, says Putin, apparently with Trump’s blessing.
But the security guarantees? That’s where far more challenging ideas, like credibility, come into play. Could Zelensky rely on the US to deliver on some NATO Article 5-type promise, to defend Ukraine if Russia breaches any peace agreement?
Putin himself might even see an opportunity to further weaken the West, by testing any such guarantees, confident they are a bluff he could call. But all that would be for the future.
For now, it looks like Zelensky will have to weigh up whether he could bring his country with him if he were to cede territory to Russia – some of it still in Ukrainian hands – or whether he and his people could bear the costs of potentially defying Trump a Nobel Peace Prize, and say no.
If he chose the latter, would the US President immediately end all remaining American support for Ukraine, in terms of military aid and intelligence sharing, for instance?
If that happened, to what extent could Zelensky’s European allies really step in and fill in the gaps left by any full US retreat?
It is an almost impossibly hard choice before him.
kraken7jmgt7yhhe2c4iyilthnhcugfylcztsdhh7otrr6jgdw667pqd
https://kraken7jmgt7yhhe2c4iyilthnhcugfylcztsdhh7otrr6jgdw667pqd0.com
Jamesbow
20 Aug 25 at 8:23 am
Этот текст призван помочь читателю расширить кругозор и получить практические знания. Мы используем простой язык, наглядные примеры и структурированное изложение, чтобы сделать обучение максимально эффективным и увлекательным.
Ознакомьтесь с аналитикой – https://www.websitescrawl.com/domain-list-6441
MarioPoild
20 Aug 25 at 8:24 am
промокод для мелбет [url=http://melbet3006.com/]http://melbet3006.com/[/url]
melbet_qxpa
20 Aug 25 at 8:24 am
Hey great blog! Does running a blog like this take a large amount of work?
I have virtually no knowledge of programming however I was hoping to start my own blog soon. Anyways,
should you have any suggestions or techniques for new blog owners
please share. I know this is off topic but I simply had to ask.
Many thanks!
Arbitrox
20 Aug 25 at 8:25 am
Tadalify [url=http://tadalify.com/#]generic cialis tadalafil 20mg reviews[/url] Tadalify
RobertCat
20 Aug 25 at 8:28 am
https://kemono.im/lobadacyfyde/madrid-kupit-kokain-mefedron-marikhuanu
Jimmybub
20 Aug 25 at 8:28 am
прогноз футбол на сегодня [url=https://prognozy-na-futbol-5.ru/]prognozy-na-futbol-5.ru[/url] .
prognozi na fytbol_ueoa
20 Aug 25 at 8:28 am
https://shootinfo.com/author/biaradokkan/?pt=ads
Samuelloofe
20 Aug 25 at 8:28 am
He has had more cordial, more productive, meetings with US President Donald Trump since that now-notorious encounter on February 28.
[url=https://kraken2trfqodidvlh4aa337cpzfrhdlfldhve5n7instad.com]kraken7jmgt7yhhe2c4iyilthnhcugfylcztsdhh7otrr6jgdw667pqd onion[/url]
But for Ukrainian President Volodymyr Zelensky, today’s meeting at the White House will surely trigger awkward memories of that very public clash with the US President almost six months ago. Navigating the treacherous waters in which he finds himself today will be no easier.
[url=https://kraken5af44k24fwzohe6fvqfgxfsee4lgydb3ayzkfhlzqhuwlo3ad.com]kraken3yvbvzmhytnrnuhsy772i6dfobofu652e27f5hx6y5cpj7rgyd.onion[/url]
Increasingly, it appears likely he will be told to give up land in exchange for some sort of security guarantees.
The land side of that “deal” will be obvious. It can be drawn on a map. Crimea: gone, says Trump. Donetsk: give all of it up, says Putin, apparently with Trump’s blessing.
But the security guarantees? That’s where far more challenging ideas, like credibility, come into play. Could Zelensky rely on the US to deliver on some NATO Article 5-type promise, to defend Ukraine if Russia breaches any peace agreement?
Putin himself might even see an opportunity to further weaken the West, by testing any such guarantees, confident they are a bluff he could call. But all that would be for the future.
For now, it looks like Zelensky will have to weigh up whether he could bring his country with him if he were to cede territory to Russia – some of it still in Ukrainian hands – or whether he and his people could bear the costs of potentially defying Trump a Nobel Peace Prize, and say no.
If he chose the latter, would the US President immediately end all remaining American support for Ukraine, in terms of military aid and intelligence sharing, for instance?
If that happened, to what extent could Zelensky’s European allies really step in and fill in the gaps left by any full US retreat?
It is an almost impossibly hard choice before him.
kraken7jmgt7yhhe2c4iyilthnhcugfylcztsdhh7otrr6jgdw667pqd onion
https://kraken2trfqodidvlh4aa337cpzfrhdlfldhve5nf7njhumwr7inst.com
Thomasslete
20 Aug 25 at 8:30 am
whoah this blog is wonderful i love reading your articles.
Keep up the good work! You know, lots of persons are searching round for this information, you could aid them greatly.
kèo kèo nhà cái
20 Aug 25 at 8:31 am
melbet casino [url=http://melbet3006.com/]melbet casino[/url]
melbet_sppa
20 Aug 25 at 8:34 am
melbet app new version [url=www.melbet3006.com]www.melbet3006.com[/url]
melbet_arpa
20 Aug 25 at 8:36 am
общение психолога анонимно онлайн
Charliesoall
20 Aug 25 at 8:36 am
Одним из самых популярных брендов считается Казино Pokerdom.
Matthewrib
20 Aug 25 at 8:36 am
cheap online viagra: buy sildenafil without prescription cheap – SildenaPeak
ElijahKic
20 Aug 25 at 8:39 am
Уникальная 10 Fruitata Wins игра доступна круглосуточно на платформе.
TimothyUndum
20 Aug 25 at 8:40 am
What’s up, of course this post is really fastidious and I have learned lot of things from it
regarding blogging. thanks.
فرق دبیری با فرهنگیان
20 Aug 25 at 8:40 am
прогнозы на сегодня футбол [url=www.prognozy-na-futbol-6.ru]прогнозы на сегодня футбол[/url] .
prognozi na fytbol_kdsn
20 Aug 25 at 8:43 am
https://shootinfo.com/author/umanafawwaz/?pt=ads
Jimmybub
20 Aug 25 at 8:49 am
Good day! Do you use Twitter? I’d like to follow you if that would
be ok. I’m undoubtedly enjoying your blog and look forward to new posts.
เว็บบาคาร่า อันดับ 1
20 Aug 25 at 8:50 am
1win вход в личный [url=https://1win22097.ru]https://1win22097.ru[/url]
1win_wxpr
20 Aug 25 at 8:50 am
There is certainly a great deal to find out about this topic.
I really like all the points you made.
sistema de gestión de vacaciones
20 Aug 25 at 8:53 am
купить обложку аттестата за 11 [url=https://arus-diplom21.ru]купить обложку аттестата за 11[/url] .
Diplomi_hrPr
20 Aug 25 at 8:54 am
Thank you, I have recently been searching for info about
this topic for a while and yours is the greatest I have found out so far.
But, what in regards to the bottom line?
Are you sure concerning the source?
evertsberg.com
20 Aug 25 at 8:54 am
Howdy this is kind of of off topic but I was wondering
if blogs use WYSIWYG editors or if you have to manually code
with HTML. I’m starting a blog soon but have no coding knowledge so I wanted to get guidance
from someone with experience. Any help would be enormously appreciated!
https://yapolo.za.com
20 Aug 25 at 8:58 am
Wonderful website. Plenty of useful information here.
I am sending it to a few friends ans additionally sharing in delicious.
And certainly, thank you to your effort!
restaurant tableware wholesale
20 Aug 25 at 9:00 am
купить аттестат 11 классов нижний новгород [url=http://arus-diplom22.ru/]купить аттестат 11 классов нижний новгород[/url] .
Diplomi_yfKt
20 Aug 25 at 9:00 am
SildenaPeak: best viagra pills in usa – generic sildenafil 100mg price
ElijahKic
20 Aug 25 at 9:00 am