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!
Hi, i think that i saw you visited my site so i came
to return the prefer?.I am trying to in finding issues to
enhance my site!I assume its ok to make use of a
few of your concepts!!
https://revwix.com
21 Jul 25 at 8:28 pm
blacksprut
блэкспрут
black sprut
блэк спрут
blacksprut вход
блэкспрут ссылка
blacksprut ссылка
blacksprut onion
блэкспрут сайт
блэкспрут вход
блэкспрут онион
блэкспрут дакрнет
blacksprut darknet
blacksprut сайт
блэкспрут зеркало
blacksprut зеркало
black sprout
blacksprut com зеркало
блэкспрут не работает
blacksprut зеркала
как зайти на blacksprut blacksprut com зеркало
RichardPep
21 Jul 25 at 8:28 pm
Игнорирование состояния пациента или попытки самостоятельно справиться с запоем могут привести к опасным последствиям для жизни и здоровья. Поэтому важно своевременно обращаться за профессиональной медицинской помощью.
Изучить вопрос глубже – [url=https://narcolog-na-dom-sankt-peterburg0.ru/]нарколог на дом срочно в санкт-петербурге[/url]
PatrickGoarm
21 Jul 25 at 8:30 pm
Современное общество сталкивается с серьёзными вызовами, связанными с зависимостями. Проблемы, возникающие в результате злоупотребления психоактивными веществами, охватывают не только здоровье, но также влияют на социальные отношения и качество жизни. Наркологическая клиника “Точка опоры” предлагает широкий спектр услуг, направленных на восстановление и реабилитацию людей, столкнувшихся с этими трудностями.
Получить дополнительные сведения – [url=https://narko-zakodirovat.ru/]вывод из запоя на дому в твери[/url]
DanielSnamn
21 Jul 25 at 8:33 pm
http://isotretinoinfromcanada.com/# Isotretinoin From Canada
TheronSnipt
21 Jul 25 at 8:36 pm
mental health support chatbot [url=http://www.mental-health21.com]http://www.mental-health21.com[/url] .
Mental Health_cjPa
21 Jul 25 at 8:40 pm
There is certainly a great deal to find out about this topic.
I love all of the points you’ve made.
mami188
21 Jul 25 at 8:44 pm
mostbet uz ga kirish [url=mostbet4075.ru]mostbet4075.ru[/url]
mostbet_avOr
21 Jul 25 at 8:48 pm
generic Cialis from India: tadalafil online no rx – tadalafil online no rx
KelvinHoosy
21 Jul 25 at 8:50 pm
pillow block buy belt tensioner rollers: Maintain optimal belt tension and smooth operation with our range of belt tensioner rollers. These rollers, designed to prevent slippage and reduce wear, extend belt life and enhance overall system efficiency. Choose from a variety of sizes and configurations to perfectly match your specific application requirements.
RobertPhasp
21 Jul 25 at 8:50 pm
Клиника «ВолгаМед» в Волгограде обеспечивает круглосуточную анонимную помощь людям, столкнувшимся с алкогольной или наркотической зависимостью. Мы предлагаем как стационарное лечение в комфортных условиях, так и выезд врача на дом в любое время суток. Конфиденциальность, профессионализм и индивидуальный подход — ключевые принципы нашей работы. В этой статье подробно рассмотрим преимущества клиники, спектр услуг, порядок организации домашнего выезда специалистов и дополнительные возможности поддержки в период реабилитации.
Получить больше информации – http://narkologicheskaya-klinika-volgograd13.ru/narkologicheskaya-klinika-klinika-pomoshh-volgograd/https://narkologicheskaya-klinika-volgograd13.ru
DustinTon
21 Jul 25 at 8:54 pm
What i don’t understood is if truth be told how you’re not actually a lot more well-appreciated than you may be now.
You’re so intelligent. You already know thus considerably in relation to this matter,
made me in my view imagine it from so many numerous angles.
Its like women and men are not interested except it is one thing to do
with Woman gaga! Your personal stuffs excellent.
At all times deal with it up!
pg77
21 Jul 25 at 8:54 pm
sertraline online: Zoloft for sale – cheap Zoloft
Leroymex
21 Jul 25 at 8:54 pm
После обращения по телефону или через сайт оператор уточняет адрес, состояние пациента и срочность выезда. Врач прибывает в течение 1–2 часов, а в экстренных случаях — в течение часа. На месте специалист проводит первичный осмотр: измеряет давление, пульс, сатурацию, температуру, уточняет длительность запоя, сопутствующие заболевания, особенности реакции на лекарства.
Выяснить больше – http://vyvod-iz-zapoya-himki5.ru/anonimnyj-vyvod-iz-zapoya-v-himkah/
BrandonRon
21 Jul 25 at 8:56 pm
lexapro tablets australia [url=https://lexapro.pro/#]lexapro 5 mg tablet price[/url] Lexapro for depression online
BurtonCix
21 Jul 25 at 8:57 pm
1win armenia [url=http://1win3072.ru/]http://1win3072.ru/[/url]
1win_wukr
21 Jul 25 at 8:58 pm
Врач назначает комплекс препаратов для нормализации состояния и устранения интоксикации. Применяются детоксикационные средства для очищения организма и облегчения абстинентного синдрома. Важным компонентом терапии являются витаминные комплексы и минералы, восстанавливающие метаболический баланс. При необходимости назначаются седативные препараты для снижения тревожности и нормализации сна. Также используются гепатопротекторы и другие средства для поддержки органов, пострадавших от алкогольной интоксикации.
Подробнее – [url=https://vyvod-iz-zapoya-krasnoyarsk55.ru/]вывод из запоя цена в красноярске[/url]
Williamsok
21 Jul 25 at 9:01 pm
карьерные стратегии презентация
JamesAllob
21 Jul 25 at 9:04 pm
https://lexapro.pro/# Lexapro for depression online
TheronSnipt
21 Jul 25 at 9:04 pm
Выбирая лечение в «Наркосфере», пациенты Балашихи и Балашихинского района могут быть уверены в высоком качестве услуг и современном подходе. Здесь применяют только сертифицированные препараты, используются новейшие методы терапии и реабилитации. Клиника работает круглосуточно, принимает пациентов в любой день недели и всегда готова к экстренному реагированию. Индивидуальный подход проявляется не только в подборе схемы лечения, но и в поддержке семьи на каждом этапе выздоровления. Весь процесс проходит в обстановке полной конфиденциальности: сведения о пациентах не передаются в сторонние организации, не оформляется наркологический учёт.
Подробнее тут – https://narkologicheskaya-klinika-balashiha5.ru/
Cletusjer
21 Jul 25 at 9:09 pm
подключить интернет в квартиру ростов
domashij-internet-rostov005.ru
домашний интернет ростов
internetrostovelini
21 Jul 25 at 9:18 pm
Добро пожаловать в Клубника Казино – место, где ваши азартные
мечты становятся реальностью.
В Клубника Казино вы найдете исключительный выбор слотов, настольных игр
и живых дилеров, готовых подарить вам незабываемые моменты.
Мы гарантируем безопасность, честность и прозрачность всех игровых процессов.
Почему Клубника турнир на деньги – это ваш идеальный выбор?
Мы предлагаем бесконечные бонусы, акции и турниры, которые дают шанс увеличить ваши выигрыши.
Кроме того, мы обеспечиваем быстрые
выплаты и 24/7 поддержку, чтобы
ваше время в казино было исключительно комфортным.
Когда стоит начать играть в Клубника Казино?
Не теряйте времени, начните прямо сейчас и
получите бонусы, которые ускорят ваш путь к победам.
В Клубника Казино вас ждут:
Не упустите шанс начать с бонусами и бесплатными вращениями, которые помогут вам сразу же
получить преимущество.
Турниры с крупными призами и невероятными возможностями для выигрыша.
Обновления игр каждый месяц,
чтобы не было скучно.
В Клубника Казино вы найдете все, что нужно для
отличной игры и крупных
выигрышей.
Клубника игры с живыми дилерами
21 Jul 25 at 9:25 pm
Excellent article. Keep posting such kind of information on your site. Im really impressed by your blog.
Hi there, You’ve done an incredible job. I will definitely digg it and in my view recommend to my friends. I’m confident they’ll be benefited from this web site.
https://novosti.ua/tech/vremennyj-nomer-telefona-v-ukraine-vozmozhnosti-i-princip-raboty
Fobertsax
21 Jul 25 at 9:26 pm
букмекерские конторы со слотами [url=https://1win3069.ru]букмекерские конторы со слотами[/url]
1win_hqkn
21 Jul 25 at 9:31 pm
вывод из запоя круглосуточно
vivod-iz-zapoya-kaluga004.ru
экстренный вывод из запоя
vivodkalugaNeT
21 Jul 25 at 9:35 pm
Следующий этап — реабилитация: индивидуальная и групповая психотерапия, восстановление эмоционального фона, обучение навыкам самоконтроля и профилактики рецидивов. При необходимости привлекаются дополнительные специалисты — психологи, психиатры, кардиологи. Важно, что в клинике «ВитаРеабилит» в Подольске большое внимание уделяется не только медицинской, но и психологической поддержке пациента и его семьи.
Детальнее – [url=https://narkologicheskaya-klinika-podolsk5.ru/]анонимная наркологическая клиника[/url]
Michaelneima
21 Jul 25 at 9:37 pm
buy ball bearings bearing manufacturer in Europe: Access a network of leading bearing manufacturers across Europe, renowned for their innovation, quality, and reliability. Explore a diverse range of bearing types, materials, and sizes to meet your specific application requirements. Partner with a trusted European manufacturer for superior bearing performance and long-lasting value.
RobertPhasp
21 Jul 25 at 9:39 pm
ai mental health app [url=https://www.mental-health22.com]https://www.mental-health22.com[/url] .
Mental Health_cbOn
21 Jul 25 at 9:40 pm
Услуга вызова нарколога на дом в Красноярске — это удобный и современный способ получения медицинской помощи. Она идеально подходит для тех, кто нуждается в профессиональной поддержке, но по каким-либо причинам не может или не хочет посещать клинику.
Углубиться в тему – [url=https://narcolog-na-dom-krasnoyarsk55.ru/]вызвать нарколога на дом красноярск[/url]
Matthewgam
21 Jul 25 at 9:50 pm
Everything is very open with a very clear description of
the challenges. It was definitely informative.
Your site is very useful. Thank you for sharing!
DirProxy
21 Jul 25 at 9:54 pm
blacksprut
блэкспрут
black sprut
блэк спрут
blacksprut вход
блэкспрут ссылка
blacksprut ссылка
blacksprut onion
блэкспрут сайт
блэкспрут вход
блэкспрут онион
блэкспрут дакрнет
blacksprut darknet
blacksprut сайт
блэкспрут зеркало
blacksprut зеркало
black sprout
blacksprut com зеркало
блэкспрут не работает
blacksprut зеркала
как зайти на blacksprut
blacksprut com зеркало
RichardPep
21 Jul 25 at 9:56 pm
Thank you, I’ve recently been looking for information approximately this topic for a while and yours is the best I’ve found out so far. But, what in regards to the conclusion? Are you sure in regards to the supply?
https://phylife.ru/kak-ispolzovat-virtualnyj-nomer-v-avstralii-dlya-biznes-proekta-v-sfere-uslug.html
IsmaelNek
21 Jul 25 at 9:58 pm
Magnificent goods from you, man. I’ve understand your stuff previous to and you are just extremely
excellent. I actually like what you have acquired here, certainly
like what you are stating and the way in which you say it.
You make it entertaining and you still take care of to keep it sensible.
I can not wait to read far more from you. This is actually a wonderful
website.
rtp koitoto
21 Jul 25 at 9:58 pm
Запой – это очень опасно, нужно срочно вызывать нарколога на дом в Волгограде! Помощь нарколога на дому – это оперативно, удобно и анонимно. Мы быстро очистим ваш организм от алкоголя и восстановим его работу. Наши специалисты готовы приехать к вам в любое время суток. Лечение комплексное: лекарства, индивидуальный подход и психологическая помощь. Первый шаг – детоксикация, мы поможем вам избавиться от алкогольной интоксикации.
Получить больше информации – [url=https://vyvod-iz-zapoya-volgograd00.ru/]вывод из запоя дешево волгоград[/url]
Charlesmew
21 Jul 25 at 9:59 pm
lexapro generic over the counter: lexapro pills for sale – lexapro online no prescription
Leroymex
21 Jul 25 at 10:02 pm
get generic propecia price [url=http://finasteridefromcanada.com/#]buying propecia for sale[/url] buy propecia
BurtonCix
21 Jul 25 at 10:04 pm
bearing manufacturer in Latvia bearing manufacturer in Europe: Access a network of leading bearing manufacturers across Europe, renowned for their innovation, quality, and reliability. Explore a diverse range of bearing types, materials, and sizes to meet your specific application requirements. Partner with a trusted European manufacturer for superior bearing performance and long-lasting value.
RobertPhasp
21 Jul 25 at 10:04 pm
hello!,I love your writing very a lot! proportion we keep
up a correspondence extra approximately your article on AOL?
I need a specialist on this space to resolve my problem.
Maybe that is you! Taking a look ahead to see you.
Click here for free money
21 Jul 25 at 10:05 pm
ai mental health app [url=http://mental-health22.com/]http://mental-health22.com/[/url] .
Mental Health_nxOn
21 Jul 25 at 10:10 pm
Hi to every one, it’s really a fastidious for me to visit this website, it consists of helpful Information.
https://womanka.com/dom/virtualnyj-nomer-telefona-preimushhestva-nedostatki-i-primenenie
LewisGuatt
21 Jul 25 at 10:14 pm
Врач-нарколог проводит комплексное обследование пациента, оценивая степень зависимости и общее состояние организма. В ходе первичного осмотра специалист не только определяет тяжесть алкогольной зависимости, но и выявляет сопутствующие патологии, которые могут повлиять на процесс лечения. Обследование включает лабораторные анализы, электрокардиограмму и другие необходимые диагностические процедуры. Врач тщательно изучает анамнез для выявления возможных противопоказаний к определенным методам терапии. Особое внимание уделяется психологическим аспектам, которые могли способствовать формированию зависимости, что помогает разработать эффективную программу лечения.
Получить больше информации – [url=https://vyvod-iz-zapoya-krasnoyarsk55.ru/]нарколог вывод из запоя красноярск[/url]
DannyFaurn
21 Jul 25 at 10:17 pm
Вызов врача-нарколога на дом в Санкт-Петербурге начинается с детального осмотра и оценки состояния пациента. Врач измеряет давление, пульс, уровень кислорода в крови и определяет степень интоксикации.
Выяснить больше – https://narcolog-na-dom-sankt-peterburg00.ru/narkolog-na-dom-czena-spb
Tomaswak
21 Jul 25 at 10:32 pm
tadalafil online no rx: Tadalafil From India – tadalafil 100mg best price
Leroymex
21 Jul 25 at 10:33 pm
Как правило, для эффективного устранения токсинов из организма и восстановления нормального функционирования всех органов и систем, назначаются специальные препараты. К таким препаратам относятся растворы для восстановления водно-солевого баланса, витамины и медикаменты для поддержания нормальной работы печени, а также препараты, стабилизирующие психоэмоциональное состояние пациента.
Выяснить больше – http://narcolog-na-dom-ektb55.ru/narkolog-na-dom-ekaterinburg-tseny/
RobertUnuck
21 Jul 25 at 10:33 pm
В условиях современного общества наркомания и алкоголизм стали актуальными и серьезными проблемами, затрагивающими не только индивидуумов, но и их семьи, сообщества. Эти заболевания не просто влияют на физическое здоровье, они наносят урон психоэмоциональному состоянию, нарушают социальные связи и ухудшают качество жизни. Наркологический центр “Новый горизонт” предлагает целостный и научно обоснованный подход к лечению зависимостей от психоактивных веществ. Мы применяем современные методы диагностики и терапии, чтобы помочь каждому пациенту обрести здоровье и вернуть себе полноценную жизнь.
Разобраться лучше – https://alko-specialist.ru
IsmaelLox
21 Jul 25 at 10:52 pm
https://finasteridefromcanada.shop/# generic Finasteride without prescription
TommyNex
21 Jul 25 at 10:57 pm
Hi there to every body, it’s my first pay a visit of this weblog; this weblog includes awesome and actually fine stuff for readers.
https://petelki.com.ua/2464-kak-ispolzovat-odnorazovyj-nomer-ukrainy-dlja-poluchenija-sms-i-registracii.html
EarnestAbent
21 Jul 25 at 11:01 pm
I was recommended this web site by my cousin.
I am not sure whether this post is written by him as no one else know such detailed about
my trouble. You are incredible! Thanks!
EagleSmartBot
21 Jul 25 at 11:07 pm
A fascinating discussion is dеfinitely worth comment. I belіeve thɑt you should write
more about this issue, it might not be a taboo matter but typіcally
people don’t discuss these subjects. To the next! Many thanks!!
Here is my web site; custom embroidery
custom embroidery
21 Jul 25 at 11:09 pm
Propecia for hair loss online: Finasteride From Canada – buying propecia pill
Leroymex
21 Jul 25 at 11:10 pm