[{"data":1,"prerenderedAt":8811},["ShallowReactive",2],{"navigation_docs":3,"-csharp-aspnet-testing-postman-professional":2949,"-csharp-aspnet-testing-postman-professional-surround":8806},[4,1640,1765,2219,2352,2559,2641,2691,2748,2782,2908,2945],{"title":5,"icon":6,"path":7,"stem":8,"children":9},"C#","i-devicon-csharp","/csharp","01.csharp",[10,13,60,90,120,202,219,253,379,404,457,650,1346,1636],{"title":11,"path":7,"stem":12},"C# Roadmap","01.csharp/index",{"title":14,"icon":15,"path":16,"stem":17,"children":18,"page":59},"Fundamentals","i-lucide-book-open","/csharp/fundamentals","01.csharp/01.fundamentals",[19,23,27,31,35,39,43,47,51,55],{"title":20,"path":21,"stem":22},"Вступ до екосистеми .NET","/csharp/fundamentals/introduction-to-ecosystem","01.csharp/01.fundamentals/01.introduction-to-ecosystem",{"title":24,"path":25,"stem":26},"Структура програми на C#","/csharp/fundamentals/program-structure","01.csharp/01.fundamentals/02.program-structure",{"title":28,"path":29,"stem":30},"Змінні та Типи Даних","/csharp/fundamentals/variables-data-types","01.csharp/01.fundamentals/03.variables-data-types",{"title":32,"path":33,"stem":34},"Масиви","/csharp/fundamentals/arrays","01.csharp/01.fundamentals/04.arrays",{"title":36,"path":37,"stem":38},"Strings & Text Handling","/csharp/fundamentals/strings-text-handling","01.csharp/01.fundamentals/05.strings-text-handling",{"title":40,"path":41,"stem":42},"Дати і Час","/csharp/fundamentals/dates-time-handling","01.csharp/01.fundamentals/06.dates-time-handling",{"title":44,"path":45,"stem":46},"Потік Керування","/csharp/fundamentals/control-flow","01.csharp/01.fundamentals/07.control-flow",{"title":48,"path":49,"stem":50},"Методи","/csharp/fundamentals/methods","01.csharp/01.fundamentals/08.methods",{"title":52,"path":53,"stem":54},"Основи Відлагодження","/csharp/fundamentals/debugging-basics","01.csharp/01.fundamentals/09.debugging-basics",{"title":56,"path":57,"stem":58},"Інтерактивна Консоль (Classic)","/csharp/fundamentals/interactive-console","01.csharp/01.fundamentals/10.interactive-console",false,{"title":61,"icon":62,"path":63,"stem":64,"children":65,"page":59},"OOP","i-lucide-box","/csharp/oop","01.csharp/02.oop",[66,70,74,78,82,86],{"title":67,"path":68,"stem":69},"Package Management (Управління Пакетами)","/csharp/oop/package-management","01.csharp/02.oop/01.package-management",{"title":71,"path":72,"stem":73},"Класи та Об'єкти","/csharp/oop/classes-objects","01.csharp/02.oop/02.classes-objects",{"title":75,"path":76,"stem":77},"Властивості та Поля","/csharp/oop/properties-fields","01.csharp/02.oop/03.properties-fields",{"title":79,"path":80,"stem":81},"Стовпи ООП","/csharp/oop/oop-pillars","01.csharp/02.oop/04.oop-pillars",{"title":83,"path":84,"stem":85},"Advanced Types","/csharp/oop/advanced-types","01.csharp/02.oop/05.advanced-types",{"title":87,"path":88,"stem":89},"Namespaces (Простори Імен)","/csharp/oop/namespaces","01.csharp/02.oop/06.namespaces",{"title":91,"icon":92,"path":93,"stem":94,"children":95,"page":59},"Advanced Core","i-lucide-zap","/csharp/advanced-core","01.csharp/03.advanced-core",[96,100,104,108,112,116],{"title":97,"path":98,"stem":99},"Generics (Узагальнення)","/csharp/advanced-core/generics","01.csharp/03.advanced-core/01.generics",{"title":101,"path":102,"stem":103},"Делегати, Події та Лямбда-вирази","/csharp/advanced-core/delegates-events-lambdas","01.csharp/03.advanced-core/02.delegates-events-lambdas",{"title":105,"path":106,"stem":107},"Interfaces Deep Dive (Інтерфейси: Поглиблений Розгляд)","/csharp/advanced-core/interfaces-deep-dive","01.csharp/03.advanced-core/03.interfaces-deep-dive",{"title":109,"path":110,"stem":111},"Обробка Винятків","/csharp/advanced-core/exception-handling","01.csharp/03.advanced-core/04.exception-handling",{"title":113,"path":114,"stem":115},"Pattern Matching","/csharp/advanced-core/pattern-matching","01.csharp/03.advanced-core/05.pattern-matching",{"title":117,"path":118,"stem":119},"Додаткові Можливості C#","/csharp/advanced-core/additional-features","01.csharp/03.advanced-core/06.additional-features",{"title":121,"icon":122,"path":123,"stem":124,"children":125,"page":59},"Architecture Best Practices","i-lucide-building-2","/csharp/architecture-best-practices","01.csharp/04.architecture-best-practices",[126,130,149,153,157,161,165,169],{"title":127,"path":128,"stem":129},"Software Design Principles (Частина 1)","/csharp/architecture-best-practices/software-design-principles","01.csharp/04.architecture-best-practices/01.software-design-principles",{"title":131,"icon":132,"path":133,"stem":134,"children":135,"page":59},"Design Patterns","i-lucide-folder","/csharp/architecture-best-practices/design-patterns","01.csharp/04.architecture-best-practices/02.design-patterns",[136],{"title":137,"icon":132,"path":138,"stem":139,"children":140,"page":59},"Creational","/csharp/architecture-best-practices/design-patterns/creational","01.csharp/04.architecture-best-practices/02.design-patterns/creational",[141,145],{"title":142,"path":143,"stem":144},"Singleton (Одинак)","/csharp/architecture-best-practices/design-patterns/creational/singleton","01.csharp/04.architecture-best-practices/02.design-patterns/creational/01.singleton",{"title":146,"path":147,"stem":148},"Builder (Будівельник)","/csharp/architecture-best-practices/design-patterns/creational/builder","01.csharp/04.architecture-best-practices/02.design-patterns/creational/02.builder",{"title":150,"path":151,"stem":152},"Building Professional CLIs","/csharp/architecture-best-practices/building-professional-clis","01.csharp/04.architecture-best-practices/03.building-professional-clis",{"title":154,"path":155,"stem":156},"Validation & Flow Control","/csharp/architecture-best-practices/validation-flow-control","01.csharp/04.architecture-best-practices/04.validation-flow-control",{"title":158,"path":159,"stem":160},"The Modern .NET Host (Microsoft.Extensions)","/csharp/architecture-best-practices/modern-dotnet-host","01.csharp/04.architecture-best-practices/05.modern-dotnet-host",{"title":162,"path":163,"stem":164},"Data Mapper: Repository та DAO патерни (Частина 1)","/csharp/architecture-best-practices/data-mapper-part1","01.csharp/04.architecture-best-practices/06.data-mapper-part1",{"title":166,"path":167,"stem":168},"Data Mapper: Repository та DAO патерни (Частина 2)","/csharp/architecture-best-practices/data-mapper-part2","01.csharp/04.architecture-best-practices/07.data-mapper-part2",{"title":170,"icon":132,"path":171,"stem":172,"children":173,"page":59},"Di Ioc","/csharp/architecture-best-practices/di-ioc","01.csharp/04.architecture-best-practices/08.di-ioc",[174,178,182,186,190,194,198],{"title":175,"path":176,"stem":177},"Проблема залежностей та Інверсія Контролю","/csharp/architecture-best-practices/di-ioc/the-dependency-problem","01.csharp/04.architecture-best-practices/08.di-ioc/01.the-dependency-problem",{"title":179,"path":180,"stem":181},"Будуємо власний Service Container","/csharp/architecture-best-practices/di-ioc/build-your-own-container","01.csharp/04.architecture-best-practices/08.di-ioc/02.build-your-own-container",{"title":183,"path":184,"stem":185},"Service Locator: Паттерн та Анти-паттерн","/csharp/architecture-best-practices/di-ioc/service-locator-pattern","01.csharp/04.architecture-best-practices/08.di-ioc/03.service-locator-pattern",{"title":187,"path":188,"stem":189},"Паттерни Dependency Injection","/csharp/architecture-best-practices/di-ioc/dependency-injection-patterns","01.csharp/04.architecture-best-practices/08.di-ioc/04.dependency-injection-patterns",{"title":191,"path":192,"stem":193},"Microsoft DI: IServiceCollection та IServiceProvider","/csharp/architecture-best-practices/di-ioc/microsoft-di-deep-dive","01.csharp/04.architecture-best-practices/08.di-ioc/05.microsoft-di-deep-dive",{"title":195,"path":196,"stem":197},"Service Lifetimes та Scopes","/csharp/architecture-best-practices/di-ioc/service-lifetimes-and-scopes","01.csharp/04.architecture-best-practices/08.di-ioc/06.service-lifetimes-and-scopes",{"title":199,"path":200,"stem":201},"DI Анти-паттерни та Найкращі Практики","/csharp/architecture-best-practices/di-ioc/di-anti-patterns-and-best-practices","01.csharp/04.architecture-best-practices/08.di-ioc/07.di-anti-patterns-and-best-practices",{"title":203,"icon":132,"path":204,"stem":205,"children":206,"page":59},"Standard Library","/csharp/standard-library","01.csharp/05.standard-library",[207,211,215],{"title":208,"path":209,"stem":210},"Collections (Колекції)","/csharp/standard-library/collections","01.csharp/05.standard-library/01.collections",{"title":212,"path":213,"stem":214},"High Performance Types (Високопродуктивні Типи)","/csharp/standard-library/high-performance-types","01.csharp/05.standard-library/02.high-performance-types",{"title":216,"path":217,"stem":218},"LINQ (Language Integrated Query)","/csharp/standard-library/linq","01.csharp/05.standard-library/03.linq",{"title":220,"icon":221,"path":222,"stem":223,"children":224,"page":59},"System Internals Concurrency","i-lucide-server","/csharp/system-internals-concurrency","01.csharp/06.system-internals-concurrency",[225,229,233,237,241,245,249],{"title":226,"path":227,"stem":228},"Memory Management","/csharp/system-internals-concurrency/memory-management","01.csharp/06.system-internals-concurrency/01.memory-management",{"title":230,"path":231,"stem":232},"Reflection API: System.Type та Метадані","/csharp/system-internals-concurrency/reflection-fundamentals","01.csharp/06.system-internals-concurrency/02.reflection-fundamentals",{"title":234,"path":235,"stem":236},"Attributes та Dynamic Language Runtime","/csharp/system-internals-concurrency/attributes-dynamic","01.csharp/06.system-internals-concurrency/03.attributes-dynamic",{"title":238,"path":239,"stem":240},"Expression Trees: Швидка Альтернатива Рефлексії","/csharp/system-internals-concurrency/expression-trees-compiled","01.csharp/06.system-internals-concurrency/04.expression-trees-compiled",{"title":242,"path":243,"stem":244},"Source Generators: Compile-Time Code Generation","/csharp/system-internals-concurrency/source-generators","01.csharp/06.system-internals-concurrency/05.source-generators",{"title":246,"path":247,"stem":248},"Multithreading Fundamentals","/csharp/system-internals-concurrency/multithreading-fundamentals","01.csharp/06.system-internals-concurrency/06.multithreading-fundamentals",{"title":250,"path":251,"stem":252},"Synchronization Primitives","/csharp/system-internals-concurrency/synchronization-primitives","01.csharp/06.system-internals-concurrency/07.synchronization-primitives",{"title":254,"icon":255,"path":256,"stem":257,"children":258,"page":59},"System Programming Windows","i-lucide-cpu","/csharp/system-programming-windows","01.csharp/07.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},"Як Працює Операційна Система","/csharp/system-programming-windows/how-os-works","01.csharp/07.system-programming-windows/01.how-os-works",{"title":264,"path":265,"stem":266},"Процеси в .NET — API та Запуск","/csharp/system-programming-windows/processes-in-dotnet","01.csharp/07.system-programming-windows/02.processes-in-dotnet",{"title":268,"path":269,"stem":270},"Процеси в .NET — IPC та Міжпроцесна Комунікація","/csharp/system-programming-windows/02a.processes-ipc","01.csharp/07.system-programming-windows/02a.processes-ipc",{"title":272,"path":273,"stem":274},"Application Domains та Збірки — AppDomain і AssemblyLoadContext","/csharp/system-programming-windows/appdomains-assemblies","01.csharp/07.system-programming-windows/03.appdomains-assemblies",{"title":276,"path":277,"stem":278},"Application Domains та Збірки — Plug-in Система з Hot-Reload","/csharp/system-programming-windows/03a.appdomains-plugin-system","01.csharp/07.system-programming-windows/03a.appdomains-plugin-system",{"title":280,"path":281,"stem":282},"Потоки — Основи та API Thread","/csharp/system-programming-windows/thread-fundamentals","01.csharp/07.system-programming-windows/04.thread-fundamentals",{"title":284,"path":285,"stem":286},"Потоки — Lifecycle, Пріоритети та Безпечне Завершення","/csharp/system-programming-windows/04a.thread-lifecycle-priorities","01.csharp/07.system-programming-windows/04a.thread-lifecycle-priorities",{"title":288,"path":289,"stem":290},"Проблеми Спільного Стану — Race Condition та Data Race","/csharp/system-programming-windows/shared-state-problems","01.csharp/07.system-programming-windows/05.shared-state-problems",{"title":292,"path":293,"stem":294},"Проблеми Спільного Стану — Memory Model та volatile","/csharp/system-programming-windows/05a.shared-state-memory-model","01.csharp/07.system-programming-windows/05a.shared-state-memory-model",{"title":296,"path":297,"stem":298},"Синхронізація — Monitor, lock та еволюція примітивів","/csharp/system-programming-windows/synchronization-fundamentals","01.csharp/07.system-programming-windows/06.synchronization-fundamentals",{"title":300,"path":301,"stem":302},"Синхронізація — Наскрізний Приклад та Deadlock Detection","/csharp/system-programming-windows/06a.synchronization-walkthrough","01.csharp/07.system-programming-windows/06a.synchronization-walkthrough",{"title":304,"path":305,"stem":306},"Синхронізація — Mutex, Semaphore та Event-Based Primitives","/csharp/system-programming-windows/synchronization-advanced","01.csharp/07.system-programming-windows/07.synchronization-advanced",{"title":308,"path":309,"stem":310},"Синхронізація — Interlocked, Volatile та Lock-Free Структури","/csharp/system-programming-windows/07a.synchronization-advanced-walkthrough","01.csharp/07.system-programming-windows/07a.synchronization-advanced-walkthrough",{"title":312,"path":313,"stem":314},"Interlocked, CAS та Lock-Free Структури","/csharp/system-programming-windows/interlocked-cas-lockfree","01.csharp/07.system-programming-windows/08.interlocked-cas-lockfree",{"title":316,"path":317,"stem":318},"Volatile, Memory Model та Spinning","/csharp/system-programming-windows/08a.volatile-memory-model","01.csharp/07.system-programming-windows/08a.volatile-memory-model",{"title":320,"path":321,"stem":322},"ThreadPool — Пул Потоків для Ефективного Виконання","/csharp/system-programming-windows/thread-pool","01.csharp/07.system-programming-windows/09.thread-pool",{"title":324,"path":325,"stem":326},"ThreadPool — Просунуті Сценарії та Внутрішня Будова","/csharp/system-programming-windows/09a.thread-pool-advanced","01.csharp/07.system-programming-windows/09a.thread-pool-advanced",{"title":328,"path":329,"stem":330},"Concurrent та Immutable Collections","/csharp/system-programming-windows/concurrent-collections","01.csharp/07.system-programming-windows/10.concurrent-collections",{"title":332,"path":333,"stem":334},"TPL, Task та Композиція — Від Thread до Task","/csharp/system-programming-windows/tpl-parallel-plinq","01.csharp/07.system-programming-windows/11.tpl-parallel-plinq",{"title":336,"path":337,"stem":338},"Parallel Class та PLINQ — Data Parallelism","/csharp/system-programming-windows/11a.tpl-parallel-plinq-advanced","01.csharp/07.system-programming-windows/11a.tpl-parallel-plinq-advanced",{"title":340,"path":341,"stem":342},"Async/Await — Фундамент Асинхронного Програмування","/csharp/system-programming-windows/async-fundamentals","01.csharp/07.system-programming-windows/12.async-fundamentals",{"title":344,"path":345,"stem":346},"SynchronizationContext та ConfigureAwait — Контекст Виконання","/csharp/system-programming-windows/async-context-configureawait","01.csharp/07.system-programming-windows/13.async-context-configureawait",{"title":348,"path":349,"stem":350},"Async — Просунуті Паттерни","/csharp/system-programming-windows/async-advanced","01.csharp/07.system-programming-windows/14.async-advanced",{"title":352,"path":353,"stem":354},"System.Threading.Channels — Async Producer-Consumer","/csharp/system-programming-windows/channels","01.csharp/07.system-programming-windows/15.channels",{"title":356,"path":357,"stem":358},"Асинхронна Синхронізація","/csharp/system-programming-windows/async-synchronization","01.csharp/07.system-programming-windows/16.async-synchronization",{"title":360,"path":361,"stem":362},"Unsafe Code та Вказівники","/csharp/system-programming-windows/unsafe-code","01.csharp/07.system-programming-windows/17.unsafe-code",{"title":364,"path":365,"stem":366},"P/Invoke та Windows API — Міст між .NET та Native Code","/csharp/system-programming-windows/pinvoke-winapi","01.csharp/07.system-programming-windows/18.pinvoke-winapi",{"title":368,"path":369,"stem":370},"Реєстр Windows — Центральна База Конфігурації Системи","/csharp/system-programming-windows/windows-registry","01.csharp/07.system-programming-windows/19.windows-registry",{"title":372,"path":373,"stem":374},"Windows Hooks, Hotkeys та Services — Глибока Інтеграція з ОС","/csharp/system-programming-windows/windows-hooks-services","01.csharp/07.system-programming-windows/20.windows-hooks-services",{"title":376,"path":377,"stem":378},"Системне Програмування C# (Windows) — 07.system-programming-windows","/csharp/system-programming-windows/implementation_plan","01.csharp/07.system-programming-windows/implementation_plan",{"title":380,"icon":132,"path":381,"stem":382,"children":383,"page":59},"Io","/csharp/io","01.csharp/08.io",[384,388,392,396,400],{"title":385,"path":386,"stem":387},"8.1.1. Основи роботи з файловою системою","/csharp/io/file-system-basics","01.csharp/08.io/01.file-system-basics",{"title":389,"path":390,"stem":391},"8.1.2. Потоки (Streams) та Серіалізація Даних","/csharp/io/streams-serialization","01.csharp/08.io/02.streams-serialization",{"title":393,"path":394,"stem":395},"8.2.1. JSON Serialization з System.Text.Json","/csharp/io/json-serialization","01.csharp/08.io/03.json-serialization",{"title":397,"path":398,"stem":399},"8.2.2. XML Serialization та LINQ to XML","/csharp/io/xml-serialization","01.csharp/08.io/04.xml-serialization",{"title":401,"path":402,"stem":403},"8.2.3. Binary Serialization: MessagePack та Protocol Buffers","/csharp/io/binary-serialization","01.csharp/08.io/05.binary-serialization",{"title":405,"icon":132,"path":406,"stem":407,"children":408,"page":59},"Ado Net","/csharp/ado-net","01.csharp/09.ado-net",[409,413,417,421,425,429,433,437,441,445,449,453],{"title":410,"path":411,"stem":412},"9.1. Введення в ADO.NET","/csharp/ado-net/introduction-to-adonet","01.csharp/09.ado-net/01.introduction-to-adonet",{"title":414,"path":415,"stem":416},"9.2. Клас DbConnection — з'єднання з базою даних","/csharp/ado-net/connection","01.csharp/09.ado-net/02.connection",{"title":418,"path":419,"stem":420},"9.3. Клас DbCommand — виконання SQL-запитів","/csharp/ado-net/command-and-queries","01.csharp/09.ado-net/03.command-and-queries",{"title":422,"path":423,"stem":424},"9.4. Клас DbDataReader — ефективне читання даних","/csharp/ado-net/datareader","01.csharp/09.ado-net/04.datareader",{"title":426,"path":427,"stem":428},"9.5. Параметризовані запити та захист від SQL Injection","/csharp/ado-net/parameters-and-sql-injection","01.csharp/09.ado-net/05.parameters-and-sql-injection",{"title":430,"path":431,"stem":432},"9.6. Транзакції в ADO.NET","/csharp/ado-net/transactions","01.csharp/09.ado-net/06.transactions",{"title":434,"path":435,"stem":436},"9.7. DbProviderFactory — провайдер-незалежний код","/csharp/ado-net/provider-factory","01.csharp/09.ado-net/07.provider-factory",{"title":438,"path":439,"stem":440},"9.8. Асинхронний доступ до даних","/csharp/ado-net/async-data-access","01.csharp/09.ado-net/08.async-data-access",{"title":442,"path":443,"stem":444},"9.9. Від'єднаний режим: DataSet, DataTable, DataRow","/csharp/ado-net/disconnected-mode-dataset","01.csharp/09.ado-net/09.disconnected-mode-dataset",{"title":446,"path":447,"stem":448},"9.10. DataAdapter — міст між DataSet та базою даних","/csharp/ado-net/data-adapter","01.csharp/09.ado-net/10.data-adapter",{"title":450,"path":451,"stem":452},"9.11. Data Mapper та Repository: Архітектура доступу до даних","/csharp/ado-net/data-mapper-repository","01.csharp/09.ado-net/11.data-mapper-repository",{"title":454,"path":455,"stem":456},"9.12. Identity Map, Unit of Work та Specification Pattern","/csharp/ado-net/advanced-patterns","01.csharp/09.ado-net/12.advanced-patterns",{"title":458,"icon":255,"path":459,"stem":460,"children":461,"page":59},"Ef Core","/csharp/ef-core","01.csharp/10.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 до об'єктів","/csharp/ef-core/what-is-orm","01.csharp/10.ef-core/01.what-is-orm",{"title":467,"path":468,"stem":469},"Перший проєкт — від нуля до CRUD","/csharp/ef-core/first-project","01.csharp/10.ef-core/02.first-project",{"title":471,"path":472,"stem":473},"DbContext — Серце EF Core","/csharp/ef-core/dbcontext-deep-dive","01.csharp/10.ef-core/03.dbcontext-deep-dive",{"title":475,"path":476,"stem":477},"Провайдери баз даних — Архітектура та Вибір СУБД","/csharp/ef-core/database-providers","01.csharp/10.ef-core/04.database-providers",{"title":479,"path":480,"stem":481},"Конвенції EF Core — Магія без конфігурації","/csharp/ef-core/conventions","01.csharp/10.ef-core/05.conventions",{"title":483,"path":484,"stem":485},"Fluent API та Data Annotations — Явна конфігурація моделі","/csharp/ef-core/fluent-api-vs-annotations","01.csharp/10.ef-core/06.fluent-api-vs-annotations",{"title":487,"path":488,"stem":489},"Зв'язки — One-to-One та One-to-Many","/csharp/ef-core/relationships-basics","01.csharp/10.ef-core/07.relationships-basics",{"title":491,"path":492,"stem":493},"Зв'язки Advanced — Many-to-Many та Складні Сценарії","/csharp/ef-core/relationships-advanced","01.csharp/10.ef-core/08.relationships-advanced",{"title":495,"path":496,"stem":497},"Властивості — Типи, Конвертери, Компаратори (Частина 1)","/csharp/ef-core/property-configuration-part1","01.csharp/10.ef-core/09.property-configuration-part1",{"title":499,"path":500,"stem":501},"Властивості — Value Comparers, Generators, Shadow Properties (Частина 2)","/csharp/ef-core/property-configuration-part2","01.csharp/10.ef-core/09.property-configuration-part2",{"title":503,"path":504,"stem":505},"Складні типи — Owned Types та Complex Types (Частина 1)","/csharp/ef-core/complex-types-owned-part1","01.csharp/10.ef-core/10.complex-types-owned-part1",{"title":507,"path":508,"stem":509},"Складні типи — Complex Types, Keyless Entities, Порівняння (Частина 2)","/csharp/ef-core/complex-types-owned-part2","01.csharp/10.ef-core/10.complex-types-owned-part2",{"title":511,"path":512,"stem":513},"JSON Columns — Складні дані у JSON (Частина 1)","/csharp/ef-core/json-columns-part1","01.csharp/10.ef-core/11.json-columns-part1",{"title":515,"path":516,"stem":517},"JSON Columns — Value Comparers, Індекси, Провайдери (Частина 2)","/csharp/ef-core/json-columns-part2","01.csharp/10.ef-core/11.json-columns-part2",{"title":519,"path":520,"stem":521},"Успадкування — Абстрактні класи та TPH (Частина 1)","/csharp/ef-core/inheritance-part1","01.csharp/10.ef-core/12.inheritance-part1",{"title":523,"path":524,"stem":525},"Успадкування — TPT, TPC та Порівняння Стратегій (Частина 2)","/csharp/ef-core/inheritance-part2","01.csharp/10.ef-core/12.inheritance-part2",{"title":527,"path":528,"stem":529,"children":530},"Індекси, Обмеження та Схема (Частина 1)","/csharp/ef-core/indexes-constraints-part1","01.csharp/10.ef-core/13.indexes-constraints-part1",[531],{"title":527,"path":528,"stem":529},{"title":533,"path":534,"stem":535,"children":536},"Індекси, Обмеження та Схема (Частина 2)","/csharp/ef-core/indexes-constraints-part2","01.csharp/10.ef-core/13.indexes-constraints-part2",[537],{"title":533,"path":534,"stem":535},{"title":539,"path":540,"stem":541},"Seed Data — Початкові Дані (Частина 1)","/csharp/ef-core/seeding-part1","01.csharp/10.ef-core/14.seeding-part1",{"title":543,"path":544,"stem":545},"Seed Data — SQL-скрипти, Bogus та Стратегії (Частина 2)","/csharp/ef-core/seeding-part2","01.csharp/10.ef-core/14.seeding-part2",{"title":547,"path":548,"stem":549},"Global Query Filters — Глобальні Фільтри (Частина 1)","/csharp/ef-core/global-query-filters-part1","01.csharp/10.ef-core/15.global-query-filters-part1",{"title":551,"path":552,"stem":553},"Global Query Filters — Підводні камені та Інтеграція (Частина 2)","/csharp/ef-core/global-query-filters-part2","01.csharp/10.ef-core/15.global-query-filters-part2",{"title":555,"path":556,"stem":557},"LINQ-запити в EF Core (Частина 1)","/csharp/ef-core/linq-queries-part1","01.csharp/10.ef-core/16.linq-queries-part1",{"title":559,"path":560,"stem":561},"LINQ-запити в EF Core (Частина 2)","/csharp/ef-core/linq-queries-part2","01.csharp/10.ef-core/16.linq-queries-part2",{"title":563,"path":564,"stem":565},"Завантаження Пов'язаних Даних (Частина 1)","/csharp/ef-core/loading-related-data-part1","01.csharp/10.ef-core/17.loading-related-data-part1",{"title":567,"path":568,"stem":569},"Завантаження Пов'язаних Даних (Частина 2)","/csharp/ef-core/loading-related-data-part2","01.csharp/10.ef-core/17.loading-related-data-part2",{"title":571,"path":572,"stem":573},"Raw SQL, Views та Stored Procedures (Частина 1)","/csharp/ef-core/raw-sql-part1","01.csharp/10.ef-core/18.raw-sql-part1",{"title":575,"path":576,"stem":577},"Raw SQL — Stored Procedures, DbFunction та Bulk Operations (Частина 2)","/csharp/ef-core/raw-sql-part2","01.csharp/10.ef-core/18.raw-sql-part2",{"title":579,"path":580,"stem":581},"Продвинуті Запити — Compiled Queries, Bulk та Оптимізація (Частина 1)","/csharp/ef-core/advanced-queries-part1","01.csharp/10.ef-core/19.advanced-queries-part1",{"title":583,"path":584,"stem":585},"Продвинуті Запити — Query Tags, Bulk та Interceptors (Частина 2)","/csharp/ef-core/advanced-queries-part2","01.csharp/10.ef-core/19.advanced-queries-part2",{"title":587,"path":588,"stem":589},"Change Tracker — Відстеження Змін (Частина 1)","/csharp/ef-core/change-tracking-part1","01.csharp/10.ef-core/20.change-tracking-part1",{"title":591,"path":592,"stem":593},"Change Tracker — Графи Об'єктів та Disconnected (Частина 2)","/csharp/ef-core/change-tracking-part2","01.csharp/10.ef-core/20.change-tracking-part2",{"title":595,"path":596,"stem":597},"Збереження Даних та Транзакції (Частина 1)","/csharp/ef-core/saving-data-part1","01.csharp/10.ef-core/21.saving-data-part1",{"title":599,"path":600,"stem":601},"Збереження Даних — Concurrency та Outbox (Частина 2)","/csharp/ef-core/saving-data-part2","01.csharp/10.ef-core/21.saving-data-part2",{"title":603,"path":604,"stem":605},"Конкурентність та Блокування (Частина 1)","/csharp/ef-core/concurrency-part1","01.csharp/10.ef-core/22.concurrency-part1",{"title":607,"path":608,"stem":609},"Конкурентність — Дедлоки та Queue Processing (Частина 2)","/csharp/ef-core/concurrency-part2","01.csharp/10.ef-core/22.concurrency-part2",{"title":611,"path":612,"stem":613},"Міграції в EF Core — Основи (Частина 1)","/csharp/ef-core/migrations-basics-part1","01.csharp/10.ef-core/23.migrations-basics-part1",{"title":615,"path":616,"stem":617},"Міграції в EF Core — Основи (Частина 2)","/csharp/ef-core/migrations-basics-part2","01.csharp/10.ef-core/23.migrations-basics-part2",{"title":619,"path":620,"stem":621},"Міграції — Просунуті Сценарії (Частина 1)","/csharp/ef-core/migrations-advanced-part1","01.csharp/10.ef-core/24.migrations-advanced-part1",{"title":623,"path":624,"stem":625},"Міграції — Просунуті Сценарії (Частина 2)","/csharp/ef-core/migrations-advanced-part2","01.csharp/10.ef-core/24.migrations-advanced-part2",{"title":627,"path":628,"stem":629},"Управління Схемою та Database-First (Частина 1)","/csharp/ef-core/schema-management-part1","01.csharp/10.ef-core/25.schema-management-part1",{"title":631,"path":632,"stem":633},"Управління Схемою та Database-First (Частина 2)","/csharp/ef-core/schema-management-part2","01.csharp/10.ef-core/25.schema-management-part2",{"title":635,"path":636,"stem":637},"Продуктивність EF Core — Основи (Частина 1)","/csharp/ef-core/performance-fundamentals-part1","01.csharp/10.ef-core/26.performance-fundamentals-part1",{"title":639,"path":640,"stem":641},"Interceptors в EF Core (Частина 1)","/csharp/ef-core/interceptors-part1","01.csharp/10.ef-core/29.interceptors-part1",{"title":643,"path":644,"stem":645},"Interceptors в EF Core — Connection, Transaction та Materialization (Частина 2)","/csharp/ef-core/interceptors-part2","01.csharp/10.ef-core/29.interceptors-part2",{"title":647,"path":648,"stem":649},"План вивчення Entity Framework Core — Повний курс","/csharp/ef-core/implementation_plan","01.csharp/10.ef-core/implementation_plan",{"title":651,"icon":652,"path":653,"stem":654,"children":655,"page":59},"ASP.NET","i-devicon-dotnetcore","/csharp/aspnet","01.csharp/11.aspnet",[656,730,791,869,927,941,967,1057,1111,1182,1212,1289],{"title":657,"icon":658,"path":659,"stem":660,"children":661,"page":59},"Minimal API","i-lucide-network","/csharp/aspnet/minimal-api","01.csharp/11.aspnet/01.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 та еволюція фреймворку","/csharp/aspnet/minimal-api/introduction","01.csharp/11.aspnet/01.minimal-api/01.introduction",{"title":667,"path":668,"stem":669},"Перший додаток на ASP.NET Core","/csharp/aspnet/minimal-api/first-application","01.csharp/11.aspnet/01.minimal-api/02.first-application",{"title":671,"path":672,"stem":673},"WebApplication, Builder та Dependency Injection","/csharp/aspnet/minimal-api/webapplication-builder","01.csharp/11.aspnet/01.minimal-api/03.webapplication-builder",{"title":675,"path":676,"stem":677},"Конвеєр запитів та Middleware","/csharp/aspnet/minimal-api/request-pipeline-middleware","01.csharp/11.aspnet/01.minimal-api/04.request-pipeline-middleware",{"title":679,"path":680,"stem":681},"Маршрутизація в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/routing-basics","01.csharp/11.aspnet/01.minimal-api/05.routing-basics",{"title":683,"path":684,"stem":685},"Маршрутизація в ASP.NET Core: Розширені можливості","/csharp/aspnet/minimal-api/routing-advanced","01.csharp/11.aspnet/01.minimal-api/06.routing-advanced",{"title":687,"path":688,"stem":689},"Статичні файли в ASP.NET Core","/csharp/aspnet/minimal-api/static-files","01.csharp/11.aspnet/01.minimal-api/07.static-files",{"title":691,"path":692,"stem":693},"Статичні Активи: MapStaticAssets (ASP.NET Core 9.0)","/csharp/aspnet/minimal-api/static-assets","01.csharp/11.aspnet/01.minimal-api/08.static-assets",{"title":695,"path":696,"stem":697},"Конфігурація в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/configuration-fundamentals","01.csharp/11.aspnet/01.minimal-api/09.configuration-fundamentals",{"title":699,"path":700,"stem":701},"Конфігурація: Паттерн Options","/csharp/aspnet/minimal-api/configuration-options","01.csharp/11.aspnet/01.minimal-api/10.configuration-options",{"title":703,"path":704,"stem":705},"Логування в ASP.NET Core: Основи","/csharp/aspnet/minimal-api/logging-basics","01.csharp/11.aspnet/01.minimal-api/11.logging-basics",{"title":707,"path":708,"stem":709},"Логування: Serilog та Middleware","/csharp/aspnet/minimal-api/logging-advanced","01.csharp/11.aspnet/01.minimal-api/12.logging-advanced",{"title":711,"path":712,"stem":713},"Управління станом: HttpContext.Items та Cookies","/csharp/aspnet/minimal-api/state-management","01.csharp/11.aspnet/01.minimal-api/13.state-management",{"title":715,"path":716,"stem":717},"Стан сесії: Sessions","/csharp/aspnet/minimal-api/session-state","01.csharp/11.aspnet/01.minimal-api/14.session-state",{"title":719,"path":720,"stem":721},"Структура проєкту: від хаосу до архітектури","/csharp/aspnet/minimal-api/project-structure","01.csharp/11.aspnet/01.minimal-api/15.project-structure",{"title":723,"path":724,"stem":725},"Scalar у Minimal API: повний проєкт і Fluent OpenAPI","/csharp/aspnet/minimal-api/scalar-openapi-fluent","01.csharp/11.aspnet/01.minimal-api/16.scalar-openapi-fluent",{"title":727,"path":728,"stem":729},"Swagger / Swashbuckle у Minimal API: окремий класичний шлях","/csharp/aspnet/minimal-api/swagger-swashbuckle","01.csharp/11.aspnet/01.minimal-api/17.swagger-swashbuckle",{"title":731,"icon":658,"path":732,"stem":733,"children":734,"page":59},"API","/csharp/aspnet/api","01.csharp/11.aspnet/02.api",[735,739,743,747,751,755,759,763,767,771,775,779,783,787],{"title":736,"path":737,"stem":738},"Що таке API. Клієнт-серверна архітектура","/csharp/aspnet/api/what-is-api","01.csharp/11.aspnet/02.api/01.what-is-api",{"title":740,"path":741,"stem":742},"Формати даних: JSON, XML, TOML та бінарні формати","/csharp/aspnet/api/data-formats","01.csharp/11.aspnet/02.api/02.data-formats",{"title":744,"path":745,"stem":746},"Парадигми API та концепція REST","/csharp/aspnet/api/api-paradigms-rest","01.csharp/11.aspnet/02.api/03.api-paradigms-rest",{"title":748,"path":749,"stem":750},"HTTP-методи, статус-коди та заголовки","/csharp/aspnet/api/http-methods-status-codes","01.csharp/11.aspnet/02.api/04.http-methods-status-codes",{"title":752,"path":753,"stem":754},"Організація HTTP API за принципами REST","/csharp/aspnet/api/rest-organizing","01.csharp/11.aspnet/02.api/05.rest-organizing",{"title":756,"path":757,"stem":758},"Номенклатура URL та CRUD-операції","/csharp/aspnet/api/url-nomenclature-crud","01.csharp/11.aspnet/02.api/06.url-nomenclature-crud",{"title":760,"path":761,"stem":762},"Правила дизайну: іменування та стандарти","/csharp/aspnet/api/api-design-naming","01.csharp/11.aspnet/02.api/07.api-design-naming",{"title":764,"path":765,"stem":766},"Валідація, ліміти та обробка помилок","/csharp/aspnet/api/api-design-validation","01.csharp/11.aspnet/02.api/08.api-design-validation",{"title":768,"path":769,"stem":770},"Обробка помилок у Minimal API","/csharp/aspnet/api/error-handling-http","01.csharp/11.aspnet/02.api/09.error-handling-http",{"title":772,"path":773,"stem":774},"Ідемпотентність та синхронізація стану","/csharp/aspnet/api/idempotency-sync","01.csharp/11.aspnet/02.api/10.idempotency-sync",{"title":776,"path":777,"stem":778},"Пагінація та організація списків","/csharp/aspnet/api/pagination-lists","01.csharp/11.aspnet/02.api/11.pagination-lists",{"title":780,"path":781,"stem":782},"Безпека API, кешування та інтернаціоналізація","/csharp/aspnet/api/security-auth","01.csharp/11.aspnet/02.api/12.security-auth",{"title":784,"path":785,"stem":786},"Процес проєктування API та документування","/csharp/aspnet/api/api-design-process","01.csharp/11.aspnet/02.api/13.api-design-process",{"title":788,"path":789,"stem":790},"OpenAPI: контракт, специфікація та документація API","/csharp/aspnet/api/openapi","01.csharp/11.aspnet/02.api/14.openapi",{"title":792,"icon":793,"path":794,"stem":795,"children":796,"page":59},"Auth","i-lucide-shield-check","/csharp/aspnet/auth","01.csharp/11.aspnet/03.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},"Основи аутентифікації та авторизації","/csharp/aspnet/auth/auth-fundamentals","01.csharp/11.aspnet/03.auth/01.auth-fundamentals",{"title":802,"path":803,"stem":804},"JWT-аутентифікація","/csharp/aspnet/auth/jwt-authentication","01.csharp/11.aspnet/03.auth/02.jwt-authentication",{"title":806,"path":807,"stem":808},"Авторизація: ролі, політики та resource-based доступ","/csharp/aspnet/auth/authorization-policies","01.csharp/11.aspnet/03.auth/03.authorization-policies",{"title":810,"path":811,"stem":812},"Cookie-аутентифікація та ASP.NET Core Identity","/csharp/aspnet/auth/cookie-auth-identity","01.csharp/11.aspnet/03.auth/04.cookie-auth-identity",{"title":814,"path":815,"stem":816},"JWT + Refresh Tokens (HttpOnly Cookie)","/csharp/aspnet/auth/04b.identity-auth-jwt","01.csharp/11.aspnet/03.auth/04b.identity-auth-jwt",{"title":818,"path":819,"stem":820},"Identity: Підтвердження Email та Скидання Пароля","/csharp/aspnet/auth/identity-email-confirmation","01.csharp/11.aspnet/03.auth/05.identity-email-confirmation",{"title":822,"path":823,"stem":824},"Identity: Двофакторна Аутентифікація (2FA)","/csharp/aspnet/auth/identity-two-factor","01.csharp/11.aspnet/03.auth/06.identity-two-factor",{"title":826,"path":827,"stem":828},"Identity: Внутрішня Архітектура та Кастомізація","/csharp/aspnet/auth/identity-internals","01.csharp/11.aspnet/03.auth/07.identity-internals",{"title":830,"path":831,"stem":832},"OAuth 2.0 та зовнішні провайдери","/csharp/aspnet/auth/oauth-external-providers","01.csharp/11.aspnet/03.auth/08.oauth-external-providers",{"title":834,"path":835,"stem":836},"Безпека на практиці: CORS, HTTPS та захист від атак","/csharp/aspnet/auth/security-hardening","01.csharp/11.aspnet/03.auth/09.security-hardening",{"title":838,"path":839,"stem":840},"Теорія OAuth 2.0: Поняття, Аналогії та Флоу","/csharp/aspnet/auth/oauth-theory","01.csharp/11.aspnet/03.auth/10.oauth-theory",{"title":842,"path":843,"stem":844},"OIDC, OAuth 2.0 та Keycloak в ASP.NET Core","/csharp/aspnet/auth/oidc-keycloak","01.csharp/11.aspnet/03.auth/10.oidc-keycloak",{"title":846,"path":847,"stem":848},"API Keys аутентифікація в ASP.NET Core","/csharp/aspnet/auth/api-keys","01.csharp/11.aspnet/03.auth/11.api-keys",{"title":850,"path":851,"stem":852},"Rate Limiting та Throttling в ASP.NET Core","/csharp/aspnet/auth/rate-limiting","01.csharp/11.aspnet/03.auth/12.rate-limiting",{"title":854,"path":855,"stem":856},"Refresh Token Rotation в ASP.NET Core","/csharp/aspnet/auth/refresh-token-rotation","01.csharp/11.aspnet/03.auth/13.refresh-token-rotation",{"title":858,"path":859,"stem":860},"Certificate Authentication та mTLS в ASP.NET Core","/csharp/aspnet/auth/certificate-auth","01.csharp/11.aspnet/03.auth/14.certificate-auth",{"title":862,"path":863,"stem":864},"RBAC, ABAC та ReBAC в ASP.NET Core","/csharp/aspnet/auth/rbac-abac-rebac","01.csharp/11.aspnet/03.auth/15.rbac-abac-rebac",{"title":866,"path":867,"stem":868},"Multi-tenancy та ізоляція даних в ASP.NET Core","/csharp/aspnet/auth/multi-tenancy","01.csharp/11.aspnet/03.auth/16.multi-tenancy",{"title":870,"icon":871,"path":872,"stem":873,"children":874,"page":59},"Нотифікації","i-lucide-bell","/csharp/aspnet/notifications","01.csharp/11.aspnet/04.notifications",[875,879,883,887,891,895,899,903,907,911,915,919,923],{"title":876,"path":877,"stem":878},"In-App нотифікації через базу даних","/csharp/aspnet/notifications/in-app-database-notifications","01.csharp/11.aspnet/04.notifications/01.in-app-database-notifications",{"title":880,"path":881,"stem":882},"Polling: Регулярний запит оновлень","/csharp/aspnet/notifications/polling","01.csharp/11.aspnet/04.notifications/02.polling",{"title":884,"path":885,"stem":886},"Server-Sent Events: Однострімовий push від сервера","/csharp/aspnet/notifications/server-sent-events","01.csharp/11.aspnet/04.notifications/03.server-sent-events",{"title":888,"path":889,"stem":890},"WebSockets: Двостороннє з'єднання в реальному часі","/csharp/aspnet/notifications/websockets","01.csharp/11.aspnet/04.notifications/04.websockets",{"title":892,"path":893,"stem":894},"SignalR: Абстракція над транспортами реального часу","/csharp/aspnet/notifications/signalr","01.csharp/11.aspnet/04.notifications/05.signalr",{"title":896,"path":897,"stem":898},"Background Services: Фонові задачі в ASP.NET Core","/csharp/aspnet/notifications/background-services","01.csharp/11.aspnet/04.notifications/06.background-services",{"title":900,"path":901,"stem":902},"Web Push нотифікації","/csharp/aspnet/notifications/web-push","01.csharp/11.aspnet/04.notifications/07.web-push",{"title":904,"path":905,"stem":906},"Email нотифікації","/csharp/aspnet/notifications/email-notifications","01.csharp/11.aspnet/04.notifications/08.email-notifications",{"title":908,"path":909,"stem":910},"Порівняння підходів: Як вибрати правильну технологію нотифікацій","/csharp/aspnet/notifications/choosing-the-right-approach","01.csharp/11.aspnet/04.notifications/09.choosing-the-right-approach",{"title":912,"path":913,"stem":914},"Hangfire: Надійне планування фонових задач","/csharp/aspnet/notifications/hangfire","01.csharp/11.aspnet/04.notifications/10.hangfire",{"title":916,"path":917,"stem":918},"Практика: Конвертація зображень у WebP через Hangfire","/csharp/aspnet/notifications/hangfire-image-webp","01.csharp/11.aspnet/04.notifications/11.hangfire-image-webp",{"title":920,"path":921,"stem":922},"Практика: Підготовка відео до HLS-стрімінгу через Hangfire","/csharp/aspnet/notifications/hangfire-video-hls","01.csharp/11.aspnet/04.notifications/12.hangfire-video-hls",{"title":924,"path":925,"stem":926},"Telegram-нотифікації: від одного повідомлення до масових розсилок і мульти-канального підходу","/csharp/aspnet/notifications/telegram-notifications","01.csharp/11.aspnet/04.notifications/13.telegram-notifications",{"title":928,"icon":929,"path":930,"stem":931,"children":932,"page":59},"Інтернаціоналізація","i-lucide-languages","/csharp/aspnet/i18n","01.csharp/11.aspnet/05.i18n",[933,937],{"title":934,"path":935,"stem":936},"Інтернаціоналізація (i18n) у Minimal API: від A до Я","/csharp/aspnet/i18n/internationalization","01.csharp/11.aspnet/05.i18n/01.internationalization",{"title":938,"path":939,"stem":940},"Humanizer: людиномовні рядки у .NET","/csharp/aspnet/i18n/humanizer","01.csharp/11.aspnet/05.i18n/02.humanizer",{"title":942,"icon":943,"path":944,"stem":945,"children":946,"page":59},"Кешування","i-lucide-layers","/csharp/aspnet/caching","01.csharp/11.aspnet/06.caching",[947,951,955,959,963],{"title":948,"path":949,"stem":950},"Огляд кешування: чотири рівні і коли що обирати","/csharp/aspnet/caching/caching","01.csharp/11.aspnet/06.caching/01.caching",{"title":952,"path":953,"stem":954},"IMemoryCache: кеш в оперативній пам'яті","/csharp/aspnet/caching/memory-cache","01.csharp/11.aspnet/06.caching/02.memory-cache",{"title":956,"path":957,"stem":958},"IDistributedCache і Redis: розподілений кеш","/csharp/aspnet/caching/distributed-cache","01.csharp/11.aspnet/06.caching/03.distributed-cache",{"title":960,"path":961,"stem":962},"Response Cache: HTTP-кешування через Cache-Control","/csharp/aspnet/caching/response-cache","01.csharp/11.aspnet/06.caching/04.response-cache",{"title":964,"path":965,"stem":966},"Output Cache: серверний кеш HTTP-відповідей (.NET 7+)","/csharp/aspnet/caching/output-cache","01.csharp/11.aspnet/06.caching/05.output-cache",{"title":968,"icon":969,"path":970,"stem":971,"children":972,"page":59},"Тестування","i-lucide-test-tube","/csharp/aspnet/testing","01.csharp/11.aspnet/07.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},"Що таке тестування? Від інтуїції до науки","/csharp/aspnet/testing/what-is-testing","01.csharp/11.aspnet/07.testing/01.what-is-testing",{"title":978,"path":979,"stem":980},"Піраміда тестування — Стратегія, а не Догма","/csharp/aspnet/testing/testing-pyramid","01.csharp/11.aspnet/07.testing/02.testing-pyramid",{"title":982,"path":983,"stem":984},"Дві Школи Тестування — Лондон проти Детройту","/csharp/aspnet/testing/testing-schools","01.csharp/11.aspnet/07.testing/03.testing-schools",{"title":986,"path":987,"stem":988},"TDD та BDD — Тести як Дизайн-інструмент","/csharp/aspnet/testing/tdd-and-bdd","01.csharp/11.aspnet/07.testing/04.tdd-and-bdd",{"title":990,"path":991,"stem":992},"Що саме тестувати — Техніки аналізу та Циклomatична складність","/csharp/aspnet/testing/what-to-test","01.csharp/11.aspnet/07.testing/05.what-to-test",{"title":994,"path":995,"stem":996},"Тестові Фреймворки — Навіщо вони і що всередині","/csharp/aspnet/testing/test-frameworks","01.csharp/11.aspnet/07.testing/06.test-frameworks",{"title":998,"path":999,"stem":1000},"xUnit — Факти, Теорії та Lifecycle тестів","/csharp/aspnet/testing/xunit-basics","01.csharp/11.aspnet/07.testing/07.xunit-basics",{"title":1002,"path":1003,"stem":1004},"xUnit Advanced — Fixtures, Кастомізація та Розширення","/csharp/aspnet/testing/xunit-advanced","01.csharp/11.aspnet/07.testing/08.xunit-advanced",{"title":1006,"path":1007,"stem":1008},"Moq — Глибоке занурення в мокування","/csharp/aspnet/testing/mocking-with-moq","01.csharp/11.aspnet/07.testing/09.mocking-with-moq",{"title":1010,"path":1011,"stem":1012},"Тестування Баз Даних — EF Core, SQLite та Testcontainers","/csharp/aspnet/testing/database-testing","01.csharp/11.aspnet/07.testing/10.database-testing",{"title":1014,"path":1015,"stem":1016},"Integration Testing — Частина 1 [Теорія та WebApplicationFactory]","/csharp/aspnet/testing/integration-testing","01.csharp/11.aspnet/07.testing/11.integration-testing",{"title":1018,"path":1019,"stem":1020},"Інтеграційне тестування — Практика","/csharp/aspnet/testing/11a.integration-testing-practice","01.csharp/11.aspnet/07.testing/11a.integration-testing-practice",{"title":1022,"path":1023,"stem":1024},"Integration Testing — Частина 2 [Просунуті Сценарії та Testcontainers]","/csharp/aspnet/testing/integration-testing-advanced","01.csharp/11.aspnet/07.testing/12.integration-testing-advanced",{"title":1026,"path":1027,"stem":1028},"Професійний Postman: Колекції, Змінні та GitHub Інтеграція","/csharp/aspnet/testing/postman-professional","01.csharp/11.aspnet/07.testing/13.postman-professional",{"title":1030,"path":1031,"stem":1032},"HttpClient у Тестах Частина 1: Архітектура та MockHttpMessageHandler","/csharp/aspnet/testing/httpclient-testing","01.csharp/11.aspnet/07.testing/14.httpclient-testing",{"title":1034,"path":1035,"stem":1036},"HttpClient у Тестах Частина 2: WireMock.Net та Resilience","/csharp/aspnet/testing/wiremock-net","01.csharp/11.aspnet/07.testing/15.wiremock-net",{"title":1038,"path":1039,"stem":1040},"Патерни та Анти-патерни Тестування: Test Smells","/csharp/aspnet/testing/testing-patterns","01.csharp/11.aspnet/07.testing/16.testing-patterns",{"title":1042,"path":1043,"stem":1044},"Просунуті інструменти: Time, Snapshots та Властивості","/csharp/aspnet/testing/advanced-testing-tools","01.csharp/11.aspnet/07.testing/17.advanced-testing-tools",{"title":1046,"path":1047,"stem":1048},"Тестування Архітектури з NetArchTest","/csharp/aspnet/testing/architecture-testing","01.csharp/11.aspnet/07.testing/18.architecture-testing",{"title":1050,"path":1051,"stem":1052},"Тестування Продуктивності: BenchmarkDotNet, NBomber та k6","/csharp/aspnet/testing/performance-testing","01.csharp/11.aspnet/07.testing/19.performance-testing",{"title":1054,"path":1055,"stem":1056},"Залишок плану для курсу \"Тестування ASP.NET Minimal API\"","/csharp/aspnet/testing/remaining_plan","01.csharp/11.aspnet/07.testing/remaining_plan",{"title":1058,"icon":1059,"path":1060,"stem":1061,"children":1062,"page":59},"Платежі","i-lucide-credit-card","/csharp/aspnet/payments","01.csharp/11.aspnet/08.payments",[1063,1067,1071,1075,1079,1083,1087,1091,1095,1099,1103,1107],{"title":1064,"path":1065,"stem":1066},"Основи платіжної інфраструктури","/csharp/aspnet/payments/payment-fundamentals","01.csharp/11.aspnet/08.payments/01.payment-fundamentals",{"title":1068,"path":1069,"stem":1070},"Методи оплати в Україні","/csharp/aspnet/payments/payment-methods-ukraine","01.csharp/11.aspnet/08.payments/02.payment-methods-ukraine",{"title":1072,"path":1073,"stem":1074},"PCI DSS та безпека платежів","/csharp/aspnet/payments/pci-dss-security","01.csharp/11.aspnet/08.payments/03.pci-dss-security",{"title":1076,"path":1077,"stem":1078},"Архітектура платіжної підсистеми","/csharp/aspnet/payments/payment-architecture","01.csharp/11.aspnet/08.payments/04.payment-architecture",{"title":1080,"path":1081,"stem":1082},"Інтеграція LiqPay (ПриватБанк)","/csharp/aspnet/payments/liqpay-integration","01.csharp/11.aspnet/08.payments/05.liqpay-integration",{"title":1084,"path":1085,"stem":1086},"Інтеграція Monobank Acquiring API","/csharp/aspnet/payments/monobank-acquiring","01.csharp/11.aspnet/08.payments/06.monobank-acquiring",{"title":1088,"path":1089,"stem":1090},"Інтеграція Stripe","/csharp/aspnet/payments/stripe-integration","01.csharp/11.aspnet/08.payments/07.stripe-integration",{"title":1092,"path":1093,"stem":1094},"Webhooks — глибоке занурення","/csharp/aspnet/payments/webhooks-deep-dive","01.csharp/11.aspnet/08.payments/08.webhooks-deep-dive",{"title":1096,"path":1097,"stem":1098},"Підписки та рекурентні платежі","/csharp/aspnet/payments/subscriptions-recurring","01.csharp/11.aspnet/08.payments/09.subscriptions-recurring",{"title":1100,"path":1101,"stem":1102},"Повернення коштів та диспути","/csharp/aspnet/payments/refunds-disputes","01.csharp/11.aspnet/08.payments/10.refunds-disputes",{"title":1104,"path":1105,"stem":1106},"Тестування платіжних інтеграцій","/csharp/aspnet/payments/testing-payments","01.csharp/11.aspnet/08.payments/11.testing-payments",{"title":1108,"path":1109,"stem":1110},"Чекліст виходу в Production","/csharp/aspnet/payments/production-checklist","01.csharp/11.aspnet/08.payments/12.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","/csharp/aspnet/libraries","01.csharp/11.aspnet/09.libraries",[1130,1134,1138,1142,1146,1150,1154,1158,1162,1166,1170,1174,1178],{"title":1131,"path":1132,"stem":1133},"Валідація з FluentValidation в ASP.NET Core","/csharp/aspnet/libraries/fluent-validation","01.csharp/11.aspnet/09.libraries/01.fluent-validation",{"title":1135,"path":1136,"stem":1137},"Маппінг об","/csharp/aspnet/libraries/mapster","01.csharp/11.aspnet/09.libraries/02.mapster",{"title":1139,"path":1140,"stem":1141},"Обробка помилок з ErrorOr та Result Pattern в ASP.NET Core","/csharp/aspnet/libraries/erroror-result-pattern","01.csharp/11.aspnet/09.libraries/03.erroror-result-pattern",{"title":1143,"path":1144,"stem":1145},"Структуроване логування з Serilog в ASP.NET Core","/csharp/aspnet/libraries/serilog","01.csharp/11.aspnet/09.libraries/04.serilog",{"title":1147,"path":1148,"stem":1149},"CQRS та Mediator з MediatR в ASP.NET Core","/csharp/aspnet/libraries/mediatr","01.csharp/11.aspnet/09.libraries/05.mediatr",{"title":1151,"path":1152,"stem":1153},"Відмовостійкість з Polly в ASP.NET Core","/csharp/aspnet/libraries/polly","01.csharp/11.aspnet/09.libraries/06.polly",{"title":1155,"path":1156,"stem":1157},"Health Checks в ASP.NET Core","/csharp/aspnet/libraries/health-checks","01.csharp/11.aspnet/09.libraries/07.health-checks",{"title":1159,"path":1160,"stem":1161},"Feature Management та Feature Flags в ASP.NET Core","/csharp/aspnet/libraries/feature-management","01.csharp/11.aspnet/09.libraries/08.feature-management",{"title":1163,"path":1164,"stem":1165},"Відправка Email з FluentEmail в ASP.NET Core","/csharp/aspnet/libraries/fluent-email","01.csharp/11.aspnet/09.libraries/09.fluent-email",{"title":1167,"path":1168,"stem":1169},"Генерація PDF з QuestPDF в ASP.NET Core","/csharp/aspnet/libraries/quest-pdf","01.csharp/11.aspnet/09.libraries/10.quest-pdf",{"title":1171,"path":1172,"stem":1173},"Генерація тестових даних з Bogus в ASP.NET Core","/csharp/aspnet/libraries/bogus","01.csharp/11.aspnet/09.libraries/11.bogus",{"title":1175,"path":1176,"stem":1177},"Humanizer та Guard Clauses в ASP.NET Core","/csharp/aspnet/libraries/humanizer-guard","01.csharp/11.aspnet/09.libraries/12.humanizer-guard",{"title":1179,"path":1180,"stem":1181},"План модуля 10.libraries — Популярні бібліотеки ASP.NET","/csharp/aspnet/libraries/plan","01.csharp/11.aspnet/09.libraries/plan",{"title":1183,"icon":1184,"path":1185,"stem":1186,"children":1187,"page":59},"Razor Pages","i-lucide-layout-template","/csharp/aspnet/razor-pages","01.csharp/11.aspnet/10.razor-pages",[1188,1192,1196,1200,1204,1208],{"title":1189,"path":1190,"stem":1191},"Від Minimal API до Razor Pages: концептуальний перехід","/csharp/aspnet/razor-pages/from-minimal-api","01.csharp/11.aspnet/10.razor-pages/01.from-minimal-api",{"title":1193,"path":1194,"stem":1195},"PageModel: логіка сторінки Razor Pages","/csharp/aspnet/razor-pages/page-model","01.csharp/11.aspnet/10.razor-pages/02.page-model",{"title":1197,"path":1198,"stem":1199},"Razor синтаксис: шаблонізатор у .cshtml","/csharp/aspnet/razor-pages/razor-syntax","01.csharp/11.aspnet/10.razor-pages/03.razor-syntax",{"title":1201,"path":1202,"stem":1203},"Tag Helpers: типізований HTML","/csharp/aspnet/razor-pages/tag-helpers","01.csharp/11.aspnet/10.razor-pages/04.tag-helpers",{"title":1205,"path":1206,"stem":1207},"Форми і валідація: повний цикл обробки даних","/csharp/aspnet/razor-pages/forms-validation","01.csharp/11.aspnet/10.razor-pages/05.forms-validation",{"title":1209,"path":1210,"stem":1211},"Практичний проєкт: TaskManager на Razor Pages","/csharp/aspnet/razor-pages/project-task-manager","01.csharp/11.aspnet/10.razor-pages/06.project-task-manager",{"title":1213,"path":1214,"stem":1215,"children":1216,"page":59},"ASP.NET Core MVC","/csharp/aspnet/mvc","01.csharp/11.aspnet/11.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: архітектура, що змінила веб","/csharp/aspnet/mvc/mvc-pattern","01.csharp/11.aspnet/11.mvc/01.mvc-pattern",{"title":1222,"path":1223,"stem":1224},"Від Razor Pages до MVC: концептуальний перехід","/csharp/aspnet/mvc/from-razor-pages","01.csharp/11.aspnet/11.mvc/02.from-razor-pages",{"title":1226,"path":1227,"stem":1228},"Controllers та Actions: серце MVC","/csharp/aspnet/mvc/controllers-actions","01.csharp/11.aspnet/11.mvc/03.controllers-actions",{"title":1230,"path":1231,"stem":1232},"Маршрутизація в MVC: Convention vs Attribute Routing","/csharp/aspnet/mvc/routing-mvc","01.csharp/11.aspnet/11.mvc/04.routing-mvc",{"title":1234,"path":1235,"stem":1236},"Model Binding: від HTTP до C#","/csharp/aspnet/mvc/model-binding","01.csharp/11.aspnet/11.mvc/05.model-binding",{"title":1238,"path":1239,"stem":1240},"Views, ViewData, ViewBag, TempData і ViewModel","/csharp/aspnet/mvc/views-viewdata-tempdata","01.csharp/11.aspnet/11.mvc/06.views-viewdata-tempdata",{"title":1242,"path":1243,"stem":1244},"Filters: аспектно-орієнтоване програмування в MVC","/csharp/aspnet/mvc/filters","01.csharp/11.aspnet/11.mvc/07.filters",{"title":1246,"path":1247,"stem":1248},"Areas: структурування великих застосунків","/csharp/aspnet/mvc/areas","01.csharp/11.aspnet/11.mvc/08.areas",{"title":1250,"path":1251,"stem":1252},"View Components: повторювані незалежні блоки UI","/csharp/aspnet/mvc/view-components","01.csharp/11.aspnet/11.mvc/09.view-components",{"title":1254,"path":1255,"stem":1256},"Display та Editor Templates","/csharp/aspnet/mvc/display-editor-templates","01.csharp/11.aspnet/11.mvc/10.display-editor-templates",{"title":1258,"path":1259,"stem":1260},"Валідація: IValidatableObject та FluentValidation","/csharp/aspnet/mvc/validation-advanced","01.csharp/11.aspnet/11.mvc/11.validation-advanced",{"title":1262,"path":1263,"stem":1264},"HTMX: інтерактивність через HTML-атрибути","/csharp/aspnet/mvc/htmx","01.csharp/11.aspnet/11.mvc/12.htmx",{"title":1266,"path":1267,"stem":1268},"HTMX у ASP.NET Core MVC: серверна інтеграція","/csharp/aspnet/mvc/ajax-htmx-mvc","01.csharp/11.aspnet/11.mvc/13.ajax-htmx-mvc",{"title":1270,"path":1271,"stem":1272},"Практичний проєкт: Каталог товарів з HTMX","/csharp/aspnet/mvc/htmx-project","01.csharp/11.aspnet/11.mvc/14.htmx-project",{"title":1274,"path":1275,"stem":1276},"Завантаження та обробка файлів","/csharp/aspnet/mvc/file-upload","01.csharp/11.aspnet/11.mvc/15.file-upload",{"title":1278,"path":1279,"stem":1280},"Глобалізація та Локалізація MVC","/csharp/aspnet/mvc/globalization-localization","01.csharp/11.aspnet/11.mvc/16.globalization-localization",{"title":1282,"path":1283,"stem":1284},"Підсумковий проєкт: Блог-платформа","/csharp/aspnet/mvc/mvc-project","01.csharp/11.aspnet/11.mvc/17.mvc-project",{"title":1286,"path":1287,"stem":1288},"План курсу: ASP.NET Core MVC","/csharp/aspnet/mvc/plan","01.csharp/11.aspnet/11.mvc/plan",{"title":1290,"path":1291,"stem":1292,"children":1293,"page":59},"Web Api","/csharp/aspnet/web-api","01.csharp/11.aspnet/12.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","/csharp/aspnet/web-api/from-minimal-api-to-controllers","01.csharp/11.aspnet/12.web-api/01.from-minimal-api-to-controllers",{"title":1299,"path":1300,"stem":1301},"ControllerBase, ActionResult\u003CT> та Response Types","/csharp/aspnet/web-api/controller-base-actionresult","01.csharp/11.aspnet/12.web-api/02.controller-base-actionresult",{"title":1303,"path":1304,"stem":1305},"Content Negotiation - JSON, XML та власні форматери","/csharp/aspnet/web-api/content-negotiation","01.csharp/11.aspnet/12.web-api/03.content-negotiation",{"title":1307,"path":1308,"stem":1309},"Версіонування API","/csharp/aspnet/web-api/api-versioning","01.csharp/11.aspnet/12.web-api/04.api-versioning",{"title":1311,"path":1312,"stem":1313},"ProblemDetails та структурована обробка помилок","/csharp/aspnet/web-api/problemdetails-error-handling","01.csharp/11.aspnet/12.web-api/05.problemdetails-error-handling",{"title":1315,"path":1316,"stem":1317},"Фільтри у Web API контексті","/csharp/aspnet/web-api/filters-for-api","01.csharp/11.aspnet/12.web-api/06.filters-for-api",{"title":1319,"path":1320,"stem":1321},"Пагінація, фільтрація та сортування","/csharp/aspnet/web-api/pagination-filtering-sorting","01.csharp/11.aspnet/12.web-api/07.pagination-filtering-sorting",{"title":1323,"path":1324,"stem":1325},"HATEOAS та Resource Expansion","/csharp/aspnet/web-api/hateoas-resource-expansion","01.csharp/11.aspnet/12.web-api/08.hateoas-resource-expansion",{"title":1327,"path":1328,"stem":1329},"Гібридна архітектура - Minimal API + Controllers","/csharp/aspnet/web-api/minimal-api-vs-controllers-hybrid","01.csharp/11.aspnet/12.web-api/09.minimal-api-vs-controllers-hybrid",{"title":1331,"path":1332,"stem":1333},"Документація API - Swashbuckle, NSwag та генерація клієнтів","/csharp/aspnet/web-api/api-documentation-generation","01.csharp/11.aspnet/12.web-api/10.api-documentation-generation",{"title":1335,"path":1336,"stem":1337},"Health Checks та моніторинг API","/csharp/aspnet/web-api/health-checks-monitoring","01.csharp/11.aspnet/12.web-api/11.health-checks-monitoring",{"title":1339,"path":1340,"stem":1341},"Підсумковий проєкт - Production-Ready REST API","/csharp/aspnet/web-api/web-api-project","01.csharp/11.aspnet/12.web-api/12.web-api-project",{"title":1343,"path":1344,"stem":1345},"План курсу: ASP.NET Core Web API (Controllers)","/csharp/aspnet/web-api/plan","01.csharp/11.aspnet/12.web-api/plan",{"title":1347,"icon":1348,"path":1349,"stem":1350,"children":1351,"page":59},"Desktop UI","i-lucide-app-window","/csharp/desktop-ui","01.csharp/12.desktop-ui",[1352,1356,1360,1364,1368,1372,1376,1380,1384,1388,1392,1396,1400,1404,1408,1412,1416,1420,1424,1428,1432,1436,1440,1444,1448,1452,1456,1460,1464,1468,1472,1476,1480,1484,1488,1492,1496,1500,1504,1508,1512,1516,1520,1524,1528,1532,1536,1540,1544,1548,1552,1556,1560,1564,1568,1572,1576,1580,1584,1588,1592,1596,1600,1604,1608,1612,1616,1620,1624,1628,1632],{"title":1353,"path":1354,"stem":1355},"Що таке десктопна розробка?","/csharp/desktop-ui/what-is-desktop-dev","01.csharp/12.desktop-ui/01.what-is-desktop-dev",{"title":1357,"path":1358,"stem":1359},"Архітектура WPF — як влаштований графічний інтерфейс","/csharp/desktop-ui/wpf-architecture","01.csharp/12.desktop-ui/02.wpf-architecture",{"title":1361,"path":1362,"stem":1363},"Перший WPF-проєкт — від нуля до вікна","/csharp/desktop-ui/first-wpf-app","01.csharp/12.desktop-ui/03.first-wpf-app",{"title":1365,"path":1366,"stem":1367},"Перший Avalonia-проєкт: WPF для всіх платформ","/csharp/desktop-ui/03a.first-avalonia-app","01.csharp/12.desktop-ui/03a.first-avalonia-app",{"title":1369,"path":1370,"stem":1371},"XAML: декларативний інтерфейс","/csharp/desktop-ui/xaml-basics","01.csharp/12.desktop-ui/04.xaml-basics",{"title":1373,"path":1374,"stem":1375},"Fluent UI у WPF — сучасний дизайн Windows 11","/csharp/desktop-ui/04a.wpf-fluent-ui","01.csharp/12.desktop-ui/04a.wpf-fluent-ui",{"title":1377,"path":1378,"stem":1379},"WPF UI — сучасна бібліотека Fluent контролів","/csharp/desktop-ui/04b.wpf-ui-library","01.csharp/12.desktop-ui/04b.wpf-ui-library",{"title":1381,"path":1382,"stem":1383},"HandyControl — велика бібліотека UI контролів для WPF","/csharp/desktop-ui/04c.handycontrol-library","01.csharp/12.desktop-ui/04c.handycontrol-library",{"title":1385,"path":1386,"stem":1387},"Простори імен та ресурси XAML","/csharp/desktop-ui/xaml-namespaces-resources","01.csharp/12.desktop-ui/05.xaml-namespaces-resources",{"title":1389,"path":1390,"stem":1391},"XAML в Avalonia: ключові відмінності від WPF","/csharp/desktop-ui/05a.avalonia-xaml-differences","01.csharp/12.desktop-ui/05a.avalonia-xaml-differences",{"title":1393,"path":1394,"stem":1395},"Розширення розмітки XAML (Markup Extensions)","/csharp/desktop-ui/xaml-markup-extensions","01.csharp/12.desktop-ui/06.xaml-markup-extensions",{"title":1397,"path":1398,"stem":1399},"Панелі Layout: StackPanel, WrapPanel, DockPanel","/csharp/desktop-ui/layout-panels-part1","01.csharp/12.desktop-ui/07.layout-panels-part1",{"title":1401,"path":1402,"stem":1403},"Grid, Canvas, UniformGrid","/csharp/desktop-ui/layout-panels-part2","01.csharp/12.desktop-ui/07.layout-panels-part2",{"title":1405,"path":1406,"stem":1407},"Просунуті техніки Layout","/csharp/desktop-ui/layout-advanced","01.csharp/12.desktop-ui/08.layout-advanced",{"title":1409,"path":1410,"stem":1411},"Адаптивний Layout та найкращі практики","/csharp/desktop-ui/layout-responsive","01.csharp/12.desktop-ui/09.layout-responsive",{"title":1413,"path":1414,"stem":1415},"Layout в Avalonia: відмінності та нові можливості","/csharp/desktop-ui/09a.layout-avalonia","01.csharp/12.desktop-ui/09a.layout-avalonia",{"title":1417,"path":1418,"stem":1419},"Button, Image, ProgressBar та інші базові контроли","/csharp/desktop-ui/basic-controls","01.csharp/12.desktop-ui/10.basic-controls",{"title":1421,"path":1422,"stem":1423},"Контроли в Avalonia: відмінності від WPF","/csharp/desktop-ui/10a.controls-avalonia","01.csharp/12.desktop-ui/10a.controls-avalonia",{"title":1425,"path":1426,"stem":1427},"Текстові контроли — TextBlock, TextBox, RichTextBox","/csharp/desktop-ui/text-controls","01.csharp/12.desktop-ui/11.text-controls",{"title":1429,"path":1430,"stem":1431},"Контроли вибору — CheckBox, RadioButton, ComboBox, ListBox, DatePicker","/csharp/desktop-ui/selection-controls","01.csharp/12.desktop-ui/12.selection-controls",{"title":1433,"path":1434,"stem":1435},"Content Model — GroupBox, Expander, TabControl, StatusBar","/csharp/desktop-ui/content-controls","01.csharp/12.desktop-ui/13.content-controls",{"title":1437,"path":1438,"stem":1439},"UI/UX принципи десктопних застосунків","/csharp/desktop-ui/13a.ui-ux-principles","01.csharp/12.desktop-ui/13a.ui-ux-principles",{"title":1441,"path":1442,"stem":1443},"Dependency Properties — Концепція та Value Resolution","/csharp/desktop-ui/dependency-properties-part1","01.csharp/12.desktop-ui/14.dependency-properties-part1",{"title":1445,"path":1446,"stem":1447},"Avalonia Property System — StyledProperty та DirectProperty","/csharp/desktop-ui/14a.avalonia-property-system","01.csharp/12.desktop-ui/14a.avalonia-property-system",{"title":1449,"path":1450,"stem":1451},"Attached Properties — Властивості без меж","/csharp/desktop-ui/attached-properties","01.csharp/12.desktop-ui/15.attached-properties",{"title":1453,"path":1454,"stem":1455},"Routed Events — Маршрутизація подій у WPF","/csharp/desktop-ui/routed-events","01.csharp/12.desktop-ui/16.routed-events",{"title":1457,"path":1458,"stem":1459},"Data Binding — Від Code-Behind до Декларативності","/csharp/desktop-ui/data-binding-basics-part1","01.csharp/12.desktop-ui/17.data-binding-basics-part1",{"title":1461,"path":1462,"stem":1463},"INotifyPropertyChanged — Живе оновлення UI","/csharp/desktop-ui/data-binding-basics-part2","01.csharp/12.desktop-ui/17.data-binding-basics-part2",{"title":1465,"path":1466,"stem":1467},"Compiled Bindings в Avalonia — Безпека на етапі компіляції","/csharp/desktop-ui/17a.avalonia-compiled-bindings","01.csharp/12.desktop-ui/17a.avalonia-compiled-bindings",{"title":1469,"path":1470,"stem":1471},"Просунутий Data Binding — ElementName, RelativeSource, MultiBinding","/csharp/desktop-ui/data-binding-advanced","01.csharp/12.desktop-ui/18.data-binding-advanced",{"title":1473,"path":1474,"stem":1475},"Value Converters — Перетворення типів даних у Data Binding","/csharp/desktop-ui/value-converters","01.csharp/12.desktop-ui/19.value-converters",{"title":1477,"path":1478,"stem":1479},"Data Templates — Візуалізація об'єктів у WPF","/csharp/desktop-ui/data-templates","01.csharp/12.desktop-ui/20.data-templates",{"title":1481,"path":1482,"stem":1483},"Collections Binding Part 1 — ObservableCollection та ItemsControl","/csharp/desktop-ui/collections-binding-part1","01.csharp/12.desktop-ui/21.collections-binding-part1",{"title":1485,"path":1486,"stem":1487},"Collections Binding Part 2 — ICollectionView, Filtering, Sorting та Virtualization","/csharp/desktop-ui/collections-binding-part2","01.csharp/12.desktop-ui/21.collections-binding-part2",{"title":1489,"path":1490,"stem":1491},"MVVM Pattern — Від Spaghetti Code до архітектури","/csharp/desktop-ui/mvvm-pattern","01.csharp/12.desktop-ui/22.mvvm-pattern",{"title":1493,"path":1494,"stem":1495},"ViewModel Implementation — Від BaseViewModel до валідації","/csharp/desktop-ui/viewmodel-implementation","01.csharp/12.desktop-ui/23.viewmodel-implementation",{"title":1497,"path":1498,"stem":1499},"Commands — Від event handlers до декларативних команд","/csharp/desktop-ui/commands","01.csharp/12.desktop-ui/24.commands",{"title":1501,"path":1502,"stem":1503},"MVVM Toolkit — MVVM без boilerplate через Source Generators","/csharp/desktop-ui/mvvm-toolkit","01.csharp/12.desktop-ui/25.mvvm-toolkit",{"title":1505,"path":1506,"stem":1507},"Messenger Pattern — Комунікація між ViewModel без прямих посилань","/csharp/desktop-ui/messenger-pattern","01.csharp/12.desktop-ui/26.messenger-pattern",{"title":1509,"path":1510,"stem":1511},"Стилі WPF — CSS для десктопу","/csharp/desktop-ui/styles-basics","01.csharp/12.desktop-ui/27.styles-basics",{"title":1513,"path":1514,"stem":1515},"CSS-like стилі Avalonia","/csharp/desktop-ui/27a.avalonia-css-styling","01.csharp/12.desktop-ui/27a.avalonia-css-styling",{"title":1517,"path":1518,"stem":1519},"Control Templates — Частина 1. Концепція та TemplateBinding","/csharp/desktop-ui/control-templates-part1","01.csharp/12.desktop-ui/28.control-templates-part1",{"title":1521,"path":1522,"stem":1523},"Control Templates — Частина 2. Named Parts та ContentPresenter","/csharp/desktop-ui/control-templates-part2","01.csharp/12.desktop-ui/28.control-templates-part2",{"title":1525,"path":1526,"stem":1527},"Control Themes в Avalonia — нова ера стилізації","/csharp/desktop-ui/28a.avalonia-control-themes","01.csharp/12.desktop-ui/28a.avalonia-control-themes",{"title":1529,"path":1530,"stem":1531},"Triggers та Visual State Manager у WPF","/csharp/desktop-ui/triggers-visual-states","01.csharp/12.desktop-ui/29.triggers-visual-states",{"title":1533,"path":1534,"stem":1535},"Pseudo-classes в Avalonia — замість WPF Triggers","/csharp/desktop-ui/29a.avalonia-pseudo-classes","01.csharp/12.desktop-ui/29a.avalonia-pseudo-classes",{"title":1537,"path":1538,"stem":1539},"Теми та ресурсні словники у WPF","/csharp/desktop-ui/resources-themes","01.csharp/12.desktop-ui/30.resources-themes",{"title":1541,"path":1542,"stem":1543},"Avalonia Themes — Fluent Design та система тематизації","/csharp/desktop-ui/30a.avalonia-themes-fluent","01.csharp/12.desktop-ui/30a.avalonia-themes-fluent",{"title":1545,"path":1546,"stem":1547},"Контроли колекцій — глибоке занурення","/csharp/desktop-ui/collection-controls","01.csharp/12.desktop-ui/31.collection-controls",{"title":1549,"path":1550,"stem":1551},"DataGrid — колонки та базове відображення","/csharp/desktop-ui/datagrid-part1","01.csharp/12.desktop-ui/32.datagrid-part1",{"title":1553,"path":1554,"stem":1555},"DataGrid — сортування, фільтрація, редагування","/csharp/desktop-ui/datagrid-part2","01.csharp/12.desktop-ui/32.datagrid-part2",{"title":1557,"path":1558,"stem":1559},"TreeView та GridView","/csharp/desktop-ui/treeview-listview","01.csharp/12.desktop-ui/33.treeview-listview",{"title":1561,"path":1562,"stem":1563},"Меню, Toolbar, ContextMenu, StatusBar","/csharp/desktop-ui/menus-toolbars","01.csharp/12.desktop-ui/34.menus-toolbars",{"title":1565,"path":1566,"stem":1567},"Навігація та керування вікнами. Частина 1: вікна та сторінки","/csharp/desktop-ui/navigation-windows-part1","01.csharp/12.desktop-ui/35.navigation-windows-part1",{"title":1569,"path":1570,"stem":1571},"Навігація та керування вікнами. Частина 2: MVVM-навігація","/csharp/desktop-ui/navigation-windows-part2","01.csharp/12.desktop-ui/35.navigation-windows-part2",{"title":1573,"path":1574,"stem":1575},"Avalonia — Навігація та діалоги","/csharp/desktop-ui/35a.avalonia-navigation-dialogs","01.csharp/12.desktop-ui/35a.avalonia-navigation-dialogs",{"title":1577,"path":1578,"stem":1579},"Діалоги та File Pickers у WPF","/csharp/desktop-ui/dialogs-file-pickers","01.csharp/12.desktop-ui/36.dialogs-file-pickers",{"title":1581,"path":1582,"stem":1583},"UserControl: компонентний підхід у WPF","/csharp/desktop-ui/user-controls","01.csharp/12.desktop-ui/37.user-controls",{"title":1585,"path":1586,"stem":1587},"Custom Controls: Lookless Controls у WPF","/csharp/desktop-ui/custom-controls","01.csharp/12.desktop-ui/38.custom-controls",{"title":1589,"path":1590,"stem":1591},"Avalonia TemplatedControl — Lookless Controls","/csharp/desktop-ui/38a.avalonia-templated-controls","01.csharp/12.desktop-ui/38a.avalonia-templated-controls",{"title":1593,"path":1594,"stem":1595},"Анімації у WPF: Storyboard та Easing Functions","/csharp/desktop-ui/animations-transitions","01.csharp/12.desktop-ui/39.animations-transitions",{"title":1597,"path":1598,"stem":1599},"Анімації в Avalonia","/csharp/desktop-ui/39a.avalonia-animations","01.csharp/12.desktop-ui/39a.avalonia-animations",{"title":1601,"path":1602,"stem":1603},"2D Графіка та Мультимедіа у WPF","/csharp/desktop-ui/media-graphics","01.csharp/12.desktop-ui/40.media-graphics",{"title":1605,"path":1606,"stem":1607},"Dependency Injection у WPF та Avalonia","/csharp/desktop-ui/di-integration","01.csharp/12.desktop-ui/41.di-integration",{"title":1609,"path":1610,"stem":1611},"SQLite та EF Core у десктопних додатках","/csharp/desktop-ui/data-persistence-part1","01.csharp/12.desktop-ui/42.data-persistence-part1",{"title":1613,"path":1614,"stem":1615},"Repository Pattern та Unit of Work","/csharp/desktop-ui/data-persistence-part2","01.csharp/12.desktop-ui/43.data-persistence-part2",{"title":1617,"path":1618,"stem":1619},"Тестування ViewModels","/csharp/desktop-ui/viewmodel-testing","01.csharp/12.desktop-ui/44.viewmodel-testing",{"title":1621,"path":1622,"stem":1623},"Avalonia Headless Testing — тестування UI без вікон","/csharp/desktop-ui/44a.avalonia-headless-testing","01.csharp/12.desktop-ui/44a.avalonia-headless-testing",{"title":1625,"path":1626,"stem":1627},"Кросплатформна розробка з Avalonia","/csharp/desktop-ui/avalonia-cross-platform","01.csharp/12.desktop-ui/45.avalonia-cross-platform",{"title":1629,"path":1630,"stem":1631},"Пакування та розгортання Avalonia додатків","/csharp/desktop-ui/avalonia-packaging-deployment","01.csharp/12.desktop-ui/46.avalonia-packaging-deployment",{"title":1633,"path":1634,"stem":1635},"Розгортання WPF застосунків","/csharp/desktop-ui/wpf-packaging-deployment","01.csharp/12.desktop-ui/47.wpf-packaging-deployment",{"title":1637,"path":1638,"stem":1639},"C# & .NET: The Ultimate Roadmap","/csharp/roadmap","01.csharp/roadmap",{"title":1641,"icon":1642,"path":1643,"stem":1644,"children":1645,"page":59},"C++","i-devicon-cplusplus","/cpp","02.cpp",[1646,1650,1654,1658,1662,1666,1670,1674,1678,1681,1685,1689,1693,1697,1701,1705,1709,1713,1717,1721,1725,1729,1733,1737,1741,1745,1749,1753,1757,1761],{"title":1647,"path":1648,"stem":1649},"Вступ у програмування та алгоритми","/cpp/intro-algorithms","02.cpp/01.intro-algorithms",{"title":1651,"path":1652,"stem":1653},"Code Style: угоди про оформлення коду","/cpp/code-style","02.cpp/02.code-style",{"title":1655,"path":1656,"stem":1657},"Середовище розробки та перший проєкт","/cpp/ide-setup","02.cpp/03.ide-setup",{"title":1659,"path":1660,"stem":1661},"Вивід даних на екран","/cpp/data-output","02.cpp/04.data-output",{"title":1663,"path":1664,"stem":1665},"Типи даних, змінні та константи","/cpp/data-types-variables","02.cpp/05.data-types-variables",{"title":1667,"path":1668,"stem":1669},"Ввід даних з клавіатури","/cpp/data-input","02.cpp/06.data-input",{"title":1671,"path":1672,"stem":1673},"Оператори, перетворення типів та логічні операції","/cpp/operators-type-conversion","02.cpp/07.operators-type-conversion",{"title":1675,"path":1676,"stem":1677},"Цикли","/cpp/loops","02.cpp/08.loops",{"title":32,"path":1679,"stem":1680},"/cpp/arrays","02.cpp/09.arrays",{"title":1682,"path":1683,"stem":1684},"Алгоритми сортування та аналіз складності","/cpp/sorting","02.cpp/10.sorting",{"title":1686,"path":1687,"stem":1688},"Алгоритми пошуку","/cpp/searching","02.cpp/11.searching",{"title":1690,"path":1691,"stem":1692},"Функції: основи","/cpp/functions-basics","02.cpp/12.functions-basics",{"title":1694,"path":1695,"stem":1696},"Функції: прототипи, область видимості та додаткові можливості","/cpp/functions-scope","02.cpp/13.functions-scope",{"title":1698,"path":1699,"stem":1700},"Функції: перевантаження та шаблони","/cpp/functions-overloading-templates","02.cpp/14.functions-overloading-templates",{"title":1702,"path":1703,"stem":1704},"Вказівники: основи","/cpp/pointers-basics","02.cpp/15.pointers-basics",{"title":1706,"path":1707,"stem":1708},"Посилання (References)","/cpp/references","02.cpp/16.references",{"title":1710,"path":1711,"stem":1712},"Вказівники, const і масиви","/cpp/pointers-const-arrays","02.cpp/17.pointers-const-arrays",{"title":1714,"path":1715,"stem":1716},"Адресна арифметика","/cpp/pointer-arithmetic","02.cpp/18.pointer-arithmetic",{"title":1718,"path":1719,"stem":1720},"Динамічна пам'ять","/cpp/dynamic-memory","02.cpp/19.dynamic-memory",{"title":1722,"path":1723,"stem":1724},"Вказівники типу void","/cpp/void-pointers","02.cpp/20.void-pointers",{"title":1726,"path":1727,"stem":1728},"Вказівники на вказівники","/cpp/pointers-to-pointers","02.cpp/21.pointers-to-pointers",{"title":1730,"path":1731,"stem":1732},"Оператор доступу до членів через вказівник (->)","/cpp/member-access-operator","02.cpp/22.member-access-operator",{"title":1734,"path":1735,"stem":1736},"Цикл for-each (Range-based for)","/cpp/foreach-loop","02.cpp/23.foreach-loop",{"title":1738,"path":1739,"stem":1740},"Вказівники на функції","/cpp/function-pointers","02.cpp/24.function-pointers",{"title":1742,"path":1743,"stem":1744},"Лямбда-вирази","/cpp/lambdas","02.cpp/25.lambdas",{"title":1746,"path":1747,"stem":1748},"Лямбда-захоплення","/cpp/lambda-captures","02.cpp/26.lambda-captures",{"title":1750,"path":1751,"stem":1752},"Еліпсис","/cpp/ellipsis","02.cpp/27.ellipsis",{"title":1754,"path":1755,"stem":1756},"Аргументи командного рядка","/cpp/command-line-arguments","02.cpp/28.command-line-arguments",{"title":1758,"path":1759,"stem":1760},"Перерахування (enum)","/cpp/enum","02.cpp/29.enum",{"title":1762,"path":1763,"stem":1764},"План навчання: Курс C++ — Продовження (Статті 29–60+)","/cpp/curriculum-plan","02.cpp/curriculum-plan",{"title":1766,"icon":1767,"path":1768,"stem":1769,"children":1770,"page":59},"JavaScript","i-devicon-javascript","/javascript","03.javascript",[1771,1797,1851,1873,2177,2215],{"title":1772,"icon":1773,"path":1774,"stem":1775,"children":1776,"page":59},"Events","i-lucide-mouse-pointer-click","/javascript/events","03.javascript/01.events",[1777,1781,1785,1789,1793],{"title":1778,"path":1779,"stem":1780},"Вступ до подій браузера","/javascript/events/intro","03.javascript/01.events/01.intro",{"title":1782,"path":1783,"stem":1784},"Бульбашковий механізм (Bubbling) та занурення (Capturing)","/javascript/events/bubbling-capturing","03.javascript/01.events/02.bubbling-capturing",{"title":1786,"path":1787,"stem":1788},"Делегування подій (Event Delegation)","/javascript/events/delegate-events","03.javascript/01.events/03.delegate-events",{"title":1790,"path":1791,"stem":1792},"Типові дії браузера та preventDefault()","/javascript/events/prevent-default","03.javascript/01.events/04.prevent-default",{"title":1794,"path":1795,"stem":1796},"Запуск користувацьких подій (Custom Events)","/javascript/events/custom-events","03.javascript/01.events/05.custom-events",{"title":1798,"icon":1799,"path":1800,"stem":1801,"children":1802,"page":59},"Network","i-lucide-globe","/javascript/network","03.javascript/02.network",[1803,1807,1811,1815,1819,1823,1827,1831,1835,1839,1843,1847],{"title":1804,"path":1805,"stem":1806},"Fetch API - Сучасний підхід до HTTP-запитів","/javascript/network/01-fetch-api","03.javascript/02.network/01-fetch-api",{"title":1808,"path":1809,"stem":1810},"FormData - Робота з формами та файлами","/javascript/network/02-formdata","03.javascript/02.network/02-formdata",{"title":1812,"path":1813,"stem":1814},"Відстеження прогресу завантаження","/javascript/network/03-download-progress","03.javascript/02.network/03-download-progress",{"title":1816,"path":1817,"stem":1818},"Переривання fetch-запитів","/javascript/network/04-abort-requests","03.javascript/02.network/04-abort-requests",{"title":1820,"path":1821,"stem":1822},"CORS - Запити між різними джерелами","/javascript/network/05-cors","03.javascript/02.network/05-cors",{"title":1824,"path":1825,"stem":1826},"Fetch API - Повний довідник опцій","/javascript/network/06-fetch-options","03.javascript/02.network/06-fetch-options",{"title":1828,"path":1829,"stem":1830},"URL Objects - Робота з посиланнями","/javascript/network/07-url-objects","03.javascript/02.network/07-url-objects",{"title":1832,"path":1833,"stem":1834},"XMLHttpRequest - AJAX та низькорівневі запити","/javascript/network/08-xmlhttprequest","03.javascript/02.network/08-xmlhttprequest",{"title":1836,"path":1837,"stem":1838},"Відновлюване завантаження файлів","/javascript/network/09-resumable-upload","03.javascript/02.network/09-resumable-upload",{"title":1840,"path":1841,"stem":1842},"Cookies, document.cookie та світ після \"Cookiepocalypse\"","/javascript/network/10-cookies","03.javascript/02.network/10-cookies",{"title":1844,"path":1845,"stem":1846},"js-cookie: Керування Cookies без Болю","/javascript/network/11-js-cookie","03.javascript/02.network/11-js-cookie",{"title":1848,"path":1849,"stem":1850},"Axios: Потужний HTTP-клієнт для JavaScript","/javascript/network/12-axios","03.javascript/02.network/12-axios",{"title":1852,"icon":1853,"path":1854,"stem":1855,"children":1856,"page":59},"Bom","i-lucide-monitor","/javascript/bom","03.javascript/03.bom",[1857,1861,1865,1869],{"title":1858,"path":1859,"stem":1860},"LocalStorage, SessionStorage та patterns збереження даних","/javascript/bom/01-localstorage","03.javascript/03.bom/01-localstorage",{"title":1862,"path":1863,"stem":1864},"Location Object - Керування адресою сторінки","/javascript/bom/02-location-object","03.javascript/03.bom/02-location-object",{"title":1866,"path":1867,"stem":1868},"History API - Керування історією браузера","/javascript/bom/03-history-api","03.javascript/03.bom/03-history-api",{"title":1870,"path":1871,"stem":1872},"Navigator Object - Ідентифікація та Можливості Пристрою","/javascript/bom/04-navigator-object","03.javascript/03.bom/04-navigator-object",{"title":1874,"icon":1875,"path":1876,"stem":1877,"children":1878},"React","i-devicon-react","/javascript/react","03.javascript/04.react/index",[1879,1880,1884,1888,1892,1896,1959,1994,2146],{"title":1874,"path":1876,"stem":1877},{"title":1881,"path":1882,"stem":1883},"Робота з Формами в React","/javascript/react/react-forms","03.javascript/04.react/01.react-forms",{"title":1885,"path":1886,"stem":1887},"React Hook Form: Професійна Робота з Формами","/javascript/react/react-hook-form","03.javascript/04.react/02.react-hook-form",{"title":1889,"path":1890,"stem":1891},"React Hook Form: Глибоке Розуміння Архітектури та Оптимізації","/javascript/react/react-hook-form-new","03.javascript/04.react/02.react-hook-form-new",{"title":1893,"path":1894,"stem":1895},"Axios та React: Професійна Архітектура Запитів","/javascript/react/data-fetching-axios","03.javascript/04.react/03.data-fetching-axios",{"title":1897,"icon":132,"path":1898,"stem":1899,"children":1900},"Tanstack Query","/javascript/react/tanstack-query","03.javascript/04.react/04.tanstack-query/index",[1901,1903,1907,1911,1915,1919,1923,1927,1931,1935,1939,1943,1947,1951,1955],{"title":1902,"path":1898,"stem":1899},"TanStack Query: Майстерність Керування Станом Сервера",{"title":1904,"path":1905,"stem":1906},"Парадигма Server State: Чому useEffect недостатньо","/javascript/react/tanstack-query/server-state-paradigm","03.javascript/04.react/04.tanstack-query/01.server-state-paradigm",{"title":1908,"path":1909,"stem":1910},"Встановлення та Налаштування: Фундамент","/javascript/react/tanstack-query/installation-and-devtools","03.javascript/04.react/04.tanstack-query/02.installation-and-devtools",{"title":1912,"path":1913,"stem":1914},"Основи Запитів та Магія Ключів","/javascript/react/tanstack-query/query-basics-and-keys","03.javascript/04.react/04.tanstack-query/03.query-basics-and-keys",{"title":1916,"path":1917,"stem":1918},"Синхронізація Даних: Життєвий Цикл Запиту","/javascript/react/tanstack-query/data-synchronization","03.javascript/04.react/04.tanstack-query/04.data-synchronization",{"title":1920,"path":1921,"stem":1922},"Мутації та Інвалідація: Зміна Даних","/javascript/react/tanstack-query/mutations-and-invalidation","03.javascript/04.react/04.tanstack-query/05.mutations-and-invalidation",{"title":1924,"path":1925,"stem":1926},"Оптимістичні Оновлення: Швидше за Світло","/javascript/react/tanstack-query/optimistic-updates","03.javascript/04.react/04.tanstack-query/06.optimistic-updates",{"title":1928,"path":1929,"stem":1930},"Пагінація та Infinite Scroll","/javascript/react/tanstack-query/pagination-and-load-more","03.javascript/04.react/04.tanstack-query/07.pagination-and-load-more",{"title":1932,"path":1933,"stem":1934},"Просунуті Патерни та Оптимізація","/javascript/react/tanstack-query/advanced-patterns","03.javascript/04.react/04.tanstack-query/08.advanced-patterns",{"title":1936,"path":1937,"stem":1938},"Архітектура та Best Practices","/javascript/react/tanstack-query/architecture-and-best-practices","03.javascript/04.react/04.tanstack-query/09.architecture-and-best-practices",{"title":1940,"path":1941,"stem":1942},"Server-Side Rendering (SSR) та Гідратація","/javascript/react/tanstack-query/server-side-rendering","03.javascript/04.react/04.tanstack-query/10.server-side-rendering",{"title":1944,"path":1945,"stem":1946},"Стратегії Тестування","/javascript/react/tanstack-query/testing-strategies","03.javascript/04.react/04.tanstack-query/11.testing-strategies",{"title":1948,"path":1949,"stem":1950},"Аутентифікація та Обробка Помилок","/javascript/react/tanstack-query/authentication-and-errors","03.javascript/04.react/04.tanstack-query/12.authentication-and-errors",{"title":1952,"path":1953,"stem":1954},"React Suspense та Майбутнє","/javascript/react/tanstack-query/react-suspense","03.javascript/04.react/04.tanstack-query/13.react-suspense",{"title":1956,"path":1957,"stem":1958},"Глибоке Занурення в Продуктивність","/javascript/react/tanstack-query/performance-deep-dive","03.javascript/04.react/04.tanstack-query/14.performance-deep-dive",{"title":1960,"icon":1875,"path":1961,"stem":1962,"children":1963},"React Router","/javascript/react/react-router","03.javascript/04.react/05.react-router/index",[1964,1966,1970,1974,1978,1982,1986,1990],{"title":1965,"path":1961,"stem":1962},"React Router: Навігаційна система сучасного вебу",{"title":1967,"path":1968,"stem":1969},"Налаштування та Базовий Роутинг","/javascript/react/react-router/setup-and-basic-routing","03.javascript/04.react/05.react-router/01.setup-and-basic-routing",{"title":1971,"path":1972,"stem":1973},"Динамічна Навігація","/javascript/react/react-router/navigation-and-links","03.javascript/04.react/05.react-router/02.navigation-and-links",{"title":1975,"path":1976,"stem":1977},"Вкладені Маршрути та Макети","/javascript/react/react-router/nested-routes-and-layouts","03.javascript/04.react/05.react-router/03.nested-routes-and-layouts",{"title":1979,"path":1980,"stem":1981},"Динамічні Маршрути та Параметри","/javascript/react/react-router/dynamic-routing","03.javascript/04.react/05.react-router/04.dynamic-routing",{"title":1983,"path":1984,"stem":1985},"Data APIs: Loaders та Actions","/javascript/react/react-router/data-loading","03.javascript/04.react/05.react-router/05.data-loading",{"title":1987,"path":1988,"stem":1989},"Просунуті Патерни","/javascript/react/react-router/advanced-patterns","03.javascript/04.react/05.react-router/06.advanced-patterns",{"title":1991,"path":1992,"stem":1993},"Legacy Routing: Компонентний підхід","/javascript/react/react-router/legacy-routing","03.javascript/04.react/05.react-router/07.legacy-routing",{"title":1995,"icon":132,"path":1996,"stem":1997,"children":1998},"Redux","/javascript/react/redux","03.javascript/04.react/06.redux/index",[1999,2001,2017,2046,2055,2076,2092,2121],{"title":2000,"path":1996,"stem":1997},"Redux: Еволюція управління станом",{"title":14,"icon":15,"path":2002,"stem":2003,"children":2004,"page":59},"/javascript/react/redux/fundamentals","03.javascript/04.react/06.redux/01.fundamentals",[2005,2009,2013],{"title":2006,"path":2007,"stem":2008},"Вступ до State Management","/javascript/react/redux/fundamentals/intro-state-management","03.javascript/04.react/06.redux/01.fundamentals/01.intro-state-management",{"title":2010,"path":2011,"stem":2012},"Філософія Redux та Три Принципи","/javascript/react/redux/fundamentals/redux-philosophy","03.javascript/04.react/06.redux/01.fundamentals/02.redux-philosophy",{"title":2014,"path":2015,"stem":2016},"Чисті функції та Іммутабельність","/javascript/react/redux/fundamentals/pure-functions-immutability","03.javascript/04.react/06.redux/01.fundamentals/03.pure-functions-immutability",{"title":2018,"icon":132,"path":2019,"stem":2020,"children":2021,"page":59},"Classic Redux","/javascript/react/redux/classic-redux","03.javascript/04.react/06.redux/02.classic-redux",[2022,2026,2030,2034,2038,2042],{"title":2023,"path":2024,"stem":2025},"Створення Store (Classic Redux)","/javascript/react/redux/classic-redux/store-setup","03.javascript/04.react/06.redux/02.classic-redux/01.store-setup",{"title":2027,"path":2028,"stem":2029},"Actions, Constants та Action Creators","/javascript/react/redux/classic-redux/actions-constants","03.javascript/04.react/06.redux/02.classic-redux/02.actions-constants",{"title":2031,"path":2032,"stem":2033},"Логіка Reducers","/javascript/react/redux/classic-redux/reducers","03.javascript/04.react/06.redux/02.classic-redux/03.reducers",{"title":2035,"path":2036,"stem":2037},"Комбінування Reducers (Root Reducer)","/javascript/react/redux/classic-redux/data-flow","03.javascript/04.react/06.redux/02.classic-redux/04.data-flow",{"title":2039,"path":2040,"stem":2041},"Підключення до React (React-Redux)","/javascript/react/redux/classic-redux/react-redux-connection","03.javascript/04.react/06.redux/02.classic-redux/05.react-redux-connection",{"title":2043,"path":2044,"stem":2045},"Middleware та Асинхронність (Redux Thunk)","/javascript/react/redux/classic-redux/middleware-thunk","03.javascript/04.react/06.redux/02.classic-redux/06.middleware-thunk",{"title":2047,"icon":132,"path":2048,"stem":2049,"children":2050,"page":59},"Transition To Rtk","/javascript/react/redux/transition-to-rtk","03.javascript/04.react/06.redux/03.transition-to-rtk",[2051],{"title":2052,"path":2053,"stem":2054},"Проблеми класичного Redux","/javascript/react/redux/transition-to-rtk/problems-with-classic","03.javascript/04.react/06.redux/03.transition-to-rtk/01.problems-with-classic",{"title":2056,"icon":132,"path":2057,"stem":2058,"children":2059,"page":59},"Redux Toolkit","/javascript/react/redux/redux-toolkit","03.javascript/04.react/06.redux/04.redux-toolkit",[2060,2064,2068,2072],{"title":2061,"path":2062,"stem":2063},"Налаштування Store з configureStore","/javascript/react/redux/redux-toolkit/configure-store","03.javascript/04.react/06.redux/04.redux-toolkit/01.configure-store",{"title":2065,"path":2066,"stem":2067},"createSlice: Революція в Redux","/javascript/react/redux/redux-toolkit/create-slice","03.javascript/04.react/06.redux/04.redux-toolkit/02.create-slice",{"title":2069,"path":2070,"stem":2071},"Асинхронність з createAsyncThunk","/javascript/react/redux/redux-toolkit/async-thunks","03.javascript/04.react/06.redux/04.redux-toolkit/03.async-thunks",{"title":2073,"path":2074,"stem":2075},"04. Entity Adapter: Керування нормалізованим станом","/javascript/react/redux/redux-toolkit/entity-adapter","03.javascript/04.react/06.redux/04.redux-toolkit/04.entity-adapter",{"title":2077,"icon":92,"path":2078,"stem":2079,"children":2080,"page":59},"Advanced","/javascript/react/redux/advanced","03.javascript/04.react/06.redux/05.advanced",[2081,2085,2089],{"title":2082,"path":2083,"stem":2084},"Мемоізація та Селектори: Повний Гайд по Reselect","/javascript/react/redux/advanced/selectors-reselect","03.javascript/04.react/06.redux/05.advanced/01.selectors-reselect",{"title":2086,"path":2087,"stem":2088},"RTK Query: Архітектура Серверного Кешу","/javascript/react/redux/advanced/rtk-query-intro","03.javascript/04.react/06.redux/05.advanced/02.rtk-query-intro",{"title":1936,"path":2090,"stem":2091},"/javascript/react/redux/advanced/architecture-best-practices","03.javascript/04.react/06.redux/05.advanced/03.architecture-best-practices",{"title":2093,"icon":132,"path":2094,"stem":2095,"children":2096,"page":59},"Project Kanban","/javascript/react/redux/project-kanban","03.javascript/04.react/06.redux/06.project-kanban",[2097,2101,2105,2109,2113,2117],{"title":2098,"path":2099,"stem":2100},"Проєкт: Kanban Board (Trello Clone)","/javascript/react/redux/project-kanban/project-overview","03.javascript/04.react/06.redux/06.project-kanban/01.project-overview",{"title":2102,"path":2103,"stem":2104},"Налаштування та Типізація","/javascript/react/redux/project-kanban/setup-and-types","03.javascript/04.react/06.redux/06.project-kanban/02.setup-and-types",{"title":2106,"path":2107,"stem":2108},"Board Slice: Серце Дошки","/javascript/react/redux/project-kanban/board-slice","03.javascript/04.react/06.redux/06.project-kanban/03.board-slice",{"title":2110,"path":2111,"stem":2112},"Логіка Drag & Drop","/javascript/react/redux/project-kanban/drag-and-drop-logic","03.javascript/04.react/06.redux/06.project-kanban/04.drag-and-drop-logic",{"title":2114,"path":2115,"stem":2116},"Інтеграція з RTK Query","/javascript/react/redux/project-kanban/rtk-query-integration","03.javascript/04.react/06.redux/06.project-kanban/05.rtk-query-integration",{"title":2118,"path":2119,"stem":2120},"Optimistic Updates","/javascript/react/redux/project-kanban/optimistic-updates","03.javascript/04.react/06.redux/06.project-kanban/06.optimistic-updates",{"title":2122,"icon":132,"path":2123,"stem":2124,"children":2125,"page":59},"Testing","/javascript/react/redux/testing","03.javascript/04.react/06.redux/07.testing",[2126,2130,2134,2138,2142],{"title":2127,"path":2128,"stem":2129},"Тестування Redux","/javascript/react/redux/testing/intro-testing","03.javascript/04.react/06.redux/07.testing/01.intro-testing",{"title":2131,"path":2132,"stem":2133},"Тестування Reducers","/javascript/react/redux/testing/testing-reducers","03.javascript/04.react/06.redux/07.testing/02.testing-reducers",{"title":2135,"path":2136,"stem":2137},"Тестування Селекторів","/javascript/react/redux/testing/testing-selectors","03.javascript/04.react/06.redux/07.testing/03.testing-selectors",{"title":2139,"path":2140,"stem":2141},"Тестування Компонентів (Integration)","/javascript/react/redux/testing/testing-components","03.javascript/04.react/06.redux/07.testing/04.testing-components",{"title":2143,"path":2144,"stem":2145},"Тестування Async Thunks","/javascript/react/redux/testing/testing-thunks","03.javascript/04.react/06.redux/07.testing/05.testing-thunks",{"title":2147,"icon":132,"path":2148,"stem":2149,"children":2150},"Ui Libraries","/javascript/react/ui-libraries","03.javascript/04.react/07.ui-libraries/index",[2151,2153,2157,2161,2165,2169,2173],{"title":2152,"path":2148,"stem":2149},"UI Бібліотеки в React",{"title":2154,"path":2155,"stem":2156},"Вступ до UI Бібліотек: Навіщо Винаходити Велосипед Двічі?","/javascript/react/ui-libraries/introduction-to-ui-libraries","03.javascript/04.react/07.ui-libraries/01.introduction-to-ui-libraries",{"title":2158,"path":2159,"stem":2160},"Філософія shadcn/ui: \"Not a Component Library\"","/javascript/react/ui-libraries/shadcn-philosophy","03.javascript/04.react/07.ui-libraries/02.shadcn-philosophy",{"title":2162,"path":2163,"stem":2164},"Установка та Налаштування shadcn/ui","/javascript/react/ui-libraries/shadcn-installation","03.javascript/04.react/07.ui-libraries/03.shadcn-installation",{"title":2166,"path":2167,"stem":2168},"Базові Компоненти shadcn/ui: Фундамент Інтерфейсу","/javascript/react/ui-libraries/shadcn-components-basics","03.javascript/04.react/07.ui-libraries/04.shadcn-components-basics",{"title":2170,"path":2171,"stem":2172},"Компоненти Форм: Побудова Інтерактивних Form","/javascript/react/ui-libraries/shadcn-components-forms","03.javascript/04.react/07.ui-libraries/05.shadcn-components-forms",{"title":2174,"path":2175,"stem":2176},"Складні Компоненти: Dialog, Dropdown, Table та Command","/javascript/react/ui-libraries/shadcn-components-advanced","03.javascript/04.react/07.ui-libraries/06.shadcn-components-advanced",{"title":2178,"icon":2179,"path":2180,"stem":2181,"children":2182,"page":59},"TypeScript","i-devicon-typescript","/javascript/typescript","03.javascript/05.typescript",[2183,2187,2191,2195,2199,2203,2207,2211],{"title":2184,"path":2185,"stem":2186},"TypeScript: Броня для вашого коду","/javascript/typescript/intro-and-basic-types","03.javascript/05.typescript/01.intro-and-basic-types",{"title":2188,"path":2189,"stem":2190},"Майстерність Моделювання Даних: Інтерфейси та Просунуті Типи","/javascript/typescript/interfaces-and-advanced-types","03.javascript/05.typescript/02.interfaces-and-advanced-types",{"title":2192,"path":2193,"stem":2194},"Алхімія Типів: Generics та Utility Types","/javascript/typescript/generics-and-utilities","03.javascript/05.typescript/03.generics-and-utilities",{"title":2196,"path":2197,"stem":2198},"Архітектура та Шаблони: Класи в TypeScript","/javascript/typescript/classes-and-oop","03.javascript/05.typescript/04.classes-and-oop",{"title":2200,"path":2201,"stem":2202},"Продакшн та Екосистема: Advanced Config & Workflow","/javascript/typescript/advanced-patterns-and-config","03.javascript/05.typescript/05.advanced-patterns-and-config",{"title":2204,"path":2205,"stem":2206},"TypeScript у світі React","/javascript/typescript/react-basics","03.javascript/05.typescript/06.react-basics",{"title":2208,"path":2209,"stem":2210},"React + TypeScript: Продвинуті патерни","/javascript/typescript/react-advanced","03.javascript/05.typescript/07.react-advanced",{"title":2212,"path":2213,"stem":2214},"React + TypeScript: Екосистема та бібліотеки","/javascript/typescript/react-ecosystem","03.javascript/05.typescript/08.react-ecosystem",{"title":2216,"path":2217,"stem":2218},"Atomic Design","/javascript/atomic-design","03.javascript/2.atomic-design",{"title":2220,"icon":2221,"path":2222,"stem":2223,"children":2224,"page":59},"Java","i-devicon-java","/java","04.java",[2225,2228,2231,2235,2239,2243,2247],{"title":162,"path":2226,"stem":2227},"/java/data-mapper-part1","04.java/01.data-mapper-part1",{"title":166,"path":2229,"stem":2230},"/java/data-mapper-part2","04.java/02.data-mapper-part2",{"title":2232,"path":2233,"stem":2234},"Service Layer: Організація бізнес-логіки","/java/service-layer","04.java/03.service-layer",{"title":2236,"path":2237,"stem":2238},"Rich Domain Model та State Pattern","/java/rich-domain-model","04.java/04.rich-domain-model",{"title":2240,"path":2241,"stem":2242},"Патерни для складної бізнес-логіки","/java/business-logic-patterns","04.java/05.business-logic-patterns",{"title":2244,"path":2245,"stem":2246},"Обробка помилок та валідація","/java/error-handling-validation","04.java/06.error-handling-validation",{"title":2248,"path":2249,"stem":2250,"children":2251,"page":59},"Проектування баз даних","/java/pr2","04.java/pr2",[2252,2256,2260,2264,2268,2272,2276,2280,2284,2288,2292,2296,2300,2304,2308,2312,2316,2320,2324,2328,2332,2336,2340,2344,2348],{"title":2253,"path":2254,"stem":2255},"Концептуальне моделювання: Мистецтво розуміння предметної області","/java/pr2/conceptual-modeling","04.java/pr2/01.conceptual-modeling",{"title":2257,"path":2258,"stem":2259},"Логічне моделювання: Від бізнес-ідей до структур даних","/java/pr2/logical-modeling","04.java/pr2/02.logical-modeling",{"title":2261,"path":2262,"stem":2263},"Нормалізація: Гігієна даних та боротьба з аномаліями","/java/pr2/normalization","04.java/pr2/03.normalization",{"title":2265,"path":2266,"stem":2267},"Фізична схема: Від абстракції до DDL","/java/pr2/physical-schema","04.java/pr2/04.physical-schema",{"title":2269,"path":2270,"stem":2271},"Архітектурна класифікація таблиць","/java/pr2/table-classification","04.java/pr2/05.table-classification",{"title":2273,"path":2274,"stem":2275},"Database Migrations: Версіонування схеми з Flyway","/java/pr2/database-migrations","04.java/pr2/06.database-migrations",{"title":2277,"path":2278,"stem":2279},"А що, якби це була не реляційна БД?","/java/pr2/beyond-relational","04.java/pr2/07.beyond-relational",{"title":2281,"path":2282,"stem":2283},"Object-Relational Impedance Mismatch: Два світи, що не хочуть дружити","/java/pr2/impedance-mismatch","04.java/pr2/09.impedance-mismatch",{"title":2285,"path":2286,"stem":2287},"JDBC: Перший контакт із базою даних","/java/pr2/jdbc-fundamentals","04.java/pr2/10.jdbc-fundamentals",{"title":2289,"path":2290,"stem":2291},"Якість коду: Spotless, SpotBugs та SonarQube","/java/pr2/10a.code-quality","04.java/pr2/10a.code-quality",{"title":2293,"path":2294,"stem":2295},"Connection Pool: Патерн Object Pool для JDBC-з'єднань","/java/pr2/connection-pool","04.java/pr2/11.connection-pool",{"title":2297,"path":2298,"stem":2299},"Row Data Gateway: Об'єкт як обгортка рядка таблиці","/java/pr2/row-data-gateway","04.java/pr2/12.row-data-gateway",{"title":2301,"path":2302,"stem":2303},"Table Data Gateway: Фасад таблиці як архітектурний відступ","/java/pr2/table-data-gateway","04.java/pr2/13.table-data-gateway",{"title":2305,"path":2306,"stem":2307},"Repository + Data Mapper: Правильна шарова архітектура з JDBC","/java/pr2/repository-data-mapper","04.java/pr2/14.repository-data-mapper",{"title":2309,"path":2310,"stem":2311},"Identity Map: Кешування сутностей у рамках сесії","/java/pr2/identity-map","04.java/pr2/15.identity-map",{"title":2313,"path":2314,"stem":2315},"Unit of Work: Відстеження змін і координація JDBC-транзакцій","/java/pr2/unit-of-work","04.java/pr2/16.unit-of-work",{"title":2317,"path":2318,"stem":2319},"Strategy: Замінювані SQL-стратегії для підтримки різних СУБД","/java/pr2/strategy-sql","04.java/pr2/17.strategy-sql",{"title":2321,"path":2322,"stem":2323},"Proxy: Lazy Loading для One-To-Many колекцій","/java/pr2/proxy-lazy-loading","04.java/pr2/18.proxy-lazy-loading",{"title":2325,"path":2326,"stem":2327},"Generic Repository через Java Reflection: анотації та динамічний SQL","/java/pr2/generic-repository-reflection","04.java/pr2/19.generic-repository-reflection",{"title":2329,"path":2330,"stem":2331},"Specification Pattern: Композиція бізнес-правил для складних запитів","/java/pr2/specification-pattern","04.java/pr2/20.specification-pattern",{"title":2333,"path":2334,"stem":2335},"Розширені можливості Specification Pattern: підзапити, агрегації та гібридний підхід","/java/pr2/20a.advanced-specifications","04.java/pr2/20a.advanced-specifications",{"title":2337,"path":2338,"stem":2339},"Асинхронність у JDBC: Від блокуючих викликів до CompletableFuture","/java/pr2/asynchronous-jdbc","04.java/pr2/21.asynchronous-jdbc",{"title":2341,"path":2342,"stem":2343},"Інтеграційне тестування JDBC-репозиторіїв: Embedded H2 та патерн AAA","/java/pr2/integration-testing-h2","04.java/pr2/22.integration-testing-h2",{"title":2345,"path":2346,"stem":2347},"Testcontainers: Тестування з реальною PostgreSQL у Docker-контейнерах","/java/pr2/integration-testing-testcontainers","04.java/pr2/23.integration-testing-testcontainers",{"title":2349,"path":2350,"stem":2351},"Модуль \"Проектування реляційних баз даних\" для 04.java/pr2","/java/pr2/implementation_plan","04.java/pr2/implementation_plan",{"title":2353,"icon":2354,"path":2355,"stem":2356,"children":2357,"page":59},"Бази даних","i-lucide-database","/databases","06.databases",[2358,2388,2411,2448,2477,2495,2529,2541,2550],{"title":2359,"icon":2360,"path":2361,"stem":2362,"children":2363,"page":59},"Intro","i-lucide-play","/databases/intro","06.databases/01.intro",[2364,2368,2372,2376,2380,2384],{"title":2365,"path":2366,"stem":2367},"Введення в теорію баз даних","/databases/intro/introduction-to-databases","06.databases/01.intro/01.introduction-to-databases",{"title":2369,"path":2370,"stem":2371},"Реляційна модель даних","/databases/intro/relational-model-theory","06.databases/01.intro/02.relational-model-theory",{"title":2373,"path":2374,"stem":2375},"ER-моделювання","/databases/intro/er-modeling","06.databases/01.intro/03.er-modeling",{"title":2377,"path":2378,"stem":2379},"Логічне проектування БД","/databases/intro/logical-schema","06.databases/01.intro/04.logical-schema",{"title":2381,"path":2382,"stem":2383},"Класифікація таблиць","/databases/intro/table-classification","06.databases/01.intro/05.table-classification",{"title":2385,"path":2386,"stem":2387},"PlantUML для баз даних","/databases/intro/plantuml-diagrams","06.databases/01.intro/06.plantuml-diagrams",{"title":2389,"icon":2354,"path":2390,"stem":2391,"children":2392,"page":59},"MS SQL Server Start","/databases/ms-sql-server-start","06.databases/02.ms-sql-server-start",[2393,2397,2403,2407],{"title":2394,"path":2395,"stem":2396},"Типи даних у MS SQL Server","/databases/ms-sql-server-start/data-types","06.databases/02.ms-sql-server-start/01.data-types",{"title":2398,"path":2399,"stem":2400,"children":2401},"Індекси у MS SQL Server","/databases/ms-sql-server-start/sql-indexes","06.databases/02.ms-sql-server-start/02.sql-indexes",[2402],{"title":2398,"path":2399,"stem":2400},{"title":2404,"path":2405,"stem":2406},"Системні бази даних MS SQL Server","/databases/ms-sql-server-start/system-databases","06.databases/02.ms-sql-server-start/03.system-databases",{"title":2408,"path":2409,"stem":2410},"Огляд мови SQL та запитів","/databases/ms-sql-server-start/sql-queries-overview","06.databases/02.ms-sql-server-start/04.sql-queries-overview",{"title":2412,"icon":2354,"path":2413,"stem":2414,"children":2415,"page":59},"SQL","/databases/sql","06.databases/03.sql",[2416,2420,2424,2428,2432,2436,2440,2444],{"title":2417,"path":2418,"stem":2419},"Налаштування демонстраційної бази даних","/databases/sql/sample-database-setup","06.databases/03.sql/00.sample-database-setup",{"title":2421,"path":2422,"stem":2423},"DDL - Створення таблиць (CREATE TABLE)","/databases/sql/ddl-create-table","06.databases/03.sql/01.ddl-create-table",{"title":2425,"path":2426,"stem":2427},"DDL - Зміна та видалення таблиць (ALTER, DROP)","/databases/sql/ddl-alter-drop-table","06.databases/03.sql/02.ddl-alter-drop-table",{"title":2429,"path":2430,"stem":2431},"SELECT запити - Основи","/databases/sql/select-queries-fundamentals","06.databases/03.sql/03.select-queries-fundamentals",{"title":2433,"path":2434,"stem":2435},"SELECT запити - Розширені можливості","/databases/sql/select-queries-advanced","06.databases/03.sql/04.select-queries-advanced",{"title":2437,"path":2438,"stem":2439},"INSERT запити - Додавання даних","/databases/sql/insert-queries","06.databases/03.sql/05.insert-queries",{"title":2441,"path":2442,"stem":2443},"UPDATE та DELETE запити","/databases/sql/update-delete-queries","06.databases/03.sql/06.update-delete-queries",{"title":2445,"path":2446,"stem":2447},"Транзакції в SQL","/databases/sql/transactions","06.databases/03.sql/07.transactions",{"title":2449,"icon":2354,"path":2450,"stem":2451,"children":2452,"page":59},"Multi Table Databases","/databases/multi-table-databases","06.databases/04.multi-table-databases",[2453,2457,2461,2465,2469,2473],{"title":2454,"path":2455,"stem":2456},"Зв'язки та нормалізація БД","/databases/multi-table-databases/relationships-and-normalization","06.databases/04.multi-table-databases/00.relationships-and-normalization",{"title":2458,"path":2459,"stem":2460},"INNER JOIN - З'єднання таблиць","/databases/multi-table-databases/inner-join","06.databases/04.multi-table-databases/01.inner-join",{"title":2462,"path":2463,"stem":2464},"OUTER JOINs - LEFT, RIGHT, FULL","/databases/multi-table-databases/outer-joins","06.databases/04.multi-table-databases/02.outer-joins",{"title":2466,"path":2467,"stem":2468},"CROSS та SELF JOINs","/databases/multi-table-databases/cross-self-joins","06.databases/04.multi-table-databases/03.cross-self-joins",{"title":2470,"path":2471,"stem":2472},"Підзапити (Subqueries)","/databases/multi-table-databases/subqueries","06.databases/04.multi-table-databases/04.subqueries",{"title":2474,"path":2475,"stem":2476},"Агрегації з JOIN","/databases/multi-table-databases/aggregations-with-joins","06.databases/04.multi-table-databases/05.aggregations-with-joins",{"title":2478,"icon":2479,"path":2480,"stem":2481,"children":2482,"page":59},"Aggregate Functions","i-lucide-calculator","/databases/aggregate-functions","06.databases/05.aggregate-functions",[2483,2487,2491],{"title":2484,"path":2485,"stem":2486},"Функції агрегування в MS SQL Server","/databases/aggregate-functions/introduction-aggregate-functions","06.databases/05.aggregate-functions/01.introduction-aggregate-functions",{"title":2488,"path":2489,"stem":2490},"Групування даних в MS SQL Server","/databases/aggregate-functions/grouping-data","06.databases/05.aggregate-functions/02.grouping-data",{"title":2492,"path":2493,"stem":2494},"Підзапити з агрегатними функціями","/databases/aggregate-functions/subqueries-aggregates","06.databases/05.aggregate-functions/03.subqueries-aggregates",{"title":2496,"icon":2497,"path":2498,"stem":2499,"children":2500,"page":59},"Тригери та зберігаємі процедури","i-lucide-database-zap","/databases/triggers-stored-procedures","06.databases/07.triggers-stored-procedures",[2501,2505,2509,2513,2517,2521,2525],{"title":2502,"path":2503,"stem":2504},"DML-тригери","/databases/triggers-stored-procedures/dml-triggers","06.databases/07.triggers-stored-procedures/01.dml-triggers",{"title":2506,"path":2507,"stem":2508},"DDL-тригери","/databases/triggers-stored-procedures/ddl-triggers","06.databases/07.triggers-stored-procedures/02.ddl-triggers",{"title":2510,"path":2511,"stem":2512},"Transact-SQL розширення","/databases/triggers-stored-procedures/transact-sql-extensions","06.databases/07.triggers-stored-procedures/03.transact-sql-extensions",{"title":2514,"path":2515,"stem":2516},"Транзакції","/databases/triggers-stored-procedures/transactions","06.databases/07.triggers-stored-procedures/04.transactions",{"title":2518,"path":2519,"stem":2520},"Зберігаємі процедури","/databases/triggers-stored-procedures/stored-procedures","06.databases/07.triggers-stored-procedures/05.stored-procedures",{"title":2522,"path":2523,"stem":2524},"Користувацькі функції","/databases/triggers-stored-procedures/user-defined-functions","06.databases/07.triggers-stored-procedures/06.user-defined-functions",{"title":2526,"path":2527,"stem":2528},"Безпека баз даних","/databases/triggers-stored-procedures/security","06.databases/07.triggers-stored-procedures/08.security",{"title":2526,"icon":793,"path":2530,"stem":2531,"children":2532,"page":59},"/databases/security","06.databases/08.security",[2533,2537],{"title":2534,"path":2535,"stem":2536},"Вступ до безпеки баз даних","/databases/security/introduction","06.databases/08.security/01.introduction",{"title":2538,"path":2539,"stem":2540},"Системні представлення та метадані","/databases/security/system-views","06.databases/08.security/02.system-views",{"title":2542,"icon":2543,"path":2544,"stem":2545,"children":2546,"page":59},"Резервне копіювання та відновлення","i-lucide-database-backup","/databases/backup-recovery","06.databases/09.backup-recovery",[2547],{"title":2542,"path":2548,"stem":2549},"/databases/backup-recovery/backup-restore","06.databases/09.backup-recovery/01.backup-restore",{"title":2551,"icon":2552,"path":2553,"stem":2554,"children":2555,"page":59},"Повнотекстовий пошук","i-lucide-search","/databases/full-text-search","06.databases/10.full-text-search",[2556],{"title":2551,"path":2557,"stem":2558},"/databases/full-text-search/full-text-search","06.databases/10.full-text-search/01.full-text-search",{"title":2560,"icon":2561,"path":2562,"stem":2563,"children":2564,"page":59},"Tools","i-lucide-wrench","/tools","07.tools",[2565],{"title":2566,"icon":2567,"path":2568,"stem":2569,"children":2570},"Docker","i-simple-icons-docker","/tools/docker","07.tools/01.docker/index",[2571,2573,2577,2581,2585,2589,2593,2597,2601,2605,2609,2613,2617,2621,2625,2629,2633,2637],{"title":2572,"path":2568,"stem":2569},"Docker: від нуля до production",{"title":2574,"path":2575,"stem":2576},"Контейнеризація — від проблеми до рішення","/tools/docker/containerization-concept","07.tools/01.docker/01.containerization-concept",{"title":2578,"path":2579,"stem":2580},"Docker — що це і навіщо?","/tools/docker/docker-what-and-why","07.tools/01.docker/02.docker-what-and-why",{"title":2582,"path":2583,"stem":2584},"Архітектура Docker Engine","/tools/docker/docker-architecture","07.tools/01.docker/03.docker-architecture",{"title":2586,"path":2587,"stem":2588},"Встановлення Docker","/tools/docker/installation","07.tools/01.docker/04.installation",{"title":2590,"path":2591,"stem":2592},"Перший контейнер — docker run","/tools/docker/first-container","07.tools/01.docker/05.first-container",{"title":2594,"path":2595,"stem":2596},"Життєвий цикл контейнера","/tools/docker/container-lifecycle","07.tools/01.docker/06.container-lifecycle",{"title":2598,"path":2599,"stem":2600},"Docker Images — фундаментальні концепції","/tools/docker/docker-images-fundamentals","07.tools/01.docker/07.docker-images-fundamentals",{"title":2602,"path":2603,"stem":2604},"Dockerfile — основи","/tools/docker/dockerfile-basics","07.tools/01.docker/08.dockerfile-basics",{"title":2606,"path":2607,"stem":2608},"Dockerfile — просунуті техніки","/tools/docker/dockerfile-advanced","07.tools/01.docker/09.dockerfile-advanced",{"title":2610,"path":2611,"stem":2612},"Build Context та кешування шарів","/tools/docker/build-context-and-cache","07.tools/01.docker/10.build-context-and-cache",{"title":2614,"path":2615,"stem":2616},"Реєстри Docker-образів","/tools/docker/image-registries","07.tools/01.docker/11.image-registries",{"title":2618,"path":2619,"stem":2620},"Контейнеризація .NET додатків","/tools/docker/dotnet-containerization","07.tools/01.docker/12.dotnet-containerization",{"title":2622,"path":2623,"stem":2624},"Томи та збереження даних","/tools/docker/volumes-and-data","07.tools/01.docker/13.volumes-and-data",{"title":2626,"path":2627,"stem":2628},"Основи мережі в Docker","/tools/docker/networking-basics","07.tools/01.docker/14.networking-basics",{"title":2630,"path":2631,"stem":2632},"Змінні оточення та конфігурація","/tools/docker/environment-and-configuration","07.tools/01.docker/15.environment-and-configuration",{"title":2634,"path":2635,"stem":2636},"Docker Compose — оркестрація контейнерів","/tools/docker/docker-compose-basics","07.tools/01.docker/16.docker-compose-basics",{"title":2638,"path":2639,"stem":2640},"Docker Compose — Multi-Service застосунки","/tools/docker/compose-multi-service","07.tools/01.docker/17.compose-multi-service",{"title":2642,"icon":2643,"path":2644,"stem":2645,"children":2646,"page":59},"Software Engineering","i-lucide-code-2","/software-engineering","09.software-engineering",[2647,2651,2655,2659,2663,2667,2671,2675,2679,2683,2687],{"title":2648,"path":2649,"stem":2650},"1. Аналіз предметної області. Експертні знання та складність","/software-engineering/intro.subdomains","09.software-engineering/01.intro.subdomains",{"title":2652,"path":2653,"stem":2654},"2. Обмежені контексти. Інтеграція обмежених контекстів","/software-engineering/integrating-limited-contexts","09.software-engineering/02.integrating-limited-contexts",{"title":2656,"path":2657,"stem":2658},"3. Реалізація простої бізнес-логіки","/software-engineering/simple","09.software-engineering/03.simple",{"title":2660,"path":2661,"stem":2662},"4. Опрацювання складної бізнес-логіки","/software-engineering/complex-business-logic","09.software-engineering/04.complex-business-logic",{"title":2664,"path":2665,"stem":2666},"5. Моделювання фактора часу. Подієво-орієнтована архітектура.","/software-engineering/modelling-the-time-factor","09.software-engineering/05.modelling-the-time-factor",{"title":2668,"path":2669,"stem":2670},"6. Архітектурні патерни","/software-engineering/architectural-patterns","09.software-engineering/06.architectural-patterns",{"title":2672,"path":2673,"stem":2674},"Паттерни взаємодії","/software-engineering/patterns-of-interaction","09.software-engineering/07.patterns-of-interaction",{"title":2676,"path":2677,"stem":2678},"Евристика проєктування","/software-engineering/design-heuristics","09.software-engineering/08.design-heuristics",{"title":2680,"path":2681,"stem":2682},"Еволюція проєктних рішень","/software-engineering/evolution-of-design-solutions","09.software-engineering/09.evolution-of-design-solutions",{"title":2684,"path":2685,"stem":2686},"EventStorming","/software-engineering/eventstorming","09.software-engineering/10.eventstorming",{"title":2688,"path":2689,"stem":2690},"DDD на практиці","/software-engineering/ddd-in-practice","09.software-engineering/11.ddd-in-practice",{"title":2692,"icon":943,"path":2693,"stem":2694,"children":2695,"page":59},"DDD","/ddd","10.ddd",[2696,2700,2704,2708,2712,2716,2720,2724,2728,2732,2736,2740,2744],{"title":2697,"path":2698,"stem":2699},"Аналіз предметної області","/ddd/domain-analysis","10.ddd/01.domain-analysis",{"title":2701,"path":2702,"stem":2703},"Експертні знання про предметну область","/ddd/domain-expert-knowledge","10.ddd/02.domain-expert-knowledge",{"title":2705,"path":2706,"stem":2707},"Як осмислити складність предметної області","/ddd/managing-domain-complexity","10.ddd/03.managing-domain-complexity",{"title":2709,"path":2710,"stem":2711},"Інтеграція обмежених контекстів","/ddd/bounded-context-integration","10.ddd/04.bounded-context-integration",{"title":2713,"path":2714,"stem":2715},"Реалізація простої бізнес-логіки","/ddd/simple-business-logic","10.ddd/05.simple-business-logic",{"title":2717,"path":2718,"stem":2719},"Обробка складної бізнес-логіки","/ddd/complex-business-logic","10.ddd/06.complex-business-logic",{"title":2721,"path":2722,"stem":2723},"Моделювання фактора часу","/ddd/time-modeling","10.ddd/07.time-modeling",{"title":2725,"path":2726,"stem":2727},"Глава 8. Архітектурні Патерни","/ddd/architectural-patterns","10.ddd/08.architectural-patterns",{"title":2729,"path":2730,"stem":2731},"Глава 9. Патерни Взаємодії","/ddd/interaction-patterns","10.ddd/09.interaction-patterns",{"title":2733,"path":2734,"stem":2735},"Глава 10. Проектні Евристики","/ddd/design-heuristics","10.ddd/10.design-heuristics",{"title":2737,"path":2738,"stem":2739},"Глава 11. Еволюція Проектних Рішень","/ddd/evolution-of-design-decisions","10.ddd/11.evolution-of-design-decisions",{"title":2741,"path":2742,"stem":2743},"Глава 12. EventStorming","/ddd/event-storming","10.ddd/12.event-storming",{"title":2745,"path":2746,"stem":2747},"Глава 13. DDD на Практиці","/ddd/ddd-in-practice","10.ddd/13.ddd-in-practice",{"title":2749,"icon":2750,"path":2751,"stem":2752,"children":2753,"page":59},"Media Streaming","i-lucide-video","/media-streaming","11.media-streaming",[2754,2758,2762,2766,2770,2774,2778],{"title":2755,"path":2756,"stem":2757},"01. Магія Стрімінгу: Що відбувається, коли ви натискаєте \"Play\"","/media-streaming/introduction","11.media-streaming/01.introduction",{"title":2759,"path":2760,"stem":2761},"02. Анатомія Медіа: Кодеки, Контейнери та Стиснення","/media-streaming/audio-video-anatomy","11.media-streaming/02.audio-video-anatomy",{"title":2763,"path":2764,"stem":2765},"03. The Gym: FFmpeg Deep Dive","/media-streaming/ffmpeg-gym","11.media-streaming/03.ffmpeg-gym",{"title":2767,"path":2768,"stem":2769},"04. HLS Protocol: HTTP Live Streaming у Деталях","/media-streaming/hls-protocol","11.media-streaming/04.hls-protocol",{"title":2771,"path":2772,"stem":2773},"05. DASH Protocol: Відкритий Стандарт","/media-streaming/dash-protocol","11.media-streaming/05.dash-protocol",{"title":2775,"path":2776,"stem":2777},"06. Масштабування: CDN та Adaptive Bitrate","/media-streaming/cdn-and-adaptive-bitrate","11.media-streaming/06.cdn-and-adaptive-bitrate",{"title":2779,"path":2780,"stem":2781},"07. Війна із Затримкою (Latency)","/media-streaming/realtime-latency","11.media-streaming/07.realtime-latency",{"title":2783,"icon":2784,"path":2785,"stem":2786,"children":2787,"page":59},"HTML & CSS","i-devicon-html5","/html-css","12.html-css",[2788,2792,2796,2800,2804,2808,2812,2816,2820,2824,2828,2832,2836,2840,2844,2848,2852,2856,2860,2864,2868,2872,2876,2880,2884,2888,2892,2896,2900,2904],{"title":2789,"path":2790,"stem":2791},"Вступ до HTML. Структура документа","/html-css/intro-html-structure","12.html-css/01.intro-html-structure",{"title":2793,"path":2794,"stem":2795},"Форматування тексту в HTML","/html-css/html-text-formatting","12.html-css/02.html-text-formatting",{"title":2797,"path":2798,"stem":2799},"Посилання та зображення в HTML","/html-css/html-links-images","12.html-css/03.html-links-images",{"title":2801,"path":2802,"stem":2803},"Списки та таблиці в HTML","/html-css/html-lists-tables","12.html-css/04.html-lists-tables",{"title":2805,"path":2806,"stem":2807},"Форми в HTML","/html-css/html-forms","12.html-css/05.html-forms",{"title":2809,"path":2810,"stem":2811},"Семантичні елементи HTML5","/html-css/html-semantic-elements","12.html-css/06.html-semantic-elements",{"title":2813,"path":2814,"stem":2815},"Мультимедіа та розширені елементи HTML","/html-css/html-multimedia-advanced","12.html-css/07.html-multimedia-advanced",{"title":2817,"path":2818,"stem":2819},"Мікророзмітка та SEO в HTML","/html-css/html-microdata-seo","12.html-css/08.html-microdata-seo",{"title":2821,"path":2822,"stem":2823},"Вступ до CSS. Селектори та специфічність","/html-css/css-intro-selectors","12.html-css/09.css-intro-selectors",{"title":2825,"path":2826,"stem":2827},"Блокова модель CSS. Відступи. Box Sizing","/html-css/css-box-model","12.html-css/10.css-box-model",{"title":2829,"path":2830,"stem":2831},"Розміри у CSS: повний довідник одиниць і ключових слів","/html-css/10a.css-sizing","12.html-css/10a.css-sizing",{"title":2833,"path":2834,"stem":2835},"Типографіка в CSS. Шрифти та текст","/html-css/css-typography","12.html-css/11.css-typography",{"title":2837,"path":2838,"stem":2839},"Кольори та фони в CSS","/html-css/css-colors-backgrounds","12.html-css/12.css-colors-backgrounds",{"title":2841,"path":2842,"stem":2843},"Тіні та фільтри в CSS","/html-css/12b.css-shadows-filters","12.html-css/12b.css-shadows-filters",{"title":2845,"path":2846,"stem":2847},"CSS Flexbox: Фундамент гнучких макетів","/html-css/css-flexbox-fundamentals","12.html-css/13.css-flexbox-fundamentals",{"title":2849,"path":2850,"stem":2851},"CSS Flexbox: Вирівнювання та Позиціонування","/html-css/css-flexbox-alignment-sizing-and-patterns","12.html-css/14.css-flexbox-alignment-sizing-and-patterns",{"title":2853,"path":2854,"stem":2855},"CSS Grid. Двовимірний макет. Частина 1","/html-css/css-layout-grid","12.html-css/15.css-layout-grid",{"title":2857,"path":2858,"stem":2859},"CSS Grid. Двовимірний макет. Частина 2","/html-css/css-layout-grid-advanced","12.html-css/16.css-layout-grid-advanced",{"title":2861,"path":2862,"stem":2863},"Позиціонування в CSS. Z-index. Stacking Context","/html-css/css-positioning","12.html-css/17.css-positioning",{"title":2865,"path":2866,"stem":2867},"CSS Анімації та Переходи","/html-css/css-animations-transitions","12.html-css/18.css-animations-transitions",{"title":2869,"path":2870,"stem":2871},"Адаптивний дизайн. Media Queries. Частина 1","/html-css/css-responsive-media-queries","12.html-css/19.css-responsive-media-queries",{"title":2873,"path":2874,"stem":2875},"Адаптивний дизайн. Частина 2: clamp(), Container Queries, @layer","/html-css/css-responsive-advanced","12.html-css/20.css-responsive-advanced",{"title":2877,"path":2878,"stem":2879},"CSS Custom Properties. Методології. Сучасний CSS","/html-css/css-variables-methodologies","12.html-css/21.css-variables-methodologies",{"title":2881,"path":2882,"stem":2883},"Сучасний CSS 2023–2025: Нові можливості","/html-css/css-modern-features","12.html-css/22.css-modern-features",{"title":2885,"path":2886,"stem":2887},"CSS Nesting, @layer, @scope та @property: нативний препроцесор","/html-css/22a.css-nesting-modern-syntax","12.html-css/22a.css-nesting-modern-syntax",{"title":2889,"path":2890,"stem":2891},"CSS для форм та інтерактивних станів","/html-css/css-forms-interactive-states","12.html-css/23.css-forms-interactive-states",{"title":2893,"path":2894,"stem":2895},"Доступність у CSS (CSS Accessibility)","/html-css/css-accessibility","12.html-css/24.css-accessibility",{"title":2897,"path":2898,"stem":2899},"CSS-функції та сучасні sizing primitives","/html-css/css-functions-sizing","12.html-css/25.css-functions-sizing",{"title":2901,"path":2902,"stem":2903},"Rendering Pipeline і CSS Performance","/html-css/css-rendering-performance","12.html-css/26.css-rendering-performance",{"title":2905,"path":2906,"stem":2907},"CSS Best Practices: типові ситуації та правильні рішення","/html-css/css-best-practices","12.html-css/27.css-best-practices",{"title":2909,"path":2910,"stem":2911,"children":2912,"page":59},"Tailwind","/tailwind","21.tailwind",[2913,2917,2921,2925,2929,2933,2937,2941],{"title":2914,"path":2915,"stem":2916},"Що таке Tailwind CSS і навіщо він потрібен","/tailwind/tailwind-intro-philosophy","21.tailwind/01.tailwind-intro-philosophy",{"title":2918,"path":2919,"stem":2920},"Встановлення та налаштування Tailwind CSS v4","/tailwind/tailwind-installation-setup","21.tailwind/02.tailwind-installation-setup",{"title":2922,"path":2923,"stem":2924},"Utility-класи: основи та система Tailwind","/tailwind/tailwind-utility-classes-core","21.tailwind/03.tailwind-utility-classes-core",{"title":2926,"path":2927,"stem":2928},"Layout: Flexbox та Grid через Tailwind","/tailwind/tailwind-flexbox-grid","21.tailwind/04.tailwind-flexbox-grid",{"title":2930,"path":2931,"stem":2932},"Кастомізація теми через @theme у Tailwind v4","/tailwind/tailwind-theme-customization","21.tailwind/05.tailwind-theme-customization",{"title":2934,"path":2935,"stem":2936},"Варіанти: hover, focus, responsive, dark mode та нові v4","/tailwind/tailwind-variants-states","21.tailwind/06.tailwind-variants-states",{"title":2938,"path":2939,"stem":2940},"Типографіка та система кольорів у Tailwind v4","/tailwind/tailwind-typography-colors","21.tailwind/07.tailwind-typography-colors",{"title":2942,"path":2943,"stem":2944},"Компоненти та повторюваність: @apply, @utility та патерни","/tailwind/tailwind-components-patterns","21.tailwind/08.tailwind-components-patterns",{"title":2946,"path":2947,"stem":2948},"Showcase Компонентів kostyl.dev","/test-new-components","98.test-new-components",{"id":2950,"title":1026,"body":2951,"description":8800,"extension":8801,"links":8802,"meta":8803,"navigation":3462,"path":1027,"seo":8804,"stem":1028,"__hash__":8805},"docs/01.csharp/11.aspnet/07.testing/13.postman-professional.md",{"type":2952,"value":2953,"toc":8750},"minimark",[2954,2958,2962,2965,2970,2973,2976,2979,2991,2994,2997,3001,3006,3009,3016,3022,3028,3034,3038,3041,3394,3397,3401,3404,3412,3415,3418,3422,3425,3429,3484,3490,3496,3516,3522,3526,3529,3716,3719,3725,3732,3743,3747,3750,3760,3764,3775,3778,3782,3788,4557,4572,4579,4583,4586,4847,4851,4862,4866,4872,4924,4928,4931,4936,5465,5470,5901,5906,6029,6034,6295,6299,6302,6306,6309,7023,7030,7034,7038,7043,7046,7072,7076,7082,7119,7122,7218,7222,7225,7386,7561,7565,7568,7572,7575,7578,7582,7864,7875,7879,7882,8402,8413,8566,8578,8582,8589,8592,8599,8606,8609,8613,8733,8735,8746],[2955,2956,1026],"h1",{"id":2957},"професійний-postman-колекції-змінні-та-github-інтеграція",[2959,2960,2961],"p",{},"Ваш API зростає. Ендпоінтів стає більше, деякі з них потребують авторизації, деякі — складного набору заголовків. Ви відкриваєте Swagger, натискаєте кнопку «Authorize», вводите токен, потім окремо перевіряєте кожен маршрут... І так кожен раз. А ваш колега на іншому кінці офісу нічого не знає про те, які запити взагалі існують, і з якими параметрами їх треба відправляти.",[2959,2963,2964],{},"Postman у руках більшості розробників — це просто «красивіший curl». Але у руках інженера, який знає його справжні можливості, — це повноцінна платформа для тестування, документування та автоматизації API. Сьогодні ми пройдемо шлях від «просто відправляю запити» до «у мене є версіонована колекція тестів, яка запускається у GitLab CI при кожному пуші».",[2966,2967,2969],"h2",{"id":2968},"чому-swagger-не-достатньо","Чому Swagger — не достатньо",[2959,2971,2972],{},"Swagger UI (або OpenAPI UI) — чудовий інструмент для одного завдання: переглянути контракт вашого API та зробити одноразовий тестовий запит. Він вбудований у проєкт, завжди актуальний, не потребує окремого встановлення. Але у нього є принципові обмеження.",[2959,2974,2975],{},"По-перше, Swagger не зберігає стан між сесіями. Кожен раз, коли ви перезавантажуєте сторінку, всі параметри, які ви вводили, зникають. Якщо вам треба перевірити 10 різних сценаріїв для одного ендпоінту, ви вводите дані вручну 10 разів.",[2959,2977,2978],{},"По-друге, Swagger не підтримує складні workflow. Якщо вам треба спочатку залогінитись, отримати токен, а потім використати цей токен у наступному запиті — Swagger змусить вас зробити це вручну, копіюючи значення з одного вікна в інше.",[2959,2980,2981,2982,2986,2987,2990],{},"По-третє, у Swagger немає автоматизованих assertions. Ви можете подивитись на відповідь сервера, але перевірити «чи дорівнює ",[2983,2984,2985],"code",{},"status"," рядку ",[2983,2988,2989],{},"\"active\"","» програматично неможливо — тільки очима.",[2959,2992,2993],{},"По-четверте, і це найважливіше — Swagger не можна передати колезі у вигляді файлу, не можна запустити у CI/CD, не можна зробити частиною командної культури тестування.",[2959,2995,2996],{},"Postman вирішує всі ці проблеми.",[2966,2998,3000],{"id":2999},"від-запиту-до-колекції","Від запиту до Колекції",[3002,3003,3005],"h3",{"id":3004},"анатомія-postman","Анатомія Postman",[2959,3007,3008],{},"Перш ніж говорити про просунуті можливості, переконаємося, що ми однаково розуміємо базові поняття.",[2959,3010,3011,3015],{},[3012,3013,3014],"strong",{},"Workspace"," — робочий простір. Може бути особистим (Personal) або командним (Team). У командному workspace всі учасники бачать одні й ті самі колекції в режимі реального часу.",[2959,3017,3018,3021],{},[3012,3019,3020],{},"Collection"," (Колекція) — структурований набір запитів, згрупованих логічно. Колекція — це аналог тестового класу у xUnit: вона групує пов'язані тести, може мати власні змінні та скрипти, і є одиницею, яку можна запустити цілком.",[2959,3023,3024,3027],{},[3012,3025,3026],{},"Request"," (Запит) — окремий HTTP-запит з усіма його параметрами: метод, URL, заголовки, тіло, налаштування автентифікації.",[2959,3029,3030,3033],{},[3012,3031,3032],{},"Folder"," (Папка) — логічна підгрупа запитів всередині колекції. Наприклад, папка «Products» може містити всі запити для роботи з продуктами, папка «Auth» — всі запити для авторизації.",[3002,3035,3037],{"id":3036},"структура-добре-організованої-колекції","Структура добре організованої колекції",[2959,3039,3040],{},"Для Minimal API зі стандартним набором CRUD-операцій типова структура виглядає так:",[3042,3043,3044,3203,3269,3302,3336],"code-tree",{},[3045,3046,3052],"pre",{"className":3047,"code":3048,"filename":3049,"language":3050,"meta":3051,"style":3051},"language-json shiki shiki-themes light-plus dark-plus dark-plus","{\n  \"info\": { \"name\": \"MyApp API\" },\n  \"item\": [\n    { \"name\": \"Auth\", \"item\": [...] },\n    { \"name\": \"Products\", \"item\": [...] },\n    { \"name\": \"Orders\", \"item\": [...] },\n    { \"name\": \"Admin\", \"item\": [...] }\n  ]\n}\n","MyApp API.postman_collection.json","json","",[2983,3053,3054,3063,3086,3095,3124,3146,3168,3191,3197],{"__ignoreMap":3051},[3055,3056,3059],"span",{"class":3057,"line":3058},"line",1,[3055,3060,3062],{"class":3061},"sHH4Y","{\n",[3055,3064,3066,3070,3073,3076,3079,3083],{"class":3057,"line":3065},2,[3055,3067,3069],{"class":3068},"sLwNe","  \"info\"",[3055,3071,3072],{"class":3061},": { ",[3055,3074,3075],{"class":3068},"\"name\"",[3055,3077,3078],{"class":3061},": ",[3055,3080,3082],{"class":3081},"sbdoH","\"MyApp API\"",[3055,3084,3085],{"class":3061}," },\n",[3055,3087,3089,3092],{"class":3057,"line":3088},3,[3055,3090,3091],{"class":3068},"  \"item\"",[3055,3093,3094],{"class":3061},": [\n",[3055,3096,3098,3101,3103,3105,3108,3111,3114,3117,3121],{"class":3057,"line":3097},4,[3055,3099,3100],{"class":3061},"    { ",[3055,3102,3075],{"class":3068},[3055,3104,3078],{"class":3061},[3055,3106,3107],{"class":3081},"\"Auth\"",[3055,3109,3110],{"class":3061},", ",[3055,3112,3113],{"class":3068},"\"item\"",[3055,3115,3116],{"class":3061},": [",[3055,3118,3120],{"class":3119},"se1LK","...",[3055,3122,3123],{"class":3061},"] },\n",[3055,3125,3127,3129,3131,3133,3136,3138,3140,3142,3144],{"class":3057,"line":3126},5,[3055,3128,3100],{"class":3061},[3055,3130,3075],{"class":3068},[3055,3132,3078],{"class":3061},[3055,3134,3135],{"class":3081},"\"Products\"",[3055,3137,3110],{"class":3061},[3055,3139,3113],{"class":3068},[3055,3141,3116],{"class":3061},[3055,3143,3120],{"class":3119},[3055,3145,3123],{"class":3061},[3055,3147,3149,3151,3153,3155,3158,3160,3162,3164,3166],{"class":3057,"line":3148},6,[3055,3150,3100],{"class":3061},[3055,3152,3075],{"class":3068},[3055,3154,3078],{"class":3061},[3055,3156,3157],{"class":3081},"\"Orders\"",[3055,3159,3110],{"class":3061},[3055,3161,3113],{"class":3068},[3055,3163,3116],{"class":3061},[3055,3165,3120],{"class":3119},[3055,3167,3123],{"class":3061},[3055,3169,3171,3173,3175,3177,3180,3182,3184,3186,3188],{"class":3057,"line":3170},7,[3055,3172,3100],{"class":3061},[3055,3174,3075],{"class":3068},[3055,3176,3078],{"class":3061},[3055,3178,3179],{"class":3081},"\"Admin\"",[3055,3181,3110],{"class":3061},[3055,3183,3113],{"class":3068},[3055,3185,3116],{"class":3061},[3055,3187,3120],{"class":3119},[3055,3189,3190],{"class":3061},"] }\n",[3055,3192,3194],{"class":3057,"line":3193},8,[3055,3195,3196],{"class":3061},"  ]\n",[3055,3198,3200],{"class":3057,"line":3199},9,[3055,3201,3202],{"class":3061},"}\n",[3045,3204,3207],{"className":3047,"code":3205,"filename":3206,"language":3050,"meta":3051,"style":3051},"{\n  \"method\": \"POST\",\n  \"url\": \"{{baseUrl}}/api/auth/register\",\n  \"body\": { \"email\": \"...\", \"password\": \"...\" }\n}\n","Auth/POST Register.request",[2983,3208,3209,3213,3226,3238,3265],{"__ignoreMap":3051},[3055,3210,3211],{"class":3057,"line":3058},[3055,3212,3062],{"class":3061},[3055,3214,3215,3218,3220,3223],{"class":3057,"line":3065},[3055,3216,3217],{"class":3068},"  \"method\"",[3055,3219,3078],{"class":3061},[3055,3221,3222],{"class":3081},"\"POST\"",[3055,3224,3225],{"class":3061},",\n",[3055,3227,3228,3231,3233,3236],{"class":3057,"line":3088},[3055,3229,3230],{"class":3068},"  \"url\"",[3055,3232,3078],{"class":3061},[3055,3234,3235],{"class":3081},"\"{{baseUrl}}/api/auth/register\"",[3055,3237,3225],{"class":3061},[3055,3239,3240,3243,3245,3248,3250,3253,3255,3258,3260,3262],{"class":3057,"line":3097},[3055,3241,3242],{"class":3068},"  \"body\"",[3055,3244,3072],{"class":3061},[3055,3246,3247],{"class":3068},"\"email\"",[3055,3249,3078],{"class":3061},[3055,3251,3252],{"class":3081},"\"...\"",[3055,3254,3110],{"class":3061},[3055,3256,3257],{"class":3068},"\"password\"",[3055,3259,3078],{"class":3061},[3055,3261,3252],{"class":3081},[3055,3263,3264],{"class":3061}," }\n",[3055,3266,3267],{"class":3057,"line":3126},[3055,3268,3202],{"class":3061},[3045,3270,3273],{"className":3047,"code":3271,"filename":3272,"language":3050,"meta":3051,"style":3051},"{\n  \"method\": \"POST\",\n  \"url\": \"{{baseUrl}}/api/auth/login\"\n}\n","Auth/POST Login.request",[2983,3274,3275,3279,3289,3298],{"__ignoreMap":3051},[3055,3276,3277],{"class":3057,"line":3058},[3055,3278,3062],{"class":3061},[3055,3280,3281,3283,3285,3287],{"class":3057,"line":3065},[3055,3282,3217],{"class":3068},[3055,3284,3078],{"class":3061},[3055,3286,3222],{"class":3081},[3055,3288,3225],{"class":3061},[3055,3290,3291,3293,3295],{"class":3057,"line":3088},[3055,3292,3230],{"class":3068},[3055,3294,3078],{"class":3061},[3055,3296,3297],{"class":3081},"\"{{baseUrl}}/api/auth/login\"\n",[3055,3299,3300],{"class":3057,"line":3097},[3055,3301,3202],{"class":3061},[3045,3303,3306],{"className":3047,"code":3304,"filename":3305,"language":3050,"meta":3051,"style":3051},"{\n  \"method\": \"GET\",\n  \"url\": \"{{baseUrl}}/api/products\"\n}\n","Products/GET All Products.request",[2983,3307,3308,3312,3323,3332],{"__ignoreMap":3051},[3055,3309,3310],{"class":3057,"line":3058},[3055,3311,3062],{"class":3061},[3055,3313,3314,3316,3318,3321],{"class":3057,"line":3065},[3055,3315,3217],{"class":3068},[3055,3317,3078],{"class":3061},[3055,3319,3320],{"class":3081},"\"GET\"",[3055,3322,3225],{"class":3061},[3055,3324,3325,3327,3329],{"class":3057,"line":3088},[3055,3326,3230],{"class":3068},[3055,3328,3078],{"class":3061},[3055,3330,3331],{"class":3081},"\"{{baseUrl}}/api/products\"\n",[3055,3333,3334],{"class":3057,"line":3097},[3055,3335,3202],{"class":3061},[3045,3337,3340],{"className":3047,"code":3338,"filename":3339,"language":3050,"meta":3051,"style":3051},"{\n  \"method\": \"POST\",\n  \"url\": \"{{baseUrl}}/api/products\",\n  \"body\": { \"name\": \"...\", \"price\": ... }\n}\n","Products/POST Create Product.request",[2983,3341,3342,3346,3356,3367,3390],{"__ignoreMap":3051},[3055,3343,3344],{"class":3057,"line":3058},[3055,3345,3062],{"class":3061},[3055,3347,3348,3350,3352,3354],{"class":3057,"line":3065},[3055,3349,3217],{"class":3068},[3055,3351,3078],{"class":3061},[3055,3353,3222],{"class":3081},[3055,3355,3225],{"class":3061},[3055,3357,3358,3360,3362,3365],{"class":3057,"line":3088},[3055,3359,3230],{"class":3068},[3055,3361,3078],{"class":3061},[3055,3363,3364],{"class":3081},"\"{{baseUrl}}/api/products\"",[3055,3366,3225],{"class":3061},[3055,3368,3369,3371,3373,3375,3377,3379,3381,3384,3386,3388],{"class":3057,"line":3097},[3055,3370,3242],{"class":3068},[3055,3372,3072],{"class":3061},[3055,3374,3075],{"class":3068},[3055,3376,3078],{"class":3061},[3055,3378,3252],{"class":3081},[3055,3380,3110],{"class":3061},[3055,3382,3383],{"class":3068},"\"price\"",[3055,3385,3078],{"class":3061},[3055,3387,3120],{"class":3119},[3055,3389,3264],{"class":3061},[3055,3391,3392],{"class":3057,"line":3126},[3055,3393,3202],{"class":3061},[2959,3395,3396],{},"Ця структура не просто для краси. Postman дозволяє запустити всю колекцію або окрему папку одним кліком (або однією командою в CLI). Папки стають одиницями тестування.",[3002,3398,3400],{"id":3399},"чому-хардкод-це-завжди-погано","Чому хардкод — це завжди погано",[2959,3402,3403],{},"Новачки в Postman роблять одну й ту саму помилку: вони хардкодять URL та токени прямо у запити.",[3045,3405,3410],{"className":3406,"code":3408,"language":3409},[3407],"language-text","POST https://my-production-api.com/api/auth/login\nAuthorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...\n","text",[2983,3411,3408],{"__ignoreMap":3051},[2959,3413,3414],{},"Ця колекція одразу стає непридатною для командної роботи (у кожного свій хост), для тестування у різних середовищах (dev, staging, production), і самі собою токени протухають. Людина повинна вручну оновлювати їх у кожному запиті. Коли запитів 50 — це пекло.",[2959,3416,3417],{},"Правильне рішення — змінні.",[2966,3419,3421],{"id":3420},"змінні-глобальні-колекційні-та-середовищні","Змінні: Глобальні, Колекційні та Середовищні",[2959,3423,3424],{},"Postman має чотири рівні змінних, які перекривають один одного від загального до конкретного. Розуміння цієї ієрархії — ключ до чистих колекцій.",[3002,3426,3428],{"id":3427},"ієрархія-змінних","Ієрархія змінних",[3430,3431,3432],"mermaid",{},[3045,3433,3436],{"className":3434,"code":3435,"language":3430,"meta":3051,"style":3051},"language-mermaid shiki shiki-themes light-plus dark-plus dark-plus","graph TD\n    A[\"🌍 Global Variables\u003Cbr/>(всі workspace, всі колекції)\"] --> B[\"📦 Collection Variables\u003Cbr/>(вся колекція, всі середовища)\"]\n    B --> C[\"🔧 Environment Variables\u003Cbr/>(конкретне середовище: dev/staging/prod)\"]\n    C --> D[\"💬 Local Variables\u003Cbr/>(один скрипт, один запит)\"]\n\n    style A fill:#64748b,stroke:#334155,color:#ffffff\n    style B fill:#3b82f6,stroke:#1d4ed8,color:#ffffff\n    style C fill:#f59e0b,stroke:#b45309,color:#ffffff\n    style D fill:#10b981,stroke:#047857,color:#ffffff\n",[2983,3437,3438,3443,3448,3453,3458,3464,3469,3474,3479],{"__ignoreMap":3051},[3055,3439,3440],{"class":3057,"line":3058},[3055,3441,3442],{},"graph TD\n",[3055,3444,3445],{"class":3057,"line":3065},[3055,3446,3447],{},"    A[\"🌍 Global Variables\u003Cbr/>(всі workspace, всі колекції)\"] --> B[\"📦 Collection Variables\u003Cbr/>(вся колекція, всі середовища)\"]\n",[3055,3449,3450],{"class":3057,"line":3088},[3055,3451,3452],{},"    B --> C[\"🔧 Environment Variables\u003Cbr/>(конкретне середовище: dev/staging/prod)\"]\n",[3055,3454,3455],{"class":3057,"line":3097},[3055,3456,3457],{},"    C --> D[\"💬 Local Variables\u003Cbr/>(один скрипт, один запит)\"]\n",[3055,3459,3460],{"class":3057,"line":3126},[3055,3461,3463],{"emptyLinePlaceholder":3462},true,"\n",[3055,3465,3466],{"class":3057,"line":3148},[3055,3467,3468],{},"    style A fill:#64748b,stroke:#334155,color:#ffffff\n",[3055,3470,3471],{"class":3057,"line":3170},[3055,3472,3473],{},"    style B fill:#3b82f6,stroke:#1d4ed8,color:#ffffff\n",[3055,3475,3476],{"class":3057,"line":3193},[3055,3477,3478],{},"    style C fill:#f59e0b,stroke:#b45309,color:#ffffff\n",[3055,3480,3481],{"class":3057,"line":3199},[3055,3482,3483],{},"    style D fill:#10b981,stroke:#047857,color:#ffffff\n",[2959,3485,3486,3489],{},[3012,3487,3488],{},"Global Variables"," — змінні, доступні у всіх колекціях у всьому workspace. Підходять для даних, які справді однакові всюди: наприклад, ім'я компанії або якийсь фіксований ключ. На практиці використовуються рідко — переважно для тимчасових «глобальних» значень у Pre-request Scripts.",[2959,3491,3492,3495],{},[3012,3493,3494],{},"Collection Variables"," — змінні, прив'язані до конкретної колекції і доступні в усіх її запитах незалежно від обраного середовища. Ідеально підходять для значень, які не змінюються між середовищами: назви ресурсів, ідентифікатори тестових об'єктів, константи.",[2959,3497,3498,3501,3502,3110,3505,3110,3508,3511,3512,3515],{},[3012,3499,3500],{},"Environment Variables"," — найважливіший рівень для ваших сценаріїв. Середовище (Environment) — це набір змінних, який ви активуєте у Postman. Ви можете мати три середовища: ",[2983,3503,3504],{},"Development",[2983,3506,3507],{},"Staging",[2983,3509,3510],{},"Production",". Кожне містить свій ",[2983,3513,3514],{},"baseUrl",", свої credentials, свої API-ключі. Переключаєте середовище — і всі ваші запити магічно починають відправляти запити на інший сервер.",[2959,3517,3518,3521],{},[3012,3519,3520],{},"Local Variables"," — змінні, які ви встановлюєте у скрипті і які існують лише під час виконання цього одного запиту. Корисні для проміжних обчислень у Pre-request Scripts.",[3002,3523,3525],{"id":3524},"налаштування-середовищ","Налаштування середовищ",[2959,3527,3528],{},"Для нашого Minimal API налаштуємо три середовища. У кожному буде повний набір необхідних змінних:",[3530,3531,3532,3603,3660],"tabs",{},[3533,3534,3535],"tabs-item",{"label":3504},[3536,3537,3538,3551],"table",{},[3539,3540,3541],"thead",{},[3542,3543,3544,3548],"tr",{},[3545,3546,3547],"th",{},"Змінна",[3545,3549,3550],{},"Значення",[3552,3553,3554,3566,3578,3590],"tbody",{},[3542,3555,3556,3561],{},[3557,3558,3559],"td",{},[2983,3560,3514],{},[3557,3562,3563],{},[2983,3564,3565],{},"http://localhost:5000",[3542,3567,3568,3573],{},[3557,3569,3570],{},[2983,3571,3572],{},"username",[3557,3574,3575],{},[2983,3576,3577],{},"admin@localhost",[3542,3579,3580,3585],{},[3557,3581,3582],{},[2983,3583,3584],{},"password",[3557,3586,3587],{},[2983,3588,3589],{},"Dev_Password_123",[3542,3591,3592,3597],{},[3557,3593,3594],{},[2983,3595,3596],{},"authToken",[3557,3598,3599],{},[3600,3601,3602],"em",{},"(порожньо — заповнюється скриптом)",[3533,3604,3605],{"label":3507},[3536,3606,3607,3615],{},[3539,3608,3609],{},[3542,3610,3611,3613],{},[3545,3612,3547],{},[3545,3614,3550],{},[3552,3616,3617,3628,3639,3650],{},[3542,3618,3619,3623],{},[3557,3620,3621],{},[2983,3622,3514],{},[3557,3624,3625],{},[2983,3626,3627],{},"https://staging.myapp.com",[3542,3629,3630,3634],{},[3557,3631,3632],{},[2983,3633,3572],{},[3557,3635,3636],{},[2983,3637,3638],{},"admin@staging.myapp.com",[3542,3640,3641,3645],{},[3557,3642,3643],{},[2983,3644,3584],{},[3557,3646,3647],{},[2983,3648,3649],{},"*(secret у Vault)*",[3542,3651,3652,3656],{},[3557,3653,3654],{},[2983,3655,3596],{},[3557,3657,3658],{},[3600,3659,3602],{},[3533,3661,3662],{"label":3510},[3536,3663,3664,3672],{},[3539,3665,3666],{},[3542,3667,3668,3670],{},[3545,3669,3547],{},[3545,3671,3550],{},[3552,3673,3674,3685,3696,3706],{},[3542,3675,3676,3680],{},[3557,3677,3678],{},[2983,3679,3514],{},[3557,3681,3682],{},[2983,3683,3684],{},"https://api.myapp.com",[3542,3686,3687,3691],{},[3557,3688,3689],{},[2983,3690,3572],{},[3557,3692,3693],{},[2983,3694,3695],{},"admin@myapp.com",[3542,3697,3698,3702],{},[3557,3699,3700],{},[2983,3701,3584],{},[3557,3703,3704],{},[2983,3705,3649],{},[3542,3707,3708,3712],{},[3557,3709,3710],{},[2983,3711,3596],{},[3557,3713,3714],{},[3600,3715,3602],{},[2959,3717,3718],{},"Після налаштування всі ваші запити використовують змінні замість хардкод-значень:",[3045,3720,3723],{"className":3721,"code":3722,"language":3409},[3407],"POST {{baseUrl}}/api/auth/login\n",[2983,3724,3722],{"__ignoreMap":3051},[2959,3726,3727,3728,3731],{},"Синтаксис подвійних фігурних дужок ",[2983,3729,3730],{},"{{variableName}}"," працює у будь-якому полі Postman: URL, заголовки, тіло запиту, параметри.",[3733,3734,3735,3738,3739,3742],"tip",{},[3012,3736,3737],{},"Секрети у змінних",": Для паролів та API-ключів позначайте змінну як ",[3012,3740,3741],{},"Secret"," у Postman (є відповідний тип). Це маскує значення у UI та запобігає їх потраплянню у логи. Ніколи не зберігайте реальні паролі у Collection Variables, які синхронізуються з хмарою.",[2966,3744,3746],{"id":3745},"pre-request-scripts-автоматизуємо-авторизацію","Pre-request Scripts: Автоматизуємо Авторизацію",[2959,3748,3749],{},"Це одна з найпотужніших функцій Postman, яку більшість розробників або не знає, або не використовує повністю.",[2959,3751,3752,3755,3756,3759],{},[3012,3753,3754],{},"Pre-request Script"," — це JavaScript-код, який Postman виконує ",[3012,3757,3758],{},"перед"," відправкою запиту. Він може робити інші HTTP-запити, обчислювати значення, встановлювати змінні.",[3002,3761,3763],{"id":3762},"проблема-токени-протухають","Проблема: токени протухають",[2959,3765,3766,3767,3770,3771,3774],{},"Уявіть типовий сценарій: ваш JWT токен діє 60 хвилин. Ви починаєте тестувати, отримуєте токен... і через годину половина ваших запитів починає повертати ",[2983,3768,3769],{},"401 Unauthorized",". Ви знову робите POST на ",[2983,3772,3773],{},"/login",", копіюєте токен, замінюєте у змінній... Це ручна, дратівлива робота.",[2959,3776,3777],{},"Правильне рішення — колекційний Pre-request Script, який автоматично перевіряє актуальність токену і оновлює його за потреби.",[3002,3779,3781],{"id":3780},"реалізація-автоматичного-логіну","Реалізація автоматичного логіну",[2959,3783,3784,3785,3787],{},"Відкрийте вашу колекцію, перейдіть на вкладку ",[3012,3786,3754],{}," (на рівні колекції, а не окремого запиту — тоді він виконується перед кожним запитом у колекції):",[3045,3789,3793],{"className":3790,"code":3791,"language":3792,"meta":3051,"style":3051},"language-javascript shiki shiki-themes light-plus dark-plus dark-plus","// Pre-request Script на рівні колекції\n// Автоматично отримує та оновлює JWT токен\n\nconst tokenExpiry = pm.collectionVariables.get(\"tokenExpiry\");\nconst authToken = pm.collectionVariables.get(\"authToken\");\nconst now = Date.now();\n\n// Перевіряємо: чи є токен і чи він ще дійсний (з запасом у 60 секунд)\nif (authToken && tokenExpiry && now \u003C (parseInt(tokenExpiry) - 60000)) {\n    // Токен ще дійсний — нічого не робимо\n    console.log(\"Token is still valid, skipping login.\");\n    return;\n}\n\n// Токен відсутній або протух — логінимось\nconsole.log(\"Token expired or missing. Performing login...\");\n\nconst baseUrl = pm.environment.get(\"baseUrl\");\nconst username = pm.environment.get(\"username\");\nconst password = pm.environment.get(\"password\");\n\n// pm.sendRequest — асинхронний запит зсередини скрипту\npm.sendRequest({\n    url: `${baseUrl}/api/auth/login`,\n    method: \"POST\",\n    header: {\n        \"Content-Type\": \"application/json\"\n    },\n    body: {\n        mode: \"raw\",\n        raw: JSON.stringify({ username, password })\n    }\n}, function(err, response) {\n    if (err) {\n        console.error(\"Login failed:\", err);\n        return;\n    }\n\n    if (response.code !== 200) {\n        console.error(\"Login returned non-200:\", response.code, response.text());\n        return;\n    }\n\n    const json = response.json();\n\n    // Зберігаємо токен у змінній колекції\n    pm.collectionVariables.set(\"authToken\", json.accessToken);\n\n    // Зберігаємо час закінчення дії токену (поточний час + 3600 секунд)\n    const expiryMs = now + (json.expiresIn * 1000 || 3600000);\n    pm.collectionVariables.set(\"tokenExpiry\", expiryMs.toString());\n\n    console.log(\"Login successful. Token cached until:\", new Date(expiryMs).toISOString());\n});\n","javascript",[2983,3794,3795,3801,3806,3810,3848,3874,3894,3898,3903,3944,3950,3968,3977,3982,3987,3993,4010,4015,4043,4070,4096,4101,4107,4120,4142,4153,4162,4174,4180,4188,4199,4225,4231,4253,4265,4287,4295,4300,4305,4326,4359,4366,4371,4376,4395,4400,4406,4436,4441,4447,4483,4512,4517,4551],{"__ignoreMap":3051},[3055,3796,3797],{"class":3057,"line":3058},[3055,3798,3800],{"class":3799},"spJ8K","// Pre-request Script на рівні колекції\n",[3055,3802,3803],{"class":3057,"line":3065},[3055,3804,3805],{"class":3799},"// Автоматично отримує та оновлює JWT токен\n",[3055,3807,3808],{"class":3057,"line":3088},[3055,3809,3463],{"emptyLinePlaceholder":3462},[3055,3811,3812,3816,3820,3823,3827,3830,3833,3835,3839,3842,3845],{"class":3057,"line":3097},[3055,3813,3815],{"class":3814},"su1O8","const",[3055,3817,3819],{"class":3818},"s-QsJ"," tokenExpiry",[3055,3821,3822],{"class":3061}," = ",[3055,3824,3826],{"class":3825},"siwwj","pm",[3055,3828,3829],{"class":3061},".",[3055,3831,3832],{"class":3825},"collectionVariables",[3055,3834,3829],{"class":3061},[3055,3836,3838],{"class":3837},"s8Opu","get",[3055,3840,3841],{"class":3061},"(",[3055,3843,3844],{"class":3081},"\"tokenExpiry\"",[3055,3846,3847],{"class":3061},");\n",[3055,3849,3850,3852,3855,3857,3859,3861,3863,3865,3867,3869,3872],{"class":3057,"line":3126},[3055,3851,3815],{"class":3814},[3055,3853,3854],{"class":3818}," authToken",[3055,3856,3822],{"class":3061},[3055,3858,3826],{"class":3825},[3055,3860,3829],{"class":3061},[3055,3862,3832],{"class":3825},[3055,3864,3829],{"class":3061},[3055,3866,3838],{"class":3837},[3055,3868,3841],{"class":3061},[3055,3870,3871],{"class":3081},"\"authToken\"",[3055,3873,3847],{"class":3061},[3055,3875,3876,3878,3881,3883,3886,3888,3891],{"class":3057,"line":3148},[3055,3877,3815],{"class":3814},[3055,3879,3880],{"class":3818}," now",[3055,3882,3822],{"class":3061},[3055,3884,3885],{"class":3825},"Date",[3055,3887,3829],{"class":3061},[3055,3889,3890],{"class":3837},"now",[3055,3892,3893],{"class":3061},"();\n",[3055,3895,3896],{"class":3057,"line":3170},[3055,3897,3463],{"emptyLinePlaceholder":3462},[3055,3899,3900],{"class":3057,"line":3193},[3055,3901,3902],{"class":3799},"// Перевіряємо: чи є токен і чи він ще дійсний (з запасом у 60 секунд)\n",[3055,3904,3905,3909,3912,3914,3917,3920,3922,3924,3927,3930,3932,3934,3937,3941],{"class":3057,"line":3199},[3055,3906,3908],{"class":3907},"sCDza","if",[3055,3910,3911],{"class":3061}," (",[3055,3913,3596],{"class":3825},[3055,3915,3916],{"class":3061}," && ",[3055,3918,3919],{"class":3825},"tokenExpiry",[3055,3921,3916],{"class":3061},[3055,3923,3890],{"class":3825},[3055,3925,3926],{"class":3061}," \u003C (",[3055,3928,3929],{"class":3837},"parseInt",[3055,3931,3841],{"class":3061},[3055,3933,3919],{"class":3825},[3055,3935,3936],{"class":3061},") - ",[3055,3938,3940],{"class":3939},"sJj4R","60000",[3055,3942,3943],{"class":3061},")) {\n",[3055,3945,3947],{"class":3057,"line":3946},10,[3055,3948,3949],{"class":3799},"    // Токен ще дійсний — нічого не робимо\n",[3055,3951,3953,3956,3958,3961,3963,3966],{"class":3057,"line":3952},11,[3055,3954,3955],{"class":3825},"    console",[3055,3957,3829],{"class":3061},[3055,3959,3960],{"class":3837},"log",[3055,3962,3841],{"class":3061},[3055,3964,3965],{"class":3081},"\"Token is still valid, skipping login.\"",[3055,3967,3847],{"class":3061},[3055,3969,3971,3974],{"class":3057,"line":3970},12,[3055,3972,3973],{"class":3907},"    return",[3055,3975,3976],{"class":3061},";\n",[3055,3978,3980],{"class":3057,"line":3979},13,[3055,3981,3202],{"class":3061},[3055,3983,3985],{"class":3057,"line":3984},14,[3055,3986,3463],{"emptyLinePlaceholder":3462},[3055,3988,3990],{"class":3057,"line":3989},15,[3055,3991,3992],{"class":3799},"// Токен відсутній або протух — логінимось\n",[3055,3994,3996,3999,4001,4003,4005,4008],{"class":3057,"line":3995},16,[3055,3997,3998],{"class":3825},"console",[3055,4000,3829],{"class":3061},[3055,4002,3960],{"class":3837},[3055,4004,3841],{"class":3061},[3055,4006,4007],{"class":3081},"\"Token expired or missing. Performing login...\"",[3055,4009,3847],{"class":3061},[3055,4011,4013],{"class":3057,"line":4012},17,[3055,4014,3463],{"emptyLinePlaceholder":3462},[3055,4016,4018,4020,4023,4025,4027,4029,4032,4034,4036,4038,4041],{"class":3057,"line":4017},18,[3055,4019,3815],{"class":3814},[3055,4021,4022],{"class":3818}," baseUrl",[3055,4024,3822],{"class":3061},[3055,4026,3826],{"class":3825},[3055,4028,3829],{"class":3061},[3055,4030,4031],{"class":3825},"environment",[3055,4033,3829],{"class":3061},[3055,4035,3838],{"class":3837},[3055,4037,3841],{"class":3061},[3055,4039,4040],{"class":3081},"\"baseUrl\"",[3055,4042,3847],{"class":3061},[3055,4044,4046,4048,4051,4053,4055,4057,4059,4061,4063,4065,4068],{"class":3057,"line":4045},19,[3055,4047,3815],{"class":3814},[3055,4049,4050],{"class":3818}," username",[3055,4052,3822],{"class":3061},[3055,4054,3826],{"class":3825},[3055,4056,3829],{"class":3061},[3055,4058,4031],{"class":3825},[3055,4060,3829],{"class":3061},[3055,4062,3838],{"class":3837},[3055,4064,3841],{"class":3061},[3055,4066,4067],{"class":3081},"\"username\"",[3055,4069,3847],{"class":3061},[3055,4071,4073,4075,4078,4080,4082,4084,4086,4088,4090,4092,4094],{"class":3057,"line":4072},20,[3055,4074,3815],{"class":3814},[3055,4076,4077],{"class":3818}," password",[3055,4079,3822],{"class":3061},[3055,4081,3826],{"class":3825},[3055,4083,3829],{"class":3061},[3055,4085,4031],{"class":3825},[3055,4087,3829],{"class":3061},[3055,4089,3838],{"class":3837},[3055,4091,3841],{"class":3061},[3055,4093,3257],{"class":3081},[3055,4095,3847],{"class":3061},[3055,4097,4099],{"class":3057,"line":4098},21,[3055,4100,3463],{"emptyLinePlaceholder":3462},[3055,4102,4104],{"class":3057,"line":4103},22,[3055,4105,4106],{"class":3799},"// pm.sendRequest — асинхронний запит зсередини скрипту\n",[3055,4108,4110,4112,4114,4117],{"class":3057,"line":4109},23,[3055,4111,3826],{"class":3825},[3055,4113,3829],{"class":3061},[3055,4115,4116],{"class":3837},"sendRequest",[3055,4118,4119],{"class":3061},"({\n",[3055,4121,4123,4126,4129,4132,4134,4137,4140],{"class":3057,"line":4122},24,[3055,4124,4125],{"class":3825},"    url:",[3055,4127,4128],{"class":3081}," `",[3055,4130,4131],{"class":3814},"${",[3055,4133,3514],{"class":3825},[3055,4135,4136],{"class":3814},"}",[3055,4138,4139],{"class":3081},"/api/auth/login`",[3055,4141,3225],{"class":3061},[3055,4143,4145,4148,4151],{"class":3057,"line":4144},25,[3055,4146,4147],{"class":3825},"    method:",[3055,4149,4150],{"class":3081}," \"POST\"",[3055,4152,3225],{"class":3061},[3055,4154,4156,4159],{"class":3057,"line":4155},26,[3055,4157,4158],{"class":3825},"    header:",[3055,4160,4161],{"class":3061}," {\n",[3055,4163,4165,4168,4171],{"class":3057,"line":4164},27,[3055,4166,4167],{"class":3081},"        \"Content-Type\"",[3055,4169,4170],{"class":3825},":",[3055,4172,4173],{"class":3081}," \"application/json\"\n",[3055,4175,4177],{"class":3057,"line":4176},28,[3055,4178,4179],{"class":3061},"    },\n",[3055,4181,4183,4186],{"class":3057,"line":4182},29,[3055,4184,4185],{"class":3825},"    body:",[3055,4187,4161],{"class":3061},[3055,4189,4191,4194,4197],{"class":3057,"line":4190},30,[3055,4192,4193],{"class":3825},"        mode:",[3055,4195,4196],{"class":3081}," \"raw\"",[3055,4198,3225],{"class":3061},[3055,4200,4202,4205,4208,4210,4213,4216,4218,4220,4222],{"class":3057,"line":4201},31,[3055,4203,4204],{"class":3825},"        raw:",[3055,4206,4207],{"class":3818}," JSON",[3055,4209,3829],{"class":3061},[3055,4211,4212],{"class":3837},"stringify",[3055,4214,4215],{"class":3061},"({ ",[3055,4217,3572],{"class":3825},[3055,4219,3110],{"class":3061},[3055,4221,3584],{"class":3825},[3055,4223,4224],{"class":3061}," })\n",[3055,4226,4228],{"class":3057,"line":4227},32,[3055,4229,4230],{"class":3061},"    }\n",[3055,4232,4234,4237,4240,4242,4245,4247,4250],{"class":3057,"line":4233},33,[3055,4235,4236],{"class":3061},"}, ",[3055,4238,4239],{"class":3814},"function",[3055,4241,3841],{"class":3061},[3055,4243,4244],{"class":3825},"err",[3055,4246,3110],{"class":3061},[3055,4248,4249],{"class":3825},"response",[3055,4251,4252],{"class":3061},") {\n",[3055,4254,4256,4259,4261,4263],{"class":3057,"line":4255},34,[3055,4257,4258],{"class":3907},"    if",[3055,4260,3911],{"class":3061},[3055,4262,4244],{"class":3825},[3055,4264,4252],{"class":3061},[3055,4266,4268,4271,4273,4276,4278,4281,4283,4285],{"class":3057,"line":4267},35,[3055,4269,4270],{"class":3825},"        console",[3055,4272,3829],{"class":3061},[3055,4274,4275],{"class":3837},"error",[3055,4277,3841],{"class":3061},[3055,4279,4280],{"class":3081},"\"Login failed:\"",[3055,4282,3110],{"class":3061},[3055,4284,4244],{"class":3825},[3055,4286,3847],{"class":3061},[3055,4288,4290,4293],{"class":3057,"line":4289},36,[3055,4291,4292],{"class":3907},"        return",[3055,4294,3976],{"class":3061},[3055,4296,4298],{"class":3057,"line":4297},37,[3055,4299,4230],{"class":3061},[3055,4301,4303],{"class":3057,"line":4302},38,[3055,4304,3463],{"emptyLinePlaceholder":3462},[3055,4306,4308,4310,4312,4314,4316,4318,4321,4324],{"class":3057,"line":4307},39,[3055,4309,4258],{"class":3907},[3055,4311,3911],{"class":3061},[3055,4313,4249],{"class":3825},[3055,4315,3829],{"class":3061},[3055,4317,2983],{"class":3825},[3055,4319,4320],{"class":3061}," !== ",[3055,4322,4323],{"class":3939},"200",[3055,4325,4252],{"class":3061},[3055,4327,4329,4331,4333,4335,4337,4340,4342,4344,4346,4348,4350,4352,4354,4356],{"class":3057,"line":4328},40,[3055,4330,4270],{"class":3825},[3055,4332,3829],{"class":3061},[3055,4334,4275],{"class":3837},[3055,4336,3841],{"class":3061},[3055,4338,4339],{"class":3081},"\"Login returned non-200:\"",[3055,4341,3110],{"class":3061},[3055,4343,4249],{"class":3825},[3055,4345,3829],{"class":3061},[3055,4347,2983],{"class":3825},[3055,4349,3110],{"class":3061},[3055,4351,4249],{"class":3825},[3055,4353,3829],{"class":3061},[3055,4355,3409],{"class":3837},[3055,4357,4358],{"class":3061},"());\n",[3055,4360,4362,4364],{"class":3057,"line":4361},41,[3055,4363,4292],{"class":3907},[3055,4365,3976],{"class":3061},[3055,4367,4369],{"class":3057,"line":4368},42,[3055,4370,4230],{"class":3061},[3055,4372,4374],{"class":3057,"line":4373},43,[3055,4375,3463],{"emptyLinePlaceholder":3462},[3055,4377,4379,4382,4385,4387,4389,4391,4393],{"class":3057,"line":4378},44,[3055,4380,4381],{"class":3814},"    const",[3055,4383,4384],{"class":3818}," json",[3055,4386,3822],{"class":3061},[3055,4388,4249],{"class":3825},[3055,4390,3829],{"class":3061},[3055,4392,3050],{"class":3837},[3055,4394,3893],{"class":3061},[3055,4396,4398],{"class":3057,"line":4397},45,[3055,4399,3463],{"emptyLinePlaceholder":3462},[3055,4401,4403],{"class":3057,"line":4402},46,[3055,4404,4405],{"class":3799},"    // Зберігаємо токен у змінній колекції\n",[3055,4407,4409,4412,4414,4416,4418,4421,4423,4425,4427,4429,4431,4434],{"class":3057,"line":4408},47,[3055,4410,4411],{"class":3825},"    pm",[3055,4413,3829],{"class":3061},[3055,4415,3832],{"class":3825},[3055,4417,3829],{"class":3061},[3055,4419,4420],{"class":3837},"set",[3055,4422,3841],{"class":3061},[3055,4424,3871],{"class":3081},[3055,4426,3110],{"class":3061},[3055,4428,3050],{"class":3825},[3055,4430,3829],{"class":3061},[3055,4432,4433],{"class":3825},"accessToken",[3055,4435,3847],{"class":3061},[3055,4437,4439],{"class":3057,"line":4438},48,[3055,4440,3463],{"emptyLinePlaceholder":3462},[3055,4442,4444],{"class":3057,"line":4443},49,[3055,4445,4446],{"class":3799},"    // Зберігаємо час закінчення дії токену (поточний час + 3600 секунд)\n",[3055,4448,4450,4452,4455,4457,4459,4462,4464,4466,4469,4472,4475,4478,4481],{"class":3057,"line":4449},50,[3055,4451,4381],{"class":3814},[3055,4453,4454],{"class":3818}," expiryMs",[3055,4456,3822],{"class":3061},[3055,4458,3890],{"class":3825},[3055,4460,4461],{"class":3061}," + (",[3055,4463,3050],{"class":3825},[3055,4465,3829],{"class":3061},[3055,4467,4468],{"class":3825},"expiresIn",[3055,4470,4471],{"class":3061}," * ",[3055,4473,4474],{"class":3939},"1000",[3055,4476,4477],{"class":3061}," || ",[3055,4479,4480],{"class":3939},"3600000",[3055,4482,3847],{"class":3061},[3055,4484,4486,4488,4490,4492,4494,4496,4498,4500,4502,4505,4507,4510],{"class":3057,"line":4485},51,[3055,4487,4411],{"class":3825},[3055,4489,3829],{"class":3061},[3055,4491,3832],{"class":3825},[3055,4493,3829],{"class":3061},[3055,4495,4420],{"class":3837},[3055,4497,3841],{"class":3061},[3055,4499,3844],{"class":3081},[3055,4501,3110],{"class":3061},[3055,4503,4504],{"class":3825},"expiryMs",[3055,4506,3829],{"class":3061},[3055,4508,4509],{"class":3837},"toString",[3055,4511,4358],{"class":3061},[3055,4513,4515],{"class":3057,"line":4514},52,[3055,4516,3463],{"emptyLinePlaceholder":3462},[3055,4518,4520,4522,4524,4526,4528,4531,4533,4536,4539,4541,4543,4546,4549],{"class":3057,"line":4519},53,[3055,4521,3955],{"class":3825},[3055,4523,3829],{"class":3061},[3055,4525,3960],{"class":3837},[3055,4527,3841],{"class":3061},[3055,4529,4530],{"class":3081},"\"Login successful. Token cached until:\"",[3055,4532,3110],{"class":3061},[3055,4534,4535],{"class":3814},"new",[3055,4537,4538],{"class":3837}," Date",[3055,4540,3841],{"class":3061},[3055,4542,4504],{"class":3825},[3055,4544,4545],{"class":3061},").",[3055,4547,4548],{"class":3837},"toISOString",[3055,4550,4358],{"class":3061},[3055,4552,4554],{"class":3057,"line":4553},54,[3055,4555,4556],{"class":3061},"});\n",[2959,4558,4559,4560,4563,4564,4567,4568,4571],{},"Тепер у кожному запиті колекції у вкладці ",[3012,4561,4562],{},"Authorization"," встановіть тип ",[2983,4565,4566],{},"Bearer Token"," і значення ",[2983,4569,4570],{},"{{authToken}}",". Postman сам підставить актуальний токен перед кожним запитом.",[3733,4573,4575,4578],{"title":4574},"Як це працює",[2983,4576,4577],{},"pm.sendRequest"," — синхронний у контексті скрипту, тобто Postman дочекається відповіді перед відправкою основного запиту. Це дозволяє будувати складні ланцюжки залежних запитів прямо у скрипті.",[3002,4580,4582],{"id":4581},"генерація-динамічних-даних","Генерація динамічних даних",[2959,4584,4585],{},"Ще один корисний сценарій Pre-request Script — генерація унікальних тестових даних перед запитом:",[3045,4587,4589],{"className":3790,"code":4588,"language":3792,"meta":3051,"style":3051},"// Генеруємо унікальний email для кожного тесту створення користувача\nconst timestamp = Date.now();\nconst uniqueEmail = `testuser_${timestamp}@example.com`;\npm.variables.set(\"uniqueEmail\", uniqueEmail);\n\n// Генеруємо UUID вручну (без бібліотек)\nfunction generateUUID() {\n    return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {\n        const r = Math.random() * 16 | 0;\n        const v = c === 'x' ? r : (r & 0x3 | 0x8);\n        return v.toString(16);\n    });\n}\npm.variables.set(\"correlationId\", generateUUID());\n",[2983,4590,4591,4596,4613,4637,4662,4666,4671,4681,4725,4757,4798,4814,4819,4823],{"__ignoreMap":3051},[3055,4592,4593],{"class":3057,"line":3058},[3055,4594,4595],{"class":3799},"// Генеруємо унікальний email для кожного тесту створення користувача\n",[3055,4597,4598,4600,4603,4605,4607,4609,4611],{"class":3057,"line":3065},[3055,4599,3815],{"class":3814},[3055,4601,4602],{"class":3818}," timestamp",[3055,4604,3822],{"class":3061},[3055,4606,3885],{"class":3825},[3055,4608,3829],{"class":3061},[3055,4610,3890],{"class":3837},[3055,4612,3893],{"class":3061},[3055,4614,4615,4617,4620,4622,4625,4627,4630,4632,4635],{"class":3057,"line":3088},[3055,4616,3815],{"class":3814},[3055,4618,4619],{"class":3818}," uniqueEmail",[3055,4621,3822],{"class":3061},[3055,4623,4624],{"class":3081},"`testuser_",[3055,4626,4131],{"class":3814},[3055,4628,4629],{"class":3825},"timestamp",[3055,4631,4136],{"class":3814},[3055,4633,4634],{"class":3081},"@example.com`",[3055,4636,3976],{"class":3061},[3055,4638,4639,4641,4643,4646,4648,4650,4652,4655,4657,4660],{"class":3057,"line":3097},[3055,4640,3826],{"class":3825},[3055,4642,3829],{"class":3061},[3055,4644,4645],{"class":3825},"variables",[3055,4647,3829],{"class":3061},[3055,4649,4420],{"class":3837},[3055,4651,3841],{"class":3061},[3055,4653,4654],{"class":3081},"\"uniqueEmail\"",[3055,4656,3110],{"class":3061},[3055,4658,4659],{"class":3825},"uniqueEmail",[3055,4661,3847],{"class":3061},[3055,4663,4664],{"class":3057,"line":3126},[3055,4665,3463],{"emptyLinePlaceholder":3462},[3055,4667,4668],{"class":3057,"line":3148},[3055,4669,4670],{"class":3799},"// Генеруємо UUID вручну (без бібліотек)\n",[3055,4672,4673,4675,4678],{"class":3057,"line":3170},[3055,4674,4239],{"class":3814},[3055,4676,4677],{"class":3837}," generateUUID",[3055,4679,4680],{"class":3061},"() {\n",[3055,4682,4683,4685,4688,4690,4693,4695,4699,4703,4706,4709,4711,4714,4716,4718,4720,4723],{"class":3057,"line":3193},[3055,4684,3973],{"class":3907},[3055,4686,4687],{"class":3081}," 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'",[3055,4689,3829],{"class":3061},[3055,4691,4692],{"class":3837},"replace",[3055,4694,3841],{"class":3061},[3055,4696,4698],{"class":4697},"shsrj","/",[3055,4700,4702],{"class":4701},"sAV2Q","[",[3055,4704,4705],{"class":4697},"xy",[3055,4707,4708],{"class":4701},"]",[3055,4710,4698],{"class":4697},[3055,4712,4713],{"class":3814},"g",[3055,4715,3110],{"class":3061},[3055,4717,4239],{"class":3814},[3055,4719,3841],{"class":3061},[3055,4721,4722],{"class":3825},"c",[3055,4724,4252],{"class":3061},[3055,4726,4727,4730,4733,4735,4738,4740,4743,4746,4749,4752,4755],{"class":3057,"line":3199},[3055,4728,4729],{"class":3814},"        const",[3055,4731,4732],{"class":3818}," r",[3055,4734,3822],{"class":3061},[3055,4736,4737],{"class":3825},"Math",[3055,4739,3829],{"class":3061},[3055,4741,4742],{"class":3837},"random",[3055,4744,4745],{"class":3061},"() * ",[3055,4747,4748],{"class":3939},"16",[3055,4750,4751],{"class":3061}," | ",[3055,4753,4754],{"class":3939},"0",[3055,4756,3976],{"class":3061},[3055,4758,4759,4761,4764,4766,4768,4771,4774,4777,4780,4783,4785,4788,4791,4793,4796],{"class":3057,"line":3946},[3055,4760,4729],{"class":3814},[3055,4762,4763],{"class":3818}," v",[3055,4765,3822],{"class":3061},[3055,4767,4722],{"class":3825},[3055,4769,4770],{"class":3061}," === ",[3055,4772,4773],{"class":3081},"'x'",[3055,4775,4776],{"class":3061}," ? ",[3055,4778,4779],{"class":3825},"r",[3055,4781,4782],{"class":3061}," : (",[3055,4784,4779],{"class":3825},[3055,4786,4787],{"class":3061}," & ",[3055,4789,4790],{"class":3939},"0x3",[3055,4792,4751],{"class":3061},[3055,4794,4795],{"class":3939},"0x8",[3055,4797,3847],{"class":3061},[3055,4799,4800,4802,4804,4806,4808,4810,4812],{"class":3057,"line":3952},[3055,4801,4292],{"class":3907},[3055,4803,4763],{"class":3825},[3055,4805,3829],{"class":3061},[3055,4807,4509],{"class":3837},[3055,4809,3841],{"class":3061},[3055,4811,4748],{"class":3939},[3055,4813,3847],{"class":3061},[3055,4815,4816],{"class":3057,"line":3970},[3055,4817,4818],{"class":3061},"    });\n",[3055,4820,4821],{"class":3057,"line":3979},[3055,4822,3202],{"class":3061},[3055,4824,4825,4827,4829,4831,4833,4835,4837,4840,4842,4845],{"class":3057,"line":3984},[3055,4826,3826],{"class":3825},[3055,4828,3829],{"class":3061},[3055,4830,4645],{"class":3825},[3055,4832,3829],{"class":3061},[3055,4834,4420],{"class":3837},[3055,4836,3841],{"class":3061},[3055,4838,4839],{"class":3081},"\"correlationId\"",[3055,4841,3110],{"class":3061},[3055,4843,4844],{"class":3837},"generateUUID",[3055,4846,4358],{"class":3061},[2966,4848,4850],{"id":4849},"tests-postman-sandbox-автоматизовані-assertions","Tests (Postman Sandbox): Автоматизовані Assertions",[2959,4852,4853,4854,4857,4858,4861],{},"Вкладка ",[3012,4855,4856],{},"Tests"," у кожному запиті — це JavaScript-код, який виконується ",[3012,4859,4860],{},"після"," отримання відповіді. Тут ви пишете перевірки (assertions) на кшталт тих, що пишуться у xUnit, але на JavaScript через API Postman.",[3002,4863,4865],{"id":4864},"поверхня-api-для-тестів","Поверхня API для тестів",[2959,4867,4868,4869,4871],{},"Об'єкт ",[2983,4870,3826],{}," — головний об'єкт Postman Sandbox. Для написання тестів нам знадобляться:",[4873,4874,4875,4886,4892,4914],"ul",{},[4876,4877,4878,4881,4882,4885],"li",{},[2983,4879,4880],{},"pm.test(name, fn)"," — декларує один тест. ",[2983,4883,4884],{},"fn"," — функція, яка кидає помилку при провалі.",[4876,4887,4888,4891],{},[2983,4889,4890],{},"pm.expect(value)"," — Chai.js-стиль assertions.",[4876,4893,4894,4897,4898,3110,4900,3110,4902,3110,4905,3110,4908,3110,4911,3829],{},[2983,4895,4896],{},"pm.response"," — об'єкт відповіді з полями ",[2983,4899,2983],{},[2983,4901,2985],{},[2983,4903,4904],{},"json()",[2983,4906,4907],{},"text()",[2983,4909,4910],{},"headers",[2983,4912,4913],{},"responseTime",[4876,4915,4916,4919,4920,4923],{},[2983,4917,4918],{},"pm.environment.set(key, value)"," / ",[2983,4921,4922],{},"pm.collectionVariables.set(key, value)"," — запис змінних.",[3002,4925,4927],{"id":4926},"повний-набір-тестів-для-crud-операцій","Повний набір тестів для CRUD-операцій",[2959,4929,4930],{},"Розглянемо, як пишуться тести для типових ендпоінтів Minimal API.",[2959,4932,4933],{},[3012,4934,4935],{},"GET /api/products — отримання списку:",[3045,4937,4939],{"className":3790,"code":4938,"language":3792,"meta":3051,"style":3051},"// Tests для GET /api/products\npm.test(\"Status code is 200\", function() {\n    pm.response.to.have.status(200);\n});\n\npm.test(\"Response time is less than 500ms\", function() {\n    pm.expect(pm.response.responseTime).to.be.below(500);\n});\n\npm.test(\"Response is an array\", function() {\n    const json = pm.response.json();\n    pm.expect(json).to.be.an(\"array\");\n});\n\npm.test(\"Each product has required fields\", function() {\n    const products = pm.response.json();\n    // Перевіряємо перший елемент (якщо список непустий)\n    if (products.length > 0) {\n        const product = products[0];\n        pm.expect(product).to.have.property(\"id\");\n        pm.expect(product).to.have.property(\"name\");\n        pm.expect(product).to.have.property(\"price\");\n        pm.expect(product.price).to.be.a(\"number\").that.is.above(0);\n    }\n});\n\npm.test(\"Content-Type is application/json\", function() {\n    pm.expect(pm.response.headers.get(\"Content-Type\")).to.include(\"application/json\");\n});\n",[2983,4940,4941,4946,4966,4994,4998,5002,5021,5063,5067,5071,5090,5110,5142,5146,5150,5169,5190,5195,5216,5234,5268,5298,5328,5384,5388,5392,5396,5415,5461],{"__ignoreMap":3051},[3055,4942,4943],{"class":3057,"line":3058},[3055,4944,4945],{"class":3799},"// Tests для GET /api/products\n",[3055,4947,4948,4950,4952,4955,4957,4960,4962,4964],{"class":3057,"line":3065},[3055,4949,3826],{"class":3825},[3055,4951,3829],{"class":3061},[3055,4953,4954],{"class":3837},"test",[3055,4956,3841],{"class":3061},[3055,4958,4959],{"class":3081},"\"Status code is 200\"",[3055,4961,3110],{"class":3061},[3055,4963,4239],{"class":3814},[3055,4965,4680],{"class":3061},[3055,4967,4968,4970,4972,4974,4976,4979,4981,4984,4986,4988,4990,4992],{"class":3057,"line":3088},[3055,4969,4411],{"class":3825},[3055,4971,3829],{"class":3061},[3055,4973,4249],{"class":3825},[3055,4975,3829],{"class":3061},[3055,4977,4978],{"class":3825},"to",[3055,4980,3829],{"class":3061},[3055,4982,4983],{"class":3825},"have",[3055,4985,3829],{"class":3061},[3055,4987,2985],{"class":3837},[3055,4989,3841],{"class":3061},[3055,4991,4323],{"class":3939},[3055,4993,3847],{"class":3061},[3055,4995,4996],{"class":3057,"line":3097},[3055,4997,4556],{"class":3061},[3055,4999,5000],{"class":3057,"line":3126},[3055,5001,3463],{"emptyLinePlaceholder":3462},[3055,5003,5004,5006,5008,5010,5012,5015,5017,5019],{"class":3057,"line":3148},[3055,5005,3826],{"class":3825},[3055,5007,3829],{"class":3061},[3055,5009,4954],{"class":3837},[3055,5011,3841],{"class":3061},[3055,5013,5014],{"class":3081},"\"Response time is less than 500ms\"",[3055,5016,3110],{"class":3061},[3055,5018,4239],{"class":3814},[3055,5020,4680],{"class":3061},[3055,5022,5023,5025,5027,5030,5032,5034,5036,5038,5040,5042,5044,5046,5048,5051,5053,5056,5058,5061],{"class":3057,"line":3170},[3055,5024,4411],{"class":3825},[3055,5026,3829],{"class":3061},[3055,5028,5029],{"class":3837},"expect",[3055,5031,3841],{"class":3061},[3055,5033,3826],{"class":3825},[3055,5035,3829],{"class":3061},[3055,5037,4249],{"class":3825},[3055,5039,3829],{"class":3061},[3055,5041,4913],{"class":3825},[3055,5043,4545],{"class":3061},[3055,5045,4978],{"class":3825},[3055,5047,3829],{"class":3061},[3055,5049,5050],{"class":3825},"be",[3055,5052,3829],{"class":3061},[3055,5054,5055],{"class":3837},"below",[3055,5057,3841],{"class":3061},[3055,5059,5060],{"class":3939},"500",[3055,5062,3847],{"class":3061},[3055,5064,5065],{"class":3057,"line":3193},[3055,5066,4556],{"class":3061},[3055,5068,5069],{"class":3057,"line":3199},[3055,5070,3463],{"emptyLinePlaceholder":3462},[3055,5072,5073,5075,5077,5079,5081,5084,5086,5088],{"class":3057,"line":3946},[3055,5074,3826],{"class":3825},[3055,5076,3829],{"class":3061},[3055,5078,4954],{"class":3837},[3055,5080,3841],{"class":3061},[3055,5082,5083],{"class":3081},"\"Response is an array\"",[3055,5085,3110],{"class":3061},[3055,5087,4239],{"class":3814},[3055,5089,4680],{"class":3061},[3055,5091,5092,5094,5096,5098,5100,5102,5104,5106,5108],{"class":3057,"line":3952},[3055,5093,4381],{"class":3814},[3055,5095,4384],{"class":3818},[3055,5097,3822],{"class":3061},[3055,5099,3826],{"class":3825},[3055,5101,3829],{"class":3061},[3055,5103,4249],{"class":3825},[3055,5105,3829],{"class":3061},[3055,5107,3050],{"class":3837},[3055,5109,3893],{"class":3061},[3055,5111,5112,5114,5116,5118,5120,5122,5124,5126,5128,5130,5132,5135,5137,5140],{"class":3057,"line":3970},[3055,5113,4411],{"class":3825},[3055,5115,3829],{"class":3061},[3055,5117,5029],{"class":3837},[3055,5119,3841],{"class":3061},[3055,5121,3050],{"class":3825},[3055,5123,4545],{"class":3061},[3055,5125,4978],{"class":3825},[3055,5127,3829],{"class":3061},[3055,5129,5050],{"class":3825},[3055,5131,3829],{"class":3061},[3055,5133,5134],{"class":3837},"an",[3055,5136,3841],{"class":3061},[3055,5138,5139],{"class":3081},"\"array\"",[3055,5141,3847],{"class":3061},[3055,5143,5144],{"class":3057,"line":3979},[3055,5145,4556],{"class":3061},[3055,5147,5148],{"class":3057,"line":3984},[3055,5149,3463],{"emptyLinePlaceholder":3462},[3055,5151,5152,5154,5156,5158,5160,5163,5165,5167],{"class":3057,"line":3989},[3055,5153,3826],{"class":3825},[3055,5155,3829],{"class":3061},[3055,5157,4954],{"class":3837},[3055,5159,3841],{"class":3061},[3055,5161,5162],{"class":3081},"\"Each product has required fields\"",[3055,5164,3110],{"class":3061},[3055,5166,4239],{"class":3814},[3055,5168,4680],{"class":3061},[3055,5170,5171,5173,5176,5178,5180,5182,5184,5186,5188],{"class":3057,"line":3995},[3055,5172,4381],{"class":3814},[3055,5174,5175],{"class":3818}," products",[3055,5177,3822],{"class":3061},[3055,5179,3826],{"class":3825},[3055,5181,3829],{"class":3061},[3055,5183,4249],{"class":3825},[3055,5185,3829],{"class":3061},[3055,5187,3050],{"class":3837},[3055,5189,3893],{"class":3061},[3055,5191,5192],{"class":3057,"line":4012},[3055,5193,5194],{"class":3799},"    // Перевіряємо перший елемент (якщо список непустий)\n",[3055,5196,5197,5199,5201,5204,5206,5209,5212,5214],{"class":3057,"line":4017},[3055,5198,4258],{"class":3907},[3055,5200,3911],{"class":3061},[3055,5202,5203],{"class":3825},"products",[3055,5205,3829],{"class":3061},[3055,5207,5208],{"class":3825},"length",[3055,5210,5211],{"class":3061}," > ",[3055,5213,4754],{"class":3939},[3055,5215,4252],{"class":3061},[3055,5217,5218,5220,5223,5225,5227,5229,5231],{"class":3057,"line":4045},[3055,5219,4729],{"class":3814},[3055,5221,5222],{"class":3818}," product",[3055,5224,3822],{"class":3061},[3055,5226,5203],{"class":3825},[3055,5228,4702],{"class":3061},[3055,5230,4754],{"class":3939},[3055,5232,5233],{"class":3061},"];\n",[3055,5235,5236,5239,5241,5243,5245,5248,5250,5252,5254,5256,5258,5261,5263,5266],{"class":3057,"line":4072},[3055,5237,5238],{"class":3825},"        pm",[3055,5240,3829],{"class":3061},[3055,5242,5029],{"class":3837},[3055,5244,3841],{"class":3061},[3055,5246,5247],{"class":3825},"product",[3055,5249,4545],{"class":3061},[3055,5251,4978],{"class":3825},[3055,5253,3829],{"class":3061},[3055,5255,4983],{"class":3825},[3055,5257,3829],{"class":3061},[3055,5259,5260],{"class":3837},"property",[3055,5262,3841],{"class":3061},[3055,5264,5265],{"class":3081},"\"id\"",[3055,5267,3847],{"class":3061},[3055,5269,5270,5272,5274,5276,5278,5280,5282,5284,5286,5288,5290,5292,5294,5296],{"class":3057,"line":4098},[3055,5271,5238],{"class":3825},[3055,5273,3829],{"class":3061},[3055,5275,5029],{"class":3837},[3055,5277,3841],{"class":3061},[3055,5279,5247],{"class":3825},[3055,5281,4545],{"class":3061},[3055,5283,4978],{"class":3825},[3055,5285,3829],{"class":3061},[3055,5287,4983],{"class":3825},[3055,5289,3829],{"class":3061},[3055,5291,5260],{"class":3837},[3055,5293,3841],{"class":3061},[3055,5295,3075],{"class":3081},[3055,5297,3847],{"class":3061},[3055,5299,5300,5302,5304,5306,5308,5310,5312,5314,5316,5318,5320,5322,5324,5326],{"class":3057,"line":4103},[3055,5301,5238],{"class":3825},[3055,5303,3829],{"class":3061},[3055,5305,5029],{"class":3837},[3055,5307,3841],{"class":3061},[3055,5309,5247],{"class":3825},[3055,5311,4545],{"class":3061},[3055,5313,4978],{"class":3825},[3055,5315,3829],{"class":3061},[3055,5317,4983],{"class":3825},[3055,5319,3829],{"class":3061},[3055,5321,5260],{"class":3837},[3055,5323,3841],{"class":3061},[3055,5325,3383],{"class":3081},[3055,5327,3847],{"class":3061},[3055,5329,5330,5332,5334,5336,5338,5340,5342,5345,5347,5349,5351,5353,5355,5358,5360,5363,5365,5368,5370,5373,5375,5378,5380,5382],{"class":3057,"line":4109},[3055,5331,5238],{"class":3825},[3055,5333,3829],{"class":3061},[3055,5335,5029],{"class":3837},[3055,5337,3841],{"class":3061},[3055,5339,5247],{"class":3825},[3055,5341,3829],{"class":3061},[3055,5343,5344],{"class":3825},"price",[3055,5346,4545],{"class":3061},[3055,5348,4978],{"class":3825},[3055,5350,3829],{"class":3061},[3055,5352,5050],{"class":3825},[3055,5354,3829],{"class":3061},[3055,5356,5357],{"class":3837},"a",[3055,5359,3841],{"class":3061},[3055,5361,5362],{"class":3081},"\"number\"",[3055,5364,4545],{"class":3061},[3055,5366,5367],{"class":3825},"that",[3055,5369,3829],{"class":3061},[3055,5371,5372],{"class":3825},"is",[3055,5374,3829],{"class":3061},[3055,5376,5377],{"class":3837},"above",[3055,5379,3841],{"class":3061},[3055,5381,4754],{"class":3939},[3055,5383,3847],{"class":3061},[3055,5385,5386],{"class":3057,"line":4122},[3055,5387,4230],{"class":3061},[3055,5389,5390],{"class":3057,"line":4144},[3055,5391,4556],{"class":3061},[3055,5393,5394],{"class":3057,"line":4155},[3055,5395,3463],{"emptyLinePlaceholder":3462},[3055,5397,5398,5400,5402,5404,5406,5409,5411,5413],{"class":3057,"line":4164},[3055,5399,3826],{"class":3825},[3055,5401,3829],{"class":3061},[3055,5403,4954],{"class":3837},[3055,5405,3841],{"class":3061},[3055,5407,5408],{"class":3081},"\"Content-Type is application/json\"",[3055,5410,3110],{"class":3061},[3055,5412,4239],{"class":3814},[3055,5414,4680],{"class":3061},[3055,5416,5417,5419,5421,5423,5425,5427,5429,5431,5433,5435,5437,5439,5441,5444,5447,5449,5451,5454,5456,5459],{"class":3057,"line":4176},[3055,5418,4411],{"class":3825},[3055,5420,3829],{"class":3061},[3055,5422,5029],{"class":3837},[3055,5424,3841],{"class":3061},[3055,5426,3826],{"class":3825},[3055,5428,3829],{"class":3061},[3055,5430,4249],{"class":3825},[3055,5432,3829],{"class":3061},[3055,5434,4910],{"class":3825},[3055,5436,3829],{"class":3061},[3055,5438,3838],{"class":3837},[3055,5440,3841],{"class":3061},[3055,5442,5443],{"class":3081},"\"Content-Type\"",[3055,5445,5446],{"class":3061},")).",[3055,5448,4978],{"class":3825},[3055,5450,3829],{"class":3061},[3055,5452,5453],{"class":3837},"include",[3055,5455,3841],{"class":3061},[3055,5457,5458],{"class":3081},"\"application/json\"",[3055,5460,3847],{"class":3061},[3055,5462,5463],{"class":3057,"line":4182},[3055,5464,4556],{"class":3061},[2959,5466,5467],{},[3012,5468,5469],{},"POST /api/products — створення ресурсу:",[3045,5471,5473],{"className":3790,"code":5472,"language":3792,"meta":3051,"style":3051},"// Tests для POST /api/products\npm.test(\"Status code is 201 Created\", function() {\n    pm.response.to.have.status(201);\n});\n\npm.test(\"Location header is present\", function() {\n    pm.expect(pm.response.headers.get(\"Location\")).to.exist;\n});\n\npm.test(\"Created product has correct name\", function() {\n    const json = pm.response.json();\n    // pm.variables.get бере з будь-якого рівня (local -> environment -> collection -> global)\n    pm.expect(json.name).to.equal(pm.variables.get(\"productName\"));\n});\n\npm.test(\"ID is a valid GUID\", function() {\n    const json = pm.response.json();\n    const guidRegex = /^[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}$/i;\n    pm.expect(json.id).to.match(guidRegex);\n});\n\n// ⭐ Зберігаємо ID для подальших тестів у workflow\npm.collectionVariables.set(\"lastCreatedProductId\", pm.response.json().id);\n",[2983,5474,5475,5480,5499,5526,5530,5534,5553,5593,5597,5601,5620,5640,5645,5691,5695,5699,5718,5738,5819,5852,5856,5860,5865],{"__ignoreMap":3051},[3055,5476,5477],{"class":3057,"line":3058},[3055,5478,5479],{"class":3799},"// Tests для POST /api/products\n",[3055,5481,5482,5484,5486,5488,5490,5493,5495,5497],{"class":3057,"line":3065},[3055,5483,3826],{"class":3825},[3055,5485,3829],{"class":3061},[3055,5487,4954],{"class":3837},[3055,5489,3841],{"class":3061},[3055,5491,5492],{"class":3081},"\"Status code is 201 Created\"",[3055,5494,3110],{"class":3061},[3055,5496,4239],{"class":3814},[3055,5498,4680],{"class":3061},[3055,5500,5501,5503,5505,5507,5509,5511,5513,5515,5517,5519,5521,5524],{"class":3057,"line":3088},[3055,5502,4411],{"class":3825},[3055,5504,3829],{"class":3061},[3055,5506,4249],{"class":3825},[3055,5508,3829],{"class":3061},[3055,5510,4978],{"class":3825},[3055,5512,3829],{"class":3061},[3055,5514,4983],{"class":3825},[3055,5516,3829],{"class":3061},[3055,5518,2985],{"class":3837},[3055,5520,3841],{"class":3061},[3055,5522,5523],{"class":3939},"201",[3055,5525,3847],{"class":3061},[3055,5527,5528],{"class":3057,"line":3097},[3055,5529,4556],{"class":3061},[3055,5531,5532],{"class":3057,"line":3126},[3055,5533,3463],{"emptyLinePlaceholder":3462},[3055,5535,5536,5538,5540,5542,5544,5547,5549,5551],{"class":3057,"line":3148},[3055,5537,3826],{"class":3825},[3055,5539,3829],{"class":3061},[3055,5541,4954],{"class":3837},[3055,5543,3841],{"class":3061},[3055,5545,5546],{"class":3081},"\"Location header is present\"",[3055,5548,3110],{"class":3061},[3055,5550,4239],{"class":3814},[3055,5552,4680],{"class":3061},[3055,5554,5555,5557,5559,5561,5563,5565,5567,5569,5571,5573,5575,5577,5579,5582,5584,5586,5588,5591],{"class":3057,"line":3170},[3055,5556,4411],{"class":3825},[3055,5558,3829],{"class":3061},[3055,5560,5029],{"class":3837},[3055,5562,3841],{"class":3061},[3055,5564,3826],{"class":3825},[3055,5566,3829],{"class":3061},[3055,5568,4249],{"class":3825},[3055,5570,3829],{"class":3061},[3055,5572,4910],{"class":3825},[3055,5574,3829],{"class":3061},[3055,5576,3838],{"class":3837},[3055,5578,3841],{"class":3061},[3055,5580,5581],{"class":3081},"\"Location\"",[3055,5583,5446],{"class":3061},[3055,5585,4978],{"class":3825},[3055,5587,3829],{"class":3061},[3055,5589,5590],{"class":3825},"exist",[3055,5592,3976],{"class":3061},[3055,5594,5595],{"class":3057,"line":3193},[3055,5596,4556],{"class":3061},[3055,5598,5599],{"class":3057,"line":3199},[3055,5600,3463],{"emptyLinePlaceholder":3462},[3055,5602,5603,5605,5607,5609,5611,5614,5616,5618],{"class":3057,"line":3946},[3055,5604,3826],{"class":3825},[3055,5606,3829],{"class":3061},[3055,5608,4954],{"class":3837},[3055,5610,3841],{"class":3061},[3055,5612,5613],{"class":3081},"\"Created product has correct name\"",[3055,5615,3110],{"class":3061},[3055,5617,4239],{"class":3814},[3055,5619,4680],{"class":3061},[3055,5621,5622,5624,5626,5628,5630,5632,5634,5636,5638],{"class":3057,"line":3952},[3055,5623,4381],{"class":3814},[3055,5625,4384],{"class":3818},[3055,5627,3822],{"class":3061},[3055,5629,3826],{"class":3825},[3055,5631,3829],{"class":3061},[3055,5633,4249],{"class":3825},[3055,5635,3829],{"class":3061},[3055,5637,3050],{"class":3837},[3055,5639,3893],{"class":3061},[3055,5641,5642],{"class":3057,"line":3970},[3055,5643,5644],{"class":3799},"    // pm.variables.get бере з будь-якого рівня (local -> environment -> collection -> global)\n",[3055,5646,5647,5649,5651,5653,5655,5657,5659,5662,5664,5666,5668,5671,5673,5675,5677,5679,5681,5683,5685,5688],{"class":3057,"line":3979},[3055,5648,4411],{"class":3825},[3055,5650,3829],{"class":3061},[3055,5652,5029],{"class":3837},[3055,5654,3841],{"class":3061},[3055,5656,3050],{"class":3825},[3055,5658,3829],{"class":3061},[3055,5660,5661],{"class":3825},"name",[3055,5663,4545],{"class":3061},[3055,5665,4978],{"class":3825},[3055,5667,3829],{"class":3061},[3055,5669,5670],{"class":3837},"equal",[3055,5672,3841],{"class":3061},[3055,5674,3826],{"class":3825},[3055,5676,3829],{"class":3061},[3055,5678,4645],{"class":3825},[3055,5680,3829],{"class":3061},[3055,5682,3838],{"class":3837},[3055,5684,3841],{"class":3061},[3055,5686,5687],{"class":3081},"\"productName\"",[3055,5689,5690],{"class":3061},"));\n",[3055,5692,5693],{"class":3057,"line":3984},[3055,5694,4556],{"class":3061},[3055,5696,5697],{"class":3057,"line":3989},[3055,5698,3463],{"emptyLinePlaceholder":3462},[3055,5700,5701,5703,5705,5707,5709,5712,5714,5716],{"class":3057,"line":3995},[3055,5702,3826],{"class":3825},[3055,5704,3829],{"class":3061},[3055,5706,4954],{"class":3837},[3055,5708,3841],{"class":3061},[3055,5710,5711],{"class":3081},"\"ID is a valid GUID\"",[3055,5713,3110],{"class":3061},[3055,5715,4239],{"class":3814},[3055,5717,4680],{"class":3061},[3055,5719,5720,5722,5724,5726,5728,5730,5732,5734,5736],{"class":3057,"line":4012},[3055,5721,4381],{"class":3814},[3055,5723,4384],{"class":3818},[3055,5725,3822],{"class":3061},[3055,5727,3826],{"class":3825},[3055,5729,3829],{"class":3061},[3055,5731,4249],{"class":3825},[3055,5733,3829],{"class":3061},[3055,5735,3050],{"class":3837},[3055,5737,3893],{"class":3061},[3055,5739,5740,5742,5745,5748,5751,5755,5757,5760,5762,5766,5769,5771,5773,5775,5778,5780,5782,5784,5786,5788,5790,5792,5794,5796,5798,5800,5802,5804,5806,5809,5812,5814,5817],{"class":3057,"line":4017},[3055,5741,4381],{"class":3814},[3055,5743,5744],{"class":3818}," guidRegex",[3055,5746,5747],{"class":3061}," =",[3055,5749,5750],{"class":4697}," /",[3055,5752,5754],{"class":5753},"siofk","^",[3055,5756,4702],{"class":4701},[3055,5758,5759],{"class":4697},"0-9a-f",[3055,5761,4708],{"class":4701},[3055,5763,5765],{"class":5764},"sYsli","{8}",[3055,5767,5768],{"class":4697},"-",[3055,5770,4702],{"class":4701},[3055,5772,5759],{"class":4697},[3055,5774,4708],{"class":4701},[3055,5776,5777],{"class":5764},"{4}",[3055,5779,5768],{"class":4697},[3055,5781,4702],{"class":4701},[3055,5783,5759],{"class":4697},[3055,5785,4708],{"class":4701},[3055,5787,5777],{"class":5764},[3055,5789,5768],{"class":4697},[3055,5791,4702],{"class":4701},[3055,5793,5759],{"class":4697},[3055,5795,4708],{"class":4701},[3055,5797,5777],{"class":5764},[3055,5799,5768],{"class":4697},[3055,5801,4702],{"class":4701},[3055,5803,5759],{"class":4697},[3055,5805,4708],{"class":4701},[3055,5807,5808],{"class":5764},"{12}",[3055,5810,5811],{"class":5753},"$",[3055,5813,4698],{"class":4697},[3055,5815,5816],{"class":3814},"i",[3055,5818,3976],{"class":3061},[3055,5820,5821,5823,5825,5827,5829,5831,5833,5836,5838,5840,5842,5845,5847,5850],{"class":3057,"line":4045},[3055,5822,4411],{"class":3825},[3055,5824,3829],{"class":3061},[3055,5826,5029],{"class":3837},[3055,5828,3841],{"class":3061},[3055,5830,3050],{"class":3825},[3055,5832,3829],{"class":3061},[3055,5834,5835],{"class":3825},"id",[3055,5837,4545],{"class":3061},[3055,5839,4978],{"class":3825},[3055,5841,3829],{"class":3061},[3055,5843,5844],{"class":3837},"match",[3055,5846,3841],{"class":3061},[3055,5848,5849],{"class":3825},"guidRegex",[3055,5851,3847],{"class":3061},[3055,5853,5854],{"class":3057,"line":4072},[3055,5855,4556],{"class":3061},[3055,5857,5858],{"class":3057,"line":4098},[3055,5859,3463],{"emptyLinePlaceholder":3462},[3055,5861,5862],{"class":3057,"line":4103},[3055,5863,5864],{"class":3799},"// ⭐ Зберігаємо ID для подальших тестів у workflow\n",[3055,5866,5867,5869,5871,5873,5875,5877,5879,5882,5884,5886,5888,5890,5892,5894,5897,5899],{"class":3057,"line":4109},[3055,5868,3826],{"class":3825},[3055,5870,3829],{"class":3061},[3055,5872,3832],{"class":3825},[3055,5874,3829],{"class":3061},[3055,5876,4420],{"class":3837},[3055,5878,3841],{"class":3061},[3055,5880,5881],{"class":3081},"\"lastCreatedProductId\"",[3055,5883,3110],{"class":3061},[3055,5885,3826],{"class":3825},[3055,5887,3829],{"class":3061},[3055,5889,4249],{"class":3825},[3055,5891,3829],{"class":3061},[3055,5893,3050],{"class":3837},[3055,5895,5896],{"class":3061},"().",[3055,5898,5835],{"class":3825},[3055,5900,3847],{"class":3061},[2959,5902,5903],{},[3012,5904,5905],{},"DELETE /api/products/:id — видалення:",[3045,5907,5909],{"className":3790,"code":5908,"language":3792,"meta":3051,"style":3051},"// Tests для DELETE /api/products/:id\npm.test(\"Status code is 204 No Content\", function() {\n    pm.response.to.have.status(204);\n});\n\npm.test(\"Response body is empty\", function() {\n    pm.expect(pm.response.text()).to.be.empty;\n});\n",[2983,5910,5911,5916,5935,5962,5966,5970,5989,6025],{"__ignoreMap":3051},[3055,5912,5913],{"class":3057,"line":3058},[3055,5914,5915],{"class":3799},"// Tests для DELETE /api/products/:id\n",[3055,5917,5918,5920,5922,5924,5926,5929,5931,5933],{"class":3057,"line":3065},[3055,5919,3826],{"class":3825},[3055,5921,3829],{"class":3061},[3055,5923,4954],{"class":3837},[3055,5925,3841],{"class":3061},[3055,5927,5928],{"class":3081},"\"Status code is 204 No Content\"",[3055,5930,3110],{"class":3061},[3055,5932,4239],{"class":3814},[3055,5934,4680],{"class":3061},[3055,5936,5937,5939,5941,5943,5945,5947,5949,5951,5953,5955,5957,5960],{"class":3057,"line":3088},[3055,5938,4411],{"class":3825},[3055,5940,3829],{"class":3061},[3055,5942,4249],{"class":3825},[3055,5944,3829],{"class":3061},[3055,5946,4978],{"class":3825},[3055,5948,3829],{"class":3061},[3055,5950,4983],{"class":3825},[3055,5952,3829],{"class":3061},[3055,5954,2985],{"class":3837},[3055,5956,3841],{"class":3061},[3055,5958,5959],{"class":3939},"204",[3055,5961,3847],{"class":3061},[3055,5963,5964],{"class":3057,"line":3097},[3055,5965,4556],{"class":3061},[3055,5967,5968],{"class":3057,"line":3126},[3055,5969,3463],{"emptyLinePlaceholder":3462},[3055,5971,5972,5974,5976,5978,5980,5983,5985,5987],{"class":3057,"line":3148},[3055,5973,3826],{"class":3825},[3055,5975,3829],{"class":3061},[3055,5977,4954],{"class":3837},[3055,5979,3841],{"class":3061},[3055,5981,5982],{"class":3081},"\"Response body is empty\"",[3055,5984,3110],{"class":3061},[3055,5986,4239],{"class":3814},[3055,5988,4680],{"class":3061},[3055,5990,5991,5993,5995,5997,5999,6001,6003,6005,6007,6009,6012,6014,6016,6018,6020,6023],{"class":3057,"line":3170},[3055,5992,4411],{"class":3825},[3055,5994,3829],{"class":3061},[3055,5996,5029],{"class":3837},[3055,5998,3841],{"class":3061},[3055,6000,3826],{"class":3825},[3055,6002,3829],{"class":3061},[3055,6004,4249],{"class":3825},[3055,6006,3829],{"class":3061},[3055,6008,3409],{"class":3837},[3055,6010,6011],{"class":3061},"()).",[3055,6013,4978],{"class":3825},[3055,6015,3829],{"class":3061},[3055,6017,5050],{"class":3825},[3055,6019,3829],{"class":3061},[3055,6021,6022],{"class":3825},"empty",[3055,6024,3976],{"class":3061},[3055,6026,6027],{"class":3057,"line":3193},[3055,6028,4556],{"class":3061},[2959,6030,6031],{},[3012,6032,6033],{},"Тестування помилкових сценаріїв:",[3045,6035,6037],{"className":3790,"code":6036,"language":3792,"meta":3051,"style":3051},"// Tests для POST /api/products з невалідними даними\npm.test(\"Status code is 400 Bad Request\", function() {\n    pm.response.to.have.status(400);\n});\n\npm.test(\"Error response has 'errors' field\", function() {\n    const json = pm.response.json();\n    pm.expect(json).to.have.property(\"errors\");\n    pm.expect(json.errors).to.be.an(\"object\");\n});\n\npm.test(\"Error mentions 'Name' field\", function() {\n    const json = pm.response.json();\n    // ASP.NET повертає errors як об'єкт {\"Name\": [\"The Name field is required.\"]}\n    pm.expect(json.errors).to.have.property(\"Name\");\n});\n",[2983,6038,6039,6044,6063,6090,6094,6098,6117,6137,6168,6204,6208,6212,6231,6251,6256,6291],{"__ignoreMap":3051},[3055,6040,6041],{"class":3057,"line":3058},[3055,6042,6043],{"class":3799},"// Tests для POST /api/products з невалідними даними\n",[3055,6045,6046,6048,6050,6052,6054,6057,6059,6061],{"class":3057,"line":3065},[3055,6047,3826],{"class":3825},[3055,6049,3829],{"class":3061},[3055,6051,4954],{"class":3837},[3055,6053,3841],{"class":3061},[3055,6055,6056],{"class":3081},"\"Status code is 400 Bad Request\"",[3055,6058,3110],{"class":3061},[3055,6060,4239],{"class":3814},[3055,6062,4680],{"class":3061},[3055,6064,6065,6067,6069,6071,6073,6075,6077,6079,6081,6083,6085,6088],{"class":3057,"line":3088},[3055,6066,4411],{"class":3825},[3055,6068,3829],{"class":3061},[3055,6070,4249],{"class":3825},[3055,6072,3829],{"class":3061},[3055,6074,4978],{"class":3825},[3055,6076,3829],{"class":3061},[3055,6078,4983],{"class":3825},[3055,6080,3829],{"class":3061},[3055,6082,2985],{"class":3837},[3055,6084,3841],{"class":3061},[3055,6086,6087],{"class":3939},"400",[3055,6089,3847],{"class":3061},[3055,6091,6092],{"class":3057,"line":3097},[3055,6093,4556],{"class":3061},[3055,6095,6096],{"class":3057,"line":3126},[3055,6097,3463],{"emptyLinePlaceholder":3462},[3055,6099,6100,6102,6104,6106,6108,6111,6113,6115],{"class":3057,"line":3148},[3055,6101,3826],{"class":3825},[3055,6103,3829],{"class":3061},[3055,6105,4954],{"class":3837},[3055,6107,3841],{"class":3061},[3055,6109,6110],{"class":3081},"\"Error response has 'errors' field\"",[3055,6112,3110],{"class":3061},[3055,6114,4239],{"class":3814},[3055,6116,4680],{"class":3061},[3055,6118,6119,6121,6123,6125,6127,6129,6131,6133,6135],{"class":3057,"line":3170},[3055,6120,4381],{"class":3814},[3055,6122,4384],{"class":3818},[3055,6124,3822],{"class":3061},[3055,6126,3826],{"class":3825},[3055,6128,3829],{"class":3061},[3055,6130,4249],{"class":3825},[3055,6132,3829],{"class":3061},[3055,6134,3050],{"class":3837},[3055,6136,3893],{"class":3061},[3055,6138,6139,6141,6143,6145,6147,6149,6151,6153,6155,6157,6159,6161,6163,6166],{"class":3057,"line":3193},[3055,6140,4411],{"class":3825},[3055,6142,3829],{"class":3061},[3055,6144,5029],{"class":3837},[3055,6146,3841],{"class":3061},[3055,6148,3050],{"class":3825},[3055,6150,4545],{"class":3061},[3055,6152,4978],{"class":3825},[3055,6154,3829],{"class":3061},[3055,6156,4983],{"class":3825},[3055,6158,3829],{"class":3061},[3055,6160,5260],{"class":3837},[3055,6162,3841],{"class":3061},[3055,6164,6165],{"class":3081},"\"errors\"",[3055,6167,3847],{"class":3061},[3055,6169,6170,6172,6174,6176,6178,6180,6182,6185,6187,6189,6191,6193,6195,6197,6199,6202],{"class":3057,"line":3199},[3055,6171,4411],{"class":3825},[3055,6173,3829],{"class":3061},[3055,6175,5029],{"class":3837},[3055,6177,3841],{"class":3061},[3055,6179,3050],{"class":3825},[3055,6181,3829],{"class":3061},[3055,6183,6184],{"class":3825},"errors",[3055,6186,4545],{"class":3061},[3055,6188,4978],{"class":3825},[3055,6190,3829],{"class":3061},[3055,6192,5050],{"class":3825},[3055,6194,3829],{"class":3061},[3055,6196,5134],{"class":3837},[3055,6198,3841],{"class":3061},[3055,6200,6201],{"class":3081},"\"object\"",[3055,6203,3847],{"class":3061},[3055,6205,6206],{"class":3057,"line":3946},[3055,6207,4556],{"class":3061},[3055,6209,6210],{"class":3057,"line":3952},[3055,6211,3463],{"emptyLinePlaceholder":3462},[3055,6213,6214,6216,6218,6220,6222,6225,6227,6229],{"class":3057,"line":3970},[3055,6215,3826],{"class":3825},[3055,6217,3829],{"class":3061},[3055,6219,4954],{"class":3837},[3055,6221,3841],{"class":3061},[3055,6223,6224],{"class":3081},"\"Error mentions 'Name' field\"",[3055,6226,3110],{"class":3061},[3055,6228,4239],{"class":3814},[3055,6230,4680],{"class":3061},[3055,6232,6233,6235,6237,6239,6241,6243,6245,6247,6249],{"class":3057,"line":3979},[3055,6234,4381],{"class":3814},[3055,6236,4384],{"class":3818},[3055,6238,3822],{"class":3061},[3055,6240,3826],{"class":3825},[3055,6242,3829],{"class":3061},[3055,6244,4249],{"class":3825},[3055,6246,3829],{"class":3061},[3055,6248,3050],{"class":3837},[3055,6250,3893],{"class":3061},[3055,6252,6253],{"class":3057,"line":3984},[3055,6254,6255],{"class":3799},"    // ASP.NET повертає errors як об'єкт {\"Name\": [\"The Name field is required.\"]}\n",[3055,6257,6258,6260,6262,6264,6266,6268,6270,6272,6274,6276,6278,6280,6282,6284,6286,6289],{"class":3057,"line":3989},[3055,6259,4411],{"class":3825},[3055,6261,3829],{"class":3061},[3055,6263,5029],{"class":3837},[3055,6265,3841],{"class":3061},[3055,6267,3050],{"class":3825},[3055,6269,3829],{"class":3061},[3055,6271,6184],{"class":3825},[3055,6273,4545],{"class":3061},[3055,6275,4978],{"class":3825},[3055,6277,3829],{"class":3061},[3055,6279,4983],{"class":3825},[3055,6281,3829],{"class":3061},[3055,6283,5260],{"class":3837},[3055,6285,3841],{"class":3061},[3055,6287,6288],{"class":3081},"\"Name\"",[3055,6290,3847],{"class":3061},[3055,6292,6293],{"class":3057,"line":3995},[3055,6294,4556],{"class":3061},[2966,6296,6298],{"id":6297},"тестування-workflow-передача-даних-між-запитами","Тестування Workflow: Передача Даних між Запитами",[2959,6300,6301],{},"Один з найпотужніших сценаріїв Postman — тестування бізнес-процесів, де результат одного запиту є вхідними даними для наступного.",[3002,6303,6305],{"id":6304},"приклад-lifecycle-тесту-продукту","Приклад: Lifecycle тесту продукту",[2959,6307,6308],{},"Уявіть, що хочемо перевірити повний цикл: створити продукт → отримати його → оновити → видалити. Кожен крок залежить від попереднього.",[6310,6311,6312,6316,6322,6391,6398,6447,6452,6550,6554,6560,6565,6696,6700,6704,6710,6791,6795,6798,6919,6923,6969,6973,6976],"steps",{},[3002,6313,6315],{"id":6314},"крок-1-post-apiproducts-create","Крок 1: POST /api/products (Create)",[2959,6317,6318,6319,6321],{},"У вкладці ",[3012,6320,3754],{}," генеруємо тестові дані:",[3045,6323,6325],{"className":3790,"code":6324,"language":3792,"meta":3051,"style":3051},"pm.variables.set(\"productName\", `Test Product ${Date.now()}`);\npm.variables.set(\"productPrice\", 99.99);\n",[2983,6326,6327,6367],{"__ignoreMap":3051},[3055,6328,6329,6331,6333,6335,6337,6339,6341,6343,6345,6348,6350,6352,6355,6357,6360,6362,6365],{"class":3057,"line":3058},[3055,6330,3826],{"class":3825},[3055,6332,3829],{"class":3061},[3055,6334,4645],{"class":3825},[3055,6336,3829],{"class":3061},[3055,6338,4420],{"class":3837},[3055,6340,3841],{"class":3061},[3055,6342,5687],{"class":3081},[3055,6344,3110],{"class":3061},[3055,6346,6347],{"class":3081},"`Test Product ",[3055,6349,4131],{"class":3814},[3055,6351,3885],{"class":3825},[3055,6353,3829],{"class":6354},"sD7JJ",[3055,6356,3890],{"class":3837},[3055,6358,6359],{"class":6354},"()",[3055,6361,4136],{"class":3814},[3055,6363,6364],{"class":3081},"`",[3055,6366,3847],{"class":3061},[3055,6368,6369,6371,6373,6375,6377,6379,6381,6384,6386,6389],{"class":3057,"line":3065},[3055,6370,3826],{"class":3825},[3055,6372,3829],{"class":3061},[3055,6374,4645],{"class":3825},[3055,6376,3829],{"class":3061},[3055,6378,4420],{"class":3837},[3055,6380,3841],{"class":3061},[3055,6382,6383],{"class":3081},"\"productPrice\"",[3055,6385,3110],{"class":3061},[3055,6387,6388],{"class":3939},"99.99",[3055,6390,3847],{"class":3061},[2959,6392,6393,6394,6397],{},"У ",[3012,6395,6396],{},"Body"," запиту:",[3045,6399,6401],{"className":3047,"code":6400,"language":3050,"meta":3051,"style":3051},"{\n    \"name\": \"{{productName}}\",\n    \"price\": {{productPrice}},\n    \"categoryId\": \"{{validCategoryId}}\"\n}\n",[2983,6402,6403,6407,6419,6433,6443],{"__ignoreMap":3051},[3055,6404,6405],{"class":3057,"line":3058},[3055,6406,3062],{"class":3061},[3055,6408,6409,6412,6414,6417],{"class":3057,"line":3065},[3055,6410,6411],{"class":3068},"    \"name\"",[3055,6413,3078],{"class":3061},[3055,6415,6416],{"class":3081},"\"{{productName}}\"",[3055,6418,3225],{"class":3061},[3055,6420,6421,6424,6427,6430],{"class":3057,"line":3088},[3055,6422,6423],{"class":3068},"    \"price\"",[3055,6425,6426],{"class":3061},": {",[3055,6428,6429],{"class":3119},"{productPrice",[3055,6431,6432],{"class":3061},"}},\n",[3055,6434,6435,6438,6440],{"class":3057,"line":3097},[3055,6436,6437],{"class":3081},"    \"categoryId\"",[3055,6439,3078],{"class":3061},[3055,6441,6442],{"class":3081},"\"{{validCategoryId}}\"\n",[3055,6444,6445],{"class":3057,"line":3126},[3055,6446,3202],{"class":3061},[2959,6448,6393,6449,6451],{},[3012,6450,4856],{}," зберігаємо ID:",[3045,6453,6455],{"className":3790,"code":6454,"language":3792,"meta":3051,"style":3051},"pm.test(\"Product created\", () => pm.response.to.have.status(201));\nconst productId = pm.response.json().id;\npm.collectionVariables.set(\"workflowProductId\", productId);\n",[2983,6456,6457,6501,6526],{"__ignoreMap":3051},[3055,6458,6459,6461,6463,6465,6467,6470,6473,6476,6479,6481,6483,6485,6487,6489,6491,6493,6495,6497,6499],{"class":3057,"line":3058},[3055,6460,3826],{"class":3825},[3055,6462,3829],{"class":3061},[3055,6464,4954],{"class":3837},[3055,6466,3841],{"class":3061},[3055,6468,6469],{"class":3081},"\"Product created\"",[3055,6471,6472],{"class":3061},", () ",[3055,6474,6475],{"class":3814},"=>",[3055,6477,6478],{"class":3825}," pm",[3055,6480,3829],{"class":3061},[3055,6482,4249],{"class":3825},[3055,6484,3829],{"class":3061},[3055,6486,4978],{"class":3825},[3055,6488,3829],{"class":3061},[3055,6490,4983],{"class":3825},[3055,6492,3829],{"class":3061},[3055,6494,2985],{"class":3837},[3055,6496,3841],{"class":3061},[3055,6498,5523],{"class":3939},[3055,6500,5690],{"class":3061},[3055,6502,6503,6505,6508,6510,6512,6514,6516,6518,6520,6522,6524],{"class":3057,"line":3065},[3055,6504,3815],{"class":3814},[3055,6506,6507],{"class":3818}," productId",[3055,6509,3822],{"class":3061},[3055,6511,3826],{"class":3825},[3055,6513,3829],{"class":3061},[3055,6515,4249],{"class":3825},[3055,6517,3829],{"class":3061},[3055,6519,3050],{"class":3837},[3055,6521,5896],{"class":3061},[3055,6523,5835],{"class":3825},[3055,6525,3976],{"class":3061},[3055,6527,6528,6530,6532,6534,6536,6538,6540,6543,6545,6548],{"class":3057,"line":3088},[3055,6529,3826],{"class":3825},[3055,6531,3829],{"class":3061},[3055,6533,3832],{"class":3825},[3055,6535,3829],{"class":3061},[3055,6537,4420],{"class":3837},[3055,6539,3841],{"class":3061},[3055,6541,6542],{"class":3081},"\"workflowProductId\"",[3055,6544,3110],{"class":3061},[3055,6546,6547],{"class":3825},"productId",[3055,6549,3847],{"class":3061},[3002,6551,6553],{"id":6552},"крок-2-get-apiproductsid-read","Крок 2: GET /api/products/:id (Read)",[2959,6555,6556,6557],{},"URL: ",[2983,6558,6559],{},"{{baseUrl}}/api/products/{{workflowProductId}}",[2959,6561,6393,6562,6564],{},[3012,6563,4856],{}," перевіряємо коректність:",[3045,6566,6568],{"className":3790,"code":6567,"language":3792,"meta":3051,"style":3051},"pm.test(\"Product retrieved successfully\", () => pm.response.to.have.status(200));\npm.test(\"Product name matches\", function() {\n    const json = pm.response.json();\n    pm.expect(json.name).to.equal(pm.collectionVariables.get(\"productName\"));\n});\n",[2983,6569,6570,6611,6630,6650,6692],{"__ignoreMap":3051},[3055,6571,6572,6574,6576,6578,6580,6583,6585,6587,6589,6591,6593,6595,6597,6599,6601,6603,6605,6607,6609],{"class":3057,"line":3058},[3055,6573,3826],{"class":3825},[3055,6575,3829],{"class":3061},[3055,6577,4954],{"class":3837},[3055,6579,3841],{"class":3061},[3055,6581,6582],{"class":3081},"\"Product retrieved successfully\"",[3055,6584,6472],{"class":3061},[3055,6586,6475],{"class":3814},[3055,6588,6478],{"class":3825},[3055,6590,3829],{"class":3061},[3055,6592,4249],{"class":3825},[3055,6594,3829],{"class":3061},[3055,6596,4978],{"class":3825},[3055,6598,3829],{"class":3061},[3055,6600,4983],{"class":3825},[3055,6602,3829],{"class":3061},[3055,6604,2985],{"class":3837},[3055,6606,3841],{"class":3061},[3055,6608,4323],{"class":3939},[3055,6610,5690],{"class":3061},[3055,6612,6613,6615,6617,6619,6621,6624,6626,6628],{"class":3057,"line":3065},[3055,6614,3826],{"class":3825},[3055,6616,3829],{"class":3061},[3055,6618,4954],{"class":3837},[3055,6620,3841],{"class":3061},[3055,6622,6623],{"class":3081},"\"Product name matches\"",[3055,6625,3110],{"class":3061},[3055,6627,4239],{"class":3814},[3055,6629,4680],{"class":3061},[3055,6631,6632,6634,6636,6638,6640,6642,6644,6646,6648],{"class":3057,"line":3088},[3055,6633,4381],{"class":3814},[3055,6635,4384],{"class":3818},[3055,6637,3822],{"class":3061},[3055,6639,3826],{"class":3825},[3055,6641,3829],{"class":3061},[3055,6643,4249],{"class":3825},[3055,6645,3829],{"class":3061},[3055,6647,3050],{"class":3837},[3055,6649,3893],{"class":3061},[3055,6651,6652,6654,6656,6658,6660,6662,6664,6666,6668,6670,6672,6674,6676,6678,6680,6682,6684,6686,6688,6690],{"class":3057,"line":3097},[3055,6653,4411],{"class":3825},[3055,6655,3829],{"class":3061},[3055,6657,5029],{"class":3837},[3055,6659,3841],{"class":3061},[3055,6661,3050],{"class":3825},[3055,6663,3829],{"class":3061},[3055,6665,5661],{"class":3825},[3055,6667,4545],{"class":3061},[3055,6669,4978],{"class":3825},[3055,6671,3829],{"class":3061},[3055,6673,5670],{"class":3837},[3055,6675,3841],{"class":3061},[3055,6677,3826],{"class":3825},[3055,6679,3829],{"class":3061},[3055,6681,3832],{"class":3825},[3055,6683,3829],{"class":3061},[3055,6685,3838],{"class":3837},[3055,6687,3841],{"class":3061},[3055,6689,5687],{"class":3081},[3055,6691,5690],{"class":3061},[3055,6693,6694],{"class":3057,"line":3126},[3055,6695,4556],{"class":3061},[3002,6697,6699],{"id":6698},"крок-3-put-apiproductsid-update","Крок 3: PUT /api/products/:id (Update)",[2959,6701,6556,6702],{},[2983,6703,6559],{},[2959,6705,6706,6707],{},"Body: ",[2983,6708,6709],{},"{\"name\": \"Updated {{productName}}\", \"price\": 149.99}",[3045,6711,6713],{"className":3790,"code":6712,"language":3792,"meta":3051,"style":3051},"pm.test(\"Product updated\", () => pm.response.to.have.status(200));\npm.collectionVariables.set(\"updatedProductName\", pm.response.json().name);\n",[2983,6714,6715,6756],{"__ignoreMap":3051},[3055,6716,6717,6719,6721,6723,6725,6728,6730,6732,6734,6736,6738,6740,6742,6744,6746,6748,6750,6752,6754],{"class":3057,"line":3058},[3055,6718,3826],{"class":3825},[3055,6720,3829],{"class":3061},[3055,6722,4954],{"class":3837},[3055,6724,3841],{"class":3061},[3055,6726,6727],{"class":3081},"\"Product updated\"",[3055,6729,6472],{"class":3061},[3055,6731,6475],{"class":3814},[3055,6733,6478],{"class":3825},[3055,6735,3829],{"class":3061},[3055,6737,4249],{"class":3825},[3055,6739,3829],{"class":3061},[3055,6741,4978],{"class":3825},[3055,6743,3829],{"class":3061},[3055,6745,4983],{"class":3825},[3055,6747,3829],{"class":3061},[3055,6749,2985],{"class":3837},[3055,6751,3841],{"class":3061},[3055,6753,4323],{"class":3939},[3055,6755,5690],{"class":3061},[3055,6757,6758,6760,6762,6764,6766,6768,6770,6773,6775,6777,6779,6781,6783,6785,6787,6789],{"class":3057,"line":3065},[3055,6759,3826],{"class":3825},[3055,6761,3829],{"class":3061},[3055,6763,3832],{"class":3825},[3055,6765,3829],{"class":3061},[3055,6767,4420],{"class":3837},[3055,6769,3841],{"class":3061},[3055,6771,6772],{"class":3081},"\"updatedProductName\"",[3055,6774,3110],{"class":3061},[3055,6776,3826],{"class":3825},[3055,6778,3829],{"class":3061},[3055,6780,4249],{"class":3825},[3055,6782,3829],{"class":3061},[3055,6784,3050],{"class":3837},[3055,6786,5896],{"class":3061},[3055,6788,5661],{"class":3825},[3055,6790,3847],{"class":3061},[3002,6792,6794],{"id":6793},"крок-4-get-apiproductsid-verify-update","Крок 4: GET /api/products/:id (Verify Update)",[2959,6796,6797],{},"Перевіряємо, що оновлення справді збереглося:",[3045,6799,6801],{"className":3790,"code":6800,"language":3792,"meta":3051,"style":3051},"pm.test(\"Updated name persisted\", function() {\n    const json = pm.response.json();\n    pm.expect(json.name).to.equal(pm.collectionVariables.get(\"updatedProductName\"));\n    pm.expect(json.price).to.equal(149.99);\n});\n",[2983,6802,6803,6822,6842,6884,6915],{"__ignoreMap":3051},[3055,6804,6805,6807,6809,6811,6813,6816,6818,6820],{"class":3057,"line":3058},[3055,6806,3826],{"class":3825},[3055,6808,3829],{"class":3061},[3055,6810,4954],{"class":3837},[3055,6812,3841],{"class":3061},[3055,6814,6815],{"class":3081},"\"Updated name persisted\"",[3055,6817,3110],{"class":3061},[3055,6819,4239],{"class":3814},[3055,6821,4680],{"class":3061},[3055,6823,6824,6826,6828,6830,6832,6834,6836,6838,6840],{"class":3057,"line":3065},[3055,6825,4381],{"class":3814},[3055,6827,4384],{"class":3818},[3055,6829,3822],{"class":3061},[3055,6831,3826],{"class":3825},[3055,6833,3829],{"class":3061},[3055,6835,4249],{"class":3825},[3055,6837,3829],{"class":3061},[3055,6839,3050],{"class":3837},[3055,6841,3893],{"class":3061},[3055,6843,6844,6846,6848,6850,6852,6854,6856,6858,6860,6862,6864,6866,6868,6870,6872,6874,6876,6878,6880,6882],{"class":3057,"line":3088},[3055,6845,4411],{"class":3825},[3055,6847,3829],{"class":3061},[3055,6849,5029],{"class":3837},[3055,6851,3841],{"class":3061},[3055,6853,3050],{"class":3825},[3055,6855,3829],{"class":3061},[3055,6857,5661],{"class":3825},[3055,6859,4545],{"class":3061},[3055,6861,4978],{"class":3825},[3055,6863,3829],{"class":3061},[3055,6865,5670],{"class":3837},[3055,6867,3841],{"class":3061},[3055,6869,3826],{"class":3825},[3055,6871,3829],{"class":3061},[3055,6873,3832],{"class":3825},[3055,6875,3829],{"class":3061},[3055,6877,3838],{"class":3837},[3055,6879,3841],{"class":3061},[3055,6881,6772],{"class":3081},[3055,6883,5690],{"class":3061},[3055,6885,6886,6888,6890,6892,6894,6896,6898,6900,6902,6904,6906,6908,6910,6913],{"class":3057,"line":3097},[3055,6887,4411],{"class":3825},[3055,6889,3829],{"class":3061},[3055,6891,5029],{"class":3837},[3055,6893,3841],{"class":3061},[3055,6895,3050],{"class":3825},[3055,6897,3829],{"class":3061},[3055,6899,5344],{"class":3825},[3055,6901,4545],{"class":3061},[3055,6903,4978],{"class":3825},[3055,6905,3829],{"class":3061},[3055,6907,5670],{"class":3837},[3055,6909,3841],{"class":3061},[3055,6911,6912],{"class":3939},"149.99",[3055,6914,3847],{"class":3061},[3055,6916,6917],{"class":3057,"line":3126},[3055,6918,4556],{"class":3061},[3002,6920,6922],{"id":6921},"крок-5-delete-apiproductsid-delete","Крок 5: DELETE /api/products/:id (Delete)",[3045,6924,6926],{"className":3790,"code":6925,"language":3792,"meta":3051,"style":3051},"pm.test(\"Product deleted\", () => pm.response.to.have.status(204));\n",[2983,6927,6928],{"__ignoreMap":3051},[3055,6929,6930,6932,6934,6936,6938,6941,6943,6945,6947,6949,6951,6953,6955,6957,6959,6961,6963,6965,6967],{"class":3057,"line":3058},[3055,6931,3826],{"class":3825},[3055,6933,3829],{"class":3061},[3055,6935,4954],{"class":3837},[3055,6937,3841],{"class":3061},[3055,6939,6940],{"class":3081},"\"Product deleted\"",[3055,6942,6472],{"class":3061},[3055,6944,6475],{"class":3814},[3055,6946,6478],{"class":3825},[3055,6948,3829],{"class":3061},[3055,6950,4249],{"class":3825},[3055,6952,3829],{"class":3061},[3055,6954,4978],{"class":3825},[3055,6956,3829],{"class":3061},[3055,6958,4983],{"class":3825},[3055,6960,3829],{"class":3061},[3055,6962,2985],{"class":3837},[3055,6964,3841],{"class":3061},[3055,6966,5959],{"class":3939},[3055,6968,5690],{"class":3061},[3002,6970,6972],{"id":6971},"крок-6-get-apiproductsid-verify-deletion","Крок 6: GET /api/products/:id (Verify Deletion)",[2959,6974,6975],{},"Переконуємося, що ресурс більше не існує:",[3045,6977,6979],{"className":3790,"code":6978,"language":3792,"meta":3051,"style":3051},"pm.test(\"Product no longer exists\", () => pm.response.to.have.status(404));\n",[2983,6980,6981],{"__ignoreMap":3051},[3055,6982,6983,6985,6987,6989,6991,6994,6996,6998,7000,7002,7004,7006,7008,7010,7012,7014,7016,7018,7021],{"class":3057,"line":3058},[3055,6984,3826],{"class":3825},[3055,6986,3829],{"class":3061},[3055,6988,4954],{"class":3837},[3055,6990,3841],{"class":3061},[3055,6992,6993],{"class":3081},"\"Product no longer exists\"",[3055,6995,6472],{"class":3061},[3055,6997,6475],{"class":3814},[3055,6999,6478],{"class":3825},[3055,7001,3829],{"class":3061},[3055,7003,4249],{"class":3825},[3055,7005,3829],{"class":3061},[3055,7007,4978],{"class":3825},[3055,7009,3829],{"class":3061},[3055,7011,4983],{"class":3825},[3055,7013,3829],{"class":3061},[3055,7015,2985],{"class":3837},[3055,7017,3841],{"class":3061},[3055,7019,7020],{"class":3939},"404",[3055,7022,5690],{"class":3061},[2959,7024,7025,7026,7029],{},"Запустивши всю папку «Product Lifecycle» через ",[3012,7027,7028],{},"Collection Runner"," (кнопка ▶ Run), Postman виконає всі 6 запитів послідовно, і після кожного — перевірить assertions. Ви побачите зведений звіт: скільки тестів пройшло, скільки провалилося.",[2966,7031,7033],{"id":7032},"collection-runner-та-newman-від-ui-до-cli","Collection Runner та Newman: Від UI до CLI",[3002,7035,7037],{"id":7036},"collection-runner-у-postman-ui","Collection Runner у Postman UI",[2959,7039,7040,7042],{},[3012,7041,7028],{}," — вбудований інструмент Postman для запуску всієї колекції або папки. Доступний через кнопку «▶ Run» у правому верхньому куті колекції.",[2959,7044,7045],{},"Ключові параметри запуску:",[4873,7047,7048,7054,7060,7066],{},[4876,7049,7050,7053],{},[3012,7051,7052],{},"Environment",": яке середовище використовувати",[4876,7055,7056,7059],{},[3012,7057,7058],{},"Iterations",": скільки разів запустити кожен запит (корисно для навантажувального тестування)",[4876,7061,7062,7065],{},[3012,7063,7064],{},"Delay",": затримка між запитами у мілісекундах",[4876,7067,7068,7071],{},[3012,7069,7070],{},"Data file",": CSV або JSON файл з даними для параметризованих тестів",[3002,7073,7075],{"id":7074},"data-driven-testing-з-файлами","Data-driven Testing з файлами",[2959,7077,7078,7079,4170],{},"Ця функція дозволяє запустити один запит з різними наборами даних. Створіть файл ",[2983,7080,7081],{},"test-data.csv",[3045,7083,7087],{"className":7084,"code":7085,"language":7086,"meta":3051,"style":3051},"language-csv shiki shiki-themes light-plus dark-plus dark-plus","productName,productPrice,expectedStatus\n\"Valid Product\",99.99,201\n\"Another Product\",0.01,201\n\"\",99.99,400\n\"Valid Name\",-1,400\n\"A very long product name that exceeds the maximum allowed characters limit in our validation\",99.99,400\n","csv",[2983,7088,7089,7094,7099,7104,7109,7114],{"__ignoreMap":3051},[3055,7090,7091],{"class":3057,"line":3058},[3055,7092,7093],{},"productName,productPrice,expectedStatus\n",[3055,7095,7096],{"class":3057,"line":3065},[3055,7097,7098],{},"\"Valid Product\",99.99,201\n",[3055,7100,7101],{"class":3057,"line":3088},[3055,7102,7103],{},"\"Another Product\",0.01,201\n",[3055,7105,7106],{"class":3057,"line":3097},[3055,7107,7108],{},"\"\",99.99,400\n",[3055,7110,7111],{"class":3057,"line":3126},[3055,7112,7113],{},"\"Valid Name\",-1,400\n",[3055,7115,7116],{"class":3057,"line":3148},[3055,7117,7118],{},"\"A very long product name that exceeds the maximum allowed characters limit in our validation\",99.99,400\n",[2959,7120,7121],{},"У Collection Runner вкажіть цей файл як Data file. Postman виконає запит 5 разів, кожного разу з іншими даними з рядка CSV. У тесті доступ до поточного рядку:",[3045,7123,7125],{"className":3790,"code":7124,"language":3792,"meta":3051,"style":3051},"pm.test(`Status is ${pm.iterationData.get(\"expectedStatus\")}`, function() {\n    pm.response.to.have.status(parseInt(pm.iterationData.get(\"expectedStatus\")));\n});\n",[2983,7126,7127,7171,7214],{"__ignoreMap":3051},[3055,7128,7129,7131,7133,7135,7137,7140,7142,7144,7146,7149,7151,7153,7155,7158,7161,7163,7165,7167,7169],{"class":3057,"line":3058},[3055,7130,3826],{"class":3825},[3055,7132,3829],{"class":3061},[3055,7134,4954],{"class":3837},[3055,7136,3841],{"class":3061},[3055,7138,7139],{"class":3081},"`Status is ",[3055,7141,4131],{"class":3814},[3055,7143,3826],{"class":3825},[3055,7145,3829],{"class":6354},[3055,7147,7148],{"class":3825},"iterationData",[3055,7150,3829],{"class":6354},[3055,7152,3838],{"class":3837},[3055,7154,3841],{"class":6354},[3055,7156,7157],{"class":3081},"\"expectedStatus\"",[3055,7159,7160],{"class":6354},")",[3055,7162,4136],{"class":3814},[3055,7164,6364],{"class":3081},[3055,7166,3110],{"class":3061},[3055,7168,4239],{"class":3814},[3055,7170,4680],{"class":3061},[3055,7172,7173,7175,7177,7179,7181,7183,7185,7187,7189,7191,7193,7195,7197,7199,7201,7203,7205,7207,7209,7211],{"class":3057,"line":3065},[3055,7174,4411],{"class":3825},[3055,7176,3829],{"class":3061},[3055,7178,4249],{"class":3825},[3055,7180,3829],{"class":3061},[3055,7182,4978],{"class":3825},[3055,7184,3829],{"class":3061},[3055,7186,4983],{"class":3825},[3055,7188,3829],{"class":3061},[3055,7190,2985],{"class":3837},[3055,7192,3841],{"class":3061},[3055,7194,3929],{"class":3837},[3055,7196,3841],{"class":3061},[3055,7198,3826],{"class":3825},[3055,7200,3829],{"class":3061},[3055,7202,7148],{"class":3825},[3055,7204,3829],{"class":3061},[3055,7206,3838],{"class":3837},[3055,7208,3841],{"class":3061},[3055,7210,7157],{"class":3081},[3055,7212,7213],{"class":3061},")));\n",[3055,7215,7216],{"class":3057,"line":3088},[3055,7217,4556],{"class":3061},[3002,7219,7221],{"id":7220},"newman-postman-у-командному-рядку","Newman: Postman у Командному Рядку",[2959,7223,7224],{},"Newman — офіційний CLI-runner для Postman колекцій. Це Node.js-бібліотека, яка дозволяє запускати колекції поза UI: у скриптах, у CI/CD, у Docker-контейнерах.",[6310,7226,7227,7231,7276,7280,7286,7292,7296,7344,7348],{},[3002,7228,7230],{"id":7229},"крок-1-встановлення-newman","Крок 1: Встановлення Newman",[7232,7233,7236,7247,7257,7260,7269],"terminal-preview",{":cursor":7234,"title":7235},"false","npm install -g newman",[7237,7238,7240,7244,7245],"div",{"className":7239},[3057],[3055,7241,5811],{"className":7242},[7243],"opacity-40"," ",[3012,7246,7235],{},[7237,7248,7250,7256],{"className":7249},[3057],[3055,7251,7255],{"className":7252},[7253,7254],"text-green-400","font-bold","✓"," newman@6.x.x installed globally",[7237,7258],{"className":7259},[3057],[7237,7261,7263,7244,7266],{"className":7262},[3057],[3055,7264,5811],{"className":7265},[7243],[3012,7267,7268],{},"npm install --save-dev newman newman-reporter-htmlextra",[7237,7270,7272,7275],{"className":7271},[3057],[3055,7273,7255],{"className":7274},[7253,7254]," Added 2 packages to devDependencies",[3002,7277,7279],{"id":7278},"крок-2-експорт-колекції-та-середовища","Крок 2: Експорт колекції та середовища",[2959,7281,7282,7283,3829],{},"У Postman: Права кнопка на колекції → Export → Collection v2.1 → збережіть як ",[2983,7284,7285],{},"collection.json",[2959,7287,7288,7289,3829],{},"Для середовища: Environments → виберіть середовище → Export → збережіть як ",[2983,7290,7291],{},"development.json",[3002,7293,7295],{"id":7294},"крок-3-базовий-запуск","Крок 3: Базовий запуск",[7232,7297,7299,7308,7315,7322,7330,7337],{":cursor":7234,"title":7298},"newman run",[7237,7300,7302,7244,7305],{"className":7301},[3057],[3055,7303,5811],{"className":7304},[7243],[3012,7306,7307],{},"newman run collection.json --environment development.json --reporters cli,junit",[7237,7309,7311,7314],{"className":7310},[3057],[3055,7312,7255],{"className":7313},[7253,7254]," Collection iterations: 1",[7237,7316,7318,7321],{"className":7317},[3057],[3055,7319,7255],{"className":7320},[7253,7254]," Requests: 15, Passed: 15, Failed: 0",[7237,7323,7325,7329],{"className":7324},[3057],[3055,7326,7255],{"className":7327},[7328,7254],"text-blue-400"," GET /api/health - 200 OK [245ms]",[7237,7331,7333,7336],{"className":7332},[3057],[3055,7334,7255],{"className":7335},[7328,7254]," POST /api/users - 201 Created [312ms]",[7237,7338,7340,7343],{"className":7339},[3057],[3055,7341,7255],{"className":7342},[7253,7254]," JUnit report saved to results/report.xml",[3002,7345,7347],{"id":7346},"крок-4-html-звіт","Крок 4: HTML-звіт",[7232,7349,7351,7360,7366,7372,7379],{":cursor":7234,"title":7350},"newman run --reporters htmlextra",[7237,7352,7354,7244,7357],{"className":7353},[3057],[3055,7355,5811],{"className":7356},[7243],[3012,7358,7359],{},"newman run collection.json --environment development.json --reporters cli,htmlextra",[7237,7361,7363,7314],{"className":7362},[3057],[3055,7364,7255],{"className":7365},[7253,7254],[7237,7367,7369,7321],{"className":7368},[3057],[3055,7370,7255],{"className":7371},[7253,7254],[7237,7373,7375,7378],{"className":7374},[3057],[3055,7376,7255],{"className":7377},[7253,7254]," HTML report: results/report.html",[7237,7380,7382,7385],{"className":7381},[3057],[3055,7383,7255],{"className":7384},[7253,7254]," Report title: MyApp API Test Report",[7232,7387,7388,7398,7401,7408,7411,7419,7427,7434,7437,7444,7451,7458,7465,7468,7475,7481,7488,7500,7503,7507,7511,7515,7519,7523,7527,7531,7535,7538,7548,7557],{":cursor":7234,"title":7298},[7237,7389,7391,7244,7394],{"className":7390},[3057],[3055,7392,5811],{"className":7393},[7243],[3012,7395,7397],{"className":7396},[7254],"newman run collection.json --environment development.json",[7237,7399],{"className":7400},[3057],[7237,7402,7404],{"className":7403},[3057],[3055,7405,7407],{"className":7406},[7328,7254],"MyApp API Collection",[7237,7409],{"className":7410},[3057],[7237,7412,7414,7418],{"className":7413},[3057],[3055,7415,7417],{"className":7416},[7243],"→"," Auth / POST Login",[7237,7420,7422,7423,7426],{"className":7421},[3057],"  ",[3055,7424,7255],{"className":7425},[7253],"  Status code is 200",[7237,7428,7422,7430,7433],{"className":7429},[3057],[3055,7431,7255],{"className":7432},[7253],"  Token received",[7237,7435],{"className":7436},[3057],[7237,7438,7440,7443],{"className":7439},[3057],[3055,7441,7417],{"className":7442},[7243]," Products / POST Create Product",[7237,7445,7422,7447,7450],{"className":7446},[3057],[3055,7448,7255],{"className":7449},[7253],"  Status code is 201 Created",[7237,7452,7422,7454,7457],{"className":7453},[3057],[3055,7455,7255],{"className":7456},[7253],"  Location header is present",[7237,7459,7422,7461,7464],{"className":7460},[3057],[3055,7462,7255],{"className":7463},[7253],"  ID is a valid GUID",[7237,7466],{"className":7467},[3057],[7237,7469,7471,7474],{"className":7470},[3057],[3055,7472,7417],{"className":7473},[7243]," Products / GET Product by ID",[7237,7476,7422,7478,7426],{"className":7477},[3057],[3055,7479,7255],{"className":7480},[7253],[7237,7482,7422,7484,7487],{"className":7483},[3057],[3055,7485,7255],{"className":7486},[7253],"  Product name matches",[7237,7489,7422,7491,7422,7496],{"className":7490},[3057],[3055,7492,7495],{"className":7493},[7494],"text-rose-400","✗",[3055,7497,7499],{"className":7498},[7494],"Response time is less than 500ms | AssertionError: expected 723 to be below 500",[7237,7501],{"className":7502},[3057],[7237,7504,7506],{"className":7505},[3057],"┌─────────────────────────┬──────────┬────────────┐",[7237,7508,7510],{"className":7509},[3057],"│                         │ executed │   failed   │",[7237,7512,7514],{"className":7513},[3057],"├─────────────────────────┼──────────┼────────────┤",[7237,7516,7518],{"className":7517},[3057],"│              iterations │        1 │          0 │",[7237,7520,7522],{"className":7521},[3057],"│                requests │        8 │          0 │",[7237,7524,7526],{"className":7525},[3057],"│            test-scripts │       16 │          0 │",[7237,7528,7530],{"className":7529},[3057],"│      prerequest-scripts │        8 │          0 │",[7237,7532,7534],{"className":7533},[3057],"│              assertions │       24 │          1 │",[7237,7536,7514],{"className":7537},[3057],[7237,7539,7541,7542,7547],{"className":7540},[3057],"│ total run duration: ",[3055,7543,7546],{"className":7544},[7545],"text-yellow-400","3.8s"," │          │            │",[7237,7549,7551,7552,7556],{"className":7550},[3057],"│ total data received: ",[3055,7553,7555],{"className":7554},[7328],"4.2kB"," │         │            │",[7237,7558,7560],{"className":7559},[3057],"└─────────────────────────┴──────────┴────────────┘",[2966,7562,7564],{"id":7563},"postman-github-integration-контроль-версій-для-api-контрактів","Postman GitHub Integration: Контроль Версій для API-Контрактів",[2959,7566,7567],{},"Це відносно нова, але революційна функція Postman. Вона дозволяє підключити ваш Postman workspace безпосередньо до GitHub-репозиторію, щоб колекції зберігались у вигляді JSON-файлів у репозиторії. Зміни в колекції автоматично комітяться у гілку.",[3002,7569,7571],{"id":7570},"навіщо-це-потрібно","Навіщо це потрібно?",[2959,7573,7574],{},"Без інтеграції з Git колекція Postman живе у хмарі Postman і недоступна тим, хто не має акаунту або не доданий до workspace. Ви не можете зробити pull request із змінами до колекції, переглянути diff, відкотити до попередньої версії.",[2959,7576,7577],{},"З Git-інтеграцією ваша колекція стає частиною репозиторію. Зміна ендпоінту в коді? Зроби PR, де в одному diff видно і зміну C#-коду, і зміну Postman-тесту для цього ендпоінту. Це справжній «Infrastructure as Code» для вашого API.",[3002,7579,7581],{"id":7580},"налаштування-інтеграції","Налаштування інтеграції",[6310,7583,7584,7588,7605,7619,7623,7626,7850,7854,7857,7861],{},[3002,7585,7587],{"id":7586},"крок-1-підключення-до-github","Крок 1: Підключення до GitHub",[2959,7589,7590,7591,7594,7595,7598,7599,7594,7602,3829],{},"В Postman: Ліва панель → вибір ім'я workspace → ",[3012,7592,7593],{},"Integrations"," → ",[3012,7596,7597],{},"Browse Integrations"," → знайдіть ",[3012,7600,7601],{},"GitHub",[3012,7603,7604],{},"Add integration",[2959,7606,7607,7608,7611,7612,7615,7616,4545],{},"Авторизуйте Postman у GitHub через OAuth. Оберіть: репозиторій, гілку (наприклад, ",[2983,7609,7610],{},"main"," або окрему ",[2983,7613,7614],{},"postman-collections","), директорію для збереження (наприклад, ",[2983,7617,7618],{},"/postman",[3002,7620,7622],{"id":7621},"крок-2-виберіть-колекцію-та-середовище","Крок 2: Виберіть колекцію та середовище",[2959,7624,7625],{},"У вікні налаштування вкажіть, які колекції та середовища синхронізувати. Postman збереже їх у вигляді JSON-файлів:",[3042,7627,7628,7695,7773],{},[3045,7629,7632],{"className":3047,"code":7630,"filename":7631,"language":3050,"meta":3051,"style":3051},"{\n  \"info\": { \"name\": \"MyApp API\" },\n  \"item\": [...],\n  \"variable\": [{ \"key\": \"baseUrl\", \"value\": \"http://localhost:5000\" }]\n}\n","postman/collections/MyApp_API.postman_collection.json",[2983,7633,7634,7638,7652,7663,7691],{"__ignoreMap":3051},[3055,7635,7636],{"class":3057,"line":3058},[3055,7637,3062],{"class":3061},[3055,7639,7640,7642,7644,7646,7648,7650],{"class":3057,"line":3065},[3055,7641,3069],{"class":3068},[3055,7643,3072],{"class":3061},[3055,7645,3075],{"class":3068},[3055,7647,3078],{"class":3061},[3055,7649,3082],{"class":3081},[3055,7651,3085],{"class":3061},[3055,7653,7654,7656,7658,7660],{"class":3057,"line":3088},[3055,7655,3091],{"class":3068},[3055,7657,3116],{"class":3061},[3055,7659,3120],{"class":3119},[3055,7661,7662],{"class":3061},"],\n",[3055,7664,7665,7668,7671,7674,7676,7678,7680,7683,7685,7688],{"class":3057,"line":3097},[3055,7666,7667],{"class":3068},"  \"variable\"",[3055,7669,7670],{"class":3061},": [{ ",[3055,7672,7673],{"class":3068},"\"key\"",[3055,7675,3078],{"class":3061},[3055,7677,4040],{"class":3081},[3055,7679,3110],{"class":3061},[3055,7681,7682],{"class":3068},"\"value\"",[3055,7684,3078],{"class":3061},[3055,7686,7687],{"class":3081},"\"http://localhost:5000\"",[3055,7689,7690],{"class":3061}," }]\n",[3055,7692,7693],{"class":3057,"line":3126},[3055,7694,3202],{"class":3061},[3045,7696,7699],{"className":3047,"code":7697,"filename":7698,"language":3050,"meta":3051,"style":3051},"{\n  \"name\": \"Development\",\n  \"variable\": [\n    { \"key\": \"baseUrl\", \"value\": \"http://localhost:5000\" },\n    { \"key\": \"apiKey\", \"value\": \"dev-key-123\" }\n  ]\n}\n","postman/environments/Development.postman_environment.json",[2983,7700,7701,7705,7717,7723,7743,7765,7769],{"__ignoreMap":3051},[3055,7702,7703],{"class":3057,"line":3058},[3055,7704,3062],{"class":3061},[3055,7706,7707,7710,7712,7715],{"class":3057,"line":3065},[3055,7708,7709],{"class":3068},"  \"name\"",[3055,7711,3078],{"class":3061},[3055,7713,7714],{"class":3081},"\"Development\"",[3055,7716,3225],{"class":3061},[3055,7718,7719,7721],{"class":3057,"line":3088},[3055,7720,7667],{"class":3068},[3055,7722,3094],{"class":3061},[3055,7724,7725,7727,7729,7731,7733,7735,7737,7739,7741],{"class":3057,"line":3097},[3055,7726,3100],{"class":3061},[3055,7728,7673],{"class":3068},[3055,7730,3078],{"class":3061},[3055,7732,4040],{"class":3081},[3055,7734,3110],{"class":3061},[3055,7736,7682],{"class":3068},[3055,7738,3078],{"class":3061},[3055,7740,7687],{"class":3081},[3055,7742,3085],{"class":3061},[3055,7744,7745,7747,7749,7751,7754,7756,7758,7760,7763],{"class":3057,"line":3126},[3055,7746,3100],{"class":3061},[3055,7748,7673],{"class":3068},[3055,7750,3078],{"class":3061},[3055,7752,7753],{"class":3081},"\"apiKey\"",[3055,7755,3110],{"class":3061},[3055,7757,7682],{"class":3068},[3055,7759,3078],{"class":3061},[3055,7761,7762],{"class":3081},"\"dev-key-123\"",[3055,7764,3264],{"class":3061},[3055,7766,7767],{"class":3057,"line":3148},[3055,7768,3196],{"class":3061},[3055,7770,7771],{"class":3057,"line":3170},[3055,7772,3202],{"class":3061},[3045,7774,7777],{"className":3047,"code":7775,"filename":7776,"language":3050,"meta":3051,"style":3051},"{\n  \"name\": \"Staging\",\n  \"variable\": [\n    { \"key\": \"baseUrl\", \"value\": \"https://staging.example.com\" },\n    { \"key\": \"apiKey\", \"value\": \"{{STAGING_API_KEY}}\" }\n  ]\n}\n","postman/environments/Staging.postman_environment.json",[2983,7778,7779,7783,7794,7800,7821,7842,7846],{"__ignoreMap":3051},[3055,7780,7781],{"class":3057,"line":3058},[3055,7782,3062],{"class":3061},[3055,7784,7785,7787,7789,7792],{"class":3057,"line":3065},[3055,7786,7709],{"class":3068},[3055,7788,3078],{"class":3061},[3055,7790,7791],{"class":3081},"\"Staging\"",[3055,7793,3225],{"class":3061},[3055,7795,7796,7798],{"class":3057,"line":3088},[3055,7797,7667],{"class":3068},[3055,7799,3094],{"class":3061},[3055,7801,7802,7804,7806,7808,7810,7812,7814,7816,7819],{"class":3057,"line":3097},[3055,7803,3100],{"class":3061},[3055,7805,7673],{"class":3068},[3055,7807,3078],{"class":3061},[3055,7809,4040],{"class":3081},[3055,7811,3110],{"class":3061},[3055,7813,7682],{"class":3068},[3055,7815,3078],{"class":3061},[3055,7817,7818],{"class":3081},"\"https://staging.example.com\"",[3055,7820,3085],{"class":3061},[3055,7822,7823,7825,7827,7829,7831,7833,7835,7837,7840],{"class":3057,"line":3126},[3055,7824,3100],{"class":3061},[3055,7826,7673],{"class":3068},[3055,7828,3078],{"class":3061},[3055,7830,7753],{"class":3081},[3055,7832,3110],{"class":3061},[3055,7834,7682],{"class":3068},[3055,7836,3078],{"class":3061},[3055,7838,7839],{"class":3081},"\"{{STAGING_API_KEY}}\"",[3055,7841,3264],{"class":3061},[3055,7843,7844],{"class":3057,"line":3148},[3055,7845,3196],{"class":3061},[3055,7847,7848],{"class":3057,"line":3170},[3055,7849,3202],{"class":3061},[3002,7851,7853],{"id":7852},"крок-3-перший-коміт","Крок 3: Перший коміт",[2959,7855,7856],{},"Після збереження інтеграції Postman зробить перший коміт до вашого репозиторію. Перевірте GitHub — там з'являться ваші файли.",[3002,7858,7860],{"id":7859},"крок-4-двостороння-синхронізація","Крок 4: Двостороння синхронізація",[2959,7862,7863],{},"Налаштуйте частоту синхронізації (наприклад, кожні 10 хвилин). Зміни з Postman UI → автоматично потрапляють у GitHub. Зміни у GitHub (наприклад, через Pull Request) → підтягуються в Postman при ручному Pull.",[7865,7866,7867,7870,7871,7874],"caution",{},[3012,7868,7869],{},"Увага до секретів!"," Змінні типу Secret (паролі, токени) ",[3012,7872,7873],{},"не синхронізуються"," з GitHub з міркувань безпеки. Це правильна поведінка. Значення звичайних (Initial value) змінних середовища — синхронізуються, тому переконайтесь, що ви не зберігаєте реальні паролі як звичайні змінні.",[2966,7876,7878],{"id":7877},"інтеграція-newman-у-cicd-github-actions","Інтеграція Newman у CI/CD (GitHub Actions)",[2959,7880,7881],{},"Тепер об'єднаємо все разом: запустимо Postman-тести автоматично при кожному push у репозиторій.",[3045,7883,7887],{"className":7884,"code":7885,"language":7886,"meta":3051,"style":3051},"language-yaml shiki shiki-themes light-plus dark-plus dark-plus","# .github/workflows/api-tests.yml\nname: API Integration Tests\n\non:\n  push:\n    branches: [ main, develop ]\n  pull_request:\n    branches: [ main ]\n\njobs:\n  api-tests:\n    name: Run Postman Collection\n    runs-on: ubuntu-latest\n\n    steps:\n      # 1. Отримуємо код\n      - name: Checkout repository\n        uses: actions/checkout@v4\n\n      # 2. Запускаємо наш ASP.NET Minimal API у Docker\n      - name: Start API\n        run: |\n          docker compose up -d --build\n          echo \"Waiting for API to be ready...\"\n          timeout 60 bash -c 'until curl -sf http://localhost:5000/health; do sleep 2; done'\n          echo \"API is ready!\"\n\n      # 3. Встановлюємо Node.js та Newman\n      - name: Setup Node.js\n        uses: actions/setup-node@v4\n        with:\n          node-version: '20'\n\n      - name: Install Newman\n        run: npm install -g newman newman-reporter-htmlextra\n\n      # 4. Запускаємо тести\n      - name: Run Postman Collection\n        run: |\n          newman run postman/collections/MyApp_API.postman_collection.json \\\n            --environment postman/environments/CI.postman_environment.json \\\n            --env-var \"authPassword=${{ secrets.CI_API_PASSWORD }}\" \\\n            --reporters cli,junit,htmlextra \\\n            --reporter-junit-export results/newman-report.xml \\\n            --reporter-htmlextra-export results/newman-report.html \\\n            --bail  # Зупинитися після першого провалу\n\n      # 5. Публікуємо результати\n      - name: Publish Test Results\n        uses: dorny/test-reporter@v1\n        if: always()  # Завжди, навіть якщо тести провалились\n        with:\n          name: Postman Tests\n          path: results/newman-report.xml\n          reporter: java-junit\n\n      - name: Upload HTML Report\n        uses: actions/upload-artifact@v4\n        if: always()\n        with:\n          name: newman-html-report\n          path: results/newman-report.html\n\n      # 6. Зупиняємо Docker\n      - name: Stop API\n        if: always()\n        run: docker compose down\n","yaml",[2983,7888,7889,7894,7905,7909,7917,7924,7942,7949,7959,7963,7970,7977,7987,7997,8001,8008,8013,8025,8035,8039,8044,8055,8065,8070,8075,8080,8085,8089,8094,8105,8114,8121,8131,8135,8146,8155,8159,8164,8174,8182,8187,8192,8197,8202,8207,8212,8217,8221,8226,8237,8246,8259,8265,8275,8285,8296,8301,8313,8323,8333,8340,8350,8360,8365,8371,8383,8392],{"__ignoreMap":3051},[3055,7890,7891],{"class":3057,"line":3058},[3055,7892,7893],{"class":3799},"# .github/workflows/api-tests.yml\n",[3055,7895,7896,7899,7901],{"class":3057,"line":3065},[3055,7897,5661],{"class":7898},"sKtos",[3055,7900,3078],{"class":3061},[3055,7902,7904],{"class":7903},"su9tN","API Integration Tests\n",[3055,7906,7907],{"class":3057,"line":3088},[3055,7908,3463],{"emptyLinePlaceholder":3462},[3055,7910,7911,7914],{"class":3057,"line":3097},[3055,7912,7913],{"class":3814},"on",[3055,7915,7916],{"class":3061},":\n",[3055,7918,7919,7922],{"class":3057,"line":3126},[3055,7920,7921],{"class":7898},"  push",[3055,7923,7916],{"class":3061},[3055,7925,7926,7929,7932,7934,7936,7939],{"class":3057,"line":3148},[3055,7927,7928],{"class":7898},"    branches",[3055,7930,7931],{"class":3061},": [ ",[3055,7933,7610],{"class":7903},[3055,7935,3110],{"class":3061},[3055,7937,7938],{"class":7903},"develop",[3055,7940,7941],{"class":3061}," ]\n",[3055,7943,7944,7947],{"class":3057,"line":3170},[3055,7945,7946],{"class":7898},"  pull_request",[3055,7948,7916],{"class":3061},[3055,7950,7951,7953,7955,7957],{"class":3057,"line":3193},[3055,7952,7928],{"class":7898},[3055,7954,7931],{"class":3061},[3055,7956,7610],{"class":7903},[3055,7958,7941],{"class":3061},[3055,7960,7961],{"class":3057,"line":3199},[3055,7962,3463],{"emptyLinePlaceholder":3462},[3055,7964,7965,7968],{"class":3057,"line":3946},[3055,7966,7967],{"class":7898},"jobs",[3055,7969,7916],{"class":3061},[3055,7971,7972,7975],{"class":3057,"line":3952},[3055,7973,7974],{"class":7898},"  api-tests",[3055,7976,7916],{"class":3061},[3055,7978,7979,7982,7984],{"class":3057,"line":3970},[3055,7980,7981],{"class":7898},"    name",[3055,7983,3078],{"class":3061},[3055,7985,7986],{"class":7903},"Run Postman Collection\n",[3055,7988,7989,7992,7994],{"class":3057,"line":3979},[3055,7990,7991],{"class":7898},"    runs-on",[3055,7993,3078],{"class":3061},[3055,7995,7996],{"class":7903},"ubuntu-latest\n",[3055,7998,7999],{"class":3057,"line":3984},[3055,8000,3463],{"emptyLinePlaceholder":3462},[3055,8002,8003,8006],{"class":3057,"line":3989},[3055,8004,8005],{"class":7898},"    steps",[3055,8007,7916],{"class":3061},[3055,8009,8010],{"class":3057,"line":3995},[3055,8011,8012],{"class":3799},"      # 1. Отримуємо код\n",[3055,8014,8015,8018,8020,8022],{"class":3057,"line":4012},[3055,8016,8017],{"class":3061},"      - ",[3055,8019,5661],{"class":7898},[3055,8021,3078],{"class":3061},[3055,8023,8024],{"class":7903},"Checkout repository\n",[3055,8026,8027,8030,8032],{"class":3057,"line":4017},[3055,8028,8029],{"class":7898},"        uses",[3055,8031,3078],{"class":3061},[3055,8033,8034],{"class":7903},"actions/checkout@v4\n",[3055,8036,8037],{"class":3057,"line":4045},[3055,8038,3463],{"emptyLinePlaceholder":3462},[3055,8040,8041],{"class":3057,"line":4072},[3055,8042,8043],{"class":3799},"      # 2. Запускаємо наш ASP.NET Minimal API у Docker\n",[3055,8045,8046,8048,8050,8052],{"class":3057,"line":4098},[3055,8047,8017],{"class":3061},[3055,8049,5661],{"class":7898},[3055,8051,3078],{"class":3061},[3055,8053,8054],{"class":7903},"Start API\n",[3055,8056,8057,8060,8062],{"class":3057,"line":4103},[3055,8058,8059],{"class":7898},"        run",[3055,8061,3078],{"class":3061},[3055,8063,8064],{"class":3907},"|\n",[3055,8066,8067],{"class":3057,"line":4109},[3055,8068,8069],{"class":7903},"          docker compose up -d --build\n",[3055,8071,8072],{"class":3057,"line":4122},[3055,8073,8074],{"class":7903},"          echo \"Waiting for API to be ready...\"\n",[3055,8076,8077],{"class":3057,"line":4144},[3055,8078,8079],{"class":7903},"          timeout 60 bash -c 'until curl -sf http://localhost:5000/health; do sleep 2; done'\n",[3055,8081,8082],{"class":3057,"line":4155},[3055,8083,8084],{"class":7903},"          echo \"API is ready!\"\n",[3055,8086,8087],{"class":3057,"line":4164},[3055,8088,3463],{"emptyLinePlaceholder":3462},[3055,8090,8091],{"class":3057,"line":4176},[3055,8092,8093],{"class":3799},"      # 3. Встановлюємо Node.js та Newman\n",[3055,8095,8096,8098,8100,8102],{"class":3057,"line":4182},[3055,8097,8017],{"class":3061},[3055,8099,5661],{"class":7898},[3055,8101,3078],{"class":3061},[3055,8103,8104],{"class":7903},"Setup Node.js\n",[3055,8106,8107,8109,8111],{"class":3057,"line":4190},[3055,8108,8029],{"class":7898},[3055,8110,3078],{"class":3061},[3055,8112,8113],{"class":7903},"actions/setup-node@v4\n",[3055,8115,8116,8119],{"class":3057,"line":4201},[3055,8117,8118],{"class":7898},"        with",[3055,8120,7916],{"class":3061},[3055,8122,8123,8126,8128],{"class":3057,"line":4227},[3055,8124,8125],{"class":7898},"          node-version",[3055,8127,3078],{"class":3061},[3055,8129,8130],{"class":7903},"'20'\n",[3055,8132,8133],{"class":3057,"line":4233},[3055,8134,3463],{"emptyLinePlaceholder":3462},[3055,8136,8137,8139,8141,8143],{"class":3057,"line":4255},[3055,8138,8017],{"class":3061},[3055,8140,5661],{"class":7898},[3055,8142,3078],{"class":3061},[3055,8144,8145],{"class":7903},"Install Newman\n",[3055,8147,8148,8150,8152],{"class":3057,"line":4267},[3055,8149,8059],{"class":7898},[3055,8151,3078],{"class":3061},[3055,8153,8154],{"class":7903},"npm install -g newman newman-reporter-htmlextra\n",[3055,8156,8157],{"class":3057,"line":4289},[3055,8158,3463],{"emptyLinePlaceholder":3462},[3055,8160,8161],{"class":3057,"line":4297},[3055,8162,8163],{"class":3799},"      # 4. Запускаємо тести\n",[3055,8165,8166,8168,8170,8172],{"class":3057,"line":4302},[3055,8167,8017],{"class":3061},[3055,8169,5661],{"class":7898},[3055,8171,3078],{"class":3061},[3055,8173,7986],{"class":7903},[3055,8175,8176,8178,8180],{"class":3057,"line":4307},[3055,8177,8059],{"class":7898},[3055,8179,3078],{"class":3061},[3055,8181,8064],{"class":3907},[3055,8183,8184],{"class":3057,"line":4328},[3055,8185,8186],{"class":7903},"          newman run postman/collections/MyApp_API.postman_collection.json \\\n",[3055,8188,8189],{"class":3057,"line":4361},[3055,8190,8191],{"class":7903},"            --environment postman/environments/CI.postman_environment.json \\\n",[3055,8193,8194],{"class":3057,"line":4368},[3055,8195,8196],{"class":7903},"            --env-var \"authPassword=${{ secrets.CI_API_PASSWORD }}\" \\\n",[3055,8198,8199],{"class":3057,"line":4373},[3055,8200,8201],{"class":7903},"            --reporters cli,junit,htmlextra \\\n",[3055,8203,8204],{"class":3057,"line":4378},[3055,8205,8206],{"class":7903},"            --reporter-junit-export results/newman-report.xml \\\n",[3055,8208,8209],{"class":3057,"line":4397},[3055,8210,8211],{"class":7903},"            --reporter-htmlextra-export results/newman-report.html \\\n",[3055,8213,8214],{"class":3057,"line":4402},[3055,8215,8216],{"class":7903},"            --bail  # Зупинитися після першого провалу\n",[3055,8218,8219],{"class":3057,"line":4408},[3055,8220,3463],{"emptyLinePlaceholder":3462},[3055,8222,8223],{"class":3057,"line":4438},[3055,8224,8225],{"class":3799},"      # 5. Публікуємо результати\n",[3055,8227,8228,8230,8232,8234],{"class":3057,"line":4443},[3055,8229,8017],{"class":3061},[3055,8231,5661],{"class":7898},[3055,8233,3078],{"class":3061},[3055,8235,8236],{"class":7903},"Publish Test Results\n",[3055,8238,8239,8241,8243],{"class":3057,"line":4449},[3055,8240,8029],{"class":7898},[3055,8242,3078],{"class":3061},[3055,8244,8245],{"class":7903},"dorny/test-reporter@v1\n",[3055,8247,8248,8251,8253,8256],{"class":3057,"line":4485},[3055,8249,8250],{"class":7898},"        if",[3055,8252,3078],{"class":3061},[3055,8254,8255],{"class":7903},"always()",[3055,8257,8258],{"class":3799},"  # Завжди, навіть якщо тести провалились\n",[3055,8260,8261,8263],{"class":3057,"line":4514},[3055,8262,8118],{"class":7898},[3055,8264,7916],{"class":3061},[3055,8266,8267,8270,8272],{"class":3057,"line":4519},[3055,8268,8269],{"class":7898},"          name",[3055,8271,3078],{"class":3061},[3055,8273,8274],{"class":7903},"Postman Tests\n",[3055,8276,8277,8280,8282],{"class":3057,"line":4553},[3055,8278,8279],{"class":7898},"          path",[3055,8281,3078],{"class":3061},[3055,8283,8284],{"class":7903},"results/newman-report.xml\n",[3055,8286,8288,8291,8293],{"class":3057,"line":8287},55,[3055,8289,8290],{"class":7898},"          reporter",[3055,8292,3078],{"class":3061},[3055,8294,8295],{"class":7903},"java-junit\n",[3055,8297,8299],{"class":3057,"line":8298},56,[3055,8300,3463],{"emptyLinePlaceholder":3462},[3055,8302,8304,8306,8308,8310],{"class":3057,"line":8303},57,[3055,8305,8017],{"class":3061},[3055,8307,5661],{"class":7898},[3055,8309,3078],{"class":3061},[3055,8311,8312],{"class":7903},"Upload HTML Report\n",[3055,8314,8316,8318,8320],{"class":3057,"line":8315},58,[3055,8317,8029],{"class":7898},[3055,8319,3078],{"class":3061},[3055,8321,8322],{"class":7903},"actions/upload-artifact@v4\n",[3055,8324,8326,8328,8330],{"class":3057,"line":8325},59,[3055,8327,8250],{"class":7898},[3055,8329,3078],{"class":3061},[3055,8331,8332],{"class":7903},"always()\n",[3055,8334,8336,8338],{"class":3057,"line":8335},60,[3055,8337,8118],{"class":7898},[3055,8339,7916],{"class":3061},[3055,8341,8343,8345,8347],{"class":3057,"line":8342},61,[3055,8344,8269],{"class":7898},[3055,8346,3078],{"class":3061},[3055,8348,8349],{"class":7903},"newman-html-report\n",[3055,8351,8353,8355,8357],{"class":3057,"line":8352},62,[3055,8354,8279],{"class":7898},[3055,8356,3078],{"class":3061},[3055,8358,8359],{"class":7903},"results/newman-report.html\n",[3055,8361,8363],{"class":3057,"line":8362},63,[3055,8364,3463],{"emptyLinePlaceholder":3462},[3055,8366,8368],{"class":3057,"line":8367},64,[3055,8369,8370],{"class":3799},"      # 6. Зупиняємо Docker\n",[3055,8372,8374,8376,8378,8380],{"class":3057,"line":8373},65,[3055,8375,8017],{"class":3061},[3055,8377,5661],{"class":7898},[3055,8379,3078],{"class":3061},[3055,8381,8382],{"class":7903},"Stop API\n",[3055,8384,8386,8388,8390],{"class":3057,"line":8385},66,[3055,8387,8250],{"class":7898},[3055,8389,3078],{"class":3061},[3055,8391,8332],{"class":7903},[3055,8393,8395,8397,8399],{"class":3057,"line":8394},67,[3055,8396,8059],{"class":7898},[3055,8398,3078],{"class":3061},[3055,8400,8401],{"class":7903},"docker compose down\n",[2959,8403,8404,8405,8408,8409,8412],{},"Для CI-середовища нам потрібний окремий файл середовища, де ",[2983,8406,8407],{},"authPassword"," буде порожнім (він підставляється через ",[2983,8410,8411],{},"--env-var"," з GitHub Secrets):",[3045,8414,8416],{"className":3047,"code":8415,"language":3050,"meta":3051,"style":3051},"{\n    \"name\": \"CI\",\n    \"values\": [\n        { \"key\": \"baseUrl\", \"value\": \"http://localhost:5000\", \"enabled\": true },\n        { \"key\": \"username\", \"value\": \"ci-admin@localhost\", \"enabled\": true },\n        { \"key\": \"password\", \"value\": \"\", \"enabled\": true },\n        { \"key\": \"authToken\", \"value\": \"\", \"enabled\": true }\n    ]\n}\n",[2983,8417,8418,8422,8433,8440,8471,8500,8529,8557,8562],{"__ignoreMap":3051},[3055,8419,8420],{"class":3057,"line":3058},[3055,8421,3062],{"class":3061},[3055,8423,8424,8426,8428,8431],{"class":3057,"line":3065},[3055,8425,6411],{"class":3068},[3055,8427,3078],{"class":3061},[3055,8429,8430],{"class":3081},"\"CI\"",[3055,8432,3225],{"class":3061},[3055,8434,8435,8438],{"class":3057,"line":3088},[3055,8436,8437],{"class":3068},"    \"values\"",[3055,8439,3094],{"class":3061},[3055,8441,8442,8445,8447,8449,8451,8453,8455,8457,8459,8461,8464,8466,8469],{"class":3057,"line":3097},[3055,8443,8444],{"class":3061},"        { ",[3055,8446,7673],{"class":3068},[3055,8448,3078],{"class":3061},[3055,8450,4040],{"class":3081},[3055,8452,3110],{"class":3061},[3055,8454,7682],{"class":3068},[3055,8456,3078],{"class":3061},[3055,8458,7687],{"class":3081},[3055,8460,3110],{"class":3061},[3055,8462,8463],{"class":3068},"\"enabled\"",[3055,8465,3078],{"class":3061},[3055,8467,8468],{"class":3814},"true",[3055,8470,3085],{"class":3061},[3055,8472,8473,8475,8477,8479,8481,8483,8485,8487,8490,8492,8494,8496,8498],{"class":3057,"line":3126},[3055,8474,8444],{"class":3061},[3055,8476,7673],{"class":3068},[3055,8478,3078],{"class":3061},[3055,8480,4067],{"class":3081},[3055,8482,3110],{"class":3061},[3055,8484,7682],{"class":3068},[3055,8486,3078],{"class":3061},[3055,8488,8489],{"class":3081},"\"ci-admin@localhost\"",[3055,8491,3110],{"class":3061},[3055,8493,8463],{"class":3068},[3055,8495,3078],{"class":3061},[3055,8497,8468],{"class":3814},[3055,8499,3085],{"class":3061},[3055,8501,8502,8504,8506,8508,8510,8512,8514,8516,8519,8521,8523,8525,8527],{"class":3057,"line":3148},[3055,8503,8444],{"class":3061},[3055,8505,7673],{"class":3068},[3055,8507,3078],{"class":3061},[3055,8509,3257],{"class":3081},[3055,8511,3110],{"class":3061},[3055,8513,7682],{"class":3068},[3055,8515,3078],{"class":3061},[3055,8517,8518],{"class":3081},"\"\"",[3055,8520,3110],{"class":3061},[3055,8522,8463],{"class":3068},[3055,8524,3078],{"class":3061},[3055,8526,8468],{"class":3814},[3055,8528,3085],{"class":3061},[3055,8530,8531,8533,8535,8537,8539,8541,8543,8545,8547,8549,8551,8553,8555],{"class":3057,"line":3170},[3055,8532,8444],{"class":3061},[3055,8534,7673],{"class":3068},[3055,8536,3078],{"class":3061},[3055,8538,3871],{"class":3081},[3055,8540,3110],{"class":3061},[3055,8542,7682],{"class":3068},[3055,8544,3078],{"class":3061},[3055,8546,8518],{"class":3081},[3055,8548,3110],{"class":3061},[3055,8550,8463],{"class":3068},[3055,8552,3078],{"class":3061},[3055,8554,8468],{"class":3814},[3055,8556,3264],{"class":3061},[3055,8558,8559],{"class":3057,"line":3193},[3055,8560,8561],{"class":3061},"    ]\n",[3055,8563,8564],{"class":3057,"line":3199},[3055,8565,3202],{"class":3061},[3733,8567,8568,8574,8575,8577],{},[3012,8569,8570,8573],{},[2983,8571,8572],{},"--bail"," vs без bail",": Флаг ",[2983,8576,8572],{}," зупиняє Newman після першого провалу — добре для швидкого feedback у CI. Без нього Newman запускає всі тести і видає повний звіт — краще для детального аналізу. Обирайте залежно від вашого workflow.",[2966,8579,8581],{"id":8580},"моніторинг-api-з-postman-monitors","Моніторинг API з Postman Monitors",[2959,8583,8584,8585,8588],{},"Раз вже ви маєте колекцію з тестами — налаштуйте ",[3012,8586,8587],{},"Monitor",". Це хмарний планувальник Postman, який запускає вашу колекцію за розкладом (наприклад, кожні 5 хвилин) і надсилає сповіщення, якщо тести провалюються.",[2959,8590,8591],{},"Це примітивний, але ефективний метод uptime-моніторингу: ви дізнаєтесь, що продакшн-API «впав», ще до того, як це помітять користувачі.",[2959,8593,8594,8595,8598],{},"Налаштування: Права кнопка на колекції → ",[3012,8596,8597],{},"Monitor Collection"," → вкажіть середовище (Production), розклад та email для сповіщень.",[8600,8601,8602,8605],"warning",{},[3012,8603,8604],{},"Обмеження безкоштовного плану",": Postman надає обмежену кількість безкоштовних викликів Monitor на місяць. Для серйозного моніторингу розгляньте комерційний план або альтернативи (наприклад, Grafana + Alertmanager).",[8607,8608],"hr",{},[2966,8610,8612],{"id":8611},"практика","Практика",[8614,8615,8616,8668,8701],"accordion",{},[8617,8618,8621,8624,8662],"accordion-item",{"icon":8619,"label":8620},"i-lucide-star","⭐ Рівень 1: Базова колекція з середовищами",[2959,8622,8623],{},"Налаштуйте Postman для вашого Minimal API:",[8625,8626,8627,8633,8642,8653,8659],"ol",{},[4876,8628,8629,8630,3829],{},"Створіть колекцію ",[2983,8631,8632],{},"MyMinimalAPI",[4876,8634,8635,8636,8638,8639,8641],{},"Налаштуйте два середовища: ",[2983,8637,3504],{}," (localhost) та ",[2983,8640,3510],{}," (якщо є).",[4876,8643,8644,8645,3110,8647,3110,8649,3110,8651,3829],{},"Визначте змінні: ",[2983,8646,3514],{},[2983,8648,3572],{},[2983,8650,3584],{},[2983,8652,3596],{},[4876,8654,8655,8656,3829],{},"Перенесіть всі існуючі ручні запити у відповідні папки колекції (Auth, Products, тощо), замінивши хардкод на змінні ",[2983,8657,8658],{},"{{baseUrl}}",[4876,8660,8661],{},"Напишіть базові тести (status code, content-type) для кожного запиту.",[2959,8663,8664,8667],{},[3012,8665,8666],{},"Очікуваний результат",": Запуск Collection Runner → все зелене, без хардкод URL та токенів.",[8617,8669,8671],{"icon":8619,"label":8670},"⭐⭐ Рівень 2: Автоматичний логін та workflow тест",[8625,8672,8673,8676,8682,8689,8695,8698],{},[4876,8674,8675],{},"Реалізуйте Pre-request Script на рівні колекції для автоматичного отримання JWT токену (з кешуванням за часом закінчення).",[4876,8677,8678,8679,8681],{},"Налаштуйте Authorization → Bearer Token → ",[2983,8680,4570],{}," на рівні колекції.",[4876,8683,8684,8685,8688],{},"Створіть папку ",[2983,8686,8687],{},"Product Lifecycle"," з 6 запитами: Create → Read → Update → Verify Update → Delete → Verify 404.",[4876,8690,8691,8692,3829],{},"Передавайте ID між запитами через ",[2983,8693,8694],{},"pm.collectionVariables",[4876,8696,8697],{},"Напишіть assertions для кожного кроку.",[4876,8699,8700],{},"Запустіть папку через Collection Runner — всі 6 кроків мають пройти послідовно.",[8617,8702,8704],{"icon":8619,"label":8703},"⭐⭐⭐ Рівень 3: Data-driven тести та Newman у CI",[8625,8705,8706,8715,8722,8725,8728],{},[4876,8707,8684,8708,8711,8712,3829],{},[2983,8709,8710],{},"Product Validation"," з одним запитом ",[2983,8713,8714],{},"POST /api/products",[4876,8716,8717,8718,8721],{},"Підготуйте ",[2983,8719,8720],{},"validation-data.csv"," з мінімум 10 рядками різних комбінацій (валідні/невалідні дані) та очікуваними статусами.",[4876,8723,8724],{},"Запустіть папку через Collection Runner з Data File — переконайтесь, що тести коректно перевіряють кожен рядок.",[4876,8726,8727],{},"Встановіть Newman та запустіть вашу колекцію з командного рядка, отримавши HTML-звіт.",[4876,8729,8730,8731,3829],{},"(Бонус) Налаштуйте GitHub Actions workflow, який запускає Newman при PR у ",[2983,8732,7610],{},[8607,8734],{},[2959,8736,8737,8738,8741,8742,8745],{},"У наступній статті ми зануримось у тестування коду, який ",[3012,8739,8740],{},"споживає"," сторонні HTTP API — навчимось ізолювати ",[2983,8743,8744],{},"HttpClient"," у юніт-тестах без жодного реального мережевого з'єднання.",[8747,8748,8749],"style",{},"html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sLwNe, html code.shiki .sLwNe{--shiki-light:#0451A5;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .se1LK, html code.shiki .se1LK{--shiki-light:#CD3131;--shiki-default:#F44747;--shiki-dark:#F44747}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 .s-QsJ, html code.shiki .s-QsJ{--shiki-light:#0070C1;--shiki-default:#4FC1FF;--shiki-dark:#4FC1FF}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .s8Opu, html code.shiki .s8Opu{--shiki-light:#795E26;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sCDza, html code.shiki .sCDza{--shiki-light:#AF00DB;--shiki-default:#CE92A4;--shiki-dark:#CE92A4}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}html pre.shiki code .shsrj, html code.shiki .shsrj{--shiki-light:#811F3F;--shiki-default:#D16969;--shiki-dark:#D16969}html pre.shiki code .sAV2Q, html code.shiki .sAV2Q{--shiki-light:#D16969;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .siofk, html code.shiki .siofk{--shiki-light:#EE0000;--shiki-default:#DCDCAA;--shiki-dark:#DCDCAA}html pre.shiki code .sYsli, html code.shiki .sYsli{--shiki-light:#000000;--shiki-default:#D7BA7D;--shiki-dark:#D7BA7D}html pre.shiki code .sD7JJ, html code.shiki .sD7JJ{--shiki-light:#000000FF;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .sKtos, html code.shiki .sKtos{--shiki-light:#800000;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .su9tN, html code.shiki .su9tN{--shiki-light:#0000FF;--shiki-default:#CE9178;--shiki-dark:#CE9178}",{"title":3051,"searchDepth":3065,"depth":3065,"links":8751},[8752,8753,8758,8762,8767,8771,8780,8789,8797,8798,8799],{"id":2968,"depth":3065,"text":2969},{"id":2999,"depth":3065,"text":3000,"children":8754},[8755,8756,8757],{"id":3004,"depth":3088,"text":3005},{"id":3036,"depth":3088,"text":3037},{"id":3399,"depth":3088,"text":3400},{"id":3420,"depth":3065,"text":3421,"children":8759},[8760,8761],{"id":3427,"depth":3088,"text":3428},{"id":3524,"depth":3088,"text":3525},{"id":3745,"depth":3065,"text":3746,"children":8763},[8764,8765,8766],{"id":3762,"depth":3088,"text":3763},{"id":3780,"depth":3088,"text":3781},{"id":4581,"depth":3088,"text":4582},{"id":4849,"depth":3065,"text":4850,"children":8768},[8769,8770],{"id":4864,"depth":3088,"text":4865},{"id":4926,"depth":3088,"text":4927},{"id":6297,"depth":3065,"text":6298,"children":8772},[8773,8774,8775,8776,8777,8778,8779],{"id":6304,"depth":3088,"text":6305},{"id":6314,"depth":3088,"text":6315},{"id":6552,"depth":3088,"text":6553},{"id":6698,"depth":3088,"text":6699},{"id":6793,"depth":3088,"text":6794},{"id":6921,"depth":3088,"text":6922},{"id":6971,"depth":3088,"text":6972},{"id":7032,"depth":3065,"text":7033,"children":8781},[8782,8783,8784,8785,8786,8787,8788],{"id":7036,"depth":3088,"text":7037},{"id":7074,"depth":3088,"text":7075},{"id":7220,"depth":3088,"text":7221},{"id":7229,"depth":3088,"text":7230},{"id":7278,"depth":3088,"text":7279},{"id":7294,"depth":3088,"text":7295},{"id":7346,"depth":3088,"text":7347},{"id":7563,"depth":3065,"text":7564,"children":8790},[8791,8792,8793,8794,8795,8796],{"id":7570,"depth":3088,"text":7571},{"id":7580,"depth":3088,"text":7581},{"id":7586,"depth":3088,"text":7587},{"id":7621,"depth":3088,"text":7622},{"id":7852,"depth":3088,"text":7853},{"id":7859,"depth":3088,"text":7860},{"id":7877,"depth":3065,"text":7878},{"id":8580,"depth":3065,"text":8581},{"id":8611,"depth":3065,"text":8612},"Від разових запитів до автоматизованих тестових сюїт — опановуємо Postman як інженерний інструмент. Колекції, змінні, Pre-request Scripts, тестові assertions та запуск у CI/CD через Newman.","md",null,{},{"title":1026,"description":8800},"cDbt8RKo3IblhDIrgnJrn89d9v7VA2cr21IGBYnp320",[8807,8809],{"title":1022,"path":1023,"stem":1024,"description":8808,"children":-1},"Глибоке тестування validation, ProblemDetails та глобальної обробки помилок. Тестування складних бізнес-сценаріїв через HTTP. WebApplicationFactory разом із Testcontainers для реальної PostgreSQL. Організація великої тест-сюїти.",{"title":1030,"path":1031,"stem":1032,"description":8810,"children":-1},"Ваш сервіс викликає платіжний шлюз. Як протестувати це, не знявши реальні гроші? Глибоко вивчаємо архітектуру HttpClient у .NET, проблему socket exhaustion, IHttpClientFactory та патерн MockHttpMessageHandler.",1777912527648]