Веб-службы Exchange: пакетная обработка с FindItemsResult <Item> - PullRequest
3 голосов
/ 30 сентября 2010

Я пишу приложение для обработки вложений электронной почты с использованием веб-служб Exchange.

Общая структура моей проблемной области следующая:

public static void Main()
{
    FindItemsResults<Item> findResults = FindItems();

    foreach (Item item in findResults)
    {
        DoSomethingWithItem(item);
    }
}

public static FindItemsResults<Item> FindItems()
{
            FindItemsResults<Item> findResults;

            ItemView view = new ItemView(10);    //batching 10 at a time
            view.OrderBy.Add(ItemSchema.DateTimeReceived, SortDirection.Ascending);
            view.PropertySet = new PropertySet(
                BasePropertySet.IdOnly,
                ItemSchema.Subject,
                ItemSchema.DateTimeReceived);

            findResults = service.FindItems(
                WellKnownFolderName.Inbox,
                new SearchFilter.SearchFilterCollection(
                    LogicalOperator.Or,
                    new SearchFilter.ContainsSubstring(ItemSchema.Subject, Properties.Settings.Default.EmailSubject)),
                view);

            //return set of emails
            return findResults;
}

Сначала это выглядело какОК - он обработал мои предыдущие тестовые примеры на отлично.Но когда я приступил к тестированию с большими нагрузками, я понял, что он обрабатывает только первые 10 изделий, поскольку я определял размер партии в 10 единиц (ItemView view = new ItemView(10)), но я не проверял дальнейшие партии.

Я мог бы просто увеличить размер пакета, но позже, быстро найдя Google, я нашел лучший пример:

public static FindItemsResults<Item> FindItems()
{
            FindItemsResults<Item> findResults;

            ItemView view = new ItemView(10, 0, OffsetBasePoint.Beginning); 
            view.OrderBy.Add(ItemSchema.DateTimeReceived, SortDirection.Ascending);
            view.PropertySet = new PropertySet(
                BasePropertySet.IdOnly,
                ItemSchema.Subject,
                ItemSchema.DateTimeReceived);

            do
            {
                findResults = service.FindItems(
                    WellKnownFolderName.Inbox,
                    new SearchFilter.SearchFilterCollection(
                        LogicalOperator.Or,
                        new SearchFilter.ContainsSubstring(ItemSchema.Subject, Properties.Settings.Default.EmailSubject)),
                    view);

                //any more batches?
                if (findResults.NextPageOffset.HasValue)
                {
                    view.Offset = findResults.NextPageOffset.Value;
                }
            }
            while (findResults.MoreAvailable);

            return findResults;
}

Это перебирает столько писем, сколько я хочу, но дляпричины, которые я пока не могу понять, цикл foreach теперь обрабатывает только первый элемент в findResults.

Несмотря на то, что findResults содержит более одного элемента (findResults.Items.Count> 1), со вторым примером, findResults.MoreAvailable= ложьЧестно говоря, я просматривал партии раньше, поэтому имеет смысл посмотреть на конец последней партии.

Но как мне сбросить findResults, чтобы он обрабатывал всю партию?Я попытался установить findResults.MoreAvailable, но он доступен только для чтения ... Чего мне не хватает?

ЗАКЛЮЧЕНИЕ:

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

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

1 Ответ

8 голосов
/ 03 октября 2010

Мне кажется, что ваша главная проблема в том, что вы пытаетесь определить функцию, которая возвращает FindItemsResults<Item>.Если вы используете сопоставление результатов, у вас будет не один объект этого типа.Вместо этого при получении каждой новой страницы FindItemsResults<Item> будет перезаписано.В следующем примере отображаются темы всех элементов из Inbox:

ItemView view = new ItemView(10, 0, OffsetBasePoint.Beginning);
view.OrderBy.Add(ItemSchema.DateTimeReceived, SortDirection.Descending);
view.PropertySet = new PropertySet(
    BasePropertySet.IdOnly,
    ItemSchema.Subject,
    ItemSchema.DateTimeReceived);

// save the folder where we will make searching to do this one time
Folder myInbox = Folder.Bind(service, WellKnownFolderName.Inbox);

FindItemsResults<Item> findResults;

do
{
    findResults = myInbox.FindItems(
        new SearchFilter.ContainsSubstring(ItemSchema.Subject,
                Properties.Settings.Default.EmailSubject)),
        view);

    foreach (Item item in findResults)
    {
        // Do something with the item.
        Console.WriteLine();
        if (item is EmailMessage)
        {
            EmailMessage em = item as EmailMessage;
            Console.WriteLine("Subject: \"{0}\"", em.Subject);
        }
        else if (item is MeetingRequest)
        {
            MeetingRequest mr = item as MeetingRequest;
            Console.WriteLine("Subject: \"{0}\"", mr.Subject);
        }
        else
        {
            // we can handle other item types
        }
    }

    //any more batches?
    if (findResults.NextPageOffset.HasValue)
    {
        view.Offset = findResults.NextPageOffset.Value;
    }
}
while (findResults.MoreAvailable);

. Код отображает темы на выходе консоли.Если вы хотите использовать EmailMessage или MeetingRequest по-другому, вам следует изменить соответствующий код.Вы также можете определить делегата, который будет что-то делать с найденным EmailMessage или MeetingRequest и вызвать делегата вместо Console.WriteLine.Если вам нужно где-то одинаковые элементы, вам нужно будет создать коллекцию, например List<Item>, заполнить ее функцией и вернуть вместо FindItemsResults<Item>, что вы делаете в настоящее время.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...