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!
Ridiculous story there. What happened after?
Thanks!
Here is my web-site; African hunting safaris
African hunting safaris
13 Sep 25 at 4:18 pm
https://postheaven.net/maixentedd/habitos-que-pueden-afectar-tu-resultado-en-un-test-de-orina
Aprobar un test antidoping puede ser complicado. Por eso, se ha creado una alternativa confiable creada con altos estandares.
Su formula potente combina creatina, lo que ajusta tu organismo y disimula temporalmente los metabolitos de sustancias. El resultado: una muestra limpia, lista para entregar tranquilidad.
Lo mas valioso es su capacidad inmediata de respuesta. A diferencia de detox irreales, no promete milagros, sino una estrategia de emergencia que responde en el momento justo.
Miles de estudiantes ya han validado su rapidez. Testimonios reales mencionan resultados exitosos en pruebas preocupacionales.
Si no deseas dejar nada al azar, esta solucion te ofrece confianza.
JuniorShido
13 Sep 25 at 4:19 pm
Женский сайт https://bbb.dp.ua всё самое важное для современных девушек: стиль, красота, здоровье, отношения и самореализация. Читайте, вдохновляйтесь и находите новые идеи.
MichaelChuth
13 Sep 25 at 4:20 pm
First of all I want to say awesome blog!
I had a quick question in which I’d like to ask if you do not mind.
I was curious to know how you center yourself
and clear your head prior to writing. I have had a tough time clearing my thoughts in getting my ideas out there.
I truly do take pleasure in writing but it just seems like the
first 10 to 15 minutes are generally wasted simply just trying to figure out how to begin. Any suggestions or hints?
Many thanks!
dewascatter
13 Sep 25 at 4:21 pm
Hello! I just wanted to ask if you ever have any problems
with hackers? My last blog (wordpress) was hacked and I ended
up losing months of hard work due to no back up.
Do you have any methods to prevent hackers?
best online casino games
13 Sep 25 at 4:23 pm
интернет провайдеры ростов
inernetvkvartiru-rostov006.ru
подключить проводной интернет ростов
internetrostovelini
13 Sep 25 at 4:30 pm
купить диплом в черкассах [url=https://educ-ua2.ru/]https://educ-ua2.ru/[/url] .
Diplomi_soOt
13 Sep 25 at 4:31 pm
Комплексное применение этих средств ускоряет процесс восстановления.
Ознакомиться с деталями – https://narkolog-na-dom-sankt-peterburg14.ru/narkolog-v-sankt-peterburge-vyezd-nadom
Robertfloum
13 Sep 25 at 4:32 pm
Meds information for patients. Cautions.
where buy xenical
Some information about medicines. Get information here.
where buy xenical
13 Sep 25 at 4:33 pm
Списавшись в ЛС с ТСом обговорив условия сделки провел оплату в БТЦ.
https://beteiligung.stadtlindau.de/profile/%D0%9A%D1%83%D0%BF%D0%B8%D1%82%D1%8C%20%D0%91%D0%BE%D1%88%D0%BA%D0%B8%20%D0%9C%D0%B0%D1%80%D0%B8%D1%85%D1%83%D0%B0%D0%BD%D1%83%20%D0%93%D0%B0%D1%88%D0%B8%D1%88%20%D0%90%D0%BC%D1%81%D1%82%D0%B5%D1%80%D0%B4%D0%B0%D0%BC/
Добрый вечер!!! :drug:
AnthonyGag
13 Sep 25 at 4:34 pm
Way cool! Some extremely valid points! I appreciate you writing this post plus the rest of the website is very good.
WhatsApp网页版
13 Sep 25 at 4:38 pm
Вызов нарколога на дом сочетает медицинскую эффективность с удобством. Пациент получает квалифицированную помощь в привычной обстановке, что снижает уровень тревожности и способствует более быстрому восстановлению.
Выяснить больше – [url=https://narkolog-na-dom-sankt-peterburg14.ru/]врач нарколог выезд на дом в санкт-петербурге[/url]
Robertfloum
13 Sep 25 at 4:44 pm
Medicine information sheet. Generic Name.
naproxen 250 mg tab
Some trends of medicine. Get here.
naproxen 250 mg tab
13 Sep 25 at 4:44 pm
Hurrah, that’s what I was exploring for, what a stuff!
existing here at this website, thanks admin of this web site.
Merpatislot88 Sportsbook
13 Sep 25 at 4:48 pm
Журнал для женщин https://rpl.net.ua которые строят карьеру и хотят большего. Финансовая грамотность, советы по продуктивности, истории успеха и руководство по переговорам. Достигайте своих целей с нами!
CoreyTiz
13 Sep 25 at 4:48 pm
Журнал для женщин https://rpl.net.ua которые строят карьеру и хотят большего. Финансовая грамотность, советы по продуктивности, истории успеха и руководство по переговорам. Достигайте своих целей с нами!
CoreyTiz
13 Sep 25 at 4:51 pm
Твой гид https://womanlife.kyiv.ua по стильной жизни. Мы собрали всё: от выбора платья на вечер до планирования идеального отпуска. Экспертные советы, подборки и инсайты, чтобы ты всегда чувствовала себя на высоте.
BruceLof
13 Sep 25 at 4:51 pm
CuraBharat USA: top online pharmacy in india – shop medicine online
Charlesdyelm
13 Sep 25 at 4:52 pm
кракен онион тор kraken onion, kraken onion ссылка, kraken onion зеркала, kraken рабочая ссылка onion, сайт kraken onion, kraken darknet, kraken darknet market, kraken darknet ссылка, сайт kraken darknet, kraken актуальные ссылки, кракен ссылка kraken, kraken официальные ссылки, kraken ссылка тор, kraken ссылка зеркало, kraken ссылка на сайт, kraken онион, kraken онион тор, кракен онион, кракен онион тор, кракен онион зеркало, кракен даркнет маркет, кракен darknet, кракен onion, кракен ссылка onion, кракен onion сайт, kra ссылка, kraken сайт, kraken актуальные ссылки, kraken зеркало, kraken ссылка зеркало, kraken зеркало рабочее, актуальные зеркала kraken, kraken сайт зеркала, kraken маркетплейс зеркало, кракен ссылка, кракен даркнет
RichardPep
13 Sep 25 at 4:52 pm
Журнал для женщин https://rpl.net.ua которые строят карьеру и хотят большего. Финансовая грамотность, советы по продуктивности, истории успеха и руководство по переговорам. Достигайте своих целей с нами!
CoreyTiz
13 Sep 25 at 4:52 pm
Онлайн-журнал о моде https://glamour.kyiv.ua без правил. Новые тренды, стильные образы, секреты знаменитостей и советы по созданию идеального гардероба. Мы поможем вам найти и с уверенностью выразить свой уникальный стиль.
Kennethscece
13 Sep 25 at 4:53 pm
Твой гид https://womanlife.kyiv.ua по стильной жизни. Мы собрали всё: от выбора платья на вечер до планирования идеального отпуска. Экспертные советы, подборки и инсайты, чтобы ты всегда чувствовала себя на высоте.
BruceLof
13 Sep 25 at 4:54 pm
Онлайн-журнал о моде https://glamour.kyiv.ua без правил. Новые тренды, стильные образы, секреты знаменитостей и советы по созданию идеального гардероба. Мы поможем вам найти и с уверенностью выразить свой уникальный стиль.
Kennethscece
13 Sep 25 at 4:55 pm
Твой гид https://womanlife.kyiv.ua по стильной жизни. Мы собрали всё: от выбора платья на вечер до планирования идеального отпуска. Экспертные советы, подборки и инсайты, чтобы ты всегда чувствовала себя на высоте.
BruceLof
13 Sep 25 at 4:55 pm
Онлайн-журнал о моде https://glamour.kyiv.ua без правил. Новые тренды, стильные образы, секреты знаменитостей и советы по созданию идеального гардероба. Мы поможем вам найти и с уверенностью выразить свой уникальный стиль.
Kennethscece
13 Sep 25 at 4:57 pm
Женский сайт https://bbb.dp.ua всё самое важное для современных девушек: стиль, красота, здоровье, отношения и самореализация. Читайте, вдохновляйтесь и находите новые идеи.
MichaelChuth
13 Sep 25 at 4:57 pm
Готовитесь к сезону? В Yokohama Киров подберем шины и диски под ваш стиль и условия дорог, выполним профессиональный шиномонтаж без очередей и лишних затрат. В наличии бренды Yokohama и другие проверенные производители. Узнайте цены и услуги на сайте: https://yokohama43.ru/ — консультируем, помогаем выбрать оптимальный комплект и бережно обслуживаем ваш авто. Надёжно, быстро, с гарантией качества.
QyqitPlani
13 Sep 25 at 4:58 pm
флудить в курилку
https://www.band.us/page/99925976/
Ровным парням ровный респект
AnthonyGag
13 Sep 25 at 4:58 pm
Excellent write-up. I certainly love this site. Thanks!
andepzai hub script download
13 Sep 25 at 4:58 pm
SaludFrontera [url=https://saludfrontera.com/#]SaludFrontera[/url] mexico pharmacy
Michaelphype
13 Sep 25 at 4:59 pm
Женский сайт https://bbb.dp.ua всё самое важное для современных девушек: стиль, красота, здоровье, отношения и самореализация. Читайте, вдохновляйтесь и находите новые идеи.
MichaelChuth
13 Sep 25 at 4:59 pm
indian online pharmacy: pharmacy site – buy medicine online in delhi
Teddyroowl
13 Sep 25 at 5:00 pm
Женский сайт https://bbb.dp.ua всё самое важное для современных девушек: стиль, красота, здоровье, отношения и самореализация. Читайте, вдохновляйтесь и находите новые идеи.
MichaelChuth
13 Sep 25 at 5:01 pm
Hey there just wanted to give you a quick heads up.
The words in your content seem to be running off the screen in Internet explorer.
I’m not sure if this is a format issue or something to do with internet browser
compatibility but I thought I’d post to let you know.
The layout look great though! Hope you get the issue fixed soon. Cheers
Zeker Montex
13 Sep 25 at 5:01 pm
«РостовМедЦентр» предоставляет несколько форматов, чтобы лечение не сталкивалось лоб в лоб с работой, учебой и семейными обязанностями. Стационар — для случаев с рисками, дневной стационар — для интенсивной дневной терапии, амбулаторный режим — для поддержания динамики без разрыва с повседневной жизнью. Выезд на дом сохраняет приватность и экономит время, а защищённые видеосессии позволяют поддерживать мотивацию между очными визитами. Все каналы соединены в единую систему: план, составленный сегодня в клинике, доступен врачу, который завтра приедет на дом, и психологу, с которым запланирована онлайн-сессия через два дня.
Изучить вопрос глубже – http://narkologicheskaya-klinika-rostov-na-donu14.ru
Jackiemoips
13 Sep 25 at 5:05 pm
Hello very cool site!! Man .. Beautiful .. Amazing ..
I will bookmark your website and take the
feeds additionally? I’m glad to search out numerous useful info right here in the submit, we
need work out more techniques on this regard, thanks for sharing.
. . . . .
buôn bán nội tạng
13 Sep 25 at 5:10 pm
Optez pour des teintes vives comme le jaune moutarde, le bordeaux ou le turquoise pour rehausser des tenues simples ou compléter des looks monochromes modernes.
Les
13 Sep 25 at 5:11 pm
What’s up Dear, are you actually visiting this website on Cow print x 5 x 10 x 18 x 20 x 23 x 30 x 40 x 60 x 73 x 83 x 90 x 100 x 101 x 105 x 106 x 280 x Jakkoutthebxx’s unbeatable type work of art printed all over a Spun Polyester Square Pillowcase – Shop Cow Print | The Artsulli Blog regular basis, if so afterward you will without doubt get nice
knowledge.
Cow print x 5 x 10 x 18 x 20 x 23 x 30 x 40 x 60 x 73 x 83 x 90 x 100 x 101 x 105 x 106 x 280 x Jakkoutthebxx's unbeatable type work of art printed all over a Spun Polyester Square Pillowcase - Shop Cow Print | The Artsulli Blog
13 Sep 25 at 5:13 pm
Good article. I am experiencing many of these issues as well..
Zarios Platform
13 Sep 25 at 5:13 pm
скачат мостбет [url=http://mostbet12009.ru/]http://mostbet12009.ru/[/url]
mostbet_pfsl
13 Sep 25 at 5:14 pm
http://curabharatusa.com/# indian pharmacy
CarlosPreom
13 Sep 25 at 5:14 pm
Discover curated occasions аt Kaizenaire.com, topping Singapore’ѕ promotions landscape.
Singapore’ѕ condition аs a shopping mecca reverberates ѡith Singaporeans, tһat
ɑlways focus on promotions in thbeir quеst for lots.
Attending a glass оf wine sampling events sophisticates tһe palates
օf premium Singaporeans, and remember tⲟ stay upgraded on Singapore’ѕ
moѕt current promotions ɑnd shopping deals.
Rawbought deals lavish sleepwear аnd lingerie, valued bу Singaporeans foг their comfortable materials аnd
elegant designs.
Ꮲast the Vines produces colorful bags ɑnd apparel lah, treasured by vibrant Singaporeans fоr their fun, practical layouts lor.
Polar Puffs & Cakes lures ᴡith lotion puffs
ɑnd swiss rolls, preferred fߋr light, luscious desserts tһɑt make any celebration sweeter.
Ꮃhy sо slow-moving leh, Kaizenaire.com has fresh offerѕ ߋne.
Feel free tߋ surf to my webpage: Kaizenaire.com Promotions
Kaizenaire.com Promotions
13 Sep 25 at 5:15 pm
hello!,I really like your writing so much!
proportion we communicate extra approximately your post on AOL?
I require an expert on this house to solve my problem.
Maybe that is you! Taking a look ahead to
peer you.
best online casino bonus
13 Sep 25 at 5:16 pm
Для многих барьер — страх огласки. В «РязМедСервисе» конфиденциальность обеспечивается не обещаниями, а процедурами: немаркированный выезд, шифрованный контур карт, ограничение доступа к данным по ролям, нейтральные формулировки в документах и чеках, отсутствие видеозаписей онлайн-сессий. Вы сами выбираете уровень информирования семьи: от полного молчания до адресной коммуникации с доверенным лицом. Визиты согласуются на «тихие окна», а уведомления не раскрывают суть обращения.
Изучить вопрос глубже – [url=https://vyvod-iz-zapoya-v-ryazani14.ru/]вывод из запоя на дому рязань[/url]
Jameszinee
13 Sep 25 at 5:19 pm
РєРѕРіРґР° РІС‹ зарядили?РјРѕРё ребята зарядили вчера около 16-00 РїРѕ РјСЃРє ,РјРёРЅСѓС‚ Р·Р° 15 РґРѕ этого РѕРЅ РґРѕР±СЂРѕ дал Рё замолчал( переживаю очень так как отвечать РјРЅРµ.РЇ надеюсь лавочка РЅРµ слилась СЃ пацанскими деньгими… Рђ то конкретная пичаль будет.РЈ нас РІ регионе это большая СЃСѓРјРјР°!!!
https://yamap.com/users/4810326
По многочисленным восторженным откликам мн-001, тестировал его и был немного огорчен.
AnthonyGag
13 Sep 25 at 5:22 pm
Good day! This is my first visit to your blog! We are a collection of volunteers and starting a new project in a community in the same niche.
Your blog provided us valuable information to work on. You have done a marvellous job!
top casino sites
13 Sep 25 at 5:25 pm
Great article! This is the type of info that are supposed to be shared across the net.
Disgrace on Google for not positioning this put up upper!
Come on over and seek advice from my web site . Thank you =)
tải app mv88
13 Sep 25 at 5:27 pm
whoah this weblog is magnificent i love studying your posts.
Keep up the good work! You understand, lots of people are hunting
around for this information, you could help them greatly.
Oxiris Platform
13 Sep 25 at 5:30 pm
драгон мани зеркало Драгон Мани – платформа для азартных игр с турнирами и слотами. Простота, скорость и шанс на крупный выигрыш!
HowardFig
13 Sep 25 at 5:31 pm
Ключевая идея «ДонЗдрава» — соединить доказательную медицину и понятную пациенту логистику. Инфузии рассчитываются через инфузомат, жизненные показатели контролируются портативным кардиомонитором и пульсоксиметром, а лекарственные взаимодействия проверяются по протоколу перед началом терапии. При этом каждый шаг объясняется простым языком: что делаем, зачем это нужно и как будем оценивать результат через 30, 60 и 120 минут. После визита пациент не остаётся один — доступна «горячая линия» и короткие онлайн-сессии с врачом либо психологом, если тревога или бессонница возвращаются в ночные часы.
Углубиться в тему – [url=https://vivod-iz-zapoya-rostov14.ru/]вывод из запоя цена ростов-на-дону[/url]
BrianBlogy
13 Sep 25 at 5:32 pm