[{"data":1,"prerenderedAt":9895},["ShallowReactive",2],{"navigation_docs":3,"-csharp-network-programming-udp-broadcast-multicast":3302,"-csharp-network-programming-udp-broadcast-multicast-surround":9890},[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":1676,"body":3304,"description":9884,"extension":9885,"links":9886,"meta":9887,"navigation":3470,"path":1677,"seo":9888,"stem":1678,"__hash__":9889},"docs\u002F01.csharp\u002F13.network-programming\u002F06.udp-broadcast-multicast.md",{"type":3305,"value":3306,"toc":9853},"minimark",[3307,3311,3316,3320,3347,3358,3369,3372,3376,3381,3390,3394,3397,3429,3433,3634,3645,3649,3697,3699,3703,3707,3719,3723,3730,3792,3809,3813,3898,3902,3912,4058,4062,4162,4164,4168,4175,4179,4189,4295,4299,4302,4356,4359,4362,4413,4417,5191,5195,5198,7252,7256,7259,8997,9001,9575,9577,9581,9584,9755,9757,9761,9830,9849],[3308,3309,1676],"h1",{"id":3310},"udp-broadcast-та-multicast",[3312,3313,3315],"h2",{"id":3314},"проблема-групової-доставки","Проблема групової доставки",[3317,3318,3319],"p",{},"У попередній статті ми вивчили базову модель UDP: один відправник — один одержувач (unicast). Але реальний мережевий світ значно складніший. Розглянемо три типові задачі:",[3321,3322,3323,3331,3341],"ol",{},[3324,3325,3326,3330],"li",{},[3327,3328,3329],"strong",{},"Виявлення сервісів",": застосунок запускається і хоче знайти всі принтери\u002Fсервери в локальній мережі — без жодних попередніх знань про їхні IP-адреси.",[3324,3332,3333,3336,3337,3340],{},[3327,3334,3335],{},"Синхронізація годинника",": сервер точного часу хоче одночасно надіслати поточний timestamp ",[3327,3338,3339],{},"тисячам"," клієнтів у мережі.",[3324,3342,3343,3346],{},[3327,3344,3345],{},"Групові оновлення",": система моніторингу розсилає сповіщення про стан усім зацікавленим підписникам.",[3317,3348,3349,3350,3353,3354,3357],{},"Для вирішення цих задач через unicast-UDP довелося б знати IP-адресу кожного одержувача та надіслати N окремих дейтаграм. Це неефективно, а іноді й неможливо. Саме для таких сценаріїв існують два механізми: ",[3327,3351,3352],{},"Broadcast"," та ",[3327,3355,3356],{},"Multicast",".",[3359,3360,3361,3364,3365,3368],"note",{},[3327,3362,3363],{},"Ключова відмінність від unicast:"," При unicast маршрутизатор передає пакет лише одному наступному вузлу. При broadcast\u002Fmulticast мережева інфраструктура сама вирішує, як доставити пакет усім цільовим хостам — відправник надсилає лише ",[3327,3366,3367],{},"один"," пакет.",[3370,3371],"hr",{},[3312,3373,3375],{"id":3374},"udp-broadcast","UDP Broadcast",[3377,3378,3380],"h3",{"id":3379},"що-таке-broadcast","Що таке Broadcast",[3317,3382,3383,3385,3386,3389],{},[3327,3384,3352],{}," — це механізм доставки, при якому одна дейтаграма надсилається ",[3327,3387,3388],{},"усім хостам у мережевому сегменті",". Пакет отримують всі пристрої, підключені до тієї ж підмережі (subnet), незалежно від того, чи хочуть вони його отримати.",[3377,3391,3393],{"id":3392},"типи-broadcast-адрес","Типи broadcast-адрес",[3317,3395,3396],{},"В IPv4 існує два типи broadcast-адрес:",[3398,3399,3400,3418],"field-group",{},[3401,3402,3405,3406,3409,3410,3413,3414,3357],"field",{"name":3403,"type":3404},"Limited Broadcast — 255.255.255.255","тип","Надсилає пакет усім хостам у ",[3327,3407,3408],{},"поточній локальній мережі",". Маршрутизатори ",[3327,3411,3412],{},"ніколи"," не пропускають такі пакети за межі підмережі. Це захист від «broadcast storm» в Інтернеті. В C# це ",[3415,3416,3417],"code",{},"IPAddress.Broadcast",[3401,3419,3405,3421,3424,3425,3428],{"name":3420,"type":3404},"Directed Broadcast — напр. 192.168.1.255",[3327,3422,3423],{},"конкретній підмережі"," (останній октет = 255 для \u002F24). Маршрутизатори ",[3327,3426,3427],{},"можуть"," передавати такі пакети між підмережами (якщо налаштовано), але більшість маршрутизаторів блокує це за замовчуванням (RFC 2644).",[3377,3430,3432],{"id":3431},"як-це-працює-на-рівні-мережі","Як це працює на рівні мережі",[3434,3435,3436],"plant-uml",{},[3437,3438,3443],"pre",{"className":3439,"code":3440,"language":3441,"meta":3442,"style":3442},"language-plantuml shiki shiki-themes light-plus dark-plus dark-plus","@startuml\nskinparam style plain\nskinparam backgroundColor #ffffff\n\nactor \"Відправник\\n192.168.1.10\" as sender\n\npackage \"Локальна мережа 192.168.1.0\u002F24\" #e3f2fd {\n    actor \"Хост A\\n192.168.1.20\" as hostA\n    actor \"Хост B\\n192.168.1.30\" as hostB\n    actor \"Хост C\\n192.168.1.40\" as hostC\n    actor \"Хост D\\n192.168.1.50\" as hostD\n\n    component \"Switch\" as sw #bbdefb\n}\n\ncomponent \"Router\" as router #fff3e0\n\nsender --> sw : UDP → 255.255.255.255:9999\\n(один пакет!)\nsw --> hostA : копія пакету\nsw --> hostB : копія пакету\nsw --> hostC : копія пакету\nsw --> hostD : копія пакету\nsw -[dashed]-> router : ❌ Блокується!\\n(маршрутизатор не\\nпропускає broadcast)\n\nnote right of router\n    Limited broadcast\n    255.255.255.255 ніколи\n    не виходить за межі\n    підмережі.\nend note\n\n@enduml\n","plantuml","",[3415,3444,3445,3453,3459,3465,3472,3478,3483,3489,3495,3501,3507,3513,3518,3524,3530,3535,3541,3546,3552,3558,3564,3570,3576,3582,3587,3593,3599,3605,3611,3617,3623,3628],{"__ignoreMap":3442},[3446,3447,3450],"span",{"class":3448,"line":3449},"line",1,[3446,3451,3452],{},"@startuml\n",[3446,3454,3456],{"class":3448,"line":3455},2,[3446,3457,3458],{},"skinparam style plain\n",[3446,3460,3462],{"class":3448,"line":3461},3,[3446,3463,3464],{},"skinparam backgroundColor #ffffff\n",[3446,3466,3468],{"class":3448,"line":3467},4,[3446,3469,3471],{"emptyLinePlaceholder":3470},true,"\n",[3446,3473,3475],{"class":3448,"line":3474},5,[3446,3476,3477],{},"actor \"Відправник\\n192.168.1.10\" as sender\n",[3446,3479,3481],{"class":3448,"line":3480},6,[3446,3482,3471],{"emptyLinePlaceholder":3470},[3446,3484,3486],{"class":3448,"line":3485},7,[3446,3487,3488],{},"package \"Локальна мережа 192.168.1.0\u002F24\" #e3f2fd {\n",[3446,3490,3492],{"class":3448,"line":3491},8,[3446,3493,3494],{},"    actor \"Хост A\\n192.168.1.20\" as hostA\n",[3446,3496,3498],{"class":3448,"line":3497},9,[3446,3499,3500],{},"    actor \"Хост B\\n192.168.1.30\" as hostB\n",[3446,3502,3504],{"class":3448,"line":3503},10,[3446,3505,3506],{},"    actor \"Хост C\\n192.168.1.40\" as hostC\n",[3446,3508,3510],{"class":3448,"line":3509},11,[3446,3511,3512],{},"    actor \"Хост D\\n192.168.1.50\" as hostD\n",[3446,3514,3516],{"class":3448,"line":3515},12,[3446,3517,3471],{"emptyLinePlaceholder":3470},[3446,3519,3521],{"class":3448,"line":3520},13,[3446,3522,3523],{},"    component \"Switch\" as sw #bbdefb\n",[3446,3525,3527],{"class":3448,"line":3526},14,[3446,3528,3529],{},"}\n",[3446,3531,3533],{"class":3448,"line":3532},15,[3446,3534,3471],{"emptyLinePlaceholder":3470},[3446,3536,3538],{"class":3448,"line":3537},16,[3446,3539,3540],{},"component \"Router\" as router #fff3e0\n",[3446,3542,3544],{"class":3448,"line":3543},17,[3446,3545,3471],{"emptyLinePlaceholder":3470},[3446,3547,3549],{"class":3448,"line":3548},18,[3446,3550,3551],{},"sender --> sw : UDP → 255.255.255.255:9999\\n(один пакет!)\n",[3446,3553,3555],{"class":3448,"line":3554},19,[3446,3556,3557],{},"sw --> hostA : копія пакету\n",[3446,3559,3561],{"class":3448,"line":3560},20,[3446,3562,3563],{},"sw --> hostB : копія пакету\n",[3446,3565,3567],{"class":3448,"line":3566},21,[3446,3568,3569],{},"sw --> hostC : копія пакету\n",[3446,3571,3573],{"class":3448,"line":3572},22,[3446,3574,3575],{},"sw --> hostD : копія пакету\n",[3446,3577,3579],{"class":3448,"line":3578},23,[3446,3580,3581],{},"sw -[dashed]-> router : ❌ Блокується!\\n(маршрутизатор не\\nпропускає broadcast)\n",[3446,3583,3585],{"class":3448,"line":3584},24,[3446,3586,3471],{"emptyLinePlaceholder":3470},[3446,3588,3590],{"class":3448,"line":3589},25,[3446,3591,3592],{},"note right of router\n",[3446,3594,3596],{"class":3448,"line":3595},26,[3446,3597,3598],{},"    Limited broadcast\n",[3446,3600,3602],{"class":3448,"line":3601},27,[3446,3603,3604],{},"    255.255.255.255 ніколи\n",[3446,3606,3608],{"class":3448,"line":3607},28,[3446,3609,3610],{},"    не виходить за межі\n",[3446,3612,3614],{"class":3448,"line":3613},29,[3446,3615,3616],{},"    підмережі.\n",[3446,3618,3620],{"class":3448,"line":3619},30,[3446,3621,3622],{},"end note\n",[3446,3624,3626],{"class":3448,"line":3625},31,[3446,3627,3471],{"emptyLinePlaceholder":3470},[3446,3629,3631],{"class":3448,"line":3630},32,[3446,3632,3633],{},"@enduml\n",[3317,3635,3636,3637,3640,3641,3644],{},"Ключовий момент: ",[3327,3638,3639],{},"switch"," (комутатор) дублює broadcast-фрейм на всі свої порти. Це відбувається на рівні Ethernet (MAC broadcast ",[3415,3642,3643],{},"FF:FF:FF:FF:FF:FF","), ще до того, як хост розпакує IP-пакет.",[3377,3646,3648],{"id":3647},"застосування-broadcast","Застосування Broadcast",[3650,3651,3652,3675],"card-group",{},[3653,3654,3656],"card",{"icon":2772,"title":3655},"🔍 Service Discovery",[3657,3658,3659,3666,3669,3672],"ul",{},[3324,3660,3661,3662,3665],{},"mDNS (Multicast DNS) — пошук ",[3415,3663,3664],{},".local"," імен",[3324,3667,3668],{},"DHCP — клієнт шукає DHCP-сервер при старті",[3324,3670,3671],{},"Ваш чат-клієнт шукає сервер у мережі",[3324,3673,3674],{},"NBNS (NetBIOS Name Service) — Windows",[3653,3676,3679],{"icon":3677,"title":3678},"i-lucide-alert-triangle","⚠️ Обмеження Broadcast",[3657,3680,3681,3688,3691,3694],{},[3324,3682,3683,3684,3687],{},"Отримують ",[3327,3685,3686],{},"усі"," хости, навіть ті, що не цікавляться",[3324,3689,3690],{},"Не перетинає межі підмережі (тільки LAN)",[3324,3692,3693],{},"Відсутній в IPv6 (замінений multicast)",[3324,3695,3696],{},"При зловживанні — «broadcast storm»",[3370,3698],{},[3312,3700,3702],{"id":3701},"udp-multicast","UDP Multicast",[3377,3704,3706],{"id":3705},"що-таке-multicast","Що таке Multicast",[3317,3708,3709,3711,3712,3715,3716,3357],{},[3327,3710,3356],{}," — це цільова групова доставка: пакет отримують ",[3327,3713,3714],{},"лише ті хости, які явно підписались"," на конкретну групу. На відміну від broadcast, multicast може перетинати межі підмережі (маршрутизовний multicast через PIM, DVMRP тощо) і є єдиним механізмом групової доставки в ",[3327,3717,3718],{},"IPv6",[3377,3720,3722],{"id":3721},"multicast-адреси-клас-d","Multicast-адреси (клас D)",[3317,3724,3725,3726,3729],{},"IPv4 резервує адреси класу D — ",[3415,3727,3728],{},"224.0.0.0 – 239.255.255.255"," — виключно для multicast. Вони поділені на діапазони:",[3731,3732,3733,3749],"table",{},[3734,3735,3736],"thead",{},[3737,3738,3739,3743,3746],"tr",{},[3740,3741,3742],"th",{},"Діапазон",[3740,3744,3745],{},"Назва",[3740,3747,3748],{},"Призначення",[3750,3751,3752,3766,3779],"tbody",{},[3737,3753,3754,3760,3763],{},[3755,3756,3757],"td",{},[3415,3758,3759],{},"224.0.0.0 – 224.0.0.255",[3755,3761,3762],{},"Link-Local",[3755,3764,3765],{},"Локальний сегмент, TTL=1. Не маршрутизується.",[3737,3767,3768,3773,3776],{},[3755,3769,3770],{},[3415,3771,3772],{},"224.0.1.0 – 238.255.255.255",[3755,3774,3775],{},"Global scope",[3755,3777,3778],{},"Глобальний multicast (для Інтернету)",[3737,3780,3781,3786,3789],{},[3755,3782,3783],{},[3415,3784,3785],{},"239.0.0.0 – 239.255.255.255",[3755,3787,3788],{},"Site-Local",[3755,3790,3791],{},"Локальний multicast для організації (аналог RFC 1918)",[3793,3794,3795,3796,3800,3801,3804,3805,3808],"tip",{},"Для локальних застосунків завжди використовуйте діапазон ",[3327,3797,3798],{},[3415,3799,3785],{}," (Site-Local). Ці адреси не маршрутизуються в Інтернет, аналогічно ",[3415,3802,3803],{},"192.168.x.x"," для unicast. Наприклад, ",[3415,3806,3807],{},"239.255.0.1"," — хороший вибір для власного сервісу.",[3377,3810,3812],{"id":3811},"відомі-multicast-адреси","Відомі multicast-адреси",[3731,3814,3815,3828],{},[3734,3816,3817],{},[3737,3818,3819,3822,3825],{},[3740,3820,3821],{},"Адреса",[3740,3823,3824],{},"Протокол",[3740,3826,3827],{},"Використання",[3750,3829,3830,3843,3856,3872,3885],{},[3737,3831,3832,3837,3840],{},[3755,3833,3834],{},[3415,3835,3836],{},"224.0.0.1",[3755,3838,3839],{},"All Hosts",[3755,3841,3842],{},"Всі хости у сегменті",[3737,3844,3845,3850,3853],{},[3755,3846,3847],{},[3415,3848,3849],{},"224.0.0.2",[3755,3851,3852],{},"All Routers",[3755,3854,3855],{},"Всі маршрутизатори",[3737,3857,3858,3863,3866],{},[3755,3859,3860],{},[3415,3861,3862],{},"224.0.0.251",[3755,3864,3865],{},"mDNS",[3755,3867,3868,3869,3871],{},"Bonjour \u002F Avahi (",[3415,3870,3664],{},")",[3737,3873,3874,3879,3882],{},[3755,3875,3876],{},[3415,3877,3878],{},"239.255.255.250",[3755,3880,3881],{},"SSDP",[3755,3883,3884],{},"UPnP, пошук пристроїв (Chromecast, TV)",[3737,3886,3887,3892,3895],{},[3755,3888,3889],{},[3415,3890,3891],{},"239.192.0.0",[3755,3893,3894],{},"OpenPGP",[3755,3896,3897],{},"Сервери ключів",[3377,3899,3901],{"id":3900},"як-це-працює-igmp","Як це працює: IGMP",[3317,3903,3904,3905,3908,3909,3911],{},"Щоб отримувати multicast-пакети, хост надсилає ",[3327,3906,3907],{},"IGMP"," (Internet Group Management Protocol) повідомлення маршрутизатору: «Хочу отримувати пакети для групи ",[3415,3910,3807],{},"». Маршрутизатор запам'ятовує це і починає пересилати відповідні multicast-пакети в цю підмережу.",[3434,3913,3914],{},[3437,3915,3917],{"className":3439,"code":3916,"language":3441,"meta":3442,"style":3442},"@startuml\nskinparam style plain\nskinparam backgroundColor #ffffff\n\nactor \"Publisher\\n(відправник)\" as pub\nactor \"Subscriber A\\n(підписаний)\" as subA\nactor \"Subscriber B\\n(підписаний)\" as subB\nactor \"Хост C\\n(не підписаний)\" as hostC\n\ncomponent \"Multicast Router\\n(з IGMP)\" as router #fff3e0\ncomponent \"Switch\" as sw #e3f2fd\n\nsubA --> router : IGMP Join\\n239.255.0.1\nsubB --> router : IGMP Join\\n239.255.0.1\n\npub --> router : UDP → 239.255.0.1:9001\\n(один пакет!)\nrouter --> sw : пересилає до підписників\nsw --> subA : ✅ отримує\nsw --> subB : ✅ отримує\nsw -[dashed]-> hostC : ❌ не отримує\\n(не підписаний)\n\nnote bottom of router\n    Маршрутизатор знає,\n    хто підписаний,\n    через IGMP-повідомлення.\n    Копії пакету створюються\n    лише там, де потрібно.\nend note\n\n@enduml\n",[3415,3918,3919,3923,3927,3931,3935,3940,3945,3950,3955,3959,3964,3969,3973,3978,3983,3987,3992,3997,4002,4007,4012,4016,4021,4026,4031,4036,4041,4046,4050,4054],{"__ignoreMap":3442},[3446,3920,3921],{"class":3448,"line":3449},[3446,3922,3452],{},[3446,3924,3925],{"class":3448,"line":3455},[3446,3926,3458],{},[3446,3928,3929],{"class":3448,"line":3461},[3446,3930,3464],{},[3446,3932,3933],{"class":3448,"line":3467},[3446,3934,3471],{"emptyLinePlaceholder":3470},[3446,3936,3937],{"class":3448,"line":3474},[3446,3938,3939],{},"actor \"Publisher\\n(відправник)\" as pub\n",[3446,3941,3942],{"class":3448,"line":3480},[3446,3943,3944],{},"actor \"Subscriber A\\n(підписаний)\" as subA\n",[3446,3946,3947],{"class":3448,"line":3485},[3446,3948,3949],{},"actor \"Subscriber B\\n(підписаний)\" as subB\n",[3446,3951,3952],{"class":3448,"line":3491},[3446,3953,3954],{},"actor \"Хост C\\n(не підписаний)\" as hostC\n",[3446,3956,3957],{"class":3448,"line":3497},[3446,3958,3471],{"emptyLinePlaceholder":3470},[3446,3960,3961],{"class":3448,"line":3503},[3446,3962,3963],{},"component \"Multicast Router\\n(з IGMP)\" as router #fff3e0\n",[3446,3965,3966],{"class":3448,"line":3509},[3446,3967,3968],{},"component \"Switch\" as sw #e3f2fd\n",[3446,3970,3971],{"class":3448,"line":3515},[3446,3972,3471],{"emptyLinePlaceholder":3470},[3446,3974,3975],{"class":3448,"line":3520},[3446,3976,3977],{},"subA --> router : IGMP Join\\n239.255.0.1\n",[3446,3979,3980],{"class":3448,"line":3526},[3446,3981,3982],{},"subB --> router : IGMP Join\\n239.255.0.1\n",[3446,3984,3985],{"class":3448,"line":3532},[3446,3986,3471],{"emptyLinePlaceholder":3470},[3446,3988,3989],{"class":3448,"line":3537},[3446,3990,3991],{},"pub --> router : UDP → 239.255.0.1:9001\\n(один пакет!)\n",[3446,3993,3994],{"class":3448,"line":3543},[3446,3995,3996],{},"router --> sw : пересилає до підписників\n",[3446,3998,3999],{"class":3448,"line":3548},[3446,4000,4001],{},"sw --> subA : ✅ отримує\n",[3446,4003,4004],{"class":3448,"line":3554},[3446,4005,4006],{},"sw --> subB : ✅ отримує\n",[3446,4008,4009],{"class":3448,"line":3560},[3446,4010,4011],{},"sw -[dashed]-> hostC : ❌ не отримує\\n(не підписаний)\n",[3446,4013,4014],{"class":3448,"line":3566},[3446,4015,3471],{"emptyLinePlaceholder":3470},[3446,4017,4018],{"class":3448,"line":3572},[3446,4019,4020],{},"note bottom of router\n",[3446,4022,4023],{"class":3448,"line":3578},[3446,4024,4025],{},"    Маршрутизатор знає,\n",[3446,4027,4028],{"class":3448,"line":3584},[3446,4029,4030],{},"    хто підписаний,\n",[3446,4032,4033],{"class":3448,"line":3589},[3446,4034,4035],{},"    через IGMP-повідомлення.\n",[3446,4037,4038],{"class":3448,"line":3595},[3446,4039,4040],{},"    Копії пакету створюються\n",[3446,4042,4043],{"class":3448,"line":3601},[3446,4044,4045],{},"    лише там, де потрібно.\n",[3446,4047,4048],{"class":3448,"line":3607},[3446,4049,3622],{},[3446,4051,4052],{"class":3448,"line":3613},[3446,4053,3471],{"emptyLinePlaceholder":3470},[3446,4055,4056],{"class":3448,"line":3619},[3446,4057,3633],{},[3377,4059,4061],{"id":4060},"broadcast-vs-multicast-vs-unicast","Broadcast vs Multicast vs Unicast",[3731,4063,4064,4078],{},[3734,4065,4066],{},[3737,4067,4068,4071,4074,4076],{},[3740,4069,4070],{},"Параметр",[3740,4072,4073],{},"Unicast",[3740,4075,3352],{},[3740,4077,3356],{},[3750,4079,4080,4094,4110,4124,4136,4148],{},[3737,4081,4082,4085,4088,4091],{},[3755,4083,4084],{},"Одержувачі",[3755,4086,4087],{},"1 конкретний",[3755,4089,4090],{},"Всі в підмережі",[3755,4092,4093],{},"Лише підписані",[3737,4095,4096,4099,4102,4105],{},[3755,4097,4098],{},"Масштабованість",[3755,4100,4101],{},"Погана (N пакетів)",[3755,4103,4104],{},"Погана (зайвий трафік)",[3755,4106,4107],{},[3327,4108,4109],{},"Відмінна",[3737,4111,4112,4115,4118,4121],{},[3755,4113,4114],{},"Маршрутизація",[3755,4116,4117],{},"✅",[3755,4119,4120],{},"❌ (в LAN only)",[3755,4122,4123],{},"✅ (при підтримці)",[3737,4125,4126,4128,4130,4133],{},[3755,4127,3718],{},[3755,4129,4117],{},[3755,4131,4132],{},"❌ (відсутній)",[3755,4134,4135],{},"✅ (обов'язковий)",[3737,4137,4138,4141,4144,4146],{},[3755,4139,4140],{},"Контроль підписки",[3755,4142,4143],{},"—",[3755,4145,4143],{},[3755,4147,3907],{},[3737,4149,4150,4153,4156,4159],{},[3755,4151,4152],{},"Типове використання",[3755,4154,4155],{},"HTTP, TCP",[3755,4157,4158],{},"DHCP, ARP",[3755,4160,4161],{},"Відео, mDNS, SSDP",[3370,4163],{},[3312,4165,4167],{"id":4166},"практичний-приклад-сервіс-виявлення-у-мережі","Практичний приклад: сервіс виявлення у мережі",[3317,4169,4170,4171,4174],{},"Побудуємо просту, але реалістичну систему ",[3327,4172,4173],{},"Network Discovery"," — консольний застосунок, що дозволяє автоматично знаходити запущені екземпляри сервісу в локальній мережі без жодної попередньої конфігурації. Ця ж схема лежить в основі Bonjour, SSDP та DNS-SD.",[3377,4176,4178],{"id":4177},"сценарій","Сценарій",[3317,4180,4181,4182,3353,4185,4188],{},"Є два консольних застосунки: ",[3327,4183,4184],{},"Server",[3327,4186,4187],{},"Client",". Клієнт не знає IP сервера. Клієнт надсилає broadcast-запит, сервер відповідає своєю адресою. Клієнт також підписується на multicast-групу, де сервер регулярно анонсує себе.",[3434,4190,4191],{},[3437,4192,4194],{"className":3439,"code":4193,"language":3441,"meta":3442,"style":3442},"@startuml\nskinparam style plain\nskinparam backgroundColor #ffffff\n\nactor \"Discovery Client\" as client\nactor \"Discovery Server A\\n192.168.1.10\" as serverA\nactor \"Discovery Server B\\n192.168.1.20\" as serverB\n\n== Broadcast Discovery ==\n\nclient -> serverA : UDP Broadcast 255.255.255.255:9876\\n\"DISCOVER?\"\nclient -> serverB : (той самий broadcast-пакет)\n\nserverA --> client : UDP unicast → client_ip:ephemeral\\n\"ANNOUNCE:ServerA:192.168.1.10:8080\"\nserverB --> client : UDP unicast → client_ip:ephemeral\\n\"ANNOUNCE:ServerB:192.168.1.20:8080\"\n\n== Multicast Announce (periodical) ==\n\nserverA -> client : UDP Multicast 239.255.99.1:9877\\n\"ANNOUNCE:ServerA:192.168.1.10:8080\"\nserverB -> client : UDP Multicast 239.255.99.1:9877\\n\"ANNOUNCE:ServerB:192.168.1.20:8080\"\n\n@enduml\n",[3415,4195,4196,4200,4204,4208,4212,4217,4222,4227,4231,4236,4240,4245,4250,4254,4259,4264,4268,4273,4277,4282,4287,4291],{"__ignoreMap":3442},[3446,4197,4198],{"class":3448,"line":3449},[3446,4199,3452],{},[3446,4201,4202],{"class":3448,"line":3455},[3446,4203,3458],{},[3446,4205,4206],{"class":3448,"line":3461},[3446,4207,3464],{},[3446,4209,4210],{"class":3448,"line":3467},[3446,4211,3471],{"emptyLinePlaceholder":3470},[3446,4213,4214],{"class":3448,"line":3474},[3446,4215,4216],{},"actor \"Discovery Client\" as client\n",[3446,4218,4219],{"class":3448,"line":3480},[3446,4220,4221],{},"actor \"Discovery Server A\\n192.168.1.10\" as serverA\n",[3446,4223,4224],{"class":3448,"line":3485},[3446,4225,4226],{},"actor \"Discovery Server B\\n192.168.1.20\" as serverB\n",[3446,4228,4229],{"class":3448,"line":3491},[3446,4230,3471],{"emptyLinePlaceholder":3470},[3446,4232,4233],{"class":3448,"line":3497},[3446,4234,4235],{},"== Broadcast Discovery ==\n",[3446,4237,4238],{"class":3448,"line":3503},[3446,4239,3471],{"emptyLinePlaceholder":3470},[3446,4241,4242],{"class":3448,"line":3509},[3446,4243,4244],{},"client -> serverA : UDP Broadcast 255.255.255.255:9876\\n\"DISCOVER?\"\n",[3446,4246,4247],{"class":3448,"line":3515},[3446,4248,4249],{},"client -> serverB : (той самий broadcast-пакет)\n",[3446,4251,4252],{"class":3448,"line":3520},[3446,4253,3471],{"emptyLinePlaceholder":3470},[3446,4255,4256],{"class":3448,"line":3526},[3446,4257,4258],{},"serverA --> client : UDP unicast → client_ip:ephemeral\\n\"ANNOUNCE:ServerA:192.168.1.10:8080\"\n",[3446,4260,4261],{"class":3448,"line":3532},[3446,4262,4263],{},"serverB --> client : UDP unicast → client_ip:ephemeral\\n\"ANNOUNCE:ServerB:192.168.1.20:8080\"\n",[3446,4265,4266],{"class":3448,"line":3537},[3446,4267,3471],{"emptyLinePlaceholder":3470},[3446,4269,4270],{"class":3448,"line":3543},[3446,4271,4272],{},"== Multicast Announce (periodical) ==\n",[3446,4274,4275],{"class":3448,"line":3548},[3446,4276,3471],{"emptyLinePlaceholder":3470},[3446,4278,4279],{"class":3448,"line":3554},[3446,4280,4281],{},"serverA -> client : UDP Multicast 239.255.99.1:9877\\n\"ANNOUNCE:ServerA:192.168.1.10:8080\"\n",[3446,4283,4284],{"class":3448,"line":3560},[3446,4285,4286],{},"serverB -> client : UDP Multicast 239.255.99.1:9877\\n\"ANNOUNCE:ServerB:192.168.1.20:8080\"\n",[3446,4288,4289],{"class":3448,"line":3566},[3446,4290,3471],{"emptyLinePlaceholder":3470},[3446,4292,4293],{"class":3448,"line":3572},[3446,4294,3633],{},[3377,4296,4298],{"id":4297},"структура-проєкту","Структура проєкту",[3317,4300,4301],{},"Створимо один Console-проєкт з двома режимами запуску — через аргумент командного рядка:",[4303,4304,4306,4319,4327,4330,4339,4348],"terminal-preview",{"title":4305},"dotnet new",[4307,4308,4310,4315,4316],"div",{"className":4309},[3448],[3446,4311,4314],{"className":4312},[4313],"opacity-40","$"," ",[3327,4317,4318],{},"dotnet new console -n NetworkDiscovery -o NetworkDiscovery",[4307,4320,4322],{"className":4321},[3448],[3446,4323,4326],{"className":4324},[4325],"text-green-400","The template \"Console App\" was created successfully.",[4307,4328],{"className":4329},[3448],[4307,4331,4333,4315,4336],{"className":4332},[3448],[3446,4334,4314],{"className":4335},[4313],[3327,4337,4338],{},"cd NetworkDiscovery",[4307,4340,4342,4315,4345],{"className":4341},[3448],[3446,4343,4314],{"className":4344},[4313],[3327,4346,4347],{},"dotnet run -- server",[4307,4349,4351],{"className":4350},[3448],[3446,4352,4355],{"className":4353},[4354],"text-blue-400","[Server] Discovery Server запущено. Ctrl+C для зупинки.",[3377,4357,3824],{"id":4358},"протокол",[3317,4360,4361],{},"Протокол гранично простий: два рядкових повідомлення.",[3731,4363,4364,4377],{},[3734,4365,4366],{},[3737,4367,4368,4371,4374],{},[3740,4369,4370],{},"Повідомлення",[3740,4372,4373],{},"Напрям",[3740,4375,4376],{},"Транспорт",[3750,4378,4379,4396],{},[3737,4380,4381,4386,4389],{},[3755,4382,4383],{},[3415,4384,4385],{},"DISCOVER?",[3755,4387,4388],{},"Client → всі",[3755,4390,4391,4315,4393],{},[3327,4392,3352],{},[3415,4394,4395],{},"255.255.255.255:9876",[3737,4397,4398,4403,4406],{},[3755,4399,4400],{},[3415,4401,4402],{},"ANNOUNCE:name:ip:port",[3755,4404,4405],{},"Server → Client",[3755,4407,4408,4409,4412],{},"Unicast (відповідь) ",[3327,4410,4411],{},"або"," Multicast",[3377,4414,4416],{"id":4415},"discoveryprotocolcs-спільні-константи","DiscoveryProtocol.cs — спільні константи",[3437,4418,4423],{"className":4419,"code":4420,"language":4421,"meta":4422,"style":3442},"language-csharp shiki shiki-themes light-plus dark-plus dark-plus","\u002F\u002F DiscoveryProtocol.cs\nnamespace NetworkDiscovery;\n\npublic static class DiscoveryProtocol\n{\n    \u002F\u002F ── Порти ────────────────────────────────────────────────────────────────\n\n    \u002F\u002F\u002F \u003Csummary>Порт для broadcast-запитів DISCOVER?\u003C\u002Fsummary>\n    public const int BroadcastPort = 9876;\n\n    \u002F\u002F\u002F \u003Csummary>Порт для multicast-анонсів ANNOUNCE\u003C\u002Fsummary>\n    public const int MulticastPort = 9877;\n\n    \u002F\u002F ── Адреси ───────────────────────────────────────────────────────────────\n\n    \u002F\u002F\u002F \u003Csummary>\n    \u002F\u002F\u002F Multicast-група для анонсів сервісу.\n    \u002F\u002F\u002F Вибрано з діапазону Site-Local (239.x.x.x),\n    \u002F\u002F\u002F щоб пакети не виходили за межі організації.\n    \u002F\u002F\u002F \u003C\u002Fsummary>\n    public const string MulticastGroupAddress = \"239.255.99.1\";\n\n    \u002F\u002F ── Команди ──────────────────────────────────────────────────────────────\n\n    public const string CmdDiscover = \"DISCOVER?\";\n    public const string CmdAnnounce = \"ANNOUNCE\";\n\n    \u002F\u002F\u002F \u003Csummary>Сервер анонсує себе в multicast кожні 5 секунд.\u003C\u002Fsummary>\n    public const int AnnounceIntervalMs = 5_000;\n\n    \u002F\u002F ── Формування та парсинг ─────────────────────────────────────────────\n\n    \u002F\u002F\u002F \u003Csummary>Формує рядок анонсу: ANNOUNCE:name:ip:port\u003C\u002Fsummary>\n    public static string FormatAnnounce(string name, string ip, int port) =>\n        $\"{CmdAnnounce}:{name}:{ip}:{port}\";\n\n    \u002F\u002F\u002F \u003Csummary>\n    \u002F\u002F\u002F Розбирає ANNOUNCE-рядок.\n    \u002F\u002F\u002F Повертає null, якщо формат некоректний.\n    \u002F\u002F\u002F \u003C\u002Fsummary>\n    public static ServiceInfo? ParseAnnounce(string raw)\n    {\n        \u002F\u002F Очікуємо рівно: ANNOUNCE:name:ip:port\n        var parts = raw.Split(':');\n        if (parts.Length != 4) return null;\n        if (parts[0] != CmdAnnounce) return null;\n        if (!int.TryParse(parts[3], out int port)) return null;\n\n        return new ServiceInfo(parts[1], parts[2], port);\n    }\n}\n\n\u002F\u002F\u002F \u003Csummary>Інформація про знайдений сервіс.\u003C\u002Fsummary>\npublic record ServiceInfo(string Name, string Ip, int Port)\n{\n    public override string ToString() => $\"{Name} @ {Ip}:{Port}\";\n}\n","csharp","showLineNumbers",[3415,4424,4425,4431,4445,4449,4463,4468,4473,4477,4482,4506,4510,4537,4555,4559,4564,4568,4573,4578,4583,4588,4593,4613,4617,4622,4626,4644,4662,4666,4685,4703,4707,4712,4716,4722,4763,4812,4817,4828,4834,4840,4851,4877,4883,4889,4916,4951,4979,5022,5027,5062,5068,5073,5078,5099,5132,5137,5186],{"__ignoreMap":3442},[3446,4426,4427],{"class":3448,"line":3449},[3446,4428,4430],{"class":4429},"spJ8K","\u002F\u002F DiscoveryProtocol.cs\n",[3446,4432,4433,4437,4441],{"class":3448,"line":3455},[3446,4434,4436],{"class":4435},"su1O8","namespace",[3446,4438,4440],{"class":4439},"sN1BT"," NetworkDiscovery",[3446,4442,4444],{"class":4443},"sHH4Y",";\n",[3446,4446,4447],{"class":3448,"line":3461},[3446,4448,3471],{"emptyLinePlaceholder":3470},[3446,4450,4451,4454,4457,4460],{"class":3448,"line":3467},[3446,4452,4453],{"class":4435},"public",[3446,4455,4456],{"class":4435}," static",[3446,4458,4459],{"class":4435}," class",[3446,4461,4462],{"class":4439}," DiscoveryProtocol\n",[3446,4464,4465],{"class":3448,"line":3474},[3446,4466,4467],{"class":4443},"{\n",[3446,4469,4470],{"class":3448,"line":3480},[3446,4471,4472],{"class":4429},"    \u002F\u002F ── Порти ────────────────────────────────────────────────────────────────\n",[3446,4474,4475],{"class":3448,"line":3485},[3446,4476,3471],{"emptyLinePlaceholder":3470},[3446,4478,4479],{"class":3448,"line":3491},[3446,4480,4481],{"class":4429},"    \u002F\u002F\u002F \u003Csummary>Порт для broadcast-запитів DISCOVER?\u003C\u002Fsummary>\n",[3446,4483,4484,4487,4490,4493,4497,4500,4504],{"class":3448,"line":3497},[3446,4485,4486],{"class":4435},"    public",[3446,4488,4489],{"class":4435}," const",[3446,4491,4492],{"class":4435}," int",[3446,4494,4496],{"class":4495},"siwwj"," BroadcastPort",[3446,4498,4499],{"class":4443}," = ",[3446,4501,4503],{"class":4502},"sJj4R","9876",[3446,4505,4444],{"class":4443},[3446,4507,4508],{"class":3448,"line":3503},[3446,4509,3471],{"emptyLinePlaceholder":3470},[3446,4511,4512,4515,4519,4523,4526,4529,4532,4534],{"class":3448,"line":3509},[3446,4513,4514],{"class":4429},"    \u002F\u002F\u002F ",[3446,4516,4518],{"class":4517},"s0P7L","\u003C",[3446,4520,4522],{"class":4521},"sKtos","summary",[3446,4524,4525],{"class":4517},">",[3446,4527,4528],{"class":4429},"Порт для multicast-анонсів ANNOUNCE",[3446,4530,4531],{"class":4517},"\u003C\u002F",[3446,4533,4522],{"class":4521},[3446,4535,4536],{"class":4517},">\n",[3446,4538,4539,4541,4543,4545,4548,4550,4553],{"class":3448,"line":3515},[3446,4540,4486],{"class":4435},[3446,4542,4489],{"class":4435},[3446,4544,4492],{"class":4435},[3446,4546,4547],{"class":4495}," MulticastPort",[3446,4549,4499],{"class":4443},[3446,4551,4552],{"class":4502},"9877",[3446,4554,4444],{"class":4443},[3446,4556,4557],{"class":3448,"line":3520},[3446,4558,3471],{"emptyLinePlaceholder":3470},[3446,4560,4561],{"class":3448,"line":3526},[3446,4562,4563],{"class":4429},"    \u002F\u002F ── Адреси ───────────────────────────────────────────────────────────────\n",[3446,4565,4566],{"class":3448,"line":3532},[3446,4567,3471],{"emptyLinePlaceholder":3470},[3446,4569,4570],{"class":3448,"line":3537},[3446,4571,4572],{"class":4429},"    \u002F\u002F\u002F \u003Csummary>\n",[3446,4574,4575],{"class":3448,"line":3543},[3446,4576,4577],{"class":4429},"    \u002F\u002F\u002F Multicast-група для анонсів сервісу.\n",[3446,4579,4580],{"class":3448,"line":3548},[3446,4581,4582],{"class":4429},"    \u002F\u002F\u002F Вибрано з діапазону Site-Local (239.x.x.x),\n",[3446,4584,4585],{"class":3448,"line":3554},[3446,4586,4587],{"class":4429},"    \u002F\u002F\u002F щоб пакети не виходили за межі організації.\n",[3446,4589,4590],{"class":3448,"line":3560},[3446,4591,4592],{"class":4429},"    \u002F\u002F\u002F \u003C\u002Fsummary>\n",[3446,4594,4595,4597,4599,4602,4605,4607,4611],{"class":3448,"line":3566},[3446,4596,4486],{"class":4435},[3446,4598,4489],{"class":4435},[3446,4600,4601],{"class":4435}," string",[3446,4603,4604],{"class":4495}," MulticastGroupAddress",[3446,4606,4499],{"class":4443},[3446,4608,4610],{"class":4609},"sbdoH","\"239.255.99.1\"",[3446,4612,4444],{"class":4443},[3446,4614,4615],{"class":3448,"line":3572},[3446,4616,3471],{"emptyLinePlaceholder":3470},[3446,4618,4619],{"class":3448,"line":3578},[3446,4620,4621],{"class":4429},"    \u002F\u002F ── Команди ──────────────────────────────────────────────────────────────\n",[3446,4623,4624],{"class":3448,"line":3584},[3446,4625,3471],{"emptyLinePlaceholder":3470},[3446,4627,4628,4630,4632,4634,4637,4639,4642],{"class":3448,"line":3589},[3446,4629,4486],{"class":4435},[3446,4631,4489],{"class":4435},[3446,4633,4601],{"class":4435},[3446,4635,4636],{"class":4495}," CmdDiscover",[3446,4638,4499],{"class":4443},[3446,4640,4641],{"class":4609},"\"DISCOVER?\"",[3446,4643,4444],{"class":4443},[3446,4645,4646,4648,4650,4652,4655,4657,4660],{"class":3448,"line":3595},[3446,4647,4486],{"class":4435},[3446,4649,4489],{"class":4435},[3446,4651,4601],{"class":4435},[3446,4653,4654],{"class":4495}," CmdAnnounce",[3446,4656,4499],{"class":4443},[3446,4658,4659],{"class":4609},"\"ANNOUNCE\"",[3446,4661,4444],{"class":4443},[3446,4663,4664],{"class":3448,"line":3601},[3446,4665,3471],{"emptyLinePlaceholder":3470},[3446,4667,4668,4670,4672,4674,4676,4679,4681,4683],{"class":3448,"line":3607},[3446,4669,4514],{"class":4429},[3446,4671,4518],{"class":4517},[3446,4673,4522],{"class":4521},[3446,4675,4525],{"class":4517},[3446,4677,4678],{"class":4429},"Сервер анонсує себе в multicast кожні 5 секунд.",[3446,4680,4531],{"class":4517},[3446,4682,4522],{"class":4521},[3446,4684,4536],{"class":4517},[3446,4686,4687,4689,4691,4693,4696,4698,4701],{"class":3448,"line":3613},[3446,4688,4486],{"class":4435},[3446,4690,4489],{"class":4435},[3446,4692,4492],{"class":4435},[3446,4694,4695],{"class":4495}," AnnounceIntervalMs",[3446,4697,4499],{"class":4443},[3446,4699,4700],{"class":4502},"5_000",[3446,4702,4444],{"class":4443},[3446,4704,4705],{"class":3448,"line":3619},[3446,4706,3471],{"emptyLinePlaceholder":3470},[3446,4708,4709],{"class":3448,"line":3625},[3446,4710,4711],{"class":4429},"    \u002F\u002F ── Формування та парсинг ─────────────────────────────────────────────\n",[3446,4713,4714],{"class":3448,"line":3630},[3446,4715,3471],{"emptyLinePlaceholder":3470},[3446,4717,4719],{"class":3448,"line":4718},33,[3446,4720,4721],{"class":4429},"    \u002F\u002F\u002F \u003Csummary>Формує рядок анонсу: ANNOUNCE:name:ip:port\u003C\u002Fsummary>\n",[3446,4723,4725,4727,4729,4731,4735,4738,4741,4744,4747,4749,4752,4754,4757,4760],{"class":3448,"line":4724},34,[3446,4726,4486],{"class":4435},[3446,4728,4456],{"class":4435},[3446,4730,4601],{"class":4435},[3446,4732,4734],{"class":4733},"s8Opu"," FormatAnnounce",[3446,4736,4737],{"class":4443},"(",[3446,4739,4740],{"class":4435},"string",[3446,4742,4743],{"class":4495}," name",[3446,4745,4746],{"class":4443},", ",[3446,4748,4740],{"class":4435},[3446,4750,4751],{"class":4495}," ip",[3446,4753,4746],{"class":4443},[3446,4755,4756],{"class":4435},"int",[3446,4758,4759],{"class":4495}," port",[3446,4761,4762],{"class":4443},") =>\n",[3446,4764,4766,4769,4773,4776,4779,4782,4784,4787,4789,4791,4793,4796,4798,4800,4802,4805,4807,4810],{"class":3448,"line":4765},35,[3446,4767,4768],{"class":4609},"        $\"",[3446,4770,4772],{"class":4771},"sD7JJ","{",[3446,4774,4775],{"class":4495},"CmdAnnounce",[3446,4777,4778],{"class":4771},"}",[3446,4780,4781],{"class":4609},":",[3446,4783,4772],{"class":4771},[3446,4785,4786],{"class":4495},"name",[3446,4788,4778],{"class":4771},[3446,4790,4781],{"class":4609},[3446,4792,4772],{"class":4771},[3446,4794,4795],{"class":4495},"ip",[3446,4797,4778],{"class":4771},[3446,4799,4781],{"class":4609},[3446,4801,4772],{"class":4771},[3446,4803,4804],{"class":4495},"port",[3446,4806,4778],{"class":4771},[3446,4808,4809],{"class":4609},"\"",[3446,4811,4444],{"class":4443},[3446,4813,4815],{"class":3448,"line":4814},36,[3446,4816,3471],{"emptyLinePlaceholder":3470},[3446,4818,4820,4822,4824,4826],{"class":3448,"line":4819},37,[3446,4821,4514],{"class":4429},[3446,4823,4518],{"class":4517},[3446,4825,4522],{"class":4521},[3446,4827,4536],{"class":4517},[3446,4829,4831],{"class":3448,"line":4830},38,[3446,4832,4833],{"class":4429},"    \u002F\u002F\u002F Розбирає ANNOUNCE-рядок.\n",[3446,4835,4837],{"class":3448,"line":4836},39,[3446,4838,4839],{"class":4429},"    \u002F\u002F\u002F Повертає null, якщо формат некоректний.\n",[3446,4841,4843,4845,4847,4849],{"class":3448,"line":4842},40,[3446,4844,4514],{"class":4429},[3446,4846,4531],{"class":4517},[3446,4848,4522],{"class":4521},[3446,4850,4536],{"class":4517},[3446,4852,4854,4856,4858,4861,4864,4867,4869,4871,4874],{"class":3448,"line":4853},41,[3446,4855,4486],{"class":4435},[3446,4857,4456],{"class":4435},[3446,4859,4860],{"class":4439}," ServiceInfo",[3446,4862,4863],{"class":4443},"? ",[3446,4865,4866],{"class":4733},"ParseAnnounce",[3446,4868,4737],{"class":4443},[3446,4870,4740],{"class":4435},[3446,4872,4873],{"class":4495}," raw",[3446,4875,4876],{"class":4443},")\n",[3446,4878,4880],{"class":3448,"line":4879},42,[3446,4881,4882],{"class":4443},"    {\n",[3446,4884,4886],{"class":3448,"line":4885},43,[3446,4887,4888],{"class":4429},"        \u002F\u002F Очікуємо рівно: ANNOUNCE:name:ip:port\n",[3446,4890,4892,4895,4898,4900,4903,4905,4908,4910,4913],{"class":3448,"line":4891},44,[3446,4893,4894],{"class":4435},"        var",[3446,4896,4897],{"class":4495}," parts",[3446,4899,4499],{"class":4443},[3446,4901,4902],{"class":4495},"raw",[3446,4904,3357],{"class":4443},[3446,4906,4907],{"class":4733},"Split",[3446,4909,4737],{"class":4443},[3446,4911,4912],{"class":4609},"':'",[3446,4914,4915],{"class":4443},");\n",[3446,4917,4919,4923,4926,4929,4931,4934,4937,4940,4943,4946,4949],{"class":3448,"line":4918},45,[3446,4920,4922],{"class":4921},"s8xlr","        if",[3446,4924,4925],{"class":4443}," (",[3446,4927,4928],{"class":4495},"parts",[3446,4930,3357],{"class":4443},[3446,4932,4933],{"class":4495},"Length",[3446,4935,4936],{"class":4443}," != ",[3446,4938,4939],{"class":4502},"4",[3446,4941,4942],{"class":4443},") ",[3446,4944,4945],{"class":4921},"return",[3446,4947,4948],{"class":4435}," null",[3446,4950,4444],{"class":4443},[3446,4952,4954,4956,4958,4960,4963,4966,4969,4971,4973,4975,4977],{"class":3448,"line":4953},46,[3446,4955,4922],{"class":4921},[3446,4957,4925],{"class":4443},[3446,4959,4928],{"class":4495},[3446,4961,4962],{"class":4443},"[",[3446,4964,4965],{"class":4502},"0",[3446,4967,4968],{"class":4443},"] != ",[3446,4970,4775],{"class":4495},[3446,4972,4942],{"class":4443},[3446,4974,4945],{"class":4921},[3446,4976,4948],{"class":4435},[3446,4978,4444],{"class":4443},[3446,4980,4982,4984,4987,4989,4991,4994,4996,4998,5000,5003,5006,5009,5011,5013,5016,5018,5020],{"class":3448,"line":4981},47,[3446,4983,4922],{"class":4921},[3446,4985,4986],{"class":4443}," (!",[3446,4988,4756],{"class":4435},[3446,4990,3357],{"class":4443},[3446,4992,4993],{"class":4733},"TryParse",[3446,4995,4737],{"class":4443},[3446,4997,4928],{"class":4495},[3446,4999,4962],{"class":4443},[3446,5001,5002],{"class":4502},"3",[3446,5004,5005],{"class":4443},"], ",[3446,5007,5008],{"class":4435},"out",[3446,5010,4492],{"class":4435},[3446,5012,4759],{"class":4495},[3446,5014,5015],{"class":4443},")) ",[3446,5017,4945],{"class":4921},[3446,5019,4948],{"class":4435},[3446,5021,4444],{"class":4443},[3446,5023,5025],{"class":3448,"line":5024},48,[3446,5026,3471],{"emptyLinePlaceholder":3470},[3446,5028,5030,5033,5036,5038,5040,5042,5044,5047,5049,5051,5053,5056,5058,5060],{"class":3448,"line":5029},49,[3446,5031,5032],{"class":4921},"        return",[3446,5034,5035],{"class":4435}," new",[3446,5037,4860],{"class":4439},[3446,5039,4737],{"class":4443},[3446,5041,4928],{"class":4495},[3446,5043,4962],{"class":4443},[3446,5045,5046],{"class":4502},"1",[3446,5048,5005],{"class":4443},[3446,5050,4928],{"class":4495},[3446,5052,4962],{"class":4443},[3446,5054,5055],{"class":4502},"2",[3446,5057,5005],{"class":4443},[3446,5059,4804],{"class":4495},[3446,5061,4915],{"class":4443},[3446,5063,5065],{"class":3448,"line":5064},50,[3446,5066,5067],{"class":4443},"    }\n",[3446,5069,5071],{"class":3448,"line":5070},51,[3446,5072,3529],{"class":4443},[3446,5074,5076],{"class":3448,"line":5075},52,[3446,5077,3471],{"emptyLinePlaceholder":3470},[3446,5079,5081,5084,5086,5088,5090,5093,5095,5097],{"class":3448,"line":5080},53,[3446,5082,5083],{"class":4429},"\u002F\u002F\u002F ",[3446,5085,4518],{"class":4517},[3446,5087,4522],{"class":4521},[3446,5089,4525],{"class":4517},[3446,5091,5092],{"class":4429},"Інформація про знайдений сервіс.",[3446,5094,4531],{"class":4517},[3446,5096,4522],{"class":4521},[3446,5098,4536],{"class":4517},[3446,5100,5102,5104,5107,5109,5111,5113,5116,5118,5120,5123,5125,5127,5130],{"class":3448,"line":5101},54,[3446,5103,4453],{"class":4435},[3446,5105,5106],{"class":4435}," record",[3446,5108,4860],{"class":4439},[3446,5110,4737],{"class":4443},[3446,5112,4740],{"class":4435},[3446,5114,5115],{"class":4495}," Name",[3446,5117,4746],{"class":4443},[3446,5119,4740],{"class":4435},[3446,5121,5122],{"class":4495}," Ip",[3446,5124,4746],{"class":4443},[3446,5126,4756],{"class":4435},[3446,5128,5129],{"class":4495}," Port",[3446,5131,4876],{"class":4443},[3446,5133,5135],{"class":3448,"line":5134},55,[3446,5136,4467],{"class":4443},[3446,5138,5140,5142,5145,5147,5150,5153,5156,5158,5161,5163,5166,5168,5171,5173,5175,5177,5180,5182,5184],{"class":3448,"line":5139},56,[3446,5141,4486],{"class":4435},[3446,5143,5144],{"class":4435}," override",[3446,5146,4601],{"class":4435},[3446,5148,5149],{"class":4733}," ToString",[3446,5151,5152],{"class":4443},"() => ",[3446,5154,5155],{"class":4609},"$\"",[3446,5157,4772],{"class":4771},[3446,5159,5160],{"class":4495},"Name",[3446,5162,4778],{"class":4771},[3446,5164,5165],{"class":4609}," @ ",[3446,5167,4772],{"class":4771},[3446,5169,5170],{"class":4495},"Ip",[3446,5172,4778],{"class":4771},[3446,5174,4781],{"class":4609},[3446,5176,4772],{"class":4771},[3446,5178,5179],{"class":4495},"Port",[3446,5181,4778],{"class":4771},[3446,5183,4809],{"class":4609},[3446,5185,4444],{"class":4443},[3446,5187,5189],{"class":3448,"line":5188},57,[3446,5190,3529],{"class":4443},[3377,5192,5194],{"id":5193},"discoveryservercs-сервер","DiscoveryServer.cs — сервер",[3317,5196,5197],{},"Сервер виконує дві функції паралельно: відповідає на broadcast-запити та регулярно анонсує себе через multicast.",[3437,5199,5201],{"className":4419,"code":5200,"language":4421,"meta":4422,"style":3442},"\u002F\u002F DiscoveryServer.cs\nnamespace NetworkDiscovery;\n\nusing System.Net;\nusing System.Net.NetworkInformation;\nusing System.Net.Sockets;\nusing System.Text;\n\n\u002F\u002F\u002F \u003Csummary>\n\u002F\u002F\u002F Сервер виявлення. Виконує дві функції паралельно:\n\u002F\u002F\u002F 1. Слухає broadcast-порт (9876) і відповідає unicast на DISCOVER? запити.\n\u002F\u002F\u002F 2. Кожні 5 секунд надсилає ANNOUNCE в multicast-групу (239.255.99.1:9877).\n\u002F\u002F\u002F \u003C\u002Fsummary>\npublic sealed class DiscoveryServer : IAsyncDisposable\n{\n    private readonly string _serviceName;\n    private readonly int _servicePort;\n    private readonly CancellationTokenSource _cts = new();\n\n    \u002F\u002F Сокет для прослуховування broadcast-запитів\n    private readonly UdpClient _broadcastListener;\n\n    \u002F\u002F Сокет для надсилання multicast-анонсів\n    private readonly UdpClient _multicastSender;\n\n    private readonly IPEndPoint _multicastEndPoint = new(\n        IPAddress.Parse(DiscoveryProtocol.MulticastGroupAddress),\n        DiscoveryProtocol.MulticastPort);\n\n    public DiscoveryServer(string serviceName, int servicePort)\n    {\n        _serviceName = serviceName;\n        _servicePort = servicePort;\n\n        \u002F\u002F ── Broadcast listener ──────────────────────────────────────────\n        _broadcastListener = new UdpClient();\n        \u002F\u002F ReuseAddress дозволяє кільком процесам слухати той самий порт\n        _broadcastListener.Client.SetSocketOption(\n            SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);\n        _broadcastListener.Client.Bind(\n            new IPEndPoint(IPAddress.Any, DiscoveryProtocol.BroadcastPort));\n        _broadcastListener.EnableBroadcast = true;\n\n        \u002F\u002F ── Multicast sender ────────────────────────────────────────────\n        _multicastSender = new UdpClient();\n        \u002F\u002F TTL=32: пакет може пройти через до 32 маршрутизаторів\n        _multicastSender.Client.SetSocketOption(\n            SocketOptionLevel.IP, SocketOptionName.MulticastTimeToLive, 32);\n    }\n\n    public async Task RunAsync()\n    {\n        string localIp = GetLocalIpAddress();\n        Console.WriteLine($\"[Server] Сервіс '{_serviceName}' на {localIp}:{_servicePort}\");\n        Console.WriteLine($\"[Server] Broadcast listener: :{DiscoveryProtocol.BroadcastPort}\");\n        Console.WriteLine($\"[Server] Multicast announcer: {DiscoveryProtocol.MulticastGroupAddress}:{DiscoveryProtocol.MulticastPort}\");\n        Console.WriteLine(\"[Server] Ctrl+C для зупинки.\");\n\n        Console.CancelKeyPress += (_, e) => { e.Cancel = true; _cts.Cancel(); };\n\n        await Task.WhenAll(\n            BroadcastListenerLoopAsync(localIp, _cts.Token),\n            MulticastAnnouncerLoopAsync(localIp, _cts.Token)\n        );\n\n        Console.WriteLine(\"[Server] Зупинено.\");\n    }\n\n    \u002F\u002F ── Відповідь на broadcast-запити ─────────────────────────────────────────\n\n    private async Task BroadcastListenerLoopAsync(string localIp, CancellationToken ct)\n    {\n        while (!ct.IsCancellationRequested)\n        {\n            UdpReceiveResult result;\n            try { result = await _broadcastListener.ReceiveAsync(ct); }\n            catch (OperationCanceledException) { break; }\n            catch (SocketException ex)\n            {\n                Console.WriteLine($\"[Server] Broadcast помилка: {ex.SocketErrorCode}\");\n                continue;\n            }\n\n            string message = Encoding.UTF8.GetString(result.Buffer).Trim();\n            Console.WriteLine($\"[Server] Broadcast від {result.RemoteEndPoint}: '{message}'\");\n\n            if (message == DiscoveryProtocol.CmdDiscover)\n            {\n                \u002F\u002F Відповідаємо unicast безпосередньо відправнику\n                string announce = DiscoveryProtocol.FormatAnnounce(\n                    _serviceName, localIp, _servicePort);\n                byte[] response = Encoding.UTF8.GetBytes(announce);\n\n                try\n                {\n                    await _broadcastListener.SendAsync(response, result.RemoteEndPoint, ct);\n                    Console.WriteLine($\"[Server] → Unicast відповідь: '{announce}'\");\n                }\n                catch (SocketException ex)\n                {\n                    Console.WriteLine($\"[Server] Помилка відповіді: {ex.SocketErrorCode}\");\n                }\n            }\n        }\n    }\n\n    \u002F\u002F ── Регулярний multicast-анонс ─────────────────────────────────────────────\n\n    private async Task MulticastAnnouncerLoopAsync(string localIp, CancellationToken ct)\n    {\n        string announce = DiscoveryProtocol.FormatAnnounce(\n            _serviceName, localIp, _servicePort);\n        byte[] data = Encoding.UTF8.GetBytes(announce);\n\n        while (!ct.IsCancellationRequested)\n        {\n            try\n            {\n                await _multicastSender.SendAsync(data, _multicastEndPoint, ct);\n                Console.WriteLine($\"[Server] Multicast → '{announce}'\");\n                await Task.Delay(DiscoveryProtocol.AnnounceIntervalMs, ct);\n            }\n            catch (OperationCanceledException) { break; }\n            catch (SocketException ex)\n            {\n                Console.WriteLine($\"[Server] Multicast помилка: {ex.SocketErrorCode}\");\n                await Task.Delay(1000, ct);\n            }\n        }\n    }\n\n    \u002F\u002F ── Визначення локальної IP-адреси ────────────────────────────────────────\n\n    private static string GetLocalIpAddress()\n    {\n        foreach (var ni in NetworkInterface.GetAllNetworkInterfaces())\n        {\n            if (ni.OperationalStatus != OperationalStatus.Up) continue;\n            if (ni.NetworkInterfaceType == NetworkInterfaceType.Loopback) continue;\n\n            foreach (var addr in ni.GetIPProperties().UnicastAddresses)\n            {\n                if (addr.Address.AddressFamily == AddressFamily.InterNetwork)\n                    return addr.Address.ToString();\n            }\n        }\n        return \"127.0.0.1\";\n    }\n\n    public async ValueTask DisposeAsync()\n    {\n        await _cts.CancelAsync();\n        _broadcastListener.Dispose();\n        _multicastSender.Dispose();\n        _cts.Dispose();\n    }\n}\n",[3415,5202,5203,5208,5216,5220,5235,5252,5269,5282,5286,5296,5301,5306,5311,5321,5339,5343,5358,5371,5391,5395,5400,5414,5418,5423,5436,5440,5459,5482,5494,5498,5520,5524,5536,5548,5552,5557,5570,5575,5590,5617,5632,5661,5676,5680,5685,5698,5703,5717,5742,5746,5750,5766,5770,5785,5830,5857,5896,5911,5916,5964,5969,5984,6005,6025,6031,6036,6052,6057,6062,6068,6073,6101,6106,6124,6130,6141,6172,6192,6207,6213,6244,6252,6258,6263,6304,6345,6350,6372,6377,6383,6403,6419,6451,6456,6462,6468,6499,6524,6530,6544,6549,6577,6582,6587,6593,6598,6603,6609,6614,6640,6645,6662,6678,6707,6712,6727,6732,6738,6743,6770,6794,6821,6826,6841,6854,6859,6887,6909,6914,6919,6924,6929,6935,6940,6954,6959,6987,6992,7023,7052,7057,7087,7092,7124,7143,7148,7153,7163,7168,7173,7188,7193,7207,7219,7230,7242,7247],{"__ignoreMap":3442},[3446,5204,5205],{"class":3448,"line":3449},[3446,5206,5207],{"class":4429},"\u002F\u002F DiscoveryServer.cs\n",[3446,5209,5210,5212,5214],{"class":3448,"line":3455},[3446,5211,4436],{"class":4435},[3446,5213,4440],{"class":4439},[3446,5215,4444],{"class":4443},[3446,5217,5218],{"class":3448,"line":3461},[3446,5219,3471],{"emptyLinePlaceholder":3470},[3446,5221,5222,5225,5228,5230,5233],{"class":3448,"line":3467},[3446,5223,5224],{"class":4921},"using",[3446,5226,5227],{"class":4439}," System",[3446,5229,3357],{"class":4443},[3446,5231,5232],{"class":4439},"Net",[3446,5234,4444],{"class":4443},[3446,5236,5237,5239,5241,5243,5245,5247,5250],{"class":3448,"line":3474},[3446,5238,5224],{"class":4921},[3446,5240,5227],{"class":4439},[3446,5242,3357],{"class":4443},[3446,5244,5232],{"class":4439},[3446,5246,3357],{"class":4443},[3446,5248,5249],{"class":4439},"NetworkInformation",[3446,5251,4444],{"class":4443},[3446,5253,5254,5256,5258,5260,5262,5264,5267],{"class":3448,"line":3480},[3446,5255,5224],{"class":4921},[3446,5257,5227],{"class":4439},[3446,5259,3357],{"class":4443},[3446,5261,5232],{"class":4439},[3446,5263,3357],{"class":4443},[3446,5265,5266],{"class":4439},"Sockets",[3446,5268,4444],{"class":4443},[3446,5270,5271,5273,5275,5277,5280],{"class":3448,"line":3485},[3446,5272,5224],{"class":4921},[3446,5274,5227],{"class":4439},[3446,5276,3357],{"class":4443},[3446,5278,5279],{"class":4439},"Text",[3446,5281,4444],{"class":4443},[3446,5283,5284],{"class":3448,"line":3491},[3446,5285,3471],{"emptyLinePlaceholder":3470},[3446,5287,5288,5290,5292,5294],{"class":3448,"line":3497},[3446,5289,5083],{"class":4429},[3446,5291,4518],{"class":4517},[3446,5293,4522],{"class":4521},[3446,5295,4536],{"class":4517},[3446,5297,5298],{"class":3448,"line":3503},[3446,5299,5300],{"class":4429},"\u002F\u002F\u002F Сервер виявлення. Виконує дві функції паралельно:\n",[3446,5302,5303],{"class":3448,"line":3509},[3446,5304,5305],{"class":4429},"\u002F\u002F\u002F 1. Слухає broadcast-порт (9876) і відповідає unicast на DISCOVER? запити.\n",[3446,5307,5308],{"class":3448,"line":3515},[3446,5309,5310],{"class":4429},"\u002F\u002F\u002F 2. Кожні 5 секунд надсилає ANNOUNCE в multicast-групу (239.255.99.1:9877).\n",[3446,5312,5313,5315,5317,5319],{"class":3448,"line":3520},[3446,5314,5083],{"class":4429},[3446,5316,4531],{"class":4517},[3446,5318,4522],{"class":4521},[3446,5320,4536],{"class":4517},[3446,5322,5323,5325,5328,5330,5333,5336],{"class":3448,"line":3526},[3446,5324,4453],{"class":4435},[3446,5326,5327],{"class":4435}," sealed",[3446,5329,4459],{"class":4435},[3446,5331,5332],{"class":4439}," DiscoveryServer",[3446,5334,5335],{"class":4443}," : ",[3446,5337,5338],{"class":4439},"IAsyncDisposable\n",[3446,5340,5341],{"class":3448,"line":3532},[3446,5342,4467],{"class":4443},[3446,5344,5345,5348,5351,5353,5356],{"class":3448,"line":3537},[3446,5346,5347],{"class":4435},"    private",[3446,5349,5350],{"class":4435}," readonly",[3446,5352,4601],{"class":4435},[3446,5354,5355],{"class":4495}," _serviceName",[3446,5357,4444],{"class":4443},[3446,5359,5360,5362,5364,5366,5369],{"class":3448,"line":3543},[3446,5361,5347],{"class":4435},[3446,5363,5350],{"class":4435},[3446,5365,4492],{"class":4435},[3446,5367,5368],{"class":4495}," _servicePort",[3446,5370,4444],{"class":4443},[3446,5372,5373,5375,5377,5380,5383,5385,5388],{"class":3448,"line":3548},[3446,5374,5347],{"class":4435},[3446,5376,5350],{"class":4435},[3446,5378,5379],{"class":4439}," CancellationTokenSource",[3446,5381,5382],{"class":4495}," _cts",[3446,5384,4499],{"class":4443},[3446,5386,5387],{"class":4435},"new",[3446,5389,5390],{"class":4443},"();\n",[3446,5392,5393],{"class":3448,"line":3554},[3446,5394,3471],{"emptyLinePlaceholder":3470},[3446,5396,5397],{"class":3448,"line":3560},[3446,5398,5399],{"class":4429},"    \u002F\u002F Сокет для прослуховування broadcast-запитів\n",[3446,5401,5402,5404,5406,5409,5412],{"class":3448,"line":3566},[3446,5403,5347],{"class":4435},[3446,5405,5350],{"class":4435},[3446,5407,5408],{"class":4439}," UdpClient",[3446,5410,5411],{"class":4495}," _broadcastListener",[3446,5413,4444],{"class":4443},[3446,5415,5416],{"class":3448,"line":3572},[3446,5417,3471],{"emptyLinePlaceholder":3470},[3446,5419,5420],{"class":3448,"line":3578},[3446,5421,5422],{"class":4429},"    \u002F\u002F Сокет для надсилання multicast-анонсів\n",[3446,5424,5425,5427,5429,5431,5434],{"class":3448,"line":3584},[3446,5426,5347],{"class":4435},[3446,5428,5350],{"class":4435},[3446,5430,5408],{"class":4439},[3446,5432,5433],{"class":4495}," _multicastSender",[3446,5435,4444],{"class":4443},[3446,5437,5438],{"class":3448,"line":3589},[3446,5439,3471],{"emptyLinePlaceholder":3470},[3446,5441,5442,5444,5446,5449,5452,5454,5456],{"class":3448,"line":3595},[3446,5443,5347],{"class":4435},[3446,5445,5350],{"class":4435},[3446,5447,5448],{"class":4439}," IPEndPoint",[3446,5450,5451],{"class":4495}," _multicastEndPoint",[3446,5453,4499],{"class":4443},[3446,5455,5387],{"class":4435},[3446,5457,5458],{"class":4443},"(\n",[3446,5460,5461,5464,5466,5469,5471,5474,5476,5479],{"class":3448,"line":3601},[3446,5462,5463],{"class":4495},"        IPAddress",[3446,5465,3357],{"class":4443},[3446,5467,5468],{"class":4733},"Parse",[3446,5470,4737],{"class":4443},[3446,5472,5473],{"class":4495},"DiscoveryProtocol",[3446,5475,3357],{"class":4443},[3446,5477,5478],{"class":4495},"MulticastGroupAddress",[3446,5480,5481],{"class":4443},"),\n",[3446,5483,5484,5487,5489,5492],{"class":3448,"line":3607},[3446,5485,5486],{"class":4495},"        DiscoveryProtocol",[3446,5488,3357],{"class":4443},[3446,5490,5491],{"class":4495},"MulticastPort",[3446,5493,4915],{"class":4443},[3446,5495,5496],{"class":3448,"line":3613},[3446,5497,3471],{"emptyLinePlaceholder":3470},[3446,5499,5500,5502,5504,5506,5508,5511,5513,5515,5518],{"class":3448,"line":3619},[3446,5501,4486],{"class":4435},[3446,5503,5332],{"class":4733},[3446,5505,4737],{"class":4443},[3446,5507,4740],{"class":4435},[3446,5509,5510],{"class":4495}," serviceName",[3446,5512,4746],{"class":4443},[3446,5514,4756],{"class":4435},[3446,5516,5517],{"class":4495}," servicePort",[3446,5519,4876],{"class":4443},[3446,5521,5522],{"class":3448,"line":3625},[3446,5523,4882],{"class":4443},[3446,5525,5526,5529,5531,5534],{"class":3448,"line":3630},[3446,5527,5528],{"class":4495},"        _serviceName",[3446,5530,4499],{"class":4443},[3446,5532,5533],{"class":4495},"serviceName",[3446,5535,4444],{"class":4443},[3446,5537,5538,5541,5543,5546],{"class":3448,"line":4718},[3446,5539,5540],{"class":4495},"        _servicePort",[3446,5542,4499],{"class":4443},[3446,5544,5545],{"class":4495},"servicePort",[3446,5547,4444],{"class":4443},[3446,5549,5550],{"class":3448,"line":4724},[3446,5551,3471],{"emptyLinePlaceholder":3470},[3446,5553,5554],{"class":3448,"line":4765},[3446,5555,5556],{"class":4429},"        \u002F\u002F ── Broadcast listener ──────────────────────────────────────────\n",[3446,5558,5559,5562,5564,5566,5568],{"class":3448,"line":4814},[3446,5560,5561],{"class":4495},"        _broadcastListener",[3446,5563,4499],{"class":4443},[3446,5565,5387],{"class":4435},[3446,5567,5408],{"class":4439},[3446,5569,5390],{"class":4443},[3446,5571,5572],{"class":3448,"line":4819},[3446,5573,5574],{"class":4429},"        \u002F\u002F ReuseAddress дозволяє кільком процесам слухати той самий порт\n",[3446,5576,5577,5579,5581,5583,5585,5588],{"class":3448,"line":4830},[3446,5578,5561],{"class":4495},[3446,5580,3357],{"class":4443},[3446,5582,4187],{"class":4495},[3446,5584,3357],{"class":4443},[3446,5586,5587],{"class":4733},"SetSocketOption",[3446,5589,5458],{"class":4443},[3446,5591,5592,5595,5597,5600,5602,5605,5607,5610,5612,5615],{"class":3448,"line":4836},[3446,5593,5594],{"class":4495},"            SocketOptionLevel",[3446,5596,3357],{"class":4443},[3446,5598,5599],{"class":4495},"Socket",[3446,5601,4746],{"class":4443},[3446,5603,5604],{"class":4495},"SocketOptionName",[3446,5606,3357],{"class":4443},[3446,5608,5609],{"class":4495},"ReuseAddress",[3446,5611,4746],{"class":4443},[3446,5613,5614],{"class":4435},"true",[3446,5616,4915],{"class":4443},[3446,5618,5619,5621,5623,5625,5627,5630],{"class":3448,"line":4842},[3446,5620,5561],{"class":4495},[3446,5622,3357],{"class":4443},[3446,5624,4187],{"class":4495},[3446,5626,3357],{"class":4443},[3446,5628,5629],{"class":4733},"Bind",[3446,5631,5458],{"class":4443},[3446,5633,5634,5637,5639,5641,5644,5646,5649,5651,5653,5655,5658],{"class":3448,"line":4853},[3446,5635,5636],{"class":4435},"            new",[3446,5638,5448],{"class":4439},[3446,5640,4737],{"class":4443},[3446,5642,5643],{"class":4495},"IPAddress",[3446,5645,3357],{"class":4443},[3446,5647,5648],{"class":4495},"Any",[3446,5650,4746],{"class":4443},[3446,5652,5473],{"class":4495},[3446,5654,3357],{"class":4443},[3446,5656,5657],{"class":4495},"BroadcastPort",[3446,5659,5660],{"class":4443},"));\n",[3446,5662,5663,5665,5667,5670,5672,5674],{"class":3448,"line":4879},[3446,5664,5561],{"class":4495},[3446,5666,3357],{"class":4443},[3446,5668,5669],{"class":4495},"EnableBroadcast",[3446,5671,4499],{"class":4443},[3446,5673,5614],{"class":4435},[3446,5675,4444],{"class":4443},[3446,5677,5678],{"class":3448,"line":4885},[3446,5679,3471],{"emptyLinePlaceholder":3470},[3446,5681,5682],{"class":3448,"line":4891},[3446,5683,5684],{"class":4429},"        \u002F\u002F ── Multicast sender ────────────────────────────────────────────\n",[3446,5686,5687,5690,5692,5694,5696],{"class":3448,"line":4918},[3446,5688,5689],{"class":4495},"        _multicastSender",[3446,5691,4499],{"class":4443},[3446,5693,5387],{"class":4435},[3446,5695,5408],{"class":4439},[3446,5697,5390],{"class":4443},[3446,5699,5700],{"class":3448,"line":4953},[3446,5701,5702],{"class":4429},"        \u002F\u002F TTL=32: пакет може пройти через до 32 маршрутизаторів\n",[3446,5704,5705,5707,5709,5711,5713,5715],{"class":3448,"line":4981},[3446,5706,5689],{"class":4495},[3446,5708,3357],{"class":4443},[3446,5710,4187],{"class":4495},[3446,5712,3357],{"class":4443},[3446,5714,5587],{"class":4733},[3446,5716,5458],{"class":4443},[3446,5718,5719,5721,5723,5726,5728,5730,5732,5735,5737,5740],{"class":3448,"line":5024},[3446,5720,5594],{"class":4495},[3446,5722,3357],{"class":4443},[3446,5724,5725],{"class":4495},"IP",[3446,5727,4746],{"class":4443},[3446,5729,5604],{"class":4495},[3446,5731,3357],{"class":4443},[3446,5733,5734],{"class":4495},"MulticastTimeToLive",[3446,5736,4746],{"class":4443},[3446,5738,5739],{"class":4502},"32",[3446,5741,4915],{"class":4443},[3446,5743,5744],{"class":3448,"line":5029},[3446,5745,5067],{"class":4443},[3446,5747,5748],{"class":3448,"line":5064},[3446,5749,3471],{"emptyLinePlaceholder":3470},[3446,5751,5752,5754,5757,5760,5763],{"class":3448,"line":5070},[3446,5753,4486],{"class":4435},[3446,5755,5756],{"class":4435}," async",[3446,5758,5759],{"class":4439}," Task",[3446,5761,5762],{"class":4733}," RunAsync",[3446,5764,5765],{"class":4443},"()\n",[3446,5767,5768],{"class":3448,"line":5075},[3446,5769,4882],{"class":4443},[3446,5771,5772,5775,5778,5780,5783],{"class":3448,"line":5080},[3446,5773,5774],{"class":4435},"        string",[3446,5776,5777],{"class":4495}," localIp",[3446,5779,4499],{"class":4443},[3446,5781,5782],{"class":4733},"GetLocalIpAddress",[3446,5784,5390],{"class":4443},[3446,5786,5787,5790,5792,5795,5797,5800,5802,5805,5807,5810,5812,5815,5817,5819,5821,5824,5826,5828],{"class":3448,"line":5101},[3446,5788,5789],{"class":4495},"        Console",[3446,5791,3357],{"class":4443},[3446,5793,5794],{"class":4733},"WriteLine",[3446,5796,4737],{"class":4443},[3446,5798,5799],{"class":4609},"$\"[Server] Сервіс '",[3446,5801,4772],{"class":4771},[3446,5803,5804],{"class":4495},"_serviceName",[3446,5806,4778],{"class":4771},[3446,5808,5809],{"class":4609},"' на ",[3446,5811,4772],{"class":4771},[3446,5813,5814],{"class":4495},"localIp",[3446,5816,4778],{"class":4771},[3446,5818,4781],{"class":4609},[3446,5820,4772],{"class":4771},[3446,5822,5823],{"class":4495},"_servicePort",[3446,5825,4778],{"class":4771},[3446,5827,4809],{"class":4609},[3446,5829,4915],{"class":4443},[3446,5831,5832,5834,5836,5838,5840,5843,5845,5847,5849,5851,5853,5855],{"class":3448,"line":5134},[3446,5833,5789],{"class":4495},[3446,5835,3357],{"class":4443},[3446,5837,5794],{"class":4733},[3446,5839,4737],{"class":4443},[3446,5841,5842],{"class":4609},"$\"[Server] Broadcast listener: :",[3446,5844,4772],{"class":4771},[3446,5846,5473],{"class":4495},[3446,5848,3357],{"class":4771},[3446,5850,5657],{"class":4495},[3446,5852,4778],{"class":4771},[3446,5854,4809],{"class":4609},[3446,5856,4915],{"class":4443},[3446,5858,5859,5861,5863,5865,5867,5870,5872,5874,5876,5878,5880,5882,5884,5886,5888,5890,5892,5894],{"class":3448,"line":5139},[3446,5860,5789],{"class":4495},[3446,5862,3357],{"class":4443},[3446,5864,5794],{"class":4733},[3446,5866,4737],{"class":4443},[3446,5868,5869],{"class":4609},"$\"[Server] Multicast announcer: ",[3446,5871,4772],{"class":4771},[3446,5873,5473],{"class":4495},[3446,5875,3357],{"class":4771},[3446,5877,5478],{"class":4495},[3446,5879,4778],{"class":4771},[3446,5881,4781],{"class":4609},[3446,5883,4772],{"class":4771},[3446,5885,5473],{"class":4495},[3446,5887,3357],{"class":4771},[3446,5889,5491],{"class":4495},[3446,5891,4778],{"class":4771},[3446,5893,4809],{"class":4609},[3446,5895,4915],{"class":4443},[3446,5897,5898,5900,5902,5904,5906,5909],{"class":3448,"line":5188},[3446,5899,5789],{"class":4495},[3446,5901,3357],{"class":4443},[3446,5903,5794],{"class":4733},[3446,5905,4737],{"class":4443},[3446,5907,5908],{"class":4609},"\"[Server] Ctrl+C для зупинки.\"",[3446,5910,4915],{"class":4443},[3446,5912,5914],{"class":3448,"line":5913},58,[3446,5915,3471],{"emptyLinePlaceholder":3470},[3446,5917,5919,5921,5923,5926,5929,5932,5934,5937,5940,5942,5944,5947,5949,5951,5954,5957,5959,5961],{"class":3448,"line":5918},59,[3446,5920,5789],{"class":4495},[3446,5922,3357],{"class":4443},[3446,5924,5925],{"class":4495},"CancelKeyPress",[3446,5927,5928],{"class":4443}," += (",[3446,5930,5931],{"class":4495},"_",[3446,5933,4746],{"class":4443},[3446,5935,5936],{"class":4495},"e",[3446,5938,5939],{"class":4443},") => { ",[3446,5941,5936],{"class":4495},[3446,5943,3357],{"class":4443},[3446,5945,5946],{"class":4495},"Cancel",[3446,5948,4499],{"class":4443},[3446,5950,5614],{"class":4435},[3446,5952,5953],{"class":4443},"; ",[3446,5955,5956],{"class":4495},"_cts",[3446,5958,3357],{"class":4443},[3446,5960,5946],{"class":4733},[3446,5962,5963],{"class":4443},"(); };\n",[3446,5965,5967],{"class":3448,"line":5966},60,[3446,5968,3471],{"emptyLinePlaceholder":3470},[3446,5970,5972,5975,5977,5979,5982],{"class":3448,"line":5971},61,[3446,5973,5974],{"class":4435},"        await",[3446,5976,5759],{"class":4495},[3446,5978,3357],{"class":4443},[3446,5980,5981],{"class":4733},"WhenAll",[3446,5983,5458],{"class":4443},[3446,5985,5987,5990,5992,5994,5996,5998,6000,6003],{"class":3448,"line":5986},62,[3446,5988,5989],{"class":4733},"            BroadcastListenerLoopAsync",[3446,5991,4737],{"class":4443},[3446,5993,5814],{"class":4495},[3446,5995,4746],{"class":4443},[3446,5997,5956],{"class":4495},[3446,5999,3357],{"class":4443},[3446,6001,6002],{"class":4495},"Token",[3446,6004,5481],{"class":4443},[3446,6006,6008,6011,6013,6015,6017,6019,6021,6023],{"class":3448,"line":6007},63,[3446,6009,6010],{"class":4733},"            MulticastAnnouncerLoopAsync",[3446,6012,4737],{"class":4443},[3446,6014,5814],{"class":4495},[3446,6016,4746],{"class":4443},[3446,6018,5956],{"class":4495},[3446,6020,3357],{"class":4443},[3446,6022,6002],{"class":4495},[3446,6024,4876],{"class":4443},[3446,6026,6028],{"class":3448,"line":6027},64,[3446,6029,6030],{"class":4443},"        );\n",[3446,6032,6034],{"class":3448,"line":6033},65,[3446,6035,3471],{"emptyLinePlaceholder":3470},[3446,6037,6039,6041,6043,6045,6047,6050],{"class":3448,"line":6038},66,[3446,6040,5789],{"class":4495},[3446,6042,3357],{"class":4443},[3446,6044,5794],{"class":4733},[3446,6046,4737],{"class":4443},[3446,6048,6049],{"class":4609},"\"[Server] Зупинено.\"",[3446,6051,4915],{"class":4443},[3446,6053,6055],{"class":3448,"line":6054},67,[3446,6056,5067],{"class":4443},[3446,6058,6060],{"class":3448,"line":6059},68,[3446,6061,3471],{"emptyLinePlaceholder":3470},[3446,6063,6065],{"class":3448,"line":6064},69,[3446,6066,6067],{"class":4429},"    \u002F\u002F ── Відповідь на broadcast-запити ─────────────────────────────────────────\n",[3446,6069,6071],{"class":3448,"line":6070},70,[3446,6072,3471],{"emptyLinePlaceholder":3470},[3446,6074,6076,6078,6080,6082,6085,6087,6089,6091,6093,6096,6099],{"class":3448,"line":6075},71,[3446,6077,5347],{"class":4435},[3446,6079,5756],{"class":4435},[3446,6081,5759],{"class":4439},[3446,6083,6084],{"class":4733}," BroadcastListenerLoopAsync",[3446,6086,4737],{"class":4443},[3446,6088,4740],{"class":4435},[3446,6090,5777],{"class":4495},[3446,6092,4746],{"class":4443},[3446,6094,6095],{"class":4439},"CancellationToken",[3446,6097,6098],{"class":4495}," ct",[3446,6100,4876],{"class":4443},[3446,6102,6104],{"class":3448,"line":6103},72,[3446,6105,4882],{"class":4443},[3446,6107,6109,6112,6114,6117,6119,6122],{"class":3448,"line":6108},73,[3446,6110,6111],{"class":4921},"        while",[3446,6113,4986],{"class":4443},[3446,6115,6116],{"class":4495},"ct",[3446,6118,3357],{"class":4443},[3446,6120,6121],{"class":4495},"IsCancellationRequested",[3446,6123,4876],{"class":4443},[3446,6125,6127],{"class":3448,"line":6126},74,[3446,6128,6129],{"class":4443},"        {\n",[3446,6131,6133,6136,6139],{"class":3448,"line":6132},75,[3446,6134,6135],{"class":4439},"            UdpReceiveResult",[3446,6137,6138],{"class":4495}," result",[3446,6140,4444],{"class":4443},[3446,6142,6144,6147,6150,6153,6155,6158,6160,6162,6165,6167,6169],{"class":3448,"line":6143},76,[3446,6145,6146],{"class":4921},"            try",[3446,6148,6149],{"class":4443}," { ",[3446,6151,6152],{"class":4495},"result",[3446,6154,4499],{"class":4443},[3446,6156,6157],{"class":4435},"await",[3446,6159,5411],{"class":4495},[3446,6161,3357],{"class":4443},[3446,6163,6164],{"class":4733},"ReceiveAsync",[3446,6166,4737],{"class":4443},[3446,6168,6116],{"class":4495},[3446,6170,6171],{"class":4443},"); }\n",[3446,6173,6175,6178,6180,6183,6186,6189],{"class":3448,"line":6174},77,[3446,6176,6177],{"class":4921},"            catch",[3446,6179,4925],{"class":4443},[3446,6181,6182],{"class":4439},"OperationCanceledException",[3446,6184,6185],{"class":4443},") { ",[3446,6187,6188],{"class":4921},"break",[3446,6190,6191],{"class":4443},"; }\n",[3446,6193,6195,6197,6199,6202,6205],{"class":3448,"line":6194},78,[3446,6196,6177],{"class":4921},[3446,6198,4925],{"class":4443},[3446,6200,6201],{"class":4439},"SocketException",[3446,6203,6204],{"class":4495}," ex",[3446,6206,4876],{"class":4443},[3446,6208,6210],{"class":3448,"line":6209},79,[3446,6211,6212],{"class":4443},"            {\n",[3446,6214,6216,6219,6221,6223,6225,6228,6230,6233,6235,6238,6240,6242],{"class":3448,"line":6215},80,[3446,6217,6218],{"class":4495},"                Console",[3446,6220,3357],{"class":4443},[3446,6222,5794],{"class":4733},[3446,6224,4737],{"class":4443},[3446,6226,6227],{"class":4609},"$\"[Server] Broadcast помилка: ",[3446,6229,4772],{"class":4771},[3446,6231,6232],{"class":4495},"ex",[3446,6234,3357],{"class":4771},[3446,6236,6237],{"class":4495},"SocketErrorCode",[3446,6239,4778],{"class":4771},[3446,6241,4809],{"class":4609},[3446,6243,4915],{"class":4443},[3446,6245,6247,6250],{"class":3448,"line":6246},81,[3446,6248,6249],{"class":4921},"                continue",[3446,6251,4444],{"class":4443},[3446,6253,6255],{"class":3448,"line":6254},82,[3446,6256,6257],{"class":4443},"            }\n",[3446,6259,6261],{"class":3448,"line":6260},83,[3446,6262,3471],{"emptyLinePlaceholder":3470},[3446,6264,6266,6269,6272,6274,6277,6279,6282,6284,6287,6289,6291,6293,6296,6299,6302],{"class":3448,"line":6265},84,[3446,6267,6268],{"class":4435},"            string",[3446,6270,6271],{"class":4495}," message",[3446,6273,4499],{"class":4443},[3446,6275,6276],{"class":4495},"Encoding",[3446,6278,3357],{"class":4443},[3446,6280,6281],{"class":4495},"UTF8",[3446,6283,3357],{"class":4443},[3446,6285,6286],{"class":4733},"GetString",[3446,6288,4737],{"class":4443},[3446,6290,6152],{"class":4495},[3446,6292,3357],{"class":4443},[3446,6294,6295],{"class":4495},"Buffer",[3446,6297,6298],{"class":4443},").",[3446,6300,6301],{"class":4733},"Trim",[3446,6303,5390],{"class":4443},[3446,6305,6307,6310,6312,6314,6316,6319,6321,6323,6325,6328,6330,6333,6335,6338,6340,6343],{"class":3448,"line":6306},85,[3446,6308,6309],{"class":4495},"            Console",[3446,6311,3357],{"class":4443},[3446,6313,5794],{"class":4733},[3446,6315,4737],{"class":4443},[3446,6317,6318],{"class":4609},"$\"[Server] Broadcast від ",[3446,6320,4772],{"class":4771},[3446,6322,6152],{"class":4495},[3446,6324,3357],{"class":4771},[3446,6326,6327],{"class":4495},"RemoteEndPoint",[3446,6329,4778],{"class":4771},[3446,6331,6332],{"class":4609},": '",[3446,6334,4772],{"class":4771},[3446,6336,6337],{"class":4495},"message",[3446,6339,4778],{"class":4771},[3446,6341,6342],{"class":4609},"'\"",[3446,6344,4915],{"class":4443},[3446,6346,6348],{"class":3448,"line":6347},86,[3446,6349,3471],{"emptyLinePlaceholder":3470},[3446,6351,6353,6356,6358,6360,6363,6365,6367,6370],{"class":3448,"line":6352},87,[3446,6354,6355],{"class":4921},"            if",[3446,6357,4925],{"class":4443},[3446,6359,6337],{"class":4495},[3446,6361,6362],{"class":4443}," == ",[3446,6364,5473],{"class":4495},[3446,6366,3357],{"class":4443},[3446,6368,6369],{"class":4495},"CmdDiscover",[3446,6371,4876],{"class":4443},[3446,6373,6375],{"class":3448,"line":6374},88,[3446,6376,6212],{"class":4443},[3446,6378,6380],{"class":3448,"line":6379},89,[3446,6381,6382],{"class":4429},"                \u002F\u002F Відповідаємо unicast безпосередньо відправнику\n",[3446,6384,6386,6389,6392,6394,6396,6398,6401],{"class":3448,"line":6385},90,[3446,6387,6388],{"class":4435},"                string",[3446,6390,6391],{"class":4495}," announce",[3446,6393,4499],{"class":4443},[3446,6395,5473],{"class":4495},[3446,6397,3357],{"class":4443},[3446,6399,6400],{"class":4733},"FormatAnnounce",[3446,6402,5458],{"class":4443},[3446,6404,6406,6409,6411,6413,6415,6417],{"class":3448,"line":6405},91,[3446,6407,6408],{"class":4495},"                    _serviceName",[3446,6410,4746],{"class":4443},[3446,6412,5814],{"class":4495},[3446,6414,4746],{"class":4443},[3446,6416,5823],{"class":4495},[3446,6418,4915],{"class":4443},[3446,6420,6422,6425,6428,6431,6433,6435,6437,6439,6441,6444,6446,6449],{"class":3448,"line":6421},92,[3446,6423,6424],{"class":4435},"                byte",[3446,6426,6427],{"class":4443},"[] ",[3446,6429,6430],{"class":4495},"response",[3446,6432,4499],{"class":4443},[3446,6434,6276],{"class":4495},[3446,6436,3357],{"class":4443},[3446,6438,6281],{"class":4495},[3446,6440,3357],{"class":4443},[3446,6442,6443],{"class":4733},"GetBytes",[3446,6445,4737],{"class":4443},[3446,6447,6448],{"class":4495},"announce",[3446,6450,4915],{"class":4443},[3446,6452,6454],{"class":3448,"line":6453},93,[3446,6455,3471],{"emptyLinePlaceholder":3470},[3446,6457,6459],{"class":3448,"line":6458},94,[3446,6460,6461],{"class":4921},"                try\n",[3446,6463,6465],{"class":3448,"line":6464},95,[3446,6466,6467],{"class":4443},"                {\n",[3446,6469,6471,6474,6476,6478,6481,6483,6485,6487,6489,6491,6493,6495,6497],{"class":3448,"line":6470},96,[3446,6472,6473],{"class":4435},"                    await",[3446,6475,5411],{"class":4495},[3446,6477,3357],{"class":4443},[3446,6479,6480],{"class":4733},"SendAsync",[3446,6482,4737],{"class":4443},[3446,6484,6430],{"class":4495},[3446,6486,4746],{"class":4443},[3446,6488,6152],{"class":4495},[3446,6490,3357],{"class":4443},[3446,6492,6327],{"class":4495},[3446,6494,4746],{"class":4443},[3446,6496,6116],{"class":4495},[3446,6498,4915],{"class":4443},[3446,6500,6502,6505,6507,6509,6511,6514,6516,6518,6520,6522],{"class":3448,"line":6501},97,[3446,6503,6504],{"class":4495},"                    Console",[3446,6506,3357],{"class":4443},[3446,6508,5794],{"class":4733},[3446,6510,4737],{"class":4443},[3446,6512,6513],{"class":4609},"$\"[Server] → Unicast відповідь: '",[3446,6515,4772],{"class":4771},[3446,6517,6448],{"class":4495},[3446,6519,4778],{"class":4771},[3446,6521,6342],{"class":4609},[3446,6523,4915],{"class":4443},[3446,6525,6527],{"class":3448,"line":6526},98,[3446,6528,6529],{"class":4443},"                }\n",[3446,6531,6533,6536,6538,6540,6542],{"class":3448,"line":6532},99,[3446,6534,6535],{"class":4921},"                catch",[3446,6537,4925],{"class":4443},[3446,6539,6201],{"class":4439},[3446,6541,6204],{"class":4495},[3446,6543,4876],{"class":4443},[3446,6545,6547],{"class":3448,"line":6546},100,[3446,6548,6467],{"class":4443},[3446,6550,6552,6554,6556,6558,6560,6563,6565,6567,6569,6571,6573,6575],{"class":3448,"line":6551},101,[3446,6553,6504],{"class":4495},[3446,6555,3357],{"class":4443},[3446,6557,5794],{"class":4733},[3446,6559,4737],{"class":4443},[3446,6561,6562],{"class":4609},"$\"[Server] Помилка відповіді: ",[3446,6564,4772],{"class":4771},[3446,6566,6232],{"class":4495},[3446,6568,3357],{"class":4771},[3446,6570,6237],{"class":4495},[3446,6572,4778],{"class":4771},[3446,6574,4809],{"class":4609},[3446,6576,4915],{"class":4443},[3446,6578,6580],{"class":3448,"line":6579},102,[3446,6581,6529],{"class":4443},[3446,6583,6585],{"class":3448,"line":6584},103,[3446,6586,6257],{"class":4443},[3446,6588,6590],{"class":3448,"line":6589},104,[3446,6591,6592],{"class":4443},"        }\n",[3446,6594,6596],{"class":3448,"line":6595},105,[3446,6597,5067],{"class":4443},[3446,6599,6601],{"class":3448,"line":6600},106,[3446,6602,3471],{"emptyLinePlaceholder":3470},[3446,6604,6606],{"class":3448,"line":6605},107,[3446,6607,6608],{"class":4429},"    \u002F\u002F ── Регулярний multicast-анонс ─────────────────────────────────────────────\n",[3446,6610,6612],{"class":3448,"line":6611},108,[3446,6613,3471],{"emptyLinePlaceholder":3470},[3446,6615,6617,6619,6621,6623,6626,6628,6630,6632,6634,6636,6638],{"class":3448,"line":6616},109,[3446,6618,5347],{"class":4435},[3446,6620,5756],{"class":4435},[3446,6622,5759],{"class":4439},[3446,6624,6625],{"class":4733}," MulticastAnnouncerLoopAsync",[3446,6627,4737],{"class":4443},[3446,6629,4740],{"class":4435},[3446,6631,5777],{"class":4495},[3446,6633,4746],{"class":4443},[3446,6635,6095],{"class":4439},[3446,6637,6098],{"class":4495},[3446,6639,4876],{"class":4443},[3446,6641,6643],{"class":3448,"line":6642},110,[3446,6644,4882],{"class":4443},[3446,6646,6648,6650,6652,6654,6656,6658,6660],{"class":3448,"line":6647},111,[3446,6649,5774],{"class":4435},[3446,6651,6391],{"class":4495},[3446,6653,4499],{"class":4443},[3446,6655,5473],{"class":4495},[3446,6657,3357],{"class":4443},[3446,6659,6400],{"class":4733},[3446,6661,5458],{"class":4443},[3446,6663,6665,6668,6670,6672,6674,6676],{"class":3448,"line":6664},112,[3446,6666,6667],{"class":4495},"            _serviceName",[3446,6669,4746],{"class":4443},[3446,6671,5814],{"class":4495},[3446,6673,4746],{"class":4443},[3446,6675,5823],{"class":4495},[3446,6677,4915],{"class":4443},[3446,6679,6681,6684,6686,6689,6691,6693,6695,6697,6699,6701,6703,6705],{"class":3448,"line":6680},113,[3446,6682,6683],{"class":4435},"        byte",[3446,6685,6427],{"class":4443},[3446,6687,6688],{"class":4495},"data",[3446,6690,4499],{"class":4443},[3446,6692,6276],{"class":4495},[3446,6694,3357],{"class":4443},[3446,6696,6281],{"class":4495},[3446,6698,3357],{"class":4443},[3446,6700,6443],{"class":4733},[3446,6702,4737],{"class":4443},[3446,6704,6448],{"class":4495},[3446,6706,4915],{"class":4443},[3446,6708,6710],{"class":3448,"line":6709},114,[3446,6711,3471],{"emptyLinePlaceholder":3470},[3446,6713,6715,6717,6719,6721,6723,6725],{"class":3448,"line":6714},115,[3446,6716,6111],{"class":4921},[3446,6718,4986],{"class":4443},[3446,6720,6116],{"class":4495},[3446,6722,3357],{"class":4443},[3446,6724,6121],{"class":4495},[3446,6726,4876],{"class":4443},[3446,6728,6730],{"class":3448,"line":6729},116,[3446,6731,6129],{"class":4443},[3446,6733,6735],{"class":3448,"line":6734},117,[3446,6736,6737],{"class":4921},"            try\n",[3446,6739,6741],{"class":3448,"line":6740},118,[3446,6742,6212],{"class":4443},[3446,6744,6746,6749,6751,6753,6755,6757,6759,6761,6764,6766,6768],{"class":3448,"line":6745},119,[3446,6747,6748],{"class":4435},"                await",[3446,6750,5433],{"class":4495},[3446,6752,3357],{"class":4443},[3446,6754,6480],{"class":4733},[3446,6756,4737],{"class":4443},[3446,6758,6688],{"class":4495},[3446,6760,4746],{"class":4443},[3446,6762,6763],{"class":4495},"_multicastEndPoint",[3446,6765,4746],{"class":4443},[3446,6767,6116],{"class":4495},[3446,6769,4915],{"class":4443},[3446,6771,6773,6775,6777,6779,6781,6784,6786,6788,6790,6792],{"class":3448,"line":6772},120,[3446,6774,6218],{"class":4495},[3446,6776,3357],{"class":4443},[3446,6778,5794],{"class":4733},[3446,6780,4737],{"class":4443},[3446,6782,6783],{"class":4609},"$\"[Server] Multicast → '",[3446,6785,4772],{"class":4771},[3446,6787,6448],{"class":4495},[3446,6789,4778],{"class":4771},[3446,6791,6342],{"class":4609},[3446,6793,4915],{"class":4443},[3446,6795,6797,6799,6801,6803,6806,6808,6810,6812,6815,6817,6819],{"class":3448,"line":6796},121,[3446,6798,6748],{"class":4435},[3446,6800,5759],{"class":4495},[3446,6802,3357],{"class":4443},[3446,6804,6805],{"class":4733},"Delay",[3446,6807,4737],{"class":4443},[3446,6809,5473],{"class":4495},[3446,6811,3357],{"class":4443},[3446,6813,6814],{"class":4495},"AnnounceIntervalMs",[3446,6816,4746],{"class":4443},[3446,6818,6116],{"class":4495},[3446,6820,4915],{"class":4443},[3446,6822,6824],{"class":3448,"line":6823},122,[3446,6825,6257],{"class":4443},[3446,6827,6829,6831,6833,6835,6837,6839],{"class":3448,"line":6828},123,[3446,6830,6177],{"class":4921},[3446,6832,4925],{"class":4443},[3446,6834,6182],{"class":4439},[3446,6836,6185],{"class":4443},[3446,6838,6188],{"class":4921},[3446,6840,6191],{"class":4443},[3446,6842,6844,6846,6848,6850,6852],{"class":3448,"line":6843},124,[3446,6845,6177],{"class":4921},[3446,6847,4925],{"class":4443},[3446,6849,6201],{"class":4439},[3446,6851,6204],{"class":4495},[3446,6853,4876],{"class":4443},[3446,6855,6857],{"class":3448,"line":6856},125,[3446,6858,6212],{"class":4443},[3446,6860,6862,6864,6866,6868,6870,6873,6875,6877,6879,6881,6883,6885],{"class":3448,"line":6861},126,[3446,6863,6218],{"class":4495},[3446,6865,3357],{"class":4443},[3446,6867,5794],{"class":4733},[3446,6869,4737],{"class":4443},[3446,6871,6872],{"class":4609},"$\"[Server] Multicast помилка: ",[3446,6874,4772],{"class":4771},[3446,6876,6232],{"class":4495},[3446,6878,3357],{"class":4771},[3446,6880,6237],{"class":4495},[3446,6882,4778],{"class":4771},[3446,6884,4809],{"class":4609},[3446,6886,4915],{"class":4443},[3446,6888,6890,6892,6894,6896,6898,6900,6903,6905,6907],{"class":3448,"line":6889},127,[3446,6891,6748],{"class":4435},[3446,6893,5759],{"class":4495},[3446,6895,3357],{"class":4443},[3446,6897,6805],{"class":4733},[3446,6899,4737],{"class":4443},[3446,6901,6902],{"class":4502},"1000",[3446,6904,4746],{"class":4443},[3446,6906,6116],{"class":4495},[3446,6908,4915],{"class":4443},[3446,6910,6912],{"class":3448,"line":6911},128,[3446,6913,6257],{"class":4443},[3446,6915,6917],{"class":3448,"line":6916},129,[3446,6918,6592],{"class":4443},[3446,6920,6922],{"class":3448,"line":6921},130,[3446,6923,5067],{"class":4443},[3446,6925,6927],{"class":3448,"line":6926},131,[3446,6928,3471],{"emptyLinePlaceholder":3470},[3446,6930,6932],{"class":3448,"line":6931},132,[3446,6933,6934],{"class":4429},"    \u002F\u002F ── Визначення локальної IP-адреси ────────────────────────────────────────\n",[3446,6936,6938],{"class":3448,"line":6937},133,[3446,6939,3471],{"emptyLinePlaceholder":3470},[3446,6941,6943,6945,6947,6949,6952],{"class":3448,"line":6942},134,[3446,6944,5347],{"class":4435},[3446,6946,4456],{"class":4435},[3446,6948,4601],{"class":4435},[3446,6950,6951],{"class":4733}," GetLocalIpAddress",[3446,6953,5765],{"class":4443},[3446,6955,6957],{"class":3448,"line":6956},135,[3446,6958,4882],{"class":4443},[3446,6960,6962,6965,6967,6970,6973,6976,6979,6981,6984],{"class":3448,"line":6961},136,[3446,6963,6964],{"class":4921},"        foreach",[3446,6966,4925],{"class":4443},[3446,6968,6969],{"class":4435},"var",[3446,6971,6972],{"class":4495}," ni",[3446,6974,6975],{"class":4921}," in",[3446,6977,6978],{"class":4495}," NetworkInterface",[3446,6980,3357],{"class":4443},[3446,6982,6983],{"class":4733},"GetAllNetworkInterfaces",[3446,6985,6986],{"class":4443},"())\n",[3446,6988,6990],{"class":3448,"line":6989},137,[3446,6991,6129],{"class":4443},[3446,6993,6995,6997,6999,7002,7004,7007,7009,7011,7013,7016,7018,7021],{"class":3448,"line":6994},138,[3446,6996,6355],{"class":4921},[3446,6998,4925],{"class":4443},[3446,7000,7001],{"class":4495},"ni",[3446,7003,3357],{"class":4443},[3446,7005,7006],{"class":4495},"OperationalStatus",[3446,7008,4936],{"class":4443},[3446,7010,7006],{"class":4495},[3446,7012,3357],{"class":4443},[3446,7014,7015],{"class":4495},"Up",[3446,7017,4942],{"class":4443},[3446,7019,7020],{"class":4921},"continue",[3446,7022,4444],{"class":4443},[3446,7024,7026,7028,7030,7032,7034,7037,7039,7041,7043,7046,7048,7050],{"class":3448,"line":7025},139,[3446,7027,6355],{"class":4921},[3446,7029,4925],{"class":4443},[3446,7031,7001],{"class":4495},[3446,7033,3357],{"class":4443},[3446,7035,7036],{"class":4495},"NetworkInterfaceType",[3446,7038,6362],{"class":4443},[3446,7040,7036],{"class":4495},[3446,7042,3357],{"class":4443},[3446,7044,7045],{"class":4495},"Loopback",[3446,7047,4942],{"class":4443},[3446,7049,7020],{"class":4921},[3446,7051,4444],{"class":4443},[3446,7053,7055],{"class":3448,"line":7054},140,[3446,7056,3471],{"emptyLinePlaceholder":3470},[3446,7058,7060,7063,7065,7067,7070,7072,7074,7076,7079,7082,7085],{"class":3448,"line":7059},141,[3446,7061,7062],{"class":4921},"            foreach",[3446,7064,4925],{"class":4443},[3446,7066,6969],{"class":4435},[3446,7068,7069],{"class":4495}," addr",[3446,7071,6975],{"class":4921},[3446,7073,6972],{"class":4495},[3446,7075,3357],{"class":4443},[3446,7077,7078],{"class":4733},"GetIPProperties",[3446,7080,7081],{"class":4443},"().",[3446,7083,7084],{"class":4495},"UnicastAddresses",[3446,7086,4876],{"class":4443},[3446,7088,7090],{"class":3448,"line":7089},142,[3446,7091,6212],{"class":4443},[3446,7093,7095,7098,7100,7103,7105,7108,7110,7113,7115,7117,7119,7122],{"class":3448,"line":7094},143,[3446,7096,7097],{"class":4921},"                if",[3446,7099,4925],{"class":4443},[3446,7101,7102],{"class":4495},"addr",[3446,7104,3357],{"class":4443},[3446,7106,7107],{"class":4495},"Address",[3446,7109,3357],{"class":4443},[3446,7111,7112],{"class":4495},"AddressFamily",[3446,7114,6362],{"class":4443},[3446,7116,7112],{"class":4495},[3446,7118,3357],{"class":4443},[3446,7120,7121],{"class":4495},"InterNetwork",[3446,7123,4876],{"class":4443},[3446,7125,7127,7130,7132,7134,7136,7138,7141],{"class":3448,"line":7126},144,[3446,7128,7129],{"class":4921},"                    return",[3446,7131,7069],{"class":4495},[3446,7133,3357],{"class":4443},[3446,7135,7107],{"class":4495},[3446,7137,3357],{"class":4443},[3446,7139,7140],{"class":4733},"ToString",[3446,7142,5390],{"class":4443},[3446,7144,7146],{"class":3448,"line":7145},145,[3446,7147,6257],{"class":4443},[3446,7149,7151],{"class":3448,"line":7150},146,[3446,7152,6592],{"class":4443},[3446,7154,7156,7158,7161],{"class":3448,"line":7155},147,[3446,7157,5032],{"class":4921},[3446,7159,7160],{"class":4609}," \"127.0.0.1\"",[3446,7162,4444],{"class":4443},[3446,7164,7166],{"class":3448,"line":7165},148,[3446,7167,5067],{"class":4443},[3446,7169,7171],{"class":3448,"line":7170},149,[3446,7172,3471],{"emptyLinePlaceholder":3470},[3446,7174,7176,7178,7180,7183,7186],{"class":3448,"line":7175},150,[3446,7177,4486],{"class":4435},[3446,7179,5756],{"class":4435},[3446,7181,7182],{"class":4439}," ValueTask",[3446,7184,7185],{"class":4733}," DisposeAsync",[3446,7187,5765],{"class":4443},[3446,7189,7191],{"class":3448,"line":7190},151,[3446,7192,4882],{"class":4443},[3446,7194,7196,7198,7200,7202,7205],{"class":3448,"line":7195},152,[3446,7197,5974],{"class":4435},[3446,7199,5382],{"class":4495},[3446,7201,3357],{"class":4443},[3446,7203,7204],{"class":4733},"CancelAsync",[3446,7206,5390],{"class":4443},[3446,7208,7210,7212,7214,7217],{"class":3448,"line":7209},153,[3446,7211,5561],{"class":4495},[3446,7213,3357],{"class":4443},[3446,7215,7216],{"class":4733},"Dispose",[3446,7218,5390],{"class":4443},[3446,7220,7222,7224,7226,7228],{"class":3448,"line":7221},154,[3446,7223,5689],{"class":4495},[3446,7225,3357],{"class":4443},[3446,7227,7216],{"class":4733},[3446,7229,5390],{"class":4443},[3446,7231,7233,7236,7238,7240],{"class":3448,"line":7232},155,[3446,7234,7235],{"class":4495},"        _cts",[3446,7237,3357],{"class":4443},[3446,7239,7216],{"class":4733},[3446,7241,5390],{"class":4443},[3446,7243,7245],{"class":3448,"line":7244},156,[3446,7246,5067],{"class":4443},[3446,7248,7250],{"class":3448,"line":7249},157,[3446,7251,3529],{"class":4443},[3377,7253,7255],{"id":7254},"discoveryclientcs-клієнт","DiscoveryClient.cs — клієнт",[3317,7257,7258],{},"Клієнт виконує дві функції: надсилає broadcast-запит та прослуховує multicast-групу для пасивного виявлення.",[3437,7260,7262],{"className":4419,"code":7261,"language":4421,"meta":4422,"style":3442},"\u002F\u002F DiscoveryClient.cs\nnamespace NetworkDiscovery;\n\nusing System.Net;\nusing System.Net.Sockets;\nusing System.Text;\n\n\u002F\u002F\u002F \u003Csummary>\n\u002F\u002F\u002F Клієнт виявлення. Виконує два типи пошуку:\n\u002F\u002F\u002F 1. Активний — надсилає broadcast DISCOVER? і збирає unicast-відповіді.\n\u002F\u002F\u002F 2. Пасивний — підписується на multicast-групу та слухає анонси.\n\u002F\u002F\u002F \u003C\u002Fsummary>\npublic sealed class DiscoveryClient : IAsyncDisposable\n{\n    private readonly CancellationTokenSource _cts = new();\n\n    \u002F\u002F Сокет для надсилання broadcast і отримання unicast-відповідей\n    private readonly UdpClient _broadcastSender;\n\n    \u002F\u002F Сокет для отримання multicast-анонсів\n    private readonly UdpClient _multicastListener;\n\n    private readonly IPEndPoint _broadcastEndPoint = new(\n        IPAddress.Broadcast, DiscoveryProtocol.BroadcastPort);\n\n    private readonly IPAddress _multicastGroup =\n        IPAddress.Parse(DiscoveryProtocol.MulticastGroupAddress);\n\n    public DiscoveryClient()\n    {\n        \u002F\u002F ── Broadcast sender (також отримує unicast-відповіді) ──────────\n        _broadcastSender = new UdpClient();\n        _broadcastSender.EnableBroadcast = true;\n        \u002F\u002F Прив'язуємо до довільного порту — ОС призначить ефемерний\n        _broadcastSender.Client.Bind(new IPEndPoint(IPAddress.Any, 0));\n\n        \u002F\u002F ── Multicast listener ──────────────────────────────────────────\n        _multicastListener = new UdpClient();\n        _multicastListener.Client.SetSocketOption(\n            SocketOptionLevel.Socket, SocketOptionName.ReuseAddress, true);\n        _multicastListener.Client.Bind(\n            new IPEndPoint(IPAddress.Any, DiscoveryProtocol.MulticastPort));\n\n        \u002F\u002F Підписуємось на multicast-групу через IGMP\n        _multicastListener.JoinMulticastGroup(_multicastGroup);\n    }\n\n    \u002F\u002F ── Активний пошук через Broadcast ───────────────────────────────────────\n\n    \u002F\u002F\u002F \u003Csummary>\n    \u002F\u002F\u002F Надсилає DISCOVER? broadcast та збирає відповіді протягом \u003Cparamref name=\"timeoutSeconds\"\u002F> секунд.\n    \u002F\u002F\u002F \u003C\u002Fsummary>\n    public async Task\u003CList\u003CServiceInfo>> DiscoverViaBroadcastAsync(int timeoutSeconds = 3)\n    {\n        var found = new List\u003CServiceInfo>();\n\n        byte[] request = Encoding.UTF8.GetBytes(DiscoveryProtocol.CmdDiscover);\n        await _broadcastSender.SendAsync(request, _broadcastEndPoint);\n        Console.WriteLine($\"[Client] Broadcast DISCOVER? надіслано → {_broadcastEndPoint}\");\n\n        using var timeoutCts = new CancellationTokenSource(\n            TimeSpan.FromSeconds(timeoutSeconds));\n\n        Console.WriteLine($\"[Client] Збираємо відповіді протягом {timeoutSeconds}с...\");\n\n        while (!timeoutCts.Token.IsCancellationRequested)\n        {\n            try\n            {\n                UdpReceiveResult result =\n                    await _broadcastSender.ReceiveAsync(timeoutCts.Token);\n\n                string raw = Encoding.UTF8.GetString(result.Buffer).Trim();\n                ServiceInfo? info = DiscoveryProtocol.ParseAnnounce(raw);\n\n                if (info is not null && !found.Any(s => s.Ip == info.Ip && s.Port == info.Port))\n                {\n                    found.Add(info);\n                    Console.ForegroundColor = ConsoleColor.Green;\n                    Console.WriteLine($\"[Client] ✅ Знайдено (broadcast): {info}\");\n                    Console.ResetColor();\n                }\n            }\n            catch (OperationCanceledException) { break; }\n            catch (SocketException ex)\n            {\n                Console.WriteLine($\"[Client] Помилка прийому: {ex.SocketErrorCode}\");\n                break;\n            }\n        }\n\n        return found;\n    }\n\n    \u002F\u002F ── Пасивний пошук через Multicast ───────────────────────────────────────\n\n    \u002F\u002F\u002F \u003Csummary>\n    \u002F\u002F\u002F Слухає multicast-групу та виводить кожен новий анонс.\n    \u002F\u002F\u002F Зупиняється при натисканні Ctrl+C або виклику DisposeAsync().\n    \u002F\u002F\u002F \u003C\u002Fsummary>\n    public async Task ListenMulticastAsync()\n    {\n        Console.WriteLine($\"[Client] Слухаємо multicast {DiscoveryProtocol.MulticastGroupAddress}:{DiscoveryProtocol.MulticastPort}...\");\n        Console.WriteLine(\"[Client] Ctrl+C для зупинки.\");\n\n        Console.CancelKeyPress += (_, e) => { e.Cancel = true; _cts.Cancel(); };\n\n        var seen = new HashSet\u003Cstring>(); \u002F\u002F щоб не дублювати у виводі\n\n        while (!_cts.Token.IsCancellationRequested)\n        {\n            UdpReceiveResult result;\n            try { result = await _multicastListener.ReceiveAsync(_cts.Token); }\n            catch (OperationCanceledException) { break; }\n            catch (SocketException ex)\n            {\n                Console.WriteLine($\"[Client] Multicast помилка: {ex.SocketErrorCode}\");\n                break;\n            }\n\n            string raw = Encoding.UTF8.GetString(result.Buffer).Trim();\n            ServiceInfo? info = DiscoveryProtocol.ParseAnnounce(raw);\n\n            if (info is not null)\n            {\n                string key = $\"{info.Ip}:{info.Port}\";\n                bool isNew = seen.Add(key);\n\n                Console.ForegroundColor = isNew ? ConsoleColor.Green : ConsoleColor.Gray;\n                string prefix = isNew ? \"✅ Новий сервіс\" : \"🔄 Анонс\";\n                Console.WriteLine($\"[Client] {prefix}: {info}\");\n                Console.ResetColor();\n            }\n        }\n\n        \u002F\u002F Відписуємось від multicast-групи при завершенні\n        _multicastListener.DropMulticastGroup(_multicastGroup);\n        Console.WriteLine(\"[Client] Зупинено.\");\n    }\n\n    public async ValueTask DisposeAsync()\n    {\n        await _cts.CancelAsync();\n        _broadcastSender.Dispose();\n        _multicastListener.Dispose();\n        _cts.Dispose();\n    }\n}\n",[3415,7263,7264,7269,7277,7281,7293,7309,7321,7325,7335,7340,7345,7350,7360,7375,7379,7395,7399,7404,7417,7421,7426,7439,7443,7460,7478,7482,7497,7515,7519,7527,7531,7536,7549,7563,7568,7600,7604,7609,7622,7636,7658,7672,7696,7700,7705,7721,7725,7729,7734,7738,7742,7747,7751,7788,7792,7813,7817,7848,7869,7892,7896,7915,7932,7936,7960,7964,7983,7987,7991,7995,8004,8024,8028,8060,8084,8088,8156,8160,8176,8197,8220,8231,8235,8239,8253,8265,8269,8296,8303,8307,8311,8315,8323,8327,8331,8336,8340,8344,8349,8354,8358,8371,8375,8415,8430,8434,8472,8476,8500,8504,8522,8526,8534,8562,8576,8588,8592,8619,8625,8629,8633,8665,8688,8692,8708,8712,8749,8773,8777,8810,8833,8866,8876,8880,8884,8888,8893,8908,8923,8927,8931,8943,8947,8959,8969,8979,8989,8993],{"__ignoreMap":3442},[3446,7265,7266],{"class":3448,"line":3449},[3446,7267,7268],{"class":4429},"\u002F\u002F DiscoveryClient.cs\n",[3446,7270,7271,7273,7275],{"class":3448,"line":3455},[3446,7272,4436],{"class":4435},[3446,7274,4440],{"class":4439},[3446,7276,4444],{"class":4443},[3446,7278,7279],{"class":3448,"line":3461},[3446,7280,3471],{"emptyLinePlaceholder":3470},[3446,7282,7283,7285,7287,7289,7291],{"class":3448,"line":3467},[3446,7284,5224],{"class":4921},[3446,7286,5227],{"class":4439},[3446,7288,3357],{"class":4443},[3446,7290,5232],{"class":4439},[3446,7292,4444],{"class":4443},[3446,7294,7295,7297,7299,7301,7303,7305,7307],{"class":3448,"line":3474},[3446,7296,5224],{"class":4921},[3446,7298,5227],{"class":4439},[3446,7300,3357],{"class":4443},[3446,7302,5232],{"class":4439},[3446,7304,3357],{"class":4443},[3446,7306,5266],{"class":4439},[3446,7308,4444],{"class":4443},[3446,7310,7311,7313,7315,7317,7319],{"class":3448,"line":3480},[3446,7312,5224],{"class":4921},[3446,7314,5227],{"class":4439},[3446,7316,3357],{"class":4443},[3446,7318,5279],{"class":4439},[3446,7320,4444],{"class":4443},[3446,7322,7323],{"class":3448,"line":3485},[3446,7324,3471],{"emptyLinePlaceholder":3470},[3446,7326,7327,7329,7331,7333],{"class":3448,"line":3491},[3446,7328,5083],{"class":4429},[3446,7330,4518],{"class":4517},[3446,7332,4522],{"class":4521},[3446,7334,4536],{"class":4517},[3446,7336,7337],{"class":3448,"line":3497},[3446,7338,7339],{"class":4429},"\u002F\u002F\u002F Клієнт виявлення. Виконує два типи пошуку:\n",[3446,7341,7342],{"class":3448,"line":3503},[3446,7343,7344],{"class":4429},"\u002F\u002F\u002F 1. Активний — надсилає broadcast DISCOVER? і збирає unicast-відповіді.\n",[3446,7346,7347],{"class":3448,"line":3509},[3446,7348,7349],{"class":4429},"\u002F\u002F\u002F 2. Пасивний — підписується на multicast-групу та слухає анонси.\n",[3446,7351,7352,7354,7356,7358],{"class":3448,"line":3515},[3446,7353,5083],{"class":4429},[3446,7355,4531],{"class":4517},[3446,7357,4522],{"class":4521},[3446,7359,4536],{"class":4517},[3446,7361,7362,7364,7366,7368,7371,7373],{"class":3448,"line":3520},[3446,7363,4453],{"class":4435},[3446,7365,5327],{"class":4435},[3446,7367,4459],{"class":4435},[3446,7369,7370],{"class":4439}," DiscoveryClient",[3446,7372,5335],{"class":4443},[3446,7374,5338],{"class":4439},[3446,7376,7377],{"class":3448,"line":3526},[3446,7378,4467],{"class":4443},[3446,7380,7381,7383,7385,7387,7389,7391,7393],{"class":3448,"line":3532},[3446,7382,5347],{"class":4435},[3446,7384,5350],{"class":4435},[3446,7386,5379],{"class":4439},[3446,7388,5382],{"class":4495},[3446,7390,4499],{"class":4443},[3446,7392,5387],{"class":4435},[3446,7394,5390],{"class":4443},[3446,7396,7397],{"class":3448,"line":3537},[3446,7398,3471],{"emptyLinePlaceholder":3470},[3446,7400,7401],{"class":3448,"line":3543},[3446,7402,7403],{"class":4429},"    \u002F\u002F Сокет для надсилання broadcast і отримання unicast-відповідей\n",[3446,7405,7406,7408,7410,7412,7415],{"class":3448,"line":3548},[3446,7407,5347],{"class":4435},[3446,7409,5350],{"class":4435},[3446,7411,5408],{"class":4439},[3446,7413,7414],{"class":4495}," _broadcastSender",[3446,7416,4444],{"class":4443},[3446,7418,7419],{"class":3448,"line":3554},[3446,7420,3471],{"emptyLinePlaceholder":3470},[3446,7422,7423],{"class":3448,"line":3560},[3446,7424,7425],{"class":4429},"    \u002F\u002F Сокет для отримання multicast-анонсів\n",[3446,7427,7428,7430,7432,7434,7437],{"class":3448,"line":3566},[3446,7429,5347],{"class":4435},[3446,7431,5350],{"class":4435},[3446,7433,5408],{"class":4439},[3446,7435,7436],{"class":4495}," _multicastListener",[3446,7438,4444],{"class":4443},[3446,7440,7441],{"class":3448,"line":3572},[3446,7442,3471],{"emptyLinePlaceholder":3470},[3446,7444,7445,7447,7449,7451,7454,7456,7458],{"class":3448,"line":3578},[3446,7446,5347],{"class":4435},[3446,7448,5350],{"class":4435},[3446,7450,5448],{"class":4439},[3446,7452,7453],{"class":4495}," _broadcastEndPoint",[3446,7455,4499],{"class":4443},[3446,7457,5387],{"class":4435},[3446,7459,5458],{"class":4443},[3446,7461,7462,7464,7466,7468,7470,7472,7474,7476],{"class":3448,"line":3584},[3446,7463,5463],{"class":4495},[3446,7465,3357],{"class":4443},[3446,7467,3352],{"class":4495},[3446,7469,4746],{"class":4443},[3446,7471,5473],{"class":4495},[3446,7473,3357],{"class":4443},[3446,7475,5657],{"class":4495},[3446,7477,4915],{"class":4443},[3446,7479,7480],{"class":3448,"line":3589},[3446,7481,3471],{"emptyLinePlaceholder":3470},[3446,7483,7484,7486,7488,7491,7494],{"class":3448,"line":3595},[3446,7485,5347],{"class":4435},[3446,7487,5350],{"class":4435},[3446,7489,7490],{"class":4439}," IPAddress",[3446,7492,7493],{"class":4495}," _multicastGroup",[3446,7495,7496],{"class":4443}," =\n",[3446,7498,7499,7501,7503,7505,7507,7509,7511,7513],{"class":3448,"line":3601},[3446,7500,5463],{"class":4495},[3446,7502,3357],{"class":4443},[3446,7504,5468],{"class":4733},[3446,7506,4737],{"class":4443},[3446,7508,5473],{"class":4495},[3446,7510,3357],{"class":4443},[3446,7512,5478],{"class":4495},[3446,7514,4915],{"class":4443},[3446,7516,7517],{"class":3448,"line":3607},[3446,7518,3471],{"emptyLinePlaceholder":3470},[3446,7520,7521,7523,7525],{"class":3448,"line":3613},[3446,7522,4486],{"class":4435},[3446,7524,7370],{"class":4733},[3446,7526,5765],{"class":4443},[3446,7528,7529],{"class":3448,"line":3619},[3446,7530,4882],{"class":4443},[3446,7532,7533],{"class":3448,"line":3625},[3446,7534,7535],{"class":4429},"        \u002F\u002F ── Broadcast sender (також отримує unicast-відповіді) ──────────\n",[3446,7537,7538,7541,7543,7545,7547],{"class":3448,"line":3630},[3446,7539,7540],{"class":4495},"        _broadcastSender",[3446,7542,4499],{"class":4443},[3446,7544,5387],{"class":4435},[3446,7546,5408],{"class":4439},[3446,7548,5390],{"class":4443},[3446,7550,7551,7553,7555,7557,7559,7561],{"class":3448,"line":4718},[3446,7552,7540],{"class":4495},[3446,7554,3357],{"class":4443},[3446,7556,5669],{"class":4495},[3446,7558,4499],{"class":4443},[3446,7560,5614],{"class":4435},[3446,7562,4444],{"class":4443},[3446,7564,7565],{"class":3448,"line":4724},[3446,7566,7567],{"class":4429},"        \u002F\u002F Прив'язуємо до довільного порту — ОС призначить ефемерний\n",[3446,7569,7570,7572,7574,7576,7578,7580,7582,7584,7586,7588,7590,7592,7594,7596,7598],{"class":3448,"line":4765},[3446,7571,7540],{"class":4495},[3446,7573,3357],{"class":4443},[3446,7575,4187],{"class":4495},[3446,7577,3357],{"class":4443},[3446,7579,5629],{"class":4733},[3446,7581,4737],{"class":4443},[3446,7583,5387],{"class":4435},[3446,7585,5448],{"class":4439},[3446,7587,4737],{"class":4443},[3446,7589,5643],{"class":4495},[3446,7591,3357],{"class":4443},[3446,7593,5648],{"class":4495},[3446,7595,4746],{"class":4443},[3446,7597,4965],{"class":4502},[3446,7599,5660],{"class":4443},[3446,7601,7602],{"class":3448,"line":4814},[3446,7603,3471],{"emptyLinePlaceholder":3470},[3446,7605,7606],{"class":3448,"line":4819},[3446,7607,7608],{"class":4429},"        \u002F\u002F ── Multicast listener ──────────────────────────────────────────\n",[3446,7610,7611,7614,7616,7618,7620],{"class":3448,"line":4830},[3446,7612,7613],{"class":4495},"        _multicastListener",[3446,7615,4499],{"class":4443},[3446,7617,5387],{"class":4435},[3446,7619,5408],{"class":4439},[3446,7621,5390],{"class":4443},[3446,7623,7624,7626,7628,7630,7632,7634],{"class":3448,"line":4836},[3446,7625,7613],{"class":4495},[3446,7627,3357],{"class":4443},[3446,7629,4187],{"class":4495},[3446,7631,3357],{"class":4443},[3446,7633,5587],{"class":4733},[3446,7635,5458],{"class":4443},[3446,7637,7638,7640,7642,7644,7646,7648,7650,7652,7654,7656],{"class":3448,"line":4842},[3446,7639,5594],{"class":4495},[3446,7641,3357],{"class":4443},[3446,7643,5599],{"class":4495},[3446,7645,4746],{"class":4443},[3446,7647,5604],{"class":4495},[3446,7649,3357],{"class":4443},[3446,7651,5609],{"class":4495},[3446,7653,4746],{"class":4443},[3446,7655,5614],{"class":4435},[3446,7657,4915],{"class":4443},[3446,7659,7660,7662,7664,7666,7668,7670],{"class":3448,"line":4853},[3446,7661,7613],{"class":4495},[3446,7663,3357],{"class":4443},[3446,7665,4187],{"class":4495},[3446,7667,3357],{"class":4443},[3446,7669,5629],{"class":4733},[3446,7671,5458],{"class":4443},[3446,7673,7674,7676,7678,7680,7682,7684,7686,7688,7690,7692,7694],{"class":3448,"line":4879},[3446,7675,5636],{"class":4435},[3446,7677,5448],{"class":4439},[3446,7679,4737],{"class":4443},[3446,7681,5643],{"class":4495},[3446,7683,3357],{"class":4443},[3446,7685,5648],{"class":4495},[3446,7687,4746],{"class":4443},[3446,7689,5473],{"class":4495},[3446,7691,3357],{"class":4443},[3446,7693,5491],{"class":4495},[3446,7695,5660],{"class":4443},[3446,7697,7698],{"class":3448,"line":4885},[3446,7699,3471],{"emptyLinePlaceholder":3470},[3446,7701,7702],{"class":3448,"line":4891},[3446,7703,7704],{"class":4429},"        \u002F\u002F Підписуємось на multicast-групу через IGMP\n",[3446,7706,7707,7709,7711,7714,7716,7719],{"class":3448,"line":4918},[3446,7708,7613],{"class":4495},[3446,7710,3357],{"class":4443},[3446,7712,7713],{"class":4733},"JoinMulticastGroup",[3446,7715,4737],{"class":4443},[3446,7717,7718],{"class":4495},"_multicastGroup",[3446,7720,4915],{"class":4443},[3446,7722,7723],{"class":3448,"line":4953},[3446,7724,5067],{"class":4443},[3446,7726,7727],{"class":3448,"line":4981},[3446,7728,3471],{"emptyLinePlaceholder":3470},[3446,7730,7731],{"class":3448,"line":5024},[3446,7732,7733],{"class":4429},"    \u002F\u002F ── Активний пошук через Broadcast ───────────────────────────────────────\n",[3446,7735,7736],{"class":3448,"line":5029},[3446,7737,3471],{"emptyLinePlaceholder":3470},[3446,7739,7740],{"class":3448,"line":5064},[3446,7741,4572],{"class":4429},[3446,7743,7744],{"class":3448,"line":5070},[3446,7745,7746],{"class":4429},"    \u002F\u002F\u002F Надсилає DISCOVER? broadcast та збирає відповіді протягом \u003Cparamref name=\"timeoutSeconds\"\u002F> секунд.\n",[3446,7748,7749],{"class":3448,"line":5075},[3446,7750,4592],{"class":4429},[3446,7752,7753,7755,7757,7759,7761,7764,7766,7769,7772,7775,7777,7779,7782,7784,7786],{"class":3448,"line":5080},[3446,7754,4486],{"class":4435},[3446,7756,5756],{"class":4435},[3446,7758,5759],{"class":4439},[3446,7760,4518],{"class":4443},[3446,7762,7763],{"class":4439},"List",[3446,7765,4518],{"class":4443},[3446,7767,7768],{"class":4439},"ServiceInfo",[3446,7770,7771],{"class":4443},">> ",[3446,7773,7774],{"class":4733},"DiscoverViaBroadcastAsync",[3446,7776,4737],{"class":4443},[3446,7778,4756],{"class":4435},[3446,7780,7781],{"class":4495}," timeoutSeconds",[3446,7783,4499],{"class":4443},[3446,7785,5002],{"class":4502},[3446,7787,4876],{"class":4443},[3446,7789,7790],{"class":3448,"line":5101},[3446,7791,4882],{"class":4443},[3446,7793,7794,7796,7799,7801,7803,7806,7808,7810],{"class":3448,"line":5134},[3446,7795,4894],{"class":4435},[3446,7797,7798],{"class":4495}," found",[3446,7800,4499],{"class":4443},[3446,7802,5387],{"class":4435},[3446,7804,7805],{"class":4439}," List",[3446,7807,4518],{"class":4443},[3446,7809,7768],{"class":4439},[3446,7811,7812],{"class":4443},">();\n",[3446,7814,7815],{"class":3448,"line":5139},[3446,7816,3471],{"emptyLinePlaceholder":3470},[3446,7818,7819,7821,7823,7826,7828,7830,7832,7834,7836,7838,7840,7842,7844,7846],{"class":3448,"line":5188},[3446,7820,6683],{"class":4435},[3446,7822,6427],{"class":4443},[3446,7824,7825],{"class":4495},"request",[3446,7827,4499],{"class":4443},[3446,7829,6276],{"class":4495},[3446,7831,3357],{"class":4443},[3446,7833,6281],{"class":4495},[3446,7835,3357],{"class":4443},[3446,7837,6443],{"class":4733},[3446,7839,4737],{"class":4443},[3446,7841,5473],{"class":4495},[3446,7843,3357],{"class":4443},[3446,7845,6369],{"class":4495},[3446,7847,4915],{"class":4443},[3446,7849,7850,7852,7854,7856,7858,7860,7862,7864,7867],{"class":3448,"line":5913},[3446,7851,5974],{"class":4435},[3446,7853,7414],{"class":4495},[3446,7855,3357],{"class":4443},[3446,7857,6480],{"class":4733},[3446,7859,4737],{"class":4443},[3446,7861,7825],{"class":4495},[3446,7863,4746],{"class":4443},[3446,7865,7866],{"class":4495},"_broadcastEndPoint",[3446,7868,4915],{"class":4443},[3446,7870,7871,7873,7875,7877,7879,7882,7884,7886,7888,7890],{"class":3448,"line":5918},[3446,7872,5789],{"class":4495},[3446,7874,3357],{"class":4443},[3446,7876,5794],{"class":4733},[3446,7878,4737],{"class":4443},[3446,7880,7881],{"class":4609},"$\"[Client] Broadcast DISCOVER? надіслано → ",[3446,7883,4772],{"class":4771},[3446,7885,7866],{"class":4495},[3446,7887,4778],{"class":4771},[3446,7889,4809],{"class":4609},[3446,7891,4915],{"class":4443},[3446,7893,7894],{"class":3448,"line":5966},[3446,7895,3471],{"emptyLinePlaceholder":3470},[3446,7897,7898,7901,7904,7907,7909,7911,7913],{"class":3448,"line":5971},[3446,7899,7900],{"class":4921},"        using",[3446,7902,7903],{"class":4435}," var",[3446,7905,7906],{"class":4495}," timeoutCts",[3446,7908,4499],{"class":4443},[3446,7910,5387],{"class":4435},[3446,7912,5379],{"class":4439},[3446,7914,5458],{"class":4443},[3446,7916,7917,7920,7922,7925,7927,7930],{"class":3448,"line":5986},[3446,7918,7919],{"class":4495},"            TimeSpan",[3446,7921,3357],{"class":4443},[3446,7923,7924],{"class":4733},"FromSeconds",[3446,7926,4737],{"class":4443},[3446,7928,7929],{"class":4495},"timeoutSeconds",[3446,7931,5660],{"class":4443},[3446,7933,7934],{"class":3448,"line":6007},[3446,7935,3471],{"emptyLinePlaceholder":3470},[3446,7937,7938,7940,7942,7944,7946,7949,7951,7953,7955,7958],{"class":3448,"line":6027},[3446,7939,5789],{"class":4495},[3446,7941,3357],{"class":4443},[3446,7943,5794],{"class":4733},[3446,7945,4737],{"class":4443},[3446,7947,7948],{"class":4609},"$\"[Client] Збираємо відповіді протягом ",[3446,7950,4772],{"class":4771},[3446,7952,7929],{"class":4495},[3446,7954,4778],{"class":4771},[3446,7956,7957],{"class":4609},"с...\"",[3446,7959,4915],{"class":4443},[3446,7961,7962],{"class":3448,"line":6033},[3446,7963,3471],{"emptyLinePlaceholder":3470},[3446,7965,7966,7968,7970,7973,7975,7977,7979,7981],{"class":3448,"line":6038},[3446,7967,6111],{"class":4921},[3446,7969,4986],{"class":4443},[3446,7971,7972],{"class":4495},"timeoutCts",[3446,7974,3357],{"class":4443},[3446,7976,6002],{"class":4495},[3446,7978,3357],{"class":4443},[3446,7980,6121],{"class":4495},[3446,7982,4876],{"class":4443},[3446,7984,7985],{"class":3448,"line":6054},[3446,7986,6129],{"class":4443},[3446,7988,7989],{"class":3448,"line":6059},[3446,7990,6737],{"class":4921},[3446,7992,7993],{"class":3448,"line":6064},[3446,7994,6212],{"class":4443},[3446,7996,7997,8000,8002],{"class":3448,"line":6070},[3446,7998,7999],{"class":4439},"                UdpReceiveResult",[3446,8001,6138],{"class":4495},[3446,8003,7496],{"class":4443},[3446,8005,8006,8008,8010,8012,8014,8016,8018,8020,8022],{"class":3448,"line":6075},[3446,8007,6473],{"class":4435},[3446,8009,7414],{"class":4495},[3446,8011,3357],{"class":4443},[3446,8013,6164],{"class":4733},[3446,8015,4737],{"class":4443},[3446,8017,7972],{"class":4495},[3446,8019,3357],{"class":4443},[3446,8021,6002],{"class":4495},[3446,8023,4915],{"class":4443},[3446,8025,8026],{"class":3448,"line":6103},[3446,8027,3471],{"emptyLinePlaceholder":3470},[3446,8029,8030,8032,8034,8036,8038,8040,8042,8044,8046,8048,8050,8052,8054,8056,8058],{"class":3448,"line":6108},[3446,8031,6388],{"class":4435},[3446,8033,4873],{"class":4495},[3446,8035,4499],{"class":4443},[3446,8037,6276],{"class":4495},[3446,8039,3357],{"class":4443},[3446,8041,6281],{"class":4495},[3446,8043,3357],{"class":4443},[3446,8045,6286],{"class":4733},[3446,8047,4737],{"class":4443},[3446,8049,6152],{"class":4495},[3446,8051,3357],{"class":4443},[3446,8053,6295],{"class":4495},[3446,8055,6298],{"class":4443},[3446,8057,6301],{"class":4733},[3446,8059,5390],{"class":4443},[3446,8061,8062,8065,8067,8070,8072,8074,8076,8078,8080,8082],{"class":3448,"line":6126},[3446,8063,8064],{"class":4439},"                ServiceInfo",[3446,8066,4863],{"class":4443},[3446,8068,8069],{"class":4495},"info",[3446,8071,4499],{"class":4443},[3446,8073,5473],{"class":4495},[3446,8075,3357],{"class":4443},[3446,8077,4866],{"class":4733},[3446,8079,4737],{"class":4443},[3446,8081,4902],{"class":4495},[3446,8083,4915],{"class":4443},[3446,8085,8086],{"class":3448,"line":6132},[3446,8087,3471],{"emptyLinePlaceholder":3470},[3446,8089,8090,8092,8094,8096,8099,8102,8104,8107,8110,8112,8114,8116,8119,8122,8124,8126,8128,8130,8132,8134,8136,8139,8141,8143,8145,8147,8149,8151,8153],{"class":3448,"line":6143},[3446,8091,7097],{"class":4921},[3446,8093,4925],{"class":4443},[3446,8095,8069],{"class":4495},[3446,8097,8098],{"class":4435}," is",[3446,8100,8101],{"class":4435}," not",[3446,8103,4948],{"class":4435},[3446,8105,8106],{"class":4443}," && !",[3446,8108,8109],{"class":4495},"found",[3446,8111,3357],{"class":4443},[3446,8113,5648],{"class":4733},[3446,8115,4737],{"class":4443},[3446,8117,8118],{"class":4495},"s",[3446,8120,8121],{"class":4443}," => ",[3446,8123,8118],{"class":4495},[3446,8125,3357],{"class":4443},[3446,8127,5170],{"class":4495},[3446,8129,6362],{"class":4443},[3446,8131,8069],{"class":4495},[3446,8133,3357],{"class":4443},[3446,8135,5170],{"class":4495},[3446,8137,8138],{"class":4443}," && ",[3446,8140,8118],{"class":4495},[3446,8142,3357],{"class":4443},[3446,8144,5179],{"class":4495},[3446,8146,6362],{"class":4443},[3446,8148,8069],{"class":4495},[3446,8150,3357],{"class":4443},[3446,8152,5179],{"class":4495},[3446,8154,8155],{"class":4443},"))\n",[3446,8157,8158],{"class":3448,"line":6174},[3446,8159,6467],{"class":4443},[3446,8161,8162,8165,8167,8170,8172,8174],{"class":3448,"line":6194},[3446,8163,8164],{"class":4495},"                    found",[3446,8166,3357],{"class":4443},[3446,8168,8169],{"class":4733},"Add",[3446,8171,4737],{"class":4443},[3446,8173,8069],{"class":4495},[3446,8175,4915],{"class":4443},[3446,8177,8178,8180,8182,8185,8187,8190,8192,8195],{"class":3448,"line":6209},[3446,8179,6504],{"class":4495},[3446,8181,3357],{"class":4443},[3446,8183,8184],{"class":4495},"ForegroundColor",[3446,8186,4499],{"class":4443},[3446,8188,8189],{"class":4495},"ConsoleColor",[3446,8191,3357],{"class":4443},[3446,8193,8194],{"class":4495},"Green",[3446,8196,4444],{"class":4443},[3446,8198,8199,8201,8203,8205,8207,8210,8212,8214,8216,8218],{"class":3448,"line":6215},[3446,8200,6504],{"class":4495},[3446,8202,3357],{"class":4443},[3446,8204,5794],{"class":4733},[3446,8206,4737],{"class":4443},[3446,8208,8209],{"class":4609},"$\"[Client] ✅ Знайдено (broadcast): ",[3446,8211,4772],{"class":4771},[3446,8213,8069],{"class":4495},[3446,8215,4778],{"class":4771},[3446,8217,4809],{"class":4609},[3446,8219,4915],{"class":4443},[3446,8221,8222,8224,8226,8229],{"class":3448,"line":6246},[3446,8223,6504],{"class":4495},[3446,8225,3357],{"class":4443},[3446,8227,8228],{"class":4733},"ResetColor",[3446,8230,5390],{"class":4443},[3446,8232,8233],{"class":3448,"line":6254},[3446,8234,6529],{"class":4443},[3446,8236,8237],{"class":3448,"line":6260},[3446,8238,6257],{"class":4443},[3446,8240,8241,8243,8245,8247,8249,8251],{"class":3448,"line":6265},[3446,8242,6177],{"class":4921},[3446,8244,4925],{"class":4443},[3446,8246,6182],{"class":4439},[3446,8248,6185],{"class":4443},[3446,8250,6188],{"class":4921},[3446,8252,6191],{"class":4443},[3446,8254,8255,8257,8259,8261,8263],{"class":3448,"line":6306},[3446,8256,6177],{"class":4921},[3446,8258,4925],{"class":4443},[3446,8260,6201],{"class":4439},[3446,8262,6204],{"class":4495},[3446,8264,4876],{"class":4443},[3446,8266,8267],{"class":3448,"line":6347},[3446,8268,6212],{"class":4443},[3446,8270,8271,8273,8275,8277,8279,8282,8284,8286,8288,8290,8292,8294],{"class":3448,"line":6352},[3446,8272,6218],{"class":4495},[3446,8274,3357],{"class":4443},[3446,8276,5794],{"class":4733},[3446,8278,4737],{"class":4443},[3446,8280,8281],{"class":4609},"$\"[Client] Помилка прийому: ",[3446,8283,4772],{"class":4771},[3446,8285,6232],{"class":4495},[3446,8287,3357],{"class":4771},[3446,8289,6237],{"class":4495},[3446,8291,4778],{"class":4771},[3446,8293,4809],{"class":4609},[3446,8295,4915],{"class":4443},[3446,8297,8298,8301],{"class":3448,"line":6374},[3446,8299,8300],{"class":4921},"                break",[3446,8302,4444],{"class":4443},[3446,8304,8305],{"class":3448,"line":6379},[3446,8306,6257],{"class":4443},[3446,8308,8309],{"class":3448,"line":6385},[3446,8310,6592],{"class":4443},[3446,8312,8313],{"class":3448,"line":6405},[3446,8314,3471],{"emptyLinePlaceholder":3470},[3446,8316,8317,8319,8321],{"class":3448,"line":6421},[3446,8318,5032],{"class":4921},[3446,8320,7798],{"class":4495},[3446,8322,4444],{"class":4443},[3446,8324,8325],{"class":3448,"line":6453},[3446,8326,5067],{"class":4443},[3446,8328,8329],{"class":3448,"line":6458},[3446,8330,3471],{"emptyLinePlaceholder":3470},[3446,8332,8333],{"class":3448,"line":6464},[3446,8334,8335],{"class":4429},"    \u002F\u002F ── Пасивний пошук через Multicast ───────────────────────────────────────\n",[3446,8337,8338],{"class":3448,"line":6470},[3446,8339,3471],{"emptyLinePlaceholder":3470},[3446,8341,8342],{"class":3448,"line":6501},[3446,8343,4572],{"class":4429},[3446,8345,8346],{"class":3448,"line":6526},[3446,8347,8348],{"class":4429},"    \u002F\u002F\u002F Слухає multicast-групу та виводить кожен новий анонс.\n",[3446,8350,8351],{"class":3448,"line":6532},[3446,8352,8353],{"class":4429},"    \u002F\u002F\u002F Зупиняється при натисканні Ctrl+C або виклику DisposeAsync().\n",[3446,8355,8356],{"class":3448,"line":6546},[3446,8357,4592],{"class":4429},[3446,8359,8360,8362,8364,8366,8369],{"class":3448,"line":6551},[3446,8361,4486],{"class":4435},[3446,8363,5756],{"class":4435},[3446,8365,5759],{"class":4439},[3446,8367,8368],{"class":4733}," ListenMulticastAsync",[3446,8370,5765],{"class":4443},[3446,8372,8373],{"class":3448,"line":6579},[3446,8374,4882],{"class":4443},[3446,8376,8377,8379,8381,8383,8385,8388,8390,8392,8394,8396,8398,8400,8402,8404,8406,8408,8410,8413],{"class":3448,"line":6584},[3446,8378,5789],{"class":4495},[3446,8380,3357],{"class":4443},[3446,8382,5794],{"class":4733},[3446,8384,4737],{"class":4443},[3446,8386,8387],{"class":4609},"$\"[Client] Слухаємо multicast ",[3446,8389,4772],{"class":4771},[3446,8391,5473],{"class":4495},[3446,8393,3357],{"class":4771},[3446,8395,5478],{"class":4495},[3446,8397,4778],{"class":4771},[3446,8399,4781],{"class":4609},[3446,8401,4772],{"class":4771},[3446,8403,5473],{"class":4495},[3446,8405,3357],{"class":4771},[3446,8407,5491],{"class":4495},[3446,8409,4778],{"class":4771},[3446,8411,8412],{"class":4609},"...\"",[3446,8414,4915],{"class":4443},[3446,8416,8417,8419,8421,8423,8425,8428],{"class":3448,"line":6589},[3446,8418,5789],{"class":4495},[3446,8420,3357],{"class":4443},[3446,8422,5794],{"class":4733},[3446,8424,4737],{"class":4443},[3446,8426,8427],{"class":4609},"\"[Client] Ctrl+C для зупинки.\"",[3446,8429,4915],{"class":4443},[3446,8431,8432],{"class":3448,"line":6595},[3446,8433,3471],{"emptyLinePlaceholder":3470},[3446,8435,8436,8438,8440,8442,8444,8446,8448,8450,8452,8454,8456,8458,8460,8462,8464,8466,8468,8470],{"class":3448,"line":6600},[3446,8437,5789],{"class":4495},[3446,8439,3357],{"class":4443},[3446,8441,5925],{"class":4495},[3446,8443,5928],{"class":4443},[3446,8445,5931],{"class":4495},[3446,8447,4746],{"class":4443},[3446,8449,5936],{"class":4495},[3446,8451,5939],{"class":4443},[3446,8453,5936],{"class":4495},[3446,8455,3357],{"class":4443},[3446,8457,5946],{"class":4495},[3446,8459,4499],{"class":4443},[3446,8461,5614],{"class":4435},[3446,8463,5953],{"class":4443},[3446,8465,5956],{"class":4495},[3446,8467,3357],{"class":4443},[3446,8469,5946],{"class":4733},[3446,8471,5963],{"class":4443},[3446,8473,8474],{"class":3448,"line":6605},[3446,8475,3471],{"emptyLinePlaceholder":3470},[3446,8477,8478,8480,8483,8485,8487,8490,8492,8494,8497],{"class":3448,"line":6611},[3446,8479,4894],{"class":4435},[3446,8481,8482],{"class":4495}," seen",[3446,8484,4499],{"class":4443},[3446,8486,5387],{"class":4435},[3446,8488,8489],{"class":4439}," HashSet",[3446,8491,4518],{"class":4443},[3446,8493,4740],{"class":4435},[3446,8495,8496],{"class":4443},">(); ",[3446,8498,8499],{"class":4429},"\u002F\u002F щоб не дублювати у виводі\n",[3446,8501,8502],{"class":3448,"line":6616},[3446,8503,3471],{"emptyLinePlaceholder":3470},[3446,8505,8506,8508,8510,8512,8514,8516,8518,8520],{"class":3448,"line":6642},[3446,8507,6111],{"class":4921},[3446,8509,4986],{"class":4443},[3446,8511,5956],{"class":4495},[3446,8513,3357],{"class":4443},[3446,8515,6002],{"class":4495},[3446,8517,3357],{"class":4443},[3446,8519,6121],{"class":4495},[3446,8521,4876],{"class":4443},[3446,8523,8524],{"class":3448,"line":6647},[3446,8525,6129],{"class":4443},[3446,8527,8528,8530,8532],{"class":3448,"line":6664},[3446,8529,6135],{"class":4439},[3446,8531,6138],{"class":4495},[3446,8533,4444],{"class":4443},[3446,8535,8536,8538,8540,8542,8544,8546,8548,8550,8552,8554,8556,8558,8560],{"class":3448,"line":6680},[3446,8537,6146],{"class":4921},[3446,8539,6149],{"class":4443},[3446,8541,6152],{"class":4495},[3446,8543,4499],{"class":4443},[3446,8545,6157],{"class":4435},[3446,8547,7436],{"class":4495},[3446,8549,3357],{"class":4443},[3446,8551,6164],{"class":4733},[3446,8553,4737],{"class":4443},[3446,8555,5956],{"class":4495},[3446,8557,3357],{"class":4443},[3446,8559,6002],{"class":4495},[3446,8561,6171],{"class":4443},[3446,8563,8564,8566,8568,8570,8572,8574],{"class":3448,"line":6709},[3446,8565,6177],{"class":4921},[3446,8567,4925],{"class":4443},[3446,8569,6182],{"class":4439},[3446,8571,6185],{"class":4443},[3446,8573,6188],{"class":4921},[3446,8575,6191],{"class":4443},[3446,8577,8578,8580,8582,8584,8586],{"class":3448,"line":6714},[3446,8579,6177],{"class":4921},[3446,8581,4925],{"class":4443},[3446,8583,6201],{"class":4439},[3446,8585,6204],{"class":4495},[3446,8587,4876],{"class":4443},[3446,8589,8590],{"class":3448,"line":6729},[3446,8591,6212],{"class":4443},[3446,8593,8594,8596,8598,8600,8602,8605,8607,8609,8611,8613,8615,8617],{"class":3448,"line":6734},[3446,8595,6218],{"class":4495},[3446,8597,3357],{"class":4443},[3446,8599,5794],{"class":4733},[3446,8601,4737],{"class":4443},[3446,8603,8604],{"class":4609},"$\"[Client] Multicast помилка: ",[3446,8606,4772],{"class":4771},[3446,8608,6232],{"class":4495},[3446,8610,3357],{"class":4771},[3446,8612,6237],{"class":4495},[3446,8614,4778],{"class":4771},[3446,8616,4809],{"class":4609},[3446,8618,4915],{"class":4443},[3446,8620,8621,8623],{"class":3448,"line":6740},[3446,8622,8300],{"class":4921},[3446,8624,4444],{"class":4443},[3446,8626,8627],{"class":3448,"line":6745},[3446,8628,6257],{"class":4443},[3446,8630,8631],{"class":3448,"line":6772},[3446,8632,3471],{"emptyLinePlaceholder":3470},[3446,8634,8635,8637,8639,8641,8643,8645,8647,8649,8651,8653,8655,8657,8659,8661,8663],{"class":3448,"line":6796},[3446,8636,6268],{"class":4435},[3446,8638,4873],{"class":4495},[3446,8640,4499],{"class":4443},[3446,8642,6276],{"class":4495},[3446,8644,3357],{"class":4443},[3446,8646,6281],{"class":4495},[3446,8648,3357],{"class":4443},[3446,8650,6286],{"class":4733},[3446,8652,4737],{"class":4443},[3446,8654,6152],{"class":4495},[3446,8656,3357],{"class":4443},[3446,8658,6295],{"class":4495},[3446,8660,6298],{"class":4443},[3446,8662,6301],{"class":4733},[3446,8664,5390],{"class":4443},[3446,8666,8667,8670,8672,8674,8676,8678,8680,8682,8684,8686],{"class":3448,"line":6823},[3446,8668,8669],{"class":4439},"            ServiceInfo",[3446,8671,4863],{"class":4443},[3446,8673,8069],{"class":4495},[3446,8675,4499],{"class":4443},[3446,8677,5473],{"class":4495},[3446,8679,3357],{"class":4443},[3446,8681,4866],{"class":4733},[3446,8683,4737],{"class":4443},[3446,8685,4902],{"class":4495},[3446,8687,4915],{"class":4443},[3446,8689,8690],{"class":3448,"line":6828},[3446,8691,3471],{"emptyLinePlaceholder":3470},[3446,8693,8694,8696,8698,8700,8702,8704,8706],{"class":3448,"line":6843},[3446,8695,6355],{"class":4921},[3446,8697,4925],{"class":4443},[3446,8699,8069],{"class":4495},[3446,8701,8098],{"class":4435},[3446,8703,8101],{"class":4435},[3446,8705,4948],{"class":4435},[3446,8707,4876],{"class":4443},[3446,8709,8710],{"class":3448,"line":6856},[3446,8711,6212],{"class":4443},[3446,8713,8714,8716,8719,8721,8723,8725,8727,8729,8731,8733,8735,8737,8739,8741,8743,8745,8747],{"class":3448,"line":6861},[3446,8715,6388],{"class":4435},[3446,8717,8718],{"class":4495}," key",[3446,8720,4499],{"class":4443},[3446,8722,5155],{"class":4609},[3446,8724,4772],{"class":4771},[3446,8726,8069],{"class":4495},[3446,8728,3357],{"class":4771},[3446,8730,5170],{"class":4495},[3446,8732,4778],{"class":4771},[3446,8734,4781],{"class":4609},[3446,8736,4772],{"class":4771},[3446,8738,8069],{"class":4495},[3446,8740,3357],{"class":4771},[3446,8742,5179],{"class":4495},[3446,8744,4778],{"class":4771},[3446,8746,4809],{"class":4609},[3446,8748,4444],{"class":4443},[3446,8750,8751,8754,8757,8759,8762,8764,8766,8768,8771],{"class":3448,"line":6889},[3446,8752,8753],{"class":4435},"                bool",[3446,8755,8756],{"class":4495}," isNew",[3446,8758,4499],{"class":4443},[3446,8760,8761],{"class":4495},"seen",[3446,8763,3357],{"class":4443},[3446,8765,8169],{"class":4733},[3446,8767,4737],{"class":4443},[3446,8769,8770],{"class":4495},"key",[3446,8772,4915],{"class":4443},[3446,8774,8775],{"class":3448,"line":6911},[3446,8776,3471],{"emptyLinePlaceholder":3470},[3446,8778,8779,8781,8783,8785,8787,8790,8793,8795,8797,8799,8801,8803,8805,8808],{"class":3448,"line":6916},[3446,8780,6218],{"class":4495},[3446,8782,3357],{"class":4443},[3446,8784,8184],{"class":4495},[3446,8786,4499],{"class":4443},[3446,8788,8789],{"class":4495},"isNew",[3446,8791,8792],{"class":4443}," ? ",[3446,8794,8189],{"class":4495},[3446,8796,3357],{"class":4443},[3446,8798,8194],{"class":4495},[3446,8800,5335],{"class":4443},[3446,8802,8189],{"class":4495},[3446,8804,3357],{"class":4443},[3446,8806,8807],{"class":4495},"Gray",[3446,8809,4444],{"class":4443},[3446,8811,8812,8814,8817,8819,8821,8823,8826,8828,8831],{"class":3448,"line":6921},[3446,8813,6388],{"class":4435},[3446,8815,8816],{"class":4495}," prefix",[3446,8818,4499],{"class":4443},[3446,8820,8789],{"class":4495},[3446,8822,8792],{"class":4443},[3446,8824,8825],{"class":4609},"\"✅ Новий сервіс\"",[3446,8827,5335],{"class":4443},[3446,8829,8830],{"class":4609},"\"🔄 Анонс\"",[3446,8832,4444],{"class":4443},[3446,8834,8835,8837,8839,8841,8843,8846,8848,8851,8853,8856,8858,8860,8862,8864],{"class":3448,"line":6926},[3446,8836,6218],{"class":4495},[3446,8838,3357],{"class":4443},[3446,8840,5794],{"class":4733},[3446,8842,4737],{"class":4443},[3446,8844,8845],{"class":4609},"$\"[Client] ",[3446,8847,4772],{"class":4771},[3446,8849,8850],{"class":4495},"prefix",[3446,8852,4778],{"class":4771},[3446,8854,8855],{"class":4609},": ",[3446,8857,4772],{"class":4771},[3446,8859,8069],{"class":4495},[3446,8861,4778],{"class":4771},[3446,8863,4809],{"class":4609},[3446,8865,4915],{"class":4443},[3446,8867,8868,8870,8872,8874],{"class":3448,"line":6931},[3446,8869,6218],{"class":4495},[3446,8871,3357],{"class":4443},[3446,8873,8228],{"class":4733},[3446,8875,5390],{"class":4443},[3446,8877,8878],{"class":3448,"line":6937},[3446,8879,6257],{"class":4443},[3446,8881,8882],{"class":3448,"line":6942},[3446,8883,6592],{"class":4443},[3446,8885,8886],{"class":3448,"line":6956},[3446,8887,3471],{"emptyLinePlaceholder":3470},[3446,8889,8890],{"class":3448,"line":6961},[3446,8891,8892],{"class":4429},"        \u002F\u002F Відписуємось від multicast-групи при завершенні\n",[3446,8894,8895,8897,8899,8902,8904,8906],{"class":3448,"line":6989},[3446,8896,7613],{"class":4495},[3446,8898,3357],{"class":4443},[3446,8900,8901],{"class":4733},"DropMulticastGroup",[3446,8903,4737],{"class":4443},[3446,8905,7718],{"class":4495},[3446,8907,4915],{"class":4443},[3446,8909,8910,8912,8914,8916,8918,8921],{"class":3448,"line":6994},[3446,8911,5789],{"class":4495},[3446,8913,3357],{"class":4443},[3446,8915,5794],{"class":4733},[3446,8917,4737],{"class":4443},[3446,8919,8920],{"class":4609},"\"[Client] Зупинено.\"",[3446,8922,4915],{"class":4443},[3446,8924,8925],{"class":3448,"line":7025},[3446,8926,5067],{"class":4443},[3446,8928,8929],{"class":3448,"line":7054},[3446,8930,3471],{"emptyLinePlaceholder":3470},[3446,8932,8933,8935,8937,8939,8941],{"class":3448,"line":7059},[3446,8934,4486],{"class":4435},[3446,8936,5756],{"class":4435},[3446,8938,7182],{"class":4439},[3446,8940,7185],{"class":4733},[3446,8942,5765],{"class":4443},[3446,8944,8945],{"class":3448,"line":7089},[3446,8946,4882],{"class":4443},[3446,8948,8949,8951,8953,8955,8957],{"class":3448,"line":7094},[3446,8950,5974],{"class":4435},[3446,8952,5382],{"class":4495},[3446,8954,3357],{"class":4443},[3446,8956,7204],{"class":4733},[3446,8958,5390],{"class":4443},[3446,8960,8961,8963,8965,8967],{"class":3448,"line":7126},[3446,8962,7540],{"class":4495},[3446,8964,3357],{"class":4443},[3446,8966,7216],{"class":4733},[3446,8968,5390],{"class":4443},[3446,8970,8971,8973,8975,8977],{"class":3448,"line":7145},[3446,8972,7613],{"class":4495},[3446,8974,3357],{"class":4443},[3446,8976,7216],{"class":4733},[3446,8978,5390],{"class":4443},[3446,8980,8981,8983,8985,8987],{"class":3448,"line":7150},[3446,8982,7235],{"class":4495},[3446,8984,3357],{"class":4443},[3446,8986,7216],{"class":4733},[3446,8988,5390],{"class":4443},[3446,8990,8991],{"class":3448,"line":7155},[3446,8992,5067],{"class":4443},[3446,8994,8995],{"class":3448,"line":7165},[3446,8996,3529],{"class":4443},[3377,8998,9000],{"id":8999},"programcs-точка-входу","Program.cs — точка входу",[3437,9002,9004],{"className":4419,"code":9003,"language":4421,"meta":4422,"style":3442},"\u002F\u002F Program.cs\nusing NetworkDiscovery;\n\nstring mode = args.Length > 0 ? args[0].ToLower() : string.Empty;\n\nswitch (mode)\n{\n    case \"server\":\n        \u002F\u002F dotnet run -- server MyService 8080\n        string name = args.Length > 1 ? args[1] : $\"Service-{Environment.MachineName}\";\n        int port    = args.Length > 2 && int.TryParse(args[2], out int p) ? p : 8080;\n\n        await using var server = new DiscoveryServer(name, port);\n        await server.RunAsync();\n        break;\n\n    case \"broadcast\":\n        \u002F\u002F Активний пошук: надсилає broadcast і чекає на відповіді\n        await using (var client = new DiscoveryClient())\n        {\n            var services = await client.DiscoverViaBroadcastAsync(timeoutSeconds: 3);\n\n            Console.WriteLine();\n            Console.WriteLine($\"=== Знайдено сервісів: {services.Count} ===\");\n            foreach (var svc in services)\n                Console.WriteLine($\"  • {svc}\");\n        }\n        break;\n\n    case \"multicast\":\n        \u002F\u002F Пасивне прослуховування multicast-анонсів\n        await using (var client = new DiscoveryClient())\n        {\n            await client.ListenMulticastAsync();\n        }\n        break;\n\n    default:\n        Console.WriteLine(\"Використання:\");\n        Console.WriteLine(\"  dotnet run -- server [name] [port]   — запустити сервер\");\n        Console.WriteLine(\"  dotnet run -- broadcast              — знайти сервери (broadcast)\");\n        Console.WriteLine(\"  dotnet run -- multicast              — слухати multicast-анонси\");\n        break;\n}\n",[3415,9005,9006,9011,9019,9023,9070,9074,9085,9089,9100,9105,9153,9210,9214,9242,9255,9262,9266,9275,9280,9301,9305,9333,9337,9347,9377,9394,9418,9422,9428,9432,9441,9446,9466,9470,9484,9488,9494,9498,9505,9520,9535,9550,9565,9571],{"__ignoreMap":3442},[3446,9007,9008],{"class":3448,"line":3449},[3446,9009,9010],{"class":4429},"\u002F\u002F Program.cs\n",[3446,9012,9013,9015,9017],{"class":3448,"line":3455},[3446,9014,5224],{"class":4921},[3446,9016,4440],{"class":4439},[3446,9018,4444],{"class":4443},[3446,9020,9021],{"class":3448,"line":3461},[3446,9022,3471],{"emptyLinePlaceholder":3470},[3446,9024,9025,9027,9030,9032,9035,9037,9039,9042,9044,9046,9048,9050,9052,9055,9058,9061,9063,9065,9068],{"class":3448,"line":3467},[3446,9026,4740],{"class":4435},[3446,9028,9029],{"class":4495}," mode",[3446,9031,4499],{"class":4443},[3446,9033,9034],{"class":4495},"args",[3446,9036,3357],{"class":4443},[3446,9038,4933],{"class":4495},[3446,9040,9041],{"class":4443}," > ",[3446,9043,4965],{"class":4502},[3446,9045,8792],{"class":4443},[3446,9047,9034],{"class":4495},[3446,9049,4962],{"class":4443},[3446,9051,4965],{"class":4502},[3446,9053,9054],{"class":4443},"].",[3446,9056,9057],{"class":4733},"ToLower",[3446,9059,9060],{"class":4443},"() : ",[3446,9062,4740],{"class":4435},[3446,9064,3357],{"class":4443},[3446,9066,9067],{"class":4495},"Empty",[3446,9069,4444],{"class":4443},[3446,9071,9072],{"class":3448,"line":3474},[3446,9073,3471],{"emptyLinePlaceholder":3470},[3446,9075,9076,9078,9080,9083],{"class":3448,"line":3480},[3446,9077,3639],{"class":4921},[3446,9079,4925],{"class":4443},[3446,9081,9082],{"class":4495},"mode",[3446,9084,4876],{"class":4443},[3446,9086,9087],{"class":3448,"line":3485},[3446,9088,4467],{"class":4443},[3446,9090,9091,9094,9097],{"class":3448,"line":3491},[3446,9092,9093],{"class":4921},"    case",[3446,9095,9096],{"class":4609}," \"server\"",[3446,9098,9099],{"class":4443},":\n",[3446,9101,9102],{"class":3448,"line":3497},[3446,9103,9104],{"class":4429},"        \u002F\u002F dotnet run -- server MyService 8080\n",[3446,9106,9107,9109,9111,9113,9115,9117,9119,9121,9123,9125,9127,9129,9131,9134,9137,9139,9142,9144,9147,9149,9151],{"class":3448,"line":3503},[3446,9108,5774],{"class":4435},[3446,9110,4743],{"class":4495},[3446,9112,4499],{"class":4443},[3446,9114,9034],{"class":4495},[3446,9116,3357],{"class":4443},[3446,9118,4933],{"class":4495},[3446,9120,9041],{"class":4443},[3446,9122,5046],{"class":4502},[3446,9124,8792],{"class":4443},[3446,9126,9034],{"class":4495},[3446,9128,4962],{"class":4443},[3446,9130,5046],{"class":4502},[3446,9132,9133],{"class":4443},"] : ",[3446,9135,9136],{"class":4609},"$\"Service-",[3446,9138,4772],{"class":4771},[3446,9140,9141],{"class":4495},"Environment",[3446,9143,3357],{"class":4771},[3446,9145,9146],{"class":4495},"MachineName",[3446,9148,4778],{"class":4771},[3446,9150,4809],{"class":4609},[3446,9152,4444],{"class":4443},[3446,9154,9155,9158,9160,9163,9165,9167,9169,9171,9173,9175,9177,9179,9181,9183,9185,9187,9189,9191,9193,9195,9198,9201,9203,9205,9208],{"class":3448,"line":3509},[3446,9156,9157],{"class":4435},"        int",[3446,9159,4759],{"class":4495},[3446,9161,9162],{"class":4443},"    = ",[3446,9164,9034],{"class":4495},[3446,9166,3357],{"class":4443},[3446,9168,4933],{"class":4495},[3446,9170,9041],{"class":4443},[3446,9172,5055],{"class":4502},[3446,9174,8138],{"class":4443},[3446,9176,4756],{"class":4435},[3446,9178,3357],{"class":4443},[3446,9180,4993],{"class":4733},[3446,9182,4737],{"class":4443},[3446,9184,9034],{"class":4495},[3446,9186,4962],{"class":4443},[3446,9188,5055],{"class":4502},[3446,9190,5005],{"class":4443},[3446,9192,5008],{"class":4435},[3446,9194,4492],{"class":4435},[3446,9196,9197],{"class":4495}," p",[3446,9199,9200],{"class":4443},") ? ",[3446,9202,3317],{"class":4495},[3446,9204,5335],{"class":4443},[3446,9206,9207],{"class":4502},"8080",[3446,9209,4444],{"class":4443},[3446,9211,9212],{"class":3448,"line":3515},[3446,9213,3471],{"emptyLinePlaceholder":3470},[3446,9215,9216,9218,9221,9223,9226,9228,9230,9232,9234,9236,9238,9240],{"class":3448,"line":3520},[3446,9217,5974],{"class":4435},[3446,9219,9220],{"class":4921}," using",[3446,9222,7903],{"class":4435},[3446,9224,9225],{"class":4495}," server",[3446,9227,4499],{"class":4443},[3446,9229,5387],{"class":4435},[3446,9231,5332],{"class":4439},[3446,9233,4737],{"class":4443},[3446,9235,4786],{"class":4495},[3446,9237,4746],{"class":4443},[3446,9239,4804],{"class":4495},[3446,9241,4915],{"class":4443},[3446,9243,9244,9246,9248,9250,9253],{"class":3448,"line":3526},[3446,9245,5974],{"class":4435},[3446,9247,9225],{"class":4495},[3446,9249,3357],{"class":4443},[3446,9251,9252],{"class":4733},"RunAsync",[3446,9254,5390],{"class":4443},[3446,9256,9257,9260],{"class":3448,"line":3532},[3446,9258,9259],{"class":4921},"        break",[3446,9261,4444],{"class":4443},[3446,9263,9264],{"class":3448,"line":3537},[3446,9265,3471],{"emptyLinePlaceholder":3470},[3446,9267,9268,9270,9273],{"class":3448,"line":3543},[3446,9269,9093],{"class":4921},[3446,9271,9272],{"class":4609}," \"broadcast\"",[3446,9274,9099],{"class":4443},[3446,9276,9277],{"class":3448,"line":3548},[3446,9278,9279],{"class":4429},"        \u002F\u002F Активний пошук: надсилає broadcast і чекає на відповіді\n",[3446,9281,9282,9284,9286,9288,9290,9293,9295,9297,9299],{"class":3448,"line":3554},[3446,9283,5974],{"class":4435},[3446,9285,9220],{"class":4921},[3446,9287,4925],{"class":4443},[3446,9289,6969],{"class":4435},[3446,9291,9292],{"class":4495}," client",[3446,9294,4499],{"class":4443},[3446,9296,5387],{"class":4435},[3446,9298,7370],{"class":4439},[3446,9300,6986],{"class":4443},[3446,9302,9303],{"class":3448,"line":3560},[3446,9304,6129],{"class":4443},[3446,9306,9307,9310,9313,9315,9317,9319,9321,9323,9325,9327,9329,9331],{"class":3448,"line":3566},[3446,9308,9309],{"class":4495},"            var",[3446,9311,9312],{"class":4495}," services",[3446,9314,4499],{"class":4443},[3446,9316,6157],{"class":4435},[3446,9318,9292],{"class":4495},[3446,9320,3357],{"class":4443},[3446,9322,7774],{"class":4733},[3446,9324,4737],{"class":4443},[3446,9326,7929],{"class":4495},[3446,9328,8855],{"class":4443},[3446,9330,5002],{"class":4502},[3446,9332,4915],{"class":4443},[3446,9334,9335],{"class":3448,"line":3572},[3446,9336,3471],{"emptyLinePlaceholder":3470},[3446,9338,9339,9341,9343,9345],{"class":3448,"line":3578},[3446,9340,6309],{"class":4495},[3446,9342,3357],{"class":4443},[3446,9344,5794],{"class":4733},[3446,9346,5390],{"class":4443},[3446,9348,9349,9351,9353,9355,9357,9360,9362,9365,9367,9370,9372,9375],{"class":3448,"line":3584},[3446,9350,6309],{"class":4495},[3446,9352,3357],{"class":4443},[3446,9354,5794],{"class":4733},[3446,9356,4737],{"class":4443},[3446,9358,9359],{"class":4609},"$\"=== Знайдено сервісів: ",[3446,9361,4772],{"class":4771},[3446,9363,9364],{"class":4495},"services",[3446,9366,3357],{"class":4771},[3446,9368,9369],{"class":4495},"Count",[3446,9371,4778],{"class":4771},[3446,9373,9374],{"class":4609}," ===\"",[3446,9376,4915],{"class":4443},[3446,9378,9379,9381,9383,9385,9388,9390,9392],{"class":3448,"line":3589},[3446,9380,7062],{"class":4733},[3446,9382,4925],{"class":4443},[3446,9384,6969],{"class":4495},[3446,9386,9387],{"class":4495}," svc",[3446,9389,6975],{"class":4435},[3446,9391,9312],{"class":4495},[3446,9393,4876],{"class":4443},[3446,9395,9396,9398,9400,9402,9404,9407,9409,9412,9414,9416],{"class":3448,"line":3595},[3446,9397,6218],{"class":4495},[3446,9399,3357],{"class":4443},[3446,9401,5794],{"class":4733},[3446,9403,4737],{"class":4443},[3446,9405,9406],{"class":4609},"$\"  • ",[3446,9408,4772],{"class":4771},[3446,9410,9411],{"class":4495},"svc",[3446,9413,4778],{"class":4771},[3446,9415,4809],{"class":4609},[3446,9417,4915],{"class":4443},[3446,9419,9420],{"class":3448,"line":3601},[3446,9421,6592],{"class":4443},[3446,9423,9424,9426],{"class":3448,"line":3607},[3446,9425,9259],{"class":4921},[3446,9427,4444],{"class":4443},[3446,9429,9430],{"class":3448,"line":3613},[3446,9431,3471],{"emptyLinePlaceholder":3470},[3446,9433,9434,9436,9439],{"class":3448,"line":3619},[3446,9435,9093],{"class":4921},[3446,9437,9438],{"class":4609}," \"multicast\"",[3446,9440,9099],{"class":4443},[3446,9442,9443],{"class":3448,"line":3625},[3446,9444,9445],{"class":4429},"        \u002F\u002F Пасивне прослуховування multicast-анонсів\n",[3446,9447,9448,9450,9452,9454,9456,9458,9460,9462,9464],{"class":3448,"line":3630},[3446,9449,5974],{"class":4435},[3446,9451,9220],{"class":4921},[3446,9453,4925],{"class":4443},[3446,9455,6969],{"class":4435},[3446,9457,9292],{"class":4495},[3446,9459,4499],{"class":4443},[3446,9461,5387],{"class":4435},[3446,9463,7370],{"class":4439},[3446,9465,6986],{"class":4443},[3446,9467,9468],{"class":3448,"line":4718},[3446,9469,6129],{"class":4443},[3446,9471,9472,9475,9477,9479,9482],{"class":3448,"line":4724},[3446,9473,9474],{"class":4435},"            await",[3446,9476,9292],{"class":4495},[3446,9478,3357],{"class":4443},[3446,9480,9481],{"class":4733},"ListenMulticastAsync",[3446,9483,5390],{"class":4443},[3446,9485,9486],{"class":3448,"line":4765},[3446,9487,6592],{"class":4443},[3446,9489,9490,9492],{"class":3448,"line":4814},[3446,9491,9259],{"class":4921},[3446,9493,4444],{"class":4443},[3446,9495,9496],{"class":3448,"line":4819},[3446,9497,3471],{"emptyLinePlaceholder":3470},[3446,9499,9500,9503],{"class":3448,"line":4830},[3446,9501,9502],{"class":4921},"    default",[3446,9504,9099],{"class":4443},[3446,9506,9507,9509,9511,9513,9515,9518],{"class":3448,"line":4836},[3446,9508,5789],{"class":4495},[3446,9510,3357],{"class":4443},[3446,9512,5794],{"class":4733},[3446,9514,4737],{"class":4443},[3446,9516,9517],{"class":4609},"\"Використання:\"",[3446,9519,4915],{"class":4443},[3446,9521,9522,9524,9526,9528,9530,9533],{"class":3448,"line":4842},[3446,9523,5789],{"class":4495},[3446,9525,3357],{"class":4443},[3446,9527,5794],{"class":4733},[3446,9529,4737],{"class":4443},[3446,9531,9532],{"class":4609},"\"  dotnet run -- server [name] [port]   — запустити сервер\"",[3446,9534,4915],{"class":4443},[3446,9536,9537,9539,9541,9543,9545,9548],{"class":3448,"line":4853},[3446,9538,5789],{"class":4495},[3446,9540,3357],{"class":4443},[3446,9542,5794],{"class":4733},[3446,9544,4737],{"class":4443},[3446,9546,9547],{"class":4609},"\"  dotnet run -- broadcast              — знайти сервери (broadcast)\"",[3446,9549,4915],{"class":4443},[3446,9551,9552,9554,9556,9558,9560,9563],{"class":3448,"line":4879},[3446,9553,5789],{"class":4495},[3446,9555,3357],{"class":4443},[3446,9557,5794],{"class":4733},[3446,9559,4737],{"class":4443},[3446,9561,9562],{"class":4609},"\"  dotnet run -- multicast              — слухати multicast-анонси\"",[3446,9564,4915],{"class":4443},[3446,9566,9567,9569],{"class":3448,"line":4885},[3446,9568,9259],{"class":4921},[3446,9570,4444],{"class":4443},[3446,9572,9573],{"class":3448,"line":4891},[3446,9574,3529],{"class":4443},[3370,9576],{},[3377,9578,9580],{"id":9579},"запуск-та-демонстрація","Запуск та демонстрація",[3317,9582,9583],{},"Для тестування відкрийте кілька термінальних вікон в одній директорії проєкту.",[9585,9586,9587,9591,9636,9640,9678,9682,9722,9726],"steps",{},[3377,9588,9590],{"id":9589},"запустіть-сервер-термінал-1","Запустіть сервер (термінал 1)",[4303,9592,9594,9603,9610,9617,9624,9628],{"title":9593},"Terminal 1 — Discovery Server",[4307,9595,9597,4315,9600],{"className":9596},[3448],[3446,9598,4314],{"className":9599},[4313],[3327,9601,9602],{},"dotnet run -- server MyApiService 8080",[4307,9604,9606],{"className":9605},[3448],[3446,9607,9609],{"className":9608},[4354],"[Server] Сервіс 'MyApiService' на 192.168.1.15:8080",[4307,9611,9613],{"className":9612},[3448],[3446,9614,9616],{"className":9615},[4354],"[Server] Broadcast listener: :9876",[4307,9618,9620],{"className":9619},[3448],[3446,9621,9623],{"className":9622},[4354],"[Server] Multicast announcer: 239.255.99.1:9877",[4307,9625,9627],{"className":9626},[3448],"[Server] Ctrl+C для зупинки.",[4307,9629,9631],{"className":9630},[3448],[3446,9632,9635],{"className":9633},[9634],"text-gray-400","[Server] Multicast → 'ANNOUNCE:MyApiService:192.168.1.15:8080'",[3377,9637,9639],{"id":9638},"знайдіть-сервер-через-broadcast-термінал-2","Знайдіть сервер через Broadcast (термінал 2)",[4303,9641,9643,9652,9656,9660,9667,9670,9674],{"title":9642},"Terminal 2 — Broadcast Discovery",[4307,9644,9646,4315,9649],{"className":9645},[3448],[3446,9647,4314],{"className":9648},[4313],[3327,9650,9651],{},"dotnet run -- broadcast",[4307,9653,9655],{"className":9654},[3448],"[Client] Broadcast DISCOVER? надіслано → 255.255.255.255:9876",[4307,9657,9659],{"className":9658},[3448],"[Client] Збираємо відповіді протягом 3с...",[4307,9661,9663],{"className":9662},[3448],[3446,9664,9666],{"className":9665},[4325],"[Client] ✅ Знайдено (broadcast): MyApiService @ 192.168.1.15:8080",[4307,9668],{"className":9669},[3448],[4307,9671,9673],{"className":9672},[3448],"=== Знайдено сервісів: 1 ===",[4307,9675,9677],{"className":9676},[3448],"  • MyApiService @ 192.168.1.15:8080",[3377,9679,9681],{"id":9680},"слухайте-multicast-анонси-термінал-3","Слухайте multicast-анонси (термінал 3)",[4303,9683,9685,9694,9698,9702,9709,9716],{"title":9684},"Terminal 3 — Multicast Listener",[4307,9686,9688,4315,9691],{"className":9687},[3448],[3446,9689,4314],{"className":9690},[4313],[3327,9692,9693],{},"dotnet run -- multicast",[4307,9695,9697],{"className":9696},[3448],"[Client] Слухаємо multicast 239.255.99.1:9877...",[4307,9699,9701],{"className":9700},[3448],"[Client] Ctrl+C для зупинки.",[4307,9703,9705],{"className":9704},[3448],[3446,9706,9708],{"className":9707},[4325],"[Client] ✅ Новий сервіс: MyApiService @ 192.168.1.15:8080",[4307,9710,9712],{"className":9711},[3448],[3446,9713,9715],{"className":9714},[9634],"[Client] 🔄 Анонс: MyApiService @ 192.168.1.15:8080",[4307,9717,9719],{"className":9718},[3448],[3446,9720,9715],{"className":9721},[9634],[3377,9723,9725],{"id":9724},"вивід-сервера-після-broadcast-запиту","Вивід сервера після broadcast-запиту",[4303,9727,9729,9735,9742,9749],{"title":9728},"Terminal 1 — Server (after broadcast)",[4307,9730,9732],{"className":9731},[3448],[3446,9733,9635],{"className":9734},[9634],[4307,9736,9738],{"className":9737},[3448],[3446,9739,9741],{"className":9740},[4354],"[Server] Broadcast від 192.168.1.7:54231: 'DISCOVER?'",[4307,9743,9745],{"className":9744},[3448],[3446,9746,9748],{"className":9747},[4325],"[Server] → Unicast відповідь: 'ANNOUNCE:MyApiService:192.168.1.15:8080'",[4307,9750,9752],{"className":9751},[3448],[3446,9753,9635],{"className":9754},[9634],[3370,9756],{},[3312,9758,9760],{"id":9759},"підсумок","Підсумок",[3650,9762,9763,9796],{},[3653,9764,9767],{"icon":9765,"title":9766},"i-lucide-radio","📡 Broadcast",[3657,9768,9769,9776,9783,9790,9793],{},[3324,9770,9771,9772,9775],{},"Доставляє пакет ",[3327,9773,9774],{},"усім"," у локальній підмережі",[3324,9777,9778,9779,9782],{},"Адреса ",[3415,9780,9781],{},"255.255.255.255"," — limited broadcast, не виходить за LAN",[3324,9784,9785,9786,9789],{},"Потрібно ",[3415,9787,9788],{},"EnableBroadcast = true"," на сокеті",[3324,9791,9792],{},"Відсутній в IPv6 — замінений multicast",[3324,9794,9795],{},"Ідеально для: DHCP, ARP, service discovery в LAN",[3653,9797,9800],{"icon":9798,"title":9799},"i-lucide-signal","📶 Multicast",[3657,9801,9802,9810,9815,9822,9825],{},[3324,9803,9804,9805,4925,9808,3871],{},"Доставляє лише ",[3327,9806,9807],{},"підписаним",[3415,9809,7713],{},[3324,9811,9812,9813],{},"Адреси класу D: ",[3415,9814,3728],{},[3324,9816,9817,9818,9821],{},"Для локальних застосунків: ",[3415,9819,9820],{},"239.x.x.x"," (Site-Local)",[3324,9823,9824],{},"Протокол підписки: IGMP (ОС керує автоматично)",[3324,9826,9827,9828],{},"При завершенні — ",[3415,9829,8901],{},[3793,9831,9832,9837],{},[3317,9833,9834],{},[3327,9835,9836],{},"Коли що обирати:",[3657,9838,9839,9844],{},[3324,9840,9841,9843],{},[3327,9842,3352],{}," — коли потрібно знайти щось у LAN одним запитом (DHCP-стиль).",[3324,9845,9846,9848],{},[3327,9847,3356],{}," — коли є постійний потік даних для групи підписників, або коли важлива масштабованість за межі одного сегменту.",[9850,9851,9852],"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 .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}html pre.shiki code .s0P7L, html code.shiki .s0P7L{--shiki-light:#800000;--shiki-default:#808080;--shiki-dark:#808080}html pre.shiki code .sKtos, html code.shiki .sKtos{--shiki-light:#800000;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sD7JJ, html code.shiki .sD7JJ{--shiki-light:#000000FF;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .s8xlr, html code.shiki .s8xlr{--shiki-light:#AF00DB;--shiki-default:#C586C0;--shiki-dark:#C586C0}",{"title":3442,"searchDepth":3455,"depth":3455,"links":9854},[9855,9856,9862,9869,9883],{"id":3314,"depth":3455,"text":3315},{"id":3374,"depth":3455,"text":3375,"children":9857},[9858,9859,9860,9861],{"id":3379,"depth":3461,"text":3380},{"id":3392,"depth":3461,"text":3393},{"id":3431,"depth":3461,"text":3432},{"id":3647,"depth":3461,"text":3648},{"id":3701,"depth":3455,"text":3702,"children":9863},[9864,9865,9866,9867,9868],{"id":3705,"depth":3461,"text":3706},{"id":3721,"depth":3461,"text":3722},{"id":3811,"depth":3461,"text":3812},{"id":3900,"depth":3461,"text":3901},{"id":4060,"depth":3461,"text":4061},{"id":4166,"depth":3455,"text":4167,"children":9870},[9871,9872,9873,9874,9875,9876,9877,9878,9879,9880,9881,9882],{"id":4177,"depth":3461,"text":4178},{"id":4297,"depth":3461,"text":4298},{"id":4358,"depth":3461,"text":3824},{"id":4415,"depth":3461,"text":4416},{"id":5193,"depth":3461,"text":5194},{"id":7254,"depth":3461,"text":7255},{"id":8999,"depth":3461,"text":9000},{"id":9579,"depth":3461,"text":9580},{"id":9589,"depth":3461,"text":9590},{"id":9638,"depth":3461,"text":9639},{"id":9680,"depth":3461,"text":9681},{"id":9724,"depth":3461,"text":9725},{"id":9759,"depth":3455,"text":9760},"Детальний розбір механізмів групової розсилки UDP — Broadcast та Multicast. Теорія, адресний простір, практичний приклад сервісу виявлення пристроїв у локальній мережі на C#.","md",null,{},{"title":1676,"description":9884},"Fr36zRzktRmB3_CHgMCuNhUFROyduPRGTu7JDs-pwJc",[9891,9893],{"title":1672,"path":1673,"stem":1674,"description":9892,"children":-1},"Глибоке вивчення UDP — від теоретичних засад протоколу до повноцінного практичного прикладу чату та ігрового сервера на C#. Академічний розбір структури дейтаграм, порівняння з TCP та реальні патерни застосування.",{"title":1680,"path":1681,"stem":1682,"description":9894,"children":-1},"Глибоке вивчення HTTP — від структури повідомлень та методів до повноцінного REST API клієнта на C#. Академічний розбір RFC 7230–7235, анатомія запитів і відповідей, статус-коди, заголовки та HttpClient екосистема .NET.",1781795350813]