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!
люстры Каждый дом начинается со света. Именно свет создает атмосферу уюта, роскоши и комфорта. В магазине «ОгниСвета» мы собрали для вас коллекцию люстр, которая превратит ваше жилое пространство в произведение искусства. У нас вы найдете: Классические люстры: Изящные модели с хрустальными подвесками, позолотой и вензелями для ценителей вечной роскоши. Они станут центральным элементом вашей гостиной или столовой. Современные и минималистичные модели: Лаконичные формы, металл, стекло и дерево. Идеальное решение для интерьеров в стиле лофт, хай-тек или сканди. Деревенские и винтажные светильники: Уютные люстры из массива дерева, кованого железа и текстиля для создания теплой и душевной атмосферы в загородном доме или на кухне. Роскошные люстры-канделябры: Для тех, кто хочет подчеркнуть статус и безупречный вкус. Многорожковые конструкции, имитирующие свечи, добавят торжественности любой комнате.
JamesInsom
8 Oct 25 at 7:40 am
Выведение из запоя в стационаре Воронежа — помощь при острых состояниях и хронической зависимости. Наши опытные наркологи используют современные методы лечения для быстрого и безопасного вывода из запоя.
Получить дополнительные сведения – [url=https://vyvod-iz-zapoya-v-stacionare-voronezh22.ru/]www.vyvod-iz-zapoya-v-stacionare-voronezh22.ru[/url]
Stevennom
8 Oct 25 at 7:40 am
Программы вывода из запоя в Самаре включают детоксикацию, медикаментозную поддержку и работу с психотерапевтом.
Получить дополнительную информацию – [url=https://vyvod-iz-zapoya-v-stacionare-samara25.ru/]vyvod-iz-zapoya-v-stacionare-samara25.ru/[/url]
Davidcar
8 Oct 25 at 7:41 am
ставки на киберспорт прогнозы на сегодня [url=prognozy-ot-professionalov5.ru]prognozy-ot-professionalov5.ru[/url] .
prognozi ot professionalov_rcSt
8 Oct 25 at 7:44 am
visit this web page link
PHP hook, building hooks in your application – Sjoerd Maessen blog at Sjoerd Maessen blog
visit this web page link
8 Oct 25 at 7:44 am
новости киберспорта [url=sport-novosti-1.ru]sport-novosti-1.ru[/url] .
sport novosti_awpa
8 Oct 25 at 7:47 am
Joined $MTAUR rush—prizes await. ICO’s tokenomics sound. Mazes challenging.
mtaur token
WilliamPargy
8 Oct 25 at 7:47 am
constantly i used to read smaller articles or reviews that also clear their motive, and
that is also happening with this post which I am reading
here.
adam and eve coupon code
8 Oct 25 at 7:49 am
В Екатеринбурге служба Stop-Alko круглосуточно помогает вывести из запоя на дому — быстро, анонимно и без постановки на учёт.
Углубиться в тему – [url=https://vyvod-iz-zapoya-ekaterinburg27.ru/]вывод из запоя клиника екатеринбург[/url]
Donaldawapy
8 Oct 25 at 7:49 am
бесплатные прогнозы на спорт [url=www.prognozy-ot-professionalov5.ru/]бесплатные прогнозы на спорт[/url] .
prognozi ot professionalov_vnSt
8 Oct 25 at 7:49 am
новости тенниса [url=http://sport-novosti-1.ru/]http://sport-novosti-1.ru/[/url] .
sport novosti_jlpa
8 Oct 25 at 7:51 am
This is my first time go to see at here and
i am actually impressed to read all at alone place.
meet ai mode
8 Oct 25 at 7:51 am
Oh my goodness! Amazing article dude! Thank you so much,
However I am having issues with your RSS. I don’t
understand why I am unable to subscribe to it. Is there anyone else having identical RSS issues?
Anyone that knows the answer can you kindly respond?
Thanks!!
bvlgari bulgari alize leoni bag pearled deer black leather
8 Oct 25 at 7:52 am
В Сочи клиника «Детокс» предлагает полный курс вывода из запоя в стационаре. Круглосуточный медицинский контроль гарантирует безопасность и эффективность лечения.
Подробнее – [url=https://vyvod-iz-zapoya-sochi23.ru/]нарколог на дом вывод из запоя[/url]
Jeffreymet
8 Oct 25 at 7:53 am
новости легкой атлетики [url=http://www.sport-novosti-1.ru]http://www.sport-novosti-1.ru[/url] .
sport novosti_nkpa
8 Oct 25 at 7:53 am
Описание
Исследовать вопрос подробнее – [url=https://vyvod-iz-zapoya-odincovo6.ru/]vyvod-iz-zapoya-ryadom[/url]
JohnnyexemE
8 Oct 25 at 7:54 am
свежие прогнозы на спорт [url=https://prognozy-ot-professionalov5.ru/]prognozy-ot-professionalov5.ru[/url] .
prognozi ot professionalov_mzSt
8 Oct 25 at 7:56 am
Prednisone tablets online USA: Prednisone without prescription USA – PredniWell Online
WillieRuivy
8 Oct 25 at 7:57 am
центр наркологической помощи [url=www.narkologicheskaya-klinika-19.ru]www.narkologicheskaya-klinika-19.ru[/url] .
narkologicheskaya klinika _tomi
8 Oct 25 at 8:00 am
sergiidima – It’s refreshing to see innovation in the art world like this.
Madeleine Sensor
8 Oct 25 at 8:01 am
I blog often and I truly appreciate your information. This article
has really peaked my interest. I am going to take a note of your
website and keep checking for new information about once a week.
I subscribed to your Feed too.
adameve discount
8 Oct 25 at 8:02 am
прогнозы футбола точные на сегодня [url=https://www.kompyuternye-prognozy-na-futbol23.ru]https://www.kompyuternye-prognozy-na-futbol23.ru[/url] .
komputernie prognozi na fytbol_qoPi
8 Oct 25 at 8:03 am
спорт новости [url=https://sport-novosti-1.ru/]sport-novosti-1.ru[/url] .
sport novosti_blpa
8 Oct 25 at 8:04 am
buy cialis online canada pharmacy
buy cialis online canada pharmacy
8 Oct 25 at 8:06 am
Торгово-строительная компания “РосКомплект” — это проверенный поставщик строительных материалов в Ижевске, действующий с 2010 года. Мы работаем с частными заказчиками, строительными фирмами и девелоперами, помогая закрывать любые задачи — от возведения кровли до благоустройства участка. В нашем ассортименте представлены металлопрокат, кровельные покрытия, фасадные панели и все необходимые комплектующие. Если вам нужен качественный материал, рекомендуем [url=https://ros18.ru/catalog/metalloprokat/armatura/]купить арматуру мм[/url] напрямую у производителя — с гарантией качества и с быстрой доставкой.
Почему клиенты доверяют именно нас? Потому что мы — это не просто поставщик, а команда практиков, которые знают, какой материал нужен именно вам. Мы предлагаем индивидуальный расчёт, правильную комплектацию листов, гарантируем качество каждой партии и соблюдаем сроки. Наши клиенты ценят ответственность, вежливый сервис и стабильность поставок.
Кроме кровельных материалов, “РосКомплект” реализует и тепличные конструкции для дач и участков. Весной и летом особенно востребованы [url=https://ros18.ru/catalog/profnastil-stalkomplekt/profnastil-krashennyy/]профнастил для крыши цена[/url] — прочные, устойчивые к нагрузкам и практичные в использовании. Мы создаём решения, которые оправдывают ожидания, помогая вам строить свой дом и участок. Оставьте заявку прямо сейчас и получите выгодное предложение от наших специалистов.
Roslviaze
8 Oct 25 at 8:07 am
Simply desire to say your article is as astounding. The
clarity in your post is just nice and i can assume you are an expert on this subject.
Fine with your permission let me to grab your feed to keep updated
with forthcoming post. Thanks a million and please carry on the gratifying work.
stahl pool eckig
8 Oct 25 at 8:07 am
Thank you a lot for sharing this with all folks you actually
recognise what you’re talking approximately! Bookmarked. Kindly additionally visit my website
=). We may have a link exchange arrangement among us
Rovixen Ai Erfahrungen
8 Oct 25 at 8:08 am
I got this website from my buddy who informed me on the topic of this
web site and at the moment this time I am browsing this website
and reading very informative articles here.
Immutable Azopt
8 Oct 25 at 8:10 am
Having read this I believed it was rather informative.
I appreciate you taking the time and energy to put this information together.
I once again find myself spending a lot of time both reading and posting comments.
But so what, it was still worth it!
spirit
8 Oct 25 at 8:11 am
новости тенниса [url=www.sport-novosti-1.ru/]www.sport-novosti-1.ru/[/url] .
sport novosti_mgpa
8 Oct 25 at 8:11 am
Hello there! Would you mind if I share your blog with my twitter group?
There’s a lot of people that I think would really enjoy your content.
Please let me know. Many thanks
Argent Cresvia Avis
8 Oct 25 at 8:14 am
716selfiebuffalo – Loved the 90s-themed room; brought back so many memories.
Serafina Mineconzo
8 Oct 25 at 8:14 am
новости киберспорта [url=www.sport-novosti-1.ru]www.sport-novosti-1.ru[/url] .
sport novosti_xipa
8 Oct 25 at 8:15 am
обзор спортивных событий [url=https://sport-novosti-1.ru/]https://sport-novosti-1.ru/[/url] .
sport novosti_rlpa
8 Oct 25 at 8:17 am
В Краснодаре клиника «Детокс» предоставляет услугу вызова нарколога на дом. Специалисты приедут к вам в течение 1–2 часов, проведут осмотр и назначат необходимое лечение. Все процедуры проводятся анонимно и с соблюдением конфиденциальности.
Получить больше информации – [url=https://narkolog-na-dom-krasnodar26.ru/]нарколог на дом анонимно[/url]
MichaelMed
8 Oct 25 at 8:17 am
It’s amazing in support of me to have a website, which is valuable
designed for my experience. thanks admin
4d
8 Oct 25 at 8:18 am
Superb, what a website it is! This website presents useful
facts to us, keep it up.
buy instagram random comments
8 Oct 25 at 8:18 am
Just grabbed some $MTAUR coins during the presale—feels like getting in on the ground floor of something huge. The audited smart contracts give me peace of mind, unlike sketchier projects. Can’t wait for the game beta to test those power-ups.
mtaur token
WilliamPargy
8 Oct 25 at 8:19 am
прогноз на теннис на сегодня от профессионалов [url=https://prognozy-ot-professionalov5.ru]https://prognozy-ot-professionalov5.ru[/url] .
prognozi ot professionalov_mlSt
8 Oct 25 at 8:22 am
Everything is very open with a really clear clarification of the challenges.
It was truly informative. Your website is useful. Many thanks for sharing!
ambien no prescription
8 Oct 25 at 8:22 am
Добро пожаловать в Клубника Казино, где каждый игрок найдет для
себя идеальные условия для выигрыша и наслаждения игрой.
Мы предлагаем широкий выбор игр, включая классические слоты, рулетку,
блэкджек и уникальные игры с живыми дилерами.
В Клубника Казино мы гарантируем полную
безопасность и прозрачность всех процессов, чтобы
ваши данные и средства были в надежных руках.
Почему Clubnika казино для Android –
лучший выбор для азартных игроков?
В нашем казино каждый игрок может рассчитывать на щедрые
бонусы, бесплатные спины и эксклюзивные предложения.
Кроме того, мы обеспечиваем быстрые выводы средств и круглосуточную поддержку, чтобы вы могли сосредоточиться на игре.
Когда вам стоит начать играть в Клубника Казино?
Зарегистрируйтесь в Клубника Казино и получите бонусы, которые сразу увеличат ваши шансы на победу.
Вот что вас ждет:
Воспользуйтесь щедрыми бонусами и бесплатными спинами,
чтобы начать игру с преимуществом.
Промо-акции и турниры с крупными призами.
Регулярные обновления и новые игры
каждый месяц.
Клубника Казино – это идеальное место для тех, кто хочет
играть и выигрывать.
casino clubnika
8 Oct 25 at 8:22 am
پروتئین کازئین چیست، کازئین نوعی پروتئین از گروه فسفو پروتئینهاست که به طور طبیعی در شیر پستانداران وجود دارد.
کازئین چیست
8 Oct 25 at 8:24 am
Услуга бесплатной юридической консультации предназначена для граждан, желающих
получить первичную правовую помощь.
Она может быть полезна в различных ситуациях,
связанных с правами и интересами отдельных
лиц. В Москве и других городах России существует множество организаций, предлагающих
такие услуги.
Где получить бесплатную юридическую консультацию?
Юридические клиники, финансируемые государством
Неправительственные организации
Сайты с онлайн-консультациями
Ключевые сферы юридической помощи
Консультация, предоставляемая бесплатно, может включать разные области права,
включая:
Правовые вопросы, касающиеся жилья
Право труда
Защита интересов потребителей
Судебные дела
Что нужно для получения консультации?
Для того чтобы воспользоваться
бесплатной юридической консультацией, следует
пройти несколько этапов:
Выбрать подходящее учреждение или адвоката.
Обратиться по телефону или онлайн через сайт.
Подготовить нужные документы и сформулировать свои вопросы.
Какие документы нужны для обращения?
При обращении за юридической
помощью важно иметь при себе:
Все документы, относящиеся к вашему делу.
Номер телефона, который можно записать для
связи.
Сжато изложенное описание вашей ситуации.
Достоинства бесплатных юридических консультаций
Бесплатные юридические консультации имеют
много плюсов:
Возможность общения с опытными юристами и
адвокатами.
Возможность получить независимое мнение по вашему вопросу.
Наличие времени для осмысления своих
прав и обязанностей.
Как правильно выбрать юриста?
При выборе юриста обратите внимание на:
Опыт в соответствующей правовой сфере.
Отзывы клиентов.
Наличие специализации на схожих делах.
Вывод
Для получения подробной информации о бесплатных юридических консультациях лучше всего посетить профилированные сайты
и обратиться за консультацией по телефону в вашу
юридическую компанию. консультации адвоката по уголовным делам бесплатно Заключение
В условиях современного мира, когда правовые аспекты становятся все более
сложными и запутанными, бесплатная юридическая консультация становится незаменимым инструментом для граждан.
Доступность правовой помощи для всех — это большой
плюс в деле защиты своих прав.
Не стоит забывать, что в критических ситуациях, требующих оперативного вмешательства, возможность
получить профессиональную юридическую консультацию без оплаты может оказать значительное влияние на исход дела.
Консультация у юриста или адвоката в подобных случаях может предотвратить множество ошибок, касающихся
правовых вопросов.
Преимущества бесплатной юридической консультации
Получение информации о своих правах и
обязанностях становится доступным;
Возможность получить помощь
в составлении документов;
Профессиональные советы по различным
правовым вопросам;
Поддержка в трудовых, жилищных и других делах;
Сбережение денег на ранних этапах решения проблемы.
Обращаясь за бесплатной юридической консультацией в Москве или других городах, вы получаете возможность понять
свои права и обязанности в различных ситуациях.
Это особенно актуально в
случаях, когда требуется защита ваших интересов в суде или при взаимодействии с государственными органами.
Независимо от того, сталкиваетесь ли
вы с трудовым спором, жилищными
вопросами или вопросами защиты прав потребителей, помощь профессионала будет неоценима.
Юридическая служба предлагает телефонные консультации, что облегчает процесс обращения за
помощью для всех.
Не упустите возможность защитить свои права и
получить грамотные рекомендации от специалистов.
Воспользуйтесь бесплатной юридической консультацией и сделайте шаг к решению
ваших правовых вопросов.
http://www.annunciogratis.net/author/lorenephili
8 Oct 25 at 8:25 am
Официальный источник Азино
три семерки обслуживает посетителей с 2010 года.
азино 777 ру
8 Oct 25 at 8:25 am
наркологическая помощь [url=http://www.narkologicheskaya-klinika-19.ru]http://www.narkologicheskaya-klinika-19.ru[/url] .
narkologicheskaya klinika _lfmi
8 Oct 25 at 8:25 am
В Сочи клиника «Детокс» предоставляет услугу вывода из запоя в стационаре. Профессиональные врачи обеспечат комфортное и безопасное лечение. Минимальная стоимость услуги — 2000 ?.
Подробнее можно узнать тут – [url=https://vyvod-iz-zapoya-sochi24.ru/]вывод из запоя на дому недорого сочи[/url]
RobertoToott
8 Oct 25 at 8:25 am
I wanted to thank you for this excellent read!!
I certainly enjoyed every bit of it. I have got you saved as a favorite
to look at new things you post…
ราชภัฏอันดับ 1
8 Oct 25 at 8:26 am
новости олимпиады [url=https://sport-novosti-1.ru/]sport-novosti-1.ru[/url] .
sport novosti_jmpa
8 Oct 25 at 8:26 am
прогнозы на спорт с описанием [url=http://prognozy-ot-professionalov5.ru/]http://prognozy-ot-professionalov5.ru/[/url] .
prognozi ot professionalov_nqSt
8 Oct 25 at 8:27 am
наркологические клиники в москве частные [url=narkologicheskaya-klinika-19.ru]narkologicheskaya-klinika-19.ru[/url] .
narkologicheskaya klinika _zlmi
8 Oct 25 at 8:29 am