DocumentDB SQL - преобразование Linq в SQL
В DocumentDB мы фактически используем SQL для запроса документов. Если мы занимаемся разработкой .NET, существует также поставщик LINQ, который можно использовать и который может генерировать соответствующий SQL из запроса LINQ.
Поддерживаемые типы данных
В DocumentDB все примитивные типы JSON поддерживаются поставщиком LINQ, включенным в DocumentDB .NET SDK, а именно:
- Numeric
- Boolean
- String
- Null
Поддерживаемое выражение
Следующие скалярные выражения поддерживаются поставщиком LINQ, включенным в DocumentDB .NET SDK.
Constant Values - Включает постоянные значения примитивных типов данных.
Property/Array Index Expressions - Выражения относятся к свойству объекта или элемента массива.
Arithmetic Expressions - Включает общие арифметические выражения для числовых и логических значений.
String Comparison Expression - Включает сравнение строкового значения с некоторым постоянным строковым значением.
Object/Array Creation Expression- Возвращает объект типа составного значения или анонимного типа или массив таких объектов. Эти значения могут быть вложенными.
Поддерживаемые операторы LINQ
Вот список поддерживаемых операторов LINQ в поставщике LINQ, включенном в DocumentDB .NET SDK.
Select - Проекции переводятся в SQL SELECT, включая построение объекта.
Where- Фильтры переводятся в SQL WHERE и поддерживают перевод между &&, || и ! к операторам SQL.
SelectMany- Позволяет раскручивать массивы к предложению SQL JOIN. Может использоваться для объединения / вложения выражений для фильтрации элементов массива.
OrderBy and OrderByDescending - Переводится в ORDER BY по возрастанию / убыванию.
CompareTo- Преобразует сравнение диапазонов. Обычно используется для строк, поскольку они не сопоставимы в .NET.
Take - Преобразуется в SQL TOP для ограничения результатов запроса.
Math Functions - Поддерживает перевод из .NET Abs, Acos, Asin, Atan, Ceiling, Cos, Exp, Floor, Log, Log10, Pow, Round, Sign, Sin, Sqrt, Tan, Truncate в эквивалентные встроенные функции SQL.
String Functions - Поддерживает перевод из .NET Concat, Contains, EndsWith, IndexOf, Count, ToLower, TrimStart, Replace, Reverse, TrimEnd, StartsWith, SubString, ToUpper в эквивалентные встроенные функции SQL.
Array Functions - Поддерживает перевод из .NET Concat, Contains и Count в эквивалентные встроенные функции SQL.
Geospatial Extension Functions - Поддерживает перевод из методов-заглушек Distance, Within, IsValid и IsValidDetailed в эквивалентные встроенные функции SQL.
User-Defined Extension Function - Поддерживает перевод из метода-заглушки UserDefinedFunctionProvider.Invoke в соответствующую пользовательскую функцию.
Miscellaneous- Поддерживает перевод операторов объединения и условных операторов. Может преобразовывать Contains в String CONTAINS, ARRAY_CONTAINS или SQL IN в зависимости от контекста.
Давайте посмотрим на пример, в котором мы будем использовать .Net SDK. Ниже приведены три документа, которые мы рассмотрим для этого примера.
Новый клиент 1
{
"name": "New Customer 1",
"address": {
"addressType": "Main Office",
"addressLine1": "123 Main Street",
"location": {
"city": "Brooklyn",
"stateProvinceName": "New York"
},
"postalCode": "11229",
"countryRegionName": "United States"
},
}
Новый клиент 2
{
"name": "New Customer 2",
"address": {
"addressType": "Main Office",
"addressLine1": "678 Main Street",
"location": {
"city": "London",
"stateProvinceName": " London "
},
"postalCode": "11229",
"countryRegionName": "United Kingdom"
},
}
Новый клиент 3
{
"name": "New Customer 3",
"address": {
"addressType": "Main Office",
"addressLine1": "12 Main Street",
"location": {
"city": "Brooklyn",
"stateProvinceName": "New York"
},
"postalCode": "11229",
"countryRegionName": "United States"
},
}
Ниже приведен код, в котором мы запрашиваем с помощью LINQ. Мы определили запрос LINQ вq, но он не будет выполняться, пока мы не запустим на нем .ToList.
private static void QueryDocumentsWithLinq(DocumentClient client) {
Console.WriteLine();
Console.WriteLine("**** Query Documents (LINQ) ****");
Console.WriteLine();
Console.WriteLine("Quering for US customers (LINQ)");
var q =
from d in client.CreateDocumentQuery<Customer>(collection.DocumentsLink)
where d.Address.CountryRegionName == "United States"
select new {
Id = d.Id,
Name = d.Name,
City = d.Address.Location.City
};
var documents = q.ToList();
Console.WriteLine("Found {0} US customers", documents.Count);
foreach (var document in documents) {
var d = document as dynamic;
Console.WriteLine(" Id: {0}; Name: {1}; City: {2}", d.Id, d.Name, d.City);
}
Console.WriteLine();
}
SDK преобразует наш запрос LINQ в синтаксис SQL для DocumentDB, генерируя предложения SELECT и WHERE на основе нашего синтаксиса LINQ.
Назовем указанные выше запросы из задачи CreateDocumentClient.
private static async Task CreateDocumentClient() {
// Create a new instance of the DocumentClient
using (var client = new DocumentClient(new Uri(EndpointUrl), AuthorizationKey)) {
database = client.CreateDatabaseQuery("SELECT * FROM c WHERE c.id =
'myfirstdb'").AsEnumerable().First();
collection = client.CreateDocumentCollectionQuery(database.CollectionsLink,
"SELECT * FROM c WHERE c.id = 'MyCollection'").AsEnumerable().First();
QueryDocumentsWithLinq(client);
}
}
Когда приведенный выше код выполняется, он дает следующий результат.
**** Query Documents (LINQ) ****
Quering for US customers (LINQ)
Found 2 US customers
Id: 7e9ad4fa-c432-4d1a-b120-58fd7113609f; Name: New Customer 1; City: Brooklyn
Id: 34e9873a-94c8-4720-9146-d63fb7840fad; Name: New Customer 1; City: Brooklyn