Заметка
Доступ к этой странице требует авторизации. Вы можете попробовать войти в систему или изменить каталог.
Доступ к этой странице требует авторизации. Вы можете попробовать сменить директорию.
В этой статье описывается обработка запросов JSON Patch в веб-API ASP.NET Core.
Поддержка исправлений JSON в веб-API ASP.NET Core основана на System.Text.Json сериализации и требует Microsoft.AspNetCore.JsonPatch.SystemTextJson пакета NuGet.
Что такое стандарт JSON Patch?
Стандарт патча JSON:
Стандартный формат описания изменений, применяемых к документу JSON.
Определяется в RFC 6902 и широко используется в API RESTful для выполнения частичных обновлений ресурсов JSON.
Описывает последовательность операций, которые изменяют документ JSON, например:
addremovereplacemovecopytest
В веб-приложениях JSON Patch обычно используется в операции PATCH для частичного обновления ресурса. Вместо отправки всего ресурса для обновления клиенты могут отправлять документ исправления JSON, содержащий только изменения. Установка исправлений уменьшает размер полезной нагрузки и повышает эффективность.
Общие сведения о стандарте исправлений JSON см. в jsonpatch.com.
Поддержка исправлений JSON в веб-API ASP.NET Core
Поддержка JSON Patch в веб-API ASP.NET Core основана на System.Text.Json сериализации, начиная с .NET 10, с реализацией Microsoft.AspNetCore.JsonPatch на основе System.Text.Json сериализации. Эта функция:
- Требуется
Microsoft.AspNetCore.JsonPatch.SystemTextJsonпакет NuGet. - Соответствует современным методикам .NET, используя библиотеку System.Text.Json , оптимизированную для .NET.
- Обеспечивает улучшенную производительность и сокращение использования памяти по сравнению с устаревшей реализацией
Newtonsoft.Json. Дополнительные сведения о устаревшейNewtonsoft.Jsonреализации см. в версии .NET 9 этой статьи.
Note
Реализация Microsoft.AspNetCore.JsonPatch, основанная на сериализации System.Text.Json, не является заменой для устаревшей реализации на основе Newtonsoft.Json. Например, ExpandoObjectон не поддерживает динамические типы.
Important
Стандарт JSON Patch имеет присущие риски безопасности. Так как эти риски присущи стандарту исправлений JSON, реализация ASP.NET Core не пытается устранить связанные с ней риски безопасности. Разработчик отвечает за безопасное применение документа исправления JSON к целевому объекту. Дополнительные сведения см. в разделе "Устранение рисков безопасности ".
Включение поддержки исправлений JSON с помощью System.Text.Json
Чтобы включить поддержку System.Text.Json JSON-патчей, установите пакет NuGet Microsoft.AspNetCore.JsonPatch.SystemTextJson.
dotnet add package Microsoft.AspNetCore.JsonPatch.SystemTextJson --prerelease
Этот пакет предоставляет JsonPatchDocument<TModel> класс для представления документа JSON Patch для объектов типа T и пользовательскую логику для сериализации и десериализации JSON Patch документов с помощью System.Text.Json. Ключевой метод класса JsonPatchDocument<TModel> — это ApplyTo(Object), который применяет операции патчей к целевому объекту типа T.
Код метода действия с применением JSON Patch
В контроллере API есть метод действия для JSON Patch, который:
- помечен атрибутом HttpPatchAttribute;
- принимает JsonPatchDocument<TModel> обычно с указанием FromBodyAttribute;
- вызывает ApplyTo(Object) для целевого документа, чтобы применить изменения.
Пример метода действия контроллера:
[HttpPatch("{id}", Name = "UpdateCustomer")]
public IActionResult Update(AppDb db, string id, [FromBody] JsonPatchDocument<Customer> patchDoc)
{
// Retrieve the customer by ID
var customer = db.Customers.FirstOrDefault(c => c.Id == id);
// Return 404 Not Found if customer doesn't exist
if (customer == null)
{
return NotFound();
}
patchDoc.ApplyTo(customer, jsonPatchError =>
{
var key = jsonPatchError.AffectedObject.GetType().Name;
ModelState.AddModelError(key, jsonPatchError.ErrorMessage);
}
);
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
}
return new ObjectResult(customer);
}
Этот код из примера приложения работает со следующими Customer моделями и Order моделями:
namespace App.Models;
public class Customer
{
public string Id { get; set; }
public string? Name { get; set; }
public string? Email { get; set; }
public string? PhoneNumber { get; set; }
public string? Address { get; set; }
public List<Order>? Orders { get; set; }
public Customer()
{
Id = Guid.NewGuid().ToString();
}
}
namespace App.Models;
public class Order
{
public string Id { get; set; }
public DateTime? OrderDate { get; set; }
public DateTime? ShipDate { get; set; }
public decimal TotalAmount { get; set; }
public Order()
{
Id = Guid.NewGuid().ToString();
}
}
Ключевые действия примера метода действия:
-
Извлечение клиента:
- Метод извлекает
Customerобъект из базы данныхAppDbс помощью предоставленного идентификатора. - Если объект
Customerне найден, возвращается ответ404 Not Found.
- Метод извлекает
-
Применение JSON-патча:
- Метод ApplyTo(Object) применяет операции исправления JSON из patchDoc к полученному
Customerобъекту. - Если ошибки возникают во время приложения исправления, например недопустимые операции или конфликты, они фиксируются делегатом обработки ошибок. Этот делегат добавляет сообщения об ошибках в
ModelState, используя имя типа затронутого объекта и сообщение об ошибке.
- Метод ApplyTo(Object) применяет операции исправления JSON из patchDoc к полученному
-
Проверка ModelState:
- После применения исправления метод проверяет
ModelStateна наличие ошибок. - Если
ModelStateнедействительно, например, из-за ошибок патча, возвращает400 Bad Requestответ с ошибками проверки.
- После применения исправления метод проверяет
-
Верните обновленного клиента:
- Если исправление успешно применено и
ModelStateдопустимо, метод возвращает обновленныйCustomerобъект в ответе.
- Если исправление успешно применено и
Пример ответа об ошибке:
В следующем примере показано тело 400 Bad Request ответа для операции JSON Patch, когда указанный путь недопустим:
{
"Customer": [
"The target location specified by path segment 'foobar' was not found."
]
}
Применение документа JSON Patch к объекту
В следующих примерах показано, как использовать ApplyTo(Object) метод для применения документа JSON Patch к объекту.
Пример: применение JsonPatchDocument<TModel> к объекту
В следующем примере показано:
- Операции
add,replaceиremove. - Операции с вложенными свойствами.
- Добавление нового элемента в массив.
- Использование преобразователя строкового перечисления JSON в документе JSON-обновления.
// Original object
var person = new Person {
FirstName = "John",
LastName = "Doe",
Email = "johndoe@gmail.com",
PhoneNumbers = [new() {Number = "123-456-7890", Type = PhoneNumberType.Mobile}],
Address = new Address
{
Street = "123 Main St",
City = "Anytown",
State = "TX"
}
};
// Raw JSON patch document
string jsonPatch = """
[
{ "op": "replace", "path": "/FirstName", "value": "Jane" },
{ "op": "remove", "path": "/Email"},
{ "op": "add", "path": "/Address/ZipCode", "value": "90210" },
{ "op": "add", "path": "/PhoneNumbers/-", "value": { "Number": "987-654-3210",
"Type": "Work" } }
]
""";
// Deserialize the JSON patch document
var patchDoc = JsonSerializer.Deserialize<JsonPatchDocument<Person>>(jsonPatch);
// Apply the JSON patch document
patchDoc!.ApplyTo(person);
// Output updated object
Console.WriteLine(JsonSerializer.Serialize(person, serializerOptions));
В предыдущем примере приводятся следующие выходные данные обновленного объекта:
{
"firstName": "Jane",
"lastName": "Doe",
"address": {
"street": "123 Main St",
"city": "Anytown",
"state": "TX",
"zipCode": "90210"
},
"phoneNumbers": [
{
"number": "123-456-7890",
"type": "Mobile"
},
{
"number": "987-654-3210",
"type": "Work"
}
]
}
Метод ApplyTo(Object) обычно следует соглашениям и параметрам System.Text.Json обработки JsonPatchDocument<TModel>, включая поведение, управляемое следующими параметрами:
- JsonNumberHandling: считываются ли числовые свойства из строк.
- PropertyNameCaseInsensitive: Являются ли имена свойств чувствительными к регистру.
Основные различия между System.Text.Json и новой JsonPatchDocument<TModel> реализацией:
- Тип среды выполнения целевого объекта, а не объявленного типа, определяет, какие свойства ApplyTo(Object) патчи.
- System.Text.Json десериализация зависит от объявленного типа для определения подходящих свойств.
Пример. Применение JsonPatchDocument с обработкой ошибок
При применении документа исправления JSON могут возникать различные ошибки. Например, целевой объект может не иметь указанного свойства, или указанное значение может быть несовместимо с типом свойства.
JSON Patch поддерживает test операцию, которая проверяет, равно ли указанное значение целевому свойству. Если это не так, возвращается ошибка.
В следующем примере показано, как корректно обрабатывать эти ошибки.
Important
Объект, переданный методу ApplyTo(Object) , изменяется на месте. Вызывающий отвечает за отмену изменений, если какая-либо операция завершается ошибкой.
// Original object
var person = new Person {
FirstName = "John",
LastName = "Doe",
Email = "johndoe@gmail.com"
};
// Raw JSON patch document
string jsonPatch = """
[
{ "op": "replace", "path": "/Email", "value": "janedoe@gmail.com"},
{ "op": "test", "path": "/FirstName", "value": "Jane" },
{ "op": "replace", "path": "/LastName", "value": "Smith" }
]
""";
// Deserialize the JSON patch document
var patchDoc = JsonSerializer.Deserialize<JsonPatchDocument<Person>>(jsonPatch);
// Apply the JSON patch document, catching any errors
Dictionary<string, string[]>? errors = null;
patchDoc!.ApplyTo(person, jsonPatchError =>
{
errors ??= new ();
var key = jsonPatchError.AffectedObject.GetType().Name;
if (!errors.ContainsKey(key))
{
errors.Add(key, new string[] { });
}
errors[key] = errors[key].Append(jsonPatchError.ErrorMessage).ToArray();
});
if (errors != null)
{
// Print the errors
foreach (var error in errors)
{
Console.WriteLine($"Error in {error.Key}: {string.Join(", ", error.Value)}");
}
}
// Output updated object
Console.WriteLine(JsonSerializer.Serialize(person, serializerOptions));
В предыдущем примере приводятся следующие выходные данные:
Error in Person: The current value 'John' at path 'FirstName' is not equal
to the test value 'Jane'.
{
"firstName": "John",
"lastName": "Smith", <<< Modified!
"email": "janedoe@gmail.com", <<< Modified!
"phoneNumbers": []
}
Снижение рисков безопасности
При использовании Microsoft.AspNetCore.JsonPatch.SystemTextJson пакета важно понимать и устранять потенциальные риски безопасности. В следующих разделах описаны выявленные риски безопасности, связанные с исправлением JSON, и рекомендуемые способы устранения рисков для обеспечения безопасного использования пакета.
Important
Это не исчерпывающий список угроз. Разработчики приложений должны проводить собственные проверки модели угроз, чтобы определить полный список конкретных приложений и при необходимости придумать соответствующие меры по устранению рисков. Например, приложения, которые делают коллекции доступными для операций исправления, должны учитывать потенциал для атак алгоритмической сложности, если эти операции вставляют или удаляют элементы в начале коллекции.
Чтобы свести к минимуму риски безопасности при интеграции функций исправлений JSON в свои приложения, разработчики должны:
- Проведение комплексных моделей угроз для их собственных приложений.
- Устранение выявленных угроз.
- Следуйте рекомендациям по устранению рисков в следующих разделах.
Отказ в обслуживании (DoS) через увеличение памяти
-
Сценарий: вредоносный клиент отправляет
copyоперацию, которая дублирует графы больших объектов несколько раз, что приводит к чрезмерному потреблению памяти. - Влияние: потенциальные условия исчерпанияOf-Memory (OOM), приводящие к сбоям в работе службы.
-
Mitigation:
- Перед вызовом ApplyTo(Object) проверьте входящие JSON-патчи на соответствие размера и структуры.
- Проверка должна быть специфичной для приложения, но пример проверки может выглядеть следующим образом:
public void Validate(JsonPatchDocument<T> patch)
{
// This is just an example. It's up to the developer to make sure that
// this case is handled properly, based on the app needs.
if (patch.Operations.Where(op=>op.OperationType == OperationType.Copy).Count()
> MaxCopyOperationsCount)
{
throw new InvalidOperationException();
}
}
Подверсия бизнес-логики
- Сценарий. Операции исправления могут управлять полями с неявными инвариантными (например, внутренними флагами, идентификаторами или вычисляемыми полями), нарушая бизнес-ограничения.
- Влияние: проблемы целостности данных и непреднамеренное поведение приложения.
-
Mitigation:
- Используйте POCOs (простые объекты CLR) с явно определёнными свойствами, которые безопасно изменять.
- Избегайте предоставления конфиденциальных или критически важных свойств безопасности в целевом объекте.
- Если объект POCO не используется, проверьте исправленный объект после применения операций, чтобы убедиться, что бизнес-правила и инварианты не нарушаются.
- Используйте POCOs (простые объекты CLR) с явно определёнными свойствами, которые безопасно изменять.
Проверка подлинности и авторизация
- Сценарий. Неавторизованные или несанкционированные клиенты отправляют вредоносные запросы на исправление JSON.
- Влияние: несанкционированный доступ к изменению конфиденциальных данных или нарушению поведения приложения.
-
Mitigation:
- Защита конечных точек, принимаюющих запросы на исправление JSON с помощью надлежащих механизмов проверки подлинности и авторизации.
- Ограничить доступ к доверенным клиентам или пользователям с соответствующими разрешениями.
Получение кода
Просмотреть или скачать образец кода. (Инструкция по скачиванию.)
Чтобы проверить этот пример, запустите приложение и отправьте HTTP-запросы со следующими параметрами:
- URL-адрес:
http://localhost:{port}/jsonpatch/jsonpatchwithmodelstate - Метод HTTP:
PATCH - заголовок:
Content-Type: application/json-patch+json; - Текст: скопируйте и вставьте один из примеров документов исправлений JSON из папки проекта JSON .
Дополнительные ресурсы
В этой статье описывается обработка запросов JSON Patch в веб-API ASP.NET Core.
Important
Стандарт JSON Patch имеет присущие риски безопасности. Эта реализация не пытается устранить эти риски безопасности. Разработчик отвечает за безопасное применение документа исправления JSON к целевому объекту. Дополнительные сведения см. в разделе "Устранение рисков безопасности ".
Установка пакета
Поддержка исправлений JSON в веб-API ASP.NET Core основана Newtonsoft.Json на пакете Microsoft.AspNetCore.Mvc.NewtonsoftJson NuGet.
Чтобы включить поддержку исправлений JSON:
Установите пакет NuGet
Microsoft.AspNetCore.Mvc.NewtonsoftJson.Вызовите процедуру AddNewtonsoftJson. Рассмотрим пример.
var builder = WebApplication.CreateBuilder(args); builder.Services.AddControllers() .AddNewtonsoftJson(); var app = builder.Build(); app.UseHttpsRedirection(); app.UseAuthorization(); app.MapControllers(); app.Run();
AddNewtonsoftJson заменяет входные и выходные средства форматирования на основе по умолчанию System.Text.Json, используемые для форматирования всего содержимого JSON. Этот метод расширения совместим со следующими методами регистрации службы MVC:
JsonPatch требует установки заголовка Content-Typeapplication/json-patch+jsonв значение .
Добавление поддержки исправления JSON при использовании System.Text.Json
Модуль форматирования входных данных на основе не поддерживает исправление System.Text.JsonJSON. Чтобы добавить поддержку исправления JSON с помощью Newtonsoft.Json, оставляя другие входные и выходные форматировщики без изменений:
Установите пакет NuGet
Microsoft.AspNetCore.Mvc.NewtonsoftJson.Обновление
Program.cs:using JsonPatchSample; using Microsoft.AspNetCore.Mvc.Formatters; var builder = WebApplication.CreateBuilder(args); builder.Services.AddControllers(options => { options.InputFormatters.Insert(0, MyJPIF.GetJsonPatchInputFormatter()); }); var app = builder.Build(); app.UseHttpsRedirection(); app.UseAuthorization(); app.MapControllers(); app.Run();using Microsoft.AspNetCore.Mvc; using Microsoft.AspNetCore.Mvc.Formatters; using Microsoft.Extensions.Options; namespace JsonPatchSample; public static class MyJPIF { public static NewtonsoftJsonPatchInputFormatter GetJsonPatchInputFormatter() { var builder = new ServiceCollection() .AddLogging() .AddMvc() .AddNewtonsoftJson() .Services.BuildServiceProvider(); return builder .GetRequiredService<IOptions<MvcOptions>>() .Value .InputFormatters .OfType<NewtonsoftJsonPatchInputFormatter>() .First(); } }
Предыдущий код создает экземпляр NewtonsoftJsonPatchInputFormatter и вставляет его в качестве первой записи в коллекцию MvcOptions.InputFormatters . Этот порядок регистрации гарантирует, что:
-
NewtonsoftJsonPatchInputFormatterобрабатывает запросы на исправление JSON. - Существующие
System.Text.Jsonвходные и форматирующие модули обрабатывают все остальные запросы и ответы JSON.
Используйте Newtonsoft.Json.JsonConvert.SerializeObject метод для сериализации JsonPatchDocument.
Метод HTTP-запроса PATCH
Методы PUT и PATCH используются для обновления существующего ресурса. Различие между ними заключается в том, что PUT заменяет весь ресурс, а PATCH указывает только изменения.
Исправление JSON
Формат JSON Patch используется для указания обновлений, применяемых к ресурсу. Документ JSON Patch содержит массив операций. Каждая операция определяет определенный тип изменения. Примеры таких изменений включают добавление элемента массива или замена значения свойства.
Например, следующие документы JSON представляют ресурс, документ исправления JSON для ресурса и результат применения операций исправления.
Пример ресурса
{
"customerName": "John",
"orders": [
{
"orderName": "Order0",
"orderType": null
},
{
"orderName": "Order1",
"orderType": null
}
]
}
Пример JSON Patch
[
{
"op": "add",
"path": "/customerName",
"value": "Barry"
},
{
"op": "add",
"path": "/orders/-",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
В приведенном выше документе JSON:
- свойство
opуказывает тип операции; - свойство
pathуказывает обновляемый элемент; - свойство
valueпредоставляет новое значение.
Ресурс после обновления
Ниже показан ресурс в том состоянии, которое он принимает после применения приведенного выше документа JSON Patch.
{
"customerName": "Barry",
"orders": [
{
"orderName": "Order0",
"orderType": null
},
{
"orderName": "Order1",
"orderType": null
},
{
"orderName": "Order2",
"orderType": null
}
]
}
Изменения, внесенные при применении документа исправления JSON к ресурсу, являются атомарными. Если любая операция в списке завершается ошибкой, операция в списке не применяется.
Синтаксис пути
Свойство Path в объекте операции содержит косые черты между уровнями. Например, "/address/zipCode".
Для указания элементов массива используются числовые индексы, начиная с нуля. Первый элемент массива addresses будет обозначаться как /addresses/0. В add конце массива используйте дефис (-) вместо номера индекса: /addresses/-
Operations
В следующей таблице перечислены поддерживаемые операции, которые определены в спецификации JSON Patch.
| Operation | Notes |
|---|---|
add |
Добавляет свойство или элемент массива. Для существующего свойства устанавливает значение. |
remove |
Удаляет свойство или элемент массива. |
replace |
Действует так же, как remove с последующим add в том же расположении. |
move |
Действует так же, как remove из источника с последующим add, в котором указаны место назначения и значение из источника. |
copy |
Действует так же, как add, в котором указаны место назначения и значение из источника. |
test |
Возвращает успешный код состояния, если значение path совпадает с предоставленным value. |
Исправление JSON в ASP.NET Core
Реализация ASP.NET Core для JSON Patch предоставляется в пакете NuGet Microsoft.AspNetCore.JsonPatch.
Код метода действия
В контроллере API есть метод действия для JSON Patch, который:
- помечен атрибутом
HttpPatch; - принимает JsonPatchDocument<TModel> обычно с указанием
[FromBody]; - вызывает ApplyTo(Object) для целевого документа, чтобы применить изменения.
Приведем пример:
[HttpPatch]
public IActionResult JsonPatchWithModelState(
[FromBody] JsonPatchDocument<Customer> patchDoc)
{
if (patchDoc != null)
{
var customer = CreateCustomer();
patchDoc.ApplyTo(customer, ModelState);
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
}
return new ObjectResult(customer);
}
else
{
return BadRequest(ModelState);
}
}
Этот код из примера приложения работает со следующей Customer моделью:
namespace JsonPatchSample.Models;
public class Customer
{
public string? CustomerName { get; set; }
public List<Order>? Orders { get; set; }
}
namespace JsonPatchSample.Models;
public class Order
{
public string OrderName { get; set; }
public string OrderType { get; set; }
}
Этот пример метода действия:
- Создает документ
Customer. - применяет изменения;
- возвращает результат в тексте ответа.
В реальном приложении код будет извлекать данные из хранилища, например из базы данных, и обновлять эту базу данных после применения исправлений.
Состояние модели
Указанный выше пример метода действия вызывает перегрузку ApplyTo, которая принимает состояние модели в качестве одного из параметров. В этом случае вы получаете в ответах сообщения об ошибках. В следующем примере показан текст ответа с кодом 400 "Неверный запрос" для операции test:
{
"Customer": [
"The current value 'John' at path 'customerName' != test value 'Nancy'."
]
}
Динамические объекты
В следующем примере метода действия показано, как применить исправление к динамическому объекту:
[HttpPatch]
public IActionResult JsonPatchForDynamic([FromBody]JsonPatchDocument patch)
{
dynamic obj = new ExpandoObject();
patch.ApplyTo(obj);
return Ok(obj);
}
Операция добавления
- Если
pathуказывает на элемент массива, новый элемент вставляется перед тем, который указан в параметреpath. - Если
pathуказывает на свойство, задается значение свойства. - Если
pathуказывает на несуществующее расположение:- Если обновляемый ресурс является динамическим объектом, добавляется свойство.
- Если обновляемый ресурс является статическим объектом, запрос завершается ошибкой.
Следующий пример документа с исправлениями устанавливает значение CustomerName и добавляет объект Order в конец массива Orders.
[
{
"op": "add",
"path": "/customerName",
"value": "Barry"
},
{
"op": "add",
"path": "/orders/-",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
Операция удаления
- Если
pathуказывает на элемент массива, элемент удаляется. - Если
pathуказывает на свойство:- Если обновляемый ресурс является динамическим объектом, свойство удаляется.
- Если ресурс для исправления является статическим объектом:
- Если свойство может принимать значения NULL, ему присваивается значение NULL.
- Если свойство не может принимать значения NULL, ему присваивается значение
default<T>.
В следующем примере наборов документов исправлений CustomerName задано значение NULL и удаляется Orders[0]:
[
{
"op": "remove",
"path": "/customerName"
},
{
"op": "remove",
"path": "/orders/0"
}
]
Операция замены
Эта операция функционально идентична операции remove с последующей add.
Следующий пример документа исправления задает значение CustomerName и заменяет Orders[0]новый Order объект:
[
{
"op": "replace",
"path": "/customerName",
"value": "Barry"
},
{
"op": "replace",
"path": "/orders/0",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
Операция перемещения
- Если
pathуказывает на элемент массива, элементfromкопируется в расположение элементаpath, а затем выполняется операцияremoveдля элементаfrom. - Если
pathуказывает на свойство, значение свойстваfromкопируется в свойствоpath, а затем выполняется операцияremoveдля свойстваfrom. - Если
pathуказывает на несуществующее свойство:- Если обновляемый ресурс является статическим объектом, запрос завершается ошибкой.
- Если исправляемый ресурс является динамическим объектом, значение
fromкопируется в местоположение, указанное параметромpath, а затем выполняется операцияremoveдля свойстваfrom.
Следующий пример документа исправления выполняет следующие действия:
- копирует значение
Orders[0].OrderNameвCustomerName; - присваивает
Orders[0].OrderNameзначение NULL; - перемещает
Orders[1]в расположение передOrders[0].
[
{
"op": "move",
"from": "/orders/0/orderName",
"path": "/customerName"
},
{
"op": "move",
"from": "/orders/1",
"path": "/orders/0"
}
]
Операция копирования
Эта операция функционально аналогична операции move без последнего шага remove.
Следующий пример документа исправления выполняет следующие действия:
- копирует значение
Orders[0].OrderNameвCustomerName; - вставляет копию
Orders[1]передOrders[0].
[
{
"op": "copy",
"from": "/orders/0/orderName",
"path": "/customerName"
},
{
"op": "copy",
"from": "/orders/1",
"path": "/orders/0"
}
]
Операция тестирования
Если значение в расположении, которое задано параметром path, отличается от предоставленного в value значения, запрос завершится ошибкой. В этом случае сбой распространяется на весь запрос PATCH, даже если все остальные операции в документе с исправлениями могли быть выполнены успешно.
Операция test традиционно используется для предотвращения обновлений при возможных конфликтах параллелизма.
Следующий пример документа с исправлениями не изменяет прежнее значение CustomerName (John), так как тест завершается ошибкой:
[
{
"op": "test",
"path": "/customerName",
"value": "Nancy"
},
{
"op": "add",
"path": "/customerName",
"value": "Barry"
}
]
Получение кода
Просмотреть или скачать образец кода. (Инструкция по скачиванию.)
Чтобы проверить этот пример, запустите приложение и отправьте HTTP-запросы со следующими параметрами:
- URL-адрес:
http://localhost:{port}/jsonpatch/jsonpatchwithmodelstate - Метод HTTP:
PATCH - заголовок:
Content-Type: application/json-patch+json; - Текст: скопируйте и вставьте один из примеров документов исправлений JSON из папки проекта JSON .
Снижение рисков безопасности
При использовании пакета Microsoft.AspNetCore.JsonPatch в реализации на основе Newtonsoft.Json, важно критически понимать и устранять потенциальные риски безопасности. В следующих разделах описаны выявленные риски безопасности, связанные с исправлением JSON, и рекомендуемые способы устранения рисков для обеспечения безопасного использования пакета.
Important
Это не исчерпывающий список угроз. Разработчики приложений должны проводить собственные проверки модели угроз, чтобы определить полный список конкретных приложений и при необходимости придумать соответствующие меры по устранению рисков. Например, приложения, которые делают коллекции доступными для операций исправления, должны учитывать потенциал для атак алгоритмической сложности, если эти операции вставляют или удаляют элементы в начале коллекции.
Выполняя комплексные модели угроз для собственных приложений и устраняя выявленные угрозы, следуя приведенным ниже рекомендациям, потребители этих пакетов могут интегрировать функции исправления JSON в свои приложения, минимизируя риски безопасности.
Отказ в обслуживании (DoS) через увеличение памяти
-
Сценарий: вредоносный клиент отправляет
copyоперацию, которая дублирует графы больших объектов несколько раз, что приводит к чрезмерному потреблению памяти. - Влияние: потенциальные условия исчерпанияOf-Memory (OOM), приводящие к сбоям в работе службы.
-
Mitigation:
- Перед вызовом
ApplyToпроверьте входящие JSON-патчи на соответствие размера и структуры. - Проверка должна быть конкретной для приложения, но пример проверки может выглядеть следующим образом:
- Перед вызовом
public void Validate(JsonPatchDocument patch)
{
// This is just an example. It's up to the developer to make sure that
// this case is handled properly, based on the app needs.
if (patch.Operations.Where(op => op.OperationType == OperationType.Copy).Count()
> MaxCopyOperationsCount)
{
throw new InvalidOperationException();
}
}
Подверсия бизнес-логики
- Сценарий. Операции исправления могут управлять полями с неявными инвариантными (например, внутренними флагами, идентификаторами или вычисляемыми полями), нарушая бизнес-ограничения.
- Влияние: проблемы целостности данных и непреднамеренное поведение приложения.
-
Mitigation:
- Используйте объекты POCO с явно определенными свойствами, безопасными для изменения.
- Избегайте предоставления конфиденциальных или критически важных свойств безопасности в целевом объекте.
- Если объект POCO не используется, проверьте исправленный объект после применения операций, чтобы убедиться, что бизнес-правила и инварианты не нарушаются.
Проверка подлинности и авторизация
- Сценарий. Неавторизованные или несанкционированные клиенты отправляют вредоносные запросы на исправление JSON.
- Влияние: несанкционированный доступ к изменению конфиденциальных данных или нарушению поведения приложения.
-
Mitigation:
- Защита конечных точек, принимаюющих запросы на исправление JSON с помощью надлежащих механизмов проверки подлинности и авторизации.
- Ограничить доступ к доверенным клиентам или пользователям с соответствующими разрешениями.
Дополнительные ресурсы
В этой статье описывается обработка запросов JSON Patch в веб-API ASP.NET Core.
Important
Стандарт JSON Patch имеет присущие риски безопасности. Поскольку эти риски присущи стандарту исправлений JSON, эта реализация не пытается снизить риски, связанные с безопасностью. Разработчик отвечает за безопасное применение документа исправления JSON к целевому объекту. Дополнительные сведения см. в разделе "Устранение рисков безопасности ".
Установка пакета
Чтобы включить поддержку исправлений JSON в приложении, выполните следующие действия.
Установите пакет NuGet
Microsoft.AspNetCore.Mvc.NewtonsoftJson.Обновите метод проекта
Startup.ConfigureServicesдля вызова AddNewtonsoftJson. Рассмотрим пример.services .AddControllersWithViews() .AddNewtonsoftJson();
AddNewtonsoftJson совместим с методами регистрации службы MVC:
Исправление JSON, AddNewtonsoftJson и System.Text.Json
AddNewtonsoftJson
System.Text.Jsonзаменяет средства форматирования входных и выходных данных на основе, используемые для форматирования всего содержимого JSON. Чтобы добавить поддержку исправления JSON с помощью Newtonsoft.Json, оставляя другие форматировщики без изменений, обновите метод проекта Startup.ConfigureServices следующим образом:
public void ConfigureServices(IServiceCollection services)
{
services.AddControllersWithViews(options =>
{
options.InputFormatters.Insert(0, GetJsonPatchInputFormatter());
});
}
private static NewtonsoftJsonPatchInputFormatter GetJsonPatchInputFormatter()
{
var builder = new ServiceCollection()
.AddLogging()
.AddMvc()
.AddNewtonsoftJson()
.Services.BuildServiceProvider();
return builder
.GetRequiredService<IOptions<MvcOptions>>()
.Value
.InputFormatters
.OfType<NewtonsoftJsonPatchInputFormatter>()
.First();
}
Для предыдущего кода требуется Microsoft.AspNetCore.Mvc.NewtonsoftJson пакет и следующие using инструкции:
using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.AspNetCore.Mvc;
using Microsoft.AspNetCore.Mvc.Formatters;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.Extensions.Options;
using System.Linq;
Используйте Newtonsoft.Json.JsonConvert.SerializeObject метод для сериализации JsonPatchDocument.
Метод HTTP-запроса PATCH
Методы PUT и PATCH используются для обновления существующего ресурса. Различие между ними заключается в том, что PUT заменяет весь ресурс, а PATCH указывает только изменения.
Исправление JSON
Формат JSON Patch используется для указания обновлений, применяемых к ресурсу. Документ JSON Patch содержит массив операций. Каждая операция определяет определенный тип изменения. Примеры таких изменений включают добавление элемента массива или замена значения свойства.
Например, следующие документы JSON представляют ресурс, документ исправления JSON для ресурса и результат применения операций исправления.
Пример ресурса
{
"customerName": "John",
"orders": [
{
"orderName": "Order0",
"orderType": null
},
{
"orderName": "Order1",
"orderType": null
}
]
}
Пример JSON Patch
[
{
"op": "add",
"path": "/customerName",
"value": "Barry"
},
{
"op": "add",
"path": "/orders/-",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
В приведенном выше документе JSON:
- свойство
opуказывает тип операции; - свойство
pathуказывает обновляемый элемент; - свойство
valueпредоставляет новое значение.
Ресурс после обновления
Ниже показан ресурс в том состоянии, которое он принимает после применения приведенного выше документа JSON Patch.
{
"customerName": "Barry",
"orders": [
{
"orderName": "Order0",
"orderType": null
},
{
"orderName": "Order1",
"orderType": null
},
{
"orderName": "Order2",
"orderType": null
}
]
}
Изменения, внесенные при применении документа исправления JSON к ресурсу, являются атомарными. Если любая операция в списке завершается ошибкой, операция в списке не применяется.
Синтаксис пути
Свойство Path в объекте операции содержит косые черты между уровнями. Например, "/address/zipCode".
Для указания элементов массива используются числовые индексы, начиная с нуля. Первый элемент массива addresses будет обозначаться как /addresses/0. В add конце массива используйте дефис (-) вместо номера индекса: /addresses/-
Operations
В следующей таблице перечислены поддерживаемые операции, которые определены в спецификации JSON Patch.
| Operation | Notes |
|---|---|
add |
Добавляет свойство или элемент массива. Для существующего свойства устанавливает значение. |
remove |
Удаляет свойство или элемент массива. |
replace |
Действует так же, как remove с последующим add в том же расположении. |
move |
Действует так же, как remove из источника с последующим add, в котором указаны место назначения и значение из источника. |
copy |
Действует так же, как add, в котором указаны место назначения и значение из источника. |
test |
Возвращает успешный код состояния, если значение path совпадает с предоставленным value. |
Исправление JSON в ASP.NET Core
Реализация ASP.NET Core для JSON Patch предоставляется в пакете NuGet Microsoft.AspNetCore.JsonPatch.
Код метода действия
В контроллере API есть метод действия для JSON Patch, который:
- помечен атрибутом
HttpPatch; - принимает
JsonPatchDocument<T>обычно с указанием[FromBody]; - вызывает
ApplyToдля целевого документа, чтобы применить изменения.
Приведем пример:
[HttpPatch]
public IActionResult JsonPatchWithModelState(
[FromBody] JsonPatchDocument<Customer> patchDoc)
{
if (patchDoc != null)
{
var customer = CreateCustomer();
patchDoc.ApplyTo(customer, ModelState);
if (!ModelState.IsValid)
{
return BadRequest(ModelState);
}
return new ObjectResult(customer);
}
else
{
return BadRequest(ModelState);
}
}
Этот код из примера приложения работает со следующей Customer моделью:
using System.Collections.Generic;
namespace JsonPatchSample.Models
{
public class Customer
{
public string CustomerName { get; set; }
public List<Order> Orders { get; set; }
}
}
namespace JsonPatchSample.Models
{
public class Order
{
public string OrderName { get; set; }
public string OrderType { get; set; }
}
}
Этот пример метода действия:
- Создает документ
Customer. - применяет изменения;
- возвращает результат в тексте ответа.
В реальном приложении код будет извлекать данные из хранилища, например из базы данных, и обновлять эту базу данных после применения исправлений.
Состояние модели
Указанный выше пример метода действия вызывает перегрузку ApplyTo, которая принимает состояние модели в качестве одного из параметров. В этом случае вы получаете в ответах сообщения об ошибках. В следующем примере показан текст ответа с кодом 400 "Неверный запрос" для операции test:
{
"Customer": [
"The current value 'John' at path 'customerName' is not equal to the test value 'Nancy'."
]
}
Динамические объекты
В следующем примере метода действия показано, как применить исправление к динамическому объекту:
[HttpPatch]
public IActionResult JsonPatchForDynamic([FromBody]JsonPatchDocument patch)
{
dynamic obj = new ExpandoObject();
patch.ApplyTo(obj);
return Ok(obj);
}
Операция добавления
- Если
pathуказывает на элемент массива, новый элемент вставляется перед тем, который указан в параметреpath. - Если
pathуказывает на свойство, задается значение свойства. - Если
pathуказывает на несуществующее расположение:- Если обновляемый ресурс является динамическим объектом, добавляется свойство.
- Если обновляемый ресурс является статическим объектом, запрос завершается ошибкой.
Следующий пример документа с исправлениями устанавливает значение CustomerName и добавляет объект Order в конец массива Orders.
[
{
"op": "add",
"path": "/customerName",
"value": "Barry"
},
{
"op": "add",
"path": "/orders/-",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
Операция удаления
- Если
pathуказывает на элемент массива, элемент удаляется. - Если
pathуказывает на свойство:- Если обновляемый ресурс является динамическим объектом, свойство удаляется.
- Если ресурс для исправления является статическим объектом:
- Если свойство может принимать значения NULL, ему присваивается значение NULL.
- Если свойство не может принимать значения NULL, ему присваивается значение
default<T>.
В следующем примере наборов документов исправлений CustomerName задано значение NULL и удаляется Orders[0]:
[
{
"op": "remove",
"path": "/customerName"
},
{
"op": "remove",
"path": "/orders/0"
}
]
Операция замены
Эта операция функционально идентична операции remove с последующей add.
Следующий пример документа исправления задает значение CustomerName и заменяет Orders[0]новый Order объект:
[
{
"op": "replace",
"path": "/customerName",
"value": "Barry"
},
{
"op": "replace",
"path": "/orders/0",
"value": {
"orderName": "Order2",
"orderType": null
}
}
]
Операция перемещения
- Если
pathуказывает на элемент массива, элементfromкопируется в расположение элементаpath, а затем выполняется операцияremoveдля элементаfrom. - Если
pathуказывает на свойство, значение свойстваfromкопируется в свойствоpath, а затем выполняется операцияremoveдля свойстваfrom. - Если
pathуказывает на несуществующее свойство:- Если обновляемый ресурс является статическим объектом, запрос завершается ошибкой.
- Если исправляемый ресурс является динамическим объектом, значение
fromкопируется в местоположение, указанное параметромpath, а затем выполняется операцияremoveдля свойстваfrom.
Следующий пример документа исправления выполняет следующие действия:
- копирует значение
Orders[0].OrderNameвCustomerName; - присваивает
Orders[0].OrderNameзначение NULL; - перемещает
Orders[1]в расположение передOrders[0].
[
{
"op": "move",
"from": "/orders/0/orderName",
"path": "/customerName"
},
{
"op": "move",
"from": "/orders/1",
"path": "/orders/0"
}
]
Операция копирования
Эта операция функционально аналогична операции move без последнего шага remove.
Следующий пример документа исправления выполняет следующие действия:
- копирует значение
Orders[0].OrderNameвCustomerName; - вставляет копию
Orders[1]передOrders[0].
[
{
"op": "copy",
"from": "/orders/0/orderName",
"path": "/customerName"
},
{
"op": "copy",
"from": "/orders/1",
"path": "/orders/0"
}
]
Операция тестирования
Если значение в расположении, которое задано параметром path, отличается от предоставленного в value значения, запрос завершится ошибкой. В этом случае сбой распространяется на весь запрос PATCH, даже если все остальные операции в документе с исправлениями могли быть выполнены успешно.
Операция test традиционно используется для предотвращения обновлений при возможных конфликтах параллелизма.
Следующий пример документа с исправлениями не изменяет прежнее значение CustomerName (John), так как тест завершается ошибкой:
[
{
"op": "test",
"path": "/customerName",
"value": "Nancy"
},
{
"op": "add",
"path": "/customerName",
"value": "Barry"
}
]
Получение кода
Просмотреть или скачать образец кода. (Инструкция по скачиванию.)
Чтобы проверить этот пример, запустите приложение и отправьте HTTP-запросы со следующими параметрами:
- URL-адрес:
http://localhost:{port}/jsonpatch/jsonpatchwithmodelstate - Метод HTTP:
PATCH - заголовок:
Content-Type: application/json-patch+json; - Текст: скопируйте и вставьте один из примеров документов исправлений JSON из папки проекта JSON .