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!
фитнес клуб москва фитнес клуб
fitnes-klub-487
14 Oct 25 at 4:51 pm
карниз для штор электрический [url=https://karniz-shtor-elektroprivodom.ru/]карниз для штор электрический[/url] .
karniz dlya shtor s elektroprivodom_xyer
14 Oct 25 at 4:51 pm
тканевый натяжной потолок самара [url=stretch-ceilings-samara-1.ru]тканевый натяжной потолок самара[/url] .
natyajnie potolki samara_wrsl
14 Oct 25 at 4:51 pm
натяжной потолок самара [url=https://www.natyazhnye-potolki-samara-1.ru]https://www.natyazhnye-potolki-samara-1.ru[/url] .
natyajnie potolki samara_evor
14 Oct 25 at 4:54 pm
Система круглосуточной наркологической помощи в «ОмскПрофи» построена на принципе постоянной готовности. В любой момент суток специалисты готовы выехать на дом, принять пациента в стационар или организовать удалённую консультацию. Это особенно важно в острых ситуациях, когда счёт идёт на часы: запой, передозировка, психоз, агрессивное поведение или резкое ухудшение самочувствия. В таких случаях промедление может стоить пациенту жизни.
Получить дополнительную информацию – [url=https://narkologicheskaya-pomoshh-omsk0.ru/]круглосуточная наркологическая помощь[/url]
DennisDycle
14 Oct 25 at 4:54 pm
Je suis envoute par Grandz Casino, est une symphonie de divertissement qui effleure. Les options de jeu au casino sont riches et ephemeres. proposant des slots de casino a theme theatral. offre un soutien qui effleure tout. repondant en un souffle spectral. fluisent comme une sonate spectrale. quand meme plus de bonus pour une harmonie voilee. En conclusion, Grandz Casino c’est un casino a explorer sans tarder pour les passionnes de casinos en ligne! De plus est une cadence visuelle envoutante. ce qui rend chaque session de casino encore plus etheree.
la grandz recree|
twirlshadowlynx6zef
14 Oct 25 at 4:55 pm
купить диплом техникума новосибирск [url=http://frei-diplom10.ru/]купить диплом техникума новосибирск[/url] .
Diplomi_gxEa
14 Oct 25 at 4:55 pm
пластиковые жалюзи с электроприводом [url=https://zhalyuzi-s-elektroprivodom77.ru]https://zhalyuzi-s-elektroprivodom77.ru[/url] .
jaluzi na okna s elektroprivodom_tjpa
14 Oct 25 at 4:55 pm
электронный карниз для штор [url=https://elektrokarnizy797.ru/]электронный карниз для штор[/url] .
elektrokarnizi_mxMl
14 Oct 25 at 4:55 pm
потолочник отзывы [url=stretch-ceilings-samara-1.ru]stretch-ceilings-samara-1.ru[/url] .
natyajnie potolki samara_xksl
14 Oct 25 at 4:58 pm
Linterest — медиа, которое отбирает действительно важное: технологии, наука, спорт, общество и практичные гиды, объясняющие сложное простыми словами. Обновления выходят регулярно, материалы выверены и снабжены понятной аналитикой и подборками, чтобы экономить ваше время. Пересмотрите информационную повестку в пользу качества: загляните на https://linterest.ru/ — удобная навигация, быстрый поиск и темы, которые помогают принимать решения и развиваться без лишнего шума.
jivovProlI
14 Oct 25 at 4:59 pm
электрокарнизы купить в москве [url=http://www.karniz-shtor-elektroprivodom.ru]электрокарнизы купить в москве[/url] .
karniz dlya shtor s elektroprivodom_pcer
14 Oct 25 at 5:04 pm
Someone essentially help to make seriously articles I’d state.
That is the very first time I frequented your web page and to this point?
I amazed with the analysis you made to create this actual submit amazing.
Great activity!
jasa maket Jakarta
14 Oct 25 at 5:05 pm
перепланировка нежилого помещения в многоквартирном доме [url=http://pereplanirovka-nezhilogo-pomeshcheniya9.ru/]перепланировка нежилого помещения в многоквартирном доме[/url] .
pereplanirovka nejilogo pomesheniya_zfKl
14 Oct 25 at 5:06 pm
жалюзи на пульте [url=www.zhalyuzi-s-elektroprivodom77.ru/]жалюзи на пульте[/url] .
jaluzi na okna s elektroprivodom_zppa
14 Oct 25 at 5:06 pm
Дизайнерский ремонт: искусство преображения пространства
Дизайн интерьера играет важную роль в создании комфортной и уютной атмосферы в доме. Сегодня мы поговорим о таком понятии, как дизайнерский ремонт, который позволяет превратить обычное жилье в уникальное пространство, отражающее индивидуальность владельца.
[url=https://designapartment.ru]дизайнерский ремонт москва[/url]
Что такое дизайнерский ремонт?
Дизайнерский ремонт — это комплекс работ, направленных на создание оригинального дизайна помещения. Это не просто обновление отделки, а полноценный творческий процесс, включающий разработку концепции, подбор материалов и мебели, а также реализацию проекта.
Ключевые особенности дизайнерского ремонта:
[url=https://designapartment.ru]дизайнерский ремонт с мебелью москва[/url]
– Индивидуальный подход к каждому проекту.
– Использование качественных материалов и современных технологий.
– Создание уникального стиля, соответствующего вкусам заказчика.
– Оптимизация пространства для максимального комфорта и функциональности.
Виды дизайнерских ремонтов
[url=https://designapartment.ru]дизайнерский ремонт однокомнатной квартиры[/url]
Существует несколько видов дизайнерских ремонтов, каждый из которых имеет свои особенности и преимущества.
#1 Дизайнерский ремонт квартиры
Это наиболее распространенный вид ремонта, подходящий для тех, кто хочет обновить интерьер своей городской квартиры. Специалисты разрабатывают проект, учитывая размеры помещений, пожелания клиента и бюджет. Такой ремонт включает перепланировку, замену коммуникаций, отделочные работы и декорирование.
Пример дизайна: светлая гостиная с панорамными окнами, минималистичный дизайн кухни и спальни в стиле лофт.
#2 Дизайнерский ремонт дома
Такой ремонт предполагает полное преобразование жилого дома, начиная от фундамента и заканчивая крышей. Здесь важно учитывать архитектурные особенности здания, климатические условия региона и предпочтения владельцев. Часто используется экодизайн, натуральные материалы и энергосберегающие технологии.
Пример дизайна: просторный холл с камином, стеклянная веранда с видом на сад, спальня в пастельных тонах.
#3 Дизайнерский ремонт виллы
Ремонт вилл требует особого подхода, поскольку такие объекты часто расположены в живописных местах и имеют большую площадь. Важно сохранить гармонию с окружающей средой, используя природные материалы и цвета. Особое внимание уделяется созданию зон отдыха, бассейнов и садов.
Пример дизайна: роскошная вилла с бассейном, открытая терраса с видами на море, спальная зона в тропическом стиле.
#4 Дизайнерский ремонт коттеджа
Коттедж отличается от обычного дома наличием придомового участка и возможностью организации дополнительных функциональных зон. Ремонт коттеджей включает работу над фасадом, ландшафтом и внутренним пространством. Стили могут варьироваться от классики до хай-тека.
Пример дизайна: двухэтажный коттедж с мансардой, гостиная-столовая в скандинавском стиле, детская комната с игровой зоной.
#5 Дизайнерский ремонт пентхауса
Пентхаус — это элитное жилье, расположенное на верхних этажах зданий с панорамными видами. Для такого типа недвижимости характерны высокие потолки, большие окна и эксклюзивные элементы декора. Проектирование пентхауса требует учета особенностей конструкции здания и пожеланий клиентов относительно приватности и удобства.
Пример дизайна: современный пентхаус с открытой планировкой, кабинет с видом на город, зона отдыха с джакузи.
Заключение
Дизайнерский ремонт — это возможность создать идеальное пространство для жизни и отдыха. Независимо от того, хотите ли вы обновить квартиру, дом, виллу, коттедж или пентхаус, профессиональный подход гарантирует вам комфорт и эстетическое удовольствие на долгие годы.
https://designapartment.ru
дизайнерский ремонт виллы москва
JeffreyTaund
14 Oct 25 at 5:09 pm
сайт натяжной потолок [url=https://stretch-ceilings-samara.ru/]stretch-ceilings-samara.ru[/url] .
natyajnie potolki samara_ckkl
14 Oct 25 at 5:09 pm
Дизайнерский ремонт: искусство преображения пространства
Дизайн интерьера играет важную роль в создании комфортной и уютной атмосферы в доме. Сегодня мы поговорим о таком понятии, как дизайнерский ремонт, который позволяет превратить обычное жилье в уникальное пространство, отражающее индивидуальность владельца.
[url=https://designapartment.ru]дизайнерский ремонт с мебелью[/url]
Что такое дизайнерский ремонт?
Дизайнерский ремонт — это комплекс работ, направленных на создание оригинального дизайна помещения. Это не просто обновление отделки, а полноценный творческий процесс, включающий разработку концепции, подбор материалов и мебели, а также реализацию проекта.
Ключевые особенности дизайнерского ремонта:
[url=https://designapartment.ru]дизайнерский ремонт виллы под ключ[/url]
– Индивидуальный подход к каждому проекту.
– Использование качественных материалов и современных технологий.
– Создание уникального стиля, соответствующего вкусам заказчика.
– Оптимизация пространства для максимального комфорта и функциональности.
Виды дизайнерских ремонтов
[url=https://designapartment.ru]дизайнерский ремонт с мебелью москва[/url]
Существует несколько видов дизайнерских ремонтов, каждый из которых имеет свои особенности и преимущества.
#1 Дизайнерский ремонт квартиры
Это наиболее распространенный вид ремонта, подходящий для тех, кто хочет обновить интерьер своей городской квартиры. Специалисты разрабатывают проект, учитывая размеры помещений, пожелания клиента и бюджет. Такой ремонт включает перепланировку, замену коммуникаций, отделочные работы и декорирование.
Пример дизайна: светлая гостиная с панорамными окнами, минималистичный дизайн кухни и спальни в стиле лофт.
#2 Дизайнерский ремонт дома
Такой ремонт предполагает полное преобразование жилого дома, начиная от фундамента и заканчивая крышей. Здесь важно учитывать архитектурные особенности здания, климатические условия региона и предпочтения владельцев. Часто используется экодизайн, натуральные материалы и энергосберегающие технологии.
Пример дизайна: просторный холл с камином, стеклянная веранда с видом на сад, спальня в пастельных тонах.
#3 Дизайнерский ремонт виллы
Ремонт вилл требует особого подхода, поскольку такие объекты часто расположены в живописных местах и имеют большую площадь. Важно сохранить гармонию с окружающей средой, используя природные материалы и цвета. Особое внимание уделяется созданию зон отдыха, бассейнов и садов.
Пример дизайна: роскошная вилла с бассейном, открытая терраса с видами на море, спальная зона в тропическом стиле.
#4 Дизайнерский ремонт коттеджа
Коттедж отличается от обычного дома наличием придомового участка и возможностью организации дополнительных функциональных зон. Ремонт коттеджей включает работу над фасадом, ландшафтом и внутренним пространством. Стили могут варьироваться от классики до хай-тека.
Пример дизайна: двухэтажный коттедж с мансардой, гостиная-столовая в скандинавском стиле, детская комната с игровой зоной.
#5 Дизайнерский ремонт пентхауса
Пентхаус — это элитное жилье, расположенное на верхних этажах зданий с панорамными видами. Для такого типа недвижимости характерны высокие потолки, большие окна и эксклюзивные элементы декора. Проектирование пентхауса требует учета особенностей конструкции здания и пожеланий клиентов относительно приватности и удобства.
Пример дизайна: современный пентхаус с открытой планировкой, кабинет с видом на город, зона отдыха с джакузи.
Заключение
Дизайнерский ремонт — это возможность создать идеальное пространство для жизни и отдыха. Независимо от того, хотите ли вы обновить квартиру, дом, виллу, коттедж или пентхаус, профессиональный подход гарантирует вам комфорт и эстетическое удовольствие на долгие годы.
https://designapartment.ru
дизайнерский ремонт с мебелью москва
RobertVex
14 Oct 25 at 5:09 pm
These are in fact great ideas in on the topic of blogging.
You have touched some fastidious factors here. Any way keep
up wrinting.
Dash Atarax 300 Recensione
14 Oct 25 at 5:14 pm
потолки натяжные самара [url=https://natyazhnye-potolki-samara-1.ru/]natyazhnye-potolki-samara-1.ru[/url] .
natyajnie potolki samara_oaor
14 Oct 25 at 5:14 pm
тканевые электрожалюзи [url=http://www.zhalyuzi-s-elektroprivodom77.ru]http://www.zhalyuzi-s-elektroprivodom77.ru[/url] .
jaluzi na okna s elektroprivodom_tipa
14 Oct 25 at 5:14 pm
Hi there every one, here every one is sharing these experience, so it’s nice to read this
website, and I used to pay a visit this weblog all the time.
http://git.tocmore.cn:3000/meridithv30578/meridith2003/wiki/Escort-outcall-prague.
14 Oct 25 at 5:15 pm
купить диплом парикмахера [url=http://rudik-diplom14.ru/]купить диплом парикмахера[/url] .
Diplomi_ydea
14 Oct 25 at 5:16 pm
новый фитнес клуб фитнес клуб с бассейном в москве
fitnes-klub-422
14 Oct 25 at 5:17 pm
Sou louco pela ressonancia de Stake Casino, oferece uma aventura que vibra como uma corda de harpa. Tem uma enxurrada de jogos de cassino irados. oferecendo lives que explodem como ecos. O suporte e um eco de eficiencia. com solucoes precisas e instantaneas. Os pagamentos sao lisos como uma corda. de vez em quando queria promocoes que vibram como sinos. Na real, Stake Casino e o point perfeito pros fas de cassino para os apaixonados por slots modernos! Como extra a navegacao e facil como um eco. elevando a imersao ao nivel de um coral.
lido stake|
twistyneonemu4zef
14 Oct 25 at 5:19 pm
потолочкин потолки натяжные [url=www.stretch-ceilings-samara-1.ru/]www.stretch-ceilings-samara-1.ru/[/url] .
natyajnie potolki samara_ycsl
14 Oct 25 at 5:21 pm
Acho simplesmente enredado IJogo Casino, tem um ritmo de jogo que se enrosca como uma cobra. A selecao de titulos e um emaranhado de prazeres. com jogos adaptados para criptomoedas. O suporte e um fio guia. com solucoes precisas e instantaneas. Os saques sao velozes como um cacador na selva. em alguns momentos mais giros gratis seriam uma loucura de selva. No geral, IJogo Casino e o point perfeito pros fas de cassino para os apaixonados por slots modernos! Alem disso o design e um espetaculo visual enredado. dando vontade de voltar como um cipo.
ijogo com paga de verdade|
twistyneonpangolin5zef
14 Oct 25 at 5:21 pm
переустройство нежилого помещения [url=https://pereplanirovka-nezhilogo-pomeshcheniya9.ru/]https://pereplanirovka-nezhilogo-pomeshcheniya9.ru/[/url] .
pereplanirovka nejilogo pomesheniya_akKl
14 Oct 25 at 5:25 pm
Theo dõi những trận so tài kịch tính của giải đá gà CPC4.
88daga phát sóng trực tiếp từ các trường
gà uy tín, mang lại trải nghiệm xem sắc nét và
chân thực. Đừng bỏ lỡ những chiến kê huyền thoại!
CPC3 Archives - Daga88: trực tiếp đá gà Thomo
14 Oct 25 at 5:26 pm
переустройство нежилого помещения [url=https://pereplanirovka-nezhilogo-pomeshcheniya9.ru/]pereplanirovka-nezhilogo-pomeshcheniya9.ru[/url] .
pereplanirovka nejilogo pomesheniya_frKl
14 Oct 25 at 5:27 pm
neue wettseiten
my web blog :: esport live wetten
esport live wetten
14 Oct 25 at 5:27 pm
После стабилизации состояния пациенту предлагается пройти этап реабилитации. В клинике он может занять от 30 до 180 дней, в зависимости от тяжести зависимости, стажа употребления и социальной ситуации. Реабилитационный блок построен по принципам когнитивной реструктуризации: меняются привычные модели мышления, формируются новые способы реагирования на стресс, прорабатываются неразрешённые психологические конфликты. Используются методы психодрамы, арт-терапии, телесно-ориентированной терапии и даже нейропсихологической гимнастики.
Детальнее – [url=https://lechenie-narkomanii-samara0.ru/]наркологическое лечение наркомания в самаре[/url]
Richardcoern
14 Oct 25 at 5:28 pm
https://www.woorips.vic.edu.au/post/2019/05/08/chemical-science-in-12
nxcpnnt
14 Oct 25 at 5:28 pm
потолочкин натяжные потолки самара [url=https://stretch-ceilings-samara.ru/]https://stretch-ceilings-samara.ru/[/url] .
natyajnie potolki samara_dmkl
14 Oct 25 at 5:28 pm
UK online pharmacy without prescription [url=https://britmedsdirect.com/#]BritMeds Direct[/url] Brit Meds Direct
Jameshoasy
14 Oct 25 at 5:29 pm
электрокарнизы для штор [url=https://elektrokarnizy797.ru/]электрокарнизы для штор[/url] .
elektrokarnizi_rfMl
14 Oct 25 at 5:29 pm
Купить диплом ВУЗа можем помочь. Купить диплом Иркутск – [url=http://diplomybox.com/kupit-diplom-irkutsk/]diplomybox.com/kupit-diplom-irkutsk[/url]
Cazrvua
14 Oct 25 at 5:29 pm
потолочкин натяжные потолки отзывы клиентов самара [url=www.natyazhnye-potolki-samara-1.ru]www.natyazhnye-potolki-samara-1.ru[/url] .
natyajnie potolki samara_qtor
14 Oct 25 at 5:32 pm
карниз электро [url=https://elektrokarnizy797.ru/]elektrokarnizy797.ru[/url] .
elektrokarnizi_nhMl
14 Oct 25 at 5:33 pm
Je trouve absolument spectral Grandz Casino, offre un spectacle de plaisir qui s’evapore. vibre avec un voile de jeux varies. incluant des tables qui dansent comme des fantomes. L’assistance du casino est chaleureuse et precise. offrant des solutions claires et instantanees. Le processus du casino est transparent et sans voile. occasionnellement des offres qui vibrent comme une cadence spectrale. Dans l’ensemble, Grandz Casino cadence comme une sonate de victoires pour les passionnes de casinos en ligne! De plus le design du casino est une fresque visuelle spectrale. facilite une experience de casino voilee.
grandz race bonus|
twirlshadowlynx6zef
14 Oct 25 at 5:35 pm
жалюзи с электроприводом [url=http://zhalyuzi-s-elektroprivodom77.ru/]жалюзи с электроприводом[/url] .
jaluzi na okna s elektroprivodom_ekpa
14 Oct 25 at 5:35 pm
https://banzaicasino77.com/
1go casino
14 Oct 25 at 5:36 pm
Just swapped BNB for $MTAUR—smooth on BSC. Referral rewards motivate sharing. Game’s power-ups via tokens strategic.
minotaurus presale
WilliamPargy
14 Oct 25 at 5:38 pm
купить диплом колледжа спб [url=frei-diplom10.ru]frei-diplom10.ru[/url] .
Diplomi_owEa
14 Oct 25 at 5:40 pm
согласование перепланировки в нежилом здании [url=www.pereplanirovka-nezhilogo-pomeshcheniya9.ru/]www.pereplanirovka-nezhilogo-pomeshcheniya9.ru/[/url] .
pereplanirovka nejilogo pomesheniya_luKl
14 Oct 25 at 5:40 pm
натяжной потолок самара [url=stretch-ceilings-samara-1.ru]натяжной потолок самара[/url] .
natyajnie potolki samara_atsl
14 Oct 25 at 5:41 pm
карниз для штор с электроприводом [url=https://www.karniz-shtor-elektroprivodom.ru]карниз для штор с электроприводом[/url] .
karniz dlya shtor s elektroprivodom_uber
14 Oct 25 at 5:42 pm
купить диплом в новоуральске [url=www.rudik-diplom9.ru/]www.rudik-diplom9.ru/[/url] .
Diplomi_kzei
14 Oct 25 at 5:42 pm
купить диплом в смоленске [url=www.rudik-diplom15.ru/]www.rudik-diplom15.ru/[/url] .
Diplomi_dbPi
14 Oct 25 at 5:43 pm
https://padlet.com/muhammadhasnainpk1_/discussion-topic-goes-here-5n3mh7gugixzfnu6/wish/NvylWE5O2dd3W0OX
rlwzlfz
14 Oct 25 at 5:44 pm