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!
Have you ever thought about including a little bit more
than just your articles? I mean, what you say is valuable and
everything. However think of if you added some great visuals or videos to give your posts more, “pop”!
Your content is excellent but with pics and clips, this blog could definitely be one of the
very best in its niche. Great blog!
Zeker Bitlijn
21 Aug 25 at 12:46 am
Hello mates, nice piece of writing and pleasant arguments commented at this
place, I am truly enjoying by these.
Trezik Forge GPT
21 Aug 25 at 12:47 am
купить медицинский диплом с занесением в реестр [url=www.arus-diplom31.ru]купить медицинский диплом с занесением в реестр[/url] .
Priobresti diplom ob obrazovanii!_chOl
21 Aug 25 at 12:49 am
Stavki Prognozy [url=https://www.stavki-prognozy-1.ru]https://www.stavki-prognozy-1.ru[/url] .
stavki prognozy_gmMa
21 Aug 25 at 12:53 am
Pills information for patients. Brand names.
where to buy cheap estradiol online
Some news about meds. Read here.
where to buy cheap estradiol online
21 Aug 25 at 12:54 am
Hi there! I know this is somewhat off topic but I was wondering which blog
platform are you using for this website? I’m getting sick and tired of WordPress because I’ve had issues with hackers and I’m looking at alternatives for another platform.
I would be fantastic if you could point me in the direction of a good platform.
Direct FLT
21 Aug 25 at 12:55 am
Hi there, i read your blog occasionally and i own a similar one and i was just wondering
if you get a lot of spam remarks? If so how do you protect
against it, any plugin or anything you can suggest?
I get so much lately it’s driving me mad so any help
is very much appreciated.
romantic comedy
21 Aug 25 at 12:56 am
I was able to find good information from your blog articles.
Here is my web page :: Artistic Stainless Steel Travel Mug
Artistic Stainless Steel Travel Mug
21 Aug 25 at 12:57 am
https://wanderlog.com/view/lspqphqiif/купить-экстази-кокаин-амфетамин-унаватуна/shared
ArturoVoift
21 Aug 25 at 12:58 am
You actually make it appear really easy together with your presentation however I to find this matter to be
really one thing that I believe I would by no means understand.
It seems too complex and very huge for me.
I am taking a look forward on your subsequent put up,
I’ll attempt to get the grasp of it!
samsongbake.xyz
21 Aug 25 at 1:00 am
Когда организм на пределе, важна срочная помощь в Санкт-Петербурге — это команда опытных наркологов, которые помогут быстро и мягко выйти из запоя без вреда для здоровья.
Углубиться в тему – [url=https://azithromycinum.ru/]вывод из запоя капельница на дому санкт-петербург[/url]
DerekSquar
21 Aug 25 at 1:05 am
Технологические аспекты производства драгоценных припою
Технологические процессы производства
высококачественного припоя из
драгоценных металлов
Ключом к успешному созданию качественных сплавов является
точное соблюдение температурного режима
на каждом этапе слития. Для достижения лучшего результата рекомендуется использовать пирометры высокой точности, способные отслеживать температурные колебания с
точностью до десятых долей градуса.
Это позволяет избежать перегрева и недостатка тепла,
что может негативно сказаться на конечном продукте.
При выборе материалов для создания сплавов старайтесь учитывать их элементы.
Например, добавление меди в золото позволяет увеличить
твердость изделия, что важно для ювелирной отрасли.
Сравнение различных комбинаций металлов также поможет
определить оптимальное соотношение, что снизит риск износа и
повысит долговечность конечного продукта.
Химический анализ исходных компонентов перед проведением операций по
смешиванию непосредственно
влияет на качество получаемого сплава.
Использование спектрометрии для
анализа помогает выявить примеси и предотвратить
их негативное воздействие на свойства сплава.
Обычно рекомендуется проводить анализ до и после литья, чтобы убедиться в стабильности характеристик.
Этап отливки требует выбора правильной формы и
процесса охлаждения. Применение
форм с хорошей теплопередачей, таких как медные или стальные, значительно повышает качество конечного сплава.
Охлаждение должно проходить равномерно для предотвращения образования трещин и улучшения механических
свойств материала.
Выбор и подготовка сырьевых материалов
для драгоценных сплавов
Оптимальный выбор компонентов
для сплавов начинается с тщательной оценки их чистоты.
Используйте металлы с высоким уровнем содержания чистого золота,
серебра и меди, так как примеси могут значительно уменьшить физические
свойства конечного продукта.
Рекомендуется проводить спектроскопический анализ для
определения состава сырья.
Купите материалы у проверенных производителей.
Этот шаг минимизирует риск получения некачественного
сырья. Обязательно ознакомьтесь с
сертификатами, подтверждающими состав и
чистоту. Отдавайте предпочтение компаниям с хорошей репутацией на рынке.
Перед использованием сырьевых компонентов требуется
их тщательная очистка. В качестве одного из методов можно использовать кислотные растворы,
такие как хлористоводородная или азотная кислота.
Помните, что процесс требует соблюдения мер безопасности, включая использование защитной одежды и
работы в хорошо проветриваемых помещениях.
Смешивание металлов для создания
сплавов должно происходить
в строгих пропорциях, определяемых
заранее. Необходимо учитывать свойства каждого компонента, чтобы достичь желаемых характеристик
готового продукта – прочности,
пластичности или коррозионной стойкости.
Для точного определения пропорций может потребоваться предварительное экспериментирование.
При упаковке материалов используйте специализированные контейнеры,
обеспечивающие защиту от влаги
и загрязнений. Это особенно важно для легкоплавких металлов, которые могут изменять свои
свойства при контакте с атмосферой.
После подготовки сырья необходимо
провести финальную проверку
на наличие дефектов, таких
как трещины или вкрапления. Такие элементы способны негативно повлиять на процесс спекания или плавления, а также на итоговые свойства сплава.
Регулярный контроль и анализ позволяют улучшить качество итогового продукта и сократить количество бракованных партий.
Процесс легирования и контроль качества
в производстве precious alloys
Для достижения необходимых свойств сплавов важно
соблюдать точные пропорции компонентов.
Рекомендуется использовать высококачественные металлы, соответствующие стандартам, чтобы минимизировать риск загрязнений.
Анализ химического состава проводите с
помощью спектрометрии, что позволяет
удостовериться в соблюдении рецептуры.
Перед началом процесса смешивания сырья всегда проводите предварительную проверку
и очистку оборудования.
Рекомендуется использовать автоматизированные
системы для адекватного контроля
температуры и времени легирования.
Это позволит добиться необходимой однородности сплава.
Контроль качества готового продукта осуществляется на нескольких этапах.
После формования проводите механическое
тестирование, включая испытания
на прочность и гибкость. Поддерживайте регистры всех тестов для последующей
оценки и аудита.
Лабораторный анализ сплавов, включая анализ
на примеси, обязательно должен следовать после термической обработки.
Использование рентгенофлуоресцентного анализа существенно
упростит процесс получения точной информации о химическом составе.
Не забывайте о важности конечной
проверки. После получения
готового изделия проводите испытания на коррозийную стойкость и температуру плавления.
Применяйте методики контроля, такие как рутинг
и бенчмаркинг, для сравнения с образцами-эталонами.
Это поможет не только в контроле,
но и в дальнейшем улучшении процесса легирования.
Feel free to visit my website :: https://rms-ekb.ru/catalog/izdeliia-iz-dragotsennykh-i-blagorodnykh-metallov/
https://rms-ekb.ru/catalog/izdeliia-iz-dragotsennykh-i-blagorodnykh-metallov/
21 Aug 25 at 1:07 am
Stavki Prognozy [url=https://stavki-prognozy-1.ru/]stavki-prognozy-1.ru[/url] .
stavki prognozy_yzMa
21 Aug 25 at 1:10 am
Наши усилия направлены на то, чтобы вернуть пациентам уверенность в себе, здоровье и качество жизни.
Получить дополнительную информацию – [url=https://tajnyj-vyvod-iz-zapoya.ru/vyvod-iz-zapoya-v-stacionare-v-omske.ru/]анонимный вывод из запоя в омске[/url]
DavidLah
21 Aug 25 at 1:14 am
Врачи клиники «ВитаЛайн» для снятия интоксикации и облегчения состояния пациента применяют исключительно качественные и проверенные лекарственные препараты, подбирая их в зависимости от особенностей ситуации и состояния здоровья пациента:
Выяснить больше – [url=https://narcolog-na-dom-novosibirsk0.ru/]вызов нарколога на дом новосибирск.[/url]
ThomasReT
21 Aug 25 at 1:16 am
https://www.metooo.io/u/689c833911f6033434f2e863
ArturoVoift
21 Aug 25 at 1:19 am
Ощутите себя капитаном, отправившись в незабываемое плавание!
К вашим услугам
катер в аренду казань цена
для захватывающих приключений.
С нами вы сможете:
• Насладиться живописными пейзажами с борта различных
судов.
• Отпраздновать знаменательное
событие в необычной обстановке.
• Провести незабываемый вечер в романтической атмосфере.
• Собраться с близкими друзьями для веселого время
провождения.
Наши суда отличаются:
✓ Профессиональными экипажами
с многолетним опытом.
✓ Повышенным уровнем комфорта для
пассажиров.
✓ Гарантией безопасности на протяжении всей поездки.
✓ Привлекательным внешним видом
и стильным дизайном.
Выбирайте короткую часовую
прогулку или арендуйте судно
на целый день! Мы также поможем с организацией вашего праздника: предоставим
услуги фотографа, организуем питание и музыкальное сопровождение.
Подарите себе и своим близким незабываемые моменты!
Забронируйте вашу водную феерию уже сегодня!
яхта в аренду в казани
21 Aug 25 at 1:20 am
Pretty! This was an incredibly wonderful post. Thank you for
supplying this info.
My web blog – heavy bass booster – jakkoutthebxx trap type beat reproduced by Artsulli
heavy bass booster - jakkoutthebxx trap type beat reproduced by Artsulli
21 Aug 25 at 1:21 am
оценка бизнеса в Москве оценочные организации
ocenochnaya-kompaniya-971
21 Aug 25 at 1:22 am
https://odysee.com/@tibelusysyqy
Charleswar
21 Aug 25 at 1:23 am
подростковый психолог онлайн консультация
Charliesoall
21 Aug 25 at 1:24 am
Затяжной запой опасен для жизни. Врачи наркологической клиники в Челябинске проводят срочный вывод из запоя — на дому или в стационаре. Анонимно, безопасно, круглосуточно.
Получить дополнительные сведения – https://vyvod-iz-zapoya-chelyabinsk13.ru
JamesUrith
21 Aug 25 at 1:24 am
This is the right blog for anybody who wishes to find out about
this topic. You realize a whole lot its almost hard to argue with you (not that I really will
need to…HaHa). You certainly put a fresh spin on a topic which has been discussed for ages.
Wonderful stuff, just wonderful!
تخمین قبولی ۱۴۰۴
21 Aug 25 at 1:25 am
https://tadalify.shop/# Tadalify
Danielchumn
21 Aug 25 at 1:26 am
Fantastic beat ! I wish to apprentice at the same time as you amenhd your website, how could i subscribe for a
weblog web site? The account helped me a appliicable deal.
I have been tiny bit familiar of this your broadcast provided bright clear concept
Gilbert
21 Aug 25 at 1:29 am
http://sildenapeak.com/# best buy viagra online
Danielchumn
21 Aug 25 at 1:31 am
украина свидетельство о браке купить [url=https://educ-ua5.ru]https://educ-ua5.ru[/url] .
Diplomi_xpKl
21 Aug 25 at 1:33 am
прогнозы на спорт бесплатно от профессионалов [url=https://prognozy-na-sport-8.ru]прогнозы на спорт бесплатно от профессионалов[/url] .
prognozi na sport_qymi
21 Aug 25 at 1:34 am
кашпо с автополивом для комнатных растений [url=kashpo-s-avtopolivom-spb.ru]кашпо с автополивом для комнатных растений[/url] .
gorshok s avtopolivom_prsr
21 Aug 25 at 1:36 am
I every time spent my half an hour to read this weblog’s content every day along with a cup of coffee.
как можно похудеть за неделю на 10 кг
21 Aug 25 at 1:39 am
When I originally commented I seem to have clicked on the -Notify me when new comments are added- checkbox and now each time a comment is added I recieve 4 emails with
the same comment. There has to be an easy method you are able to remove me from that service?
Thank you!
Botol Minum Souvenir
21 Aug 25 at 1:40 am
https://www.themeqx.com/forums/users/hictyuuiuaei/
ArturoVoift
21 Aug 25 at 1:40 am
Hi there! I know this is somewhat off topic but I was wondering if you
knew where I could find a captcha plugin for my
comment form? I’m using the same blog platform as yours and I’m having problems finding one?
Thanks a lot!
سایت مصاحبه فرهنگیان
21 Aug 25 at 1:42 am
stavkiprognozy [url=https://www.stavki-prognozy-1.ru]https://www.stavki-prognozy-1.ru[/url] .
stavki prognozy_wqMa
21 Aug 25 at 1:43 am
прогнозы на спорт бесплатные [url=https://prognozy-na-sport-8.ru/]prognozy-na-sport-8.ru[/url] .
prognozi na sport_dumi
21 Aug 25 at 1:45 am
СтавкиПрогнозы [url=www.stavki-prognozy-1.ru/]www.stavki-prognozy-1.ru/[/url] .
stavki prognozy_zfMa
21 Aug 25 at 1:47 am
вывод из запоя смоленск
vivod-iz-zapoya-smolensk011.ru
экстренный вывод из запоя смоленск
vivodzapojsmolenskNeT
21 Aug 25 at 1:51 am
Каждый день запоя увеличивает риск для жизни. Не рискуйте — специалисты в Челябинске приедут на дом и окажут экстренную помощь. Без боли, стресса и ожидания.
Разобраться лучше – [url=https://vyvod-iz-zapoya-chelyabinsk12.ru/]нарколог вывод из запоя в челябинске[/url]
JesseBut
21 Aug 25 at 1:59 am
купить диплом о высшем образовании с занесением в реестр в калуге [url=www.arus-diplom31.ru]купить диплом о высшем образовании с занесением в реестр в калуге[/url] .
Vigodno kypit diplom o visshem obrazovanii!_uqOl
21 Aug 25 at 2:01 am
https://linkin.bio/xqodoroninastanislava
Charleswar
21 Aug 25 at 2:02 am
http://webanketa.com/forms/6mrk6d9n64qpcc9mc8rp2s1g/
ArturoVoift
21 Aug 25 at 2:03 am
блэкспрут зеркало
RichardPep
21 Aug 25 at 2:09 am
Nice blog! Is your theme custom made or did you download
it from somewhere? A theme like yours with a few simple adjustements would really make my blog shine.
Please let me know where you got your design. Appreciate it
brightest solar spotlight
21 Aug 25 at 2:11 am
бесплатные высокоточные прогнозы на спорт [url=prognozy-na-sport-8.ru]бесплатные высокоточные прогнозы на спорт[/url] .
prognozi na sport_qvmi
21 Aug 25 at 2:13 am
Кодирование подходит не только для тех, кто страдает тяжёлой зависимостью с частыми рецидивами, но и для тех, кто хочет закрепить результат длительной ремиссии, повысить личную мотивацию, снизить риск срыва и облегчить адаптацию к жизни без спиртного. Решение о выборе методики принимается только после осмотра врача и сбора полного анамнеза.
Подробнее – [url=https://kodirovanie-ot-alkogolizma-dolgoprudnyj6.ru/]kodirovanie-ot-alkogolizma-ceny[/url]
Fredrichop
21 Aug 25 at 2:13 am
купить диплом в днепропетровске [url=http://educ-ua4.ru]купить диплом в днепропетровске[/url] .
Diplomi_wmPl
21 Aug 25 at 2:14 am
В нашем центре доступны следующие виды помощи:
Получить дополнительные сведения – https://narko-zakodirovat.ru/vivod-iz-zapoya-cena-v-tveri/
JeraldHor
21 Aug 25 at 2:16 am
купить аттестат об окончании 11 [url=www.arus-diplom9.ru]купить аттестат об окончании 11[/url] .
Diplomi_adEi
21 Aug 25 at 2:18 am
Stavki Prognozy [url=http://stavki-prognozy-1.ru/]http://stavki-prognozy-1.ru/[/url] .
stavki prognozy_yvMa
21 Aug 25 at 2:20 am
Самостоятельно выйти из запоя — почти невозможно. В Санкт-Петербурге врачи клиники проводят медикаментозный вывод из запоя с круглосуточным выездом. Доверяйте профессионалам.
Подробнее можно узнать тут – [url=https://vyvod-iz-zapoya-v-sankt-peterburge11.ru/]вывод из запоя вызов на дом[/url]
Peterbox
21 Aug 25 at 2:20 am