<?xml version="1.0" encoding="utf-8"?> 
<rss version="2.0"
  xmlns:itunes="http://www.itunes.com/dtds/podcast-1.0.dtd"
  xmlns:atom="http://www.w3.org/2005/Atom">

<channel>

<title>Блоги: заметки с тегом libmapi</title>
<link>https://blogengine.ru/blogs/tags/libmapi/</link>
<description>Автоматически собираемая лента заметок, написанных в блогах на Эгее</description>
<author></author>
<language>ru</language>
<generator>Aegea 11.0 (v4079e)</generator>

<itunes:subtitle>Автоматически собираемая лента заметок, написанных в блогах на Эгее</itunes:subtitle>
<itunes:image href="" />
<itunes:explicit>no</itunes:explicit>

<item>
<title>«Переговорки»: панихида</title>
<guid isPermaLink="false">119801</guid>
<link>https://bolknote.ru/all/2362/</link>
<pubDate>Mon, 28 Dec 2009 06:34:00 +0500</pubDate>
<author>Евгений Степанищев</author>
<comments>https://bolknote.ru/all/2362/</comments>
<description>
&lt;p&gt;&lt;a href="https://bolknote.ru/"&gt;Евгений Степанищев&lt;/a&gt;:&lt;/p&gt;
&lt;p&gt;Проект «&lt;a href="https://web.archive.org/web/20091210200915/http://clubs.ya.ru/company/replies.xml?item_no=20978"&gt;Переговорки&lt;/a&gt;» — внутренний веб-сервис «Яндекса» для бронирования переговорок, синхронизированный с Microsoft Exchange. Это, наверное, самый сложный проект, который мне приходилось делать за всю жизнь.&lt;/p&gt;
&lt;p&gt;Используется он теми, кто не может установить себе Outlook — ребятами с «Маками» и разнообразными «Линуксами» (а таких очень много). Кроме того, что от Exchange мы будем постепенно уходить, а интерфейс бронирования останется без изменений.&lt;/p&gt;
&lt;p&gt;Никогда я ещё так не выкладывался, так не радовался, не расстраивался, не оставался с таким упорством допоздна в офисе. Звучит просто — вытаскивать время от времени данные из Exchange и отсылать обратно новую бронь. К сожалению, MS Exchange, который использовался тогда в компании, не умел общаться с внешним миром никак иначе, кроме протокола MAPI — бинарному исчадию ада, причём не просто через MAPI, а через так называемый MAPI через RPC — диалект DCE-RPC, который называется MSRPC.&lt;/p&gt;
&lt;p&gt;Помню, первое, что пришло на ум использовать — &lt;a href="http://oss.coresecurity.com/projects/impacket.html"&gt;impacket&lt;/a&gt;, модуль для Python, который позволял работать с «сырыми» пакетами, но к нему не было документации, а примеров в сети нашлось всего ничего — несколько эксплоитов и только. Впрочем, разобраться всё-таки удалось. Правда, попробовав описать вызовы верхнего уровня, я убедился, что это работа на года и решил поискать что-нубудь ещё.&lt;/p&gt;
&lt;p&gt;Из бесплатных библиотек уровнем повыше нашлась одна-единственная — &lt;a href="https://web.archive.org/web/20090228011220/http://www.openchange.org/index.php?option=com_content&amp;task=view&amp;id=63&amp;Itemid=71"&gt;libmapi&lt;/a&gt;, поскольку она требовала наличия альфы Samba4 строго определённой версии, а на машине уже стояла Samba3 (через которую мы работали), решено было попробовать её скомпилировать. Оказалось, что такие попытки в компании уже предпринимались, но закончились неудачей. Решив что попытка не пытка, за несколько дней я её всё-таки собрал и запустил.&lt;/p&gt;
&lt;p&gt;Это и положило начало проекта «Переговорки».&lt;/p&gt;
&lt;p&gt;Процедура компиляции оказалась весьма нетривиальной, собиралась только версия из транка, и та только с моими правками. Ещё сложнее оказалось разобраться с утилитами, которые требовалось запустить после исталляции, но и эту проблему удалось победить, собрав скудные крохи по форумам и написав несколько писем разработчикам. Позднее &lt;a href="https://web.archive.org/web/20090527143405/https://peimei.ya.ru/"&gt;Сергей Белов&lt;/a&gt; сумел установить появившийся свежий дебиан-пакет в chroot-окружении, что облегчило задачу установки библиотеки на другие машины, а наш админ, Володя Емченко, вычистив из пакета всё лишнее, придумал способ установки его в произвольное место без использования chroot.&lt;/p&gt;
&lt;p&gt;Через несколько дней после того как я в первый раз скомпилировал libmapi, тестовые утилиты на Си благополучно подключились к Exchange и уже умели отпределять дают им правильные данные для авторизации или нет. Эти несколько дней ушли на то, чтобы разобраться как создавать профиль, как настраивать библиотеку и Samba4 для работы и так далее. Оставалось как-то состыковать всё это дело с Python.&lt;/p&gt;
&lt;p&gt;&lt;a href="/2009/04/29/~2120"&gt;1275 строк основного модуля&lt;/a&gt;, 647 строк файла описаний структур MAPI, 4046 констант, 457 строк для описания Pid-свойств. А сколько не заработало, сколько пришлось переписать.  Самая рутинная работа была — искать по всем исходникам libmapi и Samba описания структур, пытаться сообразить как их переписать на Python и состыковать со всеми остальными. Ребята, придумавшие модуль ctypes для Python, большие молодцы.Правда мне пришлось придумать и реализовать парсер для расширенного синтаксиса, чтобы описывать динамические структуры MAPI, не знаю как сейчас, а тогда libmapi &lt;a href="/2009/03/30/~2065"&gt;не умел читать повторяющиеся встречи&lt;/a&gt; (кажется, и теперь не умеет), так что читать и описывать их приходилось на самом низком уровне, а значит как-то хранить эти данные и уметь ими манипулировать.&lt;/p&gt;
&lt;p&gt;Ничего, пара дней &lt;a href="/2009/04/07/~2078"&gt;корпения над бинарной распечаткой&lt;/a&gt; с последующими уточнениями в противоречивой документацией от Microsoft и я написал класс, который такие встречи умеет разбирать.  Впрочем, это было уже позднее, а сначала надо было догадаться что делать после логина. Из документации у libmapi — только перечень функций и исходники на Си. Пришлось внимательно просматривать первое и читать второе.  Тут нельзя не упомянуть &lt;a href="https://web.archive.org/web/20090213183230/https://sorgoz.ya.ru/"&gt;Сергея Андрюхина&lt;/a&gt;, несколько очень ценных его идей и исследований вошли в тот код. Кстати, местами исходники на Си &lt;a href="/2008/12/26/~1969"&gt;написаны так плохо&lt;/a&gt;, что приходилось упрощать код — не всё, что можно позволить себе в Си, можно позволить себе в менее шустром «Пайтоне».&lt;/p&gt;
&lt;p&gt;Разобраться с порядком вызова этого великолепия было непросто, плюс ещё множество более мелких проблем, вроде &lt;a href="/2009/01/28/~1999"&gt;странного формата&lt;/a&gt; хранения времени (Exchange хранит число минут с 1 января 1601 года). Плюс &lt;a href="/2009/04/09/~2083"&gt;утечки libmapi&lt;/a&gt;, которые пришлось изолировать, решать проблему с передачей данных в родительский процесс, мучаться с Python 2.4, который, как оказалось, никогда не освобождает память, выделением памяти на куче «про запас» через Python API (спасибо ctypes), определением по косвенным признакам, что память закончилась, пора отдать данные в родительский и умереть, и так далее.&lt;/p&gt;
&lt;p&gt;Сначала мы научились читать freebusy информацию — просто даты и время, когда переговорка чем-то занята, потом уже — заходить в папку переговорки и читать её календарь.&lt;/p&gt;
&lt;p&gt;Класс пережил четыре рефакторинга, например, я обернул некоторые структуры классами с деструктором, чтобы было проще следить за памятью (так как ctypes работает с бинарным кодом за памятью приходилось следить самостоятельно), что, правда, всё равно не спасло от всех проблем — структуры, в некоторых случаях, нужно было уничтожать в строго определённом порядке.&lt;/p&gt;
&lt;p&gt;Спасибо &lt;a href="http://twitter.com/thasonic/"&gt;Саше Покатилову&lt;/a&gt;, который правил мой код, пока я был в отпуске и занимался Django-частью проекта, с ним же мы проектировали базу. Например, мы решили просто размножать повторяющиеся встречи на год вперёд, избежав тем самым множество проблем. Год — это так называемое «окно бронирования», дальше этого срока наш Exchange встречи не видит. Каждый день Сашин механизм достаточно оптимально размножал всречи ещё на один день дальше, мой класс при помощи полезнейшего модуля &lt;a href="http://labix.org/python-dateutil"&gt;dateutil.rrule&lt;/a&gt; переводил по достаточно хитрым правилам повторяющиеся встречи Exchange в список дат.&lt;/p&gt;
&lt;p&gt;Наверняка я не помню ещё массу сложностей, которые мы решали, например, я совсем не рассказал как мы отправляли бронь встречи по протоколу SMTP, как долго я готовил разные версии письма и нашёл-таки правильный вариант, уехал со спокойной душой в отпуск, а потом оказалось, что с русскими буквами встречи не бронировались, но Саша всё-таки нашёл решение.&lt;/p&gt;
&lt;p&gt;Думаю, если попробовать вспомнить всё, хватит на небольшую книгу… Я рад, что эту невероятную задачу удалось решить. С моей стороны, как мне кажется, это 100% моей «мощности» как программиста. Будущее покажет, но пока все задачи я буду оценивать по этой.&lt;/p&gt;
&lt;p&gt;К счастью, в новом Exchange, который появился у нас в компании, появилась поддержка SOAP и «Переговорки» переделали с использованием этого протокола. Есть предварительное согласие на выкладывание модуля, который у меня тогда получился в интернет. Пожалуй, сейчас он бесполезен (хотя, может авторы libmapi запрограммируют на его примере чтение повторяющихся встреч, попробую им его кинуть), но пусть он всё равно полежит у меня, как памятник полугоду моей жизни.&lt;/p&gt;
</description>
</item>


</channel>
</rss>