[{"data":1,"prerenderedAt":11950},["ShallowReactive",2],{"navigation_docs":3,"-csharp-network-programming-07a-http-listener":3302,"-csharp-network-programming-07a-http-listener-surround":11945},[4,1707,1896,2350,2531,2572,2779,2901,2951,3008,3042,3168,3245,3298],{"title":5,"icon":6,"path":7,"stem":8,"children":9},"C#","i-devicon-csharp","\u002Fcsharp","01.csharp",[10,13,60,90,120,202,219,253,379,404,457,650,1364,1654,1703],{"title":11,"path":7,"stem":12},"C# та .NET","01.csharp\u002Findex",{"title":14,"icon":15,"path":16,"stem":17,"children":18,"page":59},"Fundamentals","i-lucide-book-open","\u002Fcsharp\u002Ffundamentals","01.csharp\u002F01.fundamentals",[19,23,27,31,35,39,43,47,51,55],{"title":20,"path":21,"stem":22},"Вступ до екосистеми .NET","\u002Fcsharp\u002Ffundamentals\u002Fintroduction-to-ecosystem","01.csharp\u002F01.fundamentals\u002F01.introduction-to-ecosystem",{"title":24,"path":25,"stem":26},"Структура програми на C#","\u002Fcsharp\u002Ffundamentals\u002Fprogram-structure","01.csharp\u002F01.fundamentals\u002F02.program-structure",{"title":28,"path":29,"stem":30},"Змінні та Типи Даних","\u002Fcsharp\u002Ffundamentals\u002Fvariables-data-types","01.csharp\u002F01.fundamentals\u002F03.variables-data-types",{"title":32,"path":33,"stem":34},"Масиви","\u002Fcsharp\u002Ffundamentals\u002Farrays","01.csharp\u002F01.fundamentals\u002F04.arrays",{"title":36,"path":37,"stem":38},"Strings & Text Handling","\u002Fcsharp\u002Ffundamentals\u002Fstrings-text-handling","01.csharp\u002F01.fundamentals\u002F05.strings-text-handling",{"title":40,"path":41,"stem":42},"Дати і Час","\u002Fcsharp\u002Ffundamentals\u002Fdates-time-handling","01.csharp\u002F01.fundamentals\u002F06.dates-time-handling",{"title":44,"path":45,"stem":46},"Потік Керування","\u002Fcsharp\u002Ffundamentals\u002Fcontrol-flow","01.csharp\u002F01.fundamentals\u002F07.control-flow",{"title":48,"path":49,"stem":50},"Методи","\u002Fcsharp\u002Ffundamentals\u002Fmethods","01.csharp\u002F01.fundamentals\u002F08.methods",{"title":52,"path":53,"stem":54},"Основи Відлагодження","\u002Fcsharp\u002Ffundamentals\u002Fdebugging-basics","01.csharp\u002F01.fundamentals\u002F09.debugging-basics",{"title":56,"path":57,"stem":58},"Інтерактивна Консоль (Classic)","\u002Fcsharp\u002Ffundamentals\u002Finteractive-console","01.csharp\u002F01.fundamentals\u002F10.interactive-console",false,{"title":61,"icon":62,"path":63,"stem":64,"children":65,"page":59},"OOP","i-lucide-box","\u002Fcsharp\u002Foop","01.csharp\u002F02.oop",[66,70,74,78,82,86],{"title":67,"path":68,"stem":69},"Package Management (Управління Пакетами)","\u002Fcsharp\u002Foop\u002Fpackage-management","01.csharp\u002F02.oop\u002F01.package-management",{"title":71,"path":72,"stem":73},"Класи та Об'єкти","\u002Fcsharp\u002Foop\u002Fclasses-objects","01.csharp\u002F02.oop\u002F02.classes-objects",{"title":75,"path":76,"stem":77},"Властивості та Поля","\u002Fcsharp\u002Foop\u002Fproperties-fields","01.csharp\u002F02.oop\u002F03.properties-fields",{"title":79,"path":80,"stem":81},"Стовпи ООП","\u002Fcsharp\u002Foop\u002Foop-pillars","01.csharp\u002F02.oop\u002F04.oop-pillars",{"title":83,"path":84,"stem":85},"Advanced Types","\u002Fcsharp\u002Foop\u002Fadvanced-types","01.csharp\u002F02.oop\u002F05.advanced-types",{"title":87,"path":88,"stem":89},"Namespaces (Простори Імен)","\u002Fcsharp\u002Foop\u002Fnamespaces","01.csharp\u002F02.oop\u002F06.namespaces",{"title":91,"icon":92,"path":93,"stem":94,"children":95,"page":59},"Advanced Core","i-lucide-zap","\u002Fcsharp\u002Fadvanced-core","01.csharp\u002F03.advanced-core",[96,100,104,108,112,116],{"title":97,"path":98,"stem":99},"Generics (Узагальнення)","\u002Fcsharp\u002Fadvanced-core\u002Fgenerics","01.csharp\u002F03.advanced-core\u002F01.generics",{"title":101,"path":102,"stem":103},"Делегати, Події та Лямбда-вирази","\u002Fcsharp\u002Fadvanced-core\u002Fdelegates-events-lambdas","01.csharp\u002F03.advanced-core\u002F02.delegates-events-lambdas",{"title":105,"path":106,"stem":107},"Interfaces Deep Dive (Інтерфейси: Поглиблений Розгляд)","\u002Fcsharp\u002Fadvanced-core\u002Finterfaces-deep-dive","01.csharp\u002F03.advanced-core\u002F03.interfaces-deep-dive",{"title":109,"path":110,"stem":111},"Обробка Винятків","\u002Fcsharp\u002Fadvanced-core\u002Fexception-handling","01.csharp\u002F03.advanced-core\u002F04.exception-handling",{"title":113,"path":114,"stem":115},"Pattern Matching","\u002Fcsharp\u002Fadvanced-core\u002Fpattern-matching","01.csharp\u002F03.advanced-core\u002F05.pattern-matching",{"title":117,"path":118,"stem":119},"Додаткові Можливості C#","\u002Fcsharp\u002Fadvanced-core\u002Fadditional-features","01.csharp\u002F03.advanced-core\u002F06.additional-features",{"title":121,"icon":122,"path":123,"stem":124,"children":125,"page":59},"Architecture Best Practices","i-lucide-building-2","\u002Fcsharp\u002Farchitecture-best-practices","01.csharp\u002F04.architecture-best-practices",[126,130,149,153,157,161,165,169],{"title":127,"path":128,"stem":129},"Software Design Principles (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fsoftware-design-principles","01.csharp\u002F04.architecture-best-practices\u002F01.software-design-principles",{"title":131,"icon":132,"path":133,"stem":134,"children":135,"page":59},"Design Patterns","i-lucide-folder","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns",[136],{"title":137,"icon":132,"path":138,"stem":139,"children":140,"page":59},"Creational","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational",[141,145],{"title":142,"path":143,"stem":144},"Singleton (Одинак)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fsingleton","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F01.singleton",{"title":146,"path":147,"stem":148},"Builder (Будівельник)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdesign-patterns\u002Fcreational\u002Fbuilder","01.csharp\u002F04.architecture-best-practices\u002F02.design-patterns\u002Fcreational\u002F02.builder",{"title":150,"path":151,"stem":152},"Building Professional CLIs","\u002Fcsharp\u002Farchitecture-best-practices\u002Fbuilding-professional-clis","01.csharp\u002F04.architecture-best-practices\u002F03.building-professional-clis",{"title":154,"path":155,"stem":156},"Validation & Flow Control","\u002Fcsharp\u002Farchitecture-best-practices\u002Fvalidation-flow-control","01.csharp\u002F04.architecture-best-practices\u002F04.validation-flow-control",{"title":158,"path":159,"stem":160},"The Modern .NET Host (Microsoft.Extensions)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fmodern-dotnet-host","01.csharp\u002F04.architecture-best-practices\u002F05.modern-dotnet-host",{"title":162,"path":163,"stem":164},"Data Mapper: Repository та DAO патерни (Частина 1)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part1","01.csharp\u002F04.architecture-best-practices\u002F06.data-mapper-part1",{"title":166,"path":167,"stem":168},"Data Mapper: Repository та DAO патерни (Частина 2)","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdata-mapper-part2","01.csharp\u002F04.architecture-best-practices\u002F07.data-mapper-part2",{"title":170,"icon":132,"path":171,"stem":172,"children":173,"page":59},"Di Ioc","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc",[174,178,182,186,190,194,198],{"title":175,"path":176,"stem":177},"Проблема залежностей та Інверсія Контролю","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fthe-dependency-problem","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F01.the-dependency-problem",{"title":179,"path":180,"stem":181},"Будуємо власний Service Container","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fbuild-your-own-container","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F02.build-your-own-container",{"title":183,"path":184,"stem":185},"Service Locator: Паттерн та Анти-паттерн","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-locator-pattern","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F03.service-locator-pattern",{"title":187,"path":188,"stem":189},"Паттерни Dependency Injection","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdependency-injection-patterns","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F04.dependency-injection-patterns",{"title":191,"path":192,"stem":193},"Microsoft DI: IServiceCollection та IServiceProvider","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fmicrosoft-di-deep-dive","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F05.microsoft-di-deep-dive",{"title":195,"path":196,"stem":197},"Service Lifetimes та Scopes","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fservice-lifetimes-and-scopes","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F06.service-lifetimes-and-scopes",{"title":199,"path":200,"stem":201},"DI Анти-паттерни та Найкращі Практики","\u002Fcsharp\u002Farchitecture-best-practices\u002Fdi-ioc\u002Fdi-anti-patterns-and-best-practices","01.csharp\u002F04.architecture-best-practices\u002F08.di-ioc\u002F07.di-anti-patterns-and-best-practices",{"title":203,"icon":132,"path":204,"stem":205,"children":206,"page":59},"Standard Library","\u002Fcsharp\u002Fstandard-library","01.csharp\u002F05.standard-library",[207,211,215],{"title":208,"path":209,"stem":210},"Collections (Колекції)","\u002Fcsharp\u002Fstandard-library\u002Fcollections","01.csharp\u002F05.standard-library\u002F01.collections",{"title":212,"path":213,"stem":214},"High Performance Types (Високопродуктивні Типи)","\u002Fcsharp\u002Fstandard-library\u002Fhigh-performance-types","01.csharp\u002F05.standard-library\u002F02.high-performance-types",{"title":216,"path":217,"stem":218},"LINQ (Language Integrated Query)","\u002Fcsharp\u002Fstandard-library\u002Flinq","01.csharp\u002F05.standard-library\u002F03.linq",{"title":220,"icon":221,"path":222,"stem":223,"children":224,"page":59},"System Internals Concurrency","i-lucide-server","\u002Fcsharp\u002Fsystem-internals-concurrency","01.csharp\u002F06.system-internals-concurrency",[225,229,233,237,241,245,249],{"title":226,"path":227,"stem":228},"Memory Management","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmemory-management","01.csharp\u002F06.system-internals-concurrency\u002F01.memory-management",{"title":230,"path":231,"stem":232},"Reflection API: System.Type та Метадані","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Freflection-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F02.reflection-fundamentals",{"title":234,"path":235,"stem":236},"Attributes та Dynamic Language Runtime","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fattributes-dynamic","01.csharp\u002F06.system-internals-concurrency\u002F03.attributes-dynamic",{"title":238,"path":239,"stem":240},"Expression Trees: Швидка Альтернатива Рефлексії","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fexpression-trees-compiled","01.csharp\u002F06.system-internals-concurrency\u002F04.expression-trees-compiled",{"title":242,"path":243,"stem":244},"Source Generators: Compile-Time Code Generation","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsource-generators","01.csharp\u002F06.system-internals-concurrency\u002F05.source-generators",{"title":246,"path":247,"stem":248},"Multithreading Fundamentals","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fmultithreading-fundamentals","01.csharp\u002F06.system-internals-concurrency\u002F06.multithreading-fundamentals",{"title":250,"path":251,"stem":252},"Synchronization Primitives","\u002Fcsharp\u002Fsystem-internals-concurrency\u002Fsynchronization-primitives","01.csharp\u002F06.system-internals-concurrency\u002F07.synchronization-primitives",{"title":254,"icon":255,"path":256,"stem":257,"children":258,"page":59},"System Programming Windows","i-lucide-cpu","\u002Fcsharp\u002Fsystem-programming-windows","01.csharp\u002F07.system-programming-windows",[259,263,267,271,275,279,283,287,291,295,299,303,307,311,315,319,323,327,331,335,339,343,347,351,355,359,363,367,371,375],{"title":260,"path":261,"stem":262},"Як Працює Операційна Система","\u002Fcsharp\u002Fsystem-programming-windows\u002Fhow-os-works","01.csharp\u002F07.system-programming-windows\u002F01.how-os-works",{"title":264,"path":265,"stem":266},"Процеси в .NET — API та Запуск","\u002Fcsharp\u002Fsystem-programming-windows\u002Fprocesses-in-dotnet","01.csharp\u002F07.system-programming-windows\u002F02.processes-in-dotnet",{"title":268,"path":269,"stem":270},"Процеси в .NET — IPC та Міжпроцесна Комунікація","\u002Fcsharp\u002Fsystem-programming-windows\u002F02a.processes-ipc","01.csharp\u002F07.system-programming-windows\u002F02a.processes-ipc",{"title":272,"path":273,"stem":274},"Application Domains та Збірки — AppDomain і AssemblyLoadContext","\u002Fcsharp\u002Fsystem-programming-windows\u002Fappdomains-assemblies","01.csharp\u002F07.system-programming-windows\u002F03.appdomains-assemblies",{"title":276,"path":277,"stem":278},"Application Domains та Збірки — Plug-in Система з Hot-Reload","\u002Fcsharp\u002Fsystem-programming-windows\u002F03a.appdomains-plugin-system","01.csharp\u002F07.system-programming-windows\u002F03a.appdomains-plugin-system",{"title":280,"path":281,"stem":282},"Потоки — Основи та API Thread","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-fundamentals","01.csharp\u002F07.system-programming-windows\u002F04.thread-fundamentals",{"title":284,"path":285,"stem":286},"Потоки — Lifecycle, Пріоритети та Безпечне Завершення","\u002Fcsharp\u002Fsystem-programming-windows\u002F04a.thread-lifecycle-priorities","01.csharp\u002F07.system-programming-windows\u002F04a.thread-lifecycle-priorities",{"title":288,"path":289,"stem":290},"Проблеми Спільного Стану — Race Condition та Data Race","\u002Fcsharp\u002Fsystem-programming-windows\u002Fshared-state-problems","01.csharp\u002F07.system-programming-windows\u002F05.shared-state-problems",{"title":292,"path":293,"stem":294},"Проблеми Спільного Стану — Memory Model та volatile","\u002Fcsharp\u002Fsystem-programming-windows\u002F05a.shared-state-memory-model","01.csharp\u002F07.system-programming-windows\u002F05a.shared-state-memory-model",{"title":296,"path":297,"stem":298},"Синхронізація — Monitor, lock та еволюція примітивів","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-fundamentals","01.csharp\u002F07.system-programming-windows\u002F06.synchronization-fundamentals",{"title":300,"path":301,"stem":302},"Синхронізація — Наскрізний Приклад та Deadlock Detection","\u002Fcsharp\u002Fsystem-programming-windows\u002F06a.synchronization-walkthrough","01.csharp\u002F07.system-programming-windows\u002F06a.synchronization-walkthrough",{"title":304,"path":305,"stem":306},"Синхронізація — Mutex, Semaphore та Event-Based Primitives","\u002Fcsharp\u002Fsystem-programming-windows\u002Fsynchronization-advanced","01.csharp\u002F07.system-programming-windows\u002F07.synchronization-advanced",{"title":308,"path":309,"stem":310},"Синхронізація — Interlocked, Volatile та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002F07a.synchronization-advanced-walkthrough","01.csharp\u002F07.system-programming-windows\u002F07a.synchronization-advanced-walkthrough",{"title":312,"path":313,"stem":314},"Interlocked, CAS та Lock-Free Структури","\u002Fcsharp\u002Fsystem-programming-windows\u002Finterlocked-cas-lockfree","01.csharp\u002F07.system-programming-windows\u002F08.interlocked-cas-lockfree",{"title":316,"path":317,"stem":318},"Volatile, Memory Model та Spinning","\u002Fcsharp\u002Fsystem-programming-windows\u002F08a.volatile-memory-model","01.csharp\u002F07.system-programming-windows\u002F08a.volatile-memory-model",{"title":320,"path":321,"stem":322},"ThreadPool — Пул Потоків для Ефективного Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fthread-pool","01.csharp\u002F07.system-programming-windows\u002F09.thread-pool",{"title":324,"path":325,"stem":326},"ThreadPool — Просунуті Сценарії та Внутрішня Будова","\u002Fcsharp\u002Fsystem-programming-windows\u002F09a.thread-pool-advanced","01.csharp\u002F07.system-programming-windows\u002F09a.thread-pool-advanced",{"title":328,"path":329,"stem":330},"Concurrent та Immutable Collections","\u002Fcsharp\u002Fsystem-programming-windows\u002Fconcurrent-collections","01.csharp\u002F07.system-programming-windows\u002F10.concurrent-collections",{"title":332,"path":333,"stem":334},"TPL, Task та Композиція — Від Thread до Task","\u002Fcsharp\u002Fsystem-programming-windows\u002Ftpl-parallel-plinq","01.csharp\u002F07.system-programming-windows\u002F11.tpl-parallel-plinq",{"title":336,"path":337,"stem":338},"Parallel Class та PLINQ — Data Parallelism","\u002Fcsharp\u002Fsystem-programming-windows\u002F11a.tpl-parallel-plinq-advanced","01.csharp\u002F07.system-programming-windows\u002F11a.tpl-parallel-plinq-advanced",{"title":340,"path":341,"stem":342},"Async\u002FAwait — Фундамент Асинхронного Програмування","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-fundamentals","01.csharp\u002F07.system-programming-windows\u002F12.async-fundamentals",{"title":344,"path":345,"stem":346},"SynchronizationContext та ConfigureAwait — Контекст Виконання","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-context-configureawait","01.csharp\u002F07.system-programming-windows\u002F13.async-context-configureawait",{"title":348,"path":349,"stem":350},"Async — Просунуті Паттерни","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-advanced","01.csharp\u002F07.system-programming-windows\u002F14.async-advanced",{"title":352,"path":353,"stem":354},"System.Threading.Channels — Async Producer-Consumer","\u002Fcsharp\u002Fsystem-programming-windows\u002Fchannels","01.csharp\u002F07.system-programming-windows\u002F15.channels",{"title":356,"path":357,"stem":358},"Асинхронна Синхронізація","\u002Fcsharp\u002Fsystem-programming-windows\u002Fasync-synchronization","01.csharp\u002F07.system-programming-windows\u002F16.async-synchronization",{"title":360,"path":361,"stem":362},"Unsafe Code та Вказівники","\u002Fcsharp\u002Fsystem-programming-windows\u002Funsafe-code","01.csharp\u002F07.system-programming-windows\u002F17.unsafe-code",{"title":364,"path":365,"stem":366},"P\u002FInvoke та Windows API — Міст між .NET та Native Code","\u002Fcsharp\u002Fsystem-programming-windows\u002Fpinvoke-winapi","01.csharp\u002F07.system-programming-windows\u002F18.pinvoke-winapi",{"title":368,"path":369,"stem":370},"Реєстр Windows — Центральна База Конфігурації Системи","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-registry","01.csharp\u002F07.system-programming-windows\u002F19.windows-registry",{"title":372,"path":373,"stem":374},"Windows Hooks, Hotkeys та Services — Глибока Інтеграція з ОС","\u002Fcsharp\u002Fsystem-programming-windows\u002Fwindows-hooks-services","01.csharp\u002F07.system-programming-windows\u002F20.windows-hooks-services",{"title":376,"path":377,"stem":378},"Системне Програмування C# (Windows) — 07.system-programming-windows","\u002Fcsharp\u002Fsystem-programming-windows\u002Fimplementation_plan","01.csharp\u002F07.system-programming-windows\u002Fimplementation_plan",{"title":380,"icon":132,"path":381,"stem":382,"children":383,"page":59},"Io","\u002Fcsharp\u002Fio","01.csharp\u002F08.io",[384,388,392,396,400],{"title":385,"path":386,"stem":387},"8.1.1. Основи роботи з файловою системою","\u002Fcsharp\u002Fio\u002Ffile-system-basics","01.csharp\u002F08.io\u002F01.file-system-basics",{"title":389,"path":390,"stem":391},"8.1.2. Потоки (Streams) та Серіалізація Даних","\u002Fcsharp\u002Fio\u002Fstreams-serialization","01.csharp\u002F08.io\u002F02.streams-serialization",{"title":393,"path":394,"stem":395},"8.2.1. JSON Serialization з System.Text.Json","\u002Fcsharp\u002Fio\u002Fjson-serialization","01.csharp\u002F08.io\u002F03.json-serialization",{"title":397,"path":398,"stem":399},"8.2.2. XML Serialization та LINQ to XML","\u002Fcsharp\u002Fio\u002Fxml-serialization","01.csharp\u002F08.io\u002F04.xml-serialization",{"title":401,"path":402,"stem":403},"8.2.3. Binary Serialization: MessagePack та Protocol Buffers","\u002Fcsharp\u002Fio\u002Fbinary-serialization","01.csharp\u002F08.io\u002F05.binary-serialization",{"title":405,"icon":132,"path":406,"stem":407,"children":408,"page":59},"Ado Net","\u002Fcsharp\u002Fado-net","01.csharp\u002F09.ado-net",[409,413,417,421,425,429,433,437,441,445,449,453],{"title":410,"path":411,"stem":412},"9.1. Введення в ADO.NET","\u002Fcsharp\u002Fado-net\u002Fintroduction-to-adonet","01.csharp\u002F09.ado-net\u002F01.introduction-to-adonet",{"title":414,"path":415,"stem":416},"9.2. Клас DbConnection — з'єднання з базою даних","\u002Fcsharp\u002Fado-net\u002Fconnection","01.csharp\u002F09.ado-net\u002F02.connection",{"title":418,"path":419,"stem":420},"9.3. Клас DbCommand — виконання SQL-запитів","\u002Fcsharp\u002Fado-net\u002Fcommand-and-queries","01.csharp\u002F09.ado-net\u002F03.command-and-queries",{"title":422,"path":423,"stem":424},"9.4. Клас DbDataReader — ефективне читання даних","\u002Fcsharp\u002Fado-net\u002Fdatareader","01.csharp\u002F09.ado-net\u002F04.datareader",{"title":426,"path":427,"stem":428},"9.5. Параметризовані запити та захист від SQL Injection","\u002Fcsharp\u002Fado-net\u002Fparameters-and-sql-injection","01.csharp\u002F09.ado-net\u002F05.parameters-and-sql-injection",{"title":430,"path":431,"stem":432},"9.6. Транзакції в ADO.NET","\u002Fcsharp\u002Fado-net\u002Ftransactions","01.csharp\u002F09.ado-net\u002F06.transactions",{"title":434,"path":435,"stem":436},"9.7. DbProviderFactory — провайдер-незалежний код","\u002Fcsharp\u002Fado-net\u002Fprovider-factory","01.csharp\u002F09.ado-net\u002F07.provider-factory",{"title":438,"path":439,"stem":440},"9.8. Асинхронний доступ до даних","\u002Fcsharp\u002Fado-net\u002Fasync-data-access","01.csharp\u002F09.ado-net\u002F08.async-data-access",{"title":442,"path":443,"stem":444},"9.9. Від'єднаний режим: DataSet, DataTable, DataRow","\u002Fcsharp\u002Fado-net\u002Fdisconnected-mode-dataset","01.csharp\u002F09.ado-net\u002F09.disconnected-mode-dataset",{"title":446,"path":447,"stem":448},"9.10. DataAdapter — міст між DataSet та базою даних","\u002Fcsharp\u002Fado-net\u002Fdata-adapter","01.csharp\u002F09.ado-net\u002F10.data-adapter",{"title":450,"path":451,"stem":452},"9.11. Data Mapper та Repository: Архітектура доступу до даних","\u002Fcsharp\u002Fado-net\u002Fdata-mapper-repository","01.csharp\u002F09.ado-net\u002F11.data-mapper-repository",{"title":454,"path":455,"stem":456},"9.12. Identity Map, Unit of Work та Specification Pattern","\u002Fcsharp\u002Fado-net\u002Fadvanced-patterns","01.csharp\u002F09.ado-net\u002F12.advanced-patterns",{"title":458,"icon":255,"path":459,"stem":460,"children":461,"page":59},"Ef Core","\u002Fcsharp\u002Fef-core","01.csharp\u002F10.ef-core",[462,466,470,474,478,482,486,490,494,498,502,506,510,514,518,522,526,532,538,542,546,550,554,558,562,566,570,574,578,582,586,590,594,598,602,606,610,614,618,622,626,630,634,638,642,646],{"title":463,"path":464,"stem":465},"Що таке ORM? Від SQL до об'єктів","\u002Fcsharp\u002Fef-core\u002Fwhat-is-orm","01.csharp\u002F10.ef-core\u002F01.what-is-orm",{"title":467,"path":468,"stem":469},"Перший проєкт — від нуля до CRUD","\u002Fcsharp\u002Fef-core\u002Ffirst-project","01.csharp\u002F10.ef-core\u002F02.first-project",{"title":471,"path":472,"stem":473},"DbContext — Серце EF Core","\u002Fcsharp\u002Fef-core\u002Fdbcontext-deep-dive","01.csharp\u002F10.ef-core\u002F03.dbcontext-deep-dive",{"title":475,"path":476,"stem":477},"Провайдери баз даних — Архітектура та Вибір СУБД","\u002Fcsharp\u002Fef-core\u002Fdatabase-providers","01.csharp\u002F10.ef-core\u002F04.database-providers",{"title":479,"path":480,"stem":481},"Конвенції EF Core — Магія без конфігурації","\u002Fcsharp\u002Fef-core\u002Fconventions","01.csharp\u002F10.ef-core\u002F05.conventions",{"title":483,"path":484,"stem":485},"Fluent API та Data Annotations — Явна конфігурація моделі","\u002Fcsharp\u002Fef-core\u002Ffluent-api-vs-annotations","01.csharp\u002F10.ef-core\u002F06.fluent-api-vs-annotations",{"title":487,"path":488,"stem":489},"Зв'язки — One-to-One та One-to-Many","\u002Fcsharp\u002Fef-core\u002Frelationships-basics","01.csharp\u002F10.ef-core\u002F07.relationships-basics",{"title":491,"path":492,"stem":493},"Зв'язки Advanced — Many-to-Many та Складні Сценарії","\u002Fcsharp\u002Fef-core\u002Frelationships-advanced","01.csharp\u002F10.ef-core\u002F08.relationships-advanced",{"title":495,"path":496,"stem":497},"Властивості — Типи, Конвертери, Компаратори (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part1","01.csharp\u002F10.ef-core\u002F09.property-configuration-part1",{"title":499,"path":500,"stem":501},"Властивості — Value Comparers, Generators, Shadow Properties (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fproperty-configuration-part2","01.csharp\u002F10.ef-core\u002F09.property-configuration-part2",{"title":503,"path":504,"stem":505},"Складні типи — Owned Types та Complex Types (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part1","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part1",{"title":507,"path":508,"stem":509},"Складні типи — Complex Types, Keyless Entities, Порівняння (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fcomplex-types-owned-part2","01.csharp\u002F10.ef-core\u002F10.complex-types-owned-part2",{"title":511,"path":512,"stem":513},"JSON Columns — Складні дані у JSON (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part1","01.csharp\u002F10.ef-core\u002F11.json-columns-part1",{"title":515,"path":516,"stem":517},"JSON Columns — Value Comparers, Індекси, Провайдери (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fjson-columns-part2","01.csharp\u002F10.ef-core\u002F11.json-columns-part2",{"title":519,"path":520,"stem":521},"Успадкування — Абстрактні класи та TPH (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finheritance-part1","01.csharp\u002F10.ef-core\u002F12.inheritance-part1",{"title":523,"path":524,"stem":525},"Успадкування — TPT, TPC та Порівняння Стратегій (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finheritance-part2","01.csharp\u002F10.ef-core\u002F12.inheritance-part2",{"title":527,"path":528,"stem":529,"children":530},"Індекси, Обмеження та Схема (Частина 1)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part1","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part1",[531],{"title":527,"path":528,"stem":529},{"title":533,"path":534,"stem":535,"children":536},"Індекси, Обмеження та Схема (Частина 2)","\u002Fcsharp\u002Fef-core\u002Findexes-constraints-part2","01.csharp\u002F10.ef-core\u002F13.indexes-constraints-part2",[537],{"title":533,"path":534,"stem":535},{"title":539,"path":540,"stem":541},"Seed Data — Початкові Дані (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fseeding-part1","01.csharp\u002F10.ef-core\u002F14.seeding-part1",{"title":543,"path":544,"stem":545},"Seed Data — SQL-скрипти, Bogus та Стратегії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fseeding-part2","01.csharp\u002F10.ef-core\u002F14.seeding-part2",{"title":547,"path":548,"stem":549},"Global Query Filters — Глобальні Фільтри (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part1","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part1",{"title":551,"path":552,"stem":553},"Global Query Filters — Підводні камені та Інтеграція (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fglobal-query-filters-part2","01.csharp\u002F10.ef-core\u002F15.global-query-filters-part2",{"title":555,"path":556,"stem":557},"LINQ-запити в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part1","01.csharp\u002F10.ef-core\u002F16.linq-queries-part1",{"title":559,"path":560,"stem":561},"LINQ-запити в EF Core (Частина 2)","\u002Fcsharp\u002Fef-core\u002Flinq-queries-part2","01.csharp\u002F10.ef-core\u002F16.linq-queries-part2",{"title":563,"path":564,"stem":565},"Завантаження Пов'язаних Даних (Частина 1)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part1","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part1",{"title":567,"path":568,"stem":569},"Завантаження Пов'язаних Даних (Частина 2)","\u002Fcsharp\u002Fef-core\u002Floading-related-data-part2","01.csharp\u002F10.ef-core\u002F17.loading-related-data-part2",{"title":571,"path":572,"stem":573},"Raw SQL, Views та Stored Procedures (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part1","01.csharp\u002F10.ef-core\u002F18.raw-sql-part1",{"title":575,"path":576,"stem":577},"Raw SQL — Stored Procedures, DbFunction та Bulk Operations (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fraw-sql-part2","01.csharp\u002F10.ef-core\u002F18.raw-sql-part2",{"title":579,"path":580,"stem":581},"Продвинуті Запити — Compiled Queries, Bulk та Оптимізація (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part1","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part1",{"title":583,"path":584,"stem":585},"Продвинуті Запити — Query Tags, Bulk та Interceptors (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fadvanced-queries-part2","01.csharp\u002F10.ef-core\u002F19.advanced-queries-part2",{"title":587,"path":588,"stem":589},"Change Tracker — Відстеження Змін (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part1","01.csharp\u002F10.ef-core\u002F20.change-tracking-part1",{"title":591,"path":592,"stem":593},"Change Tracker — Графи Об'єктів та Disconnected (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fchange-tracking-part2","01.csharp\u002F10.ef-core\u002F20.change-tracking-part2",{"title":595,"path":596,"stem":597},"Збереження Даних та Транзакції (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part1","01.csharp\u002F10.ef-core\u002F21.saving-data-part1",{"title":599,"path":600,"stem":601},"Збереження Даних — Concurrency та Outbox (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fsaving-data-part2","01.csharp\u002F10.ef-core\u002F21.saving-data-part2",{"title":603,"path":604,"stem":605},"Конкурентність та Блокування (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part1","01.csharp\u002F10.ef-core\u002F22.concurrency-part1",{"title":607,"path":608,"stem":609},"Конкурентність — Дедлоки та Queue Processing (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fconcurrency-part2","01.csharp\u002F10.ef-core\u002F22.concurrency-part2",{"title":611,"path":612,"stem":613},"Міграції в EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part1","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part1",{"title":615,"path":616,"stem":617},"Міграції в EF Core — Основи (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-basics-part2","01.csharp\u002F10.ef-core\u002F23.migrations-basics-part2",{"title":619,"path":620,"stem":621},"Міграції — Просунуті Сценарії (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part1","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part1",{"title":623,"path":624,"stem":625},"Міграції — Просунуті Сценарії (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fmigrations-advanced-part2","01.csharp\u002F10.ef-core\u002F24.migrations-advanced-part2",{"title":627,"path":628,"stem":629},"Управління Схемою та Database-First (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part1","01.csharp\u002F10.ef-core\u002F25.schema-management-part1",{"title":631,"path":632,"stem":633},"Управління Схемою та Database-First (Частина 2)","\u002Fcsharp\u002Fef-core\u002Fschema-management-part2","01.csharp\u002F10.ef-core\u002F25.schema-management-part2",{"title":635,"path":636,"stem":637},"Продуктивність EF Core — Основи (Частина 1)","\u002Fcsharp\u002Fef-core\u002Fperformance-fundamentals-part1","01.csharp\u002F10.ef-core\u002F26.performance-fundamentals-part1",{"title":639,"path":640,"stem":641},"Interceptors в EF Core (Частина 1)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part1","01.csharp\u002F10.ef-core\u002F29.interceptors-part1",{"title":643,"path":644,"stem":645},"Interceptors в EF Core — Connection, Transaction та Materialization (Частина 2)","\u002Fcsharp\u002Fef-core\u002Finterceptors-part2","01.csharp\u002F10.ef-core\u002F29.interceptors-part2",{"title":647,"path":648,"stem":649},"План вивчення Entity Framework Core — Повний курс","\u002Fcsharp\u002Fef-core\u002Fimplementation_plan","01.csharp\u002F10.ef-core\u002Fimplementation_plan",{"title":651,"icon":652,"path":653,"stem":654,"children":655,"page":59},"ASP.NET","i-devicon-dotnetcore","\u002Fcsharp\u002Faspnet","01.csharp\u002F11.aspnet",[656,730,791,869,927,941,967,1057,1111,1182,1212,1289,1346],{"title":657,"icon":658,"path":659,"stem":660,"children":661,"page":59},"Minimal API","i-lucide-network","\u002Fcsharp\u002Faspnet\u002Fminimal-api","01.csharp\u002F11.aspnet\u002F01.minimal-api",[662,666,670,674,678,682,686,690,694,698,702,706,710,714,718,722,726],{"title":663,"path":664,"stem":665},"Вступ до ASP.NET та еволюція фреймворку","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fintroduction","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F01.introduction",{"title":667,"path":668,"stem":669},"Перший додаток на ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Ffirst-application","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F02.first-application",{"title":671,"path":672,"stem":673},"WebApplication, Builder та Dependency Injection","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fwebapplication-builder","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F03.webapplication-builder",{"title":675,"path":676,"stem":677},"Конвеєр запитів та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frequest-pipeline-middleware","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F04.request-pipeline-middleware",{"title":679,"path":680,"stem":681},"Маршрутизація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F05.routing-basics",{"title":683,"path":684,"stem":685},"Маршрутизація в ASP.NET Core: Розширені можливості","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Frouting-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F06.routing-advanced",{"title":687,"path":688,"stem":689},"Статичні файли в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-files","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F07.static-files",{"title":691,"path":692,"stem":693},"Статичні Активи: MapStaticAssets (ASP.NET Core 9.0)","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstatic-assets","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F08.static-assets",{"title":695,"path":696,"stem":697},"Конфігурація в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-fundamentals","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F09.configuration-fundamentals",{"title":699,"path":700,"stem":701},"Конфігурація: Паттерн Options","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fconfiguration-options","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F10.configuration-options",{"title":703,"path":704,"stem":705},"Логування в ASP.NET Core: Основи","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-basics","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F11.logging-basics",{"title":707,"path":708,"stem":709},"Логування: Serilog та Middleware","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Flogging-advanced","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F12.logging-advanced",{"title":711,"path":712,"stem":713},"Управління станом: HttpContext.Items та Cookies","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fstate-management","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F13.state-management",{"title":715,"path":716,"stem":717},"Стан сесії: Sessions","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fsession-state","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F14.session-state",{"title":719,"path":720,"stem":721},"Структура проєкту: від хаосу до архітектури","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fproject-structure","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F15.project-structure",{"title":723,"path":724,"stem":725},"Scalar у Minimal API: повний проєкт і Fluent OpenAPI","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fscalar-openapi-fluent","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F16.scalar-openapi-fluent",{"title":727,"path":728,"stem":729},"Swagger \u002F Swashbuckle у Minimal API: окремий класичний шлях","\u002Fcsharp\u002Faspnet\u002Fminimal-api\u002Fswagger-swashbuckle","01.csharp\u002F11.aspnet\u002F01.minimal-api\u002F17.swagger-swashbuckle",{"title":731,"icon":658,"path":732,"stem":733,"children":734,"page":59},"API","\u002Fcsharp\u002Faspnet\u002Fapi","01.csharp\u002F11.aspnet\u002F02.api",[735,739,743,747,751,755,759,763,767,771,775,779,783,787],{"title":736,"path":737,"stem":738},"Що таке API. Клієнт-серверна архітектура","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fwhat-is-api","01.csharp\u002F11.aspnet\u002F02.api\u002F01.what-is-api",{"title":740,"path":741,"stem":742},"Формати даних: JSON, XML, TOML та бінарні формати","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fdata-formats","01.csharp\u002F11.aspnet\u002F02.api\u002F02.data-formats",{"title":744,"path":745,"stem":746},"Парадигми API та концепція REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-paradigms-rest","01.csharp\u002F11.aspnet\u002F02.api\u002F03.api-paradigms-rest",{"title":748,"path":749,"stem":750},"HTTP-методи, статус-коди та заголовки","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fhttp-methods-status-codes","01.csharp\u002F11.aspnet\u002F02.api\u002F04.http-methods-status-codes",{"title":752,"path":753,"stem":754},"Організація HTTP API за принципами REST","\u002Fcsharp\u002Faspnet\u002Fapi\u002Frest-organizing","01.csharp\u002F11.aspnet\u002F02.api\u002F05.rest-organizing",{"title":756,"path":757,"stem":758},"Номенклатура URL та CRUD-операції","\u002Fcsharp\u002Faspnet\u002Fapi\u002Furl-nomenclature-crud","01.csharp\u002F11.aspnet\u002F02.api\u002F06.url-nomenclature-crud",{"title":760,"path":761,"stem":762},"Правила дизайну: іменування та стандарти","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-naming","01.csharp\u002F11.aspnet\u002F02.api\u002F07.api-design-naming",{"title":764,"path":765,"stem":766},"Валідація, ліміти та обробка помилок","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-validation","01.csharp\u002F11.aspnet\u002F02.api\u002F08.api-design-validation",{"title":768,"path":769,"stem":770},"Обробка помилок у Minimal API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Ferror-handling-http","01.csharp\u002F11.aspnet\u002F02.api\u002F09.error-handling-http",{"title":772,"path":773,"stem":774},"Ідемпотентність та синхронізація стану","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fidempotency-sync","01.csharp\u002F11.aspnet\u002F02.api\u002F10.idempotency-sync",{"title":776,"path":777,"stem":778},"Пагінація та організація списків","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fpagination-lists","01.csharp\u002F11.aspnet\u002F02.api\u002F11.pagination-lists",{"title":780,"path":781,"stem":782},"Безпека API, кешування та інтернаціоналізація","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fsecurity-auth","01.csharp\u002F11.aspnet\u002F02.api\u002F12.security-auth",{"title":784,"path":785,"stem":786},"Процес проєктування API та документування","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fapi-design-process","01.csharp\u002F11.aspnet\u002F02.api\u002F13.api-design-process",{"title":788,"path":789,"stem":790},"OpenAPI: контракт, специфікація та документація API","\u002Fcsharp\u002Faspnet\u002Fapi\u002Fopenapi","01.csharp\u002F11.aspnet\u002F02.api\u002F14.openapi",{"title":792,"icon":793,"path":794,"stem":795,"children":796,"page":59},"Auth","i-lucide-shield-check","\u002Fcsharp\u002Faspnet\u002Fauth","01.csharp\u002F11.aspnet\u002F03.auth",[797,801,805,809,813,817,821,825,829,833,837,841,845,849,853,857,861,865],{"title":798,"path":799,"stem":800},"Основи аутентифікації та авторизації","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauth-fundamentals","01.csharp\u002F11.aspnet\u002F03.auth\u002F01.auth-fundamentals",{"title":802,"path":803,"stem":804},"JWT-аутентифікація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fjwt-authentication","01.csharp\u002F11.aspnet\u002F03.auth\u002F02.jwt-authentication",{"title":806,"path":807,"stem":808},"Авторизація: ролі, політики та resource-based доступ","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fauthorization-policies","01.csharp\u002F11.aspnet\u002F03.auth\u002F03.authorization-policies",{"title":810,"path":811,"stem":812},"Cookie-аутентифікація та ASP.NET Core Identity","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcookie-auth-identity","01.csharp\u002F11.aspnet\u002F03.auth\u002F04.cookie-auth-identity",{"title":814,"path":815,"stem":816},"JWT + Refresh Tokens (HttpOnly Cookie)","\u002Fcsharp\u002Faspnet\u002Fauth\u002F04b.identity-auth-jwt","01.csharp\u002F11.aspnet\u002F03.auth\u002F04b.identity-auth-jwt",{"title":818,"path":819,"stem":820},"Identity: Підтвердження Email та Скидання Пароля","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-email-confirmation","01.csharp\u002F11.aspnet\u002F03.auth\u002F05.identity-email-confirmation",{"title":822,"path":823,"stem":824},"Identity: Двофакторна Аутентифікація (2FA)","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-two-factor","01.csharp\u002F11.aspnet\u002F03.auth\u002F06.identity-two-factor",{"title":826,"path":827,"stem":828},"Identity: Внутрішня Архітектура та Кастомізація","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fidentity-internals","01.csharp\u002F11.aspnet\u002F03.auth\u002F07.identity-internals",{"title":830,"path":831,"stem":832},"OAuth 2.0 та зовнішні провайдери","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-external-providers","01.csharp\u002F11.aspnet\u002F03.auth\u002F08.oauth-external-providers",{"title":834,"path":835,"stem":836},"Безпека на практиці: CORS, HTTPS та захист від атак","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fsecurity-hardening","01.csharp\u002F11.aspnet\u002F03.auth\u002F09.security-hardening",{"title":838,"path":839,"stem":840},"Теорія OAuth 2.0: Поняття, Аналогії та Флоу","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foauth-theory","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oauth-theory",{"title":842,"path":843,"stem":844},"OIDC, OAuth 2.0 та Keycloak в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Foidc-keycloak","01.csharp\u002F11.aspnet\u002F03.auth\u002F10.oidc-keycloak",{"title":846,"path":847,"stem":848},"API Keys аутентифікація в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fapi-keys","01.csharp\u002F11.aspnet\u002F03.auth\u002F11.api-keys",{"title":850,"path":851,"stem":852},"Rate Limiting та Throttling в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frate-limiting","01.csharp\u002F11.aspnet\u002F03.auth\u002F12.rate-limiting",{"title":854,"path":855,"stem":856},"Refresh Token Rotation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frefresh-token-rotation","01.csharp\u002F11.aspnet\u002F03.auth\u002F13.refresh-token-rotation",{"title":858,"path":859,"stem":860},"Certificate Authentication та mTLS в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fcertificate-auth","01.csharp\u002F11.aspnet\u002F03.auth\u002F14.certificate-auth",{"title":862,"path":863,"stem":864},"RBAC, ABAC та ReBAC в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Frbac-abac-rebac","01.csharp\u002F11.aspnet\u002F03.auth\u002F15.rbac-abac-rebac",{"title":866,"path":867,"stem":868},"Multi-tenancy та ізоляція даних в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fauth\u002Fmulti-tenancy","01.csharp\u002F11.aspnet\u002F03.auth\u002F16.multi-tenancy",{"title":870,"icon":871,"path":872,"stem":873,"children":874,"page":59},"Нотифікації","i-lucide-bell","\u002Fcsharp\u002Faspnet\u002Fnotifications","01.csharp\u002F11.aspnet\u002F04.notifications",[875,879,883,887,891,895,899,903,907,911,915,919,923],{"title":876,"path":877,"stem":878},"In-App нотифікації через базу даних","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fin-app-database-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F01.in-app-database-notifications",{"title":880,"path":881,"stem":882},"Polling: Регулярний запит оновлень","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fpolling","01.csharp\u002F11.aspnet\u002F04.notifications\u002F02.polling",{"title":884,"path":885,"stem":886},"Server-Sent Events: Однострімовий push від сервера","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fserver-sent-events","01.csharp\u002F11.aspnet\u002F04.notifications\u002F03.server-sent-events",{"title":888,"path":889,"stem":890},"WebSockets: Двостороннє з'єднання в реальному часі","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fwebsockets","01.csharp\u002F11.aspnet\u002F04.notifications\u002F04.websockets",{"title":892,"path":893,"stem":894},"SignalR: Абстракція над транспортами реального часу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fsignalr","01.csharp\u002F11.aspnet\u002F04.notifications\u002F05.signalr",{"title":896,"path":897,"stem":898},"Background Services: Фонові задачі в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fbackground-services","01.csharp\u002F11.aspnet\u002F04.notifications\u002F06.background-services",{"title":900,"path":901,"stem":902},"Web Push нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fweb-push","01.csharp\u002F11.aspnet\u002F04.notifications\u002F07.web-push",{"title":904,"path":905,"stem":906},"Email нотифікації","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Femail-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F08.email-notifications",{"title":908,"path":909,"stem":910},"Порівняння підходів: Як вибрати правильну технологію нотифікацій","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fchoosing-the-right-approach","01.csharp\u002F11.aspnet\u002F04.notifications\u002F09.choosing-the-right-approach",{"title":912,"path":913,"stem":914},"Hangfire: Надійне планування фонових задач","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire","01.csharp\u002F11.aspnet\u002F04.notifications\u002F10.hangfire",{"title":916,"path":917,"stem":918},"Практика: Конвертація зображень у WebP через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-image-webp","01.csharp\u002F11.aspnet\u002F04.notifications\u002F11.hangfire-image-webp",{"title":920,"path":921,"stem":922},"Практика: Підготовка відео до HLS-стрімінгу через Hangfire","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Fhangfire-video-hls","01.csharp\u002F11.aspnet\u002F04.notifications\u002F12.hangfire-video-hls",{"title":924,"path":925,"stem":926},"Telegram-нотифікації: від одного повідомлення до масових розсилок і мульти-канального підходу","\u002Fcsharp\u002Faspnet\u002Fnotifications\u002Ftelegram-notifications","01.csharp\u002F11.aspnet\u002F04.notifications\u002F13.telegram-notifications",{"title":928,"icon":929,"path":930,"stem":931,"children":932,"page":59},"Інтернаціоналізація","i-lucide-languages","\u002Fcsharp\u002Faspnet\u002Fi18n","01.csharp\u002F11.aspnet\u002F05.i18n",[933,937],{"title":934,"path":935,"stem":936},"Інтернаціоналізація (i18n) у Minimal API: від A до Я","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Finternationalization","01.csharp\u002F11.aspnet\u002F05.i18n\u002F01.internationalization",{"title":938,"path":939,"stem":940},"Humanizer: людиномовні рядки у .NET","\u002Fcsharp\u002Faspnet\u002Fi18n\u002Fhumanizer","01.csharp\u002F11.aspnet\u002F05.i18n\u002F02.humanizer",{"title":942,"icon":943,"path":944,"stem":945,"children":946,"page":59},"Кешування","i-lucide-layers","\u002Fcsharp\u002Faspnet\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching",[947,951,955,959,963],{"title":948,"path":949,"stem":950},"Огляд кешування: чотири рівні і коли що обирати","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fcaching","01.csharp\u002F11.aspnet\u002F06.caching\u002F01.caching",{"title":952,"path":953,"stem":954},"IMemoryCache: кеш в оперативній пам'яті","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fmemory-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F02.memory-cache",{"title":956,"path":957,"stem":958},"IDistributedCache і Redis: розподілений кеш","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fdistributed-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F03.distributed-cache",{"title":960,"path":961,"stem":962},"Response Cache: HTTP-кешування через Cache-Control","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Fresponse-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F04.response-cache",{"title":964,"path":965,"stem":966},"Output Cache: серверний кеш HTTP-відповідей (.NET 7+)","\u002Fcsharp\u002Faspnet\u002Fcaching\u002Foutput-cache","01.csharp\u002F11.aspnet\u002F06.caching\u002F05.output-cache",{"title":968,"icon":969,"path":970,"stem":971,"children":972,"page":59},"Тестування","i-lucide-test-tube","\u002Fcsharp\u002Faspnet\u002Ftesting","01.csharp\u002F11.aspnet\u002F07.testing",[973,977,981,985,989,993,997,1001,1005,1009,1013,1017,1021,1025,1029,1033,1037,1041,1045,1049,1053],{"title":974,"path":975,"stem":976},"Що таке тестування? Від інтуїції до науки","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-is-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F01.what-is-testing",{"title":978,"path":979,"stem":980},"Піраміда тестування — Стратегія, а не Догма","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-pyramid","01.csharp\u002F11.aspnet\u002F07.testing\u002F02.testing-pyramid",{"title":982,"path":983,"stem":984},"Дві Школи Тестування — Лондон проти Детройту","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-schools","01.csharp\u002F11.aspnet\u002F07.testing\u002F03.testing-schools",{"title":986,"path":987,"stem":988},"TDD та BDD — Тести як Дизайн-інструмент","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftdd-and-bdd","01.csharp\u002F11.aspnet\u002F07.testing\u002F04.tdd-and-bdd",{"title":990,"path":991,"stem":992},"Що саме тестувати — Техніки аналізу та Циклomatична складність","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwhat-to-test","01.csharp\u002F11.aspnet\u002F07.testing\u002F05.what-to-test",{"title":994,"path":995,"stem":996},"Тестові Фреймворки — Навіщо вони і що всередині","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftest-frameworks","01.csharp\u002F11.aspnet\u002F07.testing\u002F06.test-frameworks",{"title":998,"path":999,"stem":1000},"xUnit — Факти, Теорії та Lifecycle тестів","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-basics","01.csharp\u002F11.aspnet\u002F07.testing\u002F07.xunit-basics",{"title":1002,"path":1003,"stem":1004},"xUnit Advanced — Fixtures, Кастомізація та Розширення","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fxunit-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F08.xunit-advanced",{"title":1006,"path":1007,"stem":1008},"Moq — Глибоке занурення в мокування","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fmocking-with-moq","01.csharp\u002F11.aspnet\u002F07.testing\u002F09.mocking-with-moq",{"title":1010,"path":1011,"stem":1012},"Тестування Баз Даних — EF Core, SQLite та Testcontainers","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fdatabase-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F10.database-testing",{"title":1014,"path":1015,"stem":1016},"Integration Testing — Частина 1 [Теорія та WebApplicationFactory]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F11.integration-testing",{"title":1018,"path":1019,"stem":1020},"Інтеграційне тестування — Практика","\u002Fcsharp\u002Faspnet\u002Ftesting\u002F11a.integration-testing-practice","01.csharp\u002F11.aspnet\u002F07.testing\u002F11a.integration-testing-practice",{"title":1022,"path":1023,"stem":1024},"Integration Testing — Частина 2 [Просунуті Сценарії та Testcontainers]","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fintegration-testing-advanced","01.csharp\u002F11.aspnet\u002F07.testing\u002F12.integration-testing-advanced",{"title":1026,"path":1027,"stem":1028},"Професійний Postman: Колекції, Змінні та GitHub Інтеграція","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fpostman-professional","01.csharp\u002F11.aspnet\u002F07.testing\u002F13.postman-professional",{"title":1030,"path":1031,"stem":1032},"HttpClient у Тестах Частина 1: Архітектура та MockHttpMessageHandler","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fhttpclient-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F14.httpclient-testing",{"title":1034,"path":1035,"stem":1036},"HttpClient у Тестах Частина 2: WireMock.Net та Resilience","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fwiremock-net","01.csharp\u002F11.aspnet\u002F07.testing\u002F15.wiremock-net",{"title":1038,"path":1039,"stem":1040},"Патерни та Анти-патерни Тестування: Test Smells","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Ftesting-patterns","01.csharp\u002F11.aspnet\u002F07.testing\u002F16.testing-patterns",{"title":1042,"path":1043,"stem":1044},"Просунуті інструменти: Time, Snapshots та Властивості","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fadvanced-testing-tools","01.csharp\u002F11.aspnet\u002F07.testing\u002F17.advanced-testing-tools",{"title":1046,"path":1047,"stem":1048},"Тестування Архітектури з NetArchTest","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Farchitecture-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F18.architecture-testing",{"title":1050,"path":1051,"stem":1052},"Тестування Продуктивності: BenchmarkDotNet, NBomber та k6","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fperformance-testing","01.csharp\u002F11.aspnet\u002F07.testing\u002F19.performance-testing",{"title":1054,"path":1055,"stem":1056},"Залишок плану для курсу \"Тестування ASP.NET Minimal API\"","\u002Fcsharp\u002Faspnet\u002Ftesting\u002Fremaining_plan","01.csharp\u002F11.aspnet\u002F07.testing\u002Fremaining_plan",{"title":1058,"icon":1059,"path":1060,"stem":1061,"children":1062,"page":59},"Платежі","i-lucide-credit-card","\u002Fcsharp\u002Faspnet\u002Fpayments","01.csharp\u002F11.aspnet\u002F08.payments",[1063,1067,1071,1075,1079,1083,1087,1091,1095,1099,1103,1107],{"title":1064,"path":1065,"stem":1066},"Основи платіжної інфраструктури","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-fundamentals","01.csharp\u002F11.aspnet\u002F08.payments\u002F01.payment-fundamentals",{"title":1068,"path":1069,"stem":1070},"Методи оплати в Україні","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-methods-ukraine","01.csharp\u002F11.aspnet\u002F08.payments\u002F02.payment-methods-ukraine",{"title":1072,"path":1073,"stem":1074},"PCI DSS та безпека платежів","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpci-dss-security","01.csharp\u002F11.aspnet\u002F08.payments\u002F03.pci-dss-security",{"title":1076,"path":1077,"stem":1078},"Архітектура платіжної підсистеми","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fpayment-architecture","01.csharp\u002F11.aspnet\u002F08.payments\u002F04.payment-architecture",{"title":1080,"path":1081,"stem":1082},"Інтеграція LiqPay (ПриватБанк)","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fliqpay-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F05.liqpay-integration",{"title":1084,"path":1085,"stem":1086},"Інтеграція Monobank Acquiring API","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fmonobank-acquiring","01.csharp\u002F11.aspnet\u002F08.payments\u002F06.monobank-acquiring",{"title":1088,"path":1089,"stem":1090},"Інтеграція Stripe","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fstripe-integration","01.csharp\u002F11.aspnet\u002F08.payments\u002F07.stripe-integration",{"title":1092,"path":1093,"stem":1094},"Webhooks — глибоке занурення","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fwebhooks-deep-dive","01.csharp\u002F11.aspnet\u002F08.payments\u002F08.webhooks-deep-dive",{"title":1096,"path":1097,"stem":1098},"Підписки та рекурентні платежі","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fsubscriptions-recurring","01.csharp\u002F11.aspnet\u002F08.payments\u002F09.subscriptions-recurring",{"title":1100,"path":1101,"stem":1102},"Повернення коштів та диспути","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Frefunds-disputes","01.csharp\u002F11.aspnet\u002F08.payments\u002F10.refunds-disputes",{"title":1104,"path":1105,"stem":1106},"Тестування платіжних інтеграцій","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Ftesting-payments","01.csharp\u002F11.aspnet\u002F08.payments\u002F11.testing-payments",{"title":1108,"path":1109,"stem":1110},"Чекліст виходу в Production","\u002Fcsharp\u002Faspnet\u002Fpayments\u002Fproduction-checklist","01.csharp\u002F11.aspnet\u002F08.payments\u002F12.production-checklist",{"title":1112,"icon":1113,"items":1114,"path":1127,"stem":1128,"children":1129,"page":59},"Популярні бібліотеки","lucide:box",[1115,1116,1117,1118,1119,1120,1121,1122,1123,1124,1125,1126],"01.fluent-validation","02.mapster","03.erroror-result-pattern","04.serilog","05.mediatr","06.polly","07.health-checks","08.feature-management","09.fluent-email","10.quest-pdf","11.bogus","12.humanizer-guard","\u002Fcsharp\u002Faspnet\u002Flibraries","01.csharp\u002F11.aspnet\u002F09.libraries",[1130,1134,1138,1142,1146,1150,1154,1158,1162,1166,1170,1174,1178],{"title":1131,"path":1132,"stem":1133},"Валідація з FluentValidation в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-validation","01.csharp\u002F11.aspnet\u002F09.libraries\u002F01.fluent-validation",{"title":1135,"path":1136,"stem":1137},"Маппінг об","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmapster","01.csharp\u002F11.aspnet\u002F09.libraries\u002F02.mapster",{"title":1139,"path":1140,"stem":1141},"Обробка помилок з ErrorOr та Result Pattern в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ferroror-result-pattern","01.csharp\u002F11.aspnet\u002F09.libraries\u002F03.erroror-result-pattern",{"title":1143,"path":1144,"stem":1145},"Структуроване логування з Serilog в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fserilog","01.csharp\u002F11.aspnet\u002F09.libraries\u002F04.serilog",{"title":1147,"path":1148,"stem":1149},"CQRS та Mediator з MediatR в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fmediatr","01.csharp\u002F11.aspnet\u002F09.libraries\u002F05.mediatr",{"title":1151,"path":1152,"stem":1153},"Відмовостійкість з Polly в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fpolly","01.csharp\u002F11.aspnet\u002F09.libraries\u002F06.polly",{"title":1155,"path":1156,"stem":1157},"Health Checks в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhealth-checks","01.csharp\u002F11.aspnet\u002F09.libraries\u002F07.health-checks",{"title":1159,"path":1160,"stem":1161},"Feature Management та Feature Flags в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffeature-management","01.csharp\u002F11.aspnet\u002F09.libraries\u002F08.feature-management",{"title":1163,"path":1164,"stem":1165},"Відправка Email з FluentEmail в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Ffluent-email","01.csharp\u002F11.aspnet\u002F09.libraries\u002F09.fluent-email",{"title":1167,"path":1168,"stem":1169},"Генерація PDF з QuestPDF в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fquest-pdf","01.csharp\u002F11.aspnet\u002F09.libraries\u002F10.quest-pdf",{"title":1171,"path":1172,"stem":1173},"Генерація тестових даних з Bogus в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fbogus","01.csharp\u002F11.aspnet\u002F09.libraries\u002F11.bogus",{"title":1175,"path":1176,"stem":1177},"Humanizer та Guard Clauses в ASP.NET Core","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fhumanizer-guard","01.csharp\u002F11.aspnet\u002F09.libraries\u002F12.humanizer-guard",{"title":1179,"path":1180,"stem":1181},"План модуля 10.libraries — Популярні бібліотеки ASP.NET","\u002Fcsharp\u002Faspnet\u002Flibraries\u002Fplan","01.csharp\u002F11.aspnet\u002F09.libraries\u002Fplan",{"title":1183,"icon":1184,"path":1185,"stem":1186,"children":1187,"page":59},"Razor Pages","i-lucide-layout-template","\u002Fcsharp\u002Faspnet\u002Frazor-pages","01.csharp\u002F11.aspnet\u002F10.razor-pages",[1188,1192,1196,1200,1204,1208],{"title":1189,"path":1190,"stem":1191},"Від Minimal API до Razor Pages: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ffrom-minimal-api","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F01.from-minimal-api",{"title":1193,"path":1194,"stem":1195},"PageModel: логіка сторінки Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fpage-model","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F02.page-model",{"title":1197,"path":1198,"stem":1199},"Razor синтаксис: шаблонізатор у .cshtml","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Frazor-syntax","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F03.razor-syntax",{"title":1201,"path":1202,"stem":1203},"Tag Helpers: типізований HTML","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Ftag-helpers","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F04.tag-helpers",{"title":1205,"path":1206,"stem":1207},"Форми і валідація: повний цикл обробки даних","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fforms-validation","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F05.forms-validation",{"title":1209,"path":1210,"stem":1211},"Практичний проєкт: TaskManager на Razor Pages","\u002Fcsharp\u002Faspnet\u002Frazor-pages\u002Fproject-task-manager","01.csharp\u002F11.aspnet\u002F10.razor-pages\u002F06.project-task-manager",{"title":1213,"path":1214,"stem":1215,"children":1216,"page":59},"ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc","01.csharp\u002F11.aspnet\u002F11.mvc",[1217,1221,1225,1229,1233,1237,1241,1245,1249,1253,1257,1261,1265,1269,1273,1277,1281,1285],{"title":1218,"path":1219,"stem":1220},"Патерн MVC: архітектура, що змінила веб","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-pattern","01.csharp\u002F11.aspnet\u002F11.mvc\u002F01.mvc-pattern",{"title":1222,"path":1223,"stem":1224},"Від Razor Pages до MVC: концептуальний перехід","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffrom-razor-pages","01.csharp\u002F11.aspnet\u002F11.mvc\u002F02.from-razor-pages",{"title":1226,"path":1227,"stem":1228},"Controllers та Actions: серце MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fcontrollers-actions","01.csharp\u002F11.aspnet\u002F11.mvc\u002F03.controllers-actions",{"title":1230,"path":1231,"stem":1232},"Маршрутизація в MVC: Convention vs Attribute Routing","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Frouting-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F04.routing-mvc",{"title":1234,"path":1235,"stem":1236},"Model Binding: від HTTP до C#","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmodel-binding","01.csharp\u002F11.aspnet\u002F11.mvc\u002F05.model-binding",{"title":1238,"path":1239,"stem":1240},"Views, ViewData, ViewBag, TempData і ViewModel","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fviews-viewdata-tempdata","01.csharp\u002F11.aspnet\u002F11.mvc\u002F06.views-viewdata-tempdata",{"title":1242,"path":1243,"stem":1244},"Filters: аспектно-орієнтоване програмування в MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffilters","01.csharp\u002F11.aspnet\u002F11.mvc\u002F07.filters",{"title":1246,"path":1247,"stem":1248},"Areas: структурування великих застосунків","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fareas","01.csharp\u002F11.aspnet\u002F11.mvc\u002F08.areas",{"title":1250,"path":1251,"stem":1252},"View Components: повторювані незалежні блоки UI","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fview-components","01.csharp\u002F11.aspnet\u002F11.mvc\u002F09.view-components",{"title":1254,"path":1255,"stem":1256},"Display та Editor Templates","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fdisplay-editor-templates","01.csharp\u002F11.aspnet\u002F11.mvc\u002F10.display-editor-templates",{"title":1258,"path":1259,"stem":1260},"Валідація: IValidatableObject та FluentValidation","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fvalidation-advanced","01.csharp\u002F11.aspnet\u002F11.mvc\u002F11.validation-advanced",{"title":1262,"path":1263,"stem":1264},"HTMX: інтерактивність через HTML-атрибути","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx","01.csharp\u002F11.aspnet\u002F11.mvc\u002F12.htmx",{"title":1266,"path":1267,"stem":1268},"HTMX у ASP.NET Core MVC: серверна інтеграція","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fajax-htmx-mvc","01.csharp\u002F11.aspnet\u002F11.mvc\u002F13.ajax-htmx-mvc",{"title":1270,"path":1271,"stem":1272},"Практичний проєкт: Каталог товарів з HTMX","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fhtmx-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F14.htmx-project",{"title":1274,"path":1275,"stem":1276},"Завантаження та обробка файлів","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Ffile-upload","01.csharp\u002F11.aspnet\u002F11.mvc\u002F15.file-upload",{"title":1278,"path":1279,"stem":1280},"Глобалізація та Локалізація MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fglobalization-localization","01.csharp\u002F11.aspnet\u002F11.mvc\u002F16.globalization-localization",{"title":1282,"path":1283,"stem":1284},"Підсумковий проєкт: Блог-платформа","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fmvc-project","01.csharp\u002F11.aspnet\u002F11.mvc\u002F17.mvc-project",{"title":1286,"path":1287,"stem":1288},"План курсу: ASP.NET Core MVC","\u002Fcsharp\u002Faspnet\u002Fmvc\u002Fplan","01.csharp\u002F11.aspnet\u002F11.mvc\u002Fplan",{"title":1290,"path":1291,"stem":1292,"children":1293,"page":59},"Web Api","\u002Fcsharp\u002Faspnet\u002Fweb-api","01.csharp\u002F11.aspnet\u002F12.web-api",[1294,1298,1302,1306,1310,1314,1318,1322,1326,1330,1334,1338,1342],{"title":1295,"path":1296,"stem":1297},"Від Minimal API до Controller-based API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffrom-minimal-api-to-controllers","01.csharp\u002F11.aspnet\u002F12.web-api\u002F01.from-minimal-api-to-controllers",{"title":1299,"path":1300,"stem":1301},"ControllerBase, ActionResult\u003CT> та Response Types","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontroller-base-actionresult","01.csharp\u002F11.aspnet\u002F12.web-api\u002F02.controller-base-actionresult",{"title":1303,"path":1304,"stem":1305},"Content Negotiation - JSON, XML та власні форматери","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fcontent-negotiation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F03.content-negotiation",{"title":1307,"path":1308,"stem":1309},"Версіонування API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-versioning","01.csharp\u002F11.aspnet\u002F12.web-api\u002F04.api-versioning",{"title":1311,"path":1312,"stem":1313},"ProblemDetails та структурована обробка помилок","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fproblemdetails-error-handling","01.csharp\u002F11.aspnet\u002F12.web-api\u002F05.problemdetails-error-handling",{"title":1315,"path":1316,"stem":1317},"Фільтри у Web API контексті","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Ffilters-for-api","01.csharp\u002F11.aspnet\u002F12.web-api\u002F06.filters-for-api",{"title":1319,"path":1320,"stem":1321},"Пагінація, фільтрація та сортування","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fpagination-filtering-sorting","01.csharp\u002F11.aspnet\u002F12.web-api\u002F07.pagination-filtering-sorting",{"title":1323,"path":1324,"stem":1325},"HATEOAS та Resource Expansion","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhateoas-resource-expansion","01.csharp\u002F11.aspnet\u002F12.web-api\u002F08.hateoas-resource-expansion",{"title":1327,"path":1328,"stem":1329},"Гібридна архітектура - Minimal API + Controllers","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fminimal-api-vs-controllers-hybrid","01.csharp\u002F11.aspnet\u002F12.web-api\u002F09.minimal-api-vs-controllers-hybrid",{"title":1331,"path":1332,"stem":1333},"Документація API - Swashbuckle, NSwag та генерація клієнтів","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fapi-documentation-generation","01.csharp\u002F11.aspnet\u002F12.web-api\u002F10.api-documentation-generation",{"title":1335,"path":1336,"stem":1337},"Health Checks та моніторинг API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fhealth-checks-monitoring","01.csharp\u002F11.aspnet\u002F12.web-api\u002F11.health-checks-monitoring",{"title":1339,"path":1340,"stem":1341},"Підсумковий проєкт - Production-Ready REST API","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fweb-api-project","01.csharp\u002F11.aspnet\u002F12.web-api\u002F12.web-api-project",{"title":1343,"path":1344,"stem":1345},"План курсу: ASP.NET Core Web API (Controllers)","\u002Fcsharp\u002Faspnet\u002Fweb-api\u002Fplan","01.csharp\u002F11.aspnet\u002F12.web-api\u002Fplan",{"title":1347,"icon":1348,"path":1349,"stem":1350,"children":1351,"page":59},"Моніторинг","i-lucide-activity","\u002Fcsharp\u002Faspnet\u002Fmonitoring","01.csharp\u002F11.aspnet\u002F13.monitoring",[1352,1356,1360],{"title":1353,"path":1354,"stem":1355},"Спостережуваність: від console.log до production-систем","\u002Fcsharp\u002Faspnet\u002Fmonitoring\u002Fobservability-intro","01.csharp\u002F11.aspnet\u002F13.monitoring\u002F01.observability-intro",{"title":1357,"path":1358,"stem":1359},"Health Checks: перший рівень observability","\u002Fcsharp\u002Faspnet\u002Fmonitoring\u002Fhealth-checks","01.csharp\u002F11.aspnet\u002F13.monitoring\u002F02.health-checks",{"title":1361,"path":1362,"stem":1363},"Вбудовані метрики .NET 10 та System.Diagnostics.Metrics","\u002Fcsharp\u002Faspnet\u002Fmonitoring\u002Fdotnet-metrics","01.csharp\u002F11.aspnet\u002F13.monitoring\u002F03.dotnet-metrics",{"title":1365,"icon":1366,"path":1367,"stem":1368,"children":1369,"page":59},"Desktop UI","i-lucide-app-window","\u002Fcsharp\u002Fdesktop-ui","01.csharp\u002F12.desktop-ui",[1370,1374,1378,1382,1386,1390,1394,1398,1402,1406,1410,1414,1418,1422,1426,1430,1434,1438,1442,1446,1450,1454,1458,1462,1466,1470,1474,1478,1482,1486,1490,1494,1498,1502,1506,1510,1514,1518,1522,1526,1530,1534,1538,1542,1546,1550,1554,1558,1562,1566,1570,1574,1578,1582,1586,1590,1594,1598,1602,1606,1610,1614,1618,1622,1626,1630,1634,1638,1642,1646,1650],{"title":1371,"path":1372,"stem":1373},"Що таке десктопна розробка?","\u002Fcsharp\u002Fdesktop-ui\u002Fwhat-is-desktop-dev","01.csharp\u002F12.desktop-ui\u002F01.what-is-desktop-dev",{"title":1375,"path":1376,"stem":1377},"Архітектура WPF — як влаштований графічний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-architecture","01.csharp\u002F12.desktop-ui\u002F02.wpf-architecture",{"title":1379,"path":1380,"stem":1381},"Перший WPF-проєкт — від нуля до вікна","\u002Fcsharp\u002Fdesktop-ui\u002Ffirst-wpf-app","01.csharp\u002F12.desktop-ui\u002F03.first-wpf-app",{"title":1383,"path":1384,"stem":1385},"Перший Avalonia-проєкт: WPF для всіх платформ","\u002Fcsharp\u002Fdesktop-ui\u002F03a.first-avalonia-app","01.csharp\u002F12.desktop-ui\u002F03a.first-avalonia-app",{"title":1387,"path":1388,"stem":1389},"XAML: декларативний інтерфейс","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-basics","01.csharp\u002F12.desktop-ui\u002F04.xaml-basics",{"title":1391,"path":1392,"stem":1393},"Fluent UI у WPF — сучасний дизайн Windows 11","\u002Fcsharp\u002Fdesktop-ui\u002F04a.wpf-fluent-ui","01.csharp\u002F12.desktop-ui\u002F04a.wpf-fluent-ui",{"title":1395,"path":1396,"stem":1397},"WPF UI — сучасна бібліотека Fluent контролів","\u002Fcsharp\u002Fdesktop-ui\u002F04b.wpf-ui-library","01.csharp\u002F12.desktop-ui\u002F04b.wpf-ui-library",{"title":1399,"path":1400,"stem":1401},"HandyControl — велика бібліотека UI контролів для WPF","\u002Fcsharp\u002Fdesktop-ui\u002F04c.handycontrol-library","01.csharp\u002F12.desktop-ui\u002F04c.handycontrol-library",{"title":1403,"path":1404,"stem":1405},"Простори імен та ресурси XAML","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-namespaces-resources","01.csharp\u002F12.desktop-ui\u002F05.xaml-namespaces-resources",{"title":1407,"path":1408,"stem":1409},"XAML в Avalonia: ключові відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F05a.avalonia-xaml-differences","01.csharp\u002F12.desktop-ui\u002F05a.avalonia-xaml-differences",{"title":1411,"path":1412,"stem":1413},"Розширення розмітки XAML (Markup Extensions)","\u002Fcsharp\u002Fdesktop-ui\u002Fxaml-markup-extensions","01.csharp\u002F12.desktop-ui\u002F06.xaml-markup-extensions",{"title":1415,"path":1416,"stem":1417},"Панелі Layout: StackPanel, WrapPanel, DockPanel","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part1","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part1",{"title":1419,"path":1420,"stem":1421},"Grid, Canvas, UniformGrid","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-panels-part2","01.csharp\u002F12.desktop-ui\u002F07.layout-panels-part2",{"title":1423,"path":1424,"stem":1425},"Просунуті техніки Layout","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-advanced","01.csharp\u002F12.desktop-ui\u002F08.layout-advanced",{"title":1427,"path":1428,"stem":1429},"Адаптивний Layout та найкращі практики","\u002Fcsharp\u002Fdesktop-ui\u002Flayout-responsive","01.csharp\u002F12.desktop-ui\u002F09.layout-responsive",{"title":1431,"path":1432,"stem":1433},"Layout в Avalonia: відмінності та нові можливості","\u002Fcsharp\u002Fdesktop-ui\u002F09a.layout-avalonia","01.csharp\u002F12.desktop-ui\u002F09a.layout-avalonia",{"title":1435,"path":1436,"stem":1437},"Button, Image, ProgressBar та інші базові контроли","\u002Fcsharp\u002Fdesktop-ui\u002Fbasic-controls","01.csharp\u002F12.desktop-ui\u002F10.basic-controls",{"title":1439,"path":1440,"stem":1441},"Контроли в Avalonia: відмінності від WPF","\u002Fcsharp\u002Fdesktop-ui\u002F10a.controls-avalonia","01.csharp\u002F12.desktop-ui\u002F10a.controls-avalonia",{"title":1443,"path":1444,"stem":1445},"Текстові контроли — TextBlock, TextBox, RichTextBox","\u002Fcsharp\u002Fdesktop-ui\u002Ftext-controls","01.csharp\u002F12.desktop-ui\u002F11.text-controls",{"title":1447,"path":1448,"stem":1449},"Контроли вибору — CheckBox, RadioButton, ComboBox, ListBox, DatePicker","\u002Fcsharp\u002Fdesktop-ui\u002Fselection-controls","01.csharp\u002F12.desktop-ui\u002F12.selection-controls",{"title":1451,"path":1452,"stem":1453},"Content Model — GroupBox, Expander, TabControl, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fcontent-controls","01.csharp\u002F12.desktop-ui\u002F13.content-controls",{"title":1455,"path":1456,"stem":1457},"UI\u002FUX принципи десктопних застосунків","\u002Fcsharp\u002Fdesktop-ui\u002F13a.ui-ux-principles","01.csharp\u002F12.desktop-ui\u002F13a.ui-ux-principles",{"title":1459,"path":1460,"stem":1461},"Dependency Properties — Концепція та Value Resolution","\u002Fcsharp\u002Fdesktop-ui\u002Fdependency-properties-part1","01.csharp\u002F12.desktop-ui\u002F14.dependency-properties-part1",{"title":1463,"path":1464,"stem":1465},"Avalonia Property System — StyledProperty та DirectProperty","\u002Fcsharp\u002Fdesktop-ui\u002F14a.avalonia-property-system","01.csharp\u002F12.desktop-ui\u002F14a.avalonia-property-system",{"title":1467,"path":1468,"stem":1469},"Attached Properties — Властивості без меж","\u002Fcsharp\u002Fdesktop-ui\u002Fattached-properties","01.csharp\u002F12.desktop-ui\u002F15.attached-properties",{"title":1471,"path":1472,"stem":1473},"Routed Events — Маршрутизація подій у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Frouted-events","01.csharp\u002F12.desktop-ui\u002F16.routed-events",{"title":1475,"path":1476,"stem":1477},"Data Binding — Від Code-Behind до Декларативності","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part1","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part1",{"title":1479,"path":1480,"stem":1481},"INotifyPropertyChanged — Живе оновлення UI","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-basics-part2","01.csharp\u002F12.desktop-ui\u002F17.data-binding-basics-part2",{"title":1483,"path":1484,"stem":1485},"Compiled Bindings в Avalonia — Безпека на етапі компіляції","\u002Fcsharp\u002Fdesktop-ui\u002F17a.avalonia-compiled-bindings","01.csharp\u002F12.desktop-ui\u002F17a.avalonia-compiled-bindings",{"title":1487,"path":1488,"stem":1489},"Просунутий Data Binding — ElementName, RelativeSource, MultiBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-binding-advanced","01.csharp\u002F12.desktop-ui\u002F18.data-binding-advanced",{"title":1491,"path":1492,"stem":1493},"Value Converters — Перетворення типів даних у Data Binding","\u002Fcsharp\u002Fdesktop-ui\u002Fvalue-converters","01.csharp\u002F12.desktop-ui\u002F19.value-converters",{"title":1495,"path":1496,"stem":1497},"Data Templates — Візуалізація об'єктів у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-templates","01.csharp\u002F12.desktop-ui\u002F20.data-templates",{"title":1499,"path":1500,"stem":1501},"Collections Binding Part 1 — ObservableCollection та ItemsControl","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part1","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part1",{"title":1503,"path":1504,"stem":1505},"Collections Binding Part 2 — ICollectionView, Filtering, Sorting та Virtualization","\u002Fcsharp\u002Fdesktop-ui\u002Fcollections-binding-part2","01.csharp\u002F12.desktop-ui\u002F21.collections-binding-part2",{"title":1507,"path":1508,"stem":1509},"MVVM Pattern — Від Spaghetti Code до архітектури","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-pattern","01.csharp\u002F12.desktop-ui\u002F22.mvvm-pattern",{"title":1511,"path":1512,"stem":1513},"ViewModel Implementation — Від BaseViewModel до валідації","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-implementation","01.csharp\u002F12.desktop-ui\u002F23.viewmodel-implementation",{"title":1515,"path":1516,"stem":1517},"Commands — Від event handlers до декларативних команд","\u002Fcsharp\u002Fdesktop-ui\u002Fcommands","01.csharp\u002F12.desktop-ui\u002F24.commands",{"title":1519,"path":1520,"stem":1521},"MVVM Toolkit — MVVM без boilerplate через Source Generators","\u002Fcsharp\u002Fdesktop-ui\u002Fmvvm-toolkit","01.csharp\u002F12.desktop-ui\u002F25.mvvm-toolkit",{"title":1523,"path":1524,"stem":1525},"Messenger Pattern — Комунікація між ViewModel без прямих посилань","\u002Fcsharp\u002Fdesktop-ui\u002Fmessenger-pattern","01.csharp\u002F12.desktop-ui\u002F26.messenger-pattern",{"title":1527,"path":1528,"stem":1529},"Стилі WPF — CSS для десктопу","\u002Fcsharp\u002Fdesktop-ui\u002Fstyles-basics","01.csharp\u002F12.desktop-ui\u002F27.styles-basics",{"title":1531,"path":1532,"stem":1533},"CSS-like стилі Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F27a.avalonia-css-styling","01.csharp\u002F12.desktop-ui\u002F27a.avalonia-css-styling",{"title":1535,"path":1536,"stem":1537},"Control Templates — Частина 1. Концепція та TemplateBinding","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part1","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part1",{"title":1539,"path":1540,"stem":1541},"Control Templates — Частина 2. Named Parts та ContentPresenter","\u002Fcsharp\u002Fdesktop-ui\u002Fcontrol-templates-part2","01.csharp\u002F12.desktop-ui\u002F28.control-templates-part2",{"title":1543,"path":1544,"stem":1545},"Control Themes в Avalonia — нова ера стилізації","\u002Fcsharp\u002Fdesktop-ui\u002F28a.avalonia-control-themes","01.csharp\u002F12.desktop-ui\u002F28a.avalonia-control-themes",{"title":1547,"path":1548,"stem":1549},"Triggers та Visual State Manager у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Ftriggers-visual-states","01.csharp\u002F12.desktop-ui\u002F29.triggers-visual-states",{"title":1551,"path":1552,"stem":1553},"Pseudo-classes в Avalonia — замість WPF Triggers","\u002Fcsharp\u002Fdesktop-ui\u002F29a.avalonia-pseudo-classes","01.csharp\u002F12.desktop-ui\u002F29a.avalonia-pseudo-classes",{"title":1555,"path":1556,"stem":1557},"Теми та ресурсні словники у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fresources-themes","01.csharp\u002F12.desktop-ui\u002F30.resources-themes",{"title":1559,"path":1560,"stem":1561},"Avalonia Themes — Fluent Design та система тематизації","\u002Fcsharp\u002Fdesktop-ui\u002F30a.avalonia-themes-fluent","01.csharp\u002F12.desktop-ui\u002F30a.avalonia-themes-fluent",{"title":1563,"path":1564,"stem":1565},"Контроли колекцій — глибоке занурення","\u002Fcsharp\u002Fdesktop-ui\u002Fcollection-controls","01.csharp\u002F12.desktop-ui\u002F31.collection-controls",{"title":1567,"path":1568,"stem":1569},"DataGrid — колонки та базове відображення","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part1","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part1",{"title":1571,"path":1572,"stem":1573},"DataGrid — сортування, фільтрація, редагування","\u002Fcsharp\u002Fdesktop-ui\u002Fdatagrid-part2","01.csharp\u002F12.desktop-ui\u002F32.datagrid-part2",{"title":1575,"path":1576,"stem":1577},"TreeView та GridView","\u002Fcsharp\u002Fdesktop-ui\u002Ftreeview-listview","01.csharp\u002F12.desktop-ui\u002F33.treeview-listview",{"title":1579,"path":1580,"stem":1581},"Меню, Toolbar, ContextMenu, StatusBar","\u002Fcsharp\u002Fdesktop-ui\u002Fmenus-toolbars","01.csharp\u002F12.desktop-ui\u002F34.menus-toolbars",{"title":1583,"path":1584,"stem":1585},"Навігація та керування вікнами. Частина 1: вікна та сторінки","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part1","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part1",{"title":1587,"path":1588,"stem":1589},"Навігація та керування вікнами. Частина 2: MVVM-навігація","\u002Fcsharp\u002Fdesktop-ui\u002Fnavigation-windows-part2","01.csharp\u002F12.desktop-ui\u002F35.navigation-windows-part2",{"title":1591,"path":1592,"stem":1593},"Avalonia — Навігація та діалоги","\u002Fcsharp\u002Fdesktop-ui\u002F35a.avalonia-navigation-dialogs","01.csharp\u002F12.desktop-ui\u002F35a.avalonia-navigation-dialogs",{"title":1595,"path":1596,"stem":1597},"Діалоги та File Pickers у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fdialogs-file-pickers","01.csharp\u002F12.desktop-ui\u002F36.dialogs-file-pickers",{"title":1599,"path":1600,"stem":1601},"UserControl: компонентний підхід у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fuser-controls","01.csharp\u002F12.desktop-ui\u002F37.user-controls",{"title":1603,"path":1604,"stem":1605},"Custom Controls: Lookless Controls у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fcustom-controls","01.csharp\u002F12.desktop-ui\u002F38.custom-controls",{"title":1607,"path":1608,"stem":1609},"Avalonia TemplatedControl — Lookless Controls","\u002Fcsharp\u002Fdesktop-ui\u002F38a.avalonia-templated-controls","01.csharp\u002F12.desktop-ui\u002F38a.avalonia-templated-controls",{"title":1611,"path":1612,"stem":1613},"Анімації у WPF: Storyboard та Easing Functions","\u002Fcsharp\u002Fdesktop-ui\u002Fanimations-transitions","01.csharp\u002F12.desktop-ui\u002F39.animations-transitions",{"title":1615,"path":1616,"stem":1617},"Анімації в Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002F39a.avalonia-animations","01.csharp\u002F12.desktop-ui\u002F39a.avalonia-animations",{"title":1619,"path":1620,"stem":1621},"2D Графіка та Мультимедіа у WPF","\u002Fcsharp\u002Fdesktop-ui\u002Fmedia-graphics","01.csharp\u002F12.desktop-ui\u002F40.media-graphics",{"title":1623,"path":1624,"stem":1625},"Dependency Injection у WPF та Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Fdi-integration","01.csharp\u002F12.desktop-ui\u002F41.di-integration",{"title":1627,"path":1628,"stem":1629},"SQLite та EF Core у десктопних додатках","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part1","01.csharp\u002F12.desktop-ui\u002F42.data-persistence-part1",{"title":1631,"path":1632,"stem":1633},"Repository Pattern та Unit of Work","\u002Fcsharp\u002Fdesktop-ui\u002Fdata-persistence-part2","01.csharp\u002F12.desktop-ui\u002F43.data-persistence-part2",{"title":1635,"path":1636,"stem":1637},"Тестування ViewModels","\u002Fcsharp\u002Fdesktop-ui\u002Fviewmodel-testing","01.csharp\u002F12.desktop-ui\u002F44.viewmodel-testing",{"title":1639,"path":1640,"stem":1641},"Avalonia Headless Testing — тестування UI без вікон","\u002Fcsharp\u002Fdesktop-ui\u002F44a.avalonia-headless-testing","01.csharp\u002F12.desktop-ui\u002F44a.avalonia-headless-testing",{"title":1643,"path":1644,"stem":1645},"Кросплатформна розробка з Avalonia","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-cross-platform","01.csharp\u002F12.desktop-ui\u002F45.avalonia-cross-platform",{"title":1647,"path":1648,"stem":1649},"Пакування та розгортання Avalonia додатків","\u002Fcsharp\u002Fdesktop-ui\u002Favalonia-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F46.avalonia-packaging-deployment",{"title":1651,"path":1652,"stem":1653},"Розгортання WPF застосунків","\u002Fcsharp\u002Fdesktop-ui\u002Fwpf-packaging-deployment","01.csharp\u002F12.desktop-ui\u002F47.wpf-packaging-deployment",{"title":1655,"icon":658,"path":1656,"stem":1657,"children":1658,"page":59},"Network Programming","\u002Fcsharp\u002Fnetwork-programming","01.csharp\u002F13.network-programming",[1659,1663,1667,1671,1675,1679,1683,1687,1691,1695,1699],{"title":1660,"path":1661,"stem":1662},"Основи комп'ютерних мереж","\u002Fcsharp\u002Fnetwork-programming\u002Ffoundations","01.csharp\u002F13.network-programming\u002F01.foundations",{"title":1664,"path":1665,"stem":1666},"Модель OSI та стек TCP\u002FIP","\u002Fcsharp\u002Fnetwork-programming\u002Fosi-model","01.csharp\u002F13.network-programming\u002F02.osi-model",{"title":1668,"path":1669,"stem":1670},"IP-протокол та адресація","\u002Fcsharp\u002Fnetwork-programming\u002Fip-addressing","01.csharp\u002F13.network-programming\u002F03.ip-addressing",{"title":1672,"path":1673,"stem":1674},"UDP — протокол без з'єднання","\u002Fcsharp\u002Fnetwork-programming\u002Fudp","01.csharp\u002F13.network-programming\u002F05.udp",{"title":1676,"path":1677,"stem":1678},"UDP Broadcast та Multicast","\u002Fcsharp\u002Fnetwork-programming\u002Fudp-broadcast-multicast","01.csharp\u002F13.network-programming\u002F06.udp-broadcast-multicast",{"title":1680,"path":1681,"stem":1682},"HTTP — протокол вебу","\u002Fcsharp\u002Fnetwork-programming\u002Fhttp-fundamentals","01.csharp\u002F13.network-programming\u002F07.http-fundamentals",{"title":1684,"path":1685,"stem":1686},"HttpListener — вбудований HTTP-сервер .NET","\u002Fcsharp\u002Fnetwork-programming\u002F07a.http-listener","01.csharp\u002F13.network-programming\u002F07a.http-listener",{"title":1688,"path":1689,"stem":1690},"HTTP Advanced — cookies, аутентифікація та HTTPS","\u002Fcsharp\u002Fnetwork-programming\u002Fhttp-advanced","01.csharp\u002F13.network-programming\u002F08.http-advanced",{"title":1692,"path":1693,"stem":1694},"SMTP та протоколи електронної пошти","\u002Fcsharp\u002Fnetwork-programming\u002Fsmtp","01.csharp\u002F13.network-programming\u002F09.smtp",{"title":1696,"path":1697,"stem":1698},"WebSocket — повнодуплексний протокол реального часу","\u002Fcsharp\u002Fnetwork-programming\u002Fwebsockets","01.csharp\u002F13.network-programming\u002F10.websockets",{"title":1700,"path":1701,"stem":1702},"TLS\u002FSSL — криптографічний захист мережевих з'єднань","\u002Fcsharp\u002Fnetwork-programming\u002Ftls-ssl","01.csharp\u002F13.network-programming\u002F11.tls-ssl",{"title":1704,"path":1705,"stem":1706},"C# & .NET: The Ultimate Roadmap","\u002Fcsharp\u002Froadmap","01.csharp\u002Froadmap",{"title":1708,"icon":1709,"path":1710,"stem":1711,"children":1712,"page":59},"C++","i-devicon-cplusplus","\u002Fcpp","02.cpp",[1713,1717,1721,1725,1729,1733,1737,1741,1745,1748,1752,1756,1760,1764,1768,1772,1776,1780,1784,1788,1792,1796,1800,1804,1808,1812,1816,1820,1824,1828,1832,1836,1840,1844,1848,1852,1856,1860,1864,1868,1872,1876,1880,1884,1888,1892],{"title":1714,"path":1715,"stem":1716},"Вступ у програмування та алгоритми","\u002Fcpp\u002Fintro-algorithms","02.cpp\u002F01.intro-algorithms",{"title":1718,"path":1719,"stem":1720},"Code Style: угоди про оформлення коду","\u002Fcpp\u002Fcode-style","02.cpp\u002F02.code-style",{"title":1722,"path":1723,"stem":1724},"Середовище розробки та перший проєкт","\u002Fcpp\u002Fide-setup","02.cpp\u002F03.ide-setup",{"title":1726,"path":1727,"stem":1728},"Вивід даних на екран","\u002Fcpp\u002Fdata-output","02.cpp\u002F04.data-output",{"title":1730,"path":1731,"stem":1732},"Типи даних, змінні та константи","\u002Fcpp\u002Fdata-types-variables","02.cpp\u002F05.data-types-variables",{"title":1734,"path":1735,"stem":1736},"Ввід даних з клавіатури","\u002Fcpp\u002Fdata-input","02.cpp\u002F06.data-input",{"title":1738,"path":1739,"stem":1740},"Оператори, перетворення типів та логічні операції","\u002Fcpp\u002Foperators-type-conversion","02.cpp\u002F07.operators-type-conversion",{"title":1742,"path":1743,"stem":1744},"Цикли","\u002Fcpp\u002Floops","02.cpp\u002F08.loops",{"title":32,"path":1746,"stem":1747},"\u002Fcpp\u002Farrays","02.cpp\u002F09.arrays",{"title":1749,"path":1750,"stem":1751},"Алгоритми сортування та аналіз складності","\u002Fcpp\u002Fsorting","02.cpp\u002F10.sorting",{"title":1753,"path":1754,"stem":1755},"Алгоритми пошуку","\u002Fcpp\u002Fsearching","02.cpp\u002F11.searching",{"title":1757,"path":1758,"stem":1759},"Функції: основи","\u002Fcpp\u002Ffunctions-basics","02.cpp\u002F12.functions-basics",{"title":1761,"path":1762,"stem":1763},"Функції: прототипи, область видимості та додаткові можливості","\u002Fcpp\u002Ffunctions-scope","02.cpp\u002F13.functions-scope",{"title":1765,"path":1766,"stem":1767},"Функції: перевантаження та шаблони","\u002Fcpp\u002Ffunctions-overloading-templates","02.cpp\u002F14.functions-overloading-templates",{"title":1769,"path":1770,"stem":1771},"Вказівники: основи","\u002Fcpp\u002Fpointers-basics","02.cpp\u002F15.pointers-basics",{"title":1773,"path":1774,"stem":1775},"Посилання (References)","\u002Fcpp\u002Freferences","02.cpp\u002F16.references",{"title":1777,"path":1778,"stem":1779},"Вказівники, const і масиви","\u002Fcpp\u002Fpointers-const-arrays","02.cpp\u002F17.pointers-const-arrays",{"title":1781,"path":1782,"stem":1783},"Адресна арифметика","\u002Fcpp\u002Fpointer-arithmetic","02.cpp\u002F18.pointer-arithmetic",{"title":1785,"path":1786,"stem":1787},"Динамічна пам'ять","\u002Fcpp\u002Fdynamic-memory","02.cpp\u002F19.dynamic-memory",{"title":1789,"path":1790,"stem":1791},"Вказівники типу void","\u002Fcpp\u002Fvoid-pointers","02.cpp\u002F20.void-pointers",{"title":1793,"path":1794,"stem":1795},"Вказівники на вказівники","\u002Fcpp\u002Fpointers-to-pointers","02.cpp\u002F21.pointers-to-pointers",{"title":1797,"path":1798,"stem":1799},"Оператор доступу до членів через вказівник (->)","\u002Fcpp\u002Fmember-access-operator","02.cpp\u002F22.member-access-operator",{"title":1801,"path":1802,"stem":1803},"Цикл for-each (Range-based for)","\u002Fcpp\u002Fforeach-loop","02.cpp\u002F23.foreach-loop",{"title":1805,"path":1806,"stem":1807},"Вказівники на функції","\u002Fcpp\u002Ffunction-pointers","02.cpp\u002F24.function-pointers",{"title":1809,"path":1810,"stem":1811},"Лямбда-вирази","\u002Fcpp\u002Flambdas","02.cpp\u002F25.lambdas",{"title":1813,"path":1814,"stem":1815},"Лямбда-захоплення","\u002Fcpp\u002Flambda-captures","02.cpp\u002F26.lambda-captures",{"title":1817,"path":1818,"stem":1819},"Еліпсис","\u002Fcpp\u002Fellipsis","02.cpp\u002F27.ellipsis",{"title":1821,"path":1822,"stem":1823},"Безпечні альтернативи еліпсису","\u002Fcpp\u002F27a.ellipsis","02.cpp\u002F27a.ellipsis",{"title":1825,"path":1826,"stem":1827},"Аргументи командного рядка","\u002Fcpp\u002Fcommand-line-arguments","02.cpp\u002F28.command-line-arguments",{"title":1829,"path":1830,"stem":1831},"Перерахування (enum)","\u002Fcpp\u002Fenum","02.cpp\u002F29.enum",{"title":1833,"path":1834,"stem":1835},"Класи-перерахування (enum class)","\u002Fcpp\u002Fenum-class","02.cpp\u002F30.enum-class",{"title":1837,"path":1838,"stem":1839},"Псевдоніми типів (typedef і using)","\u002Fcpp\u002Ftype-aliases","02.cpp\u002F31.type-aliases",{"title":1841,"path":1842,"stem":1843},"Системи числення та двійкова арифметика","\u002Fcpp\u002Fnumber-systems","02.cpp\u002F32.number-systems",{"title":1845,"path":1846,"stem":1847},"Структури (struct): агрегування даних","\u002Fcpp\u002Fstruct","02.cpp\u002F33.struct",{"title":1849,"path":1850,"stem":1851},"Структури у функціях","\u002Fcpp\u002Fstruct-functions","02.cpp\u002F34.struct-functions",{"title":1853,"path":1854,"stem":1855},"Масиви структур і вкладені структури","\u002Fcpp\u002Fstruct-arrays","02.cpp\u002F35.struct-arrays",{"title":1857,"path":1858,"stem":1859},"Патерни struct та межі застосування","\u002Fcpp\u002Fstruct-patterns","02.cpp\u002F36.struct-patterns",{"title":1861,"path":1862,"stem":1863},"Символи та таблиця ASCII","\u002Fcpp\u002Fascii-characters","02.cpp\u002F37.ascii-characters",{"title":1865,"path":1866,"stem":1867},"Unicode та кодування UTF","\u002Fcpp\u002Funicode-utf","02.cpp\u002F38.unicode-utf",{"title":1869,"path":1870,"stem":1871},"C-style рядки","\u002Fcpp\u002Fc-strings","02.cpp\u002F39.c-strings",{"title":1873,"path":1874,"stem":1875},"Вступ до std::string","\u002Fcpp\u002Fstd-string-intro","02.cpp\u002F40.std-string-intro",{"title":1877,"path":1878,"stem":1879},"Довжина, ємність та доступ до символів std::string","\u002Fcpp\u002Fstd-string-capacity-access","02.cpp\u002F41.std-string-capacity-access",{"title":1881,"path":1882,"stem":1883},"Модифікація std::string: присвоювання, додавання, вставка, видалення та заміна","\u002Fcpp\u002Fstd-string-modification","02.cpp\u002F42.std-string-modification",{"title":1885,"path":1886,"stem":1887},"Пошук у std::string: find, npos та практичні патерни","\u002Fcpp\u002Fstd-string-search","02.cpp\u002F43.std-string-search",{"title":1889,"path":1890,"stem":1891},"std::string_view: невласницький погляд на рядок без копіювання","\u002Fcpp\u002Fstd-string-view","02.cpp\u002F44.std-string-view",{"title":1893,"path":1894,"stem":1895},"План навчання: Курс C++ — Продовження (Статті 29–60+)","\u002Fcpp\u002Fcurriculum-plan","02.cpp\u002Fcurriculum-plan",{"title":1897,"icon":1898,"path":1899,"stem":1900,"children":1901,"page":59},"JavaScript","i-devicon-javascript","\u002Fjavascript","03.javascript",[1902,1928,1982,2004,2308,2346],{"title":1903,"icon":1904,"path":1905,"stem":1906,"children":1907,"page":59},"Events","i-lucide-mouse-pointer-click","\u002Fjavascript\u002Fevents","03.javascript\u002F01.events",[1908,1912,1916,1920,1924],{"title":1909,"path":1910,"stem":1911},"Вступ до подій браузера","\u002Fjavascript\u002Fevents\u002Fintro","03.javascript\u002F01.events\u002F01.intro",{"title":1913,"path":1914,"stem":1915},"Бульбашковий механізм (Bubbling) та занурення (Capturing)","\u002Fjavascript\u002Fevents\u002Fbubbling-capturing","03.javascript\u002F01.events\u002F02.bubbling-capturing",{"title":1917,"path":1918,"stem":1919},"Делегування подій (Event Delegation)","\u002Fjavascript\u002Fevents\u002Fdelegate-events","03.javascript\u002F01.events\u002F03.delegate-events",{"title":1921,"path":1922,"stem":1923},"Типові дії браузера та preventDefault()","\u002Fjavascript\u002Fevents\u002Fprevent-default","03.javascript\u002F01.events\u002F04.prevent-default",{"title":1925,"path":1926,"stem":1927},"Запуск користувацьких подій (Custom Events)","\u002Fjavascript\u002Fevents\u002Fcustom-events","03.javascript\u002F01.events\u002F05.custom-events",{"title":1929,"icon":1930,"path":1931,"stem":1932,"children":1933,"page":59},"Network","i-lucide-globe","\u002Fjavascript\u002Fnetwork","03.javascript\u002F02.network",[1934,1938,1942,1946,1950,1954,1958,1962,1966,1970,1974,1978],{"title":1935,"path":1936,"stem":1937},"Fetch API - Сучасний підхід до HTTP-запитів","\u002Fjavascript\u002Fnetwork\u002F01-fetch-api","03.javascript\u002F02.network\u002F01-fetch-api",{"title":1939,"path":1940,"stem":1941},"FormData - Робота з формами та файлами","\u002Fjavascript\u002Fnetwork\u002F02-formdata","03.javascript\u002F02.network\u002F02-formdata",{"title":1943,"path":1944,"stem":1945},"Відстеження прогресу завантаження","\u002Fjavascript\u002Fnetwork\u002F03-download-progress","03.javascript\u002F02.network\u002F03-download-progress",{"title":1947,"path":1948,"stem":1949},"Переривання fetch-запитів","\u002Fjavascript\u002Fnetwork\u002F04-abort-requests","03.javascript\u002F02.network\u002F04-abort-requests",{"title":1951,"path":1952,"stem":1953},"CORS - Запити між різними джерелами","\u002Fjavascript\u002Fnetwork\u002F05-cors","03.javascript\u002F02.network\u002F05-cors",{"title":1955,"path":1956,"stem":1957},"Fetch API - Повний довідник опцій","\u002Fjavascript\u002Fnetwork\u002F06-fetch-options","03.javascript\u002F02.network\u002F06-fetch-options",{"title":1959,"path":1960,"stem":1961},"URL Objects - Робота з посиланнями","\u002Fjavascript\u002Fnetwork\u002F07-url-objects","03.javascript\u002F02.network\u002F07-url-objects",{"title":1963,"path":1964,"stem":1965},"XMLHttpRequest - AJAX та низькорівневі запити","\u002Fjavascript\u002Fnetwork\u002F08-xmlhttprequest","03.javascript\u002F02.network\u002F08-xmlhttprequest",{"title":1967,"path":1968,"stem":1969},"Відновлюване завантаження файлів","\u002Fjavascript\u002Fnetwork\u002F09-resumable-upload","03.javascript\u002F02.network\u002F09-resumable-upload",{"title":1971,"path":1972,"stem":1973},"Cookies, document.cookie та світ після \"Cookiepocalypse\"","\u002Fjavascript\u002Fnetwork\u002F10-cookies","03.javascript\u002F02.network\u002F10-cookies",{"title":1975,"path":1976,"stem":1977},"js-cookie: Керування Cookies без Болю","\u002Fjavascript\u002Fnetwork\u002F11-js-cookie","03.javascript\u002F02.network\u002F11-js-cookie",{"title":1979,"path":1980,"stem":1981},"Axios: Потужний HTTP-клієнт для JavaScript","\u002Fjavascript\u002Fnetwork\u002F12-axios","03.javascript\u002F02.network\u002F12-axios",{"title":1983,"icon":1984,"path":1985,"stem":1986,"children":1987,"page":59},"Bom","i-lucide-monitor","\u002Fjavascript\u002Fbom","03.javascript\u002F03.bom",[1988,1992,1996,2000],{"title":1989,"path":1990,"stem":1991},"LocalStorage, SessionStorage та patterns збереження даних","\u002Fjavascript\u002Fbom\u002F01-localstorage","03.javascript\u002F03.bom\u002F01-localstorage",{"title":1993,"path":1994,"stem":1995},"Location Object - Керування адресою сторінки","\u002Fjavascript\u002Fbom\u002F02-location-object","03.javascript\u002F03.bom\u002F02-location-object",{"title":1997,"path":1998,"stem":1999},"History API - Керування історією браузера","\u002Fjavascript\u002Fbom\u002F03-history-api","03.javascript\u002F03.bom\u002F03-history-api",{"title":2001,"path":2002,"stem":2003},"Navigator Object - Ідентифікація та Можливості Пристрою","\u002Fjavascript\u002Fbom\u002F04-navigator-object","03.javascript\u002F03.bom\u002F04-navigator-object",{"title":2005,"icon":2006,"path":2007,"stem":2008,"children":2009},"React","i-devicon-react","\u002Fjavascript\u002Freact","03.javascript\u002F04.react\u002Findex",[2010,2011,2015,2019,2023,2027,2090,2125,2277],{"title":2005,"path":2007,"stem":2008},{"title":2012,"path":2013,"stem":2014},"Робота з Формами в React","\u002Fjavascript\u002Freact\u002Freact-forms","03.javascript\u002F04.react\u002F01.react-forms",{"title":2016,"path":2017,"stem":2018},"React Hook Form: Професійна Робота з Формами","\u002Fjavascript\u002Freact\u002Freact-hook-form","03.javascript\u002F04.react\u002F02.react-hook-form",{"title":2020,"path":2021,"stem":2022},"React Hook Form: Глибоке Розуміння Архітектури та Оптимізації","\u002Fjavascript\u002Freact\u002Freact-hook-form-new","03.javascript\u002F04.react\u002F02.react-hook-form-new",{"title":2024,"path":2025,"stem":2026},"Axios та React: Професійна Архітектура Запитів","\u002Fjavascript\u002Freact\u002Fdata-fetching-axios","03.javascript\u002F04.react\u002F03.data-fetching-axios",{"title":2028,"icon":132,"path":2029,"stem":2030,"children":2031},"Tanstack Query","\u002Fjavascript\u002Freact\u002Ftanstack-query","03.javascript\u002F04.react\u002F04.tanstack-query\u002Findex",[2032,2034,2038,2042,2046,2050,2054,2058,2062,2066,2070,2074,2078,2082,2086],{"title":2033,"path":2029,"stem":2030},"TanStack Query: Майстерність Керування Станом Сервера",{"title":2035,"path":2036,"stem":2037},"Парадигма Server State: Чому useEffect недостатньо","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-state-paradigm","03.javascript\u002F04.react\u002F04.tanstack-query\u002F01.server-state-paradigm",{"title":2039,"path":2040,"stem":2041},"Встановлення та Налаштування: Фундамент","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Finstallation-and-devtools","03.javascript\u002F04.react\u002F04.tanstack-query\u002F02.installation-and-devtools",{"title":2043,"path":2044,"stem":2045},"Основи Запитів та Магія Ключів","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fquery-basics-and-keys","03.javascript\u002F04.react\u002F04.tanstack-query\u002F03.query-basics-and-keys",{"title":2047,"path":2048,"stem":2049},"Синхронізація Даних: Життєвий Цикл Запиту","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fdata-synchronization","03.javascript\u002F04.react\u002F04.tanstack-query\u002F04.data-synchronization",{"title":2051,"path":2052,"stem":2053},"Мутації та Інвалідація: Зміна Даних","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fmutations-and-invalidation","03.javascript\u002F04.react\u002F04.tanstack-query\u002F05.mutations-and-invalidation",{"title":2055,"path":2056,"stem":2057},"Оптимістичні Оновлення: Швидше за Світло","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Foptimistic-updates","03.javascript\u002F04.react\u002F04.tanstack-query\u002F06.optimistic-updates",{"title":2059,"path":2060,"stem":2061},"Пагінація та Infinite Scroll","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fpagination-and-load-more","03.javascript\u002F04.react\u002F04.tanstack-query\u002F07.pagination-and-load-more",{"title":2063,"path":2064,"stem":2065},"Просунуті Патерни та Оптимізація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F04.tanstack-query\u002F08.advanced-patterns",{"title":2067,"path":2068,"stem":2069},"Архітектура та Best Practices","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Farchitecture-and-best-practices","03.javascript\u002F04.react\u002F04.tanstack-query\u002F09.architecture-and-best-practices",{"title":2071,"path":2072,"stem":2073},"Server-Side Rendering (SSR) та Гідратація","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fserver-side-rendering","03.javascript\u002F04.react\u002F04.tanstack-query\u002F10.server-side-rendering",{"title":2075,"path":2076,"stem":2077},"Стратегії Тестування","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Ftesting-strategies","03.javascript\u002F04.react\u002F04.tanstack-query\u002F11.testing-strategies",{"title":2079,"path":2080,"stem":2081},"Аутентифікація та Обробка Помилок","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fauthentication-and-errors","03.javascript\u002F04.react\u002F04.tanstack-query\u002F12.authentication-and-errors",{"title":2083,"path":2084,"stem":2085},"React Suspense та Майбутнє","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Freact-suspense","03.javascript\u002F04.react\u002F04.tanstack-query\u002F13.react-suspense",{"title":2087,"path":2088,"stem":2089},"Глибоке Занурення в Продуктивність","\u002Fjavascript\u002Freact\u002Ftanstack-query\u002Fperformance-deep-dive","03.javascript\u002F04.react\u002F04.tanstack-query\u002F14.performance-deep-dive",{"title":2091,"icon":2006,"path":2092,"stem":2093,"children":2094},"React Router","\u002Fjavascript\u002Freact\u002Freact-router","03.javascript\u002F04.react\u002F05.react-router\u002Findex",[2095,2097,2101,2105,2109,2113,2117,2121],{"title":2096,"path":2092,"stem":2093},"React Router: Навігаційна система сучасного вебу",{"title":2098,"path":2099,"stem":2100},"Налаштування та Базовий Роутинг","\u002Fjavascript\u002Freact\u002Freact-router\u002Fsetup-and-basic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F01.setup-and-basic-routing",{"title":2102,"path":2103,"stem":2104},"Динамічна Навігація","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnavigation-and-links","03.javascript\u002F04.react\u002F05.react-router\u002F02.navigation-and-links",{"title":2106,"path":2107,"stem":2108},"Вкладені Маршрути та Макети","\u002Fjavascript\u002Freact\u002Freact-router\u002Fnested-routes-and-layouts","03.javascript\u002F04.react\u002F05.react-router\u002F03.nested-routes-and-layouts",{"title":2110,"path":2111,"stem":2112},"Динамічні Маршрути та Параметри","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdynamic-routing","03.javascript\u002F04.react\u002F05.react-router\u002F04.dynamic-routing",{"title":2114,"path":2115,"stem":2116},"Data APIs: Loaders та Actions","\u002Fjavascript\u002Freact\u002Freact-router\u002Fdata-loading","03.javascript\u002F04.react\u002F05.react-router\u002F05.data-loading",{"title":2118,"path":2119,"stem":2120},"Просунуті Патерни","\u002Fjavascript\u002Freact\u002Freact-router\u002Fadvanced-patterns","03.javascript\u002F04.react\u002F05.react-router\u002F06.advanced-patterns",{"title":2122,"path":2123,"stem":2124},"Legacy Routing: Компонентний підхід","\u002Fjavascript\u002Freact\u002Freact-router\u002Flegacy-routing","03.javascript\u002F04.react\u002F05.react-router\u002F07.legacy-routing",{"title":2126,"icon":132,"path":2127,"stem":2128,"children":2129},"Redux","\u002Fjavascript\u002Freact\u002Fredux","03.javascript\u002F04.react\u002F06.redux\u002Findex",[2130,2132,2148,2177,2186,2207,2223,2252],{"title":2131,"path":2127,"stem":2128},"Redux: Еволюція управління станом",{"title":14,"icon":15,"path":2133,"stem":2134,"children":2135,"page":59},"\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals",[2136,2140,2144],{"title":2137,"path":2138,"stem":2139},"Вступ до State Management","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fintro-state-management","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F01.intro-state-management",{"title":2141,"path":2142,"stem":2143},"Філософія Redux та Три Принципи","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fredux-philosophy","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F02.redux-philosophy",{"title":2145,"path":2146,"stem":2147},"Чисті функції та Іммутабельність","\u002Fjavascript\u002Freact\u002Fredux\u002Ffundamentals\u002Fpure-functions-immutability","03.javascript\u002F04.react\u002F06.redux\u002F01.fundamentals\u002F03.pure-functions-immutability",{"title":2149,"icon":132,"path":2150,"stem":2151,"children":2152,"page":59},"Classic Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux",[2153,2157,2161,2165,2169,2173],{"title":2154,"path":2155,"stem":2156},"Створення Store (Classic Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fstore-setup","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F01.store-setup",{"title":2158,"path":2159,"stem":2160},"Actions, Constants та Action Creators","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Factions-constants","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F02.actions-constants",{"title":2162,"path":2163,"stem":2164},"Логіка Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freducers","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F03.reducers",{"title":2166,"path":2167,"stem":2168},"Комбінування Reducers (Root Reducer)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fdata-flow","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F04.data-flow",{"title":2170,"path":2171,"stem":2172},"Підключення до React (React-Redux)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Freact-redux-connection","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F05.react-redux-connection",{"title":2174,"path":2175,"stem":2176},"Middleware та Асинхронність (Redux Thunk)","\u002Fjavascript\u002Freact\u002Fredux\u002Fclassic-redux\u002Fmiddleware-thunk","03.javascript\u002F04.react\u002F06.redux\u002F02.classic-redux\u002F06.middleware-thunk",{"title":2178,"icon":132,"path":2179,"stem":2180,"children":2181,"page":59},"Transition To Rtk","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk",[2182],{"title":2183,"path":2184,"stem":2185},"Проблеми класичного Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftransition-to-rtk\u002Fproblems-with-classic","03.javascript\u002F04.react\u002F06.redux\u002F03.transition-to-rtk\u002F01.problems-with-classic",{"title":2187,"icon":132,"path":2188,"stem":2189,"children":2190,"page":59},"Redux Toolkit","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit",[2191,2195,2199,2203],{"title":2192,"path":2193,"stem":2194},"Налаштування Store з configureStore","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fconfigure-store","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F01.configure-store",{"title":2196,"path":2197,"stem":2198},"createSlice: Революція в Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fcreate-slice","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F02.create-slice",{"title":2200,"path":2201,"stem":2202},"Асинхронність з createAsyncThunk","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fasync-thunks","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F03.async-thunks",{"title":2204,"path":2205,"stem":2206},"04. Entity Adapter: Керування нормалізованим станом","\u002Fjavascript\u002Freact\u002Fredux\u002Fredux-toolkit\u002Fentity-adapter","03.javascript\u002F04.react\u002F06.redux\u002F04.redux-toolkit\u002F04.entity-adapter",{"title":2208,"icon":92,"path":2209,"stem":2210,"children":2211,"page":59},"Advanced","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced",[2212,2216,2220],{"title":2213,"path":2214,"stem":2215},"Мемоізація та Селектори: Повний Гайд по Reselect","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Fselectors-reselect","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F01.selectors-reselect",{"title":2217,"path":2218,"stem":2219},"RTK Query: Архітектура Серверного Кешу","\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Frtk-query-intro","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F02.rtk-query-intro",{"title":2067,"path":2221,"stem":2222},"\u002Fjavascript\u002Freact\u002Fredux\u002Fadvanced\u002Farchitecture-best-practices","03.javascript\u002F04.react\u002F06.redux\u002F05.advanced\u002F03.architecture-best-practices",{"title":2224,"icon":132,"path":2225,"stem":2226,"children":2227,"page":59},"Project Kanban","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban",[2228,2232,2236,2240,2244,2248],{"title":2229,"path":2230,"stem":2231},"Проєкт: Kanban Board (Trello Clone)","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fproject-overview","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F01.project-overview",{"title":2233,"path":2234,"stem":2235},"Налаштування та Типізація","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fsetup-and-types","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F02.setup-and-types",{"title":2237,"path":2238,"stem":2239},"Board Slice: Серце Дошки","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fboard-slice","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F03.board-slice",{"title":2241,"path":2242,"stem":2243},"Логіка Drag & Drop","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Fdrag-and-drop-logic","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F04.drag-and-drop-logic",{"title":2245,"path":2246,"stem":2247},"Інтеграція з RTK Query","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Frtk-query-integration","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F05.rtk-query-integration",{"title":2249,"path":2250,"stem":2251},"Optimistic Updates","\u002Fjavascript\u002Freact\u002Fredux\u002Fproject-kanban\u002Foptimistic-updates","03.javascript\u002F04.react\u002F06.redux\u002F06.project-kanban\u002F06.optimistic-updates",{"title":2253,"icon":132,"path":2254,"stem":2255,"children":2256,"page":59},"Testing","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting","03.javascript\u002F04.react\u002F06.redux\u002F07.testing",[2257,2261,2265,2269,2273],{"title":2258,"path":2259,"stem":2260},"Тестування Redux","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Fintro-testing","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F01.intro-testing",{"title":2262,"path":2263,"stem":2264},"Тестування Reducers","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-reducers","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F02.testing-reducers",{"title":2266,"path":2267,"stem":2268},"Тестування Селекторів","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-selectors","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F03.testing-selectors",{"title":2270,"path":2271,"stem":2272},"Тестування Компонентів (Integration)","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-components","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F04.testing-components",{"title":2274,"path":2275,"stem":2276},"Тестування Async Thunks","\u002Fjavascript\u002Freact\u002Fredux\u002Ftesting\u002Ftesting-thunks","03.javascript\u002F04.react\u002F06.redux\u002F07.testing\u002F05.testing-thunks",{"title":2278,"icon":132,"path":2279,"stem":2280,"children":2281},"Ui Libraries","\u002Fjavascript\u002Freact\u002Fui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002Findex",[2282,2284,2288,2292,2296,2300,2304],{"title":2283,"path":2279,"stem":2280},"UI Бібліотеки в React",{"title":2285,"path":2286,"stem":2287},"Вступ до UI Бібліотек: Навіщо Винаходити Велосипед Двічі?","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fintroduction-to-ui-libraries","03.javascript\u002F04.react\u002F07.ui-libraries\u002F01.introduction-to-ui-libraries",{"title":2289,"path":2290,"stem":2291},"Філософія shadcn\u002Fui: \"Not a Component Library\"","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-philosophy","03.javascript\u002F04.react\u002F07.ui-libraries\u002F02.shadcn-philosophy",{"title":2293,"path":2294,"stem":2295},"Установка та Налаштування shadcn\u002Fui","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-installation","03.javascript\u002F04.react\u002F07.ui-libraries\u002F03.shadcn-installation",{"title":2297,"path":2298,"stem":2299},"Базові Компоненти shadcn\u002Fui: Фундамент Інтерфейсу","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-basics","03.javascript\u002F04.react\u002F07.ui-libraries\u002F04.shadcn-components-basics",{"title":2301,"path":2302,"stem":2303},"Компоненти Форм: Побудова Інтерактивних Form","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-forms","03.javascript\u002F04.react\u002F07.ui-libraries\u002F05.shadcn-components-forms",{"title":2305,"path":2306,"stem":2307},"Складні Компоненти: Dialog, Dropdown, Table та Command","\u002Fjavascript\u002Freact\u002Fui-libraries\u002Fshadcn-components-advanced","03.javascript\u002F04.react\u002F07.ui-libraries\u002F06.shadcn-components-advanced",{"title":2309,"icon":2310,"path":2311,"stem":2312,"children":2313,"page":59},"TypeScript","i-devicon-typescript","\u002Fjavascript\u002Ftypescript","03.javascript\u002F05.typescript",[2314,2318,2322,2326,2330,2334,2338,2342],{"title":2315,"path":2316,"stem":2317},"TypeScript: Броня для вашого коду","\u002Fjavascript\u002Ftypescript\u002Fintro-and-basic-types","03.javascript\u002F05.typescript\u002F01.intro-and-basic-types",{"title":2319,"path":2320,"stem":2321},"Майстерність Моделювання Даних: Інтерфейси та Просунуті Типи","\u002Fjavascript\u002Ftypescript\u002Finterfaces-and-advanced-types","03.javascript\u002F05.typescript\u002F02.interfaces-and-advanced-types",{"title":2323,"path":2324,"stem":2325},"Алхімія Типів: Generics та Utility Types","\u002Fjavascript\u002Ftypescript\u002Fgenerics-and-utilities","03.javascript\u002F05.typescript\u002F03.generics-and-utilities",{"title":2327,"path":2328,"stem":2329},"Архітектура та Шаблони: Класи в TypeScript","\u002Fjavascript\u002Ftypescript\u002Fclasses-and-oop","03.javascript\u002F05.typescript\u002F04.classes-and-oop",{"title":2331,"path":2332,"stem":2333},"Продакшн та Екосистема: Advanced Config & Workflow","\u002Fjavascript\u002Ftypescript\u002Fadvanced-patterns-and-config","03.javascript\u002F05.typescript\u002F05.advanced-patterns-and-config",{"title":2335,"path":2336,"stem":2337},"TypeScript у світі React","\u002Fjavascript\u002Ftypescript\u002Freact-basics","03.javascript\u002F05.typescript\u002F06.react-basics",{"title":2339,"path":2340,"stem":2341},"React + TypeScript: Продвинуті патерни","\u002Fjavascript\u002Ftypescript\u002Freact-advanced","03.javascript\u002F05.typescript\u002F07.react-advanced",{"title":2343,"path":2344,"stem":2345},"React + TypeScript: Екосистема та бібліотеки","\u002Fjavascript\u002Ftypescript\u002Freact-ecosystem","03.javascript\u002F05.typescript\u002F08.react-ecosystem",{"title":2347,"path":2348,"stem":2349},"Atomic Design","\u002Fjavascript\u002Fatomic-design","03.javascript\u002F2.atomic-design",{"title":2351,"icon":2352,"path":2353,"stem":2354,"children":2355,"page":59},"Java","i-devicon-java","\u002Fjava","04.java",[2356,2359,2362,2366,2370,2374,2378],{"title":162,"path":2357,"stem":2358},"\u002Fjava\u002Fdata-mapper-part1","04.java\u002F01.data-mapper-part1",{"title":166,"path":2360,"stem":2361},"\u002Fjava\u002Fdata-mapper-part2","04.java\u002F02.data-mapper-part2",{"title":2363,"path":2364,"stem":2365},"Service Layer: Організація бізнес-логіки","\u002Fjava\u002Fservice-layer","04.java\u002F03.service-layer",{"title":2367,"path":2368,"stem":2369},"Rich Domain Model та State Pattern","\u002Fjava\u002Frich-domain-model","04.java\u002F04.rich-domain-model",{"title":2371,"path":2372,"stem":2373},"Патерни для складної бізнес-логіки","\u002Fjava\u002Fbusiness-logic-patterns","04.java\u002F05.business-logic-patterns",{"title":2375,"path":2376,"stem":2377},"Обробка помилок та валідація","\u002Fjava\u002Ferror-handling-validation","04.java\u002F06.error-handling-validation",{"title":2379,"path":2380,"stem":2381,"children":2382,"page":59},"Проектування баз даних","\u002Fjava\u002Fpr2","04.java\u002Fpr2",[2383,2387,2391,2395,2399,2403,2407,2411,2415,2419,2423,2427,2431,2435,2439,2443,2447,2451,2455,2459,2463,2467,2471,2475,2479,2483,2487,2491,2495,2499,2503,2507,2511,2515,2519,2523,2527],{"title":2384,"path":2385,"stem":2386},"Концептуальне моделювання: Мистецтво розуміння предметної області","\u002Fjava\u002Fpr2\u002Fconceptual-modeling","04.java\u002Fpr2\u002F01.conceptual-modeling",{"title":2388,"path":2389,"stem":2390},"Логічне моделювання: Від бізнес-ідей до структур даних","\u002Fjava\u002Fpr2\u002Flogical-modeling","04.java\u002Fpr2\u002F02.logical-modeling",{"title":2392,"path":2393,"stem":2394},"Нормалізація: Гігієна даних та боротьба з аномаліями","\u002Fjava\u002Fpr2\u002Fnormalization","04.java\u002Fpr2\u002F03.normalization",{"title":2396,"path":2397,"stem":2398},"Фізична схема: Від абстракції до DDL","\u002Fjava\u002Fpr2\u002Fphysical-schema","04.java\u002Fpr2\u002F04.physical-schema",{"title":2400,"path":2401,"stem":2402},"Архітектурна класифікація таблиць","\u002Fjava\u002Fpr2\u002Ftable-classification","04.java\u002Fpr2\u002F05.table-classification",{"title":2404,"path":2405,"stem":2406},"Database Migrations: Версіонування схеми з Flyway","\u002Fjava\u002Fpr2\u002Fdatabase-migrations","04.java\u002Fpr2\u002F06.database-migrations",{"title":2408,"path":2409,"stem":2410},"А що, якби це була не реляційна БД?","\u002Fjava\u002Fpr2\u002Fbeyond-relational","04.java\u002Fpr2\u002F07.beyond-relational",{"title":2412,"path":2413,"stem":2414},"Object-Relational Impedance Mismatch: Два світи, що не хочуть дружити","\u002Fjava\u002Fpr2\u002Fimpedance-mismatch","04.java\u002Fpr2\u002F09.impedance-mismatch",{"title":2416,"path":2417,"stem":2418},"JDBC: Перший контакт із базою даних","\u002Fjava\u002Fpr2\u002Fjdbc-fundamentals","04.java\u002Fpr2\u002F10.jdbc-fundamentals",{"title":2420,"path":2421,"stem":2422},"Якість коду: Spotless, SpotBugs та SonarQube","\u002Fjava\u002Fpr2\u002F10a.code-quality","04.java\u002Fpr2\u002F10a.code-quality",{"title":2424,"path":2425,"stem":2426},"Connection Pool: Патерн Object Pool для JDBC-з'єднань","\u002Fjava\u002Fpr2\u002Fconnection-pool","04.java\u002Fpr2\u002F11.connection-pool",{"title":2428,"path":2429,"stem":2430},"Row Data Gateway: Об'єкт як обгортка рядка таблиці","\u002Fjava\u002Fpr2\u002Frow-data-gateway","04.java\u002Fpr2\u002F12.row-data-gateway",{"title":2432,"path":2433,"stem":2434},"Table Data Gateway: Фасад таблиці як архітектурний відступ","\u002Fjava\u002Fpr2\u002Ftable-data-gateway","04.java\u002Fpr2\u002F13.table-data-gateway",{"title":2436,"path":2437,"stem":2438},"Repository + Data Mapper: Правильна шарова архітектура з JDBC","\u002Fjava\u002Fpr2\u002Frepository-data-mapper","04.java\u002Fpr2\u002F14.repository-data-mapper",{"title":2440,"path":2441,"stem":2442},"Identity Map: Кешування сутностей у рамках сесії","\u002Fjava\u002Fpr2\u002Fidentity-map","04.java\u002Fpr2\u002F15.identity-map",{"title":2444,"path":2445,"stem":2446},"Unit of Work: Відстеження змін і координація JDBC-транзакцій","\u002Fjava\u002Fpr2\u002Funit-of-work","04.java\u002Fpr2\u002F16.unit-of-work",{"title":2448,"path":2449,"stem":2450},"Strategy: Замінювані SQL-стратегії для підтримки різних СУБД","\u002Fjava\u002Fpr2\u002Fstrategy-sql","04.java\u002Fpr2\u002F17.strategy-sql",{"title":2452,"path":2453,"stem":2454},"Proxy: Lazy Loading для One-To-Many колекцій","\u002Fjava\u002Fpr2\u002Fproxy-lazy-loading","04.java\u002Fpr2\u002F18.proxy-lazy-loading",{"title":2456,"path":2457,"stem":2458},"Generic Repository через Java Reflection: анотації та динамічний SQL","\u002Fjava\u002Fpr2\u002Fgeneric-repository-reflection","04.java\u002Fpr2\u002F19.generic-repository-reflection",{"title":2460,"path":2461,"stem":2462},"Specification Pattern: Композиція бізнес-правил для складних запитів","\u002Fjava\u002Fpr2\u002Fspecification-pattern","04.java\u002Fpr2\u002F20.specification-pattern",{"title":2464,"path":2465,"stem":2466},"Розширені можливості Specification Pattern: підзапити, агрегації та гібридний підхід","\u002Fjava\u002Fpr2\u002F20a.advanced-specifications","04.java\u002Fpr2\u002F20a.advanced-specifications",{"title":2468,"path":2469,"stem":2470},"Асинхронність у JDBC: Від блокуючих викликів до CompletableFuture","\u002Fjava\u002Fpr2\u002Fasynchronous-jdbc","04.java\u002Fpr2\u002F21.asynchronous-jdbc",{"title":2472,"path":2473,"stem":2474},"Інтеграційне тестування JDBC-репозиторіїв: Embedded H2 та патерн AAA","\u002Fjava\u002Fpr2\u002Fintegration-testing-h2","04.java\u002Fpr2\u002F22.integration-testing-h2",{"title":2476,"path":2477,"stem":2478},"Testcontainers: Тестування з реальною PostgreSQL у Docker-контейнерах","\u002Fjava\u002Fpr2\u002Fintegration-testing-testcontainers","04.java\u002Fpr2\u002F23.integration-testing-testcontainers",{"title":2480,"path":2481,"stem":2482},"Google Guice: Впровадження залежностей у JavaFX-проєкті","\u002Fjava\u002Fpr2\u002Fdependency-injection-guice","04.java\u002Fpr2\u002F24.dependency-injection-guice",{"title":2484,"path":2485,"stem":2486},"JavaFX: Основи побудови графічних інтерфейсів","\u002Fjava\u002Fpr2\u002Fjavafx-fundamentals","04.java\u002Fpr2\u002F25.javafx-fundamentals",{"title":2488,"path":2489,"stem":2490},"Properties та Bindings: Реактивність у JavaFX","\u002Fjava\u002Fpr2\u002Fjavafx-properties-bindings","04.java\u002Fpr2\u002F26.javafx-properties-bindings",{"title":2492,"path":2493,"stem":2494},"MVC vs MVP vs MVVM: Еволюція архітектурних патернів UI","\u002Fjava\u002Fpr2\u002Fui-architecture-patterns","04.java\u002Fpr2\u002F27.ui-architecture-patterns",{"title":2496,"path":2497,"stem":2498},"MVVM на практиці: Побудова ViewModel","\u002Fjava\u002Fpr2\u002Fmvvm-viewmodel-implementation","04.java\u002Fpr2\u002F28.mvvm-viewmodel-implementation",{"title":2500,"path":2501,"stem":2502},"View та Controller: Зв'язування з ViewModel через FXML","\u002Fjava\u002Fpr2\u002Fmvvm-view-controller","04.java\u002Fpr2\u002F29.mvvm-view-controller",{"title":2504,"path":2505,"stem":2506},"Інтеграція MVVM з Guice: Автоматична ін'єкція залежностей","\u002Fjava\u002Fpr2\u002Fmvvm-guice-integration","04.java\u002Fpr2\u002F30.mvvm-guice-integration",{"title":2508,"path":2509,"stem":2510},"Валідація та обробка помилок у MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-validation-error-handling","04.java\u002Fpr2\u002F31.mvvm-validation-error-handling",{"title":2512,"path":2513,"stem":2514},"Навігація та управління екранами у JavaFX MVVM","\u002Fjava\u002Fpr2\u002Fmvvm-navigation-screen-management","04.java\u002Fpr2\u002F32.mvvm-navigation-screen-management",{"title":2516,"path":2517,"stem":2518},"Тестування JavaFX MVVM-додатків","\u002Fjava\u002Fpr2\u002Fmvvm-testing","04.java\u002Fpr2\u002F33.mvvm-testing",{"title":2520,"path":2521,"stem":2522},"Стилізація та теми у JavaFX: CSS та User Experience","\u002Fjava\u002Fpr2\u002Fjavafx-styling-themes","04.java\u002Fpr2\u002F34.javafx-styling-themes",{"title":2524,"path":2525,"stem":2526},"AtlantaFX: Сучасні теми для JavaFX додатків","\u002Fjava\u002Fpr2\u002Fatlantafx-modern-themes","04.java\u002Fpr2\u002F35.atlantafx-modern-themes",{"title":2528,"path":2529,"stem":2530},"Пакування та розповсюдження JavaFX-додатків","\u002Fjava\u002Fpr2\u002Fjar-packaging-distribution","04.java\u002Fpr2\u002F36.jar-packaging-distribution",{"title":2532,"icon":2533,"path":2534,"stem":2535,"children":2536,"page":59},"Python","i-devicon-python","\u002Fpython","05.python",[2537,2541,2544,2548,2552,2556,2560,2564,2568],{"title":2538,"path":2539,"stem":2540},"Модулі, Пакети та Віртуальні Середовища","\u002Fpython\u002Fmodules-packages-venv","05.python\u002F00.modules-packages-venv",{"title":71,"path":2542,"stem":2543},"\u002Fpython\u002Fclasses-objects","05.python\u002F01.classes-objects",{"title":2545,"path":2546,"stem":2547},"Інкапсуляція, Керування Доступом та Властивості","\u002Fpython\u002Fencapsulation","05.python\u002F02.encapsulation",{"title":2549,"path":2550,"stem":2551},"Наслідування, MRO та суперсила super()","\u002Fpython\u002Finheritance-mro","05.python\u002F03.inheritance-mro",{"title":2553,"path":2554,"stem":2555},"Абстракція — ABC проти Статичних Протоколів (PEP 544)","\u002Fpython\u002Fabstraction-protocols","05.python\u002F04.abstraction-protocols",{"title":2557,"path":2558,"stem":2559},"Магічні методи (Dunder) та Емуляція протоколів","\u002Fpython\u002Fdunder-methods","05.python\u002F05.dunder-methods",{"title":2561,"path":2562,"stem":2563},"Декоратори та Керування життєвим циклом методів","\u002Fpython\u002Fdecorators-static-class","05.python\u002F06.decorators-static-class",{"title":2565,"path":2566,"stem":2567},"📦 Повний посібник з модулів, пакетів та віртуальних середовищ у Python","\u002Fpython\u002Flesson_9","05.python\u002Flesson_9",{"title":2569,"path":2570,"stem":2571},"[object Object]","\u002Fpython\u002Foop-plan","05.python\u002Foop-plan",{"title":2573,"icon":2574,"path":2575,"stem":2576,"children":2577,"page":59},"Бази даних","i-lucide-database","\u002Fdatabases","06.databases",[2578,2608,2631,2668,2697,2715,2749,2761,2770],{"title":2579,"icon":2580,"path":2581,"stem":2582,"children":2583,"page":59},"Intro","i-lucide-play","\u002Fdatabases\u002Fintro","06.databases\u002F01.intro",[2584,2588,2592,2596,2600,2604],{"title":2585,"path":2586,"stem":2587},"Введення в теорію баз даних","\u002Fdatabases\u002Fintro\u002Fintroduction-to-databases","06.databases\u002F01.intro\u002F01.introduction-to-databases",{"title":2589,"path":2590,"stem":2591},"Реляційна модель даних","\u002Fdatabases\u002Fintro\u002Frelational-model-theory","06.databases\u002F01.intro\u002F02.relational-model-theory",{"title":2593,"path":2594,"stem":2595},"ER-моделювання","\u002Fdatabases\u002Fintro\u002Fer-modeling","06.databases\u002F01.intro\u002F03.er-modeling",{"title":2597,"path":2598,"stem":2599},"Логічне проектування БД","\u002Fdatabases\u002Fintro\u002Flogical-schema","06.databases\u002F01.intro\u002F04.logical-schema",{"title":2601,"path":2602,"stem":2603},"Класифікація таблиць","\u002Fdatabases\u002Fintro\u002Ftable-classification","06.databases\u002F01.intro\u002F05.table-classification",{"title":2605,"path":2606,"stem":2607},"PlantUML для баз даних","\u002Fdatabases\u002Fintro\u002Fplantuml-diagrams","06.databases\u002F01.intro\u002F06.plantuml-diagrams",{"title":2609,"icon":2574,"path":2610,"stem":2611,"children":2612,"page":59},"MS SQL Server Start","\u002Fdatabases\u002Fms-sql-server-start","06.databases\u002F02.ms-sql-server-start",[2613,2617,2623,2627],{"title":2614,"path":2615,"stem":2616},"Типи даних у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fdata-types","06.databases\u002F02.ms-sql-server-start\u002F01.data-types",{"title":2618,"path":2619,"stem":2620,"children":2621},"Індекси у MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-indexes","06.databases\u002F02.ms-sql-server-start\u002F02.sql-indexes",[2622],{"title":2618,"path":2619,"stem":2620},{"title":2624,"path":2625,"stem":2626},"Системні бази даних MS SQL Server","\u002Fdatabases\u002Fms-sql-server-start\u002Fsystem-databases","06.databases\u002F02.ms-sql-server-start\u002F03.system-databases",{"title":2628,"path":2629,"stem":2630},"Огляд мови SQL та запитів","\u002Fdatabases\u002Fms-sql-server-start\u002Fsql-queries-overview","06.databases\u002F02.ms-sql-server-start\u002F04.sql-queries-overview",{"title":2632,"icon":2574,"path":2633,"stem":2634,"children":2635,"page":59},"SQL","\u002Fdatabases\u002Fsql","06.databases\u002F03.sql",[2636,2640,2644,2648,2652,2656,2660,2664],{"title":2637,"path":2638,"stem":2639},"Налаштування демонстраційної бази даних","\u002Fdatabases\u002Fsql\u002Fsample-database-setup","06.databases\u002F03.sql\u002F00.sample-database-setup",{"title":2641,"path":2642,"stem":2643},"DDL - Створення таблиць (CREATE TABLE)","\u002Fdatabases\u002Fsql\u002Fddl-create-table","06.databases\u002F03.sql\u002F01.ddl-create-table",{"title":2645,"path":2646,"stem":2647},"DDL - Зміна та видалення таблиць (ALTER, DROP)","\u002Fdatabases\u002Fsql\u002Fddl-alter-drop-table","06.databases\u002F03.sql\u002F02.ddl-alter-drop-table",{"title":2649,"path":2650,"stem":2651},"SELECT запити - Основи","\u002Fdatabases\u002Fsql\u002Fselect-queries-fundamentals","06.databases\u002F03.sql\u002F03.select-queries-fundamentals",{"title":2653,"path":2654,"stem":2655},"SELECT запити - Розширені можливості","\u002Fdatabases\u002Fsql\u002Fselect-queries-advanced","06.databases\u002F03.sql\u002F04.select-queries-advanced",{"title":2657,"path":2658,"stem":2659},"INSERT запити - Додавання даних","\u002Fdatabases\u002Fsql\u002Finsert-queries","06.databases\u002F03.sql\u002F05.insert-queries",{"title":2661,"path":2662,"stem":2663},"UPDATE та DELETE запити","\u002Fdatabases\u002Fsql\u002Fupdate-delete-queries","06.databases\u002F03.sql\u002F06.update-delete-queries",{"title":2665,"path":2666,"stem":2667},"Транзакції в SQL","\u002Fdatabases\u002Fsql\u002Ftransactions","06.databases\u002F03.sql\u002F07.transactions",{"title":2669,"icon":2574,"path":2670,"stem":2671,"children":2672,"page":59},"Multi Table Databases","\u002Fdatabases\u002Fmulti-table-databases","06.databases\u002F04.multi-table-databases",[2673,2677,2681,2685,2689,2693],{"title":2674,"path":2675,"stem":2676},"Зв'язки та нормалізація БД","\u002Fdatabases\u002Fmulti-table-databases\u002Frelationships-and-normalization","06.databases\u002F04.multi-table-databases\u002F00.relationships-and-normalization",{"title":2678,"path":2679,"stem":2680},"INNER JOIN - З'єднання таблиць","\u002Fdatabases\u002Fmulti-table-databases\u002Finner-join","06.databases\u002F04.multi-table-databases\u002F01.inner-join",{"title":2682,"path":2683,"stem":2684},"OUTER JOINs - LEFT, RIGHT, FULL","\u002Fdatabases\u002Fmulti-table-databases\u002Fouter-joins","06.databases\u002F04.multi-table-databases\u002F02.outer-joins",{"title":2686,"path":2687,"stem":2688},"CROSS та SELF JOINs","\u002Fdatabases\u002Fmulti-table-databases\u002Fcross-self-joins","06.databases\u002F04.multi-table-databases\u002F03.cross-self-joins",{"title":2690,"path":2691,"stem":2692},"Підзапити (Subqueries)","\u002Fdatabases\u002Fmulti-table-databases\u002Fsubqueries","06.databases\u002F04.multi-table-databases\u002F04.subqueries",{"title":2694,"path":2695,"stem":2696},"Агрегації з JOIN","\u002Fdatabases\u002Fmulti-table-databases\u002Faggregations-with-joins","06.databases\u002F04.multi-table-databases\u002F05.aggregations-with-joins",{"title":2698,"icon":2699,"path":2700,"stem":2701,"children":2702,"page":59},"Aggregate Functions","i-lucide-calculator","\u002Fdatabases\u002Faggregate-functions","06.databases\u002F05.aggregate-functions",[2703,2707,2711],{"title":2704,"path":2705,"stem":2706},"Функції агрегування в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fintroduction-aggregate-functions","06.databases\u002F05.aggregate-functions\u002F01.introduction-aggregate-functions",{"title":2708,"path":2709,"stem":2710},"Групування даних в MS SQL Server","\u002Fdatabases\u002Faggregate-functions\u002Fgrouping-data","06.databases\u002F05.aggregate-functions\u002F02.grouping-data",{"title":2712,"path":2713,"stem":2714},"Підзапити з агрегатними функціями","\u002Fdatabases\u002Faggregate-functions\u002Fsubqueries-aggregates","06.databases\u002F05.aggregate-functions\u002F03.subqueries-aggregates",{"title":2716,"icon":2717,"path":2718,"stem":2719,"children":2720,"page":59},"Тригери та зберігаємі процедури","i-lucide-database-zap","\u002Fdatabases\u002Ftriggers-stored-procedures","06.databases\u002F07.triggers-stored-procedures",[2721,2725,2729,2733,2737,2741,2745],{"title":2722,"path":2723,"stem":2724},"DML-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fdml-triggers","06.databases\u002F07.triggers-stored-procedures\u002F01.dml-triggers",{"title":2726,"path":2727,"stem":2728},"DDL-тригери","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fddl-triggers","06.databases\u002F07.triggers-stored-procedures\u002F02.ddl-triggers",{"title":2730,"path":2731,"stem":2732},"Transact-SQL розширення","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransact-sql-extensions","06.databases\u002F07.triggers-stored-procedures\u002F03.transact-sql-extensions",{"title":2734,"path":2735,"stem":2736},"Транзакції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Ftransactions","06.databases\u002F07.triggers-stored-procedures\u002F04.transactions",{"title":2738,"path":2739,"stem":2740},"Зберігаємі процедури","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fstored-procedures","06.databases\u002F07.triggers-stored-procedures\u002F05.stored-procedures",{"title":2742,"path":2743,"stem":2744},"Користувацькі функції","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fuser-defined-functions","06.databases\u002F07.triggers-stored-procedures\u002F06.user-defined-functions",{"title":2746,"path":2747,"stem":2748},"Безпека баз даних","\u002Fdatabases\u002Ftriggers-stored-procedures\u002Fsecurity","06.databases\u002F07.triggers-stored-procedures\u002F08.security",{"title":2746,"icon":793,"path":2750,"stem":2751,"children":2752,"page":59},"\u002Fdatabases\u002Fsecurity","06.databases\u002F08.security",[2753,2757],{"title":2754,"path":2755,"stem":2756},"Вступ до безпеки баз даних","\u002Fdatabases\u002Fsecurity\u002Fintroduction","06.databases\u002F08.security\u002F01.introduction",{"title":2758,"path":2759,"stem":2760},"Системні представлення та метадані","\u002Fdatabases\u002Fsecurity\u002Fsystem-views","06.databases\u002F08.security\u002F02.system-views",{"title":2762,"icon":2763,"path":2764,"stem":2765,"children":2766,"page":59},"Резервне копіювання та відновлення","i-lucide-database-backup","\u002Fdatabases\u002Fbackup-recovery","06.databases\u002F09.backup-recovery",[2767],{"title":2762,"path":2768,"stem":2769},"\u002Fdatabases\u002Fbackup-recovery\u002Fbackup-restore","06.databases\u002F09.backup-recovery\u002F01.backup-restore",{"title":2771,"icon":2772,"path":2773,"stem":2774,"children":2775,"page":59},"Повнотекстовий пошук","i-lucide-search","\u002Fdatabases\u002Ffull-text-search","06.databases\u002F10.full-text-search",[2776],{"title":2771,"path":2777,"stem":2778},"\u002Fdatabases\u002Ffull-text-search\u002Ffull-text-search","06.databases\u002F10.full-text-search\u002F01.full-text-search",{"title":2780,"icon":2781,"path":2782,"stem":2783,"children":2784,"page":59},"Tools","i-lucide-wrench","\u002Ftools","07.tools",[2785,2861],{"title":2786,"icon":2787,"path":2788,"stem":2789,"children":2790},"Docker","i-simple-icons-docker","\u002Ftools\u002Fdocker","07.tools\u002F01.docker\u002Findex",[2791,2793,2797,2801,2805,2809,2813,2817,2821,2825,2829,2833,2837,2841,2845,2849,2853,2857],{"title":2792,"path":2788,"stem":2789},"Docker: від нуля до production",{"title":2794,"path":2795,"stem":2796},"Контейнеризація — від проблеми до рішення","\u002Ftools\u002Fdocker\u002Fcontainerization-concept","07.tools\u002F01.docker\u002F01.containerization-concept",{"title":2798,"path":2799,"stem":2800},"Docker — що це і навіщо?","\u002Ftools\u002Fdocker\u002Fdocker-what-and-why","07.tools\u002F01.docker\u002F02.docker-what-and-why",{"title":2802,"path":2803,"stem":2804},"Архітектура Docker Engine","\u002Ftools\u002Fdocker\u002Fdocker-architecture","07.tools\u002F01.docker\u002F03.docker-architecture",{"title":2806,"path":2807,"stem":2808},"Встановлення Docker","\u002Ftools\u002Fdocker\u002Finstallation","07.tools\u002F01.docker\u002F04.installation",{"title":2810,"path":2811,"stem":2812},"Перший контейнер — docker run","\u002Ftools\u002Fdocker\u002Ffirst-container","07.tools\u002F01.docker\u002F05.first-container",{"title":2814,"path":2815,"stem":2816},"Життєвий цикл контейнера","\u002Ftools\u002Fdocker\u002Fcontainer-lifecycle","07.tools\u002F01.docker\u002F06.container-lifecycle",{"title":2818,"path":2819,"stem":2820},"Docker Images — фундаментальні концепції","\u002Ftools\u002Fdocker\u002Fdocker-images-fundamentals","07.tools\u002F01.docker\u002F07.docker-images-fundamentals",{"title":2822,"path":2823,"stem":2824},"Dockerfile — основи","\u002Ftools\u002Fdocker\u002Fdockerfile-basics","07.tools\u002F01.docker\u002F08.dockerfile-basics",{"title":2826,"path":2827,"stem":2828},"Dockerfile — просунуті техніки","\u002Ftools\u002Fdocker\u002Fdockerfile-advanced","07.tools\u002F01.docker\u002F09.dockerfile-advanced",{"title":2830,"path":2831,"stem":2832},"Build Context та кешування шарів","\u002Ftools\u002Fdocker\u002Fbuild-context-and-cache","07.tools\u002F01.docker\u002F10.build-context-and-cache",{"title":2834,"path":2835,"stem":2836},"Реєстри Docker-образів","\u002Ftools\u002Fdocker\u002Fimage-registries","07.tools\u002F01.docker\u002F11.image-registries",{"title":2838,"path":2839,"stem":2840},"Контейнеризація .NET додатків","\u002Ftools\u002Fdocker\u002Fdotnet-containerization","07.tools\u002F01.docker\u002F12.dotnet-containerization",{"title":2842,"path":2843,"stem":2844},"Томи та збереження даних","\u002Ftools\u002Fdocker\u002Fvolumes-and-data","07.tools\u002F01.docker\u002F13.volumes-and-data",{"title":2846,"path":2847,"stem":2848},"Основи мережі в Docker","\u002Ftools\u002Fdocker\u002Fnetworking-basics","07.tools\u002F01.docker\u002F14.networking-basics",{"title":2850,"path":2851,"stem":2852},"Змінні оточення та конфігурація","\u002Ftools\u002Fdocker\u002Fenvironment-and-configuration","07.tools\u002F01.docker\u002F15.environment-and-configuration",{"title":2854,"path":2855,"stem":2856},"Docker Compose — оркестрація контейнерів","\u002Ftools\u002Fdocker\u002Fdocker-compose-basics","07.tools\u002F01.docker\u002F16.docker-compose-basics",{"title":2858,"path":2859,"stem":2860},"Docker Compose — Multi-Service застосунки","\u002Ftools\u002Fdocker\u002Fcompose-multi-service","07.tools\u002F01.docker\u002F17.compose-multi-service",{"title":2862,"icon":2863,"path":2864,"stem":2865,"children":2866},"Kubernetes","simple-icons:kubernetes","\u002Ftools\u002Fkubernetes","07.tools\u002F02.kubernetes\u002Findex",[2867,2869,2873,2877,2881,2885,2889,2893,2897],{"title":2868,"path":2864,"stem":2865},"Kubernetes: від розробки до production",{"title":2870,"path":2871,"stem":2872},"Kubernetes — коли Docker Compose більше не вистачає","\u002Ftools\u002Fkubernetes\u002Fwhy-kubernetes","07.tools\u002F02.kubernetes\u002F01.why-kubernetes",{"title":2874,"path":2875,"stem":2876},"Архітектура Kubernetes — анатомія кластера","\u002Ftools\u002Fkubernetes\u002Fkubernetes-architecture","07.tools\u002F02.kubernetes\u002F02.kubernetes-architecture",{"title":2878,"path":2879,"stem":2880},"Локальне середовище — minikube, kind та k3s","\u002Ftools\u002Fkubernetes\u002Flocal-environment","07.tools\u002F02.kubernetes\u002F03.local-environment",{"title":2882,"path":2883,"stem":2884},"Pod — атомарна одиниця Kubernetes","\u002Ftools\u002Fkubernetes\u002Fpods-and-containers","07.tools\u002F02.kubernetes\u002F04.pods-and-containers",{"title":2886,"path":2887,"stem":2888},"Патерни використання Pod","\u002Ftools\u002Fkubernetes\u002Fpod-patterns","07.tools\u002F02.kubernetes\u002F05.pod-patterns",{"title":2890,"path":2891,"stem":2892},"Deployment — декларативне управління Pod","\u002Ftools\u002Fkubernetes\u002Fdeployment-basics","07.tools\u002F02.kubernetes\u002F06.deployment-basics",{"title":2894,"path":2895,"stem":2896},"Rolling Updates та управління життєвим циклом Deployment","\u002Ftools\u002Fkubernetes\u002Fdeployment-rolling-updates","07.tools\u002F02.kubernetes\u002F07.deployment-rolling-updates",{"title":2898,"path":2899,"stem":2900},"Service — мережева абстракція для Pod","\u002Ftools\u002Fkubernetes\u002Fservices-networking","07.tools\u002F02.kubernetes\u002F08.services-networking",{"title":2902,"icon":2903,"path":2904,"stem":2905,"children":2906,"page":59},"Software Engineering","i-lucide-code-2","\u002Fsoftware-engineering","09.software-engineering",[2907,2911,2915,2919,2923,2927,2931,2935,2939,2943,2947],{"title":2908,"path":2909,"stem":2910},"1. Аналіз предметної області. Експертні знання та складність","\u002Fsoftware-engineering\u002Fintro-subdomains","09.software-engineering\u002F01.intro-subdomains",{"title":2912,"path":2913,"stem":2914},"2. Обмежені контексти. Інтеграція обмежених контекстів","\u002Fsoftware-engineering\u002Fintegrating-limited-contexts","09.software-engineering\u002F02.integrating-limited-contexts",{"title":2916,"path":2917,"stem":2918},"3. Реалізація простої бізнес-логіки","\u002Fsoftware-engineering\u002Fsimple","09.software-engineering\u002F03.simple",{"title":2920,"path":2921,"stem":2922},"4. Опрацювання складної бізнес-логіки","\u002Fsoftware-engineering\u002Fcomplex-business-logic","09.software-engineering\u002F04.complex-business-logic",{"title":2924,"path":2925,"stem":2926},"5. Моделювання фактора часу. Подієво-орієнтована архітектура.","\u002Fsoftware-engineering\u002Fmodelling-the-time-factor","09.software-engineering\u002F05.modelling-the-time-factor",{"title":2928,"path":2929,"stem":2930},"6. Архітектурні патерни","\u002Fsoftware-engineering\u002Farchitectural-patterns","09.software-engineering\u002F06.architectural-patterns",{"title":2932,"path":2933,"stem":2934},"Паттерни взаємодії","\u002Fsoftware-engineering\u002Fpatterns-of-interaction","09.software-engineering\u002F07.patterns-of-interaction",{"title":2936,"path":2937,"stem":2938},"Евристика проєктування","\u002Fsoftware-engineering\u002Fdesign-heuristics","09.software-engineering\u002F08.design-heuristics",{"title":2940,"path":2941,"stem":2942},"Еволюція проєктних рішень","\u002Fsoftware-engineering\u002Fevolution-of-design-solutions","09.software-engineering\u002F09.evolution-of-design-solutions",{"title":2944,"path":2945,"stem":2946},"EventStorming","\u002Fsoftware-engineering\u002Feventstorming","09.software-engineering\u002F10.eventstorming",{"title":2948,"path":2949,"stem":2950},"DDD на практиці","\u002Fsoftware-engineering\u002Fddd-in-practice","09.software-engineering\u002F11.ddd-in-practice",{"title":2952,"icon":943,"path":2953,"stem":2954,"children":2955,"page":59},"DDD","\u002Fddd","10.ddd",[2956,2960,2964,2968,2972,2976,2980,2984,2988,2992,2996,3000,3004],{"title":2957,"path":2958,"stem":2959},"Аналіз предметної області","\u002Fddd\u002Fdomain-analysis","10.ddd\u002F01.domain-analysis",{"title":2961,"path":2962,"stem":2963},"Експертні знання про предметну область","\u002Fddd\u002Fdomain-expert-knowledge","10.ddd\u002F02.domain-expert-knowledge",{"title":2965,"path":2966,"stem":2967},"Як осмислити складність предметної області","\u002Fddd\u002Fmanaging-domain-complexity","10.ddd\u002F03.managing-domain-complexity",{"title":2969,"path":2970,"stem":2971},"Інтеграція обмежених контекстів","\u002Fddd\u002Fbounded-context-integration","10.ddd\u002F04.bounded-context-integration",{"title":2973,"path":2974,"stem":2975},"Реалізація простої бізнес-логіки","\u002Fddd\u002Fsimple-business-logic","10.ddd\u002F05.simple-business-logic",{"title":2977,"path":2978,"stem":2979},"Обробка складної бізнес-логіки","\u002Fddd\u002Fcomplex-business-logic","10.ddd\u002F06.complex-business-logic",{"title":2981,"path":2982,"stem":2983},"Моделювання фактора часу","\u002Fddd\u002Ftime-modeling","10.ddd\u002F07.time-modeling",{"title":2985,"path":2986,"stem":2987},"Глава 8. Архітектурні Патерни","\u002Fddd\u002Farchitectural-patterns","10.ddd\u002F08.architectural-patterns",{"title":2989,"path":2990,"stem":2991},"Глава 9. Патерни Взаємодії","\u002Fddd\u002Finteraction-patterns","10.ddd\u002F09.interaction-patterns",{"title":2993,"path":2994,"stem":2995},"Глава 10. Проектні Евристики","\u002Fddd\u002Fdesign-heuristics","10.ddd\u002F10.design-heuristics",{"title":2997,"path":2998,"stem":2999},"Глава 11. Еволюція Проектних Рішень","\u002Fddd\u002Fevolution-of-design-decisions","10.ddd\u002F11.evolution-of-design-decisions",{"title":3001,"path":3002,"stem":3003},"Глава 12. EventStorming","\u002Fddd\u002Fevent-storming","10.ddd\u002F12.event-storming",{"title":3005,"path":3006,"stem":3007},"Глава 13. DDD на Практиці","\u002Fddd\u002Fddd-in-practice","10.ddd\u002F13.ddd-in-practice",{"title":3009,"icon":3010,"path":3011,"stem":3012,"children":3013,"page":59},"Media Streaming","i-lucide-video","\u002Fmedia-streaming","11.media-streaming",[3014,3018,3022,3026,3030,3034,3038],{"title":3015,"path":3016,"stem":3017},"01. Магія Стрімінгу: Що відбувається, коли ви натискаєте \"Play\"","\u002Fmedia-streaming\u002Fintroduction","11.media-streaming\u002F01.introduction",{"title":3019,"path":3020,"stem":3021},"02. Анатомія Медіа: Кодеки, Контейнери та Стиснення","\u002Fmedia-streaming\u002Faudio-video-anatomy","11.media-streaming\u002F02.audio-video-anatomy",{"title":3023,"path":3024,"stem":3025},"03. The Gym: FFmpeg Deep Dive","\u002Fmedia-streaming\u002Fffmpeg-gym","11.media-streaming\u002F03.ffmpeg-gym",{"title":3027,"path":3028,"stem":3029},"04. HLS Protocol: HTTP Live Streaming у Деталях","\u002Fmedia-streaming\u002Fhls-protocol","11.media-streaming\u002F04.hls-protocol",{"title":3031,"path":3032,"stem":3033},"05. DASH Protocol: Відкритий Стандарт","\u002Fmedia-streaming\u002Fdash-protocol","11.media-streaming\u002F05.dash-protocol",{"title":3035,"path":3036,"stem":3037},"06. Масштабування: CDN та Adaptive Bitrate","\u002Fmedia-streaming\u002Fcdn-and-adaptive-bitrate","11.media-streaming\u002F06.cdn-and-adaptive-bitrate",{"title":3039,"path":3040,"stem":3041},"07. Війна із Затримкою (Latency)","\u002Fmedia-streaming\u002Frealtime-latency","11.media-streaming\u002F07.realtime-latency",{"title":3043,"icon":3044,"path":3045,"stem":3046,"children":3047,"page":59},"HTML & CSS","i-devicon-html5","\u002Fhtml-css","12.html-css",[3048,3052,3056,3060,3064,3068,3072,3076,3080,3084,3088,3092,3096,3100,3104,3108,3112,3116,3120,3124,3128,3132,3136,3140,3144,3148,3152,3156,3160,3164],{"title":3049,"path":3050,"stem":3051},"Вступ до HTML. Структура документа","\u002Fhtml-css\u002Fintro-html-structure","12.html-css\u002F01.intro-html-structure",{"title":3053,"path":3054,"stem":3055},"Форматування тексту в HTML","\u002Fhtml-css\u002Fhtml-text-formatting","12.html-css\u002F02.html-text-formatting",{"title":3057,"path":3058,"stem":3059},"Посилання та зображення в HTML","\u002Fhtml-css\u002Fhtml-links-images","12.html-css\u002F03.html-links-images",{"title":3061,"path":3062,"stem":3063},"Списки та таблиці в HTML","\u002Fhtml-css\u002Fhtml-lists-tables","12.html-css\u002F04.html-lists-tables",{"title":3065,"path":3066,"stem":3067},"Форми в HTML","\u002Fhtml-css\u002Fhtml-forms","12.html-css\u002F05.html-forms",{"title":3069,"path":3070,"stem":3071},"Семантичні елементи HTML5","\u002Fhtml-css\u002Fhtml-semantic-elements","12.html-css\u002F06.html-semantic-elements",{"title":3073,"path":3074,"stem":3075},"Мультимедіа та розширені елементи HTML","\u002Fhtml-css\u002Fhtml-multimedia-advanced","12.html-css\u002F07.html-multimedia-advanced",{"title":3077,"path":3078,"stem":3079},"Мікророзмітка та SEO в HTML","\u002Fhtml-css\u002Fhtml-microdata-seo","12.html-css\u002F08.html-microdata-seo",{"title":3081,"path":3082,"stem":3083},"Вступ до CSS. Селектори та специфічність","\u002Fhtml-css\u002Fcss-intro-selectors","12.html-css\u002F09.css-intro-selectors",{"title":3085,"path":3086,"stem":3087},"Блокова модель CSS. Відступи. Box Sizing","\u002Fhtml-css\u002Fcss-box-model","12.html-css\u002F10.css-box-model",{"title":3089,"path":3090,"stem":3091},"Розміри у CSS: повний довідник одиниць і ключових слів","\u002Fhtml-css\u002F10a.css-sizing","12.html-css\u002F10a.css-sizing",{"title":3093,"path":3094,"stem":3095},"Типографіка в CSS. Шрифти та текст","\u002Fhtml-css\u002Fcss-typography","12.html-css\u002F11.css-typography",{"title":3097,"path":3098,"stem":3099},"Кольори та фони в CSS","\u002Fhtml-css\u002Fcss-colors-backgrounds","12.html-css\u002F12.css-colors-backgrounds",{"title":3101,"path":3102,"stem":3103},"Тіні та фільтри в CSS","\u002Fhtml-css\u002F12b.css-shadows-filters","12.html-css\u002F12b.css-shadows-filters",{"title":3105,"path":3106,"stem":3107},"CSS Flexbox: Фундамент гнучких макетів","\u002Fhtml-css\u002Fcss-flexbox-fundamentals","12.html-css\u002F13.css-flexbox-fundamentals",{"title":3109,"path":3110,"stem":3111},"CSS Flexbox: Вирівнювання та Позиціонування","\u002Fhtml-css\u002Fcss-flexbox-alignment-sizing-and-patterns","12.html-css\u002F14.css-flexbox-alignment-sizing-and-patterns",{"title":3113,"path":3114,"stem":3115},"CSS Grid. Двовимірний макет. Частина 1","\u002Fhtml-css\u002Fcss-layout-grid","12.html-css\u002F15.css-layout-grid",{"title":3117,"path":3118,"stem":3119},"CSS Grid. Двовимірний макет. Частина 2","\u002Fhtml-css\u002Fcss-layout-grid-advanced","12.html-css\u002F16.css-layout-grid-advanced",{"title":3121,"path":3122,"stem":3123},"Позиціонування в CSS. Z-index. Stacking Context","\u002Fhtml-css\u002Fcss-positioning","12.html-css\u002F17.css-positioning",{"title":3125,"path":3126,"stem":3127},"CSS Анімації та Переходи","\u002Fhtml-css\u002Fcss-animations-transitions","12.html-css\u002F18.css-animations-transitions",{"title":3129,"path":3130,"stem":3131},"Адаптивний дизайн. Media Queries. Частина 1","\u002Fhtml-css\u002Fcss-responsive-media-queries","12.html-css\u002F19.css-responsive-media-queries",{"title":3133,"path":3134,"stem":3135},"Адаптивний дизайн. Частина 2: clamp(), Container Queries, @layer","\u002Fhtml-css\u002Fcss-responsive-advanced","12.html-css\u002F20.css-responsive-advanced",{"title":3137,"path":3138,"stem":3139},"CSS Custom Properties. Методології. Сучасний CSS","\u002Fhtml-css\u002Fcss-variables-methodologies","12.html-css\u002F21.css-variables-methodologies",{"title":3141,"path":3142,"stem":3143},"Сучасний CSS 2023–2025: Нові можливості","\u002Fhtml-css\u002Fcss-modern-features","12.html-css\u002F22.css-modern-features",{"title":3145,"path":3146,"stem":3147},"CSS Nesting, @layer, @scope та @property: нативний препроцесор","\u002Fhtml-css\u002F22a.css-nesting-modern-syntax","12.html-css\u002F22a.css-nesting-modern-syntax",{"title":3149,"path":3150,"stem":3151},"CSS для форм та інтерактивних станів","\u002Fhtml-css\u002Fcss-forms-interactive-states","12.html-css\u002F23.css-forms-interactive-states",{"title":3153,"path":3154,"stem":3155},"Доступність у CSS (CSS Accessibility)","\u002Fhtml-css\u002Fcss-accessibility","12.html-css\u002F24.css-accessibility",{"title":3157,"path":3158,"stem":3159},"CSS-функції та сучасні sizing primitives","\u002Fhtml-css\u002Fcss-functions-sizing","12.html-css\u002F25.css-functions-sizing",{"title":3161,"path":3162,"stem":3163},"Rendering Pipeline і CSS Performance","\u002Fhtml-css\u002Fcss-rendering-performance","12.html-css\u002F26.css-rendering-performance",{"title":3165,"path":3166,"stem":3167},"CSS Best Practices: типові ситуації та правильні рішення","\u002Fhtml-css\u002Fcss-best-practices","12.html-css\u002F27.css-best-practices",{"title":3169,"path":3170,"stem":3171,"children":3172,"page":59},"AWS","\u002Faws","13.aws",[3173,3177,3181,3185,3189,3193,3197,3201,3205,3209,3213,3217,3221,3225,3229,3233,3237,3241],{"title":3174,"path":3175,"stem":3176},"Реєстрація AWS акаунту та студентські програми","\u002Faws\u002Faccount-registration","13.aws\u002F00.account-registration",{"title":3178,"path":3179,"stem":3180},"Вступ до хмарних обчислень та AWS","\u002Faws\u002Fintroduction-to-cloud","13.aws\u002F01.introduction-to-cloud",{"title":3182,"path":3183,"stem":3184},"AWS IAM — Identity and Access Management","\u002Faws\u002Fiam","13.aws\u002F02.iam",{"title":3186,"path":3187,"stem":3188},"AWS IAM CLI — Довідник команд","\u002Faws\u002F02a.iam-doc","13.aws\u002F02a.iam-doc",{"title":3190,"path":3191,"stem":3192},"Docker та контейнеризація в AWS — ECR, ECS та Fargate","\u002Faws\u002Fdocker-ecs","13.aws\u002F03.docker-ecs",{"title":3194,"path":3195,"stem":3196},"AWS ECR \u002F ECS CLI — Довідник команд","\u002Faws\u002F03a.docker-ecs-doc","13.aws\u002F03a.docker-ecs-doc",{"title":3198,"path":3199,"stem":3200},"Amazon EC2 — Elastic Compute Cloud","\u002Faws\u002Fec2","13.aws\u002F04.ec2",{"title":3202,"path":3203,"stem":3204},"AWS EC2 CLI — Довідник команд","\u002Faws\u002F04a.ec2-doc","13.aws\u002F04a.ec2-doc",{"title":3206,"path":3207,"stem":3208},"Elastic Load Balancing та Auto Scaling","\u002Faws\u002Falb-asg","13.aws\u002F05.alb-asg",{"title":3210,"path":3211,"stem":3212},"Amazon S3 — Simple Storage Service","\u002Faws\u002Fs3","13.aws\u002F06.s3",{"title":3214,"path":3215,"stem":3216},"Amazon CloudFront — Content Delivery Network","\u002Faws\u002Fcloudfront","13.aws\u002F07.cloudfront",{"title":3218,"path":3219,"stem":3220},"Amazon RDS — Relational Database Service","\u002Faws\u002Frds","13.aws\u002F08.rds",{"title":3222,"path":3223,"stem":3224},"Amazon DynamoDB — NoSQL Database","\u002Faws\u002Fdynamodb","13.aws\u002F09.dynamodb",{"title":3226,"path":3227,"stem":3228},"AWS Lambda та Serverless Compute","\u002Faws\u002Flambda","13.aws\u002F10.lambda",{"title":3230,"path":3231,"stem":3232},"Amazon Bedrock - Foundation Models, RAG та Agents","\u002Faws\u002Fbedrock","13.aws\u002F22.bedrock",{"title":3234,"path":3235,"stem":3236},"Amazon Rekognition - Комп'ютерний зір","\u002Faws\u002Frekognition","13.aws\u002F23.rekognition",{"title":3238,"path":3239,"stem":3240},"Amazon Textract - Інтелектуальний аналіз документів","\u002Faws\u002Ftextract","13.aws\u002F24.textract",{"title":3242,"path":3243,"stem":3244},"Amazon Polly, Transcribe, Comprehend та Translate","\u002Faws\u002Faudio-nlp-services","13.aws\u002F25.audio-nlp-services",{"title":3246,"path":3247,"stem":3248,"children":3249,"page":59},"Tailwind","\u002Ftailwind","21.tailwind",[3250,3254,3258,3262,3266,3270,3274,3278,3282,3286,3290,3294],{"title":3251,"path":3252,"stem":3253},"Що таке Tailwind CSS і навіщо він потрібен","\u002Ftailwind\u002Ftailwind-intro-philosophy","21.tailwind\u002F01.tailwind-intro-philosophy",{"title":3255,"path":3256,"stem":3257},"Встановлення та налаштування Tailwind CSS v4","\u002Ftailwind\u002Ftailwind-installation-setup","21.tailwind\u002F02.tailwind-installation-setup",{"title":3259,"path":3260,"stem":3261},"Utility-класи: основи та система Tailwind","\u002Ftailwind\u002Ftailwind-utility-classes-core","21.tailwind\u002F03.tailwind-utility-classes-core",{"title":3263,"path":3264,"stem":3265},"Layout: Flexbox та Grid через Tailwind","\u002Ftailwind\u002Ftailwind-flexbox-grid","21.tailwind\u002F04.tailwind-flexbox-grid",{"title":3267,"path":3268,"stem":3269},"Кастомізація теми через @theme у Tailwind v4","\u002Ftailwind\u002Ftailwind-theme-customization","21.tailwind\u002F05.tailwind-theme-customization",{"title":3271,"path":3272,"stem":3273},"Варіанти: hover, focus, responsive, dark mode та нові v4","\u002Ftailwind\u002Ftailwind-variants-states","21.tailwind\u002F06.tailwind-variants-states",{"title":3275,"path":3276,"stem":3277},"Типографіка та система кольорів у Tailwind v4","\u002Ftailwind\u002Ftailwind-typography-colors","21.tailwind\u002F07.tailwind-typography-colors",{"title":3279,"path":3280,"stem":3281},"Компоненти та повторюваність: @apply, @utility та патерни","\u002Ftailwind\u002Ftailwind-components-patterns","21.tailwind\u002F08.tailwind-components-patterns",{"title":3283,"path":3284,"stem":3285},"Темна тема та система дизайн-токенів у Tailwind v4","\u002Ftailwind\u002Ftailwind-dark-mode-theming","21.tailwind\u002F09.tailwind-dark-mode-theming",{"title":3287,"path":3288,"stem":3289},"Довільні значення та контейнерні запити у Tailwind v4","\u002Ftailwind\u002Ftailwind-arbitrary-container-queries","21.tailwind\u002F10.tailwind-arbitrary-container-queries",{"title":3291,"path":3292,"stem":3293},"Анімації, трансформації та 3D у Tailwind v4","\u002Ftailwind\u002Ftailwind-animations-transforms","21.tailwind\u002F11.tailwind-animations-transforms",{"title":3295,"path":3296,"stem":3297},"Tailwind CLI, PostCSS та інтеграція з фреймворками","\u002Ftailwind\u002Ftailwind-cli-tooling","21.tailwind\u002F12.tailwind-cli-tooling",{"title":3299,"path":3300,"stem":3301},"Тестування компонентів діаграм","\u002Ftest-components","98.test-components",{"id":3303,"title":1684,"body":3304,"description":11939,"extension":11940,"links":11941,"meta":11942,"navigation":3474,"path":1685,"seo":11943,"stem":1686,"__hash__":11944},"docs\u002F01.csharp\u002F13.network-programming\u002F07a.http-listener.md",{"type":3305,"value":3306,"toc":11915},"minimark",[3307,3311,3316,3329,3342,3368,3373,3379,3388,3397,3406,3412,3421,3424,3428,3437,3629,3633,3648,3693,3708,3789,3791,3795,3799,3931,3935,4026,4028,4032,4037,4080,4082,4086,4091,4300,4302,4306,4311,4468,4470,4474,4479,4626,4648,4650,4654,4660,7211,7215,7218,7233,7254,7275,7284,7290,7292,7296,7299,11337,11341,11749,11752,11765,11789,11802,11819,11827,11829,11833,11838,11901,11911],[3308,3309,1684],"h1",{"id":3310},"httplistener-вбудований-http-сервер-net",[3312,3313,3315],"h2",{"id":3314},"що-таке-httplistener-і-навіщо-він-потрібен","Що таке HttpListener і навіщо він потрібен",[3317,3318,3319,3320,3324,3325,3328],"p",{},"У попередньому розділі ми вивчили ",[3321,3322,3323],"code",{},"HttpClient"," — клієнтську сторону HTTP. Але .NET надає й серверну сторону без будь-яких фреймворків: клас ",[3321,3326,3327],{},"System.Net.HttpListener",". Він дозволяє отримувати та обробляти HTTP-запити безпосередньо у будь-якому .NET-застосунку — без ASP.NET Core, без IIS, без Kestrel.",[3317,3330,3331,3334,3335,3338,3339,3341],{},[3321,3332,3333],{},"HttpListener"," є тонкою керованою обгорткою навколо операційної системи: на Windows він використовує ",[3321,3336,3337],{},"HTTP.sys"," — системний HTTP-сервер, вбудований у ядро ОС. На Linux і macOS ",[3321,3340,3333],{}," використовує власну реалізацію на основі сокетів. У всіх випадках результат однаковий: ваш процес може відповідати на HTTP-запити безпосередньо із зазначеного URL-префіксу.",[3343,3344,3345,3349,3350,3352,3353,3356,3357,3360,3361,3364,3365,3367],"note",{},[3346,3347,3348],"strong",{},"Ключова різниця від ASP.NET Core:"," ",[3321,3351,3333],{}," — це ",[3346,3354,3355],{},"примітивний"," HTTP-сервер. Він не надає middleware pipeline, dependency injection, маршрутизації, model binding, Razor Pages чи будь-чого подібного. Це просто «слухач», що дає вам сирий ",[3321,3358,3359],{},"HttpListenerRequest"," і ",[3321,3362,3363],{},"HttpListenerResponse"," — далі ви будуєте всю логіку самостійно. Саме тому ",[3321,3366,3333],{}," цінний як навчальний інструмент і як компонент для специфічних вбудованих сценаріїв.",[3369,3370,3372],"h3",{"id":3371},"сценарії-застосування","Сценарії застосування",[3317,3374,3375,3376,3378],{},"Незважаючи на простоту (або завдяки їй), ",[3321,3377,3333],{}," є правильним інструментом у кількох ситуаціях:",[3317,3380,3381,3384,3385,3387],{},[3346,3382,3383],{},"Тестові та mock-сервери."," У unit і integration тестах часто потрібен реальний HTTP-сервер, що відповідає на конкретні запити. ",[3321,3386,3333],{}," запускається за мілісекунди і не тягне за собою весь стек ASP.NET Core.",[3317,3389,3390,3393,3394,3396],{},[3346,3391,3392],{},"Утиліти командного рядка."," Якщо ваша CLI-утиліта повинна тимчасово відкрити HTTP-ендпоінт (наприклад, для OAuth callback або webhooks), ",[3321,3395,3333],{}," — найлегше рішення.",[3317,3398,3399,3402,3403,3405],{},[3346,3400,3401],{},"Вбудовані системи та IoT."," На пристроях з обмеженими ресурсами (Windows IoT, embedded .NET) повноцінний ASP.NET Core може бути надмірним. ",[3321,3404,3333],{}," дає мінімальний HTTP-сервер із мінімальними залежностями.",[3317,3407,3408,3411],{},[3346,3409,3410],{},"Адміністративні інтерфейси."," Десктопні застосунки іноді відкривають HTTP-ендпоінт для управління або моніторингу без запуску окремого сервера.",[3317,3413,3414,3417,3418,3420],{},[3346,3415,3416],{},"Навчання."," Робота з ",[3321,3419,3333],{}," дозволяє зрозуміти HTTP без «магії» фреймворків: обробка заголовків, відповіді, статус-коди, потокова передача — все вручну.",[3422,3423],"hr",{},[3312,3425,3427],{"id":3426},"архітектура-та-принцип-роботи","Архітектура та принцип роботи",[3317,3429,3430,3432,3433,3436],{},[3321,3431,3333],{}," будується на концепції ",[3346,3434,3435],{},"URL-префіксів"," (URL prefixes). Ви реєструєте один або кілька префіксів, і ОС спрямовує відповідні HTTP-запити до вашого процесу.",[3438,3439,3440],"plant-uml",{},[3441,3442,3447],"pre",{"className":3443,"code":3444,"language":3445,"meta":3446,"style":3446},"language-plantuml shiki shiki-themes light-plus dark-plus dark-plus","@startuml\nskinparam style plain\nskinparam backgroundColor #ffffff\n\nactor \"HTTP Client\\n(browser \u002F curl)\" as client\n\nparticipant \"OS Network Stack\\n(HTTP.sys \u002F Socket)\" as os #fff3e0\n\nparticipant \"HttpListener\\nQueue\" as queue #e3f2fd\n\nparticipant \"Your Application\\nThreadPool\" as app #e8f5e9\n\nclient -> os : HTTP Request\\nGET \u002Fapi\u002Fdata HTTP\u002F1.1\nos -> os : Перевірка префіксів:\\nhttp:\u002F\u002Flocalhost:8080\u002Fapi\u002F\nos -> queue : Поміщає запит\\nу внутрішню чергу\n\napp -> queue : GetContextAsync()\\nабо GetContext()\nqueue --> app : HttpListenerContext\\n(Request + Response)\n\napp -> app : Обробляє запит:\\n• читає заголовки\\n• читає тіло\\n• формує відповідь\n\napp -> client : HTTP Response\\n(через Response.OutputStream)\n\nnote over queue\n  Черга запитів HttpListener\n  є внутрішньою — ОС або\n  бібліотека буферизує\n  вхідні з'єднання.\nend note\n\n@enduml\n","plantuml","",[3321,3448,3449,3457,3463,3469,3476,3482,3487,3493,3498,3504,3509,3515,3520,3526,3532,3538,3543,3549,3555,3560,3566,3571,3577,3582,3588,3594,3600,3606,3612,3618,3623],{"__ignoreMap":3446},[3450,3451,3454],"span",{"class":3452,"line":3453},"line",1,[3450,3455,3456],{},"@startuml\n",[3450,3458,3460],{"class":3452,"line":3459},2,[3450,3461,3462],{},"skinparam style plain\n",[3450,3464,3466],{"class":3452,"line":3465},3,[3450,3467,3468],{},"skinparam backgroundColor #ffffff\n",[3450,3470,3472],{"class":3452,"line":3471},4,[3450,3473,3475],{"emptyLinePlaceholder":3474},true,"\n",[3450,3477,3479],{"class":3452,"line":3478},5,[3450,3480,3481],{},"actor \"HTTP Client\\n(browser \u002F curl)\" as client\n",[3450,3483,3485],{"class":3452,"line":3484},6,[3450,3486,3475],{"emptyLinePlaceholder":3474},[3450,3488,3490],{"class":3452,"line":3489},7,[3450,3491,3492],{},"participant \"OS Network Stack\\n(HTTP.sys \u002F Socket)\" as os #fff3e0\n",[3450,3494,3496],{"class":3452,"line":3495},8,[3450,3497,3475],{"emptyLinePlaceholder":3474},[3450,3499,3501],{"class":3452,"line":3500},9,[3450,3502,3503],{},"participant \"HttpListener\\nQueue\" as queue #e3f2fd\n",[3450,3505,3507],{"class":3452,"line":3506},10,[3450,3508,3475],{"emptyLinePlaceholder":3474},[3450,3510,3512],{"class":3452,"line":3511},11,[3450,3513,3514],{},"participant \"Your Application\\nThreadPool\" as app #e8f5e9\n",[3450,3516,3518],{"class":3452,"line":3517},12,[3450,3519,3475],{"emptyLinePlaceholder":3474},[3450,3521,3523],{"class":3452,"line":3522},13,[3450,3524,3525],{},"client -> os : HTTP Request\\nGET \u002Fapi\u002Fdata HTTP\u002F1.1\n",[3450,3527,3529],{"class":3452,"line":3528},14,[3450,3530,3531],{},"os -> os : Перевірка префіксів:\\nhttp:\u002F\u002Flocalhost:8080\u002Fapi\u002F\n",[3450,3533,3535],{"class":3452,"line":3534},15,[3450,3536,3537],{},"os -> queue : Поміщає запит\\nу внутрішню чергу\n",[3450,3539,3541],{"class":3452,"line":3540},16,[3450,3542,3475],{"emptyLinePlaceholder":3474},[3450,3544,3546],{"class":3452,"line":3545},17,[3450,3547,3548],{},"app -> queue : GetContextAsync()\\nабо GetContext()\n",[3450,3550,3552],{"class":3452,"line":3551},18,[3450,3553,3554],{},"queue --> app : HttpListenerContext\\n(Request + Response)\n",[3450,3556,3558],{"class":3452,"line":3557},19,[3450,3559,3475],{"emptyLinePlaceholder":3474},[3450,3561,3563],{"class":3452,"line":3562},20,[3450,3564,3565],{},"app -> app : Обробляє запит:\\n• читає заголовки\\n• читає тіло\\n• формує відповідь\n",[3450,3567,3569],{"class":3452,"line":3568},21,[3450,3570,3475],{"emptyLinePlaceholder":3474},[3450,3572,3574],{"class":3452,"line":3573},22,[3450,3575,3576],{},"app -> client : HTTP Response\\n(через Response.OutputStream)\n",[3450,3578,3580],{"class":3452,"line":3579},23,[3450,3581,3475],{"emptyLinePlaceholder":3474},[3450,3583,3585],{"class":3452,"line":3584},24,[3450,3586,3587],{},"note over queue\n",[3450,3589,3591],{"class":3452,"line":3590},25,[3450,3592,3593],{},"  Черга запитів HttpListener\n",[3450,3595,3597],{"class":3452,"line":3596},26,[3450,3598,3599],{},"  є внутрішньою — ОС або\n",[3450,3601,3603],{"class":3452,"line":3602},27,[3450,3604,3605],{},"  бібліотека буферизує\n",[3450,3607,3609],{"class":3452,"line":3608},28,[3450,3610,3611],{},"  вхідні з'єднання.\n",[3450,3613,3615],{"class":3452,"line":3614},29,[3450,3616,3617],{},"end note\n",[3450,3619,3621],{"class":3452,"line":3620},30,[3450,3622,3475],{"emptyLinePlaceholder":3474},[3450,3624,3626],{"class":3452,"line":3625},31,[3450,3627,3628],{},"@enduml\n",[3369,3630,3632],{"id":3631},"url-префікси-основна-концепція","URL-префікси: основна концепція",[3317,3634,3635,3636,3639,3640,3643,3644,3647],{},"Префікс — це рядок вигляду ",[3321,3637,3638],{},"scheme:\u002F\u002Fhost:port\u002Fpath\u002F",". Зверніть увагу на ",[3346,3641,3642],{},"обов'язковий"," завершальний слеш ",[3321,3645,3646],{},"\u002F",". Декілька правил префіксів:",[3649,3650,3651,3658,3667,3677,3687],"ul",{},[3652,3653,3654,3657],"li",{},[3321,3655,3656],{},"http:\u002F\u002Flocalhost:8080\u002F"," — усі запити на localhost:8080",[3652,3659,3660,3663,3664],{},[3321,3661,3662],{},"http:\u002F\u002Flocalhost:8080\u002Fapi\u002F"," — лише шляхи, що починаються з ",[3321,3665,3666],{},"\u002Fapi\u002F",[3652,3668,3669,3672,3673,3676],{},[3321,3670,3671],{},"http:\u002F\u002F+:8080\u002F"," — усі інтерфейси на порту 8080 (еквівалент ",[3321,3674,3675],{},"0.0.0.0",")",[3652,3678,3679,3682,3683,3686],{},[3321,3680,3681],{},"http:\u002F\u002F*:8080\u002F"," — аналогічно ",[3321,3684,3685],{},"+",", але менш суворий пріоритет",[3652,3688,3689,3692],{},[3321,3690,3691],{},"https:\u002F\u002F+:443\u002F"," — HTTPS на всіх інтерфейсах (потрібен сертифікат)",[3317,3694,3695,3696,3360,3698,3701,3702,3704,3705,3707],{},"Символи ",[3321,3697,3685],{},[3321,3699,3700],{},"*"," є сильними та слабкими wildcard відповідно. ",[3321,3703,3685],{}," обирається, якщо є і ",[3321,3706,3685],{},", і конкретний хост на тому ж порту.",[3709,3710,3711,3724,3783],"warning",{},[3317,3712,3713,3714,3716,3717,3719,3720,3723],{},"На Windows для прослуховування не-localhost адрес (особливо ",[3321,3715,3685],{}," або ",[3321,3718,3700],{},") без прав адміністратора необхідно попередньо зареєструвати URL-простір командою ",[3321,3721,3722],{},"netsh",":",[3441,3725,3729],{"className":3726,"code":3727,"language":3728,"meta":3446,"style":3446},"language-powershell shiki shiki-themes light-plus dark-plus dark-plus","# Дозволити поточному користувачу прослуховувати http:\u002F\u002F+:8080\u002Fapi\u002F\nnetsh http add urlacl url=http:\u002F\u002F+:8080\u002Fapi\u002F user=DOMAIN\\username\n\n# Перегляд зареєстрованих префіксів\nnetsh http show urlacl\n\n# Видалення\nnetsh http delete urlacl url=http:\u002F\u002F+:8080\u002Fapi\u002F\n","powershell",[3321,3730,3731,3737,3750,3754,3759,3764,3768,3773],{"__ignoreMap":3446},[3450,3732,3733],{"class":3452,"line":3453},[3450,3734,3736],{"class":3735},"spJ8K","# Дозволити поточному користувачу прослуховувати http:\u002F\u002F+:8080\u002Fapi\u002F\n",[3450,3738,3739,3743,3747],{"class":3452,"line":3459},[3450,3740,3742],{"class":3741},"sHH4Y","netsh http add urlacl url=http:\u002F\u002F+:",[3450,3744,3746],{"class":3745},"sJj4R","8080",[3450,3748,3749],{"class":3741},"\u002Fapi\u002F user=DOMAIN\\username\n",[3450,3751,3752],{"class":3452,"line":3465},[3450,3753,3475],{"emptyLinePlaceholder":3474},[3450,3755,3756],{"class":3452,"line":3471},[3450,3757,3758],{"class":3735},"# Перегляд зареєстрованих префіксів\n",[3450,3760,3761],{"class":3452,"line":3478},[3450,3762,3763],{"class":3741},"netsh http show urlacl\n",[3450,3765,3766],{"class":3452,"line":3484},[3450,3767,3475],{"emptyLinePlaceholder":3474},[3450,3769,3770],{"class":3452,"line":3489},[3450,3771,3772],{"class":3735},"# Видалення\n",[3450,3774,3775,3778,3780],{"class":3452,"line":3495},[3450,3776,3777],{"class":3741},"netsh http delete urlacl url=http:\u002F\u002F+:",[3450,3779,3746],{"class":3745},[3450,3781,3782],{"class":3741},"\u002Fapi\u002F\n",[3317,3784,3785,3788],{},[3321,3786,3787],{},"localhost"," — виняток: для нього права адміністратора не потрібні.",[3422,3790],{},[3312,3792,3794],{"id":3793},"api-класу-httplistener","API класу HttpListener",[3369,3796,3798],{"id":3797},"клас-httplistener","Клас HttpListener",[3800,3801,3802,3822,3846,3878,3886,3895,3907],"field-group",{},[3803,3804,3807,3808,3810,3811,3814,3815,3818,3819,3821],"field",{"name":3805,"type":3806},"Prefixes","HttpListenerPrefixCollection","Колекція URL-префіксів, на яких ",[3321,3809,3333],{}," буде приймати запити. Префікси додаються методом ",[3321,3812,3813],{},"Add(string)"," до виклику ",[3321,3816,3817],{},"Start()",". Зміна після ",[3321,3820,3817],{}," можлива, але залежить від реалізації (HTTP.sys підтримує динамічне додавання).",[3803,3823,3826,3827,3830,3831,3833,3834,3837,3838,3841,3842,3845],{"name":3824,"type":3825},"IsListening","bool (readonly)","Повертає ",[3321,3828,3829],{},"true",", якщо ",[3321,3832,3817],{}," було викликано і ",[3321,3835,3836],{},"Stop()"," \u002F ",[3321,3839,3840],{},"Abort()"," ще не виконано. Корисно для перевірки стану перед викликом ",[3321,3843,3844],{},"GetContext()",".",[3803,3847,3850,3851,3854,3855,3858,3859,3858,3862,3858,3865,3858,3868,3871,3872,3874,3875,3845],{"name":3848,"type":3849},"AuthenticationSchemes","AuthenticationSchemes (enum)","Схема аутентифікації на рівні HTTP: ",[3321,3852,3853],{},"None"," (за замовчуванням), ",[3321,3856,3857],{},"Basic",", ",[3321,3860,3861],{},"Digest",[3321,3863,3864],{},"Ntlm",[3321,3866,3867],{},"Negotiate",[3321,3869,3870],{},"Anonymous",". Можна комбінувати прапорці. При ",[3321,3873,3857],{}," клієнт надсилає ",[3321,3876,3877],{},"Authorization: Basic base64(user:password)",[3803,3879,3882,3883,3885],{"name":3880,"type":3881},"AuthenticationSchemeSelectorDelegate","Func\u003CHttpListenerRequest, AuthenticationSchemes>","Динамічний вибір схеми аутентифікації залежно від запиту. Дозволяє вимагати Basic Auth для одних шляхів і ",[3321,3884,3870],{}," для інших.",[3803,3887,3890,3891,3894],{"name":3888,"type":3889},"Realm","string","Рядок realm для Basic та Digest аутентифікації. Відображається браузером у діалозі входу: ",[3321,3892,3893],{},"Authorization Required — realm",". Типово: назва застосунку або домену.",[3803,3896,3899,3900,3902,3903,3906],{"name":3897,"type":3898},"IgnoreWriteExceptions","bool","Якщо ",[3321,3901,3829],{}," — виключення при записі у ",[3321,3904,3905],{},"Response.OutputStream"," (наприклад, клієнт розірвав з'єднання) ігноруються. Корисно при потоковій передачі, щоб один відключений клієнт не руйнував цикл обробки.",[3803,3908,3911,3912,3858,3915,3858,3918,3858,3921,3858,3924,3858,3927,3930],{"name":3909,"type":3910},"TimeoutManager","HttpListenerTimeoutManager","Управління таймаутами з'єднань: ",[3321,3913,3914],{},"DrainEntityBody",[3321,3916,3917],{},"EntityBody",[3321,3919,3920],{},"HeaderWait",[3321,3922,3923],{},"IdleConnection",[3321,3925,3926],{},"MinSendBytesPerSecond",[3321,3928,3929],{},"RequestQueue",". Дозволяє тонко налаштувати поведінку при повільних або зависаючих клієнтах.",[3369,3932,3934],{"id":3933},"методи-httplistener","Методи HttpListener",[3800,3936,3937,3948,3966,3969,3979,4001,4019],{},[3803,3938,3940,3941,3944,3945,3947],{"name":3817,"type":3939},"void","Відкриває прослуховування на зареєстрованих префіксах. На Windows реєструє префікси у HTTP.sys. Кидає ",[3321,3942,3943],{},"HttpListenerException",", якщо порт зайнятий або недостатньо прав. Після ",[3321,3946,3817],{}," префікси можна лише переглядати (зміна залежить від реалізації).",[3803,3949,3950,3951,3837,3953,3956,3957,3960,3961,3963,3964,3845],{"name":3836,"type":3939},"Зупиняє прослуховування нових з'єднань. Вже прийняті запити можна завершити. ",[3321,3952,3844],{},[3321,3954,3955],{},"GetContextAsync()"," повернуть ",[3321,3958,3959],{},"null"," або кинуть виключення. Після ",[3321,3962,3836],{}," можна знову викликати ",[3321,3965,3817],{},[3803,3967,3968],{"name":3840,"type":3939},"Негайно скасовує всі з'єднання та ресурси без очікування завершення поточних запитів. Використовуйте для аварійного відключення.",[3803,3970,3972,3973,3975,3976,3978],{"name":3971,"type":3939},"Close()","Еквівалентний ",[3321,3974,3840],{}," + звільнення ресурсів. Після ",[3321,3977,3971],{}," об'єкт не може бути перезапущений.",[3803,3980,3982,3985,3986,3988,3989,3646,3992,3995,3996,3998,3999,3845],{"name":3844,"type":3981},"HttpListenerContext",[3346,3983,3984],{},"Синхронний"," метод. Блокує поточний потік до отримання наступного запиту. Повертає ",[3321,3987,3981],{}," із парою ",[3321,3990,3991],{},"Request",[3321,3993,3994],{},"Response",". При ",[3321,3997,3836],{}," під час очікування кидає ",[3321,4000,3943],{},[3803,4002,4004,4007,4008,4011,4012,4015,4016,4018],{"name":3955,"type":4003},"Task\u003CHttpListenerContext>",[3346,4005,4006],{},"Асинхронний"," метод. Повертає ",[3321,4009,4010],{},"Task",", що завершується при надходженні запиту. Є правильним підходом для серверів, оскільки не блокує потік. Підтримує ",[3321,4013,4014],{},"CancellationToken"," через ",[3321,4017,4010],{},"-обгортку.",[3803,4020,4023,4024,3845],{"name":4021,"type":4022},"BeginGetContext() \u002F EndGetContext()","IAsyncResult \u002F HttpListenerContext","Застарілий APM (Asynchronous Programming Model) підхід. Не використовуйте у новому коді — є ",[3321,4025,3955],{},[3422,4027],{},[3312,4029,4031],{"id":4030},"api-класу-httplistenercontext","API класу HttpListenerContext",[3317,4033,4034,4036],{},[3321,4035,3981],{}," — це «контейнер» одного HTTP-обміну: запит + відповідь.",[3800,4038,4039,4043,4053],{},[3803,4040,4042],{"name":3991,"type":4041},"HttpListenerRequest (readonly)","Вхідний HTTP-запит від клієнта. Містить URL, метод, заголовки, тіло та інформацію про клієнта.",[3803,4044,4046,4047,3716,4049,4052],{"name":3994,"type":4045},"HttpListenerResponse (readonly)","Відповідь, що буде надіслана клієнту. Ви заповнюєте статус, заголовки та тіло. ",[3321,4048,3971],{},[3321,4050,4051],{},"OutputStream.Close()"," надсилає відповідь клієнту.",[3803,4054,4057,4058,4060,4061,4063,4064,3646,4066,4068,4069,4072,4073,4076,4077,3845],{"name":4055,"type":4056},"User","IPrincipal? (readonly)","Автентифікований користувач. Не ",[3321,4059,3959],{}," лише якщо ",[3321,4062,3848],{}," не ",[3321,4065,3853],{},[3321,4067,3870],{}," і клієнт надав валідні облікові дані. Містить ",[3321,4070,4071],{},"Identity"," з ",[3321,4074,4075],{},"Name"," та ",[3321,4078,4079],{},"AuthenticationType",[3422,4081],{},[3312,4083,4085],{"id":4084},"api-класу-httplistenerrequest","API класу HttpListenerRequest",[3317,4087,4088,4090],{},[3321,4089,3359],{}," надає доступ до всіх аспектів вхідного HTTP-запиту:",[3800,4092,4093,4119,4128,4136,4149,4160,4185,4201,4217,4227,4236,4242,4251,4263,4272,4280,4288],{},[3803,4094,4096,4097,3858,4100,3858,4103,3858,4106,3858,4109,3858,4112,3858,4115,4118],{"name":4095,"type":3889},"HttpMethod","HTTP-метод запиту: ",[3321,4098,4099],{},"\"GET\"",[3321,4101,4102],{},"\"POST\"",[3321,4104,4105],{},"\"PUT\"",[3321,4107,4108],{},"\"DELETE\"",[3321,4110,4111],{},"\"PATCH\"",[3321,4113,4114],{},"\"HEAD\"",[3321,4116,4117],{},"\"OPTIONS\"",". Завжди у верхньому регістрі.",[3803,4120,4123,4124,4127],{"name":4121,"type":4122},"Url","Uri","Повний URL запиту, включно зі схемою, хостом, портом та шляхом: ",[3321,4125,4126],{},"http:\u002F\u002Flocalhost:8080\u002Fapi\u002Fusers?page=1",". Включає query string.",[3803,4129,4131,4132,4135],{"name":4130,"type":3889},"RawUrl","Сирий URL без схеми та хосту: ",[3321,4133,4134],{},"\u002Fapi\u002Fusers?page=1",". Зручний для маршрутизації.",[3803,4137,4140,4141,4144,4145,4148],{"name":4138,"type":4139},"QueryString","NameValueCollection","Розібрані параметри query string. ",[3321,4142,4143],{},"request.QueryString[\"page\"]"," → ",[3321,4146,4147],{},"\"1\"",". При кількох значеннях одного ключа повертає їх через кому.",[3803,4150,4153,4154,3716,4157,3845],{"name":4151,"type":4152},"Headers","WebHeaderCollection","Всі заголовки запиту. Регістронезалежний доступ: ",[3321,4155,4156],{},"request.Headers[\"Content-Type\"]",[3321,4158,4159],{},"request.Headers[HttpRequestHeader.ContentType]",[3803,4161,4164,4165,3837,4168,4171,4172,3837,4175,4178,4179,3716,4182,3845],{"name":4162,"type":4163},"InputStream","Stream","Потік тіла запиту. Для ",[3321,4166,4167],{},"GET",[3321,4169,4170],{},"HEAD"," — порожній. Для ",[3321,4173,4174],{},"POST",[3321,4176,4177],{},"PUT"," — містить тіло. Читайте за допомогою ",[3321,4180,4181],{},"StreamReader",[3321,4183,4184],{},"JsonSerializer.DeserializeAsync()",[3803,4186,4189,4190,4193,4194,4197,4198,4200],{"name":4187,"type":4188},"ContentType","string?","Значення заголовку ",[3321,4191,4192],{},"Content-Type",": ",[3321,4195,4196],{},"\"application\u002Fjson; charset=utf-8\"",". Може бути ",[3321,4199,3959],{}," якщо заголовок відсутній.",[3803,4202,4205,4206,4209,4210,4213,4214,4216],{"name":4203,"type":4204},"ContentLength64","long","Довжина тіла запиту в байтах (заголовок ",[3321,4207,4208],{},"Content-Length","). ",[3321,4211,4212],{},"-1"," якщо ",[3321,4215,4208],{}," не вказано (може бути chunked).",[3803,4218,4220,4222,4223,4226],{"name":4219,"type":3898},"IsAuthenticated",[3321,4221,3829],{}," якщо аутентифікація пройшла успішно (доступний ",[3321,4224,4225],{},"context.User",").",[3803,4228,4231,4232,4235],{"name":4229,"type":4230},"RemoteEndPoint","IPEndPoint","IP-адреса та порт клієнта: ",[3321,4233,4234],{},"192.168.1.100:54321",". Для визначення джерела запиту. При наявності проксі — адреса проксі, а не кінцевого клієнта.",[3803,4237,4239,4241],{"name":4238,"type":3898},"IsLocal",[3321,4240,3829],{}," якщо запит надійшов з того самого хосту (localhost). Зручно для захисту адміністративних ендпоінтів.",[3803,4243,4245,4247,4248,4250],{"name":4244,"type":3898},"IsSecureConnection",[3321,4246,3829],{}," для HTTPS-з'єднань. Корисно, якщо ",[3321,4249,3333],{}," обслуговує обидва протоколи.",[3803,4252,4255,4256,4259,4260,3845],{"name":4253,"type":4254},"Cookies","CookieCollection","Cookies із заголовку ",[3321,4257,4258],{},"Cookie",". Доступ за іменем: ",[3321,4261,4262],{},"request.Cookies[\"session\"]?.Value",[3803,4264,4267,4268,4271],{"name":4265,"type":4266},"AcceptTypes","string[]?","Розібраний заголовок ",[3321,4269,4270],{},"Accept",". Масив типів, які клієнт готовий прийняти.",[3803,4273,4275,4276,4279],{"name":4274,"type":4188},"UserAgent","Заголовок ",[3321,4277,4278],{},"User-Agent"," — ідентифікатор клієнтського застосунку.",[3803,4281,4283,4284,4287],{"name":4282,"type":4266},"UserLanguages","Розібраний ",[3321,4285,4286],{},"Accept-Language",". Мови у порядку пріоритету клієнта.",[3803,4289,4291,4293,4294,4297,4298,3845],{"name":4290,"type":3898},"HasEntityBody",[3321,4292,3829],{}," якщо запит містить тіло (ContentLength > 0 або ",[3321,4295,4296],{},"Transfer-Encoding: chunked","). Перевіряйте перед читанням ",[3321,4299,4162],{},[3422,4301],{},[3312,4303,4305],{"id":4304},"api-класу-httplistenerresponse","API класу HttpListenerResponse",[3317,4307,4308,4310],{},[3321,4309,3363],{}," є вашим «пером» для написання HTTP-відповіді:",[3800,4312,4313,4328,4342,4353,4365,4377,4389,4400,4411,4421,4431,4442,4447,4454,4465],{},[3803,4314,4317,4318,4321,4322,4193,4325,3845],{"name":4315,"type":4316},"StatusCode","int","HTTP статус-код відповіді. За замовчуванням ",[3321,4319,4320],{},"200",". Встановлюйте до початку запису у ",[3321,4323,4324],{},"OutputStream",[3321,4326,4327],{},"response.StatusCode = 404",[3803,4329,4331,4332,3858,4335,4338,4339,4341],{"name":4330,"type":3889},"StatusDescription","Reason phrase для статус-коду: ",[3321,4333,4334],{},"\"OK\"",[3321,4336,4337],{},"\"Not Found\"",". Автоматично встановлюється при зміні ",[3321,4340,4315],{},", але можна перевизначити. У HTTP\u002F2 ігнорується.",[3803,4343,4275,4344,4346,4347,3858,4349,4352],{"name":4187,"type":3889},[3321,4345,4192],{}," відповіді: ",[3321,4348,4196],{},[3321,4350,4351],{},"\"text\u002Fhtml; charset=utf-8\"",". Встановлюйте завжди, щоб клієнт знав, як інтерпретувати тіло.",[3803,4354,4355,4356,4358,4359,4362,4363,3845],{"name":4203,"type":4204},"Довжина тіла відповіді у байтах. Якщо встановлено — відповідь передається з ",[3321,4357,4208],{},". Якщо не встановлено і ",[3321,4360,4361],{},"SendChunked = false"," — буферизується до ",[3321,4364,4051],{},[3803,4366,4369,4370,4373,4374,3845],{"name":4367,"type":4368},"ContentEncoding","Encoding","Кодування тексту для відповіді. Впливає на ",[3321,4371,4372],{},"Content-Type; charset=",". За замовчуванням ",[3321,4375,4376],{},"UTF-8",[3803,4378,4379,4380,4382,4383,3716,4386,3845],{"name":4151,"type":4152},"Заголовки відповіді. Встановлюйте до запису у ",[3321,4381,4324],{},". ",[3321,4384,4385],{},"response.Headers.Add(\"X-Custom-Header\", \"value\")",[3321,4387,4388],{},"response.Headers[HttpResponseHeader.CacheControl] = \"no-store\"",[3803,4390,4391,4392,4395,4396,4399],{"name":4324,"type":4163},"Потік для запису тіла відповіді. Після запису всіх даних закрийте потік: ",[3321,4393,4394],{},"response.OutputStream.Close()"," або через ",[3321,4397,4398],{},"using",". Закриття потоку надсилає відповідь клієнту.",[3803,4401,3899,4403,4405,4406,4408,4409,3845],{"name":4402,"type":3898},"SendChunked",[3321,4404,3829],{}," — використовується ",[3321,4407,4296],{},". Дозволяє надсилати відповідь до того, як відомий її повний розмір. Не можна використовувати разом із ",[3321,4410,4203],{},[3803,4412,4413,4414,4417,4418,3845],{"name":4253,"type":4254},"Cookies для встановлення у відповіді (заголовок ",[3321,4415,4416],{},"Set-Cookie","). Додавайте через ",[3321,4419,4420],{},"response.Cookies.Add(new Cookie(\"name\", \"value\"))",[3803,4422,4424,4425,4428,4429,3845],{"name":4423,"type":3898},"KeepAlive","Чи зберігати TCP-з'єднання для наступного запиту (",[3321,4426,4427],{},"Connection: keep-alive","). За замовчуванням ",[3321,4430,3829],{},[3803,4432,4434,4435,4438,4439,3845],{"name":4433,"type":3889},"RedirectLocation","Встановлює заголовок ",[3321,4436,4437],{},"Location"," для редиректу. Зручніше використовувати ",[3321,4440,4441],{},"response.Redirect(url)",[3803,4443,4444,4445,3845],{"name":3971,"type":3939},"Завершує відповідь і закриває з'єднання (або повертає у пул для keep-alive). Еквівалент ",[3321,4446,4051],{},[3803,4448,4450,4451,3845],{"name":4449,"type":3939},"Close(byte[] responseEntity, bool willBlock)","Надсилає масив байт як тіло відповіді та завершує. Зручно для коротких відповідей без окремого ",[3321,4452,4453],{},"Write",[3803,4455,4457,4458,4076,4461,4464],{"name":4456,"type":3939},"Redirect(string url)","Встановлює ",[3321,4459,4460],{},"StatusCode = 302",[3321,4462,4463],{},"Location: url",". Клієнт автоматично перейде за вказаним URL.",[3803,4466,4467],{"name":3840,"type":3939},"Негайно обриває з'єднання без надсилання відповіді. Використовуйте для відхилення небажаних з'єднань.",[3422,4469],{},[3312,4471,4473],{"id":4472},"схема-lifecycle-одного-http-запиту","Схема lifecycle одного HTTP-запиту",[3317,4475,4476,4477,3723],{},"Розуміння послідовності операцій критично важливо для правильного використання ",[3321,4478,3333],{},[3438,4480,4481],{},[3441,4482,4484],{"className":3443,"code":4483,"language":3445,"meta":3446,"style":3446},"@startuml\nskinparam style plain\nskinparam backgroundColor #ffffff\n\nstart\n\n:HttpListenerContext context =\\nawait listener.GetContextAsync();|\n\n:HttpListenerRequest request = context.Request;\nHttpListenerResponse response = context.Response;|\n\nif (Перевірка методу, шляху, аутентифікації) then (OK)\n  :Читаємо тіло:\\nrequest.InputStream|\n\n  :Бізнес-логіка|\n\n  :Встановлюємо відповідь:\\nresponse.StatusCode\\nresponse.ContentType\\nresponse.Headers|\n\n  :Записуємо тіло:\\nresponse.OutputStream.Write()|\n\n  :response.OutputStream.Close()\\nабо response.Close()|\nelse (Помилка)\n  :response.StatusCode = 400\u002F403\u002F404\u002F500|\n  :response.Close()|\nendif\n\n:Повертаємось до GetContextAsync() —\\nочікуємо наступний запит|\n\nstop\n\n@enduml\n",[3321,4485,4486,4490,4494,4498,4502,4507,4511,4516,4520,4525,4530,4534,4539,4544,4548,4553,4557,4562,4566,4571,4575,4580,4585,4590,4595,4600,4604,4609,4613,4618,4622],{"__ignoreMap":3446},[3450,4487,4488],{"class":3452,"line":3453},[3450,4489,3456],{},[3450,4491,4492],{"class":3452,"line":3459},[3450,4493,3462],{},[3450,4495,4496],{"class":3452,"line":3465},[3450,4497,3468],{},[3450,4499,4500],{"class":3452,"line":3471},[3450,4501,3475],{"emptyLinePlaceholder":3474},[3450,4503,4504],{"class":3452,"line":3478},[3450,4505,4506],{},"start\n",[3450,4508,4509],{"class":3452,"line":3484},[3450,4510,3475],{"emptyLinePlaceholder":3474},[3450,4512,4513],{"class":3452,"line":3489},[3450,4514,4515],{},":HttpListenerContext context =\\nawait listener.GetContextAsync();|\n",[3450,4517,4518],{"class":3452,"line":3495},[3450,4519,3475],{"emptyLinePlaceholder":3474},[3450,4521,4522],{"class":3452,"line":3500},[3450,4523,4524],{},":HttpListenerRequest request = context.Request;\n",[3450,4526,4527],{"class":3452,"line":3506},[3450,4528,4529],{},"HttpListenerResponse response = context.Response;|\n",[3450,4531,4532],{"class":3452,"line":3511},[3450,4533,3475],{"emptyLinePlaceholder":3474},[3450,4535,4536],{"class":3452,"line":3517},[3450,4537,4538],{},"if (Перевірка методу, шляху, аутентифікації) then (OK)\n",[3450,4540,4541],{"class":3452,"line":3522},[3450,4542,4543],{},"  :Читаємо тіло:\\nrequest.InputStream|\n",[3450,4545,4546],{"class":3452,"line":3528},[3450,4547,3475],{"emptyLinePlaceholder":3474},[3450,4549,4550],{"class":3452,"line":3534},[3450,4551,4552],{},"  :Бізнес-логіка|\n",[3450,4554,4555],{"class":3452,"line":3540},[3450,4556,3475],{"emptyLinePlaceholder":3474},[3450,4558,4559],{"class":3452,"line":3545},[3450,4560,4561],{},"  :Встановлюємо відповідь:\\nresponse.StatusCode\\nresponse.ContentType\\nresponse.Headers|\n",[3450,4563,4564],{"class":3452,"line":3551},[3450,4565,3475],{"emptyLinePlaceholder":3474},[3450,4567,4568],{"class":3452,"line":3557},[3450,4569,4570],{},"  :Записуємо тіло:\\nresponse.OutputStream.Write()|\n",[3450,4572,4573],{"class":3452,"line":3562},[3450,4574,3475],{"emptyLinePlaceholder":3474},[3450,4576,4577],{"class":3452,"line":3568},[3450,4578,4579],{},"  :response.OutputStream.Close()\\nабо response.Close()|\n",[3450,4581,4582],{"class":3452,"line":3573},[3450,4583,4584],{},"else (Помилка)\n",[3450,4586,4587],{"class":3452,"line":3579},[3450,4588,4589],{},"  :response.StatusCode = 400\u002F403\u002F404\u002F500|\n",[3450,4591,4592],{"class":3452,"line":3584},[3450,4593,4594],{},"  :response.Close()|\n",[3450,4596,4597],{"class":3452,"line":3590},[3450,4598,4599],{},"endif\n",[3450,4601,4602],{"class":3452,"line":3596},[3450,4603,3475],{"emptyLinePlaceholder":3474},[3450,4605,4606],{"class":3452,"line":3602},[3450,4607,4608],{},":Повертаємось до GetContextAsync() —\\nочікуємо наступний запит|\n",[3450,4610,4611],{"class":3452,"line":3608},[3450,4612,3475],{"emptyLinePlaceholder":3474},[3450,4614,4615],{"class":3452,"line":3614},[3450,4616,4617],{},"stop\n",[3450,4619,4620],{"class":3452,"line":3620},[3450,4621,3475],{"emptyLinePlaceholder":3474},[3450,4623,4624],{"class":3452,"line":3625},[3450,4625,3628],{},[4627,4628,4629,4632,4633,3716,4635,4638,4639,4642,4643,4395,4646,3845],"caution",{},[3346,4630,4631],{},"Критична помилка початківців:"," якщо ви не викликаєте ",[3321,4634,4394],{},[3321,4636,4637],{},"response.Close()"," — клієнт буде нескінченно чекати відповіді, а ресурси з'єднання не звільняться. ",[3346,4640,4641],{},"Завжди"," закривайте відповідь у блоці ",[3321,4644,4645],{},"try\u002Ffinally",[3321,4647,4398],{},[3422,4649],{},[3312,4651,4653],{"id":4652},"приклад-1-статичний-файловий-сервер","Приклад 1: Статичний файловий сервер",[3317,4655,4656,4657,4659],{},"Перший приклад демонструє базову роботу з ",[3321,4658,3333],{},": роздача статичних файлів із директорії. Це повноцінний HTTP-сервер для статичного контенту — HTML, CSS, JS, зображень. Він демонструє ключові патерни: цикл обробки запитів, визначення MIME-типів, обробку помилок та коректне завершення роботи.",[3441,4661,4666],{"className":4662,"code":4663,"language":4664,"meta":4665,"style":3446},"language-csharp shiki shiki-themes light-plus dark-plus dark-plus","using System.Net;\nusing System.Text;\n\n\u002F\u002F ── Налаштування сервера ────────────────────────────────────────────────────\nstring wwwroot = Path.GetFullPath(\".\u002Fwwwroot\");  \u002F\u002F Директорія зі статичними файлами\nstring prefix  = \"http:\u002F\u002Flocalhost:8080\u002F\";\n\nif (!Directory.Exists(wwwroot))\n    Directory.CreateDirectory(wwwroot);\n\n\u002F\u002F Таблиця MIME-типів для розширень файлів\nvar mimeTypes = new Dictionary\u003Cstring, string>(StringComparer.OrdinalIgnoreCase)\n{\n    [\".html\"] = \"text\u002Fhtml; charset=utf-8\",\n    [\".htm\"]  = \"text\u002Fhtml; charset=utf-8\",\n    [\".css\"]  = \"text\u002Fcss; charset=utf-8\",\n    [\".js\"]   = \"application\u002Fjavascript; charset=utf-8\",\n    [\".json\"] = \"application\u002Fjson; charset=utf-8\",\n    [\".png\"]  = \"image\u002Fpng\",\n    [\".jpg\"]  = \"image\u002Fjpeg\",\n    [\".jpeg\"] = \"image\u002Fjpeg\",\n    [\".gif\"]  = \"image\u002Fgif\",\n    [\".svg\"]  = \"image\u002Fsvg+xml\",\n    [\".ico\"]  = \"image\u002Fx-icon\",\n    [\".txt\"]  = \"text\u002Fplain; charset=utf-8\",\n    [\".xml\"]  = \"application\u002Fxml; charset=utf-8\",\n    [\".pdf\"]  = \"application\u002Fpdf\",\n    [\".woff2\"]= \"font\u002Fwoff2\",\n};\n\n\u002F\u002F ── Ініціалізація та запуск HttpListener ────────────────────────────────────\nusing var listener = new HttpListener();\nlistener.Prefixes.Add(prefix);\n\n\u002F\u002F CancellationTokenSource для коректного завершення по Ctrl+C\nusing var cts = new CancellationTokenSource();\n\nConsole.CancelKeyPress += (_, e) =>\n{\n    e.Cancel = true;   \u002F\u002F Запобігаємо негайному завершенню процесу\n    cts.Cancel();      \u002F\u002F Сигналізуємо про необхідність зупинки\n};\n\nlistener.Start();\nConsole.WriteLine($\"📁 Файловий сервер запущено: {prefix}\");\nConsole.WriteLine($\"   Директорія: {wwwroot}\");\nConsole.WriteLine($\"   Натисніть Ctrl+C для зупинки.\");\n\n\u002F\u002F ── Основний цикл обробки запитів ───────────────────────────────────────────\n\u002F\u002F Декілька паралельних обробників — сервер не чекає завершення одного\n\u002F\u002F запиту перед прийняттям наступного (Task.Run для кожного контексту)\ntry\n{\n    while (!cts.Token.IsCancellationRequested)\n    {\n        HttpListenerContext context;\n        try\n        {\n            \u002F\u002F GetContextAsync не приймає CancellationToken безпосередньо —\n            \u002F\u002F використовуємо Task.WhenAny для скасування\n            Task\u003CHttpListenerContext> getContextTask = listener.GetContextAsync();\n            Task cancelTask = Task.Delay(Timeout.Infinite, cts.Token);\n\n            Task completed = await Task.WhenAny(getContextTask, cancelTask);\n\n            if (completed == cancelTask)\n                break; \u002F\u002F Отримали сигнал зупинки\n\n            context = await getContextTask;\n        }\n        catch (HttpListenerException) when (cts.Token.IsCancellationRequested)\n        {\n            break; \u002F\u002F Listener зупинено через Stop()\n        }\n\n        \u002F\u002F Обробляємо запит у окремому завданні — не блокуємо прийом наступного\n        _ = Task.Run(() => HandleRequestAsync(context, wwwroot, mimeTypes));\n    }\n}\nfinally\n{\n    listener.Stop();\n    Console.WriteLine(\"\\n🛑 Сервер зупинено.\");\n}\n\n\u002F\u002F ── Обробник одного запиту ──────────────────────────────────────────────────\nstatic async Task HandleRequestAsync(\n    HttpListenerContext context,\n    string wwwroot,\n    Dictionary\u003Cstring, string> mimeTypes)\n{\n    HttpListenerRequest  req = context.Request;\n    HttpListenerResponse res = context.Response;\n\n    Console.WriteLine($\"[{DateTime.Now:HH:mm:ss}] {req.HttpMethod} {req.RawUrl} \" +\n                      $\"← {req.RemoteEndPoint}\");\n\n    try\n    {\n        \u002F\u002F HEAD — ідентичний GET, але без тіла відповіді\n        if (req.HttpMethod != \"GET\" && req.HttpMethod != \"HEAD\")\n        {\n            await SendErrorAsync(res, 405, \"Method Not Allowed\");\n            res.Headers.Add(\"Allow\", \"GET, HEAD\");\n            return;\n        }\n\n        \u002F\u002F Визначаємо локальний шлях до файлу\n        \u002F\u002F RawUrl: \"\u002Findex.html\" або \"\u002F\" або \"\u002Fcss\u002Fstyle.css\"\n        string urlPath = req.Url!.AbsolutePath;\n\n        \u002F\u002F Захист від path traversal: \"..\" у шляху може вийти за межі wwwroot\n        string localPath = Path.GetFullPath(\n            Path.Combine(wwwroot, urlPath.TrimStart('\u002F')));\n\n        if (!localPath.StartsWith(wwwroot, StringComparison.OrdinalIgnoreCase))\n        {\n            await SendErrorAsync(res, 403, \"Forbidden\");\n            return;\n        }\n\n        \u002F\u002F Якщо шлях вказує на директорію — шукаємо index.html\n        if (Directory.Exists(localPath))\n            localPath = Path.Combine(localPath, \"index.html\");\n\n        if (!File.Exists(localPath))\n        {\n            await SendErrorAsync(res, 404,\n                $\"File not found: {urlPath}\");\n            return;\n        }\n\n        \u002F\u002F Визначаємо MIME-тип за розширенням\n        string ext = Path.GetExtension(localPath);\n        string contentType = mimeTypes.TryGetValue(ext, out string? mime)\n            ? mime\n            : \"application\u002Foctet-stream\";\n\n        \u002F\u002F Відкриваємо файл та налаштовуємо відповідь\n        await using FileStream fileStream = File.OpenRead(localPath);\n\n        res.StatusCode  = 200;\n        res.ContentType = contentType;\n        res.ContentLength64 = fileStream.Length;\n\n        \u002F\u002F Заголовки кешування: дозволяємо кешувати на 1 годину\n        res.Headers[HttpResponseHeader.CacheControl] = \"public, max-age=3600\";\n        res.Headers[\"Last-Modified\"] =\n            File.GetLastWriteTimeUtc(localPath).ToString(\"R\");\n\n        \u002F\u002F HEAD-запит: заголовки без тіла\n        if (req.HttpMethod == \"HEAD\")\n        {\n            res.Close();\n            return;\n        }\n\n        \u002F\u002F Потокова передача файлу — не завантажуємо повністю у пам'ять\n        await fileStream.CopyToAsync(res.OutputStream);\n    }\n    catch (Exception ex)\n    {\n        Console.Error.WriteLine($\"Помилка обробки {req.RawUrl}: {ex.Message}\");\n        try { await SendErrorAsync(res, 500, \"Internal Server Error\"); }\n        catch { \u002F* Відповідь вже могла бути частково надіслана *\u002F }\n    }\n    finally\n    {\n        \u002F\u002F ОБОВ'ЯЗКОВО — завжди закривати відповідь!\n        try { res.OutputStream.Close(); } catch { }\n    }\n}\n\n\u002F\u002F ── Допоміжний метод для надсилання помилок ────────────────────────────────\nstatic async Task SendErrorAsync(\n    HttpListenerResponse res, int statusCode, string message)\n{\n    byte[] body = Encoding.UTF8.GetBytes($\"\"\"\n        \u003C!DOCTYPE html>\n        \u003Chtml lang=\"uk\">\u003Chead>\u003Cmeta charset=\"UTF-8\">\n        \u003Ctitle>{statusCode}\u003C\u002Ftitle>\u003C\u002Fhead>\n        \u003Cbody>\n        \u003Ch1>{statusCode}\u003C\u002Fh1>\n        \u003Cp>{System.Net.WebUtility.HtmlEncode(message)}\u003C\u002Fp>\n        \u003C\u002Fbody>\u003C\u002Fhtml>\n        \"\"\");\n\n    res.StatusCode   = statusCode;\n    res.ContentType  = \"text\u002Fhtml; charset=utf-8\";\n    res.ContentLength64 = body.Length;\n\n    await res.OutputStream.WriteAsync(body);\n    res.OutputStream.Close();\n}\n","csharp","showLineNumbers",[3321,4667,4668,4685,4698,4702,4707,4741,4756,4760,4784,4801,4805,4810,4849,4854,4870,4884,4898,4913,4926,4940,4954,4967,4981,4995,5009,5023,5037,5051,5066,5071,5075,5080,5101,5123,5128,5134,5153,5158,5183,5188,5209,5225,5230,5235,5247,5276,5300,5316,5321,5327,5333,5339,5345,5350,5373,5379,5390,5396,5402,5408,5414,5441,5478,5483,5515,5520,5539,5551,5556,5571,5577,5607,5612,5623,5628,5633,5639,5677,5683,5689,5695,5700,5713,5736,5741,5746,5752,5769,5779,5789,5809,5814,5833,5852,5857,5929,5949,5954,5960,5965,5971,6005,6010,6036,6062,6070,6075,6080,6086,6092,6117,6122,6128,6146,6179,6184,6214,6219,6242,6249,6254,6259,6265,6284,6309,6314,6334,6339,6357,6373,6380,6385,6390,6396,6419,6457,6466,6477,6482,6488,6518,6523,6539,6555,6576,6581,6587,6614,6631,6658,6663,6669,6688,6693,6705,6712,6717,6722,6728,6750,6755,6771,6776,6824,6854,6867,6872,6878,6883,6889,6915,6920,6925,6930,6936,6949,6972,6977,7008,7014,7020,7036,7042,7057,7093,7099,7107,7112,7129,7144,7163,7168,7191,7206],{"__ignoreMap":3446},[3450,4669,4670,4673,4677,4679,4682],{"class":3452,"line":3453},[3450,4671,4398],{"class":4672},"s8xlr",[3450,4674,4676],{"class":4675},"sN1BT"," System",[3450,4678,3845],{"class":3741},[3450,4680,4681],{"class":4675},"Net",[3450,4683,4684],{"class":3741},";\n",[3450,4686,4687,4689,4691,4693,4696],{"class":3452,"line":3459},[3450,4688,4398],{"class":4672},[3450,4690,4676],{"class":4675},[3450,4692,3845],{"class":3741},[3450,4694,4695],{"class":4675},"Text",[3450,4697,4684],{"class":3741},[3450,4699,4700],{"class":3452,"line":3465},[3450,4701,3475],{"emptyLinePlaceholder":3474},[3450,4703,4704],{"class":3452,"line":3471},[3450,4705,4706],{"class":3735},"\u002F\u002F ── Налаштування сервера ────────────────────────────────────────────────────\n",[3450,4708,4709,4712,4716,4719,4722,4724,4728,4731,4735,4738],{"class":3452,"line":3478},[3450,4710,3889],{"class":4711},"su1O8",[3450,4713,4715],{"class":4714},"siwwj"," wwwroot",[3450,4717,4718],{"class":3741}," = ",[3450,4720,4721],{"class":4714},"Path",[3450,4723,3845],{"class":3741},[3450,4725,4727],{"class":4726},"s8Opu","GetFullPath",[3450,4729,4730],{"class":3741},"(",[3450,4732,4734],{"class":4733},"sbdoH","\".\u002Fwwwroot\"",[3450,4736,4737],{"class":3741},");  ",[3450,4739,4740],{"class":3735},"\u002F\u002F Директорія зі статичними файлами\n",[3450,4742,4743,4745,4748,4751,4754],{"class":3452,"line":3484},[3450,4744,3889],{"class":4711},[3450,4746,4747],{"class":4714}," prefix",[3450,4749,4750],{"class":3741},"  = ",[3450,4752,4753],{"class":4733},"\"http:\u002F\u002Flocalhost:8080\u002F\"",[3450,4755,4684],{"class":3741},[3450,4757,4758],{"class":3452,"line":3489},[3450,4759,3475],{"emptyLinePlaceholder":3474},[3450,4761,4762,4765,4768,4771,4773,4776,4778,4781],{"class":3452,"line":3495},[3450,4763,4764],{"class":4672},"if",[3450,4766,4767],{"class":3741}," (!",[3450,4769,4770],{"class":4714},"Directory",[3450,4772,3845],{"class":3741},[3450,4774,4775],{"class":4726},"Exists",[3450,4777,4730],{"class":3741},[3450,4779,4780],{"class":4714},"wwwroot",[3450,4782,4783],{"class":3741},"))\n",[3450,4785,4786,4789,4791,4794,4796,4798],{"class":3452,"line":3500},[3450,4787,4788],{"class":4714},"    Directory",[3450,4790,3845],{"class":3741},[3450,4792,4793],{"class":4726},"CreateDirectory",[3450,4795,4730],{"class":3741},[3450,4797,4780],{"class":4714},[3450,4799,4800],{"class":3741},");\n",[3450,4802,4803],{"class":3452,"line":3506},[3450,4804,3475],{"emptyLinePlaceholder":3474},[3450,4806,4807],{"class":3452,"line":3511},[3450,4808,4809],{"class":3735},"\u002F\u002F Таблиця MIME-типів для розширень файлів\n",[3450,4811,4812,4815,4818,4820,4823,4826,4829,4831,4833,4835,4838,4841,4843,4846],{"class":3452,"line":3517},[3450,4813,4814],{"class":4711},"var",[3450,4816,4817],{"class":4714}," mimeTypes",[3450,4819,4718],{"class":3741},[3450,4821,4822],{"class":4711},"new",[3450,4824,4825],{"class":4675}," Dictionary",[3450,4827,4828],{"class":3741},"\u003C",[3450,4830,3889],{"class":4711},[3450,4832,3858],{"class":3741},[3450,4834,3889],{"class":4711},[3450,4836,4837],{"class":3741},">(",[3450,4839,4840],{"class":4714},"StringComparer",[3450,4842,3845],{"class":3741},[3450,4844,4845],{"class":4714},"OrdinalIgnoreCase",[3450,4847,4848],{"class":3741},")\n",[3450,4850,4851],{"class":3452,"line":3522},[3450,4852,4853],{"class":3741},"{\n",[3450,4855,4856,4859,4862,4865,4867],{"class":3452,"line":3528},[3450,4857,4858],{"class":3741},"    [",[3450,4860,4861],{"class":4733},"\".html\"",[3450,4863,4864],{"class":3741},"] = ",[3450,4866,4351],{"class":4733},[3450,4868,4869],{"class":3741},",\n",[3450,4871,4872,4874,4877,4880,4882],{"class":3452,"line":3534},[3450,4873,4858],{"class":3741},[3450,4875,4876],{"class":4733},"\".htm\"",[3450,4878,4879],{"class":3741},"]  = ",[3450,4881,4351],{"class":4733},[3450,4883,4869],{"class":3741},[3450,4885,4886,4888,4891,4893,4896],{"class":3452,"line":3540},[3450,4887,4858],{"class":3741},[3450,4889,4890],{"class":4733},"\".css\"",[3450,4892,4879],{"class":3741},[3450,4894,4895],{"class":4733},"\"text\u002Fcss; charset=utf-8\"",[3450,4897,4869],{"class":3741},[3450,4899,4900,4902,4905,4908,4911],{"class":3452,"line":3545},[3450,4901,4858],{"class":3741},[3450,4903,4904],{"class":4733},"\".js\"",[3450,4906,4907],{"class":3741},"]   = ",[3450,4909,4910],{"class":4733},"\"application\u002Fjavascript; charset=utf-8\"",[3450,4912,4869],{"class":3741},[3450,4914,4915,4917,4920,4922,4924],{"class":3452,"line":3551},[3450,4916,4858],{"class":3741},[3450,4918,4919],{"class":4733},"\".json\"",[3450,4921,4864],{"class":3741},[3450,4923,4196],{"class":4733},[3450,4925,4869],{"class":3741},[3450,4927,4928,4930,4933,4935,4938],{"class":3452,"line":3557},[3450,4929,4858],{"class":3741},[3450,4931,4932],{"class":4733},"\".png\"",[3450,4934,4879],{"class":3741},[3450,4936,4937],{"class":4733},"\"image\u002Fpng\"",[3450,4939,4869],{"class":3741},[3450,4941,4942,4944,4947,4949,4952],{"class":3452,"line":3562},[3450,4943,4858],{"class":3741},[3450,4945,4946],{"class":4733},"\".jpg\"",[3450,4948,4879],{"class":3741},[3450,4950,4951],{"class":4733},"\"image\u002Fjpeg\"",[3450,4953,4869],{"class":3741},[3450,4955,4956,4958,4961,4963,4965],{"class":3452,"line":3568},[3450,4957,4858],{"class":3741},[3450,4959,4960],{"class":4733},"\".jpeg\"",[3450,4962,4864],{"class":3741},[3450,4964,4951],{"class":4733},[3450,4966,4869],{"class":3741},[3450,4968,4969,4971,4974,4976,4979],{"class":3452,"line":3573},[3450,4970,4858],{"class":3741},[3450,4972,4973],{"class":4733},"\".gif\"",[3450,4975,4879],{"class":3741},[3450,4977,4978],{"class":4733},"\"image\u002Fgif\"",[3450,4980,4869],{"class":3741},[3450,4982,4983,4985,4988,4990,4993],{"class":3452,"line":3579},[3450,4984,4858],{"class":3741},[3450,4986,4987],{"class":4733},"\".svg\"",[3450,4989,4879],{"class":3741},[3450,4991,4992],{"class":4733},"\"image\u002Fsvg+xml\"",[3450,4994,4869],{"class":3741},[3450,4996,4997,4999,5002,5004,5007],{"class":3452,"line":3584},[3450,4998,4858],{"class":3741},[3450,5000,5001],{"class":4733},"\".ico\"",[3450,5003,4879],{"class":3741},[3450,5005,5006],{"class":4733},"\"image\u002Fx-icon\"",[3450,5008,4869],{"class":3741},[3450,5010,5011,5013,5016,5018,5021],{"class":3452,"line":3590},[3450,5012,4858],{"class":3741},[3450,5014,5015],{"class":4733},"\".txt\"",[3450,5017,4879],{"class":3741},[3450,5019,5020],{"class":4733},"\"text\u002Fplain; charset=utf-8\"",[3450,5022,4869],{"class":3741},[3450,5024,5025,5027,5030,5032,5035],{"class":3452,"line":3596},[3450,5026,4858],{"class":3741},[3450,5028,5029],{"class":4733},"\".xml\"",[3450,5031,4879],{"class":3741},[3450,5033,5034],{"class":4733},"\"application\u002Fxml; charset=utf-8\"",[3450,5036,4869],{"class":3741},[3450,5038,5039,5041,5044,5046,5049],{"class":3452,"line":3602},[3450,5040,4858],{"class":3741},[3450,5042,5043],{"class":4733},"\".pdf\"",[3450,5045,4879],{"class":3741},[3450,5047,5048],{"class":4733},"\"application\u002Fpdf\"",[3450,5050,4869],{"class":3741},[3450,5052,5053,5055,5058,5061,5064],{"class":3452,"line":3608},[3450,5054,4858],{"class":3741},[3450,5056,5057],{"class":4733},"\".woff2\"",[3450,5059,5060],{"class":3741},"]= ",[3450,5062,5063],{"class":4733},"\"font\u002Fwoff2\"",[3450,5065,4869],{"class":3741},[3450,5067,5068],{"class":3452,"line":3614},[3450,5069,5070],{"class":3741},"};\n",[3450,5072,5073],{"class":3452,"line":3620},[3450,5074,3475],{"emptyLinePlaceholder":3474},[3450,5076,5077],{"class":3452,"line":3625},[3450,5078,5079],{"class":3735},"\u002F\u002F ── Ініціалізація та запуск HttpListener ────────────────────────────────────\n",[3450,5081,5083,5085,5088,5091,5093,5095,5098],{"class":3452,"line":5082},32,[3450,5084,4398],{"class":4672},[3450,5086,5087],{"class":4711}," var",[3450,5089,5090],{"class":4714}," listener",[3450,5092,4718],{"class":3741},[3450,5094,4822],{"class":4711},[3450,5096,5097],{"class":4675}," HttpListener",[3450,5099,5100],{"class":3741},"();\n",[3450,5102,5104,5107,5109,5111,5113,5116,5118,5121],{"class":3452,"line":5103},33,[3450,5105,5106],{"class":4714},"listener",[3450,5108,3845],{"class":3741},[3450,5110,3805],{"class":4714},[3450,5112,3845],{"class":3741},[3450,5114,5115],{"class":4726},"Add",[3450,5117,4730],{"class":3741},[3450,5119,5120],{"class":4714},"prefix",[3450,5122,4800],{"class":3741},[3450,5124,5126],{"class":3452,"line":5125},34,[3450,5127,3475],{"emptyLinePlaceholder":3474},[3450,5129,5131],{"class":3452,"line":5130},35,[3450,5132,5133],{"class":3735},"\u002F\u002F CancellationTokenSource для коректного завершення по Ctrl+C\n",[3450,5135,5137,5139,5141,5144,5146,5148,5151],{"class":3452,"line":5136},36,[3450,5138,4398],{"class":4672},[3450,5140,5087],{"class":4711},[3450,5142,5143],{"class":4714}," cts",[3450,5145,4718],{"class":3741},[3450,5147,4822],{"class":4711},[3450,5149,5150],{"class":4675}," CancellationTokenSource",[3450,5152,5100],{"class":3741},[3450,5154,5156],{"class":3452,"line":5155},37,[3450,5157,3475],{"emptyLinePlaceholder":3474},[3450,5159,5161,5164,5166,5169,5172,5175,5177,5180],{"class":3452,"line":5160},38,[3450,5162,5163],{"class":4714},"Console",[3450,5165,3845],{"class":3741},[3450,5167,5168],{"class":4714},"CancelKeyPress",[3450,5170,5171],{"class":3741}," += (",[3450,5173,5174],{"class":4714},"_",[3450,5176,3858],{"class":3741},[3450,5178,5179],{"class":4714},"e",[3450,5181,5182],{"class":3741},") =>\n",[3450,5184,5186],{"class":3452,"line":5185},39,[3450,5187,4853],{"class":3741},[3450,5189,5191,5194,5196,5199,5201,5203,5206],{"class":3452,"line":5190},40,[3450,5192,5193],{"class":4714},"    e",[3450,5195,3845],{"class":3741},[3450,5197,5198],{"class":4714},"Cancel",[3450,5200,4718],{"class":3741},[3450,5202,3829],{"class":4711},[3450,5204,5205],{"class":3741},";   ",[3450,5207,5208],{"class":3735},"\u002F\u002F Запобігаємо негайному завершенню процесу\n",[3450,5210,5212,5215,5217,5219,5222],{"class":3452,"line":5211},41,[3450,5213,5214],{"class":4714},"    cts",[3450,5216,3845],{"class":3741},[3450,5218,5198],{"class":4726},[3450,5220,5221],{"class":3741},"();      ",[3450,5223,5224],{"class":3735},"\u002F\u002F Сигналізуємо про необхідність зупинки\n",[3450,5226,5228],{"class":3452,"line":5227},42,[3450,5229,5070],{"class":3741},[3450,5231,5233],{"class":3452,"line":5232},43,[3450,5234,3475],{"emptyLinePlaceholder":3474},[3450,5236,5238,5240,5242,5245],{"class":3452,"line":5237},44,[3450,5239,5106],{"class":4714},[3450,5241,3845],{"class":3741},[3450,5243,5244],{"class":4726},"Start",[3450,5246,5100],{"class":3741},[3450,5248,5250,5252,5254,5257,5259,5262,5266,5268,5271,5274],{"class":3452,"line":5249},45,[3450,5251,5163],{"class":4714},[3450,5253,3845],{"class":3741},[3450,5255,5256],{"class":4726},"WriteLine",[3450,5258,4730],{"class":3741},[3450,5260,5261],{"class":4733},"$\"📁 Файловий сервер запущено: ",[3450,5263,5265],{"class":5264},"sD7JJ","{",[3450,5267,5120],{"class":4714},[3450,5269,5270],{"class":5264},"}",[3450,5272,5273],{"class":4733},"\"",[3450,5275,4800],{"class":3741},[3450,5277,5279,5281,5283,5285,5287,5290,5292,5294,5296,5298],{"class":3452,"line":5278},46,[3450,5280,5163],{"class":4714},[3450,5282,3845],{"class":3741},[3450,5284,5256],{"class":4726},[3450,5286,4730],{"class":3741},[3450,5288,5289],{"class":4733},"$\"   Директорія: ",[3450,5291,5265],{"class":5264},[3450,5293,4780],{"class":4714},[3450,5295,5270],{"class":5264},[3450,5297,5273],{"class":4733},[3450,5299,4800],{"class":3741},[3450,5301,5303,5305,5307,5309,5311,5314],{"class":3452,"line":5302},47,[3450,5304,5163],{"class":4714},[3450,5306,3845],{"class":3741},[3450,5308,5256],{"class":4726},[3450,5310,4730],{"class":3741},[3450,5312,5313],{"class":4733},"$\"   Натисніть Ctrl+C для зупинки.\"",[3450,5315,4800],{"class":3741},[3450,5317,5319],{"class":3452,"line":5318},48,[3450,5320,3475],{"emptyLinePlaceholder":3474},[3450,5322,5324],{"class":3452,"line":5323},49,[3450,5325,5326],{"class":3735},"\u002F\u002F ── Основний цикл обробки запитів ───────────────────────────────────────────\n",[3450,5328,5330],{"class":3452,"line":5329},50,[3450,5331,5332],{"class":3735},"\u002F\u002F Декілька паралельних обробників — сервер не чекає завершення одного\n",[3450,5334,5336],{"class":3452,"line":5335},51,[3450,5337,5338],{"class":3735},"\u002F\u002F запиту перед прийняттям наступного (Task.Run для кожного контексту)\n",[3450,5340,5342],{"class":3452,"line":5341},52,[3450,5343,5344],{"class":4672},"try\n",[3450,5346,5348],{"class":3452,"line":5347},53,[3450,5349,4853],{"class":3741},[3450,5351,5353,5356,5358,5361,5363,5366,5368,5371],{"class":3452,"line":5352},54,[3450,5354,5355],{"class":4672},"    while",[3450,5357,4767],{"class":3741},[3450,5359,5360],{"class":4714},"cts",[3450,5362,3845],{"class":3741},[3450,5364,5365],{"class":4714},"Token",[3450,5367,3845],{"class":3741},[3450,5369,5370],{"class":4714},"IsCancellationRequested",[3450,5372,4848],{"class":3741},[3450,5374,5376],{"class":3452,"line":5375},55,[3450,5377,5378],{"class":3741},"    {\n",[3450,5380,5382,5385,5388],{"class":3452,"line":5381},56,[3450,5383,5384],{"class":4675},"        HttpListenerContext",[3450,5386,5387],{"class":4714}," context",[3450,5389,4684],{"class":3741},[3450,5391,5393],{"class":3452,"line":5392},57,[3450,5394,5395],{"class":4672},"        try\n",[3450,5397,5399],{"class":3452,"line":5398},58,[3450,5400,5401],{"class":3741},"        {\n",[3450,5403,5405],{"class":3452,"line":5404},59,[3450,5406,5407],{"class":3735},"            \u002F\u002F GetContextAsync не приймає CancellationToken безпосередньо —\n",[3450,5409,5411],{"class":3452,"line":5410},60,[3450,5412,5413],{"class":3735},"            \u002F\u002F використовуємо Task.WhenAny для скасування\n",[3450,5415,5417,5420,5422,5424,5427,5430,5432,5434,5436,5439],{"class":3452,"line":5416},61,[3450,5418,5419],{"class":4675},"            Task",[3450,5421,4828],{"class":3741},[3450,5423,3981],{"class":4675},[3450,5425,5426],{"class":3741},"> ",[3450,5428,5429],{"class":4714},"getContextTask",[3450,5431,4718],{"class":3741},[3450,5433,5106],{"class":4714},[3450,5435,3845],{"class":3741},[3450,5437,5438],{"class":4726},"GetContextAsync",[3450,5440,5100],{"class":3741},[3450,5442,5444,5446,5449,5451,5453,5455,5458,5460,5463,5465,5468,5470,5472,5474,5476],{"class":3452,"line":5443},62,[3450,5445,5419],{"class":4675},[3450,5447,5448],{"class":4714}," cancelTask",[3450,5450,4718],{"class":3741},[3450,5452,4010],{"class":4714},[3450,5454,3845],{"class":3741},[3450,5456,5457],{"class":4726},"Delay",[3450,5459,4730],{"class":3741},[3450,5461,5462],{"class":4714},"Timeout",[3450,5464,3845],{"class":3741},[3450,5466,5467],{"class":4714},"Infinite",[3450,5469,3858],{"class":3741},[3450,5471,5360],{"class":4714},[3450,5473,3845],{"class":3741},[3450,5475,5365],{"class":4714},[3450,5477,4800],{"class":3741},[3450,5479,5481],{"class":3452,"line":5480},63,[3450,5482,3475],{"emptyLinePlaceholder":3474},[3450,5484,5486,5488,5491,5493,5496,5499,5501,5504,5506,5508,5510,5513],{"class":3452,"line":5485},64,[3450,5487,5419],{"class":4675},[3450,5489,5490],{"class":4714}," completed",[3450,5492,4718],{"class":3741},[3450,5494,5495],{"class":4711},"await",[3450,5497,5498],{"class":4714}," Task",[3450,5500,3845],{"class":3741},[3450,5502,5503],{"class":4726},"WhenAny",[3450,5505,4730],{"class":3741},[3450,5507,5429],{"class":4714},[3450,5509,3858],{"class":3741},[3450,5511,5512],{"class":4714},"cancelTask",[3450,5514,4800],{"class":3741},[3450,5516,5518],{"class":3452,"line":5517},65,[3450,5519,3475],{"emptyLinePlaceholder":3474},[3450,5521,5523,5526,5529,5532,5535,5537],{"class":3452,"line":5522},66,[3450,5524,5525],{"class":4672},"            if",[3450,5527,5528],{"class":3741}," (",[3450,5530,5531],{"class":4714},"completed",[3450,5533,5534],{"class":3741}," == ",[3450,5536,5512],{"class":4714},[3450,5538,4848],{"class":3741},[3450,5540,5542,5545,5548],{"class":3452,"line":5541},67,[3450,5543,5544],{"class":4672},"                break",[3450,5546,5547],{"class":3741},"; ",[3450,5549,5550],{"class":3735},"\u002F\u002F Отримали сигнал зупинки\n",[3450,5552,5554],{"class":3452,"line":5553},68,[3450,5555,3475],{"emptyLinePlaceholder":3474},[3450,5557,5559,5562,5564,5566,5569],{"class":3452,"line":5558},69,[3450,5560,5561],{"class":4714},"            context",[3450,5563,4718],{"class":3741},[3450,5565,5495],{"class":4711},[3450,5567,5568],{"class":4714}," getContextTask",[3450,5570,4684],{"class":3741},[3450,5572,5574],{"class":3452,"line":5573},70,[3450,5575,5576],{"class":3741},"        }\n",[3450,5578,5580,5583,5585,5587,5590,5593,5595,5597,5599,5601,5603,5605],{"class":3452,"line":5579},71,[3450,5581,5582],{"class":4672},"        catch",[3450,5584,5528],{"class":3741},[3450,5586,3943],{"class":4675},[3450,5588,5589],{"class":3741},") ",[3450,5591,5592],{"class":4672},"when",[3450,5594,5528],{"class":3741},[3450,5596,5360],{"class":4714},[3450,5598,3845],{"class":3741},[3450,5600,5365],{"class":4714},[3450,5602,3845],{"class":3741},[3450,5604,5370],{"class":4714},[3450,5606,4848],{"class":3741},[3450,5608,5610],{"class":3452,"line":5609},72,[3450,5611,5401],{"class":3741},[3450,5613,5615,5618,5620],{"class":3452,"line":5614},73,[3450,5616,5617],{"class":4672},"            break",[3450,5619,5547],{"class":3741},[3450,5621,5622],{"class":3735},"\u002F\u002F Listener зупинено через Stop()\n",[3450,5624,5626],{"class":3452,"line":5625},74,[3450,5627,5576],{"class":3741},[3450,5629,5631],{"class":3452,"line":5630},75,[3450,5632,3475],{"emptyLinePlaceholder":3474},[3450,5634,5636],{"class":3452,"line":5635},76,[3450,5637,5638],{"class":3735},"        \u002F\u002F Обробляємо запит у окремому завданні — не блокуємо прийом наступного\n",[3450,5640,5642,5645,5647,5649,5651,5654,5657,5660,5662,5665,5667,5669,5671,5674],{"class":3452,"line":5641},77,[3450,5643,5644],{"class":4714},"        _",[3450,5646,4718],{"class":3741},[3450,5648,4010],{"class":4714},[3450,5650,3845],{"class":3741},[3450,5652,5653],{"class":4726},"Run",[3450,5655,5656],{"class":3741},"(() => ",[3450,5658,5659],{"class":4726},"HandleRequestAsync",[3450,5661,4730],{"class":3741},[3450,5663,5664],{"class":4714},"context",[3450,5666,3858],{"class":3741},[3450,5668,4780],{"class":4714},[3450,5670,3858],{"class":3741},[3450,5672,5673],{"class":4714},"mimeTypes",[3450,5675,5676],{"class":3741},"));\n",[3450,5678,5680],{"class":3452,"line":5679},78,[3450,5681,5682],{"class":3741},"    }\n",[3450,5684,5686],{"class":3452,"line":5685},79,[3450,5687,5688],{"class":3741},"}\n",[3450,5690,5692],{"class":3452,"line":5691},80,[3450,5693,5694],{"class":4672},"finally\n",[3450,5696,5698],{"class":3452,"line":5697},81,[3450,5699,4853],{"class":3741},[3450,5701,5703,5706,5708,5711],{"class":3452,"line":5702},82,[3450,5704,5705],{"class":4714},"    listener",[3450,5707,3845],{"class":3741},[3450,5709,5710],{"class":4726},"Stop",[3450,5712,5100],{"class":3741},[3450,5714,5716,5719,5721,5723,5725,5727,5731,5734],{"class":3452,"line":5715},83,[3450,5717,5718],{"class":4714},"    Console",[3450,5720,3845],{"class":3741},[3450,5722,5256],{"class":4726},[3450,5724,4730],{"class":3741},[3450,5726,5273],{"class":4733},[3450,5728,5730],{"class":5729},"sjcCO","\\n",[3450,5732,5733],{"class":4733},"🛑 Сервер зупинено.\"",[3450,5735,4800],{"class":3741},[3450,5737,5739],{"class":3452,"line":5738},84,[3450,5740,5688],{"class":3741},[3450,5742,5744],{"class":3452,"line":5743},85,[3450,5745,3475],{"emptyLinePlaceholder":3474},[3450,5747,5749],{"class":3452,"line":5748},86,[3450,5750,5751],{"class":3735},"\u002F\u002F ── Обробник одного запиту ──────────────────────────────────────────────────\n",[3450,5753,5755,5758,5761,5763,5766],{"class":3452,"line":5754},87,[3450,5756,5757],{"class":4711},"static",[3450,5759,5760],{"class":4711}," async",[3450,5762,5498],{"class":4675},[3450,5764,5765],{"class":4726}," HandleRequestAsync",[3450,5767,5768],{"class":3741},"(\n",[3450,5770,5772,5775,5777],{"class":3452,"line":5771},88,[3450,5773,5774],{"class":4675},"    HttpListenerContext",[3450,5776,5387],{"class":4714},[3450,5778,4869],{"class":3741},[3450,5780,5782,5785,5787],{"class":3452,"line":5781},89,[3450,5783,5784],{"class":4711},"    string",[3450,5786,4715],{"class":4714},[3450,5788,4869],{"class":3741},[3450,5790,5792,5795,5797,5799,5801,5803,5805,5807],{"class":3452,"line":5791},90,[3450,5793,5794],{"class":4675},"    Dictionary",[3450,5796,4828],{"class":3741},[3450,5798,3889],{"class":4711},[3450,5800,3858],{"class":3741},[3450,5802,3889],{"class":4711},[3450,5804,5426],{"class":3741},[3450,5806,5673],{"class":4714},[3450,5808,4848],{"class":3741},[3450,5810,5812],{"class":3452,"line":5811},91,[3450,5813,4853],{"class":3741},[3450,5815,5817,5820,5823,5825,5827,5829,5831],{"class":3452,"line":5816},92,[3450,5818,5819],{"class":4675},"    HttpListenerRequest",[3450,5821,5822],{"class":4714},"  req",[3450,5824,4718],{"class":3741},[3450,5826,5664],{"class":4714},[3450,5828,3845],{"class":3741},[3450,5830,3991],{"class":4714},[3450,5832,4684],{"class":3741},[3450,5834,5836,5839,5842,5844,5846,5848,5850],{"class":3452,"line":5835},93,[3450,5837,5838],{"class":4675},"    HttpListenerResponse",[3450,5840,5841],{"class":4714}," res",[3450,5843,4718],{"class":3741},[3450,5845,5664],{"class":4714},[3450,5847,3845],{"class":3741},[3450,5849,3994],{"class":4714},[3450,5851,4684],{"class":3741},[3450,5853,5855],{"class":3452,"line":5854},94,[3450,5856,3475],{"emptyLinePlaceholder":3474},[3450,5858,5860,5862,5864,5866,5868,5871,5873,5876,5878,5881,5883,5886,5888,5891,5893,5896,5898,5901,5903,5906,5908,5910,5912,5915,5917,5919,5921,5923,5926],{"class":3452,"line":5859},95,[3450,5861,5718],{"class":4714},[3450,5863,3845],{"class":3741},[3450,5865,5256],{"class":4726},[3450,5867,4730],{"class":3741},[3450,5869,5870],{"class":4733},"$\"[",[3450,5872,5265],{"class":5264},[3450,5874,5875],{"class":4714},"DateTime",[3450,5877,3845],{"class":5264},[3450,5879,5880],{"class":4714},"Now",[3450,5882,3723],{"class":3741},[3450,5884,5885],{"class":4714},"HH",[3450,5887,3723],{"class":3741},[3450,5889,5890],{"class":4714},"mm",[3450,5892,3723],{"class":3741},[3450,5894,5895],{"class":4714},"ss",[3450,5897,5270],{"class":5264},[3450,5899,5900],{"class":4733},"] ",[3450,5902,5265],{"class":5264},[3450,5904,5905],{"class":4714},"req",[3450,5907,3845],{"class":5264},[3450,5909,4095],{"class":4714},[3450,5911,5270],{"class":5264},[3450,5913,5914],{"class":5264}," {",[3450,5916,5905],{"class":4714},[3450,5918,3845],{"class":5264},[3450,5920,4130],{"class":4714},[3450,5922,5270],{"class":5264},[3450,5924,5925],{"class":4733}," \"",[3450,5927,5928],{"class":3741}," +\n",[3450,5930,5932,5935,5937,5939,5941,5943,5945,5947],{"class":3452,"line":5931},96,[3450,5933,5934],{"class":4733},"                      $\"← ",[3450,5936,5265],{"class":5264},[3450,5938,5905],{"class":4714},[3450,5940,3845],{"class":5264},[3450,5942,4229],{"class":4714},[3450,5944,5270],{"class":5264},[3450,5946,5273],{"class":4733},[3450,5948,4800],{"class":3741},[3450,5950,5952],{"class":3452,"line":5951},97,[3450,5953,3475],{"emptyLinePlaceholder":3474},[3450,5955,5957],{"class":3452,"line":5956},98,[3450,5958,5959],{"class":4672},"    try\n",[3450,5961,5963],{"class":3452,"line":5962},99,[3450,5964,5378],{"class":3741},[3450,5966,5968],{"class":3452,"line":5967},100,[3450,5969,5970],{"class":3735},"        \u002F\u002F HEAD — ідентичний GET, але без тіла відповіді\n",[3450,5972,5974,5977,5979,5981,5983,5985,5988,5990,5993,5995,5997,5999,6001,6003],{"class":3452,"line":5973},101,[3450,5975,5976],{"class":4672},"        if",[3450,5978,5528],{"class":3741},[3450,5980,5905],{"class":4714},[3450,5982,3845],{"class":3741},[3450,5984,4095],{"class":4714},[3450,5986,5987],{"class":3741}," != ",[3450,5989,4099],{"class":4733},[3450,5991,5992],{"class":3741}," && ",[3450,5994,5905],{"class":4714},[3450,5996,3845],{"class":3741},[3450,5998,4095],{"class":4714},[3450,6000,5987],{"class":3741},[3450,6002,4114],{"class":4733},[3450,6004,4848],{"class":3741},[3450,6006,6008],{"class":3452,"line":6007},102,[3450,6009,5401],{"class":3741},[3450,6011,6013,6016,6019,6021,6024,6026,6029,6031,6034],{"class":3452,"line":6012},103,[3450,6014,6015],{"class":4711},"            await",[3450,6017,6018],{"class":4726}," SendErrorAsync",[3450,6020,4730],{"class":3741},[3450,6022,6023],{"class":4714},"res",[3450,6025,3858],{"class":3741},[3450,6027,6028],{"class":3745},"405",[3450,6030,3858],{"class":3741},[3450,6032,6033],{"class":4733},"\"Method Not Allowed\"",[3450,6035,4800],{"class":3741},[3450,6037,6039,6042,6044,6046,6048,6050,6052,6055,6057,6060],{"class":3452,"line":6038},104,[3450,6040,6041],{"class":4714},"            res",[3450,6043,3845],{"class":3741},[3450,6045,4151],{"class":4714},[3450,6047,3845],{"class":3741},[3450,6049,5115],{"class":4726},[3450,6051,4730],{"class":3741},[3450,6053,6054],{"class":4733},"\"Allow\"",[3450,6056,3858],{"class":3741},[3450,6058,6059],{"class":4733},"\"GET, HEAD\"",[3450,6061,4800],{"class":3741},[3450,6063,6065,6068],{"class":3452,"line":6064},105,[3450,6066,6067],{"class":4672},"            return",[3450,6069,4684],{"class":3741},[3450,6071,6073],{"class":3452,"line":6072},106,[3450,6074,5576],{"class":3741},[3450,6076,6078],{"class":3452,"line":6077},107,[3450,6079,3475],{"emptyLinePlaceholder":3474},[3450,6081,6083],{"class":3452,"line":6082},108,[3450,6084,6085],{"class":3735},"        \u002F\u002F Визначаємо локальний шлях до файлу\n",[3450,6087,6089],{"class":3452,"line":6088},109,[3450,6090,6091],{"class":3735},"        \u002F\u002F RawUrl: \"\u002Findex.html\" або \"\u002F\" або \"\u002Fcss\u002Fstyle.css\"\n",[3450,6093,6095,6098,6101,6103,6105,6107,6109,6112,6115],{"class":3452,"line":6094},110,[3450,6096,6097],{"class":4711},"        string",[3450,6099,6100],{"class":4714}," urlPath",[3450,6102,4718],{"class":3741},[3450,6104,5905],{"class":4714},[3450,6106,3845],{"class":3741},[3450,6108,4121],{"class":4714},[3450,6110,6111],{"class":3741},"!.",[3450,6113,6114],{"class":4714},"AbsolutePath",[3450,6116,4684],{"class":3741},[3450,6118,6120],{"class":3452,"line":6119},111,[3450,6121,3475],{"emptyLinePlaceholder":3474},[3450,6123,6125],{"class":3452,"line":6124},112,[3450,6126,6127],{"class":3735},"        \u002F\u002F Захист від path traversal: \"..\" у шляху може вийти за межі wwwroot\n",[3450,6129,6131,6133,6136,6138,6140,6142,6144],{"class":3452,"line":6130},113,[3450,6132,6097],{"class":4711},[3450,6134,6135],{"class":4714}," localPath",[3450,6137,4718],{"class":3741},[3450,6139,4721],{"class":4714},[3450,6141,3845],{"class":3741},[3450,6143,4727],{"class":4726},[3450,6145,5768],{"class":3741},[3450,6147,6149,6152,6154,6157,6159,6161,6163,6166,6168,6171,6173,6176],{"class":3452,"line":6148},114,[3450,6150,6151],{"class":4714},"            Path",[3450,6153,3845],{"class":3741},[3450,6155,6156],{"class":4726},"Combine",[3450,6158,4730],{"class":3741},[3450,6160,4780],{"class":4714},[3450,6162,3858],{"class":3741},[3450,6164,6165],{"class":4714},"urlPath",[3450,6167,3845],{"class":3741},[3450,6169,6170],{"class":4726},"TrimStart",[3450,6172,4730],{"class":3741},[3450,6174,6175],{"class":4733},"'\u002F'",[3450,6177,6178],{"class":3741},")));\n",[3450,6180,6182],{"class":3452,"line":6181},115,[3450,6183,3475],{"emptyLinePlaceholder":3474},[3450,6185,6187,6189,6191,6194,6196,6199,6201,6203,6205,6208,6210,6212],{"class":3452,"line":6186},116,[3450,6188,5976],{"class":4672},[3450,6190,4767],{"class":3741},[3450,6192,6193],{"class":4714},"localPath",[3450,6195,3845],{"class":3741},[3450,6197,6198],{"class":4726},"StartsWith",[3450,6200,4730],{"class":3741},[3450,6202,4780],{"class":4714},[3450,6204,3858],{"class":3741},[3450,6206,6207],{"class":4714},"StringComparison",[3450,6209,3845],{"class":3741},[3450,6211,4845],{"class":4714},[3450,6213,4783],{"class":3741},[3450,6215,6217],{"class":3452,"line":6216},117,[3450,6218,5401],{"class":3741},[3450,6220,6222,6224,6226,6228,6230,6232,6235,6237,6240],{"class":3452,"line":6221},118,[3450,6223,6015],{"class":4711},[3450,6225,6018],{"class":4726},[3450,6227,4730],{"class":3741},[3450,6229,6023],{"class":4714},[3450,6231,3858],{"class":3741},[3450,6233,6234],{"class":3745},"403",[3450,6236,3858],{"class":3741},[3450,6238,6239],{"class":4733},"\"Forbidden\"",[3450,6241,4800],{"class":3741},[3450,6243,6245,6247],{"class":3452,"line":6244},119,[3450,6246,6067],{"class":4672},[3450,6248,4684],{"class":3741},[3450,6250,6252],{"class":3452,"line":6251},120,[3450,6253,5576],{"class":3741},[3450,6255,6257],{"class":3452,"line":6256},121,[3450,6258,3475],{"emptyLinePlaceholder":3474},[3450,6260,6262],{"class":3452,"line":6261},122,[3450,6263,6264],{"class":3735},"        \u002F\u002F Якщо шлях вказує на директорію — шукаємо index.html\n",[3450,6266,6268,6270,6272,6274,6276,6278,6280,6282],{"class":3452,"line":6267},123,[3450,6269,5976],{"class":4672},[3450,6271,5528],{"class":3741},[3450,6273,4770],{"class":4714},[3450,6275,3845],{"class":3741},[3450,6277,4775],{"class":4726},[3450,6279,4730],{"class":3741},[3450,6281,6193],{"class":4714},[3450,6283,4783],{"class":3741},[3450,6285,6287,6290,6292,6294,6296,6298,6300,6302,6304,6307],{"class":3452,"line":6286},124,[3450,6288,6289],{"class":4714},"            localPath",[3450,6291,4718],{"class":3741},[3450,6293,4721],{"class":4714},[3450,6295,3845],{"class":3741},[3450,6297,6156],{"class":4726},[3450,6299,4730],{"class":3741},[3450,6301,6193],{"class":4714},[3450,6303,3858],{"class":3741},[3450,6305,6306],{"class":4733},"\"index.html\"",[3450,6308,4800],{"class":3741},[3450,6310,6312],{"class":3452,"line":6311},125,[3450,6313,3475],{"emptyLinePlaceholder":3474},[3450,6315,6317,6319,6321,6324,6326,6328,6330,6332],{"class":3452,"line":6316},126,[3450,6318,5976],{"class":4672},[3450,6320,4767],{"class":3741},[3450,6322,6323],{"class":4714},"File",[3450,6325,3845],{"class":3741},[3450,6327,4775],{"class":4726},[3450,6329,4730],{"class":3741},[3450,6331,6193],{"class":4714},[3450,6333,4783],{"class":3741},[3450,6335,6337],{"class":3452,"line":6336},127,[3450,6338,5401],{"class":3741},[3450,6340,6342,6344,6346,6348,6350,6352,6355],{"class":3452,"line":6341},128,[3450,6343,6015],{"class":4711},[3450,6345,6018],{"class":4726},[3450,6347,4730],{"class":3741},[3450,6349,6023],{"class":4714},[3450,6351,3858],{"class":3741},[3450,6353,6354],{"class":3745},"404",[3450,6356,4869],{"class":3741},[3450,6358,6360,6363,6365,6367,6369,6371],{"class":3452,"line":6359},129,[3450,6361,6362],{"class":4733},"                $\"File not found: ",[3450,6364,5265],{"class":5264},[3450,6366,6165],{"class":4714},[3450,6368,5270],{"class":5264},[3450,6370,5273],{"class":4733},[3450,6372,4800],{"class":3741},[3450,6374,6376,6378],{"class":3452,"line":6375},130,[3450,6377,6067],{"class":4672},[3450,6379,4684],{"class":3741},[3450,6381,6383],{"class":3452,"line":6382},131,[3450,6384,5576],{"class":3741},[3450,6386,6388],{"class":3452,"line":6387},132,[3450,6389,3475],{"emptyLinePlaceholder":3474},[3450,6391,6393],{"class":3452,"line":6392},133,[3450,6394,6395],{"class":3735},"        \u002F\u002F Визначаємо MIME-тип за розширенням\n",[3450,6397,6399,6401,6404,6406,6408,6410,6413,6415,6417],{"class":3452,"line":6398},134,[3450,6400,6097],{"class":4711},[3450,6402,6403],{"class":4714}," ext",[3450,6405,4718],{"class":3741},[3450,6407,4721],{"class":4714},[3450,6409,3845],{"class":3741},[3450,6411,6412],{"class":4726},"GetExtension",[3450,6414,4730],{"class":3741},[3450,6416,6193],{"class":4714},[3450,6418,4800],{"class":3741},[3450,6420,6422,6424,6427,6429,6431,6433,6436,6438,6441,6443,6446,6449,6452,6455],{"class":3452,"line":6421},135,[3450,6423,6097],{"class":4711},[3450,6425,6426],{"class":4714}," contentType",[3450,6428,4718],{"class":3741},[3450,6430,5673],{"class":4714},[3450,6432,3845],{"class":3741},[3450,6434,6435],{"class":4726},"TryGetValue",[3450,6437,4730],{"class":3741},[3450,6439,6440],{"class":4714},"ext",[3450,6442,3858],{"class":3741},[3450,6444,6445],{"class":4711},"out",[3450,6447,6448],{"class":4711}," string",[3450,6450,6451],{"class":3741},"? ",[3450,6453,6454],{"class":4714},"mime",[3450,6456,4848],{"class":3741},[3450,6458,6460,6463],{"class":3452,"line":6459},136,[3450,6461,6462],{"class":3741},"            ? ",[3450,6464,6465],{"class":4714},"mime\n",[3450,6467,6469,6472,6475],{"class":3452,"line":6468},137,[3450,6470,6471],{"class":3741},"            : ",[3450,6473,6474],{"class":4733},"\"application\u002Foctet-stream\"",[3450,6476,4684],{"class":3741},[3450,6478,6480],{"class":3452,"line":6479},138,[3450,6481,3475],{"emptyLinePlaceholder":3474},[3450,6483,6485],{"class":3452,"line":6484},139,[3450,6486,6487],{"class":3735},"        \u002F\u002F Відкриваємо файл та налаштовуємо відповідь\n",[3450,6489,6491,6494,6497,6500,6503,6505,6507,6509,6512,6514,6516],{"class":3452,"line":6490},140,[3450,6492,6493],{"class":4711},"        await",[3450,6495,6496],{"class":4672}," using",[3450,6498,6499],{"class":4675}," FileStream",[3450,6501,6502],{"class":4714}," fileStream",[3450,6504,4718],{"class":3741},[3450,6506,6323],{"class":4714},[3450,6508,3845],{"class":3741},[3450,6510,6511],{"class":4726},"OpenRead",[3450,6513,4730],{"class":3741},[3450,6515,6193],{"class":4714},[3450,6517,4800],{"class":3741},[3450,6519,6521],{"class":3452,"line":6520},141,[3450,6522,3475],{"emptyLinePlaceholder":3474},[3450,6524,6526,6529,6531,6533,6535,6537],{"class":3452,"line":6525},142,[3450,6527,6528],{"class":4714},"        res",[3450,6530,3845],{"class":3741},[3450,6532,4315],{"class":4714},[3450,6534,4750],{"class":3741},[3450,6536,4320],{"class":3745},[3450,6538,4684],{"class":3741},[3450,6540,6542,6544,6546,6548,6550,6553],{"class":3452,"line":6541},143,[3450,6543,6528],{"class":4714},[3450,6545,3845],{"class":3741},[3450,6547,4187],{"class":4714},[3450,6549,4718],{"class":3741},[3450,6551,6552],{"class":4714},"contentType",[3450,6554,4684],{"class":3741},[3450,6556,6558,6560,6562,6564,6566,6569,6571,6574],{"class":3452,"line":6557},144,[3450,6559,6528],{"class":4714},[3450,6561,3845],{"class":3741},[3450,6563,4203],{"class":4714},[3450,6565,4718],{"class":3741},[3450,6567,6568],{"class":4714},"fileStream",[3450,6570,3845],{"class":3741},[3450,6572,6573],{"class":4714},"Length",[3450,6575,4684],{"class":3741},[3450,6577,6579],{"class":3452,"line":6578},145,[3450,6580,3475],{"emptyLinePlaceholder":3474},[3450,6582,6584],{"class":3452,"line":6583},146,[3450,6585,6586],{"class":3735},"        \u002F\u002F Заголовки кешування: дозволяємо кешувати на 1 годину\n",[3450,6588,6590,6592,6594,6596,6599,6602,6604,6607,6609,6612],{"class":3452,"line":6589},147,[3450,6591,6528],{"class":4714},[3450,6593,3845],{"class":3741},[3450,6595,4151],{"class":4714},[3450,6597,6598],{"class":3741},"[",[3450,6600,6601],{"class":4714},"HttpResponseHeader",[3450,6603,3845],{"class":3741},[3450,6605,6606],{"class":4714},"CacheControl",[3450,6608,4864],{"class":3741},[3450,6610,6611],{"class":4733},"\"public, max-age=3600\"",[3450,6613,4684],{"class":3741},[3450,6615,6617,6619,6621,6623,6625,6628],{"class":3452,"line":6616},148,[3450,6618,6528],{"class":4714},[3450,6620,3845],{"class":3741},[3450,6622,4151],{"class":4714},[3450,6624,6598],{"class":3741},[3450,6626,6627],{"class":4733},"\"Last-Modified\"",[3450,6629,6630],{"class":3741},"] =\n",[3450,6632,6634,6637,6639,6642,6644,6646,6648,6651,6653,6656],{"class":3452,"line":6633},149,[3450,6635,6636],{"class":4714},"            File",[3450,6638,3845],{"class":3741},[3450,6640,6641],{"class":4726},"GetLastWriteTimeUtc",[3450,6643,4730],{"class":3741},[3450,6645,6193],{"class":4714},[3450,6647,4226],{"class":3741},[3450,6649,6650],{"class":4726},"ToString",[3450,6652,4730],{"class":3741},[3450,6654,6655],{"class":4733},"\"R\"",[3450,6657,4800],{"class":3741},[3450,6659,6661],{"class":3452,"line":6660},150,[3450,6662,3475],{"emptyLinePlaceholder":3474},[3450,6664,6666],{"class":3452,"line":6665},151,[3450,6667,6668],{"class":3735},"        \u002F\u002F HEAD-запит: заголовки без тіла\n",[3450,6670,6672,6674,6676,6678,6680,6682,6684,6686],{"class":3452,"line":6671},152,[3450,6673,5976],{"class":4672},[3450,6675,5528],{"class":3741},[3450,6677,5905],{"class":4714},[3450,6679,3845],{"class":3741},[3450,6681,4095],{"class":4714},[3450,6683,5534],{"class":3741},[3450,6685,4114],{"class":4733},[3450,6687,4848],{"class":3741},[3450,6689,6691],{"class":3452,"line":6690},153,[3450,6692,5401],{"class":3741},[3450,6694,6696,6698,6700,6703],{"class":3452,"line":6695},154,[3450,6697,6041],{"class":4714},[3450,6699,3845],{"class":3741},[3450,6701,6702],{"class":4726},"Close",[3450,6704,5100],{"class":3741},[3450,6706,6708,6710],{"class":3452,"line":6707},155,[3450,6709,6067],{"class":4672},[3450,6711,4684],{"class":3741},[3450,6713,6715],{"class":3452,"line":6714},156,[3450,6716,5576],{"class":3741},[3450,6718,6720],{"class":3452,"line":6719},157,[3450,6721,3475],{"emptyLinePlaceholder":3474},[3450,6723,6725],{"class":3452,"line":6724},158,[3450,6726,6727],{"class":3735},"        \u002F\u002F Потокова передача файлу — не завантажуємо повністю у пам'ять\n",[3450,6729,6731,6733,6735,6737,6740,6742,6744,6746,6748],{"class":3452,"line":6730},159,[3450,6732,6493],{"class":4711},[3450,6734,6502],{"class":4714},[3450,6736,3845],{"class":3741},[3450,6738,6739],{"class":4726},"CopyToAsync",[3450,6741,4730],{"class":3741},[3450,6743,6023],{"class":4714},[3450,6745,3845],{"class":3741},[3450,6747,4324],{"class":4714},[3450,6749,4800],{"class":3741},[3450,6751,6753],{"class":3452,"line":6752},160,[3450,6754,5682],{"class":3741},[3450,6756,6758,6761,6763,6766,6769],{"class":3452,"line":6757},161,[3450,6759,6760],{"class":4672},"    catch",[3450,6762,5528],{"class":3741},[3450,6764,6765],{"class":4675},"Exception",[3450,6767,6768],{"class":4714}," ex",[3450,6770,4848],{"class":3741},[3450,6772,6774],{"class":3452,"line":6773},162,[3450,6775,5378],{"class":3741},[3450,6777,6779,6782,6784,6787,6789,6791,6793,6796,6798,6800,6802,6804,6806,6808,6810,6813,6815,6818,6820,6822],{"class":3452,"line":6778},163,[3450,6780,6781],{"class":4714},"        Console",[3450,6783,3845],{"class":3741},[3450,6785,6786],{"class":4714},"Error",[3450,6788,3845],{"class":3741},[3450,6790,5256],{"class":4726},[3450,6792,4730],{"class":3741},[3450,6794,6795],{"class":4733},"$\"Помилка обробки ",[3450,6797,5265],{"class":5264},[3450,6799,5905],{"class":4714},[3450,6801,3845],{"class":5264},[3450,6803,4130],{"class":4714},[3450,6805,5270],{"class":5264},[3450,6807,4193],{"class":4733},[3450,6809,5265],{"class":5264},[3450,6811,6812],{"class":4714},"ex",[3450,6814,3845],{"class":5264},[3450,6816,6817],{"class":4714},"Message",[3450,6819,5270],{"class":5264},[3450,6821,5273],{"class":4733},[3450,6823,4800],{"class":3741},[3450,6825,6827,6830,6833,6835,6837,6839,6841,6843,6846,6848,6851],{"class":3452,"line":6826},164,[3450,6828,6829],{"class":4672},"        try",[3450,6831,6832],{"class":3741}," { ",[3450,6834,5495],{"class":4711},[3450,6836,6018],{"class":4726},[3450,6838,4730],{"class":3741},[3450,6840,6023],{"class":4714},[3450,6842,3858],{"class":3741},[3450,6844,6845],{"class":3745},"500",[3450,6847,3858],{"class":3741},[3450,6849,6850],{"class":4733},"\"Internal Server Error\"",[3450,6852,6853],{"class":3741},"); }\n",[3450,6855,6857,6859,6861,6864],{"class":3452,"line":6856},165,[3450,6858,5582],{"class":4672},[3450,6860,6832],{"class":3741},[3450,6862,6863],{"class":3735},"\u002F* Відповідь вже могла бути частково надіслана *\u002F",[3450,6865,6866],{"class":3741}," }\n",[3450,6868,6870],{"class":3452,"line":6869},166,[3450,6871,5682],{"class":3741},[3450,6873,6875],{"class":3452,"line":6874},167,[3450,6876,6877],{"class":4672},"    finally\n",[3450,6879,6881],{"class":3452,"line":6880},168,[3450,6882,5378],{"class":3741},[3450,6884,6886],{"class":3452,"line":6885},169,[3450,6887,6888],{"class":3735},"        \u002F\u002F ОБОВ'ЯЗКОВО — завжди закривати відповідь!\n",[3450,6890,6892,6894,6896,6898,6900,6902,6904,6906,6909,6912],{"class":3452,"line":6891},170,[3450,6893,6829],{"class":4672},[3450,6895,6832],{"class":3741},[3450,6897,6023],{"class":4714},[3450,6899,3845],{"class":3741},[3450,6901,4324],{"class":4714},[3450,6903,3845],{"class":3741},[3450,6905,6702],{"class":4726},[3450,6907,6908],{"class":3741},"(); } ",[3450,6910,6911],{"class":4672},"catch",[3450,6913,6914],{"class":3741}," { }\n",[3450,6916,6918],{"class":3452,"line":6917},171,[3450,6919,5682],{"class":3741},[3450,6921,6923],{"class":3452,"line":6922},172,[3450,6924,5688],{"class":3741},[3450,6926,6928],{"class":3452,"line":6927},173,[3450,6929,3475],{"emptyLinePlaceholder":3474},[3450,6931,6933],{"class":3452,"line":6932},174,[3450,6934,6935],{"class":3735},"\u002F\u002F ── Допоміжний метод для надсилання помилок ────────────────────────────────\n",[3450,6937,6939,6941,6943,6945,6947],{"class":3452,"line":6938},175,[3450,6940,5757],{"class":4711},[3450,6942,5760],{"class":4711},[3450,6944,5498],{"class":4675},[3450,6946,6018],{"class":4726},[3450,6948,5768],{"class":3741},[3450,6950,6952,6954,6956,6958,6960,6963,6965,6967,6970],{"class":3452,"line":6951},176,[3450,6953,5838],{"class":4675},[3450,6955,5841],{"class":4714},[3450,6957,3858],{"class":3741},[3450,6959,4316],{"class":4711},[3450,6961,6962],{"class":4714}," statusCode",[3450,6964,3858],{"class":3741},[3450,6966,3889],{"class":4711},[3450,6968,6969],{"class":4714}," message",[3450,6971,4848],{"class":3741},[3450,6973,6975],{"class":3452,"line":6974},177,[3450,6976,4853],{"class":3741},[3450,6978,6980,6983,6986,6989,6991,6993,6995,6998,7000,7003,7005],{"class":3452,"line":6979},178,[3450,6981,6982],{"class":4711},"    byte",[3450,6984,6985],{"class":3741},"[] ",[3450,6987,6988],{"class":4714},"body",[3450,6990,4718],{"class":3741},[3450,6992,4368],{"class":4714},[3450,6994,3845],{"class":3741},[3450,6996,6997],{"class":4714},"UTF8",[3450,6999,3845],{"class":3741},[3450,7001,7002],{"class":4726},"GetBytes",[3450,7004,4730],{"class":3741},[3450,7006,7007],{"class":4733},"$\"\"\"\n",[3450,7009,7011],{"class":3452,"line":7010},179,[3450,7012,7013],{"class":4733},"        \u003C!DOCTYPE html>\n",[3450,7015,7017],{"class":3452,"line":7016},180,[3450,7018,7019],{"class":4733},"        \u003Chtml lang=\"uk\">\u003Chead>\u003Cmeta charset=\"UTF-8\">\n",[3450,7021,7023,7026,7028,7031,7033],{"class":3452,"line":7022},181,[3450,7024,7025],{"class":4733},"        \u003Ctitle>",[3450,7027,5265],{"class":5264},[3450,7029,7030],{"class":4714},"statusCode",[3450,7032,5270],{"class":5264},[3450,7034,7035],{"class":4733},"\u003C\u002Ftitle>\u003C\u002Fhead>\n",[3450,7037,7039],{"class":3452,"line":7038},182,[3450,7040,7041],{"class":4733},"        \u003Cbody>\n",[3450,7043,7045,7048,7050,7052,7054],{"class":3452,"line":7044},183,[3450,7046,7047],{"class":4733},"        \u003Ch1>",[3450,7049,5265],{"class":5264},[3450,7051,7030],{"class":4714},[3450,7053,5270],{"class":5264},[3450,7055,7056],{"class":4733},"\u003C\u002Fh1>\n",[3450,7058,7060,7063,7065,7068,7070,7072,7074,7077,7079,7082,7084,7087,7090],{"class":3452,"line":7059},184,[3450,7061,7062],{"class":4733},"        \u003Cp>",[3450,7064,5265],{"class":5264},[3450,7066,7067],{"class":4714},"System",[3450,7069,3845],{"class":5264},[3450,7071,4681],{"class":4714},[3450,7073,3845],{"class":5264},[3450,7075,7076],{"class":4714},"WebUtility",[3450,7078,3845],{"class":5264},[3450,7080,7081],{"class":4726},"HtmlEncode",[3450,7083,4730],{"class":5264},[3450,7085,7086],{"class":4714},"message",[3450,7088,7089],{"class":5264},")}",[3450,7091,7092],{"class":4733},"\u003C\u002Fp>\n",[3450,7094,7096],{"class":3452,"line":7095},185,[3450,7097,7098],{"class":4733},"        \u003C\u002Fbody>\u003C\u002Fhtml>\n",[3450,7100,7102,7105],{"class":3452,"line":7101},186,[3450,7103,7104],{"class":4733},"        \"\"\"",[3450,7106,4800],{"class":3741},[3450,7108,7110],{"class":3452,"line":7109},187,[3450,7111,3475],{"emptyLinePlaceholder":3474},[3450,7113,7115,7118,7120,7122,7125,7127],{"class":3452,"line":7114},188,[3450,7116,7117],{"class":4714},"    res",[3450,7119,3845],{"class":3741},[3450,7121,4315],{"class":4714},[3450,7123,7124],{"class":3741},"   = ",[3450,7126,7030],{"class":4714},[3450,7128,4684],{"class":3741},[3450,7130,7132,7134,7136,7138,7140,7142],{"class":3452,"line":7131},189,[3450,7133,7117],{"class":4714},[3450,7135,3845],{"class":3741},[3450,7137,4187],{"class":4714},[3450,7139,4750],{"class":3741},[3450,7141,4351],{"class":4733},[3450,7143,4684],{"class":3741},[3450,7145,7147,7149,7151,7153,7155,7157,7159,7161],{"class":3452,"line":7146},190,[3450,7148,7117],{"class":4714},[3450,7150,3845],{"class":3741},[3450,7152,4203],{"class":4714},[3450,7154,4718],{"class":3741},[3450,7156,6988],{"class":4714},[3450,7158,3845],{"class":3741},[3450,7160,6573],{"class":4714},[3450,7162,4684],{"class":3741},[3450,7164,7166],{"class":3452,"line":7165},191,[3450,7167,3475],{"emptyLinePlaceholder":3474},[3450,7169,7171,7174,7176,7178,7180,7182,7185,7187,7189],{"class":3452,"line":7170},192,[3450,7172,7173],{"class":4711},"    await",[3450,7175,5841],{"class":4714},[3450,7177,3845],{"class":3741},[3450,7179,4324],{"class":4714},[3450,7181,3845],{"class":3741},[3450,7183,7184],{"class":4726},"WriteAsync",[3450,7186,4730],{"class":3741},[3450,7188,6988],{"class":4714},[3450,7190,4800],{"class":3741},[3450,7192,7194,7196,7198,7200,7202,7204],{"class":3452,"line":7193},193,[3450,7195,7117],{"class":4714},[3450,7197,3845],{"class":3741},[3450,7199,4324],{"class":4714},[3450,7201,3845],{"class":3741},[3450,7203,6702],{"class":4726},[3450,7205,5100],{"class":3741},[3450,7207,7209],{"class":3452,"line":7208},194,[3450,7210,5688],{"class":3741},[3369,7212,7214],{"id":7213},"що-демонструє-цей-приклад","Що демонструє цей приклад",[3317,7216,7217],{},"Розберемо ключові аспекти реалізації:",[3317,7219,7220,7226,7227,7229,7230,7232],{},[3346,7221,7222,7223,3845],{},"Паралельна обробка через ",[3321,7224,7225],{},"Task.Run"," Кожен запит обробляється у власному Task. Якби ми ",[3321,7228,5495],{},"-ували ",[3321,7231,5659],{}," у головному циклі — сервер обробляв би лише один запит одночасно. Файловий сервер повинен обробляти декілька паралельних завантажень.",[3317,7234,7235,3349,7241,7243,7244,7246,7247,7250,7251,7253],{},[3346,7236,7237,7238,3845],{},"Скасування через ",[3321,7239,7240],{},"Task.WhenAny",[3321,7242,3955],{}," не приймає ",[3321,7245,4014],{},". Обхід: ",[3321,7248,7249],{},"Task.WhenAny(getContextTask, Task.Delay(Timeout.Infinite, cts.Token))",". Якщо ",[3321,7252,5512],{}," завершується першою — виходимо з циклу.",[3317,7255,7256,7259,7260,7263,7264,7267,7268,7270,7271,7274],{},[3346,7257,7258],{},"Захист від Path Traversal атаки."," URL ",[3321,7261,7262],{},"\u002F..\u002F..\u002Fetc\u002Fpasswd"," після ",[3321,7265,7266],{},"Path.GetFullPath()"," може вийти за межі ",[3321,7269,4780],{},". Перевірка ",[3321,7272,7273],{},"localPath.StartsWith(wwwroot)"," запобігає цьому.",[3317,7276,7277,3349,7280,7283],{},[3346,7278,7279],{},"Потокова передача файлів.",[3321,7281,7282],{},"fileStream.CopyToAsync(res.OutputStream)"," передає файл по частинах без завантаження у пам'ять. Критично для великих файлів.",[3317,7285,7286,7289],{},[3346,7287,7288],{},"HEAD-метод."," Клієнти (браузери, curl) використовують HEAD для отримання метаданих (розмір, тип, дата) без завантаження тіла. Наш сервер коректно відповідає без тіла.",[3422,7291],{},[3312,7293,7295],{"id":7294},"приклад-2-мінімальний-rest-api-сервер","Приклад 2: Мінімальний REST API сервер",[3317,7297,7298],{},"Другий приклад значно складніший: повноцінний REST API для управління колекцією задач (To-Do list). Він демонструє маршрутизацію, обробку різних HTTP-методів, читання тіла запиту (JSON), запис JSON-відповідей, обробку помилок та базову аутентифікацію.",[3441,7300,7302],{"className":4662,"code":7301,"language":4664,"meta":4665,"style":3446},"using System.Collections.Concurrent;\nusing System.Net;\nusing System.Text;\nusing System.Text.Json;\nusing System.Text.Json.Serialization;\nusing System.Text.RegularExpressions;\n\n\u002F\u002F ── Модель даних ─────────────────────────────────────────────────────────────\nrecord TodoItem(\n    int Id,\n    string Title,\n    bool IsCompleted,\n    DateTimeOffset CreatedAt\n);\n\n\u002F\u002F ── Сховище в пам'яті ────────────────────────────────────────────────────────\n\u002F\u002F ConcurrentDictionary забезпечує thread-safe доступ без явних lock-ів\nvar store = new ConcurrentDictionary\u003Cint, TodoItem>();\nint nextId = 1; \u002F\u002F Лічильник ID — інкрементується через Interlocked\n\n\u002F\u002F Заповнюємо початковими даними\nfor (int i = 1; i \u003C= 3; i++)\n    store[i] = new TodoItem(i, $\"Завдання #{i}\", i == 1, DateTimeOffset.UtcNow);\nnextId = 4;\n\n\u002F\u002F ── Налаштування серіалізації ────────────────────────────────────────────────\nvar jsonOptions = new JsonSerializerOptions\n{\n    PropertyNamingPolicy = JsonNamingPolicy.CamelCase,\n    WriteIndented = true,\n    DefaultIgnoreCondition = JsonIgnoreCondition.WhenWritingNull\n};\n\n\u002F\u002F ── Маршрути: регулярні вирази для REST URL ──────────────────────────────────\n\u002F\u002F REST API маршрути:\n\u002F\u002F   GET    \u002Ftodos        → список усіх задач\n\u002F\u002F   POST   \u002Ftodos        → створення нової задачі\n\u002F\u002F   GET    \u002Ftodos\u002F{id}   → отримання конкретної задачі\n\u002F\u002F   PUT    \u002Ftodos\u002F{id}   → повне оновлення задачі\n\u002F\u002F   PATCH  \u002Ftodos\u002F{id}   → часткове оновлення (лише IsCompleted)\n\u002F\u002F   DELETE \u002Ftodos\u002F{id}   → видалення задачі\nstatic readonly Regex RouteCollection = new(@\"^\u002Ftodos\u002F?$\",\n    RegexOptions.Compiled | RegexOptions.IgnoreCase);\nstatic readonly Regex RouteItem = new(@\"^\u002Ftodos\u002F(?\u003Cid>\\d+)\u002F?$\",\n    RegexOptions.Compiled | RegexOptions.IgnoreCase);\n\n\u002F\u002F ── Ініціалізація HttpListener ───────────────────────────────────────────────\nusing var listener = new HttpListener();\nlistener.Prefixes.Add(\"http:\u002F\u002Flocalhost:8081\u002F\");\n\n\u002F\u002F Basic Auth: логін admin, пароль secret (для прикладу)\nlistener.AuthenticationSchemes = AuthenticationSchemes.Basic;\nlistener.Realm = \"TodoAPI\";\n\nusing var cts = new CancellationTokenSource();\nConsole.CancelKeyPress += (_, e) => { e.Cancel = true; cts.Cancel(); };\n\nlistener.Start();\nConsole.WriteLine(\"📋 Todo REST API запущено: http:\u002F\u002Flocalhost:8081\u002F\");\nConsole.WriteLine(\"   Аутентифікація: admin \u002F secret\");\nConsole.WriteLine(\"   Натисніть Ctrl+C для зупинки.\\n\");\n\n\u002F\u002F ── Основний цикл ────────────────────────────────────────────────────────────\ntry\n{\n    while (!cts.Token.IsCancellationRequested)\n    {\n        Task\u003CHttpListenerContext> getCtx = listener.GetContextAsync();\n        Task cancel = Task.Delay(Timeout.Infinite, cts.Token);\n\n        if (await Task.WhenAny(getCtx, cancel) == cancel) break;\n\n        HttpListenerContext ctx = await getCtx;\n        _ = Task.Run(() => ProcessRequestAsync(ctx));\n    }\n}\nfinally\n{\n    listener.Stop();\n    Console.WriteLine(\"\\n🛑 API сервер зупинено.\");\n}\n\n\u002F\u002F ── Диспетчер запитів ────────────────────────────────────────────────────────\nasync Task ProcessRequestAsync(HttpListenerContext ctx)\n{\n    HttpListenerRequest  req = ctx.Request;\n    HttpListenerResponse res = ctx.Response;\n\n    \u002F\u002F ─ Логування ──────────────────────────────────────────────────────────\n    Console.WriteLine($\"  [{DateTime.Now:HH:mm:ss}] {req.HttpMethod} {req.RawUrl}\");\n\n    try\n    {\n        \u002F\u002F ─ Аутентифікація ─────────────────────────────────────────────────\n        \u002F\u002F HttpListener перевіряє Basic Auth заголовок автоматично\n        \u002F\u002F і заповнює ctx.User при успіху\n        if (!ctx.User.Identity.IsAuthenticated)\n        {\n            await WriteJsonAsync(res, 401,\n                new { error = \"Unauthorized\", message = \"Потрібна Basic аутентифікація\" });\n            return;\n        }\n\n        \u002F\u002F Перевіряємо облікові дані (у реальності — із БД або Identity)\n        var identity = (HttpListenerBasicIdentity)ctx.User.Identity;\n        if (identity.Password != \"secret\")\n        {\n            await WriteJsonAsync(res, 403,\n                new { error = \"Forbidden\", message = \"Невірний пароль\" });\n            return;\n        }\n\n        \u002F\u002F ─ CORS Headers ───────────────────────────────────────────────────\n        res.Headers.Add(\"Access-Control-Allow-Origin\", \"*\");\n        res.Headers.Add(\"Access-Control-Allow-Methods\", \"GET, POST, PUT, PATCH, DELETE\");\n        res.Headers.Add(\"Access-Control-Allow-Headers\", \"Content-Type, Authorization\");\n\n        \u002F\u002F OPTIONS preflight (браузерний CORS)\n        if (req.HttpMethod == \"OPTIONS\")\n        {\n            res.StatusCode = 204;\n            res.Close();\n            return;\n        }\n\n        \u002F\u002F ─ Маршрутизація ──────────────────────────────────────────────────\n        string path = req.Url!.AbsolutePath;\n\n        if (RouteCollection.IsMatch(path))\n        {\n            await HandleCollectionAsync(req, res);\n        }\n        else if (RouteItem.Match(path) is { Success: true } match)\n        {\n            int id = int.Parse(match.Groups[\"id\"].Value);\n            await HandleItemAsync(req, res, id);\n        }\n        else\n        {\n            await WriteJsonAsync(res, 404,\n                new { error = \"Not Found\", path });\n        }\n    }\n    catch (Exception ex)\n    {\n        Console.Error.WriteLine($\"  ❌ Помилка: {ex.Message}\");\n        try\n        {\n            await WriteJsonAsync(res, 500,\n                new { error = \"Internal Server Error\", message = ex.Message });\n        }\n        catch { }\n    }\n    finally\n    {\n        try { res.OutputStream.Close(); } catch { }\n    }\n}\n\n\u002F\u002F ── GET \u002Ftodos, POST \u002Ftodos ───────────────────────────────────────────────────\nasync Task HandleCollectionAsync(HttpListenerRequest req, HttpListenerResponse res)\n{\n    switch (req.HttpMethod)\n    {\n        case \"GET\":\n        {\n            \u002F\u002F Фільтрація: ?completed=true|false\n            string? completedFilter = req.QueryString[\"completed\"];\n            IEnumerable\u003CTodoItem> items = store.Values.OrderBy(t => t.Id);\n\n            if (completedFilter is not null &&\n                bool.TryParse(completedFilter, out bool filterValue))\n                items = items.Where(t => t.IsCompleted == filterValue);\n\n            await WriteJsonAsync(res, 200, items.ToArray());\n            break;\n        }\n\n        case \"POST\":\n        {\n            if (!req.HasEntityBody)\n            {\n                await WriteJsonAsync(res, 400,\n                    new { error = \"Bad Request\", message = \"Тіло запиту обов'язкове\" });\n                return;\n            }\n\n            \u002F\u002F Читаємо та десеріалізуємо JSON-тіло\n            using var reader = new StreamReader(req.InputStream,\n                req.ContentEncoding ?? Encoding.UTF8);\n            string body = await reader.ReadToEndAsync();\n\n            using var doc = JsonDocument.Parse(body);\n            string? title = doc.RootElement\n                .GetProperty(\"title\").GetString();\n\n            if (string.IsNullOrWhiteSpace(title))\n            {\n                await WriteJsonAsync(res, 422,\n                    new { error = \"Unprocessable Entity\",\n                          message = \"Поле 'title' обов'язкове та не може бути порожнім\" });\n                return;\n            }\n\n            \u002F\u002F Атомарна операція збільшення лічильника ID\n            int id = Interlocked.Increment(ref nextId);\n            var item = new TodoItem(id, title.Trim(), false, DateTimeOffset.UtcNow);\n            store[id] = item;\n\n            \u002F\u002F 201 Created + заголовок Location з URL нового ресурсу\n            res.Headers.Add(\"Location\", $\"\u002Ftodos\u002F{id}\");\n            await WriteJsonAsync(res, 201, item);\n            break;\n        }\n\n        default:\n            res.Headers.Add(\"Allow\", \"GET, POST\");\n            await WriteJsonAsync(res, 405,\n                new { error = \"Method Not Allowed\" });\n            break;\n    }\n}\n\n\u002F\u002F ── GET\u002FPUT\u002FPATCH\u002FDELETE \u002Ftodos\u002F{id} ─────────────────────────────────────────\nasync Task HandleItemAsync(\n    HttpListenerRequest req, HttpListenerResponse res, int id)\n{\n    \u002F\u002F Перевіряємо існування ресурсу\n    if (!store.TryGetValue(id, out TodoItem? existing))\n    {\n        await WriteJsonAsync(res, 404,\n            new { error = \"Not Found\", id });\n        return;\n    }\n\n    switch (req.HttpMethod)\n    {\n        case \"GET\":\n            await WriteJsonAsync(res, 200, existing);\n            break;\n\n        case \"PUT\":\n        {\n            \u002F\u002F Повне оновлення — вимагає всі поля\n            using var reader = new StreamReader(req.InputStream,\n                req.ContentEncoding ?? Encoding.UTF8);\n            string body = await reader.ReadToEndAsync();\n            using var doc = JsonDocument.Parse(body);\n\n            string? title = doc.RootElement.GetProperty(\"title\").GetString();\n            bool isCompleted = doc.RootElement\n                .GetProperty(\"isCompleted\").GetBoolean();\n\n            if (string.IsNullOrWhiteSpace(title))\n            {\n                await WriteJsonAsync(res, 422,\n                    new { error = \"Unprocessable Entity\", field = \"title\" });\n                return;\n            }\n\n            var updated = existing with\n            {\n                Title = title.Trim(),\n                IsCompleted = isCompleted\n            };\n            store[id] = updated;\n            await WriteJsonAsync(res, 200, updated);\n            break;\n        }\n\n        case \"PATCH\":\n        {\n            \u002F\u002F Часткове оновлення — тільки isCompleted\n            using var reader = new StreamReader(req.InputStream,\n                req.ContentEncoding ?? Encoding.UTF8);\n            string body = await reader.ReadToEndAsync();\n            using var doc = JsonDocument.Parse(body);\n\n            bool? isCompleted = null;\n            if (doc.RootElement.TryGetProperty(\"isCompleted\", out var val))\n                isCompleted = val.GetBoolean();\n\n            if (isCompleted is null)\n            {\n                await WriteJsonAsync(res, 400,\n                    new { error = \"Bad Request\",\n                          message = \"Хоча б одне поле має бути вказано\" });\n                return;\n            }\n\n            var patched = existing with { IsCompleted = isCompleted.Value };\n            store[id] = patched;\n            await WriteJsonAsync(res, 200, patched);\n            break;\n        }\n\n        case \"DELETE\":\n        {\n            store.TryRemove(id, out _);\n            res.StatusCode = 204;  \u002F\u002F No Content — успішно видалено, тіло відсутнє\n            res.Close();\n            break;\n        }\n\n        default:\n            res.Headers.Add(\"Allow\", \"GET, PUT, PATCH, DELETE\");\n            await WriteJsonAsync(res, 405,\n                new { error = \"Method Not Allowed\" });\n            break;\n    }\n}\n\n\u002F\u002F ── Допоміжний метод: серіалізація та відправлення JSON ──────────────────────\nasync Task WriteJsonAsync\u003CT>(HttpListenerResponse res, int statusCode, T data)\n{\n    byte[] body = JsonSerializer.SerializeToUtf8Bytes(data, jsonOptions);\n\n    res.StatusCode   = statusCode;\n    res.ContentType  = \"application\u002Fjson; charset=utf-8\";\n    res.ContentLength64 = body.Length;\n\n    \u002F\u002F Стандартні заголовки безпеки та відсутності кешу для API\n    res.Headers[HttpResponseHeader.CacheControl] = \"no-store\";\n    res.Headers.Add(\"X-Content-Type-Options\", \"nosniff\");\n\n    await res.OutputStream.WriteAsync(body);\n}\n",[3321,7303,7304,7322,7334,7346,7363,7384,7401,7405,7410,7420,7430,7439,7449,7457,7461,7465,7470,7475,7501,7518,7522,7527,7561,7613,7625,7629,7634,7648,7652,7669,7680,7695,7699,7703,7708,7713,7718,7723,7728,7733,7738,7743,7767,7790,7812,7830,7834,7839,7855,7874,7878,7883,7901,7916,7920,7936,7976,7980,7990,8005,8020,8039,8043,8048,8052,8056,8074,8078,8102,8135,8139,8174,8178,8194,8218,8222,8226,8230,8234,8244,8263,8267,8271,8276,8294,8298,8314,8330,8334,8339,8400,8404,8408,8412,8417,8422,8427,8449,8453,8471,8498,8504,8508,8512,8517,8545,8566,8570,8586,8609,8615,8619,8623,8628,8652,8676,8700,8704,8709,8727,8731,8746,8756,8762,8766,8770,8775,8796,8800,8821,8825,8842,8846,8890,8894,8933,8955,8959,8964,8968,8984,9002,9006,9010,9022,9026,9057,9061,9065,9081,9107,9111,9117,9121,9125,9129,9151,9155,9159,9163,9168,9191,9195,9210,9214,9225,9229,9234,9260,9306,9310,9330,9356,9390,9394,9420,9426,9430,9434,9443,9447,9461,9466,9484,9509,9516,9521,9525,9530,9557,9577,9597,9601,9625,9644,9665,9670,9690,9695,9713,9729,9742,9749,9754,9759,9765,9791,9835,9852,9857,9863,9896,9918,9925,9930,9935,9943,9967,9984,9999,10006,10011,10016,10021,10027,10038,10059,10064,10070,10101,10106,10123,10143,10151,10156,10161,10176,10181,10190,10211,10218,10223,10233,10238,10244,10269,10288,10307,10330,10335,10367,10384,10403,10408,10427,10432,10449,10472,10479,10484,10489,10504,10509,10526,10537,10543,10559,10580,10587,10592,10597,10607,10612,10618,10643,10662,10681,10704,10709,10725,10758,10775,10780,10795,10800,10817,10832,10844,10851,10856,10861,10891,10907,10928,10935,10940,10945,10955,10960,10983,11002,11013,11020,11025,11030,11037,11061,11078,11093,11100,11105,11110,11115,11121,11156,11161,11192,11197,11212,11227,11246,11251,11257,11281,11306,11311,11332],{"__ignoreMap":3446},[3450,7305,7306,7308,7310,7312,7315,7317,7320],{"class":3452,"line":3453},[3450,7307,4398],{"class":4672},[3450,7309,4676],{"class":4675},[3450,7311,3845],{"class":3741},[3450,7313,7314],{"class":4675},"Collections",[3450,7316,3845],{"class":3741},[3450,7318,7319],{"class":4675},"Concurrent",[3450,7321,4684],{"class":3741},[3450,7323,7324,7326,7328,7330,7332],{"class":3452,"line":3459},[3450,7325,4398],{"class":4672},[3450,7327,4676],{"class":4675},[3450,7329,3845],{"class":3741},[3450,7331,4681],{"class":4675},[3450,7333,4684],{"class":3741},[3450,7335,7336,7338,7340,7342,7344],{"class":3452,"line":3465},[3450,7337,4398],{"class":4672},[3450,7339,4676],{"class":4675},[3450,7341,3845],{"class":3741},[3450,7343,4695],{"class":4675},[3450,7345,4684],{"class":3741},[3450,7347,7348,7350,7352,7354,7356,7358,7361],{"class":3452,"line":3471},[3450,7349,4398],{"class":4672},[3450,7351,4676],{"class":4675},[3450,7353,3845],{"class":3741},[3450,7355,4695],{"class":4675},[3450,7357,3845],{"class":3741},[3450,7359,7360],{"class":4675},"Json",[3450,7362,4684],{"class":3741},[3450,7364,7365,7367,7369,7371,7373,7375,7377,7379,7382],{"class":3452,"line":3478},[3450,7366,4398],{"class":4672},[3450,7368,4676],{"class":4675},[3450,7370,3845],{"class":3741},[3450,7372,4695],{"class":4675},[3450,7374,3845],{"class":3741},[3450,7376,7360],{"class":4675},[3450,7378,3845],{"class":3741},[3450,7380,7381],{"class":4675},"Serialization",[3450,7383,4684],{"class":3741},[3450,7385,7386,7388,7390,7392,7394,7396,7399],{"class":3452,"line":3484},[3450,7387,4398],{"class":4672},[3450,7389,4676],{"class":4675},[3450,7391,3845],{"class":3741},[3450,7393,4695],{"class":4675},[3450,7395,3845],{"class":3741},[3450,7397,7398],{"class":4675},"RegularExpressions",[3450,7400,4684],{"class":3741},[3450,7402,7403],{"class":3452,"line":3489},[3450,7404,3475],{"emptyLinePlaceholder":3474},[3450,7406,7407],{"class":3452,"line":3495},[3450,7408,7409],{"class":3735},"\u002F\u002F ── Модель даних ─────────────────────────────────────────────────────────────\n",[3450,7411,7412,7415,7418],{"class":3452,"line":3500},[3450,7413,7414],{"class":4711},"record",[3450,7416,7417],{"class":4675}," TodoItem",[3450,7419,5768],{"class":3741},[3450,7421,7422,7425,7428],{"class":3452,"line":3506},[3450,7423,7424],{"class":4711},"    int",[3450,7426,7427],{"class":4714}," Id",[3450,7429,4869],{"class":3741},[3450,7431,7432,7434,7437],{"class":3452,"line":3511},[3450,7433,5784],{"class":4711},[3450,7435,7436],{"class":4714}," Title",[3450,7438,4869],{"class":3741},[3450,7440,7441,7444,7447],{"class":3452,"line":3517},[3450,7442,7443],{"class":4711},"    bool",[3450,7445,7446],{"class":4714}," IsCompleted",[3450,7448,4869],{"class":3741},[3450,7450,7451,7454],{"class":3452,"line":3522},[3450,7452,7453],{"class":4675},"    DateTimeOffset",[3450,7455,7456],{"class":4714}," CreatedAt\n",[3450,7458,7459],{"class":3452,"line":3528},[3450,7460,4800],{"class":3741},[3450,7462,7463],{"class":3452,"line":3534},[3450,7464,3475],{"emptyLinePlaceholder":3474},[3450,7466,7467],{"class":3452,"line":3540},[3450,7468,7469],{"class":3735},"\u002F\u002F ── Сховище в пам'яті ────────────────────────────────────────────────────────\n",[3450,7471,7472],{"class":3452,"line":3545},[3450,7473,7474],{"class":3735},"\u002F\u002F ConcurrentDictionary забезпечує thread-safe доступ без явних lock-ів\n",[3450,7476,7477,7479,7482,7484,7486,7489,7491,7493,7495,7498],{"class":3452,"line":3551},[3450,7478,4814],{"class":4711},[3450,7480,7481],{"class":4714}," store",[3450,7483,4718],{"class":3741},[3450,7485,4822],{"class":4711},[3450,7487,7488],{"class":4675}," ConcurrentDictionary",[3450,7490,4828],{"class":3741},[3450,7492,4316],{"class":4711},[3450,7494,3858],{"class":3741},[3450,7496,7497],{"class":4675},"TodoItem",[3450,7499,7500],{"class":3741},">();\n",[3450,7502,7503,7505,7508,7510,7513,7515],{"class":3452,"line":3557},[3450,7504,4316],{"class":4711},[3450,7506,7507],{"class":4714}," nextId",[3450,7509,4718],{"class":3741},[3450,7511,7512],{"class":3745},"1",[3450,7514,5547],{"class":3741},[3450,7516,7517],{"class":3735},"\u002F\u002F Лічильник ID — інкрементується через Interlocked\n",[3450,7519,7520],{"class":3452,"line":3562},[3450,7521,3475],{"emptyLinePlaceholder":3474},[3450,7523,7524],{"class":3452,"line":3568},[3450,7525,7526],{"class":3735},"\u002F\u002F Заповнюємо початковими даними\n",[3450,7528,7529,7532,7534,7536,7539,7541,7543,7545,7548,7551,7554,7556,7558],{"class":3452,"line":3573},[3450,7530,7531],{"class":4672},"for",[3450,7533,5528],{"class":3741},[3450,7535,4316],{"class":4711},[3450,7537,7538],{"class":4714}," i",[3450,7540,4718],{"class":3741},[3450,7542,7512],{"class":3745},[3450,7544,5547],{"class":3741},[3450,7546,7547],{"class":4714},"i",[3450,7549,7550],{"class":3741}," \u003C= ",[3450,7552,7553],{"class":3745},"3",[3450,7555,5547],{"class":3741},[3450,7557,7547],{"class":4714},[3450,7559,7560],{"class":3741},"++)\n",[3450,7562,7563,7566,7568,7570,7572,7574,7576,7578,7580,7582,7585,7587,7589,7591,7593,7595,7597,7599,7601,7603,7606,7608,7611],{"class":3452,"line":3579},[3450,7564,7565],{"class":4714},"    store",[3450,7567,6598],{"class":3741},[3450,7569,7547],{"class":4714},[3450,7571,4864],{"class":3741},[3450,7573,4822],{"class":4711},[3450,7575,7417],{"class":4675},[3450,7577,4730],{"class":3741},[3450,7579,7547],{"class":4714},[3450,7581,3858],{"class":3741},[3450,7583,7584],{"class":4733},"$\"Завдання #",[3450,7586,5265],{"class":5264},[3450,7588,7547],{"class":4714},[3450,7590,5270],{"class":5264},[3450,7592,5273],{"class":4733},[3450,7594,3858],{"class":3741},[3450,7596,7547],{"class":4714},[3450,7598,5534],{"class":3741},[3450,7600,7512],{"class":3745},[3450,7602,3858],{"class":3741},[3450,7604,7605],{"class":4714},"DateTimeOffset",[3450,7607,3845],{"class":3741},[3450,7609,7610],{"class":4714},"UtcNow",[3450,7612,4800],{"class":3741},[3450,7614,7615,7618,7620,7623],{"class":3452,"line":3584},[3450,7616,7617],{"class":4714},"nextId",[3450,7619,4718],{"class":3741},[3450,7621,7622],{"class":3745},"4",[3450,7624,4684],{"class":3741},[3450,7626,7627],{"class":3452,"line":3590},[3450,7628,3475],{"emptyLinePlaceholder":3474},[3450,7630,7631],{"class":3452,"line":3596},[3450,7632,7633],{"class":3735},"\u002F\u002F ── Налаштування серіалізації ────────────────────────────────────────────────\n",[3450,7635,7636,7638,7641,7643,7645],{"class":3452,"line":3602},[3450,7637,4814],{"class":4711},[3450,7639,7640],{"class":4714}," jsonOptions",[3450,7642,4718],{"class":3741},[3450,7644,4822],{"class":4711},[3450,7646,7647],{"class":4675}," JsonSerializerOptions\n",[3450,7649,7650],{"class":3452,"line":3608},[3450,7651,4853],{"class":3741},[3450,7653,7654,7657,7659,7662,7664,7667],{"class":3452,"line":3614},[3450,7655,7656],{"class":4714},"    PropertyNamingPolicy",[3450,7658,4718],{"class":3741},[3450,7660,7661],{"class":4714},"JsonNamingPolicy",[3450,7663,3845],{"class":3741},[3450,7665,7666],{"class":4714},"CamelCase",[3450,7668,4869],{"class":3741},[3450,7670,7671,7674,7676,7678],{"class":3452,"line":3620},[3450,7672,7673],{"class":4714},"    WriteIndented",[3450,7675,4718],{"class":3741},[3450,7677,3829],{"class":4711},[3450,7679,4869],{"class":3741},[3450,7681,7682,7685,7687,7690,7692],{"class":3452,"line":3625},[3450,7683,7684],{"class":4714},"    DefaultIgnoreCondition",[3450,7686,4718],{"class":3741},[3450,7688,7689],{"class":4714},"JsonIgnoreCondition",[3450,7691,3845],{"class":3741},[3450,7693,7694],{"class":4714},"WhenWritingNull\n",[3450,7696,7697],{"class":3452,"line":5082},[3450,7698,5070],{"class":3741},[3450,7700,7701],{"class":3452,"line":5103},[3450,7702,3475],{"emptyLinePlaceholder":3474},[3450,7704,7705],{"class":3452,"line":5125},[3450,7706,7707],{"class":3735},"\u002F\u002F ── Маршрути: регулярні вирази для REST URL ──────────────────────────────────\n",[3450,7709,7710],{"class":3452,"line":5130},[3450,7711,7712],{"class":3735},"\u002F\u002F REST API маршрути:\n",[3450,7714,7715],{"class":3452,"line":5136},[3450,7716,7717],{"class":3735},"\u002F\u002F   GET    \u002Ftodos        → список усіх задач\n",[3450,7719,7720],{"class":3452,"line":5155},[3450,7721,7722],{"class":3735},"\u002F\u002F   POST   \u002Ftodos        → створення нової задачі\n",[3450,7724,7725],{"class":3452,"line":5160},[3450,7726,7727],{"class":3735},"\u002F\u002F   GET    \u002Ftodos\u002F{id}   → отримання конкретної задачі\n",[3450,7729,7730],{"class":3452,"line":5185},[3450,7731,7732],{"class":3735},"\u002F\u002F   PUT    \u002Ftodos\u002F{id}   → повне оновлення задачі\n",[3450,7734,7735],{"class":3452,"line":5190},[3450,7736,7737],{"class":3735},"\u002F\u002F   PATCH  \u002Ftodos\u002F{id}   → часткове оновлення (лише IsCompleted)\n",[3450,7739,7740],{"class":3452,"line":5211},[3450,7741,7742],{"class":3735},"\u002F\u002F   DELETE \u002Ftodos\u002F{id}   → видалення задачі\n",[3450,7744,7745,7747,7750,7753,7756,7758,7760,7762,7765],{"class":3452,"line":5227},[3450,7746,5757],{"class":4711},[3450,7748,7749],{"class":4711}," readonly",[3450,7751,7752],{"class":4675}," Regex",[3450,7754,7755],{"class":4714}," RouteCollection",[3450,7757,4718],{"class":3741},[3450,7759,4822],{"class":4711},[3450,7761,4730],{"class":3741},[3450,7763,7764],{"class":4733},"@\"^\u002Ftodos\u002F?$\"",[3450,7766,4869],{"class":3741},[3450,7768,7769,7772,7774,7777,7780,7783,7785,7788],{"class":3452,"line":5232},[3450,7770,7771],{"class":4714},"    RegexOptions",[3450,7773,3845],{"class":3741},[3450,7775,7776],{"class":4714},"Compiled",[3450,7778,7779],{"class":3741}," | ",[3450,7781,7782],{"class":4714},"RegexOptions",[3450,7784,3845],{"class":3741},[3450,7786,7787],{"class":4714},"IgnoreCase",[3450,7789,4800],{"class":3741},[3450,7791,7792,7794,7796,7798,7801,7803,7805,7807,7810],{"class":3452,"line":5237},[3450,7793,5757],{"class":4711},[3450,7795,7749],{"class":4711},[3450,7797,7752],{"class":4675},[3450,7799,7800],{"class":4714}," RouteItem",[3450,7802,4718],{"class":3741},[3450,7804,4822],{"class":4711},[3450,7806,4730],{"class":3741},[3450,7808,7809],{"class":4733},"@\"^\u002Ftodos\u002F(?\u003Cid>\\d+)\u002F?$\"",[3450,7811,4869],{"class":3741},[3450,7813,7814,7816,7818,7820,7822,7824,7826,7828],{"class":3452,"line":5249},[3450,7815,7771],{"class":4714},[3450,7817,3845],{"class":3741},[3450,7819,7776],{"class":4714},[3450,7821,7779],{"class":3741},[3450,7823,7782],{"class":4714},[3450,7825,3845],{"class":3741},[3450,7827,7787],{"class":4714},[3450,7829,4800],{"class":3741},[3450,7831,7832],{"class":3452,"line":5278},[3450,7833,3475],{"emptyLinePlaceholder":3474},[3450,7835,7836],{"class":3452,"line":5302},[3450,7837,7838],{"class":3735},"\u002F\u002F ── Ініціалізація HttpListener ───────────────────────────────────────────────\n",[3450,7840,7841,7843,7845,7847,7849,7851,7853],{"class":3452,"line":5318},[3450,7842,4398],{"class":4672},[3450,7844,5087],{"class":4711},[3450,7846,5090],{"class":4714},[3450,7848,4718],{"class":3741},[3450,7850,4822],{"class":4711},[3450,7852,5097],{"class":4675},[3450,7854,5100],{"class":3741},[3450,7856,7857,7859,7861,7863,7865,7867,7869,7872],{"class":3452,"line":5323},[3450,7858,5106],{"class":4714},[3450,7860,3845],{"class":3741},[3450,7862,3805],{"class":4714},[3450,7864,3845],{"class":3741},[3450,7866,5115],{"class":4726},[3450,7868,4730],{"class":3741},[3450,7870,7871],{"class":4733},"\"http:\u002F\u002Flocalhost:8081\u002F\"",[3450,7873,4800],{"class":3741},[3450,7875,7876],{"class":3452,"line":5329},[3450,7877,3475],{"emptyLinePlaceholder":3474},[3450,7879,7880],{"class":3452,"line":5335},[3450,7881,7882],{"class":3735},"\u002F\u002F Basic Auth: логін admin, пароль secret (для прикладу)\n",[3450,7884,7885,7887,7889,7891,7893,7895,7897,7899],{"class":3452,"line":5341},[3450,7886,5106],{"class":4714},[3450,7888,3845],{"class":3741},[3450,7890,3848],{"class":4714},[3450,7892,4718],{"class":3741},[3450,7894,3848],{"class":4714},[3450,7896,3845],{"class":3741},[3450,7898,3857],{"class":4714},[3450,7900,4684],{"class":3741},[3450,7902,7903,7905,7907,7909,7911,7914],{"class":3452,"line":5347},[3450,7904,5106],{"class":4714},[3450,7906,3845],{"class":3741},[3450,7908,3888],{"class":4714},[3450,7910,4718],{"class":3741},[3450,7912,7913],{"class":4733},"\"TodoAPI\"",[3450,7915,4684],{"class":3741},[3450,7917,7918],{"class":3452,"line":5352},[3450,7919,3475],{"emptyLinePlaceholder":3474},[3450,7921,7922,7924,7926,7928,7930,7932,7934],{"class":3452,"line":5375},[3450,7923,4398],{"class":4672},[3450,7925,5087],{"class":4711},[3450,7927,5143],{"class":4714},[3450,7929,4718],{"class":3741},[3450,7931,4822],{"class":4711},[3450,7933,5150],{"class":4675},[3450,7935,5100],{"class":3741},[3450,7937,7938,7940,7942,7944,7946,7948,7950,7952,7955,7957,7959,7961,7963,7965,7967,7969,7971,7973],{"class":3452,"line":5381},[3450,7939,5163],{"class":4714},[3450,7941,3845],{"class":3741},[3450,7943,5168],{"class":4714},[3450,7945,5171],{"class":3741},[3450,7947,5174],{"class":4714},[3450,7949,3858],{"class":3741},[3450,7951,5179],{"class":4714},[3450,7953,7954],{"class":3741},") => { ",[3450,7956,5179],{"class":4714},[3450,7958,3845],{"class":3741},[3450,7960,5198],{"class":4714},[3450,7962,4718],{"class":3741},[3450,7964,3829],{"class":4711},[3450,7966,5547],{"class":3741},[3450,7968,5360],{"class":4714},[3450,7970,3845],{"class":3741},[3450,7972,5198],{"class":4726},[3450,7974,7975],{"class":3741},"(); };\n",[3450,7977,7978],{"class":3452,"line":5392},[3450,7979,3475],{"emptyLinePlaceholder":3474},[3450,7981,7982,7984,7986,7988],{"class":3452,"line":5398},[3450,7983,5106],{"class":4714},[3450,7985,3845],{"class":3741},[3450,7987,5244],{"class":4726},[3450,7989,5100],{"class":3741},[3450,7991,7992,7994,7996,7998,8000,8003],{"class":3452,"line":5404},[3450,7993,5163],{"class":4714},[3450,7995,3845],{"class":3741},[3450,7997,5256],{"class":4726},[3450,7999,4730],{"class":3741},[3450,8001,8002],{"class":4733},"\"📋 Todo REST API запущено: http:\u002F\u002Flocalhost:8081\u002F\"",[3450,8004,4800],{"class":3741},[3450,8006,8007,8009,8011,8013,8015,8018],{"class":3452,"line":5410},[3450,8008,5163],{"class":4714},[3450,8010,3845],{"class":3741},[3450,8012,5256],{"class":4726},[3450,8014,4730],{"class":3741},[3450,8016,8017],{"class":4733},"\"   Аутентифікація: admin \u002F secret\"",[3450,8019,4800],{"class":3741},[3450,8021,8022,8024,8026,8028,8030,8033,8035,8037],{"class":3452,"line":5416},[3450,8023,5163],{"class":4714},[3450,8025,3845],{"class":3741},[3450,8027,5256],{"class":4726},[3450,8029,4730],{"class":3741},[3450,8031,8032],{"class":4733},"\"   Натисніть Ctrl+C для зупинки.",[3450,8034,5730],{"class":5729},[3450,8036,5273],{"class":4733},[3450,8038,4800],{"class":3741},[3450,8040,8041],{"class":3452,"line":5443},[3450,8042,3475],{"emptyLinePlaceholder":3474},[3450,8044,8045],{"class":3452,"line":5480},[3450,8046,8047],{"class":3735},"\u002F\u002F ── Основний цикл ────────────────────────────────────────────────────────────\n",[3450,8049,8050],{"class":3452,"line":5485},[3450,8051,5344],{"class":4672},[3450,8053,8054],{"class":3452,"line":5517},[3450,8055,4853],{"class":3741},[3450,8057,8058,8060,8062,8064,8066,8068,8070,8072],{"class":3452,"line":5522},[3450,8059,5355],{"class":4672},[3450,8061,4767],{"class":3741},[3450,8063,5360],{"class":4714},[3450,8065,3845],{"class":3741},[3450,8067,5365],{"class":4714},[3450,8069,3845],{"class":3741},[3450,8071,5370],{"class":4714},[3450,8073,4848],{"class":3741},[3450,8075,8076],{"class":3452,"line":5541},[3450,8077,5378],{"class":3741},[3450,8079,8080,8083,8085,8087,8089,8092,8094,8096,8098,8100],{"class":3452,"line":5553},[3450,8081,8082],{"class":4675},"        Task",[3450,8084,4828],{"class":3741},[3450,8086,3981],{"class":4675},[3450,8088,5426],{"class":3741},[3450,8090,8091],{"class":4714},"getCtx",[3450,8093,4718],{"class":3741},[3450,8095,5106],{"class":4714},[3450,8097,3845],{"class":3741},[3450,8099,5438],{"class":4726},[3450,8101,5100],{"class":3741},[3450,8103,8104,8106,8109,8111,8113,8115,8117,8119,8121,8123,8125,8127,8129,8131,8133],{"class":3452,"line":5558},[3450,8105,8082],{"class":4675},[3450,8107,8108],{"class":4714}," cancel",[3450,8110,4718],{"class":3741},[3450,8112,4010],{"class":4714},[3450,8114,3845],{"class":3741},[3450,8116,5457],{"class":4726},[3450,8118,4730],{"class":3741},[3450,8120,5462],{"class":4714},[3450,8122,3845],{"class":3741},[3450,8124,5467],{"class":4714},[3450,8126,3858],{"class":3741},[3450,8128,5360],{"class":4714},[3450,8130,3845],{"class":3741},[3450,8132,5365],{"class":4714},[3450,8134,4800],{"class":3741},[3450,8136,8137],{"class":3452,"line":5573},[3450,8138,3475],{"emptyLinePlaceholder":3474},[3450,8140,8141,8143,8145,8147,8149,8151,8153,8155,8157,8159,8162,8165,8167,8169,8172],{"class":3452,"line":5579},[3450,8142,5976],{"class":4672},[3450,8144,5528],{"class":3741},[3450,8146,5495],{"class":4711},[3450,8148,5498],{"class":4714},[3450,8150,3845],{"class":3741},[3450,8152,5503],{"class":4726},[3450,8154,4730],{"class":3741},[3450,8156,8091],{"class":4714},[3450,8158,3858],{"class":3741},[3450,8160,8161],{"class":4714},"cancel",[3450,8163,8164],{"class":3741},") == ",[3450,8166,8161],{"class":4714},[3450,8168,5589],{"class":3741},[3450,8170,8171],{"class":4672},"break",[3450,8173,4684],{"class":3741},[3450,8175,8176],{"class":3452,"line":5609},[3450,8177,3475],{"emptyLinePlaceholder":3474},[3450,8179,8180,8182,8185,8187,8189,8192],{"class":3452,"line":5614},[3450,8181,5384],{"class":4675},[3450,8183,8184],{"class":4714}," ctx",[3450,8186,4718],{"class":3741},[3450,8188,5495],{"class":4711},[3450,8190,8191],{"class":4714}," getCtx",[3450,8193,4684],{"class":3741},[3450,8195,8196,8198,8200,8202,8204,8206,8208,8211,8213,8216],{"class":3452,"line":5625},[3450,8197,5644],{"class":4714},[3450,8199,4718],{"class":3741},[3450,8201,4010],{"class":4714},[3450,8203,3845],{"class":3741},[3450,8205,5653],{"class":4726},[3450,8207,5656],{"class":3741},[3450,8209,8210],{"class":4726},"ProcessRequestAsync",[3450,8212,4730],{"class":3741},[3450,8214,8215],{"class":4714},"ctx",[3450,8217,5676],{"class":3741},[3450,8219,8220],{"class":3452,"line":5630},[3450,8221,5682],{"class":3741},[3450,8223,8224],{"class":3452,"line":5635},[3450,8225,5688],{"class":3741},[3450,8227,8228],{"class":3452,"line":5641},[3450,8229,5694],{"class":4672},[3450,8231,8232],{"class":3452,"line":5679},[3450,8233,4853],{"class":3741},[3450,8235,8236,8238,8240,8242],{"class":3452,"line":5685},[3450,8237,5705],{"class":4714},[3450,8239,3845],{"class":3741},[3450,8241,5710],{"class":4726},[3450,8243,5100],{"class":3741},[3450,8245,8246,8248,8250,8252,8254,8256,8258,8261],{"class":3452,"line":5691},[3450,8247,5718],{"class":4714},[3450,8249,3845],{"class":3741},[3450,8251,5256],{"class":4726},[3450,8253,4730],{"class":3741},[3450,8255,5273],{"class":4733},[3450,8257,5730],{"class":5729},[3450,8259,8260],{"class":4733},"🛑 API сервер зупинено.\"",[3450,8262,4800],{"class":3741},[3450,8264,8265],{"class":3452,"line":5697},[3450,8266,5688],{"class":3741},[3450,8268,8269],{"class":3452,"line":5702},[3450,8270,3475],{"emptyLinePlaceholder":3474},[3450,8272,8273],{"class":3452,"line":5715},[3450,8274,8275],{"class":3735},"\u002F\u002F ── Диспетчер запитів ────────────────────────────────────────────────────────\n",[3450,8277,8278,8281,8283,8286,8288,8290,8292],{"class":3452,"line":5738},[3450,8279,8280],{"class":4711},"async",[3450,8282,5498],{"class":4675},[3450,8284,8285],{"class":4726}," ProcessRequestAsync",[3450,8287,4730],{"class":3741},[3450,8289,3981],{"class":4675},[3450,8291,8184],{"class":4714},[3450,8293,4848],{"class":3741},[3450,8295,8296],{"class":3452,"line":5743},[3450,8297,4853],{"class":3741},[3450,8299,8300,8302,8304,8306,8308,8310,8312],{"class":3452,"line":5748},[3450,8301,5819],{"class":4675},[3450,8303,5822],{"class":4714},[3450,8305,4718],{"class":3741},[3450,8307,8215],{"class":4714},[3450,8309,3845],{"class":3741},[3450,8311,3991],{"class":4714},[3450,8313,4684],{"class":3741},[3450,8315,8316,8318,8320,8322,8324,8326,8328],{"class":3452,"line":5754},[3450,8317,5838],{"class":4675},[3450,8319,5841],{"class":4714},[3450,8321,4718],{"class":3741},[3450,8323,8215],{"class":4714},[3450,8325,3845],{"class":3741},[3450,8327,3994],{"class":4714},[3450,8329,4684],{"class":3741},[3450,8331,8332],{"class":3452,"line":5771},[3450,8333,3475],{"emptyLinePlaceholder":3474},[3450,8335,8336],{"class":3452,"line":5781},[3450,8337,8338],{"class":3735},"    \u002F\u002F ─ Логування ──────────────────────────────────────────────────────────\n",[3450,8340,8341,8343,8345,8347,8349,8352,8354,8356,8358,8360,8362,8364,8366,8368,8370,8372,8374,8376,8378,8380,8382,8384,8386,8388,8390,8392,8394,8396,8398],{"class":3452,"line":5791},[3450,8342,5718],{"class":4714},[3450,8344,3845],{"class":3741},[3450,8346,5256],{"class":4726},[3450,8348,4730],{"class":3741},[3450,8350,8351],{"class":4733},"$\"  [",[3450,8353,5265],{"class":5264},[3450,8355,5875],{"class":4714},[3450,8357,3845],{"class":5264},[3450,8359,5880],{"class":4714},[3450,8361,3723],{"class":3741},[3450,8363,5885],{"class":4714},[3450,8365,3723],{"class":3741},[3450,8367,5890],{"class":4714},[3450,8369,3723],{"class":3741},[3450,8371,5895],{"class":4714},[3450,8373,5270],{"class":5264},[3450,8375,5900],{"class":4733},[3450,8377,5265],{"class":5264},[3450,8379,5905],{"class":4714},[3450,8381,3845],{"class":5264},[3450,8383,4095],{"class":4714},[3450,8385,5270],{"class":5264},[3450,8387,5914],{"class":5264},[3450,8389,5905],{"class":4714},[3450,8391,3845],{"class":5264},[3450,8393,4130],{"class":4714},[3450,8395,5270],{"class":5264},[3450,8397,5273],{"class":4733},[3450,8399,4800],{"class":3741},[3450,8401,8402],{"class":3452,"line":5811},[3450,8403,3475],{"emptyLinePlaceholder":3474},[3450,8405,8406],{"class":3452,"line":5816},[3450,8407,5959],{"class":4672},[3450,8409,8410],{"class":3452,"line":5835},[3450,8411,5378],{"class":3741},[3450,8413,8414],{"class":3452,"line":5854},[3450,8415,8416],{"class":3735},"        \u002F\u002F ─ Аутентифікація ─────────────────────────────────────────────────\n",[3450,8418,8419],{"class":3452,"line":5859},[3450,8420,8421],{"class":3735},"        \u002F\u002F HttpListener перевіряє Basic Auth заголовок автоматично\n",[3450,8423,8424],{"class":3452,"line":5931},[3450,8425,8426],{"class":3735},"        \u002F\u002F і заповнює ctx.User при успіху\n",[3450,8428,8429,8431,8433,8435,8437,8439,8441,8443,8445,8447],{"class":3452,"line":5951},[3450,8430,5976],{"class":4672},[3450,8432,4767],{"class":3741},[3450,8434,8215],{"class":4714},[3450,8436,3845],{"class":3741},[3450,8438,4055],{"class":4714},[3450,8440,3845],{"class":3741},[3450,8442,4071],{"class":4714},[3450,8444,3845],{"class":3741},[3450,8446,4219],{"class":4714},[3450,8448,4848],{"class":3741},[3450,8450,8451],{"class":3452,"line":5956},[3450,8452,5401],{"class":3741},[3450,8454,8455,8457,8460,8462,8464,8466,8469],{"class":3452,"line":5962},[3450,8456,6015],{"class":4711},[3450,8458,8459],{"class":4726}," WriteJsonAsync",[3450,8461,4730],{"class":3741},[3450,8463,6023],{"class":4714},[3450,8465,3858],{"class":3741},[3450,8467,8468],{"class":3745},"401",[3450,8470,4869],{"class":3741},[3450,8472,8473,8476,8478,8481,8483,8486,8488,8490,8492,8495],{"class":3452,"line":5967},[3450,8474,8475],{"class":4711},"                new",[3450,8477,6832],{"class":3741},[3450,8479,8480],{"class":4714},"error",[3450,8482,4718],{"class":3741},[3450,8484,8485],{"class":4733},"\"Unauthorized\"",[3450,8487,3858],{"class":3741},[3450,8489,7086],{"class":4714},[3450,8491,4718],{"class":3741},[3450,8493,8494],{"class":4733},"\"Потрібна Basic аутентифікація\"",[3450,8496,8497],{"class":3741}," });\n",[3450,8499,8500,8502],{"class":3452,"line":5973},[3450,8501,6067],{"class":4672},[3450,8503,4684],{"class":3741},[3450,8505,8506],{"class":3452,"line":6007},[3450,8507,5576],{"class":3741},[3450,8509,8510],{"class":3452,"line":6012},[3450,8511,3475],{"emptyLinePlaceholder":3474},[3450,8513,8514],{"class":3452,"line":6038},[3450,8515,8516],{"class":3735},"        \u002F\u002F Перевіряємо облікові дані (у реальності — із БД або Identity)\n",[3450,8518,8519,8522,8525,8528,8531,8533,8535,8537,8539,8541,8543],{"class":3452,"line":6064},[3450,8520,8521],{"class":4711},"        var",[3450,8523,8524],{"class":4714}," identity",[3450,8526,8527],{"class":3741}," = (",[3450,8529,8530],{"class":4675},"HttpListenerBasicIdentity",[3450,8532,3676],{"class":3741},[3450,8534,8215],{"class":4714},[3450,8536,3845],{"class":3741},[3450,8538,4055],{"class":4714},[3450,8540,3845],{"class":3741},[3450,8542,4071],{"class":4714},[3450,8544,4684],{"class":3741},[3450,8546,8547,8549,8551,8554,8556,8559,8561,8564],{"class":3452,"line":6072},[3450,8548,5976],{"class":4672},[3450,8550,5528],{"class":3741},[3450,8552,8553],{"class":4714},"identity",[3450,8555,3845],{"class":3741},[3450,8557,8558],{"class":4714},"Password",[3450,8560,5987],{"class":3741},[3450,8562,8563],{"class":4733},"\"secret\"",[3450,8565,4848],{"class":3741},[3450,8567,8568],{"class":3452,"line":6077},[3450,8569,5401],{"class":3741},[3450,8571,8572,8574,8576,8578,8580,8582,8584],{"class":3452,"line":6082},[3450,8573,6015],{"class":4711},[3450,8575,8459],{"class":4726},[3450,8577,4730],{"class":3741},[3450,8579,6023],{"class":4714},[3450,8581,3858],{"class":3741},[3450,8583,6234],{"class":3745},[3450,8585,4869],{"class":3741},[3450,8587,8588,8590,8592,8594,8596,8598,8600,8602,8604,8607],{"class":3452,"line":6088},[3450,8589,8475],{"class":4711},[3450,8591,6832],{"class":3741},[3450,8593,8480],{"class":4714},[3450,8595,4718],{"class":3741},[3450,8597,6239],{"class":4733},[3450,8599,3858],{"class":3741},[3450,8601,7086],{"class":4714},[3450,8603,4718],{"class":3741},[3450,8605,8606],{"class":4733},"\"Невірний пароль\"",[3450,8608,8497],{"class":3741},[3450,8610,8611,8613],{"class":3452,"line":6094},[3450,8612,6067],{"class":4672},[3450,8614,4684],{"class":3741},[3450,8616,8617],{"class":3452,"line":6119},[3450,8618,5576],{"class":3741},[3450,8620,8621],{"class":3452,"line":6124},[3450,8622,3475],{"emptyLinePlaceholder":3474},[3450,8624,8625],{"class":3452,"line":6130},[3450,8626,8627],{"class":3735},"        \u002F\u002F ─ CORS Headers ───────────────────────────────────────────────────\n",[3450,8629,8630,8632,8634,8636,8638,8640,8642,8645,8647,8650],{"class":3452,"line":6148},[3450,8631,6528],{"class":4714},[3450,8633,3845],{"class":3741},[3450,8635,4151],{"class":4714},[3450,8637,3845],{"class":3741},[3450,8639,5115],{"class":4726},[3450,8641,4730],{"class":3741},[3450,8643,8644],{"class":4733},"\"Access-Control-Allow-Origin\"",[3450,8646,3858],{"class":3741},[3450,8648,8649],{"class":4733},"\"*\"",[3450,8651,4800],{"class":3741},[3450,8653,8654,8656,8658,8660,8662,8664,8666,8669,8671,8674],{"class":3452,"line":6181},[3450,8655,6528],{"class":4714},[3450,8657,3845],{"class":3741},[3450,8659,4151],{"class":4714},[3450,8661,3845],{"class":3741},[3450,8663,5115],{"class":4726},[3450,8665,4730],{"class":3741},[3450,8667,8668],{"class":4733},"\"Access-Control-Allow-Methods\"",[3450,8670,3858],{"class":3741},[3450,8672,8673],{"class":4733},"\"GET, POST, PUT, PATCH, DELETE\"",[3450,8675,4800],{"class":3741},[3450,8677,8678,8680,8682,8684,8686,8688,8690,8693,8695,8698],{"class":3452,"line":6186},[3450,8679,6528],{"class":4714},[3450,8681,3845],{"class":3741},[3450,8683,4151],{"class":4714},[3450,8685,3845],{"class":3741},[3450,8687,5115],{"class":4726},[3450,8689,4730],{"class":3741},[3450,8691,8692],{"class":4733},"\"Access-Control-Allow-Headers\"",[3450,8694,3858],{"class":3741},[3450,8696,8697],{"class":4733},"\"Content-Type, Authorization\"",[3450,8699,4800],{"class":3741},[3450,8701,8702],{"class":3452,"line":6216},[3450,8703,3475],{"emptyLinePlaceholder":3474},[3450,8705,8706],{"class":3452,"line":6221},[3450,8707,8708],{"class":3735},"        \u002F\u002F OPTIONS preflight (браузерний CORS)\n",[3450,8710,8711,8713,8715,8717,8719,8721,8723,8725],{"class":3452,"line":6244},[3450,8712,5976],{"class":4672},[3450,8714,5528],{"class":3741},[3450,8716,5905],{"class":4714},[3450,8718,3845],{"class":3741},[3450,8720,4095],{"class":4714},[3450,8722,5534],{"class":3741},[3450,8724,4117],{"class":4733},[3450,8726,4848],{"class":3741},[3450,8728,8729],{"class":3452,"line":6251},[3450,8730,5401],{"class":3741},[3450,8732,8733,8735,8737,8739,8741,8744],{"class":3452,"line":6256},[3450,8734,6041],{"class":4714},[3450,8736,3845],{"class":3741},[3450,8738,4315],{"class":4714},[3450,8740,4718],{"class":3741},[3450,8742,8743],{"class":3745},"204",[3450,8745,4684],{"class":3741},[3450,8747,8748,8750,8752,8754],{"class":3452,"line":6261},[3450,8749,6041],{"class":4714},[3450,8751,3845],{"class":3741},[3450,8753,6702],{"class":4726},[3450,8755,5100],{"class":3741},[3450,8757,8758,8760],{"class":3452,"line":6267},[3450,8759,6067],{"class":4672},[3450,8761,4684],{"class":3741},[3450,8763,8764],{"class":3452,"line":6286},[3450,8765,5576],{"class":3741},[3450,8767,8768],{"class":3452,"line":6311},[3450,8769,3475],{"emptyLinePlaceholder":3474},[3450,8771,8772],{"class":3452,"line":6316},[3450,8773,8774],{"class":3735},"        \u002F\u002F ─ Маршрутизація ──────────────────────────────────────────────────\n",[3450,8776,8777,8779,8782,8784,8786,8788,8790,8792,8794],{"class":3452,"line":6336},[3450,8778,6097],{"class":4711},[3450,8780,8781],{"class":4714}," path",[3450,8783,4718],{"class":3741},[3450,8785,5905],{"class":4714},[3450,8787,3845],{"class":3741},[3450,8789,4121],{"class":4714},[3450,8791,6111],{"class":3741},[3450,8793,6114],{"class":4714},[3450,8795,4684],{"class":3741},[3450,8797,8798],{"class":3452,"line":6341},[3450,8799,3475],{"emptyLinePlaceholder":3474},[3450,8801,8802,8804,8806,8809,8811,8814,8816,8819],{"class":3452,"line":6359},[3450,8803,5976],{"class":4672},[3450,8805,5528],{"class":3741},[3450,8807,8808],{"class":4714},"RouteCollection",[3450,8810,3845],{"class":3741},[3450,8812,8813],{"class":4726},"IsMatch",[3450,8815,4730],{"class":3741},[3450,8817,8818],{"class":4714},"path",[3450,8820,4783],{"class":3741},[3450,8822,8823],{"class":3452,"line":6375},[3450,8824,5401],{"class":3741},[3450,8826,8827,8829,8832,8834,8836,8838,8840],{"class":3452,"line":6382},[3450,8828,6015],{"class":4711},[3450,8830,8831],{"class":4726}," HandleCollectionAsync",[3450,8833,4730],{"class":3741},[3450,8835,5905],{"class":4714},[3450,8837,3858],{"class":3741},[3450,8839,6023],{"class":4714},[3450,8841,4800],{"class":3741},[3450,8843,8844],{"class":3452,"line":6387},[3450,8845,5576],{"class":3741},[3450,8847,8848,8851,8854,8856,8859,8861,8864,8866,8868,8870,8873,8875,8878,8880,8882,8885,8888],{"class":3452,"line":6392},[3450,8849,8850],{"class":4672},"        else",[3450,8852,8853],{"class":4672}," if",[3450,8855,5528],{"class":3741},[3450,8857,8858],{"class":4714},"RouteItem",[3450,8860,3845],{"class":3741},[3450,8862,8863],{"class":4726},"Match",[3450,8865,4730],{"class":3741},[3450,8867,8818],{"class":4714},[3450,8869,5589],{"class":3741},[3450,8871,8872],{"class":4711},"is",[3450,8874,6832],{"class":3741},[3450,8876,8877],{"class":4714},"Success",[3450,8879,4193],{"class":3741},[3450,8881,3829],{"class":4711},[3450,8883,8884],{"class":3741}," } ",[3450,8886,8887],{"class":4714},"match",[3450,8889,4848],{"class":3741},[3450,8891,8892],{"class":3452,"line":6398},[3450,8893,5401],{"class":3741},[3450,8895,8896,8899,8902,8904,8906,8908,8911,8913,8915,8917,8920,8922,8925,8928,8931],{"class":3452,"line":6421},[3450,8897,8898],{"class":4711},"            int",[3450,8900,8901],{"class":4714}," id",[3450,8903,4718],{"class":3741},[3450,8905,4316],{"class":4711},[3450,8907,3845],{"class":3741},[3450,8909,8910],{"class":4726},"Parse",[3450,8912,4730],{"class":3741},[3450,8914,8887],{"class":4714},[3450,8916,3845],{"class":3741},[3450,8918,8919],{"class":4714},"Groups",[3450,8921,6598],{"class":3741},[3450,8923,8924],{"class":4733},"\"id\"",[3450,8926,8927],{"class":3741},"].",[3450,8929,8930],{"class":4714},"Value",[3450,8932,4800],{"class":3741},[3450,8934,8935,8937,8940,8942,8944,8946,8948,8950,8953],{"class":3452,"line":6459},[3450,8936,6015],{"class":4711},[3450,8938,8939],{"class":4726}," HandleItemAsync",[3450,8941,4730],{"class":3741},[3450,8943,5905],{"class":4714},[3450,8945,3858],{"class":3741},[3450,8947,6023],{"class":4714},[3450,8949,3858],{"class":3741},[3450,8951,8952],{"class":4714},"id",[3450,8954,4800],{"class":3741},[3450,8956,8957],{"class":3452,"line":6468},[3450,8958,5576],{"class":3741},[3450,8960,8961],{"class":3452,"line":6479},[3450,8962,8963],{"class":4672},"        else\n",[3450,8965,8966],{"class":3452,"line":6484},[3450,8967,5401],{"class":3741},[3450,8969,8970,8972,8974,8976,8978,8980,8982],{"class":3452,"line":6490},[3450,8971,6015],{"class":4711},[3450,8973,8459],{"class":4726},[3450,8975,4730],{"class":3741},[3450,8977,6023],{"class":4714},[3450,8979,3858],{"class":3741},[3450,8981,6354],{"class":3745},[3450,8983,4869],{"class":3741},[3450,8985,8986,8988,8990,8992,8994,8996,8998,9000],{"class":3452,"line":6520},[3450,8987,8475],{"class":4711},[3450,8989,6832],{"class":3741},[3450,8991,8480],{"class":4714},[3450,8993,4718],{"class":3741},[3450,8995,4337],{"class":4733},[3450,8997,3858],{"class":3741},[3450,8999,8818],{"class":4714},[3450,9001,8497],{"class":3741},[3450,9003,9004],{"class":3452,"line":6525},[3450,9005,5576],{"class":3741},[3450,9007,9008],{"class":3452,"line":6541},[3450,9009,5682],{"class":3741},[3450,9011,9012,9014,9016,9018,9020],{"class":3452,"line":6557},[3450,9013,6760],{"class":4672},[3450,9015,5528],{"class":3741},[3450,9017,6765],{"class":4675},[3450,9019,6768],{"class":4714},[3450,9021,4848],{"class":3741},[3450,9023,9024],{"class":3452,"line":6578},[3450,9025,5378],{"class":3741},[3450,9027,9028,9030,9032,9034,9036,9038,9040,9043,9045,9047,9049,9051,9053,9055],{"class":3452,"line":6583},[3450,9029,6781],{"class":4714},[3450,9031,3845],{"class":3741},[3450,9033,6786],{"class":4714},[3450,9035,3845],{"class":3741},[3450,9037,5256],{"class":4726},[3450,9039,4730],{"class":3741},[3450,9041,9042],{"class":4733},"$\"  ❌ Помилка: ",[3450,9044,5265],{"class":5264},[3450,9046,6812],{"class":4714},[3450,9048,3845],{"class":5264},[3450,9050,6817],{"class":4714},[3450,9052,5270],{"class":5264},[3450,9054,5273],{"class":4733},[3450,9056,4800],{"class":3741},[3450,9058,9059],{"class":3452,"line":6589},[3450,9060,5395],{"class":4672},[3450,9062,9063],{"class":3452,"line":6616},[3450,9064,5401],{"class":3741},[3450,9066,9067,9069,9071,9073,9075,9077,9079],{"class":3452,"line":6633},[3450,9068,6015],{"class":4711},[3450,9070,8459],{"class":4726},[3450,9072,4730],{"class":3741},[3450,9074,6023],{"class":4714},[3450,9076,3858],{"class":3741},[3450,9078,6845],{"class":3745},[3450,9080,4869],{"class":3741},[3450,9082,9083,9085,9087,9089,9091,9093,9095,9097,9099,9101,9103,9105],{"class":3452,"line":6660},[3450,9084,8475],{"class":4711},[3450,9086,6832],{"class":3741},[3450,9088,8480],{"class":4714},[3450,9090,4718],{"class":3741},[3450,9092,6850],{"class":4733},[3450,9094,3858],{"class":3741},[3450,9096,7086],{"class":4714},[3450,9098,4718],{"class":3741},[3450,9100,6812],{"class":4714},[3450,9102,3845],{"class":3741},[3450,9104,6817],{"class":4714},[3450,9106,8497],{"class":3741},[3450,9108,9109],{"class":3452,"line":6665},[3450,9110,5576],{"class":3741},[3450,9112,9113,9115],{"class":3452,"line":6671},[3450,9114,5582],{"class":4672},[3450,9116,6914],{"class":3741},[3450,9118,9119],{"class":3452,"line":6690},[3450,9120,5682],{"class":3741},[3450,9122,9123],{"class":3452,"line":6695},[3450,9124,6877],{"class":4672},[3450,9126,9127],{"class":3452,"line":6707},[3450,9128,5378],{"class":3741},[3450,9130,9131,9133,9135,9137,9139,9141,9143,9145,9147,9149],{"class":3452,"line":6714},[3450,9132,6829],{"class":4672},[3450,9134,6832],{"class":3741},[3450,9136,6023],{"class":4714},[3450,9138,3845],{"class":3741},[3450,9140,4324],{"class":4714},[3450,9142,3845],{"class":3741},[3450,9144,6702],{"class":4726},[3450,9146,6908],{"class":3741},[3450,9148,6911],{"class":4672},[3450,9150,6914],{"class":3741},[3450,9152,9153],{"class":3452,"line":6719},[3450,9154,5682],{"class":3741},[3450,9156,9157],{"class":3452,"line":6724},[3450,9158,5688],{"class":3741},[3450,9160,9161],{"class":3452,"line":6730},[3450,9162,3475],{"emptyLinePlaceholder":3474},[3450,9164,9165],{"class":3452,"line":6752},[3450,9166,9167],{"class":3735},"\u002F\u002F ── GET \u002Ftodos, POST \u002Ftodos ───────────────────────────────────────────────────\n",[3450,9169,9170,9172,9174,9176,9178,9180,9183,9185,9187,9189],{"class":3452,"line":6757},[3450,9171,8280],{"class":4711},[3450,9173,5498],{"class":4675},[3450,9175,8831],{"class":4726},[3450,9177,4730],{"class":3741},[3450,9179,3359],{"class":4675},[3450,9181,9182],{"class":4714}," req",[3450,9184,3858],{"class":3741},[3450,9186,3363],{"class":4675},[3450,9188,5841],{"class":4714},[3450,9190,4848],{"class":3741},[3450,9192,9193],{"class":3452,"line":6773},[3450,9194,4853],{"class":3741},[3450,9196,9197,9200,9202,9204,9206,9208],{"class":3452,"line":6778},[3450,9198,9199],{"class":4672},"    switch",[3450,9201,5528],{"class":3741},[3450,9203,5905],{"class":4714},[3450,9205,3845],{"class":3741},[3450,9207,4095],{"class":4714},[3450,9209,4848],{"class":3741},[3450,9211,9212],{"class":3452,"line":6826},[3450,9213,5378],{"class":3741},[3450,9215,9216,9219,9222],{"class":3452,"line":6856},[3450,9217,9218],{"class":4672},"        case",[3450,9220,9221],{"class":4733}," \"GET\"",[3450,9223,9224],{"class":3741},":\n",[3450,9226,9227],{"class":3452,"line":6869},[3450,9228,5401],{"class":3741},[3450,9230,9231],{"class":3452,"line":6874},[3450,9232,9233],{"class":3735},"            \u002F\u002F Фільтрація: ?completed=true|false\n",[3450,9235,9236,9239,9241,9244,9246,9248,9250,9252,9254,9257],{"class":3452,"line":6880},[3450,9237,9238],{"class":4711},"            string",[3450,9240,6451],{"class":3741},[3450,9242,9243],{"class":4714},"completedFilter",[3450,9245,4718],{"class":3741},[3450,9247,5905],{"class":4714},[3450,9249,3845],{"class":3741},[3450,9251,4138],{"class":4714},[3450,9253,6598],{"class":3741},[3450,9255,9256],{"class":4733},"\"completed\"",[3450,9258,9259],{"class":3741},"];\n",[3450,9261,9262,9265,9267,9269,9271,9274,9276,9279,9281,9284,9286,9289,9291,9294,9297,9299,9301,9304],{"class":3452,"line":6885},[3450,9263,9264],{"class":4675},"            IEnumerable",[3450,9266,4828],{"class":3741},[3450,9268,7497],{"class":4675},[3450,9270,5426],{"class":3741},[3450,9272,9273],{"class":4714},"items",[3450,9275,4718],{"class":3741},[3450,9277,9278],{"class":4714},"store",[3450,9280,3845],{"class":3741},[3450,9282,9283],{"class":4714},"Values",[3450,9285,3845],{"class":3741},[3450,9287,9288],{"class":4726},"OrderBy",[3450,9290,4730],{"class":3741},[3450,9292,9293],{"class":4714},"t",[3450,9295,9296],{"class":3741}," => ",[3450,9298,9293],{"class":4714},[3450,9300,3845],{"class":3741},[3450,9302,9303],{"class":4714},"Id",[3450,9305,4800],{"class":3741},[3450,9307,9308],{"class":3452,"line":6891},[3450,9309,3475],{"emptyLinePlaceholder":3474},[3450,9311,9312,9314,9316,9318,9321,9324,9327],{"class":3452,"line":6917},[3450,9313,5525],{"class":4672},[3450,9315,5528],{"class":3741},[3450,9317,9243],{"class":4714},[3450,9319,9320],{"class":4711}," is",[3450,9322,9323],{"class":4711}," not",[3450,9325,9326],{"class":4711}," null",[3450,9328,9329],{"class":3741}," &&\n",[3450,9331,9332,9335,9337,9340,9342,9344,9346,9348,9351,9354],{"class":3452,"line":6922},[3450,9333,9334],{"class":4711},"                bool",[3450,9336,3845],{"class":3741},[3450,9338,9339],{"class":4726},"TryParse",[3450,9341,4730],{"class":3741},[3450,9343,9243],{"class":4714},[3450,9345,3858],{"class":3741},[3450,9347,6445],{"class":4711},[3450,9349,9350],{"class":4711}," bool",[3450,9352,9353],{"class":4714}," filterValue",[3450,9355,4783],{"class":3741},[3450,9357,9358,9361,9363,9365,9367,9370,9372,9374,9376,9378,9380,9383,9385,9388],{"class":3452,"line":6927},[3450,9359,9360],{"class":4714},"                items",[3450,9362,4718],{"class":3741},[3450,9364,9273],{"class":4714},[3450,9366,3845],{"class":3741},[3450,9368,9369],{"class":4726},"Where",[3450,9371,4730],{"class":3741},[3450,9373,9293],{"class":4714},[3450,9375,9296],{"class":3741},[3450,9377,9293],{"class":4714},[3450,9379,3845],{"class":3741},[3450,9381,9382],{"class":4714},"IsCompleted",[3450,9384,5534],{"class":3741},[3450,9386,9387],{"class":4714},"filterValue",[3450,9389,4800],{"class":3741},[3450,9391,9392],{"class":3452,"line":6932},[3450,9393,3475],{"emptyLinePlaceholder":3474},[3450,9395,9396,9398,9400,9402,9404,9406,9408,9410,9412,9414,9417],{"class":3452,"line":6938},[3450,9397,6015],{"class":4711},[3450,9399,8459],{"class":4726},[3450,9401,4730],{"class":3741},[3450,9403,6023],{"class":4714},[3450,9405,3858],{"class":3741},[3450,9407,4320],{"class":3745},[3450,9409,3858],{"class":3741},[3450,9411,9273],{"class":4714},[3450,9413,3845],{"class":3741},[3450,9415,9416],{"class":4726},"ToArray",[3450,9418,9419],{"class":3741},"());\n",[3450,9421,9422,9424],{"class":3452,"line":6951},[3450,9423,5617],{"class":4672},[3450,9425,4684],{"class":3741},[3450,9427,9428],{"class":3452,"line":6974},[3450,9429,5576],{"class":3741},[3450,9431,9432],{"class":3452,"line":6979},[3450,9433,3475],{"emptyLinePlaceholder":3474},[3450,9435,9436,9438,9441],{"class":3452,"line":7010},[3450,9437,9218],{"class":4672},[3450,9439,9440],{"class":4733}," \"POST\"",[3450,9442,9224],{"class":3741},[3450,9444,9445],{"class":3452,"line":7016},[3450,9446,5401],{"class":3741},[3450,9448,9449,9451,9453,9455,9457,9459],{"class":3452,"line":7022},[3450,9450,5525],{"class":4672},[3450,9452,4767],{"class":3741},[3450,9454,5905],{"class":4714},[3450,9456,3845],{"class":3741},[3450,9458,4290],{"class":4714},[3450,9460,4848],{"class":3741},[3450,9462,9463],{"class":3452,"line":7038},[3450,9464,9465],{"class":3741},"            {\n",[3450,9467,9468,9471,9473,9475,9477,9479,9482],{"class":3452,"line":7044},[3450,9469,9470],{"class":4711},"                await",[3450,9472,8459],{"class":4726},[3450,9474,4730],{"class":3741},[3450,9476,6023],{"class":4714},[3450,9478,3858],{"class":3741},[3450,9480,9481],{"class":3745},"400",[3450,9483,4869],{"class":3741},[3450,9485,9486,9489,9491,9493,9495,9498,9500,9502,9504,9507],{"class":3452,"line":7059},[3450,9487,9488],{"class":4711},"                    new",[3450,9490,6832],{"class":3741},[3450,9492,8480],{"class":4714},[3450,9494,4718],{"class":3741},[3450,9496,9497],{"class":4733},"\"Bad Request\"",[3450,9499,3858],{"class":3741},[3450,9501,7086],{"class":4714},[3450,9503,4718],{"class":3741},[3450,9505,9506],{"class":4733},"\"Тіло запиту обов'язкове\"",[3450,9508,8497],{"class":3741},[3450,9510,9511,9514],{"class":3452,"line":7095},[3450,9512,9513],{"class":4672},"                return",[3450,9515,4684],{"class":3741},[3450,9517,9518],{"class":3452,"line":7101},[3450,9519,9520],{"class":3741},"            }\n",[3450,9522,9523],{"class":3452,"line":7109},[3450,9524,3475],{"emptyLinePlaceholder":3474},[3450,9526,9527],{"class":3452,"line":7114},[3450,9528,9529],{"class":3735},"            \u002F\u002F Читаємо та десеріалізуємо JSON-тіло\n",[3450,9531,9532,9535,9537,9540,9542,9544,9547,9549,9551,9553,9555],{"class":3452,"line":7131},[3450,9533,9534],{"class":4672},"            using",[3450,9536,5087],{"class":4711},[3450,9538,9539],{"class":4714}," reader",[3450,9541,4718],{"class":3741},[3450,9543,4822],{"class":4711},[3450,9545,9546],{"class":4675}," StreamReader",[3450,9548,4730],{"class":3741},[3450,9550,5905],{"class":4714},[3450,9552,3845],{"class":3741},[3450,9554,4162],{"class":4714},[3450,9556,4869],{"class":3741},[3450,9558,9559,9562,9564,9566,9569,9571,9573,9575],{"class":3452,"line":7146},[3450,9560,9561],{"class":4714},"                req",[3450,9563,3845],{"class":3741},[3450,9565,4367],{"class":4714},[3450,9567,9568],{"class":3741}," ?? ",[3450,9570,4368],{"class":4714},[3450,9572,3845],{"class":3741},[3450,9574,6997],{"class":4714},[3450,9576,4800],{"class":3741},[3450,9578,9579,9581,9584,9586,9588,9590,9592,9595],{"class":3452,"line":7165},[3450,9580,9238],{"class":4711},[3450,9582,9583],{"class":4714}," body",[3450,9585,4718],{"class":3741},[3450,9587,5495],{"class":4711},[3450,9589,9539],{"class":4714},[3450,9591,3845],{"class":3741},[3450,9593,9594],{"class":4726},"ReadToEndAsync",[3450,9596,5100],{"class":3741},[3450,9598,9599],{"class":3452,"line":7170},[3450,9600,3475],{"emptyLinePlaceholder":3474},[3450,9602,9603,9605,9607,9610,9612,9615,9617,9619,9621,9623],{"class":3452,"line":7193},[3450,9604,9534],{"class":4672},[3450,9606,5087],{"class":4711},[3450,9608,9609],{"class":4714}," doc",[3450,9611,4718],{"class":3741},[3450,9613,9614],{"class":4714},"JsonDocument",[3450,9616,3845],{"class":3741},[3450,9618,8910],{"class":4726},[3450,9620,4730],{"class":3741},[3450,9622,6988],{"class":4714},[3450,9624,4800],{"class":3741},[3450,9626,9627,9629,9631,9634,9636,9639,9641],{"class":3452,"line":7208},[3450,9628,9238],{"class":4711},[3450,9630,6451],{"class":3741},[3450,9632,9633],{"class":4714},"title",[3450,9635,4718],{"class":3741},[3450,9637,9638],{"class":4714},"doc",[3450,9640,3845],{"class":3741},[3450,9642,9643],{"class":4714},"RootElement\n",[3450,9645,9647,9650,9653,9655,9658,9660,9663],{"class":3452,"line":9646},195,[3450,9648,9649],{"class":3741},"                .",[3450,9651,9652],{"class":4726},"GetProperty",[3450,9654,4730],{"class":3741},[3450,9656,9657],{"class":4733},"\"title\"",[3450,9659,4226],{"class":3741},[3450,9661,9662],{"class":4726},"GetString",[3450,9664,5100],{"class":3741},[3450,9666,9668],{"class":3452,"line":9667},196,[3450,9669,3475],{"emptyLinePlaceholder":3474},[3450,9671,9673,9675,9677,9679,9681,9684,9686,9688],{"class":3452,"line":9672},197,[3450,9674,5525],{"class":4672},[3450,9676,5528],{"class":3741},[3450,9678,3889],{"class":4711},[3450,9680,3845],{"class":3741},[3450,9682,9683],{"class":4726},"IsNullOrWhiteSpace",[3450,9685,4730],{"class":3741},[3450,9687,9633],{"class":4714},[3450,9689,4783],{"class":3741},[3450,9691,9693],{"class":3452,"line":9692},198,[3450,9694,9465],{"class":3741},[3450,9696,9698,9700,9702,9704,9706,9708,9711],{"class":3452,"line":9697},199,[3450,9699,9470],{"class":4711},[3450,9701,8459],{"class":4726},[3450,9703,4730],{"class":3741},[3450,9705,6023],{"class":4714},[3450,9707,3858],{"class":3741},[3450,9709,9710],{"class":3745},"422",[3450,9712,4869],{"class":3741},[3450,9714,9716,9718,9720,9722,9724,9727],{"class":3452,"line":9715},200,[3450,9717,9488],{"class":4711},[3450,9719,6832],{"class":3741},[3450,9721,8480],{"class":4714},[3450,9723,4718],{"class":3741},[3450,9725,9726],{"class":4733},"\"Unprocessable Entity\"",[3450,9728,4869],{"class":3741},[3450,9730,9732,9735,9737,9740],{"class":3452,"line":9731},201,[3450,9733,9734],{"class":4714},"                          message",[3450,9736,4718],{"class":3741},[3450,9738,9739],{"class":4733},"\"Поле 'title' обов'язкове та не може бути порожнім\"",[3450,9741,8497],{"class":3741},[3450,9743,9745,9747],{"class":3452,"line":9744},202,[3450,9746,9513],{"class":4672},[3450,9748,4684],{"class":3741},[3450,9750,9752],{"class":3452,"line":9751},203,[3450,9753,9520],{"class":3741},[3450,9755,9757],{"class":3452,"line":9756},204,[3450,9758,3475],{"emptyLinePlaceholder":3474},[3450,9760,9762],{"class":3452,"line":9761},205,[3450,9763,9764],{"class":3735},"            \u002F\u002F Атомарна операція збільшення лічильника ID\n",[3450,9766,9768,9770,9772,9774,9777,9779,9782,9784,9787,9789],{"class":3452,"line":9767},206,[3450,9769,8898],{"class":4711},[3450,9771,8901],{"class":4714},[3450,9773,4718],{"class":3741},[3450,9775,9776],{"class":4714},"Interlocked",[3450,9778,3845],{"class":3741},[3450,9780,9781],{"class":4726},"Increment",[3450,9783,4730],{"class":3741},[3450,9785,9786],{"class":4711},"ref",[3450,9788,7507],{"class":4714},[3450,9790,4800],{"class":3741},[3450,9792,9794,9797,9800,9802,9804,9806,9808,9810,9812,9814,9816,9819,9822,9825,9827,9829,9831,9833],{"class":3452,"line":9793},207,[3450,9795,9796],{"class":4711},"            var",[3450,9798,9799],{"class":4714}," item",[3450,9801,4718],{"class":3741},[3450,9803,4822],{"class":4711},[3450,9805,7417],{"class":4675},[3450,9807,4730],{"class":3741},[3450,9809,8952],{"class":4714},[3450,9811,3858],{"class":3741},[3450,9813,9633],{"class":4714},[3450,9815,3845],{"class":3741},[3450,9817,9818],{"class":4726},"Trim",[3450,9820,9821],{"class":3741},"(), ",[3450,9823,9824],{"class":4711},"false",[3450,9826,3858],{"class":3741},[3450,9828,7605],{"class":4714},[3450,9830,3845],{"class":3741},[3450,9832,7610],{"class":4714},[3450,9834,4800],{"class":3741},[3450,9836,9838,9841,9843,9845,9847,9850],{"class":3452,"line":9837},208,[3450,9839,9840],{"class":4714},"            store",[3450,9842,6598],{"class":3741},[3450,9844,8952],{"class":4714},[3450,9846,4864],{"class":3741},[3450,9848,9849],{"class":4714},"item",[3450,9851,4684],{"class":3741},[3450,9853,9855],{"class":3452,"line":9854},209,[3450,9856,3475],{"emptyLinePlaceholder":3474},[3450,9858,9860],{"class":3452,"line":9859},210,[3450,9861,9862],{"class":3735},"            \u002F\u002F 201 Created + заголовок Location з URL нового ресурсу\n",[3450,9864,9866,9868,9870,9872,9874,9876,9878,9881,9883,9886,9888,9890,9892,9894],{"class":3452,"line":9865},211,[3450,9867,6041],{"class":4714},[3450,9869,3845],{"class":3741},[3450,9871,4151],{"class":4714},[3450,9873,3845],{"class":3741},[3450,9875,5115],{"class":4726},[3450,9877,4730],{"class":3741},[3450,9879,9880],{"class":4733},"\"Location\"",[3450,9882,3858],{"class":3741},[3450,9884,9885],{"class":4733},"$\"\u002Ftodos\u002F",[3450,9887,5265],{"class":5264},[3450,9889,8952],{"class":4714},[3450,9891,5270],{"class":5264},[3450,9893,5273],{"class":4733},[3450,9895,4800],{"class":3741},[3450,9897,9899,9901,9903,9905,9907,9909,9912,9914,9916],{"class":3452,"line":9898},212,[3450,9900,6015],{"class":4711},[3450,9902,8459],{"class":4726},[3450,9904,4730],{"class":3741},[3450,9906,6023],{"class":4714},[3450,9908,3858],{"class":3741},[3450,9910,9911],{"class":3745},"201",[3450,9913,3858],{"class":3741},[3450,9915,9849],{"class":4714},[3450,9917,4800],{"class":3741},[3450,9919,9921,9923],{"class":3452,"line":9920},213,[3450,9922,5617],{"class":4672},[3450,9924,4684],{"class":3741},[3450,9926,9928],{"class":3452,"line":9927},214,[3450,9929,5576],{"class":3741},[3450,9931,9933],{"class":3452,"line":9932},215,[3450,9934,3475],{"emptyLinePlaceholder":3474},[3450,9936,9938,9941],{"class":3452,"line":9937},216,[3450,9939,9940],{"class":4672},"        default",[3450,9942,9224],{"class":3741},[3450,9944,9946,9948,9950,9952,9954,9956,9958,9960,9962,9965],{"class":3452,"line":9945},217,[3450,9947,6041],{"class":4714},[3450,9949,3845],{"class":3741},[3450,9951,4151],{"class":4714},[3450,9953,3845],{"class":3741},[3450,9955,5115],{"class":4726},[3450,9957,4730],{"class":3741},[3450,9959,6054],{"class":4733},[3450,9961,3858],{"class":3741},[3450,9963,9964],{"class":4733},"\"GET, POST\"",[3450,9966,4800],{"class":3741},[3450,9968,9970,9972,9974,9976,9978,9980,9982],{"class":3452,"line":9969},218,[3450,9971,6015],{"class":4711},[3450,9973,8459],{"class":4726},[3450,9975,4730],{"class":3741},[3450,9977,6023],{"class":4714},[3450,9979,3858],{"class":3741},[3450,9981,6028],{"class":3745},[3450,9983,4869],{"class":3741},[3450,9985,9987,9989,9991,9993,9995,9997],{"class":3452,"line":9986},219,[3450,9988,8475],{"class":4711},[3450,9990,6832],{"class":3741},[3450,9992,8480],{"class":4714},[3450,9994,4718],{"class":3741},[3450,9996,6033],{"class":4733},[3450,9998,8497],{"class":3741},[3450,10000,10002,10004],{"class":3452,"line":10001},220,[3450,10003,5617],{"class":4672},[3450,10005,4684],{"class":3741},[3450,10007,10009],{"class":3452,"line":10008},221,[3450,10010,5682],{"class":3741},[3450,10012,10014],{"class":3452,"line":10013},222,[3450,10015,5688],{"class":3741},[3450,10017,10019],{"class":3452,"line":10018},223,[3450,10020,3475],{"emptyLinePlaceholder":3474},[3450,10022,10024],{"class":3452,"line":10023},224,[3450,10025,10026],{"class":3735},"\u002F\u002F ── GET\u002FPUT\u002FPATCH\u002FDELETE \u002Ftodos\u002F{id} ─────────────────────────────────────────\n",[3450,10028,10030,10032,10034,10036],{"class":3452,"line":10029},225,[3450,10031,8280],{"class":4711},[3450,10033,5498],{"class":4675},[3450,10035,8939],{"class":4726},[3450,10037,5768],{"class":3741},[3450,10039,10041,10043,10045,10047,10049,10051,10053,10055,10057],{"class":3452,"line":10040},226,[3450,10042,5819],{"class":4675},[3450,10044,9182],{"class":4714},[3450,10046,3858],{"class":3741},[3450,10048,3363],{"class":4675},[3450,10050,5841],{"class":4714},[3450,10052,3858],{"class":3741},[3450,10054,4316],{"class":4711},[3450,10056,8901],{"class":4714},[3450,10058,4848],{"class":3741},[3450,10060,10062],{"class":3452,"line":10061},227,[3450,10063,4853],{"class":3741},[3450,10065,10067],{"class":3452,"line":10066},228,[3450,10068,10069],{"class":3735},"    \u002F\u002F Перевіряємо існування ресурсу\n",[3450,10071,10073,10076,10078,10080,10082,10084,10086,10088,10090,10092,10094,10096,10099],{"class":3452,"line":10072},229,[3450,10074,10075],{"class":4672},"    if",[3450,10077,4767],{"class":3741},[3450,10079,9278],{"class":4714},[3450,10081,3845],{"class":3741},[3450,10083,6435],{"class":4726},[3450,10085,4730],{"class":3741},[3450,10087,8952],{"class":4714},[3450,10089,3858],{"class":3741},[3450,10091,6445],{"class":4711},[3450,10093,7417],{"class":4675},[3450,10095,6451],{"class":3741},[3450,10097,10098],{"class":4714},"existing",[3450,10100,4783],{"class":3741},[3450,10102,10104],{"class":3452,"line":10103},230,[3450,10105,5378],{"class":3741},[3450,10107,10109,10111,10113,10115,10117,10119,10121],{"class":3452,"line":10108},231,[3450,10110,6493],{"class":4711},[3450,10112,8459],{"class":4726},[3450,10114,4730],{"class":3741},[3450,10116,6023],{"class":4714},[3450,10118,3858],{"class":3741},[3450,10120,6354],{"class":3745},[3450,10122,4869],{"class":3741},[3450,10124,10126,10129,10131,10133,10135,10137,10139,10141],{"class":3452,"line":10125},232,[3450,10127,10128],{"class":4711},"            new",[3450,10130,6832],{"class":3741},[3450,10132,8480],{"class":4714},[3450,10134,4718],{"class":3741},[3450,10136,4337],{"class":4733},[3450,10138,3858],{"class":3741},[3450,10140,8952],{"class":4714},[3450,10142,8497],{"class":3741},[3450,10144,10146,10149],{"class":3452,"line":10145},233,[3450,10147,10148],{"class":4672},"        return",[3450,10150,4684],{"class":3741},[3450,10152,10154],{"class":3452,"line":10153},234,[3450,10155,5682],{"class":3741},[3450,10157,10159],{"class":3452,"line":10158},235,[3450,10160,3475],{"emptyLinePlaceholder":3474},[3450,10162,10164,10166,10168,10170,10172,10174],{"class":3452,"line":10163},236,[3450,10165,9199],{"class":4672},[3450,10167,5528],{"class":3741},[3450,10169,5905],{"class":4714},[3450,10171,3845],{"class":3741},[3450,10173,4095],{"class":4714},[3450,10175,4848],{"class":3741},[3450,10177,10179],{"class":3452,"line":10178},237,[3450,10180,5378],{"class":3741},[3450,10182,10184,10186,10188],{"class":3452,"line":10183},238,[3450,10185,9218],{"class":4672},[3450,10187,9221],{"class":4733},[3450,10189,9224],{"class":3741},[3450,10191,10193,10195,10197,10199,10201,10203,10205,10207,10209],{"class":3452,"line":10192},239,[3450,10194,6015],{"class":4711},[3450,10196,8459],{"class":4726},[3450,10198,4730],{"class":3741},[3450,10200,6023],{"class":4714},[3450,10202,3858],{"class":3741},[3450,10204,4320],{"class":3745},[3450,10206,3858],{"class":3741},[3450,10208,10098],{"class":4714},[3450,10210,4800],{"class":3741},[3450,10212,10214,10216],{"class":3452,"line":10213},240,[3450,10215,5617],{"class":4672},[3450,10217,4684],{"class":3741},[3450,10219,10221],{"class":3452,"line":10220},241,[3450,10222,3475],{"emptyLinePlaceholder":3474},[3450,10224,10226,10228,10231],{"class":3452,"line":10225},242,[3450,10227,9218],{"class":4672},[3450,10229,10230],{"class":4733}," \"PUT\"",[3450,10232,9224],{"class":3741},[3450,10234,10236],{"class":3452,"line":10235},243,[3450,10237,5401],{"class":3741},[3450,10239,10241],{"class":3452,"line":10240},244,[3450,10242,10243],{"class":3735},"            \u002F\u002F Повне оновлення — вимагає всі поля\n",[3450,10245,10247,10249,10251,10253,10255,10257,10259,10261,10263,10265,10267],{"class":3452,"line":10246},245,[3450,10248,9534],{"class":4672},[3450,10250,5087],{"class":4711},[3450,10252,9539],{"class":4714},[3450,10254,4718],{"class":3741},[3450,10256,4822],{"class":4711},[3450,10258,9546],{"class":4675},[3450,10260,4730],{"class":3741},[3450,10262,5905],{"class":4714},[3450,10264,3845],{"class":3741},[3450,10266,4162],{"class":4714},[3450,10268,4869],{"class":3741},[3450,10270,10272,10274,10276,10278,10280,10282,10284,10286],{"class":3452,"line":10271},246,[3450,10273,9561],{"class":4714},[3450,10275,3845],{"class":3741},[3450,10277,4367],{"class":4714},[3450,10279,9568],{"class":3741},[3450,10281,4368],{"class":4714},[3450,10283,3845],{"class":3741},[3450,10285,6997],{"class":4714},[3450,10287,4800],{"class":3741},[3450,10289,10291,10293,10295,10297,10299,10301,10303,10305],{"class":3452,"line":10290},247,[3450,10292,9238],{"class":4711},[3450,10294,9583],{"class":4714},[3450,10296,4718],{"class":3741},[3450,10298,5495],{"class":4711},[3450,10300,9539],{"class":4714},[3450,10302,3845],{"class":3741},[3450,10304,9594],{"class":4726},[3450,10306,5100],{"class":3741},[3450,10308,10310,10312,10314,10316,10318,10320,10322,10324,10326,10328],{"class":3452,"line":10309},248,[3450,10311,9534],{"class":4672},[3450,10313,5087],{"class":4711},[3450,10315,9609],{"class":4714},[3450,10317,4718],{"class":3741},[3450,10319,9614],{"class":4714},[3450,10321,3845],{"class":3741},[3450,10323,8910],{"class":4726},[3450,10325,4730],{"class":3741},[3450,10327,6988],{"class":4714},[3450,10329,4800],{"class":3741},[3450,10331,10333],{"class":3452,"line":10332},249,[3450,10334,3475],{"emptyLinePlaceholder":3474},[3450,10336,10338,10340,10342,10344,10346,10348,10350,10353,10355,10357,10359,10361,10363,10365],{"class":3452,"line":10337},250,[3450,10339,9238],{"class":4711},[3450,10341,6451],{"class":3741},[3450,10343,9633],{"class":4714},[3450,10345,4718],{"class":3741},[3450,10347,9638],{"class":4714},[3450,10349,3845],{"class":3741},[3450,10351,10352],{"class":4714},"RootElement",[3450,10354,3845],{"class":3741},[3450,10356,9652],{"class":4726},[3450,10358,4730],{"class":3741},[3450,10360,9657],{"class":4733},[3450,10362,4226],{"class":3741},[3450,10364,9662],{"class":4726},[3450,10366,5100],{"class":3741},[3450,10368,10370,10373,10376,10378,10380,10382],{"class":3452,"line":10369},251,[3450,10371,10372],{"class":4711},"            bool",[3450,10374,10375],{"class":4714}," isCompleted",[3450,10377,4718],{"class":3741},[3450,10379,9638],{"class":4714},[3450,10381,3845],{"class":3741},[3450,10383,9643],{"class":4714},[3450,10385,10387,10389,10391,10393,10396,10398,10401],{"class":3452,"line":10386},252,[3450,10388,9649],{"class":3741},[3450,10390,9652],{"class":4726},[3450,10392,4730],{"class":3741},[3450,10394,10395],{"class":4733},"\"isCompleted\"",[3450,10397,4226],{"class":3741},[3450,10399,10400],{"class":4726},"GetBoolean",[3450,10402,5100],{"class":3741},[3450,10404,10406],{"class":3452,"line":10405},253,[3450,10407,3475],{"emptyLinePlaceholder":3474},[3450,10409,10411,10413,10415,10417,10419,10421,10423,10425],{"class":3452,"line":10410},254,[3450,10412,5525],{"class":4672},[3450,10414,5528],{"class":3741},[3450,10416,3889],{"class":4711},[3450,10418,3845],{"class":3741},[3450,10420,9683],{"class":4726},[3450,10422,4730],{"class":3741},[3450,10424,9633],{"class":4714},[3450,10426,4783],{"class":3741},[3450,10428,10430],{"class":3452,"line":10429},255,[3450,10431,9465],{"class":3741},[3450,10433,10435,10437,10439,10441,10443,10445,10447],{"class":3452,"line":10434},256,[3450,10436,9470],{"class":4711},[3450,10438,8459],{"class":4726},[3450,10440,4730],{"class":3741},[3450,10442,6023],{"class":4714},[3450,10444,3858],{"class":3741},[3450,10446,9710],{"class":3745},[3450,10448,4869],{"class":3741},[3450,10450,10452,10454,10456,10458,10460,10462,10464,10466,10468,10470],{"class":3452,"line":10451},257,[3450,10453,9488],{"class":4711},[3450,10455,6832],{"class":3741},[3450,10457,8480],{"class":4714},[3450,10459,4718],{"class":3741},[3450,10461,9726],{"class":4733},[3450,10463,3858],{"class":3741},[3450,10465,3803],{"class":4714},[3450,10467,4718],{"class":3741},[3450,10469,9657],{"class":4733},[3450,10471,8497],{"class":3741},[3450,10473,10475,10477],{"class":3452,"line":10474},258,[3450,10476,9513],{"class":4672},[3450,10478,4684],{"class":3741},[3450,10480,10482],{"class":3452,"line":10481},259,[3450,10483,9520],{"class":3741},[3450,10485,10487],{"class":3452,"line":10486},260,[3450,10488,3475],{"emptyLinePlaceholder":3474},[3450,10490,10492,10494,10497,10499,10501],{"class":3452,"line":10491},261,[3450,10493,9796],{"class":4711},[3450,10495,10496],{"class":4714}," updated",[3450,10498,4718],{"class":3741},[3450,10500,10098],{"class":4714},[3450,10502,10503],{"class":4711}," with\n",[3450,10505,10507],{"class":3452,"line":10506},262,[3450,10508,9465],{"class":3741},[3450,10510,10512,10515,10517,10519,10521,10523],{"class":3452,"line":10511},263,[3450,10513,10514],{"class":4714},"                Title",[3450,10516,4718],{"class":3741},[3450,10518,9633],{"class":4714},[3450,10520,3845],{"class":3741},[3450,10522,9818],{"class":4726},[3450,10524,10525],{"class":3741},"(),\n",[3450,10527,10529,10532,10534],{"class":3452,"line":10528},264,[3450,10530,10531],{"class":4714},"                IsCompleted",[3450,10533,4718],{"class":3741},[3450,10535,10536],{"class":4714},"isCompleted\n",[3450,10538,10540],{"class":3452,"line":10539},265,[3450,10541,10542],{"class":3741},"            };\n",[3450,10544,10546,10548,10550,10552,10554,10557],{"class":3452,"line":10545},266,[3450,10547,9840],{"class":4714},[3450,10549,6598],{"class":3741},[3450,10551,8952],{"class":4714},[3450,10553,4864],{"class":3741},[3450,10555,10556],{"class":4714},"updated",[3450,10558,4684],{"class":3741},[3450,10560,10562,10564,10566,10568,10570,10572,10574,10576,10578],{"class":3452,"line":10561},267,[3450,10563,6015],{"class":4711},[3450,10565,8459],{"class":4726},[3450,10567,4730],{"class":3741},[3450,10569,6023],{"class":4714},[3450,10571,3858],{"class":3741},[3450,10573,4320],{"class":3745},[3450,10575,3858],{"class":3741},[3450,10577,10556],{"class":4714},[3450,10579,4800],{"class":3741},[3450,10581,10583,10585],{"class":3452,"line":10582},268,[3450,10584,5617],{"class":4672},[3450,10586,4684],{"class":3741},[3450,10588,10590],{"class":3452,"line":10589},269,[3450,10591,5576],{"class":3741},[3450,10593,10595],{"class":3452,"line":10594},270,[3450,10596,3475],{"emptyLinePlaceholder":3474},[3450,10598,10600,10602,10605],{"class":3452,"line":10599},271,[3450,10601,9218],{"class":4672},[3450,10603,10604],{"class":4733}," \"PATCH\"",[3450,10606,9224],{"class":3741},[3450,10608,10610],{"class":3452,"line":10609},272,[3450,10611,5401],{"class":3741},[3450,10613,10615],{"class":3452,"line":10614},273,[3450,10616,10617],{"class":3735},"            \u002F\u002F Часткове оновлення — тільки isCompleted\n",[3450,10619,10621,10623,10625,10627,10629,10631,10633,10635,10637,10639,10641],{"class":3452,"line":10620},274,[3450,10622,9534],{"class":4672},[3450,10624,5087],{"class":4711},[3450,10626,9539],{"class":4714},[3450,10628,4718],{"class":3741},[3450,10630,4822],{"class":4711},[3450,10632,9546],{"class":4675},[3450,10634,4730],{"class":3741},[3450,10636,5905],{"class":4714},[3450,10638,3845],{"class":3741},[3450,10640,4162],{"class":4714},[3450,10642,4869],{"class":3741},[3450,10644,10646,10648,10650,10652,10654,10656,10658,10660],{"class":3452,"line":10645},275,[3450,10647,9561],{"class":4714},[3450,10649,3845],{"class":3741},[3450,10651,4367],{"class":4714},[3450,10653,9568],{"class":3741},[3450,10655,4368],{"class":4714},[3450,10657,3845],{"class":3741},[3450,10659,6997],{"class":4714},[3450,10661,4800],{"class":3741},[3450,10663,10665,10667,10669,10671,10673,10675,10677,10679],{"class":3452,"line":10664},276,[3450,10666,9238],{"class":4711},[3450,10668,9583],{"class":4714},[3450,10670,4718],{"class":3741},[3450,10672,5495],{"class":4711},[3450,10674,9539],{"class":4714},[3450,10676,3845],{"class":3741},[3450,10678,9594],{"class":4726},[3450,10680,5100],{"class":3741},[3450,10682,10684,10686,10688,10690,10692,10694,10696,10698,10700,10702],{"class":3452,"line":10683},277,[3450,10685,9534],{"class":4672},[3450,10687,5087],{"class":4711},[3450,10689,9609],{"class":4714},[3450,10691,4718],{"class":3741},[3450,10693,9614],{"class":4714},[3450,10695,3845],{"class":3741},[3450,10697,8910],{"class":4726},[3450,10699,4730],{"class":3741},[3450,10701,6988],{"class":4714},[3450,10703,4800],{"class":3741},[3450,10705,10707],{"class":3452,"line":10706},278,[3450,10708,3475],{"emptyLinePlaceholder":3474},[3450,10710,10712,10714,10716,10719,10721,10723],{"class":3452,"line":10711},279,[3450,10713,10372],{"class":4711},[3450,10715,6451],{"class":3741},[3450,10717,10718],{"class":4714},"isCompleted",[3450,10720,4718],{"class":3741},[3450,10722,3959],{"class":4711},[3450,10724,4684],{"class":3741},[3450,10726,10728,10730,10732,10734,10736,10738,10740,10743,10745,10747,10749,10751,10753,10756],{"class":3452,"line":10727},280,[3450,10729,5525],{"class":4672},[3450,10731,5528],{"class":3741},[3450,10733,9638],{"class":4714},[3450,10735,3845],{"class":3741},[3450,10737,10352],{"class":4714},[3450,10739,3845],{"class":3741},[3450,10741,10742],{"class":4726},"TryGetProperty",[3450,10744,4730],{"class":3741},[3450,10746,10395],{"class":4733},[3450,10748,3858],{"class":3741},[3450,10750,6445],{"class":4711},[3450,10752,5087],{"class":4711},[3450,10754,10755],{"class":4714}," val",[3450,10757,4783],{"class":3741},[3450,10759,10761,10764,10766,10769,10771,10773],{"class":3452,"line":10760},281,[3450,10762,10763],{"class":4714},"                isCompleted",[3450,10765,4718],{"class":3741},[3450,10767,10768],{"class":4714},"val",[3450,10770,3845],{"class":3741},[3450,10772,10400],{"class":4726},[3450,10774,5100],{"class":3741},[3450,10776,10778],{"class":3452,"line":10777},282,[3450,10779,3475],{"emptyLinePlaceholder":3474},[3450,10781,10783,10785,10787,10789,10791,10793],{"class":3452,"line":10782},283,[3450,10784,5525],{"class":4672},[3450,10786,5528],{"class":3741},[3450,10788,10718],{"class":4714},[3450,10790,9320],{"class":4711},[3450,10792,9326],{"class":4711},[3450,10794,4848],{"class":3741},[3450,10796,10798],{"class":3452,"line":10797},284,[3450,10799,9465],{"class":3741},[3450,10801,10803,10805,10807,10809,10811,10813,10815],{"class":3452,"line":10802},285,[3450,10804,9470],{"class":4711},[3450,10806,8459],{"class":4726},[3450,10808,4730],{"class":3741},[3450,10810,6023],{"class":4714},[3450,10812,3858],{"class":3741},[3450,10814,9481],{"class":3745},[3450,10816,4869],{"class":3741},[3450,10818,10820,10822,10824,10826,10828,10830],{"class":3452,"line":10819},286,[3450,10821,9488],{"class":4711},[3450,10823,6832],{"class":3741},[3450,10825,8480],{"class":4714},[3450,10827,4718],{"class":3741},[3450,10829,9497],{"class":4733},[3450,10831,4869],{"class":3741},[3450,10833,10835,10837,10839,10842],{"class":3452,"line":10834},287,[3450,10836,9734],{"class":4714},[3450,10838,4718],{"class":3741},[3450,10840,10841],{"class":4733},"\"Хоча б одне поле має бути вказано\"",[3450,10843,8497],{"class":3741},[3450,10845,10847,10849],{"class":3452,"line":10846},288,[3450,10848,9513],{"class":4672},[3450,10850,4684],{"class":3741},[3450,10852,10854],{"class":3452,"line":10853},289,[3450,10855,9520],{"class":3741},[3450,10857,10859],{"class":3452,"line":10858},290,[3450,10860,3475],{"emptyLinePlaceholder":3474},[3450,10862,10864,10866,10869,10871,10873,10876,10878,10880,10882,10884,10886,10888],{"class":3452,"line":10863},291,[3450,10865,9796],{"class":4711},[3450,10867,10868],{"class":4714}," patched",[3450,10870,4718],{"class":3741},[3450,10872,10098],{"class":4714},[3450,10874,10875],{"class":4711}," with",[3450,10877,6832],{"class":3741},[3450,10879,9382],{"class":4714},[3450,10881,4718],{"class":3741},[3450,10883,10718],{"class":4714},[3450,10885,3845],{"class":3741},[3450,10887,8930],{"class":4714},[3450,10889,10890],{"class":3741}," };\n",[3450,10892,10894,10896,10898,10900,10902,10905],{"class":3452,"line":10893},292,[3450,10895,9840],{"class":4714},[3450,10897,6598],{"class":3741},[3450,10899,8952],{"class":4714},[3450,10901,4864],{"class":3741},[3450,10903,10904],{"class":4714},"patched",[3450,10906,4684],{"class":3741},[3450,10908,10910,10912,10914,10916,10918,10920,10922,10924,10926],{"class":3452,"line":10909},293,[3450,10911,6015],{"class":4711},[3450,10913,8459],{"class":4726},[3450,10915,4730],{"class":3741},[3450,10917,6023],{"class":4714},[3450,10919,3858],{"class":3741},[3450,10921,4320],{"class":3745},[3450,10923,3858],{"class":3741},[3450,10925,10904],{"class":4714},[3450,10927,4800],{"class":3741},[3450,10929,10931,10933],{"class":3452,"line":10930},294,[3450,10932,5617],{"class":4672},[3450,10934,4684],{"class":3741},[3450,10936,10938],{"class":3452,"line":10937},295,[3450,10939,5576],{"class":3741},[3450,10941,10943],{"class":3452,"line":10942},296,[3450,10944,3475],{"emptyLinePlaceholder":3474},[3450,10946,10948,10950,10953],{"class":3452,"line":10947},297,[3450,10949,9218],{"class":4672},[3450,10951,10952],{"class":4733}," \"DELETE\"",[3450,10954,9224],{"class":3741},[3450,10956,10958],{"class":3452,"line":10957},298,[3450,10959,5401],{"class":3741},[3450,10961,10963,10965,10967,10970,10972,10974,10976,10978,10981],{"class":3452,"line":10962},299,[3450,10964,9840],{"class":4714},[3450,10966,3845],{"class":3741},[3450,10968,10969],{"class":4726},"TryRemove",[3450,10971,4730],{"class":3741},[3450,10973,8952],{"class":4714},[3450,10975,3858],{"class":3741},[3450,10977,6445],{"class":4711},[3450,10979,10980],{"class":4714}," _",[3450,10982,4800],{"class":3741},[3450,10984,10986,10988,10990,10992,10994,10996,10999],{"class":3452,"line":10985},300,[3450,10987,6041],{"class":4714},[3450,10989,3845],{"class":3741},[3450,10991,4315],{"class":4714},[3450,10993,4718],{"class":3741},[3450,10995,8743],{"class":3745},[3450,10997,10998],{"class":3741},";  ",[3450,11000,11001],{"class":3735},"\u002F\u002F No Content — успішно видалено, тіло відсутнє\n",[3450,11003,11005,11007,11009,11011],{"class":3452,"line":11004},301,[3450,11006,6041],{"class":4714},[3450,11008,3845],{"class":3741},[3450,11010,6702],{"class":4726},[3450,11012,5100],{"class":3741},[3450,11014,11016,11018],{"class":3452,"line":11015},302,[3450,11017,5617],{"class":4672},[3450,11019,4684],{"class":3741},[3450,11021,11023],{"class":3452,"line":11022},303,[3450,11024,5576],{"class":3741},[3450,11026,11028],{"class":3452,"line":11027},304,[3450,11029,3475],{"emptyLinePlaceholder":3474},[3450,11031,11033,11035],{"class":3452,"line":11032},305,[3450,11034,9940],{"class":4672},[3450,11036,9224],{"class":3741},[3450,11038,11040,11042,11044,11046,11048,11050,11052,11054,11056,11059],{"class":3452,"line":11039},306,[3450,11041,6041],{"class":4714},[3450,11043,3845],{"class":3741},[3450,11045,4151],{"class":4714},[3450,11047,3845],{"class":3741},[3450,11049,5115],{"class":4726},[3450,11051,4730],{"class":3741},[3450,11053,6054],{"class":4733},[3450,11055,3858],{"class":3741},[3450,11057,11058],{"class":4733},"\"GET, PUT, PATCH, DELETE\"",[3450,11060,4800],{"class":3741},[3450,11062,11064,11066,11068,11070,11072,11074,11076],{"class":3452,"line":11063},307,[3450,11065,6015],{"class":4711},[3450,11067,8459],{"class":4726},[3450,11069,4730],{"class":3741},[3450,11071,6023],{"class":4714},[3450,11073,3858],{"class":3741},[3450,11075,6028],{"class":3745},[3450,11077,4869],{"class":3741},[3450,11079,11081,11083,11085,11087,11089,11091],{"class":3452,"line":11080},308,[3450,11082,8475],{"class":4711},[3450,11084,6832],{"class":3741},[3450,11086,8480],{"class":4714},[3450,11088,4718],{"class":3741},[3450,11090,6033],{"class":4733},[3450,11092,8497],{"class":3741},[3450,11094,11096,11098],{"class":3452,"line":11095},309,[3450,11097,5617],{"class":4672},[3450,11099,4684],{"class":3741},[3450,11101,11103],{"class":3452,"line":11102},310,[3450,11104,5682],{"class":3741},[3450,11106,11108],{"class":3452,"line":11107},311,[3450,11109,5688],{"class":3741},[3450,11111,11113],{"class":3452,"line":11112},312,[3450,11114,3475],{"emptyLinePlaceholder":3474},[3450,11116,11118],{"class":3452,"line":11117},313,[3450,11119,11120],{"class":3735},"\u002F\u002F ── Допоміжний метод: серіалізація та відправлення JSON ──────────────────────\n",[3450,11122,11124,11126,11128,11130,11132,11135,11137,11139,11141,11143,11145,11147,11149,11151,11154],{"class":3452,"line":11123},314,[3450,11125,8280],{"class":4711},[3450,11127,5498],{"class":4675},[3450,11129,8459],{"class":4726},[3450,11131,4828],{"class":3741},[3450,11133,11134],{"class":4675},"T",[3450,11136,4837],{"class":3741},[3450,11138,3363],{"class":4675},[3450,11140,5841],{"class":4714},[3450,11142,3858],{"class":3741},[3450,11144,4316],{"class":4711},[3450,11146,6962],{"class":4714},[3450,11148,3858],{"class":3741},[3450,11150,11134],{"class":4675},[3450,11152,11153],{"class":4714}," data",[3450,11155,4848],{"class":3741},[3450,11157,11159],{"class":3452,"line":11158},315,[3450,11160,4853],{"class":3741},[3450,11162,11164,11166,11168,11170,11172,11175,11177,11180,11182,11185,11187,11190],{"class":3452,"line":11163},316,[3450,11165,6982],{"class":4711},[3450,11167,6985],{"class":3741},[3450,11169,6988],{"class":4714},[3450,11171,4718],{"class":3741},[3450,11173,11174],{"class":4714},"JsonSerializer",[3450,11176,3845],{"class":3741},[3450,11178,11179],{"class":4726},"SerializeToUtf8Bytes",[3450,11181,4730],{"class":3741},[3450,11183,11184],{"class":4714},"data",[3450,11186,3858],{"class":3741},[3450,11188,11189],{"class":4714},"jsonOptions",[3450,11191,4800],{"class":3741},[3450,11193,11195],{"class":3452,"line":11194},317,[3450,11196,3475],{"emptyLinePlaceholder":3474},[3450,11198,11200,11202,11204,11206,11208,11210],{"class":3452,"line":11199},318,[3450,11201,7117],{"class":4714},[3450,11203,3845],{"class":3741},[3450,11205,4315],{"class":4714},[3450,11207,7124],{"class":3741},[3450,11209,7030],{"class":4714},[3450,11211,4684],{"class":3741},[3450,11213,11215,11217,11219,11221,11223,11225],{"class":3452,"line":11214},319,[3450,11216,7117],{"class":4714},[3450,11218,3845],{"class":3741},[3450,11220,4187],{"class":4714},[3450,11222,4750],{"class":3741},[3450,11224,4196],{"class":4733},[3450,11226,4684],{"class":3741},[3450,11228,11230,11232,11234,11236,11238,11240,11242,11244],{"class":3452,"line":11229},320,[3450,11231,7117],{"class":4714},[3450,11233,3845],{"class":3741},[3450,11235,4203],{"class":4714},[3450,11237,4718],{"class":3741},[3450,11239,6988],{"class":4714},[3450,11241,3845],{"class":3741},[3450,11243,6573],{"class":4714},[3450,11245,4684],{"class":3741},[3450,11247,11249],{"class":3452,"line":11248},321,[3450,11250,3475],{"emptyLinePlaceholder":3474},[3450,11252,11254],{"class":3452,"line":11253},322,[3450,11255,11256],{"class":3735},"    \u002F\u002F Стандартні заголовки безпеки та відсутності кешу для API\n",[3450,11258,11260,11262,11264,11266,11268,11270,11272,11274,11276,11279],{"class":3452,"line":11259},323,[3450,11261,7117],{"class":4714},[3450,11263,3845],{"class":3741},[3450,11265,4151],{"class":4714},[3450,11267,6598],{"class":3741},[3450,11269,6601],{"class":4714},[3450,11271,3845],{"class":3741},[3450,11273,6606],{"class":4714},[3450,11275,4864],{"class":3741},[3450,11277,11278],{"class":4733},"\"no-store\"",[3450,11280,4684],{"class":3741},[3450,11282,11284,11286,11288,11290,11292,11294,11296,11299,11301,11304],{"class":3452,"line":11283},324,[3450,11285,7117],{"class":4714},[3450,11287,3845],{"class":3741},[3450,11289,4151],{"class":4714},[3450,11291,3845],{"class":3741},[3450,11293,5115],{"class":4726},[3450,11295,4730],{"class":3741},[3450,11297,11298],{"class":4733},"\"X-Content-Type-Options\"",[3450,11300,3858],{"class":3741},[3450,11302,11303],{"class":4733},"\"nosniff\"",[3450,11305,4800],{"class":3741},[3450,11307,11309],{"class":3452,"line":11308},325,[3450,11310,3475],{"emptyLinePlaceholder":3474},[3450,11312,11314,11316,11318,11320,11322,11324,11326,11328,11330],{"class":3452,"line":11313},326,[3450,11315,7173],{"class":4711},[3450,11317,5841],{"class":4714},[3450,11319,3845],{"class":3741},[3450,11321,4324],{"class":4714},[3450,11323,3845],{"class":3741},[3450,11325,7184],{"class":4726},[3450,11327,4730],{"class":3741},[3450,11329,6988],{"class":4714},[3450,11331,4800],{"class":3741},[3450,11333,11335],{"class":3452,"line":11334},327,[3450,11336,5688],{"class":3741},[3369,11338,11340],{"id":11339},"тестування-api-за-допомогою-curl","Тестування API за допомогою curl",[11342,11343,11344,11403,11471,11595,11641],"tabs",{},[11345,11346,11348],"tabs-item",{"label":11347},"GET — список",[3441,11349,11353],{"className":11350,"code":11351,"language":11352,"meta":3446,"style":3446},"language-bash shiki shiki-themes light-plus dark-plus dark-plus","# Список усіх задач\ncurl -u admin:secret http:\u002F\u002Flocalhost:8081\u002Ftodos | json_pp\n\n# Тільки невиконані\ncurl -u admin:secret \"http:\u002F\u002Flocalhost:8081\u002Ftodos?completed=false\" | json_pp\n","bash",[3321,11354,11355,11360,11379,11383,11388],{"__ignoreMap":3446},[3450,11356,11357],{"class":3452,"line":3453},[3450,11358,11359],{"class":3735},"# Список усіх задач\n",[3450,11361,11362,11365,11368,11371,11374,11376],{"class":3452,"line":3459},[3450,11363,11364],{"class":4726},"curl",[3450,11366,11367],{"class":4711}," -u",[3450,11369,11370],{"class":4733}," admin:secret",[3450,11372,11373],{"class":4733}," http:\u002F\u002Flocalhost:8081\u002Ftodos",[3450,11375,7779],{"class":3741},[3450,11377,11378],{"class":4726},"json_pp\n",[3450,11380,11381],{"class":3452,"line":3465},[3450,11382,3475],{"emptyLinePlaceholder":3474},[3450,11384,11385],{"class":3452,"line":3471},[3450,11386,11387],{"class":3735},"# Тільки невиконані\n",[3450,11389,11390,11392,11394,11396,11399,11401],{"class":3452,"line":3478},[3450,11391,11364],{"class":4726},[3450,11393,11367],{"class":4711},[3450,11395,11370],{"class":4733},[3450,11397,11398],{"class":4733}," \"http:\u002F\u002Flocalhost:8081\u002Ftodos?completed=false\"",[3450,11400,7779],{"class":3741},[3450,11402,11378],{"class":4726},[11345,11404,11406],{"label":11405},"POST — створення",[3441,11407,11409],{"className":11350,"code":11408,"language":11352,"meta":3446,"style":3446},"curl -u admin:secret \\\n  -X POST \\\n  -H \"Content-Type: application\u002Fjson\" \\\n  -d '{\"title\": \"Вивчити HttpListener\"}' \\\n  http:\u002F\u002Flocalhost:8081\u002Ftodos\n\n# Відповідь: 201 Created\n# Location: \u002Ftodos\u002F4\n",[3321,11410,11411,11422,11432,11442,11452,11457,11461,11466],{"__ignoreMap":3446},[3450,11412,11413,11415,11417,11419],{"class":3452,"line":3453},[3450,11414,11364],{"class":4726},[3450,11416,11367],{"class":4711},[3450,11418,11370],{"class":4733},[3450,11420,11421],{"class":5729}," \\\n",[3450,11423,11424,11427,11430],{"class":3452,"line":3459},[3450,11425,11426],{"class":4711},"  -X",[3450,11428,11429],{"class":4733}," POST",[3450,11431,11421],{"class":5729},[3450,11433,11434,11437,11440],{"class":3452,"line":3465},[3450,11435,11436],{"class":4711},"  -H",[3450,11438,11439],{"class":4733}," \"Content-Type: application\u002Fjson\"",[3450,11441,11421],{"class":5729},[3450,11443,11444,11447,11450],{"class":3452,"line":3471},[3450,11445,11446],{"class":4711},"  -d",[3450,11448,11449],{"class":4733}," '{\"title\": \"Вивчити HttpListener\"}'",[3450,11451,11421],{"class":5729},[3450,11453,11454],{"class":3452,"line":3478},[3450,11455,11456],{"class":4733},"  http:\u002F\u002Flocalhost:8081\u002Ftodos\n",[3450,11458,11459],{"class":3452,"line":3484},[3450,11460,3475],{"emptyLinePlaceholder":3474},[3450,11462,11463],{"class":3452,"line":3489},[3450,11464,11465],{"class":3735},"# Відповідь: 201 Created\n",[3450,11467,11468],{"class":3452,"line":3495},[3450,11469,11470],{"class":3735},"# Location: \u002Ftodos\u002F4\n",[11345,11472,11474],{"label":11473},"GET, PUT, PATCH — ресурс",[3441,11475,11477],{"className":11350,"code":11476,"language":11352,"meta":3446,"style":3446},"# GET конкретної задачі\ncurl -u admin:secret http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F1\n\n# PUT — повне оновлення\ncurl -u admin:secret \\\n  -X PUT \\\n  -H \"Content-Type: application\u002Fjson\" \\\n  -d '{\"title\": \"Оновлена задача\", \"isCompleted\": true}' \\\n  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F1\n\n# PATCH — часткове оновлення (лише статус)\ncurl -u admin:secret \\\n  -X PATCH \\\n  -H \"Content-Type: application\u002Fjson\" \\\n  -d '{\"isCompleted\": true}' \\\n  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F2\n",[3321,11478,11479,11484,11495,11499,11504,11514,11523,11531,11540,11545,11549,11554,11564,11573,11581,11590],{"__ignoreMap":3446},[3450,11480,11481],{"class":3452,"line":3453},[3450,11482,11483],{"class":3735},"# GET конкретної задачі\n",[3450,11485,11486,11488,11490,11492],{"class":3452,"line":3459},[3450,11487,11364],{"class":4726},[3450,11489,11367],{"class":4711},[3450,11491,11370],{"class":4733},[3450,11493,11494],{"class":4733}," http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F1\n",[3450,11496,11497],{"class":3452,"line":3465},[3450,11498,3475],{"emptyLinePlaceholder":3474},[3450,11500,11501],{"class":3452,"line":3471},[3450,11502,11503],{"class":3735},"# PUT — повне оновлення\n",[3450,11505,11506,11508,11510,11512],{"class":3452,"line":3478},[3450,11507,11364],{"class":4726},[3450,11509,11367],{"class":4711},[3450,11511,11370],{"class":4733},[3450,11513,11421],{"class":5729},[3450,11515,11516,11518,11521],{"class":3452,"line":3484},[3450,11517,11426],{"class":4711},[3450,11519,11520],{"class":4733}," PUT",[3450,11522,11421],{"class":5729},[3450,11524,11525,11527,11529],{"class":3452,"line":3489},[3450,11526,11436],{"class":4711},[3450,11528,11439],{"class":4733},[3450,11530,11421],{"class":5729},[3450,11532,11533,11535,11538],{"class":3452,"line":3495},[3450,11534,11446],{"class":4711},[3450,11536,11537],{"class":4733}," '{\"title\": \"Оновлена задача\", \"isCompleted\": true}'",[3450,11539,11421],{"class":5729},[3450,11541,11542],{"class":3452,"line":3500},[3450,11543,11544],{"class":4733},"  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F1\n",[3450,11546,11547],{"class":3452,"line":3506},[3450,11548,3475],{"emptyLinePlaceholder":3474},[3450,11550,11551],{"class":3452,"line":3511},[3450,11552,11553],{"class":3735},"# PATCH — часткове оновлення (лише статус)\n",[3450,11555,11556,11558,11560,11562],{"class":3452,"line":3517},[3450,11557,11364],{"class":4726},[3450,11559,11367],{"class":4711},[3450,11561,11370],{"class":4733},[3450,11563,11421],{"class":5729},[3450,11565,11566,11568,11571],{"class":3452,"line":3522},[3450,11567,11426],{"class":4711},[3450,11569,11570],{"class":4733}," PATCH",[3450,11572,11421],{"class":5729},[3450,11574,11575,11577,11579],{"class":3452,"line":3528},[3450,11576,11436],{"class":4711},[3450,11578,11439],{"class":4733},[3450,11580,11421],{"class":5729},[3450,11582,11583,11585,11588],{"class":3452,"line":3534},[3450,11584,11446],{"class":4711},[3450,11586,11587],{"class":4733}," '{\"isCompleted\": true}'",[3450,11589,11421],{"class":5729},[3450,11591,11592],{"class":3452,"line":3540},[3450,11593,11594],{"class":4733},"  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F2\n",[11345,11596,11598],{"label":11597},"DELETE — видалення",[3441,11599,11601],{"className":11350,"code":11600,"language":11352,"meta":3446,"style":3446},"curl -u admin:secret \\\n  -X DELETE \\\n  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F3\n\n# Відповідь: 204 No Content\n# Тіло відсутнє\n",[3321,11602,11603,11613,11622,11627,11631,11636],{"__ignoreMap":3446},[3450,11604,11605,11607,11609,11611],{"class":3452,"line":3453},[3450,11606,11364],{"class":4726},[3450,11608,11367],{"class":4711},[3450,11610,11370],{"class":4733},[3450,11612,11421],{"class":5729},[3450,11614,11615,11617,11620],{"class":3452,"line":3459},[3450,11616,11426],{"class":4711},[3450,11618,11619],{"class":4733}," DELETE",[3450,11621,11421],{"class":5729},[3450,11623,11624],{"class":3452,"line":3465},[3450,11625,11626],{"class":4733},"  http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F3\n",[3450,11628,11629],{"class":3452,"line":3471},[3450,11630,3475],{"emptyLinePlaceholder":3474},[3450,11632,11633],{"class":3452,"line":3478},[3450,11634,11635],{"class":3735},"# Відповідь: 204 No Content\n",[3450,11637,11638],{"class":3452,"line":3484},[3450,11639,11640],{"class":3735},"# Тіло відсутнє\n",[11345,11642,11644],{"label":11643},"Помилки",[3441,11645,11647],{"className":11350,"code":11646,"language":11352,"meta":3446,"style":3446},"# 401 — без аутентифікації\ncurl http:\u002F\u002Flocalhost:8081\u002Ftodos\n\n# 403 — невірний пароль\ncurl -u admin:wrong http:\u002F\u002Flocalhost:8081\u002Ftodos\n\n# 404 — ресурс не існує\ncurl -u admin:secret http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F999\n\n# 422 — некоректне тіло\ncurl -u admin:secret \\\n  -X POST \\\n  -H \"Content-Type: application\u002Fjson\" \\\n  -d '{\"title\": \"\"}' \\\n  http:\u002F\u002Flocalhost:8081\u002Ftodos\n",[3321,11648,11649,11654,11661,11665,11670,11681,11685,11690,11701,11705,11710,11720,11728,11736,11745],{"__ignoreMap":3446},[3450,11650,11651],{"class":3452,"line":3453},[3450,11652,11653],{"class":3735},"# 401 — без аутентифікації\n",[3450,11655,11656,11658],{"class":3452,"line":3459},[3450,11657,11364],{"class":4726},[3450,11659,11660],{"class":4733}," http:\u002F\u002Flocalhost:8081\u002Ftodos\n",[3450,11662,11663],{"class":3452,"line":3465},[3450,11664,3475],{"emptyLinePlaceholder":3474},[3450,11666,11667],{"class":3452,"line":3471},[3450,11668,11669],{"class":3735},"# 403 — невірний пароль\n",[3450,11671,11672,11674,11676,11679],{"class":3452,"line":3478},[3450,11673,11364],{"class":4726},[3450,11675,11367],{"class":4711},[3450,11677,11678],{"class":4733}," admin:wrong",[3450,11680,11660],{"class":4733},[3450,11682,11683],{"class":3452,"line":3484},[3450,11684,3475],{"emptyLinePlaceholder":3474},[3450,11686,11687],{"class":3452,"line":3489},[3450,11688,11689],{"class":3735},"# 404 — ресурс не існує\n",[3450,11691,11692,11694,11696,11698],{"class":3452,"line":3495},[3450,11693,11364],{"class":4726},[3450,11695,11367],{"class":4711},[3450,11697,11370],{"class":4733},[3450,11699,11700],{"class":4733}," http:\u002F\u002Flocalhost:8081\u002Ftodos\u002F999\n",[3450,11702,11703],{"class":3452,"line":3500},[3450,11704,3475],{"emptyLinePlaceholder":3474},[3450,11706,11707],{"class":3452,"line":3506},[3450,11708,11709],{"class":3735},"# 422 — некоректне тіло\n",[3450,11711,11712,11714,11716,11718],{"class":3452,"line":3511},[3450,11713,11364],{"class":4726},[3450,11715,11367],{"class":4711},[3450,11717,11370],{"class":4733},[3450,11719,11421],{"class":5729},[3450,11721,11722,11724,11726],{"class":3452,"line":3517},[3450,11723,11426],{"class":4711},[3450,11725,11429],{"class":4733},[3450,11727,11421],{"class":5729},[3450,11729,11730,11732,11734],{"class":3452,"line":3522},[3450,11731,11436],{"class":4711},[3450,11733,11439],{"class":4733},[3450,11735,11421],{"class":5729},[3450,11737,11738,11740,11743],{"class":3452,"line":3528},[3450,11739,11446],{"class":4711},[3450,11741,11742],{"class":4733}," '{\"title\": \"\"}'",[3450,11744,11421],{"class":5729},[3450,11746,11747],{"class":3452,"line":3534},[3450,11748,11456],{"class":4733},[3369,11750,7214],{"id":11751},"що-демонструє-цей-приклад-1",[3317,11753,11754,11757,11758,11761,11762,3845],{},[3346,11755,11756],{},"Маршрутизація через регулярні вирази"," є ручною реалізацією того, що ASP.NET Core робить автоматично. ",[3321,11759,11760],{},"RouteItem.Match(path).Groups[\"id\"]"," витягує числовий ідентифікатор із URL ",[3321,11763,11764],{},"\u002Ftodos\u002F42",[3317,11766,11767,11770,11771,11774,11775,11778,11779,11782,11783,4072,11785,4076,11787,3845],{},[3346,11768,11769],{},"Basic аутентифікація через HttpListener."," При ",[3321,11772,11773],{},"AuthenticationSchemes = Basic",", HTTP.sys автоматично перевіряє наявність заголовку ",[3321,11776,11777],{},"Authorization: Basic base64(user:pass)"," і відповідає ",[3321,11780,11781],{},"401 WWW-Authenticate: Basic realm=\"...\""," якщо він відсутній. Успішно автентифікований контекст містить ",[3321,11784,8530],{},[3321,11786,4075],{},[3321,11788,8558],{},[3317,11790,11791,11794,11795,11797,11798,11801],{},[3346,11792,11793],{},"PATCH vs PUT."," Стандарт REST: ",[3321,11796,4177],{}," — повна заміна ресурсу (усі поля обов'язкові), ",[3321,11799,11800],{},"PATCH"," — часткове оновлення (лише змінені поля). Наша реалізація коректно розрізняє їх.",[3317,11803,11804,11807,11808,11810,11811,11814,11815,11818],{},[3346,11805,11806],{},"204 No Content для DELETE."," Успішне видалення не повертає тіло — лише статус ",[3321,11809,8743],{},". Намагання встановити ",[3321,11812,11813],{},"ContentLength64 = 0"," і відповісти ",[3321,11816,11817],{},"200 OK"," із порожнім тілом — поширена помилка.",[3317,11820,11821,11826],{},[3346,11822,11823],{},[3321,11824,11825],{},"Interlocked.Increment"," для атомарного збільшення лічильника ID в умовах паралельної обробки запитів. Без нього два одночасних POST могли б отримати однаковий ID через race condition.",[3422,11828],{},[3312,11830,11832],{"id":11831},"підсумок","Підсумок",[3317,11834,11835,11837],{},[3321,11836,3333],{}," є потужним, але низькорівневим інструментом. У цьому розділі ми вивчили:",[3649,11839,11840,11849,11864,11878,11889,11895],{},[3652,11841,11842,11845,11846,11848],{},[3346,11843,11844],{},"Архітектуру"," та місце ",[3321,11847,3333],{}," у .NET: тонка обгортка над HTTP.sys \u002F сокетами",[3652,11850,11851,4193,11854,3858,11856,3858,11858,11860,11861,11863],{},[3346,11852,11853],{},"URL-префікси",[3321,11855,3787],{},[3321,11857,3685],{},[3321,11859,3700],{},", необхідність ",[3321,11862,3722],{}," для не-localhost адрес",[3652,11865,11866,4193,11869,3858,11871,3858,11873,3858,11875,11877],{},[3346,11867,11868],{},"Повний API",[3321,11870,3333],{},[3321,11872,3981],{},[3321,11874,3359],{},[3321,11876,3363],{}," — всі властивості та методи з описом",[3652,11879,11880,11883,11884,11886,11887],{},[3346,11881,11882],{},"Lifecycle запиту",": від ",[3321,11885,3955],{}," до ",[3321,11888,4637],{},[3652,11890,11891,11894],{},[3346,11892,11893],{},"Приклад 1",": статичний файловий сервер із MIME-типами, path traversal захистом, потоковою передачею та підтримкою HEAD",[3652,11896,11897,11900],{},[3346,11898,11899],{},"Приклад 2",": REST API сервер із маршрутизацією, Basic Auth, CRUD операціями, JSON серіалізацією та стандартними HTTP-кодами",[3343,11902,11903,11904,11907,11908,11910],{},"Для продакшн-сервісів використовуйте ",[3346,11905,11906],{},"ASP.NET Core Kestrel"," — він надає middleware pipeline, dependency injection, автоматичний HTTPS, HTTP\u002F2, production-grade performance та екосистему NuGet. ",[3321,11909,3333],{}," залишається цінним для утиліт, тестових серверів та навчання основам HTTP без абстракцій фреймворку.",[11912,11913,11914],"style",{},"html .light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html.light .shiki span {color: var(--shiki-light);background: var(--shiki-light-bg);font-style: var(--shiki-light-font-style);font-weight: var(--shiki-light-font-weight);text-decoration: var(--shiki-light-text-decoration);}html .default .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .shiki span {color: var(--shiki-default);background: var(--shiki-default-bg);font-style: var(--shiki-default-font-style);font-weight: var(--shiki-default-font-weight);text-decoration: var(--shiki-default-text-decoration);}html .dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html.dark .shiki span {color: var(--shiki-dark);background: var(--shiki-dark-bg);font-style: var(--shiki-dark-font-style);font-weight: var(--shiki-dark-font-weight);text-decoration: var(--shiki-dark-text-decoration);}html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .sD7JJ, html code.shiki .sD7JJ{--shiki-light:#000000FF;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sjcCO, html code.shiki .sjcCO{--shiki-light:#EE0000;--shiki-default:#D7BA7D;--shiki-dark:#D7BA7D}",{"title":3446,"searchDepth":3459,"depth":3459,"links":11916},[11917,11920,11923,11927,11928,11929,11930,11931,11934,11938],{"id":3314,"depth":3459,"text":3315,"children":11918},[11919],{"id":3371,"depth":3465,"text":3372},{"id":3426,"depth":3459,"text":3427,"children":11921},[11922],{"id":3631,"depth":3465,"text":3632},{"id":3793,"depth":3459,"text":3794,"children":11924},[11925,11926],{"id":3797,"depth":3465,"text":3798},{"id":3933,"depth":3465,"text":3934},{"id":4030,"depth":3459,"text":4031},{"id":4084,"depth":3459,"text":4085},{"id":4304,"depth":3459,"text":4305},{"id":4472,"depth":3459,"text":4473},{"id":4652,"depth":3459,"text":4653,"children":11932},[11933],{"id":7213,"depth":3465,"text":7214},{"id":7294,"depth":3459,"text":7295,"children":11935},[11936,11937],{"id":11339,"depth":3465,"text":11340},{"id":11751,"depth":3465,"text":7214},{"id":11831,"depth":3459,"text":11832},"Детальний розгляд класу HttpListener — нативного HTTP-сервера .NET без ASP.NET Core. Архітектура, повний API, конфігурація, аутентифікація, обробка запитів та два практичні приклади — статичний файловий сервер та REST API.","md",null,{},{"title":1684,"description":11939},"KO5luTALSqC5N2NDmrhrio7hjCsF66P8u6DIqpEJnBU",[11946,11948],{"title":1680,"path":1681,"stem":1682,"description":11947,"children":-1},"Глибоке вивчення HTTP — від структури повідомлень та методів до повноцінного REST API клієнта на C#. Академічний розбір RFC 7230–7235, анатомія запитів і відповідей, статус-коди, заголовки та HttpClient екосистема .NET.",{"title":1688,"path":1689,"stem":1690,"description":11949,"children":-1},"Глибоке вивчення механізмів стану в HTTP — cookies, атаки XSS\u002FCSRF\u002Fsession-fixation, Basic\u002FDigest\u002FBearer аутентифікація, JWT алгоритми, OAuth 2.0 з усіма grant-types, PKCE, TLS Handshake, HSTS, HTTP Security Headers, кешування ETag\u002FLast-Modified\u002Fstale-while-revalidate, CORS з credentialed-запитами та content negotiation.",1781795351246]