Linq to SharePoint. Получение данных из другой коллекции сайтов

Сегодня я покажу как выполнять Cross-SiteCollection запросы с использованием Linq to SharePoint.

По умолчанию, Linq to SharePoint работает только в пределах одной коллекции узлов (SPSite). Это ограничение связано с объектом SPServerDataConnection, предоставляющего в Linq to SharePoint подключение к спискам/библиотекам документов. Если конкретней, то проблема в конструкторе:

  1. public SPServerDataConnection(string url)
  2. {
  3.     if (SPContext.Current != null)
  4.     {
  5.         this.defaultSite = SPContext.Current.Site;
  6.         this.defaultWeb = (SPContext.Current.Web.Url == url) 
  7.             ? SPContext.Current.Web 
  8.             : this.defaultSite.OpenWeb(new Uri(url).PathAndQuery);
  9.     }
  10.     else
  11.     {
  12.         this.defaultSite = new SPSite(url);
  13.         this.defaultWeb = this.defaultSite.OpenWeb(new Uri(url).PathAndQuery);
  14.     }
  15.     if (!this.defaultWeb.Exists)
  16.     {
  17.         throw new ArgumentException(Resources.GetString("CannotFindWeb",
  18.                           new object[] { url }));
  19.     }
  20.     this.defaultWebUrl = this.defaultWeb.ServerRelativeUrl;
  21.     this.openedWebs = new Dictionary<string, SPWeb>();
  22.     this.openedWebs.Add(this.defaultWebUrl, this.defaultWeb);
  23. }

Т.е. если SPContext доступен, то за пределы коллекции сайтов "выйти" нельзя. Трюк, позволяющий обойти это ограничение аналогичен трюку при работе с Linq to SharePoint в анонимном режиме.

Нам необходимо обнулить текущий контекст, чтобы спровоцировать SharePoint создать новый объект SPSite.

Cross-SiteCollection Repository

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

  1. protected BaseRepository(string listName, string webUrl, bool readOnly, bool crossSite)
  2. {
  3.     ReadOnly = readOnly;
  4.     ListName = listName;
  5.     WebUrl = webUrl;
  6.  
  7.     var ctx = SPContext.Current;
  8.     IsAnonymous = ctx != null && SPContext.Current.Web.CurrentUser == null;
  9.     if (crossSite)
  10.     {
  11.         var backupCtx = HttpContext.Current;
  12.         // Сбрасываем контекст
  13.         HttpContext.Current = null;
  14.         // Инициализируем контекст и прочее
  15.         InitializeParameters();
  16.         HttpContext.Current = backupCtx;
  17.     }
  18.     else
  19.     {
  20.         InitializeParameters();
  21.     }
  22. }

Применение

В демонстрационном проекте я создал веб-часть CrossSiteCollectionDataQuery, демонстрирующую работу с данными, расположенными в другой коллекции сайтов. Код метода CreateChildControls в нем выглядит вот так (частично):

  1. // Другая коллекция сайтов
  2. var repository1 = new EmployeeRepository("http://SharePointServer/linq"truetrue);
  3. var items1 = repository1.GetEntityCollection();
  4. foreach (var employee in items1)
  5. {
  6.     // Выводим информацию о сотруднике
  7.     RenderEmployeeInfo(employee);
  8. }
  9. Controls.Add(new LiteralControl("***<br/>"));
  10. // Текущая коллекция сайтов
  11. var repository2 = new EmployeeRepository("http://SharePointServer/sites/asc"truefalse);
  12. var items2 = repository2.GetEntityCollection();
  13. foreach (var employee in items2)
  14. {
  15.     // Выводим информацию о сотруднике
  16.     RenderEmployeeInfo(employee);
  17. }

И результат работы:

Виталий Жуков

Виталий Жуков

Техлид, Архитектор, Разработчик, Microsoft MVP. Более 20 лет опыта в области системной интеграции и разработки программного обеспечения. Специализируюсь на проектировании и внедрении масштабируемых высокопроизводительных программных решений в различных отраслях.

Смотрите также

Развертывание списков и библиотек с помощью SPFx-решений

Развертывание списков и библиотек с помощью SPFx-решений

SharePoint. Drag-and-Drop Загрузчик файлов

SharePoint. Drag-and-Drop Загрузчик файлов

CSOM. Загрузка файлов

CSOM. Загрузка файлов

SharePoint List REST API. Часть 2

SharePoint List REST API. Часть 2

SharePoint Framework. Создание веб-части на Angular

SharePoint Framework. Создание веб-части на Angular

SharePoint List REST API. Часть 1

SharePoint List REST API. Часть 1

Презентация с доклада о SharePoint Framework

Презентация с доклада о SharePoint Framework

SharePoint Framework. Создаем AngularJS 1.x Client WebPart

SharePoint Framework. Создаем AngularJS 1.x Client WebPart

SharePoint. Регистрация CSS и JavaScript с помощью DelegateControl

SharePoint. Регистрация CSS и JavaScript с помощью DelegateControl

SharePoint. Расширяем REST API

SharePoint. Расширяем REST API

SharePoint Excel Services. Создаем кредитный калькулятор

SharePoint Excel Services. Создаем кредитный калькулятор

SharePoint Ribbon API. Использование ToggleButton

SharePoint Ribbon API. Использование ToggleButton

SharePoint 2013. How To: настройка входящей почты для разработчиков

SharePoint 2013. How To: настройка входящей почты для разработчиков

Мифы и правда о Linq to SharePoint

Мифы и правда о Linq to SharePoint

5 особенностей SPSiteDataQuery

5 особенностей SPSiteDataQuery

SharePoint 2013. Введение в SharePoint App. Часть 2

SharePoint 2013. Введение в SharePoint App. Часть 2

SharePoint 2013. Введение в SharePoint App. Часть 1

SharePoint 2013. Введение в SharePoint App. Часть 1

Превью для веб-части в SharePoint 2010/2013

Превью для веб-части в SharePoint 2010/2013

SharePoint 2013. Еще немного о новых контролах

SharePoint 2013. Еще немного о новых контролах

SharePoint 2013. Контрол ClientPeoplePicker

SharePoint 2013. Контрол ClientPeoplePicker

SharePoint 2013. Контрол ImageCrop

SharePoint 2013. Контрол ImageCrop

SharePoint 2013. Тип поля Geolocation

SharePoint 2013. Тип поля Geolocation

Создание типа поля в SharePoint

Создание типа поля в SharePoint

SharePoint 2010. Длительные операции с обновляемым статусом

SharePoint 2010. Длительные операции с обновляемым статусом

Linq to SharePoint. Создаем ContentIterator

Linq to SharePoint. Создаем ContentIterator

Linq to SharePoint. Версионность

Linq to SharePoint. Версионность

SharePoint. Получение URL-адреса иконки для документа

SharePoint. Получение URL-адреса иконки для документа

SharePoint 2010. PostBack для Fluent Ribbon API

SharePoint 2010. PostBack для Fluent Ribbon API

Linq to SharePoint. Блокировка документов

Linq to SharePoint. Блокировка документов

Linq to SharePoint. Паттерн Repository

Linq to SharePoint. Паттерн Repository

Linq to SharePoint. Получение мета-данных списка

Linq to SharePoint. Получение мета-данных списка

Linq to SharePoint. Мапинг полей

Linq to SharePoint. Мапинг полей

Linq to SharePoint. Формирование данных для ProcessBatchData

Linq to SharePoint. Формирование данных для ProcessBatchData

Linq to SharePoint. Сравнение производительности с Camlex.NET

Linq to SharePoint. Сравнение производительности с Camlex.NET

Linq to SharePoint. Часть 5. Поля Choice и MultiChoice

Linq to SharePoint. Часть 5. Поля Choice и MultiChoice

Linq to SharePoint. Часть 4. Dynamic LINQ

Linq to SharePoint. Часть 4. Dynamic LINQ

Linq to SharePoint. Особенности. Часть 3

Linq to SharePoint. Особенности. Часть 3

Linq to SharePoint. Особенности. Часть 2

Linq to SharePoint. Особенности. Часть 2

SharePoint 2010. PeopleEditor. Установка значения

SharePoint 2010. PeopleEditor. Установка значения

SharePoint 2010. Настройка входящей почты для кастомного списка

SharePoint 2010. Настройка входящей почты для кастомного списка

Linq to Sharepoint. Особенности

Linq to Sharepoint. Особенности

EntityFramework. Оптимистические блокировки

EntityFramework. Оптимистические блокировки