Собрать все почтовые ящики в один. Как собрать письма из разных ящиков в одном месте. Как отдельно увидеть все письма с подключенной почты

По моему мнению, Gmail обладает одним из самых удобных веб интерфейсов для работы с электронной почтой. Помимо просто приятного и удобного интерфейса, создатели не забыли и о функциональной части. Цепочки писем, создание различных папок, фильтры и т.д. Все это сделало Gmail одним из самых популярных почтовых сервисов в мире. Многие не знают о такой замечательной возможности, как сборщик почты. Например у вас есть один или несколько почтовых ящиков на вашем доменном имени , но стандартный интерфейс webmail и почтовые клиенты вас не устраивают (Mozilla Thunderbird , Microsoft Outlook , The Bat ). В таком случае можно воспользоваться сборщиком почты и обрабатывать всю почту через Gmail, как это сделать мы сейчас рассмотрим.

  1. Необходим аккаунт в Gmail, процесс регистрации довольно прост и думаю и разобраться с этим не составит труда.
  2. После регистрации, в Gmail необходимо перейти в настройки
  3. В настройка выбираем пункт "Аккаунты и импорт"

  4. В этом разделе нас интересует два пункта: "Отправлять письма как:" и "Проверить почту из других аккаунтов (с помощью POP3):". В них есть ссылки "Добавить другой свой адрес электронной почты" и "Добавьте свой почтовый аккаунт POP3".

  5. Для того, что бы собирать почту с другого почтового ящика, нажимаем на "Добавьте свой почтовый аккаунт POP3". Появляется новое окно, в котором нужно ввести адрес почтового ящика и нажать "Далее". После появляется несколько полей, которые необходимо заполнить:
    Имя пользователя
    Пароль
    POP-сервер и порт — адрес POP сервера и порт для подключения к серверу, с которого будет собираться почта. Эту информацию вы можете узнать в поддержке вашего почтового сервиса, обычно он опубликован в разделе "Вопросы и ответы".
    Сохранять копии полученных писем на сервере — если галочка не будет установлена, после загрузки писем, эти письма будут удалены из почтового ящика. .
    Всегда использовать защищенное соединение (SSL) при получении писем — следует устанавливать, если ваш почтовый сервис для работы использует защищенное соединение SSL .
    Присвоить ярлык входящим сообщениям — ярлык, это визуальная отметка возле письма. Удобно использовать для понимания, на какой почтовый ящик было отправлено письмо.
    Архивировать входящие письма (пропустить входящие) — письма собранные с этого почтового ящика, будут архивироваться, во входящих их не будет.

  6. Нажимаем "Добавить аккаунт", после чего, Gmail отправляет проверочное письмо на указанный вами почтовый ящик. В письме на выбор будет два варианта проверки, ссылка и код. По ссылке нужно просто перейти, а код вставить в поле подтверждения. Проще перейти по ссылке, код видимо отправляется на всякий случай, вдруг ссылка будет отображена в письме некорректно.
  7. Ну вот собственно и все, теперь наш аккаунт на Gmail автоматически собирает письма с другого почтового ящика. Если есть необходимость отвечать с этого или другого адреса используя тот же Gmail, в настройках нажимаем на ссылку "Добавить другой свой адрес электронной почты" Заполняем поля:
    "Имя:" — можно вводить любые данные, например ваше имя.
    "Адрес электронной почты:" — указываем почтовый адрес, от имени которого мы будем отвечать.
    Галочка " Использовать как псевдоним." — этот адрес будет использован в качестве псевдонима вашего основного адреса и при отправке письма на него, оно будет отображено у вас во входящих, более подробно описано в справке google по ссылке "Подробнее...".
    Укажите другой адрес в поле "Ответить" (необязательно) — по умолчанию, ответ приходит на тот же адрес, который использовался для отправки. Нажав на ссылку, вы можете указать другой почтовый ящик, на который будут доставлять письма, отправленые на этот адрес.
    Нажимаем "Дальше"

  8. Далее мы видим еще несколько полей, которые необходимо заполнить.
    Сервер SMTP и Порт — эту информацию следует узнать у почтового сервиса.
    Имя пользователя — обычно это адрес электронно почты, это логин для доступа к почтовому ящику.
    Пароль — пароль доступа к почтовому ящику.
    Защищенное соединение с использованием TLS (рекомендуется) и Защищенное соединение с использованием SSL — второй вариант (SSL) можно выбирать только если ваш почтовый сервис использует этот тип защищенного соединения. В основном можно выбрать первый пункт.

    После нажатия "Добавить аккаунт", будет отправлено проверочное письмо, по аналогии с добавлением почтового ящика для сбора почты. Подтверждаем и все готово.

Ну собственно вот и все, таким образом можно настроить сборку писем и ответ из других ящиков через Gmail. Небольшой нюанс, график сбора почты из других почтовых ящиков Gmail формирует самостоятельно, отдельно для каждого такого почтового ящика. То есть отправленное письмо на почтовый ящик, с которого вы собираете почту, во входящих в интерфейсе Gmail отобразится не сразу, а только после проверки. Периодичность проверки зависит от количества и частоты поступления писем на этот почтовый ящик почтовый ящик, чем чаще поступают письма, тем чаще будет происходить проверка. Изменить этот параметр нельзя никак, но в настройках можно принудительно запустить проверку.

Разумеется, всё должно быть распараллелено. Тут на сцену выходит моя любимая библиотека TPL DataFlow .

Забирать почту будем по POP3. Все «модные штучки» IMAP в данной задаче излишни - надо как можно быстрее и проще забрать исходник письма и удалить его на сервера. POP3 тут хватит за глаза. Используем OpenPop.NET .

В качестве факультатива прикрутим мониторинг в Zabbix . (Мы же собрались работать 24/7 и выдавать хваленую скорость - нужно следить за этим).

Поехали

Создаем обычное консольное приложение. Открываем NuGet консоль и ставим все нужные пакеты:

Install-Package Nlog Install-Package OpenPop.NET Install-Package TopShelf Install-Package Microsoft.TPL.DataFlow
Переходим в папку проекта, создаем App.Debug.config и App.Release.config. Выгружаем проект из студии, открываем его код (Здесь и далее TopCrawler.csproj). В секцию с конфигом добавляем:

Конфигурации

App.config App.config


А ниже собственный таргет для MSBuild:

Transform target

$(TargetFileName).config


Лично я привык именно таким способом - по старинке - добавлять трансформацию конфигов для разделения сред.
Для удобства предлагаю strongly-type конфиги. Отдельный класс будет читать конфигурацию. (О теоретических аспектах такого решения можно пообщаться в комментах). Конфиги, логи, мониторинг - отличный повод реализовать паттерн Singleton.

Создаем в проекте одноименную папку (должен же быть порядок). Внутри создаем 3 класса - Config, Logger, Zabbix. Наш логгер:

Logger

static class Logger { public static NLog.Logger Log { get; private set; } public static NLog.Logger Archive { get; private set; } static Logger() { Log = LogManager.GetLogger("Global"); Archive = LogManager.GetLogger("Archivator"); } }


Мониторинг с помощью Zabbix заслуживает отдельного поста, поэтому я просто оставлю тут класс, реализующий агента:

Zabbix

namespace TopCrawler.Singleton { ///

/// Singleton: zabbix sender class /// static class Zabbix { public static ZabbixSender Sender { get; private set; } static Zabbix() { Sender = new ZabbixSender(Config.ZabbixServer, Config.ZabbixPort); } } struct ZabbixItem { public string Host; public string Key; public string Value; } class ZabbixSender { internal struct SendItem { // ReSharper disable InconsistentNaming - Zabbix is case sensitive public string host; public string key; public string value; public string clock; // ReSharper restore InconsistentNaming } #pragma warning disable 0649 internal struct ZabbixResponse { public string Response; public string Info; } #pragma warning restore 0649 #region --- Constants --- public const string DefaultHeader = "ZBXD\x01"; public const string SendRequest = "sender data"; public const int DefaultTimeout = 10000; #endregion #region --- Fields --- private readonly DateTime _dtUnixMinTime = DateTime.SpecifyKind(new DateTime(1970, 1, 1), DateTimeKind.Utc); private readonly int _timeout; private readonly string _zabbixserver; private readonly int _zabbixport; #endregion #region --- Constructors --- public ZabbixSender(string zabbixserver, int zabbixport) : this(zabbixserver, zabbixport, DefaultTimeout) { } public ZabbixSender(string zabbixserver, int zabbixport, int timeout) { _zabbixserver = zabbixserver; _zabbixport = zabbixport; _timeout = timeout; } #endregion #region --- Methods --- public string SendData(ZabbixItem itm) { return SendData(new List(1) { itm }); } public string SendData(List lstData) { try { var serializer = new JavaScriptSerializer(); var values = new List(lstData.Count); values.AddRange(lstData.Select(itm => new SendItem { host = itm.Host, key = itm.Key, value = itm.Value, clock = Math.Floor((DateTime.Now.ToUniversalTime() - _dtUnixMinTime).TotalSeconds).ToString(CultureInfo.InvariantCulture) })); var json = serializer.Serialize(new { request = SendRequest, data = values.ToArray() }); var header = Encoding.ASCII.GetBytes(DefaultHeader); var length = BitConverter.GetBytes((long)json.Length); var data = Encoding.ASCII.GetBytes(json); var packet = new byte; Buffer.BlockCopy(header, 0, packet, 0, header.Length); Buffer.BlockCopy(length, 0, packet, header.Length, length.Length); Buffer.BlockCopy(data, 0, packet, header.Length + length.Length, data.Length); using (var socket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp)) { socket.Connect(_zabbixserver, _zabbixport); socket.Send(packet); //Header var buffer = new byte; ReceivData(socket, buffer, 0, buffer.Length, _timeout); if (DefaultHeader != Encoding.ASCII.GetString(buffer, 0, buffer.Length)) throw new Exception("Invalid header"); //Message length buffer = new byte; ReceivData(socket, buffer, 0, buffer.Length, _timeout); var dataLength = BitConverter.ToInt32(buffer, 0); if (dataLength == 0) throw new Exception("Invalid data length"); //Message buffer = new byte; ReceivData(socket, buffer, 0, buffer.Length, _timeout); var response = serializer.Deserialize(Encoding.ASCII.GetString(buffer, 0, buffer.Length)); return string.Format("Response: {0}, Info: {1}", response.Response, response.Info); } } catch (Exception e) { return string.Format("Exception: {0}", e); } } private static void ReceivData(Socket pObjSocket, byte buffer, int offset, int size, int timeout) { var startTickCount = Environment.TickCount; var received = 0; do { if (Environment.TickCount > startTickCount + timeout) throw new TimeoutException(); try { received += pObjSocket.Receive(buffer, offset + received, size - received, SocketFlags.None); } catch (SocketException ex) { if (ex.SocketErrorCode == SocketError.WouldBlock || ex.SocketErrorCode == SocketError.IOPending || ex.SocketErrorCode == SocketError.NoBufferSpaceAvailable) Thread.Sleep(30); else throw; } } while (received < size); } #endregion } }


Конфиги… Пора уже делать хоть что-то интересное. Во-первых, в конфигах будем хранить ящики, которые мы опрашиваем. Во вторых настройки DataFlow. Предлагаю так:

Конфиги



Итак, хост и порт куда конектится, юзер и пароль - тут всё понятно. Дальше тип ящика. Допустим, служба используется маркетингом (как и другими отделами). У них есть ящики, куда сваливаются автоответы на рассылки, а также отчеты о спаме FBL . Сам ящик уже категоризирует письмо, поэтому для таких ситуаций сразу задаем тип ящика. С настройками DataFlow будет понятно дальше, когда начнем создавать объекты. Тут у нас будут собственные секции в конфиге. Мануалов куча как это сделать, поэтому просто покажу результат:

Определяем типы

#region --- Types --- static class MailboxType { public const string Bo = "bo"; public const string Crm = "crm"; public const string Fbl = "fbl"; public const string Bounce = "bounce"; } class MailboxInfo { public string Type { get; set; } public string Hostname { get; set; } public string User { get; set; } public string Password { get; set; } public int Port { get; set; } } class DataBlockOptions { public int Maxdop { get; set; } public int BoundedCapacity { get; set; } public DataBlockOptions() { Maxdop = 1; BoundedCapacity = 1; } } #endregion


Создаем секции

///

/// Custom config section /// public class CustomSettingsConfigSection: ConfigurationSection { public CredentialsCollection CredentialItems { get { return base["CredentialsList"] as CredentialsCollection; } } public DataBlockOptionsCollection DataFlowOptionsItems { get { return base["DataFlowOptionsList"] as DataBlockOptionsCollection; } } }


///

/// Custom collection - credentials list /// public class CredentialsCollection: ConfigurationElementCollection, IEnumerable { protected override ConfigurationElement CreateNewElement() { return new CredentialsElement(); } protected override object GetElementKey(ConfigurationElement element) { return ((CredentialsElement)element).Username; } public CredentialsElement this { get { return BaseGet(index) as CredentialsElement; } } public new IEnumerator < Count; i++) { yield return BaseGet(i) as CredentialsElement; } } } /// /// Custom credentials item /// public class CredentialsElement: ConfigurationElement { public string Hostname { get { return base["hostname"] as string; } } public string Username { get { return base["username"] as string; } } public string Password { get { return base["password"] as string; } } public string Type { get { return base["type"] as string; } } public string Port { get { return base["port"] as string; } } } /// /// Custom collection - DataBlock options list /// public class DataBlockOptionsCollection: ConfigurationElementCollection, IEnumerable { protected override ConfigurationElement CreateNewElement() { return new DataBlockOptionsElement(); } protected override object GetElementKey(ConfigurationElement element) { return ((DataBlockOptionsElement)element).Name; } public CredentialsElement this { get { return BaseGet(index) as CredentialsElement; } } public new IEnumerator GetEnumerator() { for (var i = 0; i < Count; i++) { yield return BaseGet(i) as DataBlockOptionsElement; } } } /// /// Custom DataBlock options item /// public class DataBlockOptionsElement: ConfigurationElement { public string Name { get { return base["name"] as string; } } public string Maxdop { get { return base["maxdop"] as string; } } public string BoundedCapacity { get { return base["boundedcapacity"] as string; } } }


Полную реализацию конфига писать не буду, подразумевается, что в процессе разработки туда будут добавляться нужные нам параметры.

Наши кастомные настройки прочитаем так:

Читаем

public List CredentialsList { get; private set; } public Dictionary DataFlowOptionsList { get; private set; } ... static Config() { try { var customConfig = (CustomSettingsConfigSection)ConfigurationManager.GetSection("CustomSettings"); //Get mailboxes foreach (var item in customConfig.CredentialItems) CredentialsList.Add(new MailboxInfo { Hostname = item.Hostname, Port = Convert.ToInt32(item.Port), User = item.Username, Type = item.Type, Password = item.Password }); //Get DataFlow settings foreach (var item in customConfig.DataFlowOptionsItems) DataFlowOptionsList.Add(item.Name, new DataBlockOptions { Maxdop = Convert.ToInt32(item.Maxdop), BoundedCapacity = Convert.ToInt32(item.BoundedCapacity) }); } catch (Exception ex) { Logger.Log.Fatal("Error at reading config: {0}", ex.Message); throw; } }


Как-то очень затянуто получается, а мы даже не дошли до самого интересного.

Опустим пока обвязку из TopShelf, счетчики производительности, общение с БД и перейдем к делу! Создаем класс Crawler - ядро. Для начала читаем почту:

Private volatile bool _stopPipeline; ... public void Start() { do { var getMailsTasks = _config.CredentialsList.Select(credentials => Task.Run(() => GetMails(credentials))).ToList(); foreach (var task in getMailsTasks) task.Wait(); Thread.Sleep(2000); } while (!_stopPipeline); //Stop pipeline - wait for completion of all endpoints //Тут будет остановка DataFlow конвейера if (_stopPipeline) Logger.Log.Warn("Pipeline has been stopped by user"); }
Вот тут лень взяла свое и я решил не заморачиваться - если ящиков порядка 20-30 можно под каждый запустить таск и не париться о количестве потоков. (Разрешаю закидать помидорами.)

Переходим к самому чтению:

Private void GetMails(MailboxInfo info) { try { using (var client = new Pop3Client()) {
Сразу посчитаем тайминги доступа к ящику - пригодится для диагностики сети и загруженности сервера.

//Get Zabbix metrics var stopwatch = new Stopwatch(); stopwatch.Start(); //Get mail count client.Connect(info.Hostname, info.Port, false); client.Authenticate(info.User, info.Password); stopwatch.Stop();
Отправляем данные в Zabbix. Всё просто - указываем имя хоста (как оно заведено в Zabbix), ключ (опять таки строго, как в Zabbix) и строковое значение.

//Send it to Zabbix Zabbix.Sender.SendData(new ZabbixItem { Host = Config.HostKey, Key = info.Type + Config.TimingKey, Value = stopwatch.ElapsedMilliseconds.ToString() }); Logger.Log.Debug("Send [{0}] timing to Zabbix: connected to "{1}" as "{2}", timing {3}ms", info.Type, info.Hostname, info.User, stopwatch.ElapsedMilliseconds); var count = client.GetMessageCount(); if (count == 0) return; Logger.Log.Debug("We"ve got new {0} messages in "{1}"", count, info.User); //Send messages to sorting block for (var i = 0; i < count; i++) { try { var mailInfo = new MessageInfo { IsSpam = false, Mail = client.GetMessage(i + 1), Type = MessageType.UNKNOWN, Subtype = null, Recipient = null, Mailbox = info }; Logger.Log.Debug("Download message from "{0}". Size: {1}b", info.User, mailInfo.Mail.RawMessage.Length);
DataFlow pipeline будет создана при создании класса Crawler. Считаем, что наш первый этап - отсортировать письмо.

While (!_sortMailDataBlock.Post(mailInfo)) Thread.Sleep(500);
Видите, как просто - сам конвейер один. Все таски, читающие почту, кидают туда сообщения по одному. Если блок занят, Post вернет false и мы просто подождем пока он не освободится. Текущий потом в это время продолжает работать. Вот это я называю параллелизм без забот.

Сообщение ушло на конвейер, теперь его можно со спокойной душой сохранить в RAW архив (да-да! всё, что читаем - сохраняем в файловый архив. Служба поддержки нам потом скажет спасибо).

Настроим, например, ротацию архива:

NLog.config



Потом на него можно натравить logStash , но это уже другая история…

//Save every mail to archive Logger.Log.Debug("Archive message"); Logger.Archive.Info(Functions.MessageToString(mailInfo.Mail)); } catch (Exception ex) { Logger.Log.Error("Parse email error: {0}", ex.Message); Functions.ErrorsCounters.Increment(); //Archive mail anyway Logger.Log.Debug("Archive message"); Logger.Archive.Info(Encoding.Default.GetString(client.GetMessageAsBytes(i + 1))); } if (_config.DeleteMail) client.DeleteMessage(i + 1); if (_stopPipeline) break; } Logger.Log.Debug("Done with "{0}"", info.User); } } catch (Exception ex) { Logger.Log.Error("General error - type: {0}, message: {1}", ex, ex.Message); Functions.ErrorsCounters.Increment(); } }
Здесь мы использовали статические счетчики ошибок (в разрезе типов ящиков), где ErrorsCounters - это:

Public static Dictionary ErrorsCounters = new Dictionary();
А сами счетчики можно сделать так:

Counter.cs

class Counter { private long _counter; public Counter() { _counter = 0; } public void Increment() { Interlocked.Increment(ref _counter); } public long Read() { return _counter; } public long Refresh() { return Interlocked.Exchange(ref _counter, 0); } public void Add(long value) { Interlocked.Add(ref _counter, value); } public void Set(long value) { Interlocked.Exchange(ref _counter, value); } }


Перейдем к созданию конвейера. Допустим, у нас есть ящики, куда сыпятся автоответы. Такие письма надо распарсить (что за автоответ, от кого, по какой рассылке и т.д.) и сложить результат в хранилище (БД). Допустим, есть ящики, куда падают FBL отчеты. Такие письма сразу складываем в базу. Все прочие письма считаем «полезными» - их надо проверить на спам и отправить во внешнюю систему, например, CRM.

Как вы уже поняли, данный пример в основном рассматривает применение сборщика для задач маркетинга - сбор статистики по доставке почты, информация о спаме.

Итак, мы определились с рабочим потоком. Объявляем необходимые блоки в классе Crawler:

Class MessageInfo { public bool IsSpam { get; set; } public Message Mail { get; set; } public string Subtype { get; set; } public string Recipient { get; set; } public MessageType Type { get; set; } public MailboxInfo Mailbox { get; set; } } class Crawler { //Pipeline private TransformBlock _sortMailDataBlock; private TransformBlock _spamFilterDataBlock; private TransformBlock _checkBounceDataBlock; private TransformBlock _identifyDataBlock; private ActionBlock _addToCrmDataBlock; private ActionBlock _addToFblDataBlock; private ActionBlock _addToBounceDataBlock; ...
Создаем метод инициализации и создаем блоки конвейера (для инициализации блоков используем наши замечательные секции из конфигов):

Public void Init() { //*** Create pipeline *** //Create TransformBlock to get message type var blockOptions = _config.GetDataBlockOptions("_sortMailDataBlock"); _sortMailDataBlock = new TransformBlock(mail => SortMail(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create TransformBlock to filter spam blockOptions = _config.GetDataBlockOptions("_spamFilterDataBlock"); _spamFilterDataBlock = new TransformBlock(mail => FilterSpam(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create TransformBlock to sort bounces blockOptions = _config.GetDataBlockOptions("_checkBounceDataBlock"); _checkBounceDataBlock = new TransformBlock(mail => BounceTypeCheck(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create TransformBlock to identify bounce owner blockOptions = _config.GetDataBlockOptions("_identifyDataBlock"); _identifyDataBlock = new TransformBlock(mail => GetRecipient(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create ActionBlock to send mail to CRM blockOptions = _config.GetDataBlockOptions("_addToCrmDataBlock"); _addToCrmDataBlock = new ActionBlock(mail => AddToCrm(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create ActionBlock to send FBL to MailWH blockOptions = _config.GetDataBlockOptions("_addToFblDataBlock"); _addToFblDataBlock = new ActionBlock(mail => AddToFbl(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity }); //Create ActionBlock to send Bounce to MailWH blockOptions = _config.GetDataBlockOptions("_addToBounceDataBlock"); _addToBounceDataBlock = new ActionBlock(mail => AddToBounce(mail), new ExecutionDataflowBlockOptions { MaxDegreeOfParallelism = blockOptions.Maxdop, BoundedCapacity = blockOptions.BoundedCapacity });
Собираем конвейер в соответствии с нашей схемой:

//*** Build pipeline *** _sortMailDataBlock.LinkTo(_spamFilterDataBlock, info => info.Type == MessageType.GENERAL); _sortMailDataBlock.LinkTo(_addToFblDataBlock, info => info.Type == MessageType.FBL); _sortMailDataBlock.LinkTo(_checkBounceDataBlock, info => info.Type == MessageType.BOUNCE); _sortMailDataBlock.LinkTo(DataflowBlock.NullTarget(), info => info.Type == MessageType.UNKNOWN); /*STUB*/ _checkBounceDataBlock.LinkTo(_identifyDataBlock); _identifyDataBlock.LinkTo(_addToBounceDataBlock); _spamFilterDataBlock.LinkTo(_addToCrmDataBlock, info => !info.IsSpam); _spamFilterDataBlock.LinkTo(DataflowBlock.NullTarget(), info => info.IsSpam); /*STUB*/
Как видим, всё предельно просто - связываем блок со следующим (с возможностью задания условия связи). Все блоки исполняются параллельно. Каждый блок имеет степень параллелизма и емкость (с помощью емкости можно регулировать очередь перед блоком, то есть блок сообщение принял, но еще не обрабатывает). Таким образом, можно задавать высокую степень параллелизма для «сложных» и долгих операций, как, например, парсинг содержимого письма.

Не буду описывать матчасть DataFlow, лучше всё прочесть в первоисточнике TPL DataFlow .

SortMailDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_spamFilterDataBlock).Fault(t.Exception); else _spamFilterDataBlock.Complete(); }); _sortMailDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_addToFblDataBlock).Fault(t.Exception); else _addToFblDataBlock.Complete(); }); _sortMailDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_checkBounceDataBlock).Fault(t.Exception); else _checkBounceDataBlock.Complete(); }); _spamFilterDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_addToCrmDataBlock).Fault(t.Exception); else _addToCrmDataBlock.Complete(); }); _checkBounceDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_identifyDataBlock).Fault(t.Exception); else _identifyDataBlock.Complete(); }); _identifyDataBlock.Completion.ContinueWith(t => { if (t.IsFaulted) ((IDataflowBlock)_addToBounceDataBlock).Fault(t.Exception); else _addToBounceDataBlock.Complete(); }); }
Всё, на самом деле конвейер уже работает, можно постить в него сообщения. Осталось только остановить его дополнив наш метод Start:

Start

public void Start() { do { var getMailsTasks = _config.CredentialsList.Select(credentials => Task.Run(() => GetMails(credentials))).ToList(); foreach (var task in getMailsTasks) task.Wait(); Thread.Sleep(2000); } while (!_stopPipeline); //Stop pipeline - wait for completion of all endpoints _sortMailDataBlock.Complete(); _addToCrmDataBlock.Completion.Wait(); _addToFblDataBlock.Completion.Wait(); _addToBounceDataBlock.Completion.Wait(); if (_stopPipeline) Logger.Log.Warn("Pipeline has been stopped by user"); }


Переходим к делегатам.
Сортировка… Ну, допустим у нас всё просто (усложнить то всегда успеем):

Private MessageInfo SortMail(MessageInfo mail) { switch (mail.Mailbox.Type) { case MailboxType.Crm: mail.Type = MessageType.GENERAL; break; case MailboxType.Bounce: mail.Type = MessageType.BOUNCE; break; case MailboxType.Fbl: mail.Type = MessageType.FBL; break; } return mail; }
Спам фильтр. Это на домашнюю работу - используйте SpamAssassin .
Вот вам делегат:

Private MessageInfo FilterSpam(MessageInfo mail) { //TODO: Add SpamAssassin logic return mail; }
И классы для работы с API SpamAssassin (ссылка на проект).
А мы переходим к парсингу писем. Парсим мы автоответы. Тут вступает в дело MEF.
Создаем проект (dll) с интерфейсами для наших плагинов (Назовем Interfaces).
Добавляем интерфейс:

Public interface ICondition { string Check(Message mimeMessage); } public interface IConditionMetadata { Type Type { get; } }
И… всё. Наш TopCrawler зависит от этого проекта и проект с плагинами тоже будет использовать его.
Создаем новый проект (тоже dll), назовем Conditions.
Добавим типы автоответов:

#region --- Types --- static class BounceType { public const string Full = "BounceTypeFull"; public const string Timeout = "BounceTypeTimeout"; public const string Refused = "BounceTypeRefused"; public const string NotFound = "BounceTypeNotFound"; public const string Inactive = "BounceTypeInactive"; public const string OutOfOffice = "BounceTypeOutOfOffice"; public const string HostNotFound = "BounceTypeHostNotFound"; public const string NotAuthorized = "BounceTypeNotAuthorized"; public const string ManyConnections = "BounceTypeManyConnections"; } #endregion
И классы, реализующие наш интерфейс:

Public class ConditionNotFound1: ICondition { public string Check(Message mimeMessage) { if (!mimeMessage.MessagePart.IsMultiPart) return null; const string pattern = "Diagnostic-Code:.+smtp.+550"; var regexp = new Regex(pattern, RegexOptions.IgnoreCase); return mimeMessage.MessagePart.MessageParts.Any(part => part.ContentType.MediaType == "message/delivery-status" && regexp.IsMatch(part.GetBodyAsText())) ? BounceType.NotFound: null; } } ... public class ConditionTimeout2: ICondition { return BounceType.Timeout; } ...
Как вы заметилиб всё дело в атрибутах. С помощью них плагины и будут загружены.
Возвращаемся к нашему проекту и загружаем плагины:

Class Crawler { ... //Plugins public IEnumerable> BounceTypeConditions { get; set; } private void LoadPlugins() { try { var container = new CompositionContainer(new DirectoryCatalog(_config.PluginDirectory), true); container.ComposeParts(this); } catch (Exception ex) { Logger.Log.Error("Unable to load plugins: {0}", ex.Message); } } ...
LoadPlugins дергаем в конструкторе нашего класса. Объяснять подробно про механизм загрузки не буду - гугл справится лучше.

Переходим к нашему делегату проверки типа Bounce. Условия будут применяться по очереди, пока не сработает первое - исключающий метод:

Private MessageInfo BounceTypeCheck(MessageInfo mailInfo) { try { foreach (var condition in BounceTypeConditions) { var res = condition.Value.Check(mailInfo.Mail); if (res == null) continue; mailInfo.Subtype = res; Logger.Log.Debug("Bounce type condition [{0}] triggered for message [{1}]", condition.Metadata.Type, mailInfo.Mail.Headers.MessageId); break; } } catch (Exception ex) { Logger.Log.Error("Failed to determine bounce type for message "{0}": {1}", mailInfo.Mail.Headers.MessageId, ex.Message); Logger.ErrorsCounters.Increment(); } return mailInfo; }
Таким образомб если появляется новая логикаб достаточно просто добавить в проект с плагинами новый класс, реализующий наш интерфейс и - вуаля! Пример второго плагина по определению отправителя письма прикладывать не буду - итак уже длинный пост (Автоответ сгенерировал сам сервер, поэтому отправителя тоже надо распарсить из заголовков письма) .

С записью результатов в БД тоже ничего необычного. Например, так:

Private void AddToBounce(MessageInfo mail) { try { MailWH.BounceAdd(mail); Functions.ProcessedCounters.Increment(); Functions.Log.Debug("Send Bounce to MailWH"); } catch (Exception ex) { Functions.Log.Error("Error saving Bounce message "{0}" to MailWH: {1}", mail.Mail.Headers.MessageId, ex.Message); Functions.ErrorsCounters.Increment(); } }

BounceAdd

public static long BounceAdd(MessageInfo message) { using (var conn = new SqlConnection(ConnectionString)) using (var cmd = new SqlDataAdapter("BounceAdd", conn)) { var body = message.Mail.FindFirstPlainTextVersion() == null ? message.Mail.FindFirstHtmlVersion().GetBodyAsText() : message.Mail.FindFirstPlainTextVersion().GetBodyAsText(); var outId = new SqlParameter("@ID", SqlDbType.BigInt) { Direction = ParameterDirection.Output }; cmd.SelectCommand.CommandType = CommandType.StoredProcedure; cmd.SelectCommand.Parameters.Add(new SqlParameter("@RawMessage", message.Mail.RawMessage)); cmd.SelectCommand.Parameters.Add(new SqlParameter("@Message", body)); cmd.SelectCommand.Parameters.Add(new SqlParameter("@Subject", message.Mail.Headers.Subject ?? "")); cmd.SelectCommand.Parameters.Add(new SqlParameter("@MessageID", message.Mail.Headers.MessageId ?? "")); cmd.SelectCommand.Parameters.Add(new SqlParameter("@AddressTo", message.Mail.Headers.To.Address ?? "")); cmd.SelectCommand.Parameters.Add(new SqlParameter("@AddressFrom", message.Mail.Headers.From.Address ?? "")); cmd.SelectCommand.Parameters.Add(new SqlParameter("@DateRecieved", DateTime.Now)); cmd.SelectCommand.Parameters.Add(new SqlParameter("@BounceTypeSysName", (object)message.Subtype ?? DBNull.Value)); cmd.SelectCommand.Parameters.Add(new SqlParameter("@SourceFrom", (object)message.Recipient ?? DBNull.Value)); // TODO: Add ListId support cmd.SelectCommand.Parameters.Add(new SqlParameter("@ListId", DBNull.Value)); cmd.SelectCommand.Parameters.Add(outId); conn.Open(); cmd.SelectCommand.ExecuteNonQuery(); return outId.Value as long? ?? 0; } }


Простите, что не успел показать TopShelf - пост и так уже слишком раздулся.

Выводы

В этом уроке мы узнали, что задача сбора почты может оказаться не такой простой. Разработанное ядро позволяет быстро добавлять новые шаги процесса - DataFlow-блоки, не затрагивая существующую логику. Подсистема плагинов позволяет быстро наращивать скриптоподобную логику парсинга, а сам DataFlow распараллеливает все вычисления (причем мы имеем возможность гибко настраивать многопоточность под конкретную машину). TopShelf дает нам возможность запускать сервис как в режиме службы, так и в консольном режиме для облегчения отладки.

Фух… Если будет интересно, дальше расскажу, как поставить это на рельсы Continious Integration, настроить автобилды и выпуск релиза через VS Release Management .

Теги: Добавить метки

Если автоматически настроить соединение сборщика с сервером невозможно

Первый сбор почты произойдет в течение нескольких минут после того, как вы настроите сборщик. Сборщик обращается к другому почтовому серверу от нескольких раз в час до раза в несколько дней. Вручную установить или изменить время сбора нельзя. Оно зависит от используемого протокола (POP3 или IMAP), а также от особенностей работы сервера, с которого собирается почта. Именно поэтому письма могут приходить с задержкой.

Собрать почту с другого ящика на Яндексе

Собирать новые письма

Если вы хотите собирать почту с другого вашего ящика на Яндексе, вы можете настроить в нем правило пересылки . После подтверждения правила все новые письма будут перенаправляться в ваш текущий ящик.

Забрать старую почту

Чтобы собрать старые письма с другого ящика на Яндексе:

Выбрать папку для сбора писем

    Подтвердите удаление.

Отредактировать настройки сборщика

Чтобы отключить сборщик, перейдите на страницу Сбор почты и нажмите ссылку выкл .

Если сборщик больше не нужен, нажмите на название сборщика, а затем на ссылку Удалить .

Написать письмо от имени подключенного ящика

Если вы хотите, чтобы письмо для получателя пришло с вашего адреса электронной почты на другом сервисе (например, Gmail или Mail.Ru), добавьте этот адрес на странице Адреса электронной почты и подтвердите доступ к нему. После подтверждения вы сможете выбрать в поле От кого не только алиас вашего адреса на Яндексе , но и адрес почты на другом сервисе.

Внимание. Если вы добавляете адрес ящика на другом почтовом сервисе, настройте сбор писем с него в ящик на Яндексе. Без настроенного сборщика отправлять письма с такого адреса не получится. При настройке сборщика вместо логина укажите адрес полностью, например [email protected] .

Если даже при корректно настроенном сборщике у вас не получается отправлять письма с адреса на другом почтовом сервисе, это значит, что правила безопасности этого сервиса запрещают отправлять письма через другие почтовые сервисы, в том числе и через Яндекс.Почту.

Что можно делать с собранными письмами

С собранными письмами можно делать все то же самое, что и с другими письмами в ящике, но есть особенности:

Удаление писем

Если удалить собранное письмо, оно удалится только из того ящика, куда было собрано. В исходном ящике письмо останется.

Правила обработки писем

Правила обработки работают только для писем, собранных в папку Входящие . Если письмо собрано в другую папку, правило для него не сработает.

Черный список

Черный список не работает для собранных с другого вашего ящика писем. Чтобы заблокировать такие письма, перейдите в почтовый ящик, с которого собираются письма, и в нем настройте черный список.

Не работает сбор писем

Это пошаговое руководство поможет вам решить проблемы, связанные с работой сборщика писем. Прежде чем приступить, откройте Яндекс.Почту в новой вкладке , чтобы выполнять рекомендации.

Выберите проблему:

Уточните, сборщик раньше собирал письма?

    Отключите сборщик почты: перейдите на страницу Сбор почты и нажмите ссылку выкл .

    Подождите 2-3 часа.

    Включите сборщик обратно.

Смените пароль в почтовом ящике, с которого собираются письма. После\\n этого подождите 2-3 часа - столько времени требуется, чтобы новый пароль\\n точно заработал.\\n

    \\n \\n

    Нажмите на название сборщика в списке подключенных ящиков, а затем\\n нажмите ссылку Настроить .

    \\n \\n

    В Параметрах соединения введите новый\\n пароль.

    \\n \\n
\\n \\n \\n Письма не собираются с ящика на Gmail \\n \\n
    \\n

    На странице Безопасность и вход установите переключатель\\n Ненадежные приложения в положение Ненадежные\\n приложения разрешены .

    \\n \\n \\n
\\n

Подключите сборщик почты заново, уже с новым паролем:

    \\n \\n

    Нажмите на название сборщика в списке подключенных ящиков, а затем нажмите ссылку\\n Настроить .

    \\n \\n

    В Параметрах соединения введите новый пароль.

    \\n \\n
\\n \\n \\n \\n\\n

Проверьте, что в настройках вашего почтового ящика на Яндексе включен доступ\\\\n по протоколам POP3 или IMAP:

\\\\n

Убедитесь, что в настройках почтового ящика, с которого собираются письма,\\\\n активирован доступ по протоколам POP3 или IMAP.

\\\\n \\\\n \\\\n

Rambler

\\\\n \\\\n \\\\n \\\\n

Gmail

    \\\\n
  1. Пересылки и\\\\n POP/IMAP .

    \\\\n

    В блоке Доступ по протоколу POP (пункт\\\\n Состояние ) установите переключатель в положение\\\\n Включить POP для всех писем (даже для тех, которые уже\\\\n загружены) .

    \\\\n

    В блоке Доступ по протоколу IMAP (пункт\\\\n Состояние ) установите переключатель в положение\\\\n Включить IMAP .

    \\\\n \\\\n
\\\\n

\\\\n \\\\n \\\\n \\\\n

Mail.ru

В почтовом сервисе Mail.ru доступ к протоколам IMAP и POP3 активирован по\\\\n умолчанию.

\\\\n \\\\n \\\\n \\\\n

Outlook.com (hotmail.com)

Почтовый сервис Outlook.com может передавать письма только по протоколу POP. Чтобы\\\\n активировать доступ по протоколу POP:

\\\\n \\\\n \\\\n \\\\n

Yahoo

Почтовый сервис Yahoo может передавать письма только по протоколу IMAP. Этот протокол\\\\n включен по умолчанию.

\\\\n \\\\n \\\\n \\\\n\\\\n

Если вашего почтового сервиса нет в списке, обратитесь за информацией в службу\\\\n поддержки этого сервиса.

Возможно, возникли проблемы на том почтовом сервисе, с\\\\\\\\n ящика на котором собираются письма. Чтобы решить проблему,\\\\\\\\n настройте пересылку с того почтового ящика, с которого\\\\\\\\n собираются письма. Перед настройкой пересылки отключите\\\\\\\\n сборщик на Яндексе - иначе письма будут приходить в\\\\\\\\n двойном экземпляре.\\\\\\\\n

Чтобы узнать, как настроить пересылку писем с другого\\\\\\\\n почтового ящика в ящик на Яндексе, обратитесь к разделу\\\\\\\\n помощи другого почтового сервиса.

\\\\\\\\n "}]}}\\\\">

Возможно, возникли проблемы на том почтовом сервисе, с\\\\n ящика на котором собираются письма. Чтобы решить проблему,\\\\n настройте пересылку с того почтового ящика, с которого\\\\n собираются письма. Перед настройкой пересылки отключите\\\\n сборщик на Яндексе - иначе письма будут приходить в\\\\n двойном экземпляре.


\\\\n \\\\n "}]}}\\">
    \\n \\n

    \\n \\n \\n

    Rambler

    \\n \\n \\n \\n

    Gmail

      \\n
    1. Перейдите в настройки ящика Gmail, а затем на вкладку Пересылки и\\n POP/IMAP .

      \\n

      В блоке Доступ по протоколу POP .

      \\n

      В блоке Доступ по протоколу IMAP

      \\n \\n
    \\n

    \\n \\n \\n \\n

    Mail.ru

    \\n \\n \\n \\n

    Outlook.com (hotmail.com)

    \\n \\n \\n \\n

    Yahoo

    \\n \\n \\n \\n\\n

    Чтобы узнать, как настроить пересылку писем с другого\\\\n почтового ящика в ящик на Яндексе, обратитесь к разделу\\\\n помощи другого почтового сервиса.

    \\\\n "}]}}\\">


    \\n
\\n

\\n\\n "}]}}\">

Смените пароль в почтовом ящике, с которого собираются письма. После этого подождите 2-3 часа - столько времени требуется, чтобы новый пароль точно заработал.

Подключите сборщик почты заново, уже с новым паролем:

    В Параметрах соединения введите новый пароль.

Письма не собираются с ящика на Gmail

Если вы настроили сбор писем с ящика на Gmail:

    На странице Безопасность и вход установите переключатель Ненадежные приложения в положение Ненадежные приложения разрешены .

Подключите сборщик почты заново, уже с новым паролем:

    Нажмите на название сборщика в списке подключенных ящиков, а затем нажмите ссылку Настроить .

    В Параметрах соединения введите новый пароль.

Проверьте, что в настройках вашего почтового ящика на Яндексе включен доступ\\n по протоколам POP3 или IMAP:

\\n

Убедитесь, что в настройках почтового ящика, с которого собираются письма,\\n активирован доступ по протоколам POP3 или IMAP.

\\n \\n \\n

Rambler

\\n \\n \\n \\n

Gmail

    \\n
  1. Перейдите в настройки ящика Gmail, а затем на вкладку Пересылки и\\n POP/IMAP .

    \\n

    В блоке Доступ по протоколу POP (пункт\\n Состояние ) установите переключатель в положение\\n Включить POP для всех писем (даже для тех, которые уже\\n загружены) .

    \\n

    В блоке Доступ по протоколу IMAP (пункт\\n Состояние ) установите переключатель в положение\\n Включить IMAP .

    \\n \\n
\\n

\\n \\n \\n \\n

Mail.ru

В почтовом сервисе Mail.ru доступ к протоколам IMAP и POP3 активирован по\\n умолчанию.

\\n \\n \\n \\n

Outlook.com (hotmail.com)

Почтовый сервис Outlook.com может передавать письма только по протоколу POP. Чтобы\\n активировать доступ по протоколу POP:

\\n \\n \\n \\n

Yahoo

Почтовый сервис Yahoo может передавать письма только по протоколу IMAP. Этот протокол\\n включен по умолчанию.

\\n \\n \\n \\n\\n

Если вашего почтового сервиса нет в списке, обратитесь за информацией в службу\\n поддержки этого сервиса.

Возможно, возникли проблемы на том почтовом сервисе, с\\\\n ящика на котором собираются письма. Чтобы решить проблему,\\\\n настройте пересылку с того почтового ящика, с которого\\\\n собираются письма. Перед настройкой пересылки отключите\\\\n сборщик на Яндексе - иначе письма будут приходить в\\\\n двойном экземпляре.\\\\n

Чтобы узнать, как настроить пересылку писем с другого\\\\n почтового ящика в ящик на Яндексе, обратитесь к разделу\\\\n помощи другого почтового сервиса.

\\\\n "}]}}\\">

Возможно, возникли проблемы на том почтовом сервисе, с\\n ящика на котором собираются письма. Чтобы решить проблему,\\n настройте пересылку с того почтового ящика, с которого\\n собираются письма. Перед настройкой пересылки отключите\\n сборщик на Яндексе - иначе письма будут приходить в\\n двойном экземпляре.

Чтобы узнать, как настроить пересылку писем с другого\\n почтового ящика в ящик на Яндексе, обратитесь к разделу\\n помощи другого почтового сервиса.


\\n \\n "}]}}\">

    Rambler

    Gmail

    1. Перейдите в настройки ящика Gmail, а затем на вкладку Пересылки и POP/IMAP .

      В блоке Доступ по протоколу POP .

      В блоке Доступ по протоколу IMAP

    Mail.ru

    Outlook.com (hotmail.com)

    Yahoo

    Возможно, возникли проблемы на том почтовом сервисе, с\\n ящика на котором собираются письма. Чтобы решить проблему,\\n настройте пересылку с того почтового ящика, с которого\\n собираются письма. Перед настройкой пересылки отключите\\n сборщик на Яндексе - иначе письма будут приходить в\\n двойном экземпляре.\\n

    Чтобы узнать, как настроить пересылку писем с другого\\n почтового ящика в ящик на Яндексе, обратитесь к разделу\\n помощи другого почтового сервиса.

    \\n "}]}}\">

    Возможно, возникли проблемы на том почтовом сервисе, с ящика на котором собираются письма. Чтобы решить проблему, настройте пересылку с того почтового ящика, с которого собираются письма. Перед настройкой пересылки отключите сборщик на Яндексе - иначе письма будут приходить в двойном экземпляре.

    Чтобы узнать, как настроить пересылку писем с другого почтового ящика в ящик на Яндексе, обратитесь к разделу помощи другого почтового сервиса.




    Проверьте, что в настройках вашего почтового ящика на Яндексе включен доступ по протоколам POP3 или IMAP:

    Убедитесь, что в настройках почтового ящика, с которого собираются письма, активирован доступ по протоколам POP3 или IMAP.

    Rambler

    Gmail

    1. Перейдите в настройки ящика Gmail, а затем на вкладку Пересылки и POP/IMAP .

      В блоке Доступ по протоколу POP (пункт Состояние ) установите переключатель в положение Включить POP для всех писем (даже для тех, которые уже загружены) .

      В блоке Доступ по протоколу IMAP (пункт Состояние ) установите переключатель в положение Включить IMAP .

    Mail.ru

    В почтовом сервисе Mail.ru доступ к протоколам IMAP и POP3 активирован по умолчанию.

    Outlook.com (hotmail.com)

    Почтовый сервис Outlook.com может передавать письма только по протоколу POP. Чтобы активировать доступ по протоколу POP:

    Yahoo

    Почтовый сервис Yahoo может передавать письма только по протоколу IMAP. Этот протокол включен по умолчанию.

    Если вашего почтового сервиса нет в списке, обратитесь за информацией в службу поддержки этого сервиса.

    Проверьте, не установлен ли в настройках ящика запрет на скачивание\\n писем из конкретных папок или писем за конкретный период.\\n

    Почтовая служба Outlook.com (hotmail.com, live.com) разрешает\\n пересылать письма только из папки Входящие .\\n Если вы хранили письма в другой папке, переместите их в папку\\n Входящие .

    \\n "}]}}\">

    Проверьте, не установлен ли в настройках ящика запрет на скачивание писем из конкретных папок или писем за конкретный период.

    Почтовая служба Outlook.com (hotmail.com, live.com) разрешает пересылать письма только из папки Входящие . Если вы хранили письма в другой папке, переместите их в папку Входящие .


    Проверьте, что в настройках вашего почтового ящика на Яндексе включен доступ по протоколам POP3 или IMAP:

    Убедитесь, что в настройках почтового ящика, с которого собираются письма, активирован доступ по протоколам POP3 или IMAP.

    Rambler

    Gmail

    1. Перейдите в настройки ящика Gmail, а затем на вкладку Пересылки и POP/IMAP .

      В блоке Доступ по протоколу POP (пункт Состояние ) установите переключатель в положение Включить POP для всех писем (даже для тех, которые уже загружены) .

      В блоке Доступ по протоколу IMAP (пункт Состояние ) установите переключатель в положение Включить IMAP .

    Mail.ru

    В почтовом сервисе Mail.ru доступ к протоколам IMAP и POP3 активирован по умолчанию.

    Outlook.com (hotmail.com)

    Почтовый сервис Outlook.com может передавать письма только по протоколу POP. Чтобы активировать доступ по протоколу POP:

    Yahoo

    Почтовый сервис Yahoo может передавать письма только по протоколу IMAP. Этот протокол включен по умолчанию.

    Если вашего почтового сервиса нет в списке, обратитесь за информацией в службу поддержки этого сервиса.

    Проверьте правильность вводимого пароля - наберите его в текстовом\\n редакторе, скопируйте и вставьте в поле ввода пароля. Если вы уверены, что\\n пароль верный, но сборщик не подключается, попробуйте авторизоваться в\\n ящике, с которого хотите собирать письма.\\n "}]}}\">

    Проверьте правильность вводимого пароля - наберите его в текстовом редакторе, скопируйте и вставьте в поле ввода пароля. Если вы уверены, что пароль верный, но сборщик не подключается, попробуйте авторизоваться в ящике, с которого хотите собирать письма.


Скорость работы сборщика зависит от количества писем в том ящике, из которого собираются письма, и от работы сервера, на котором находится этот ящик. Задержки в сборе писем могут также происходить из-за особенностей протоколов, по которым работает сборщик.

Чтобы получать письма мгновенно, настройте переадресацию из другого ящика в ящик на Яндексе. Перед настройкой переадресации отключите сборщик - иначе письма будут приходить в двойном экземпляре.

Выбрать папку для сбора писем нельзя в том случае, если вы настроили сборщик по протоколу IMAP - он не предусматривает возможности собирать письма в заданную папку, т.к. копирует структуру папок полностью. Вы можете изменить протокол на POP3, для этого:

    Нажмите на адрес нужного сборщика.

    Подтвердите удаление.

После этого в настройках появится возможность выбрать папку для сбора писем.

В настоящее время большинство пользователей имеют несколько почтовых ящиков для электронной почты, расположенных на разных серверах (например на mail.ru или yandex.ru). И, чтобы проверить почту, приходиться заходить на каждый почтовый ящик, вводить логин и пароль. Это требует определённых временных затрат, тем более что не все ящики для электронной почты одинаково значимы. Поэтому возникает потребность объединения ящиков для электронной почты, расположенных даже на разных серверах, в один более важный, чтобы можно было проверить всю почту, заходя только в один ящик. Рассмотрим данную возможность на примере mail.ru, объединим несколько ящиков для электронной почты в один.

Для того, чтобы собрать всю почту в один ящик на mail.ru необходимо:

  • Зайти на свою почту mail.ru и нажать "Настройки" в правом верхнем углу;
  • Затем перейти в "Почта из других ящиков";
  • Потом выбрать сервер (в нашем случае yandex.ru), на котором расположен почтовый ящик, который Вы хотите присоединить, ввести адрес электронной почты и пароль и нажать "Добавить";

  • После подтверждения логина и пароля выбрать папку, куда будут сохраняться электронные письма с данного почтового ящика и нажать "Сохранить".

    Теперь электронные письма, приходящие на почтовый ящик на сервере yandex.ru, Вы можете получать и читать в ящике на сервере mail.ru. Таким образом Вы сможете собрать всю электронную почту из нескольких ящиков, расположенных на разных серверах, в один и значительно экономить своё время на чтении электронной почты.

    Если статья "как получать всю почту в один почтовый ящик на примере mail.ru" оказалась для Вас полезной, пожалуйста поделитесь ей со своими друзьями

  • Для правильной и высоко эффективной работы на разных досках, вам нужно иметь целый пакет почтовых ящиков, для этого вам нужно создать себе 10-20 почтовых ящиков. Из-за некоторых проблем, на Mail.ru, Gmail.ru, Gmail.com лучше их не делать . Проблемы эти заключаются в том, что вышеперечисленные почтовые сервисы с некоторых пор стали усиленно бороться со спамом. И они как раз Орифлейм (письма как-нибудь связанные с ним) приравняли к спаму.


    Соответственно, вы рискуете в один прекрасный момент перестать получать письма от соискателей, или вдруг у вас появятся проблемы с отправкой писем, или неожиданно безобидное отправленное вами письмо от кого-то к кому-то, не дойдёт. Хотя, многие консультанты работают с этими почтовыми сервисами и особых проблем не наблюдают, но лучше, не рисковать.


    Работать с ящиками удобнее, через почтовую программу, например, The Bat или Mozilla Thunderbird (приложение к браузеру Мозиле), или через другие аналогичные, чтобы не работать через браузер, так как одновременно открыть в браузере десяток почтовых ящиков – не получится. А любая почтовая программа позволит вам без проблем работать с письмами со всех ящиков.

    Обратите внимание, что названия вами созданных для работы на досках ваших почтовых ящиков, НЕ должны быть похожими между собой названиями, тем более, если вы будете регистрироваться в один присест - не сходя со стула. Иначе администраторы досок объявлений вас элементарно вычислят и удалят после вашей регистрации всех ваших аккаунтов. Так как на любой серьёзной доске можно иметь от одного человека только один аккаунт.


    А можно поступить ещё проще и работать всего с одной почтой, которая будет сама собирать письма со всех ваших почтовых адресов. Почта на Яндексе с этим прекрасно справится. Настройка простейшая и много времени не займёт.

    Настройка Яндекс-почты для сборки писем с разных ящиков

    Для начала вы входите через браузер в свою заранее приготовленную Яндекс-почту для сбора писем со всех ящиков. Потом справа вверху находите обозначенный на первой картинке символ и нажимаете на него.

    После этого у вас откроется выпадающее поле, обозначенное на втором скрине. Вам нужно нажать на ссылку «СБОР ПОЧТЫ С ДРУГИХ ЯЩИКОВ».


    Затем вы попадёте на страницу настройки почтовых ящиков, с которых ваша почта на Яндексе будет собирать письма со всех ваших Е-мейлов. Далее всё интуитивно понятно. Есть, правда, одно ограничение - одна Яндекс-почта может собирать все сообщения максимум с 10 разных почтовых ящиков, но это вообще не проблема, например, можно для сбора почты сделать 2 или 3 почтовых ящика на Яндексе.


    Следующим шагом будет активация в настройках ящиков-сборщиков на Яндексе настройки доступа к этим почтовым ящикам по протоколам IMAP и POP3. На двух скринах ниже видно, как добраться до этих настроек.



    На всех ящиках с которых ваша Яндекс-почта будет собирать все письма, нужно также поступить аналогичным образом, предоставив доступ к этим ящикам по протоколам IMAP и POP3. Там настройки делаются аналогично. Правда при сборе писем таким методом через Яндекс-сборщик, имеется маленький не всегда удобный момент - сборка писем не происходит моментально, она осуществляется в течении суток всего по нескольку раз. Но, думаю, это не столь критично.


    Для моментального же получения почты на одну почту-сборщик на Яндексе, нужно настроить не сбор писем, а переадресацию (пересылку). Это так же делается не сложно. Посмотреть осуществляемые настройки почты для переадесации, можно на этой странице подсказок Яндекса.