Рамис Хамидуллин

Рамис Хамидуллин

С нами с 28 июня 2017; Место в рейтинге пользователей: #620
Anton
13 января 2022, 20:48
1
+6
Всем привет!

Я борюсь таким образом:
1. Проверяю в сниппете для FormIt с помощи регулярки на наличие ссылки типа site.com И email типа name@mail.com в поле текстового сообщения + в каждом поле такую проверку делаю (Имя, Телефон, Название фирмы, все поля чекаю). Если обнаружена ссылка, то возвращаю false и сообщение, что «Ссылки или электронные адреса в данном поле не разрешены. Вы можете приложить Word, Excel, PDF или текстовый файл с подробным описанием, ссылками и email.».
2. Использую сниппет csrfhelper:
<input type="hidden" name="csrf_token" value="[[!csrfhelper? &key=`contact-us`]]">
А в вызов FormIt добавляю хук csrfhelper_formit.
3. Разумеется, добавляю простейшую проверку на пустоту скрытое поле URL (можно назвать поле любым человекоподобным словом)
<input class="form-control visually-hidden" type="text" name="url" value="">
4. Добавляю таймер на страницу, который не разрешает отправлять форму, если ты менее 15 секунд на странице (обычно спам-боты отправляют спам моментально после загрузки страницы, менее чем за 1-5 секунд) и показываю пользователю сообщение об ошибке «Слишком быстро нажата кнопка Отправить. Подождите, пожалуйста, 15 секунд и отправьте еще раз.»
В форме Fenom-тэг с текущим временем в формате UNIX. Сравниваем его в антиспам-сниппете.
<input class="form-control visually-hidden" type="text" name="formtime" value="{time()}">
5. На всякий случай вырезаю все html-тэги из всех полей :stripTags
6. В доработке хук, который чекает прилагаемый файл через форму: только pdf, word, excel, txt, zip, png, jpg; не больше 25 мегабайт (вроде как стандартное ограничение всеми почтовиками). Если проверка не пройдена, то показываем вежливое сообщение, что мы принимаем только такие файлы и не больше 25 мб.
7. Не забываем про проверку на обязательность заполнения полей со стороны клиента с помощью тэга required. А также проставляем все необходимы типы данных в инпутах: type=«email», type=«number» или же type=«text» во всех полях!

Мое мнение — самое главное вернуть очень понятный человеку и вежливый ответ, что здесь ссылки не разрешены в целях борьбы со спамом, Вы можете приложить файл со всеми ссылками и адресами.

{$_modx->runSnippet('!FormIt', [

	'hooks' => 'antiSpam,spam,csrfhelper_formit,email',

	'spamCheckIp' => '1',
	'csrfKey' => 'contact-us',

	'validate' => 'name:stripTags:required,email:stripTags:required,message:stripTags:required,url:blank',

	'successMessage' => '<div class="alert alert-success mt-3" role="alert"><b>Спасибо!</b> Ваше сообщение было отправлено.</div>'

])}
Сам сниппет antiSpam:
<?php
$name = $hook->getValue('name');
$phone = $hook->getValue('phone');
$email = $hook->getValue('email');
$message = $hook->getValue('message');
$contact_attachment = $hook->getValue('contact_attachment');

$regExpGSM = '/(\d{3})(?:(?:-*|\s*|\.*|\)*|\)\s*))(\d{3})(?:(?:-*|\s*|\.*|\)*))(\d{3})/';
$regExpURL = '/[-a-zA-Z0-9@:%._\+~#=]{2,256}\.[a-z]{2,6}\b[-a-zA-Z0-9@:%_\+.~#?&=]*/i';
$regExpEmail = '/(?<=\b)\w([\w\.\-_0-9])*(@| at )[\w0-9][\w\-_0-9]*((\.| DOT )[\w\-_0-9]+)+(?=\b)/';
$regExpExt = '/^([A-Za-z0-9-_.])+\.(png|jpe?g|gif|bmp)$/';

/*Check regExp email*/
if (preg_match($regExpEmail, $name)) {
    $hook->addError('emailInName', 'Email-адреса не разрешены в поле Имя. Вы можете приложить Word, Excel, PDF или текстовый файл с подробным описанием, ссылками и электронной почтой.');
    return false;
} elseif (preg_match($regExpEmail, $message)) {
    $hook->addError('emailInMessage', 'Email-адреса не разрешены в поле Сообщения. Вы можете приложить Word, Excel, PDF или текстовый файл с подробным описанием, ссылками и электронной почтой.');
    return false;

    /*Check regExp URL's*/
} elseif (preg_match($regExpURL, $name)) {
    $hook->addError('linkInName', 'Ссылки на сайты не разрешены в поле Имя. Вы можете приложить Word, Excel, PDF или текстовый файл с подробным описанием, ссылками и электронной почтой.');
    return false;
} elseif (preg_match($regExpURL, $message)) {
    $hook->addError('linkInMessage', 'Ссылки на сайты не разрешены в поле Сообщения. Вы можете приложить Word, Excel, PDF или текстовый файл с подробным описанием, ссылками и электронной почтой.');
    return false;
	
/*И так далее по всем полям чекаю регулярками выше*/

/*Проверяем, что не слишком быстро нажимается кнопка "Отправить"*/
} elseif ($_POST['formtime'] > time() - 15) {
    $hook->addError('fastSending', 'Слишком быстро нажата кнопка Отправить. Подождите, пожалуйста, 15 секунд и отправьте еще раз.');
    return false;

} else {
    return true;
}
В самой форме плейсхолдеры для вывода человекопонятных ошибок:
[[!+fi.error.csrf_token:notempty=`
<div class="alert alert-warning mt-3 mb-2">[[!+fi.error.csrf_token]]</div>
`]]
[[!+fi.error.fastSending:notempty=`
<div class="alert alert-warning mt-3 mb-2">[[!+fi.error.fastSending]]</div>
`]]

[[+fi.statusMessage]]
[[+fi.successMessage]]

<input type="text" class="form-control"
       placeholder="Ваше имя *"
       name="name"
       value="[[!+fi.name]]"
       required>
<span class="error_name">[[+fi.error.name]]</span>
До этого у меня была всего-лишь одна проверка regExp на наличие ссылки и email в поле Сообщения (message). Спама почти не было, но стал приходить спам типа Buy our viagra on usa-pharma.shop, пропускались такие ссылки. Более длинная регулярка уже иногда блочила любой текст, если юзер пишет сообщение и после точки сразу пишет слово нового предложения.

Добавил проверку на 15 секундное ожидание (человеку просто не реально раньше 15й секунды заполнить все поля и отправить форму). Можно поставить 5 секунд.

После этого стали приходить сообщение только от человеков, никакого спама.

Не отловленные ошибки надо бы логгировать (у меня был этот кусок кода для логгирования не отправленных форм через FormIt, но не могу найти, он в старых хуках был).

Ну и можно по IP блочить вдобавок, как выше уже предложено.

Я не очень силен в PHP, как смог так и написал, поэтому большая просьба помочь хорошо доработать данный сниппет/хук и поделиться им здесь.
Даниил
26 февраля 2018, 13:42
1
0
Долго сражался с API. Все ему не слава богу. То половина ресурсов создается, а половина нет, то ТВ не заполняются, то вообще не работает ибо бубен шамаский не тот… В общем победил вот таким кодом. Делюсь, вдруг кому тож пригодится.

foreach($addArr as $key => $val){

    $rss = $modx->newObject('modResource');
    $rss->set('template', 17);                     // Назначаем ему нужный шаблон
    $rss->set('isfolder', 1);                         // Указываем, что это не контейнер   
    $rss->set('published', 1);                      // Неопубликован
    $rss->set('createdon', time());              // Время создания
    $rss->set('pagetitle', $val[0]);               // Заголовок
    $rss->set('alias', strtolower($val[0]));    // Псевдоним
    $rss->set('parent', $startID);                 // Родительский ресурс
    $rss->save(); 
    
    $nid = $rss->get('id');      //получаем ID созданного ресурса
    
    $tvcid = $modx->getObject('modTemplateVar',array('name'=>'cid'));       // Указываем имя нужного TV
    $tvcid->setValue($nid, $key);       // Записываем в TV значение
    $tvcid->save();

    unset($rss); unset($tvcid);       // Удаляем переменные
    $modx->cacheManager->clearCache();      // Кэш тоже чистим

}
Володя
22 июня 2017, 09:51
2
+1
с шаблонизатором все проще, можно писать прям в шаблоне все
{extends 'file:templates/category.tpl'}

{block 'initialize'}
    {parent}

    {var $tag = $_modx->getPlaceholder('vp.tag')}
    {var $_modx->resource.longtitle = $tag}

    {var $crumbs = 0}
    {var $parents = 0}
    {var $resources = '!msProducts'|snippet:[
    'parents' => 0,
    'optionFilters' => json_encode([
    'tags' => $tag
    ]),
    'returnIds' => 1
    ]}

    {if $resources is empty}
        {var $url = ('site_start'|option) | url : ['scheme' => 'full'] : []}
        {$_modx->sendRedirect($url, ['responseCode' => 'HTTP/1.1 404 Not Found'])}
    {/if}

{/block}
Leo
Leo
20 января 2017, 17:58
1
0
Здравствуйте!

1. Можно вывести ваш компонент в простом формате? типа — город (картинка состояния погоды) +-градусы

2. И как там с блокировками импорта данных? в примере из тикета о простом выводе погоды импорт блокирован яндексом
http://export.yandex.ru/weather-ng/forecasts/$cityId.xml
Володя
05 августа 2016, 22:35
21
+5
можно добавить так
создать плагин на pdoToolsOnFenomInit, в нем добавить модификатор detector
<?php
switch ($modx->event->name) {
    case 'pdoToolsOnFenomInit':
    if (!$fenom = $modx->getOption('fenom', $scriptProperties)) {
        return;
    }
    if (!$MobileDetect = $modx->getService('mobiledetect', 'MobileDetect', MODX_CORE_PATH . 'components/mobiledetect/')) {
	    return;
    }
    
    $key = $MobileDetect->config['force_browser_variable'];
    $device = !empty($_GET) && array_key_exists($key, $_GET)
		? $modx->stripTags($_GET[$key])
		: '';
	if (empty($device)) {
	    $device = $MobileDetect->getSettings();
    }
    if (empty($device)) {
        $detector = $MobileDetect->getDetector();
        $device = ($detector->isMobile() ? ($detector->isTablet() ? 'tablet' : 'mobile') : 'standard');
        $MobileDetect->saveSettings($device);
    }

    $fenom->addModifier("detector", function ($value) use ($device) {
        return $value == $device;
    });

    break;
}

и теперь в любом месте можно делать так
{if 'mobile'|detector}
mobile
{/if}

{if 'tablet'|detector}
tablet
{/if}

{if 'standard'|detector}
standard
{/if}
Воеводский Михаил
16 мая 2016, 17:01
1
0
Из бесплатного ничего адекватного не было найдено.
Сергей Шлоков
23 апреля 2016, 11:10
5
+3
Я для себя сделал такой плагин. Сильно не заморачивался. Просто скопировал код из Ace.
<?php
switch ($modx->event->name) {
    case 'OnDocFormPrerender':
        if (!$modx->controller->resourceArray) {
            return;
        }
        $field = 'modx-resource-introtext';
        $mimeType = $modx->getObject('modContentType', $modx->controller->resourceArray['content_type'])->get('mime_type');
        if ($mimeType == 'text/html' && $modx->getOption('pdotools_fenom_parser')) {
            $mimeType = 'text/x-smarty';
        }
		$modxTags = 1;
		$script = "MODx.ux.Ace.replaceComponent('$field', '$mimeType', $modxTags);";
		$script .= "MODx.ux.Ace.replaceTextAreas(Ext.query('.modx-richtext'));";
		$modx->controller->addHtml('<script>Ext.onReady(function() {' . $script . '});</script>');
        break;
    default:
        return;
}
Наумов Алексей
17 октября 2014, 15:46
4
+2
Пример с MIGx db, в общих чертах, код взят с одного из сайтов кусками, и для вашего сайта его надо будет переработать.

1. Настраиваем MIGx

1.1. Создаем таблицы в БД и объекты
— Ставим MIGx, как написано в документации к нему!
Я назвал этот «полукомпонент» xqa, все имена/префиксы делал с этим названием…
— Открываем MIGx в админке, и на первой вкладке создаем package xqa
— берем эту xml
<?xml version="1.0" encoding="UTF-8"?>
<model package="xqa" baseClass="xPDOObject" platform="mysql" defaultEngine="MyISAM" version="1.1">
	<object class="xQa" table="xqa" extends="xPDOSimpleObject" >
	<field key="resource_id" dbtype="int" precision="11" phptype="integer" null="false" default="0" />
	<field key="author" dbtype="varchar" precision="255" phptype="string" null="false" default="" />
	<field key="author_email" dbtype="varchar" precision="255" phptype="string" null="false" default="" />
	<field key="question" dbtype="text" phptype="string" />
	<field key="answer" dbtype="text" phptype="string" />
	<field key="notify" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="published" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="createdby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="createdon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="editedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="editedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="deleted" dbtype="tinyint" precision="1" attributes="unsigned" phptype="integer" null="false" default="0" />
	<field key="deletedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="deletedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<field key="publishedon" dbtype="datetime" phptype="datetime" null="false" />
	<field key="publishedby" dbtype="int" precision="10" phptype="integer" null="false" default="0" />
	<index alias="PRIMARY" name="PRIMARY" primary="true" unique="true">
		<column key="id" collation="A" null="false" />
	</index>
	<aggregate alias="Resource" class="modResource" local="resource_id" foreign="id" cardinality="one" owner="foreign" />
	</object>
</model>
и на вкладке xml schema ждем save, сохраняя ее.

— теперь создаем таблицы в БД на вкладке create Tables и создаем классы на вкладке parse Schema.

Таким образом мы получаем таблицу в БД и xpdo классы для работы с этой таблицей.

Структура моей таблицы:
resource_id - документ, к которому "привязан" вопрос
author - имя автора
author_email - его email
question - текст вопроса
answer - текст ответа
notify - галочка, символизирующая что нужно отправить уведомление автору о том, что не его вопрос появился ответ

---- с колонками ниже MIGx работает САМ, т.е. мы в коде ничего с ними не делаем, они сами заполняются
published
createdby
createdon
editedby
editedon
deleted
deletedon
deletedby
publishedon
publishedby

1.2 Создаем MIGx настройку.
Идем в MIGx — вкладка MIGx (вторая вкладка с настройками)

Делаем новую xqa и по ней правой кнопкой — Экспорт/Импорт, вставляем следующий код:
{
  "formtabs":[
    {
      "MIGX_id":1,
      "caption":"\u0412\u043e\u043f\u0440\u043e\u0441-\u043e\u0442\u0432\u0435\u0442",
      "print_before_tabs":"0",
      "fields":[
        {
          "MIGX_id":1,
          "field":"author",
          "caption":"\u0410\u0432\u0442\u043e\u0440",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":4,
          "field":"author_email",
          "caption":"E-mail",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":2,
          "field":"question",
          "caption":"\u0412\u043e\u043f\u0440\u043e\u0441",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"textarea",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":3,
          "field":"answer",
          "caption":"\u041e\u0442\u0432\u0435\u0442",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"textarea",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"",
          "default":""
        },
        {
          "MIGX_id":6,
          "field":"notify",
          "caption":"\u0423\u0432\u0435\u0434\u043e\u043c\u0438\u0442\u044c \u0430\u0432\u0442\u043e\u0440\u0430 \u043e\u0431 \u043e\u0442\u0432\u0435\u0442\u0435?",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"listbox",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"\u041d\u0435\u0442==||\u0414\u0430==1",
          "default":""
        },
        {
          "MIGX_id":5,
          "field":"published",
          "caption":"\u041e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d",
          "description":"",
          "description_is_code":"0",
          "inputTV":"",
          "inputTVtype":"listbox",
          "validation":"",
          "configs":"",
          "sourceFrom":"config",
          "sources":"[]",
          "inputOptionValues":"\u041d\u0435\u0442==||\u0414\u0430==1",
          "default":""
        }
      ]
    }
  ],
  "contextmenus":"update||publish||unpublish||recall_remove_delete",
  "actionbuttons":"addItem||toggletrash",
  "columnbuttons":"",
  "filters":"[]",
  "extended":{
    "migx_add":"\u0414\u043e\u0431\u0430\u0432\u0438\u0442\u044c \u0432\u043e\u043f\u0440\u043e\u0441",
    "formcaption":"",
    "update_win_title":"\u0420\u0435\u0434\u0430\u043a\u0442\u0438\u0440\u043e\u0432\u0430\u0442\u044c \u0432\u043e\u043f\u0440\u043e\u0441",
    "win_id":"xqa",
    "maxRecords":"",
    "addNewItemAt":"top",
    "multiple_formtabs":"",
    "extrahandlers":"",
    "packageName":"xqa",
    "classname":"xQa",
    "task":"",
    "getlistsort":"createdon",
    "getlistsortdir":"DESC",
    "use_custom_prefix":"0",
    "prefix":"",
    "grid":"",
    "gridload_mode":2,
    "check_resid":1,
    "check_resid_TV":"",
    "join_alias":"",
    "has_jointable":"yes",
    "getlistwhere":"",
    "joins":"",
    "cmpmaincaption":"",
    "cmptabcaption":"",
    "cmptabdescription":"",
    "cmptabcontroller":"",
    "winbuttons":"",
    "onsubmitsuccess":"",
    "submitparams":""
  },
  "columns":[
    {
      "MIGX_id":7,
      "header":"ID",
      "dataIndex":"id",
      "width":"",
      "sortable":"false",
      "show_in_grid":"0",
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":6,
      "header":"\u0414\u0430\u0442\u0430 \u0441\u043e\u0437\u0434\u0430\u043d\u0438\u044f",
      "dataIndex":"createdon",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"this.renderDate",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":1,
      "header":"\u0410\u0432\u0442\u043e\u0440",
      "dataIndex":"author",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":8,
      "header":"E-mail",
      "dataIndex":"author_email",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":2,
      "header":"\u0412\u043e\u043f\u0440\u043e\u0441",
      "dataIndex":"question",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":3,
      "header":"\u041e\u0442\u0432\u0435\u0442",
      "dataIndex":"answer",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":9,
      "header":"\u0423\u0432\u0435\u0434\u043e\u043c\u0438\u0442\u044c?",
      "dataIndex":"notify",
      "width":"",
      "sortable":"false",
      "show_in_grid":1,
      "renderer":"this.renderCrossTick",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":4,
      "header":"\u041e\u043f\u0443\u0431\u043b\u0438\u043a\u043e\u0432\u0430\u043d",
      "dataIndex":"published",
      "width":"",
      "sortable":true,
      "show_in_grid":1,
      "renderer":"this.renderCrossTick",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    },
    {
      "MIGX_id":5,
      "header":"\u0423\u0434\u0430\u043b\u0435\u043d",
      "dataIndex":"deleted",
      "width":"",
      "sortable":"false",
      "show_in_grid":"0",
      "renderer":"",
      "clickaction":"",
      "selectorconfig":"",
      "renderchunktpl":"",
      "renderoptions":"[]"
    }
  ]
}
Сохраняем и ради интереса жмем редактировать, смотрим какие поля и колонки у нас есть.

2. Делаем стандартную форму на Formit, в которой должны быть поля author, author_email, question. Как вы ее реализуете — ваши заботы.

Что важно для нас, у Formit указываем хук xQaAdd:
&hooks=`email,xQaAdd`
Хук email я тоже использую, что бы пришло администратору сайта уведомление «Добавлен новый вопрос...».

Код хука примерно такой, т.е. мы просто создаем новый объект xQa (вопрос):
$resource_id = $modx->resource->get('id');

$modx->addPackage('xqa', MODX_CORE_PATH.'components/xqa/model/');

$qa = $modx->newObject('xQa');
$qa->set('author', strip_tags($hook->getValue('author')));
$qa->set('author_email', strip_tags($hook->getValue('author_email')));
$qa->set('question', strip_tags($hook->getValue('question')));
$qa->set('resource_id', $resource_id);
$qa->set('published', 0);
$qa->set('createdby', 1);
$qa->set('createdon', strtotime("now"));
$qa->save();

return true;
3. Делаем TV поле с типом ввода migxdb, указываем Конфигурацию xqa (мы ее выше создавали), назначаем TV нужному шаблону.

У ресурса должна появится таблица:


4. Уведомление пользователя об ответе:
Создаем плагин notifyVisitors на событие OnDocFormSave:

<?php
if($modx->event->name != 'OnDocFormSave'){
  return false;
}

$template = $resource->get('template');

switch($template){
  case 7:
  case 15:
    // Вопрос-ответ
    $modx->addPackage('xqa', MODX_CORE_PATH.'components/xqa/model/');
    $qas = $modx->getCollection('xQa', array('deleted' => 0, 'notify' => 1, 'resource_id' => $resource->get('id')));
    foreach($qas as $qa){
      $to = $qa->get('author_email');
      if(!empty($to)){
	// ЗДЕСЬ отправляем письмо, объект $qa у нас есть
      }
      $qa->set('notify', 0);
      $qa->save();
    }
    break;
  default:
    return false;
}
Что делает плагин: при сохранении ресурса получаем все xQa, у которых стоит галочка «Оповестить автора об ответе», далее, например, отправляем письмо (я НЕ ПРИВОЖУ этот код, напишите его сами), ставим notify в 0.