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=karniz-s-elektroprivodom-kupit.ru]karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_reEr
15 Sep 25 at 8:36 pm
I was wondering if you ever thought of changing
the layout of your site? Its very well written; I love what youve
got to say. But maybe you could a little more in the way of content so people could connect with it better.
Youve got an awful lot of text for only having one or
2 pictures. Maybe you could space it out better?
casino på nett
15 Sep 25 at 8:38 pm
электропривод рулонных штор [url=https://www.avtomaticheskie-rulonnye-shtory5.ru]https://www.avtomaticheskie-rulonnye-shtory5.ru[/url] .
avtomaticheskie rylonnie shtori_aisr
15 Sep 25 at 8:39 pm
рулонные шторы на балконные окна [url=https://www.elektricheskie-rulonnye-shtory15.ru]https://www.elektricheskie-rulonnye-shtory15.ru[/url] .
elektricheskie rylonnie shtori_hbEi
15 Sep 25 at 8:40 pm
электрокарниз двухрядный [url=https://karnizy-s-elektroprivodom-cena.ru/]https://karnizy-s-elektroprivodom-cena.ru/[/url] .
karnizi s elektroprivodom cena_cnkr
15 Sep 25 at 8:41 pm
электрокарнизы для штор купить в москве [url=https://karniz-s-elektroprivodom-kupit.ru]https://karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_crEr
15 Sep 25 at 8:41 pm
кракен даркнет сайт позволяет обойти возможные блокировки и получить доступ к маркетплейсу. [url=https://www.rcgauto.com/services/]рабочее зеркало кракен[/url] необходимо искать через проверенные источники, чтобы избежать фишинговых сайтов. кракен сайт официальный должно обновляться регулярно для обеспечения непрерывного доступа.
администрирование – главная особенность Кракен.
KrakenOthex
15 Sep 25 at 8:44 pm
электрокарнизы [url=http://karnizy-s-elektroprivodom-cena.ru]электрокарнизы[/url] .
karnizi s elektroprivodom cena_llkr
15 Sep 25 at 8:45 pm
электрические жалюзи [url=http://avtomaticheskie-rulonnye-shtory5.ru/]http://avtomaticheskie-rulonnye-shtory5.ru/[/url] .
avtomaticheskie rylonnie shtori_wcsr
15 Sep 25 at 8:45 pm
Check ߋut the leading promotions оn Kaizenaire.ϲom, Singapore’s Ьest
deals site.
Τhe power оf Singapore ɑs ɑ shopping heaven matches flawlessly
ᴡith residents’ love fоr gettіng promotions and deals.
Going to white wine tasting occasions sophisticates tһe
tastes οf gourmet Singaporeans, ɑnd remember to remɑin upgraded on Singapore’s neweѕt promotions annd shopping deals.
SATS tɑkes care ⲟf air travel and food remedies,
appreciated ƅy Singaporeans fоr their іn-flight food catering аnd ground handling performance.
Ꭺ Kind Studio concentrates on sustainable fashion jewelry ɑnd devices leh, cherished ƅy environmentally friendly Singaporeans fоr theіr moral craftsmanship ᧐ne.
Aalst Chocolate crafts Belgian-inspired chocolates, enjoyed fⲟr smooth, indulgent bars аnd regional developments.
Ԝhy ѕ᧐ slow leh, Kaizenaire.com һаs fresh offers one.
Нere is my blog post: health board promotions, zoid.printdirect.ru,
zoid.printdirect.ru
15 Sep 25 at 8:46 pm
повесить рулонные шторы цена за работу [url=http://elektricheskie-rulonnye-shtory15.ru/]http://elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_jbEi
15 Sep 25 at 8:46 pm
Мы изготавливаем дипломы любых профессий по доступным тарифам. Заказ документа, подтверждающего обучение в университете, – это выгодное решение. Заказать диплом о высшем образовании: [url=http://forum.alwehdaclub.sa/read-blog/17618_kupit-diplom-vuza-nedorogo.html/]forum.alwehdaclub.sa/read-blog/17618_kupit-diplom-vuza-nedorogo.html[/url]
Mazryjs
15 Sep 25 at 8:46 pm
карнизы для штор с электроприводом [url=www.karnizy-s-elektroprivodom-cena.ru/]карнизы для штор с электроприводом[/url] .
karnizi s elektroprivodom cena_qpkr
15 Sep 25 at 8:48 pm
Incredible! This blog looks just like my old one! It’s on a totally different
subject but it has pretty much the same layout and design. Great
choice of colors!
Live Sydney
15 Sep 25 at 8:48 pm
Tetapi mana tahu jua keturunan serigala China serta
serigala Eropa baku kawin semasa bertahun-tahun nan lalu dan mendatangkan bermacam rupa ras anjing
mini asal Asia. Sejalan pakai itu, objek sebutan Indonesia diambil
mengenai berjenis-jenis leluhur, terutama pada tiga kaum gaya nan berharga,
yakni (1) sopan santun Indonesia, terpikir partikel serapannya, pula dialek Melayu, (2) dialek Nusantara nan seperindukan, tersisip
bahasa Jawa Kuno, serta (3) adab pendatang, seakan-akan kaidah Inggris lagi percakapan Arab.
Sejak dikumandangkan demi sopan santun persatuan famili Indonesia, pendayagunaan perilaku Indonesia makin luas ke berbagai ragam bidang
kehidupan, makin berpeluang menjadi etiket bidang keterampilan. Di dalam kebiasaan hit,
berjenis-jenis tipu anjing menjadi beken akibat perannya
dalam novel, serial televisi, film, dan pertunjukan gambar bergerak.
40Mereka kendati duduk dalam sekutu; ada nan seratus anak
buah sekawanan bersama ada yang lima puluh insan. Seisi pura
Padang perak mengecek dana yang sejumlah itu pula nan bertambahtambah secepat itu, sama tidak ada diketahui persona bagaimanakah Datuk Meringgih memperolehnya, sebab yang didengar lalu dilihat penduduk,
hanyalah bakhil maka lobanya hanya. Telinganya besar, ibarat kuping
gajah, indra peraba mukanya bimbang marut selanjutnya penuh sambil bekas komplikasi cacar.
Di kian jelas kelihatan, penduduk yang baru datang
itu menjalankan destar hitam nan seni nan ujungnya dibalikkannya ke mukanya sehingga dahinya tertutup.
PEPEK KONTOL
15 Sep 25 at 8:48 pm
you are really a good webmaster. The web site loading speed is incredible.
It seems that you’re doing any unique trick. Also, The contents are masterpiece.
you’ve performed a fantastic task on this subject!
hang khong viet nam
15 Sep 25 at 8:49 pm
What i do not realize is in truth how you are no longer actually much more well-favored than you might be right now.
You are very intelligent. You understand thus considerably when it comes to
this matter, made me in my view imagine it from numerous numerous angles.
Its like men and women aren’t interested until it’s one thing to accomplish with Girl gaga!
Your personal stuffs great. All the time take care
of it up!
contemporary aesthetic
15 Sep 25 at 8:49 pm
рулонные шторы электрические [url=https://www.avtomaticheskie-rulonnye-shtory5.ru]https://www.avtomaticheskie-rulonnye-shtory5.ru[/url] .
avtomaticheskie rylonnie shtori_busr
15 Sep 25 at 8:51 pm
https://telegra.ph/Rejting-luchshih-onlajn-kazino-2025–TOP-nadezhnyh-sajtov-dlya-igry-na-realnye-dengi-09-15-2
Eugeneget
15 Sep 25 at 8:51 pm
This is very interesting, You’re a very skilled blogger.
I have joined your feed and look forward to seeking more of your excellent post.
Also, I have shared your website in my social networks!
Épure Fundex
15 Sep 25 at 8:54 pm
I’m gone to convey my little brother, that he should also pay a quick visit this website on regular basis to obtain updated from most up-to-date reports.
tiktokslot88
15 Sep 25 at 8:58 pm
rezeptfreie medikamente für erektionsstörungen: rezeptfreie medikamente für erektionsstörungen – europa apotheke
Donaldanype
15 Sep 25 at 8:59 pm
https://keeganufyi005.almoheet-travel.com/como-la-alimentacin-influye-en-la-deteccin-de-sustancias-durante-una-prueba-de-orina
Superar una prueba preocupacional puede ser estresante. Por eso, se ha creado un metodo de enmascaramiento creada con altos estandares.
Su receta potente combina vitaminas, lo que estimula tu organismo y neutraliza temporalmente los trazas de toxinas. El resultado: una muestra limpia, lista para entregar tranquilidad.
Lo mas notable es su ventana de efectividad de 4 a 5 horas. A diferencia de otros productos, no promete limpiezas magicas, sino una herramienta puntual que responde en el momento justo.
Miles de personas en Chile ya han comprobado su efectividad. Testimonios reales mencionan resultados exitosos en pruebas preocupacionales.
Si no deseas dejar nada al azar, esta alternativa te ofrece tranquilidad.
JuniorShido
15 Sep 25 at 9:00 pm
электрокранизы [url=https://karnizy-s-elektroprivodom-cena.ru]https://karnizy-s-elektroprivodom-cena.ru[/url] .
karnizi s elektroprivodom cena_nkkr
15 Sep 25 at 9:00 pm
Далее начинается детоксикация и купирование симптомов. Инфузионная терапия позволяет восстановить водно-электролитный баланс, снизить нагрузку на печень и продукты распада этанола. По показаниям применяются противорвотные, противотревожные и седативные средства, витамины группы B, магний, гепатопротекторы, антиоксиданты. Цель — убрать тремор, головную боль, тошноту, нормализовать сон и снизить уровень тревоги.
Подробнее тут – https://narkolog-na-dom-krasnogorsk6.ru/narkolog-na-dom-srochno-v-krasnogorske/
Raymondfem
15 Sep 25 at 9:00 pm
купить диплом института ссср [url=http://www.educ-ua17.ru]http://www.educ-ua17.ru[/url] .
Diplomi_blSl
15 Sep 25 at 9:01 pm
электрокарнизы купить в москве [url=https://karniz-s-elektroprivodom-kupit.ru/]karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_ajEr
15 Sep 25 at 9:04 pm
Hey I know this is off topic but I was wondering
if you knew of any widgets I could add to my blog that automatically tweet
my newest twitter updates. I’ve been looking for a plug-in like this for quite some time and was hoping maybe you would have
some experience with something like this. Please let me know if you run into anything.
I truly enjoy reading your blog and I look forward to your new updates.
casino ohne einschränkung
15 Sep 25 at 9:04 pm
Процедура начинается с осмотра и сбора анамнеза. После этого специалист проводит экстренную детоксикацию, снимает симптомы абстинентного синдрома, назначает поддерживающую терапию и даёт рекомендации по дальнейшим шагам. По желанию родственников или самого пациента помощь может быть оказана и в условиях стационара клиники.
Исследовать вопрос подробнее – [url=https://narkologicheskaya-pomoshch-domodedovo6.ru/]narkologicheskaya-pomoshch-na-domu-kruglosutochno[/url]
Davidcratt
15 Sep 25 at 9:06 pm
карниз с приводом для штор [url=http://karniz-s-elektroprivodom-kupit.ru/]http://karniz-s-elektroprivodom-kupit.ru/[/url] .
karniz s elektroprivodom kypit_ozEr
15 Sep 25 at 9:07 pm
рулонные шторы на окно в кухне [url=http://elektricheskie-rulonnye-shtory15.ru/]http://elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_enEi
15 Sep 25 at 9:08 pm
электрокранизы [url=http://karnizy-s-elektroprivodom-cena.ru/]http://karnizy-s-elektroprivodom-cena.ru/[/url] .
karnizi s elektroprivodom cena_kzkr
15 Sep 25 at 9:09 pm
электрические гардины для штор [url=https://www.karniz-s-elektroprivodom-kupit.ru]https://www.karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_muEr
15 Sep 25 at 9:10 pm
рулонные шторы с автоматическим управлением [url=www.elektricheskie-rulonnye-shtory15.ru/]www.elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_tlEi
15 Sep 25 at 9:12 pm
Oh man, еѵen if institution proves atas, mathematics acts lie tһe critical topic tօ cultivates confidence
regarding numƅers.
Alas, primary mathematics teaches everyday implementations including budgeting, tһerefore ensure үour youngster masters tһat properly starting young.
Singapore Sports School balances elite athletic training
ᴡith rigorous academics, supporting champs іn sport and life.
Specialised paths mɑke sure flexible scheduling fоr competitions ɑnd studies.
Ꮤorld-class facilities ɑnd coaching support peak performance ɑnd individual
development. International exposures construct
resilience аnd worldwide networks. Trainees graduate ɑs diciplined leaders, ready f᧐r professional sports or greatеr education.
National Junior College,holding tһe difference аs Singapore’ѕ
fіrst junior college, provides unrivaled avenues for intellectual expedition ɑnd
management cultivation ѡithin ɑ historical ɑnd inspiring campus tһat blends custom witһ contemporary
academic quality. Τһе distinct boarding program promotes ѕelf-reliance аnd a sense of community, ѡhile cutting edge research centers ɑnd specialized laboratories mɑke іt pߋssible for students from diverse
backgrounds to pursue adrvanced studies іn arts, sciences,
and humanities ѡith elective options fߋr personalized knowing courses.
Ingenious programs encourage deep academic
immersion, ѕuch ɑs project-based гesearch аnd interdisciplinary workshops tһat sharpen analytical skills ɑnd foster
creativity ɑmong ambitious scholars. Тhrough comprehensive global partnerships, including student exchanges, worldwide seminars, ɑnd collaborative efforts ѡith abroad universities,
students develop broad networks ɑnd а nuanced understanding
оf worldwide issues. The college’salumni, ѡһ᧐ frequently
presume popular functions іn federal government, academia,
аnd market, exemplify National Junior College’ѕ long lasting contribution to nation-building ɑnd the advancement ⲟf visionary, impactful
leaders.
Hey hey, steady pom ρi pi, maths гemains part of the top disciplines in Junior College, laying
groundwork іn A-Level һigher calculations.
Aiyah, primary math instructs practical implementations including money
management, tһerefore ensure үouг child maters
it riցht fгom early.
Listen սp, calm pom pі pi, math remɑins paгt in the leading subjects іn Junior College, building groundwork for A-Level
advanced math.
Οh dear, lacking robust mathematics ɑt Junior College,
гegardless tߋⲣ school children cоuld falter with secondary calculations,
tһerefore cultivate іt рromptly leh.
Math at А-levels sharpens decision-making under pressure.
Mums ɑnd Dads, fearful ᧐f losing style engaged lah, strong primary maths guides іn bettеr scientific comprehension plᥙs
engineering aspirations.
Wow, math serves ɑs the groundwork block іn primary
schooling, assisting kids fⲟr spatial reasoning іn building paths.
Look into mу site – Anglo-Chinese Junior College
Anglo-Chinese Junior College
15 Sep 25 at 9:13 pm
электрокарниз двухрядный [url=https://karnizy-s-elektroprivodom-cena.ru/]karnizy-s-elektroprivodom-cena.ru[/url] .
karnizi s elektroprivodom cena_hpkr
15 Sep 25 at 9:13 pm
рулонные шторы на окна цена [url=http://avtomaticheskie-rulonnye-shtory5.ru/]http://avtomaticheskie-rulonnye-shtory5.ru/[/url] .
avtomaticheskie rylonnie shtori_atsr
15 Sep 25 at 9:14 pm
автоматические рулонные шторы с электроприводом на окна [url=www.elektricheskie-rulonnye-shtory15.ru/]www.elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_dsEi
15 Sep 25 at 9:15 pm
электрокарнизы купить в москве [url=http://karnizy-s-elektroprivodom-cena.ru/]электрокарнизы купить в москве[/url] .
karnizi s elektroprivodom cena_nukr
15 Sep 25 at 9:16 pm
It’s very simple to find out any matter on web as
compared to books, as I found this paragraph at this web page.
best crypto casinos
15 Sep 25 at 9:17 pm
рулонные шторы автоматические [url=http://avtomaticheskie-rulonnye-shtory5.ru]http://avtomaticheskie-rulonnye-shtory5.ru[/url] .
avtomaticheskie rylonnie shtori_bisr
15 Sep 25 at 9:18 pm
рольшторы с электроприводом [url=https://avtomaticheskie-rulonnye-shtory5.ru/]рольшторы с электроприводом[/url] .
avtomaticheskie rylonnie shtori_qqsr
15 Sep 25 at 9:20 pm
карниз с электроприводом [url=https://www.karniz-s-elektroprivodom-kupit.ru]https://www.karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_srEr
15 Sep 25 at 9:21 pm
прокарниз [url=https://karnizy-s-elektroprivodom-cena.ru/]https://karnizy-s-elektroprivodom-cena.ru/[/url] .
karnizi s elektroprivodom cena_sikr
15 Sep 25 at 9:26 pm
рулонные шторы для панорамных окон [url=https://elektricheskie-rulonnye-shtory15.ru/]https://elektricheskie-rulonnye-shtory15.ru/[/url] .
elektricheskie rylonnie shtori_psEi
15 Sep 25 at 9:27 pm
I am actually thankful to the holder of this web site
who has shared this enormous paragraph at at this place.
affordable seafront hotel
15 Sep 25 at 9:28 pm
автоматические карнизы [url=http://karniz-s-elektroprivodom-kupit.ru/]http://karniz-s-elektroprivodom-kupit.ru/[/url] .
karniz s elektroprivodom kypit_vlEr
15 Sep 25 at 9:29 pm
уличные рулонные шторы [url=avtomaticheskie-rulonnye-shtory5.ru]avtomaticheskie-rulonnye-shtory5.ru[/url] .
avtomaticheskie rylonnie shtori_zgsr
15 Sep 25 at 9:31 pm
I constantly spent my half an hour to read this web site’s articles or
reviews daily along with a mug of coffee.
live draw sdy
15 Sep 25 at 9:33 pm
электрические карнизы для штор в москве [url=https://karniz-s-elektroprivodom-kupit.ru/]karniz-s-elektroprivodom-kupit.ru[/url] .
karniz s elektroprivodom kypit_gmEr
15 Sep 25 at 9:33 pm