Возвращаемся к проекту на C#.
Мы остановились на списке материалов.
Реализуем получение списка материалов выбранного продукта с помощью HTTP-запроса к АПИ, которое мы реализовали в прошлой теме.
Напоминаю, GET-запрос на локальный компьютер:
GET http://localhost:8080/Material?product_id=1
Authorization: Basic esmirnov 111103
На C# нам надо решить две задачи:
Для HTTP-запросов будем использовать встроенный класс HttpClient
var client = new HttpClient();
В АПИ мы использовали "базовую" авторизацию. В C# я не нашёл встроенных библиотек для облегчения формирования заголовка для "базовой" авторизации
Формируем base64-кодированную строку:
var basic = Convert.ToBase64String(
ASCIIEncoding.ASCII.GetBytes("esmirnov:111103"));
И добавляем заголовок нашему http-клиенту:
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
Теперь можно запрашивать данные.
У HttpClient все методы асинхронные, нужно понимать как работают async/await. Я для облегчения вашей работы нарисовал синхронную реализацию:
// в параметрах URL
private string GetString(string url){
var basic = Convert.ToBase64String(
ASCIIEncoding.ASCII.GetBytes("esmirnov:111103"));
var client = new HttpClient();
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
return client.GetStringAsync(url).Result;
}
Сначала допишем интерфейс нашего поставщика данных:
interface IDataProvider
{
...
IEnumerable<MaterialTC> GetMaterials(int ProductId);
}
Здесь я класс назвал не Material, а MaterialTC, потому что АПИ возвращает не весь объект Material, а только название и количество (TitleCount).
И реализуем его:
[DataContract]
internal class MaterialTC
{
[DataMember]
public string Title { get; set; }
[DataMember]
public int Count { get; set; }
}
// В ответе у нас не сразу массив материалов, а два вложенных объекта, поэтому надо их тоже описать
[DataContract]
internal class Notice
{
[DataMember]
public Material[] data { get; set; }
}
[DataContract]
internal class Answer
{
[DataMember]
public Notice notice { get; set; }
}
public IEnumerable<MaterialTC> GetMaterials(int ProductId) {
var result = new List<MaterialTC>();
var resp = GetString($"http://localhost:8080/Material?product_id={ProductId}");
var Serializer = new DataContractJsonSerializer(typeof(Answer));
using (var sr = new StreamReader(new MemoryStream(Encoding.UTF8.GetBytes(resp))))
{
var answer = (Answer)Serializer.ReadObject(sr.BaseStream);
foreach (MaterialTC material in answer.notice.data)
{
result.Add(material);
}
}
return result;
}
Оказывается на WorldSkills можно использовать не только "голый" .NET Framework, но и библиотеки из других компонентов Visual Studio.
В пространстве имён System.Web.Script.Serialization есть класс JavaScriptSerializer, который выглядит попроще чем классическая реализация:
В пакет разработки C# не входит библиотека System.Web.Extensions (в которой и находится System.Web.Script.Serialization). Нужно в "Обозревателе решений" добавить в "Ссылки" библиотеку Сборки -> Платформа -> System.Web.Extensions
// целевые классы нам по прежнему нужны, но уже без всяких аннотаций
internal class MaterialTC
{
public string Title { get; set; }
public int Count { get; set; }
}
internal class Notice
{
public Material[] data;
}
internal class Answer
{
public Notice notice;
}
// в месте, где нам нужно распарсить JSON создаем сериализатор и разбираем строку
var serializer = new JavaScriptSerializer();
var answer = serializer.Deserialize<Answer>("тут ваша JSON-строка");
// и ВСЁ
Для удаления данных в REST API используется http-метод DELETE. (Но никто не запрещает использовать и другие методы)
Запрос в REST клиенте выглядит примерно так, идентификатор удаляемой записи передаём параметрами:
DELETE {{url}}/Product?id=131
^^^^^^^
Authorization: Basic esmirnov 111103
Строку запроса, надеюсь, сформируете сами.
Для вызова http-метода DELETE используется метод DeleteAsync:
var basic = Convert.ToBase64String(
ASCIIEncoding.ASCII.GetBytes("esmirnov:111103"));
var client = new HttpClient();
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
client.DeleteAsync($"http://localhost:8080/Product?id={id}").Result;
В проекте C# нарисуйте форму добавления продажи (в окне продукции)
В DataProvider добавьте метод AddProductSale, который на входе получает экземпляр класса ProductSale и реализуйте отправку POST-запроса в локальный PHP-сервер, который добавит эту продажу в соответствующую таблицу
// сначала запихиваем объект в JSON-строку.
var jsonString = serializer.Serialize(NewProductSale);
// создаём контент для запроса
var json = new StringContent(jsonString, Encoding.UTF8, "application/json");
// и вызываем POST-запрос
var client = new HttpClient();
// не забываем добавить авторизацию
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Basic", basic);
var result = client.PostAsync("http://localhost:8080/ProductSale", json).Result;
// Console.WriteLine(result.Content.ReadAsStringAsync().Result);