[{"data":1,"prerenderedAt":10119},["ShallowReactive",2],{"navigation_docs":3,"-csharp-ef-core-schema-management-part2":2949,"-csharp-ef-core-schema-management-part2-surround":10114},[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":631,"body":2951,"description":10108,"extension":10109,"links":10110,"meta":10111,"navigation":3082,"path":632,"seo":10112,"stem":633,"__hash__":10113},"docs/01.csharp/10.ef-core/25.schema-management-part2.md",{"type":2952,"value":2953,"toc":10055},"minimark",[2954,2959,2972,2975,2980,2988,3011,3020,3030,3040,3045,3051,3055,3786,3790,4273,4277,4280,4471,4813,4824,4826,4830,4833,4837,4861,4865,4954,4983,4987,4993,4997,5264,5268,5425,5429,5432,5495,5498,5500,5504,5514,5517,5521,5548,5562,5566,5573,5577,5594,5757,5760,5897,5901,5908,5939,5943,5949,5953,6044,6046,6050,6053,6079,6083,6323,6326,6491,6494,7011,7015,7102,7109,7188,7270,7274,7755,7759,8379,8383,8512,8514,8518,8521,8525,8789,8793,9285,9299,9303,9309,9589,9591,9595,9599,9717,9721,9814,9818,9861,9863,9867,9870,9875,9939,9944,9995,10007,10009,10013,10051],[2955,2956,2958],"h1",{"id":2957},"управління-схемою-partial-classes-re-scaffolding-та-multi-database","Управління Схемою: Partial Classes, Re-scaffolding та Multi-Database",[2960,2961,2962],"blockquote",{},[2963,2964,2965,2966,2971],"p",{},"Це продовження статті ",[2967,2968,2970],"a",{"href":2969},"/csharp/ef-core/25.schema-management-part1","«Управління Схемою: Частина 1»",". Читайте послідовно.",[2973,2974],"hr",{},[2976,2977,2979],"h2",{"id":2978},"partial-classes-розширення-без-ризику","Partial Classes: розширення без ризику",[2963,2981,2982,2983,2987],{},"Перша частина показала що scaffold генерує ",[2984,2985,2986],"code",{},"partial class",". Це ключова деталь що робить Database-First підхід практичним у довгостроковій перспективі.",[2963,2989,2990,2994,2995,2998,2999,3002,3003,3006,3007,3010],{},[2991,2992,2993],"strong",{},"Проблема",": ви отримали згенерований ",[2984,2996,2997],{},"Product.cs",". Додали метод ",[2984,3000,3001],{},"CalculateDiscount()",". Схема БД змінилась, треба re-scaffold з ",[2984,3004,3005],{},"--force",". Ваш метод ",[2991,3008,3009],{},"перезаписано",".",[2963,3012,3013,3016,3017,3019],{},[2991,3014,3015],{},"Рішення",": ",[2984,3018,2986],{}," у C# дозволяє розрізати один клас на кілька файлів. Компілятор збирає їх в один клас. Scaffold генерує один файл — ви пишете розширення у іншому файлі що ніколи не перезаписується.",[3021,3022,3027],"pre",{"className":3023,"code":3025,"language":3026},[3024],"language-text","Product.cs              ← згенерований (перезаписується при --force)\nProduct.Extensions.cs   ← ваш розширення (НІКОЛИ не перезаписується)\n","text",[2984,3028,3025],{"__ignoreMap":3029},"",[2963,3031,3032,3033,3036,3037,3010],{},"обидва: ",[2984,3034,3035],{},"public partial class Product { ... }"," → компілятор зберіть в один ",[2984,3038,3039],{},"Product",[3041,3042,3044],"h3",{"id":3043},"структура-файлів-у-database-first-проєкті","Структура файлів у Database-First проєкті",[3021,3046,3049],{"className":3047,"code":3048,"language":3026},[3024],"Infrastructure/\n├── Entities/                         ← Згенеровані (перезаписуються при re-scaffold)\n│   ├── Product.cs                    ← auto-generated\n│   ├── Category.cs                   ← auto-generated\n│   └── Order.cs                      ← auto-generated\n├── Entities/Extensions/              ← Ваші розширення (НІКОЛИ не чіпати при scaffold)\n│   ├── Product.Extensions.cs         ← бізнес-логіка Product\n│   ├── Category.Extensions.cs        ← бізнес-логіка Category\n│   └── Order.Extensions.cs           ← бізнес-логіка Order\n└── ShopDbContext.cs                  ← згенерований\n└── ShopDbContextExtensions.cs        ← ваші розширення DbContext\n",[2984,3050,3048],{"__ignoreMap":3029},[3041,3052,3054],{"id":3053},"приклади-rozширень-через-partial-classes","Приклади rozширень через Partial Classes",[3021,3056,3060],{"className":3057,"code":3058,"language":3059,"meta":3029,"style":3029},"language-csharp shiki shiki-themes light-plus dark-plus dark-plus","// Infrastructure/Entities/Extensions/Product.Extensions.cs\n// Цей файл НІКОЛИ не перезаписується scaffold\n\nnamespace YourApp.Infrastructure.Entities;\n\n// Той самий namespace і тип що у згенерованому Product.cs\npublic partial class Product\n{\n    // 1. Обчислювані властивості (computed properties)\n    public decimal FinalPrice =>\n        Discount.HasValue\n            ? Price - (Price * Discount.Value / 100)\n            : Price;\n\n    public bool IsDiscounted => Discount.HasValue && Discount.Value > 0;\n\n    // 2. Доменна логіка (методи)\n    public void ApplyDiscount(decimal discountPercent)\n    {\n        if (discountPercent \u003C 0 || discountPercent > 100)\n            throw new ArgumentOutOfRangeException(nameof(discountPercent),\n                \"Discount must be between 0 and 100.\");\n\n        Discount = discountPercent;\n    }\n\n    public void Activate()  => IsActive = true;\n    public void Deactivate() => IsActive = false;\n\n    // 3. Форматування і відображення\n    public string DisplayName => $\"{Name} ({Category?.Name ?? \"Uncategorized\"})\";\n\n    // 4. Валідація (може бути використана у тестах і бізнес-логіці)\n    public IReadOnlyList\u003Cstring> Validate()\n    {\n        var errors = new List\u003Cstring>();\n\n        if (string.IsNullOrWhiteSpace(Name))\n            errors.Add(\"Name is required.\");\n\n        if (Price \u003C= 0)\n            errors.Add(\"Price must be positive.\");\n\n        if (CategoryId \u003C= 0)\n            errors.Add(\"Category is required.\");\n\n        return errors;\n    }\n\n    // 5. Override ToString для кращого відлагодження\n    public override string ToString() =>\n        $\"Product[{Id}]: {Name} @ {Price:C} (Active: {IsActive})\";\n}\n","csharp",[2984,3061,3062,3071,3077,3084,3109,3114,3120,3135,3141,3147,3163,3174,3209,3219,3224,3262,3267,3273,3296,3302,3331,3355,3365,3370,3382,3388,3393,3416,3438,3443,3449,3502,3507,3513,3536,3541,3565,3570,3591,3609,3614,3630,3646,3651,3667,3683,3688,3698,3703,3708,3714,3730,3780],{"__ignoreMap":3029},[3063,3064,3067],"span",{"class":3065,"line":3066},"line",1,[3063,3068,3070],{"class":3069},"spJ8K","// Infrastructure/Entities/Extensions/Product.Extensions.cs\n",[3063,3072,3074],{"class":3065,"line":3073},2,[3063,3075,3076],{"class":3069},"// Цей файл НІКОЛИ не перезаписується scaffold\n",[3063,3078,3080],{"class":3065,"line":3079},3,[3063,3081,3083],{"emptyLinePlaceholder":3082},true,"\n",[3063,3085,3087,3091,3095,3098,3101,3103,3106],{"class":3065,"line":3086},4,[3063,3088,3090],{"class":3089},"su1O8","namespace",[3063,3092,3094],{"class":3093},"sN1BT"," YourApp",[3063,3096,3010],{"class":3097},"sHH4Y",[3063,3099,3100],{"class":3093},"Infrastructure",[3063,3102,3010],{"class":3097},[3063,3104,3105],{"class":3093},"Entities",[3063,3107,3108],{"class":3097},";\n",[3063,3110,3112],{"class":3065,"line":3111},5,[3063,3113,3083],{"emptyLinePlaceholder":3082},[3063,3115,3117],{"class":3065,"line":3116},6,[3063,3118,3119],{"class":3069},"// Той самий namespace і тип що у згенерованому Product.cs\n",[3063,3121,3123,3126,3129,3132],{"class":3065,"line":3122},7,[3063,3124,3125],{"class":3089},"public",[3063,3127,3128],{"class":3089}," partial",[3063,3130,3131],{"class":3089}," class",[3063,3133,3134],{"class":3093}," Product\n",[3063,3136,3138],{"class":3065,"line":3137},8,[3063,3139,3140],{"class":3097},"{\n",[3063,3142,3144],{"class":3065,"line":3143},9,[3063,3145,3146],{"class":3069},"    // 1. Обчислювані властивості (computed properties)\n",[3063,3148,3150,3153,3156,3160],{"class":3065,"line":3149},10,[3063,3151,3152],{"class":3089},"    public",[3063,3154,3155],{"class":3089}," decimal",[3063,3157,3159],{"class":3158},"siwwj"," FinalPrice",[3063,3161,3162],{"class":3097}," =>\n",[3063,3164,3166,3169,3171],{"class":3065,"line":3165},11,[3063,3167,3168],{"class":3158},"        Discount",[3063,3170,3010],{"class":3097},[3063,3172,3173],{"class":3158},"HasValue\n",[3063,3175,3177,3180,3183,3186,3188,3191,3194,3196,3199,3202,3206],{"class":3065,"line":3176},12,[3063,3178,3179],{"class":3097},"            ? ",[3063,3181,3182],{"class":3158},"Price",[3063,3184,3185],{"class":3097}," - (",[3063,3187,3182],{"class":3158},[3063,3189,3190],{"class":3097}," * ",[3063,3192,3193],{"class":3158},"Discount",[3063,3195,3010],{"class":3097},[3063,3197,3198],{"class":3158},"Value",[3063,3200,3201],{"class":3097}," / ",[3063,3203,3205],{"class":3204},"sJj4R","100",[3063,3207,3208],{"class":3097},")\n",[3063,3210,3212,3215,3217],{"class":3065,"line":3211},13,[3063,3213,3214],{"class":3097},"            : ",[3063,3216,3182],{"class":3158},[3063,3218,3108],{"class":3097},[3063,3220,3222],{"class":3065,"line":3221},14,[3063,3223,3083],{"emptyLinePlaceholder":3082},[3063,3225,3227,3229,3232,3235,3238,3240,3242,3245,3248,3250,3252,3254,3257,3260],{"class":3065,"line":3226},15,[3063,3228,3152],{"class":3089},[3063,3230,3231],{"class":3089}," bool",[3063,3233,3234],{"class":3158}," IsDiscounted",[3063,3236,3237],{"class":3097}," => ",[3063,3239,3193],{"class":3158},[3063,3241,3010],{"class":3097},[3063,3243,3244],{"class":3158},"HasValue",[3063,3246,3247],{"class":3097}," && ",[3063,3249,3193],{"class":3158},[3063,3251,3010],{"class":3097},[3063,3253,3198],{"class":3158},[3063,3255,3256],{"class":3097}," > ",[3063,3258,3259],{"class":3204},"0",[3063,3261,3108],{"class":3097},[3063,3263,3265],{"class":3065,"line":3264},16,[3063,3266,3083],{"emptyLinePlaceholder":3082},[3063,3268,3270],{"class":3065,"line":3269},17,[3063,3271,3272],{"class":3069},"    // 2. Доменна логіка (методи)\n",[3063,3274,3276,3278,3281,3285,3288,3291,3294],{"class":3065,"line":3275},18,[3063,3277,3152],{"class":3089},[3063,3279,3280],{"class":3089}," void",[3063,3282,3284],{"class":3283},"s8Opu"," ApplyDiscount",[3063,3286,3287],{"class":3097},"(",[3063,3289,3290],{"class":3089},"decimal",[3063,3292,3293],{"class":3158}," discountPercent",[3063,3295,3208],{"class":3097},[3063,3297,3299],{"class":3065,"line":3298},19,[3063,3300,3301],{"class":3097},"    {\n",[3063,3303,3305,3309,3312,3315,3318,3320,3323,3325,3327,3329],{"class":3065,"line":3304},20,[3063,3306,3308],{"class":3307},"sCDza","        if",[3063,3310,3311],{"class":3097}," (",[3063,3313,3314],{"class":3158},"discountPercent",[3063,3316,3317],{"class":3097}," \u003C ",[3063,3319,3259],{"class":3204},[3063,3321,3322],{"class":3097}," || ",[3063,3324,3314],{"class":3158},[3063,3326,3256],{"class":3097},[3063,3328,3205],{"class":3204},[3063,3330,3208],{"class":3097},[3063,3332,3334,3337,3340,3343,3345,3348,3350,3352],{"class":3065,"line":3333},21,[3063,3335,3336],{"class":3307},"            throw",[3063,3338,3339],{"class":3089}," new",[3063,3341,3342],{"class":3093}," ArgumentOutOfRangeException",[3063,3344,3287],{"class":3097},[3063,3346,3347],{"class":3089},"nameof",[3063,3349,3287],{"class":3097},[3063,3351,3314],{"class":3158},[3063,3353,3354],{"class":3097},"),\n",[3063,3356,3358,3362],{"class":3065,"line":3357},22,[3063,3359,3361],{"class":3360},"sbdoH","                \"Discount must be between 0 and 100.\"",[3063,3363,3364],{"class":3097},");\n",[3063,3366,3368],{"class":3065,"line":3367},23,[3063,3369,3083],{"emptyLinePlaceholder":3082},[3063,3371,3373,3375,3378,3380],{"class":3065,"line":3372},24,[3063,3374,3168],{"class":3158},[3063,3376,3377],{"class":3097}," = ",[3063,3379,3314],{"class":3158},[3063,3381,3108],{"class":3097},[3063,3383,3385],{"class":3065,"line":3384},25,[3063,3386,3387],{"class":3097},"    }\n",[3063,3389,3391],{"class":3065,"line":3390},26,[3063,3392,3083],{"emptyLinePlaceholder":3082},[3063,3394,3396,3398,3400,3403,3406,3409,3411,3414],{"class":3065,"line":3395},27,[3063,3397,3152],{"class":3089},[3063,3399,3280],{"class":3089},[3063,3401,3402],{"class":3283}," Activate",[3063,3404,3405],{"class":3097},"()  => ",[3063,3407,3408],{"class":3158},"IsActive",[3063,3410,3377],{"class":3097},[3063,3412,3413],{"class":3089},"true",[3063,3415,3108],{"class":3097},[3063,3417,3419,3421,3423,3426,3429,3431,3433,3436],{"class":3065,"line":3418},28,[3063,3420,3152],{"class":3089},[3063,3422,3280],{"class":3089},[3063,3424,3425],{"class":3283}," Deactivate",[3063,3427,3428],{"class":3097},"() => ",[3063,3430,3408],{"class":3158},[3063,3432,3377],{"class":3097},[3063,3434,3435],{"class":3089},"false",[3063,3437,3108],{"class":3097},[3063,3439,3441],{"class":3065,"line":3440},29,[3063,3442,3083],{"emptyLinePlaceholder":3082},[3063,3444,3446],{"class":3065,"line":3445},30,[3063,3447,3448],{"class":3069},"    // 3. Форматування і відображення\n",[3063,3450,3452,3454,3457,3460,3462,3465,3469,3472,3475,3477,3479,3482,3485,3487,3489,3492,3495,3497,3500],{"class":3065,"line":3451},31,[3063,3453,3152],{"class":3089},[3063,3455,3456],{"class":3089}," string",[3063,3458,3459],{"class":3158}," DisplayName",[3063,3461,3237],{"class":3097},[3063,3463,3464],{"class":3360},"$\"",[3063,3466,3468],{"class":3467},"sD7JJ","{",[3063,3470,3471],{"class":3158},"Name",[3063,3473,3474],{"class":3467},"}",[3063,3476,3311],{"class":3360},[3063,3478,3468],{"class":3467},[3063,3480,3481],{"class":3158},"Category",[3063,3483,3484],{"class":3097},"?",[3063,3486,3010],{"class":3467},[3063,3488,3471],{"class":3158},[3063,3490,3491],{"class":3097}," ??",[3063,3493,3494],{"class":3360}," \"Uncategorized\"",[3063,3496,3474],{"class":3467},[3063,3498,3499],{"class":3360},")\"",[3063,3501,3108],{"class":3097},[3063,3503,3505],{"class":3065,"line":3504},32,[3063,3506,3083],{"emptyLinePlaceholder":3082},[3063,3508,3510],{"class":3065,"line":3509},33,[3063,3511,3512],{"class":3069},"    // 4. Валідація (може бути використана у тестах і бізнес-логіці)\n",[3063,3514,3516,3518,3521,3524,3527,3530,3533],{"class":3065,"line":3515},34,[3063,3517,3152],{"class":3089},[3063,3519,3520],{"class":3093}," IReadOnlyList",[3063,3522,3523],{"class":3097},"\u003C",[3063,3525,3526],{"class":3089},"string",[3063,3528,3529],{"class":3097},"> ",[3063,3531,3532],{"class":3283},"Validate",[3063,3534,3535],{"class":3097},"()\n",[3063,3537,3539],{"class":3065,"line":3538},35,[3063,3540,3301],{"class":3097},[3063,3542,3544,3547,3550,3552,3555,3558,3560,3562],{"class":3065,"line":3543},36,[3063,3545,3546],{"class":3089},"        var",[3063,3548,3549],{"class":3158}," errors",[3063,3551,3377],{"class":3097},[3063,3553,3554],{"class":3089},"new",[3063,3556,3557],{"class":3093}," List",[3063,3559,3523],{"class":3097},[3063,3561,3526],{"class":3089},[3063,3563,3564],{"class":3097},">();\n",[3063,3566,3568],{"class":3065,"line":3567},37,[3063,3569,3083],{"emptyLinePlaceholder":3082},[3063,3571,3573,3575,3577,3579,3581,3584,3586,3588],{"class":3065,"line":3572},38,[3063,3574,3308],{"class":3307},[3063,3576,3311],{"class":3097},[3063,3578,3526],{"class":3089},[3063,3580,3010],{"class":3097},[3063,3582,3583],{"class":3283},"IsNullOrWhiteSpace",[3063,3585,3287],{"class":3097},[3063,3587,3471],{"class":3158},[3063,3589,3590],{"class":3097},"))\n",[3063,3592,3594,3597,3599,3602,3604,3607],{"class":3065,"line":3593},39,[3063,3595,3596],{"class":3158},"            errors",[3063,3598,3010],{"class":3097},[3063,3600,3601],{"class":3283},"Add",[3063,3603,3287],{"class":3097},[3063,3605,3606],{"class":3360},"\"Name is required.\"",[3063,3608,3364],{"class":3097},[3063,3610,3612],{"class":3065,"line":3611},40,[3063,3613,3083],{"emptyLinePlaceholder":3082},[3063,3615,3617,3619,3621,3623,3626,3628],{"class":3065,"line":3616},41,[3063,3618,3308],{"class":3307},[3063,3620,3311],{"class":3097},[3063,3622,3182],{"class":3158},[3063,3624,3625],{"class":3097}," \u003C= ",[3063,3627,3259],{"class":3204},[3063,3629,3208],{"class":3097},[3063,3631,3633,3635,3637,3639,3641,3644],{"class":3065,"line":3632},42,[3063,3634,3596],{"class":3158},[3063,3636,3010],{"class":3097},[3063,3638,3601],{"class":3283},[3063,3640,3287],{"class":3097},[3063,3642,3643],{"class":3360},"\"Price must be positive.\"",[3063,3645,3364],{"class":3097},[3063,3647,3649],{"class":3065,"line":3648},43,[3063,3650,3083],{"emptyLinePlaceholder":3082},[3063,3652,3654,3656,3658,3661,3663,3665],{"class":3065,"line":3653},44,[3063,3655,3308],{"class":3307},[3063,3657,3311],{"class":3097},[3063,3659,3660],{"class":3158},"CategoryId",[3063,3662,3625],{"class":3097},[3063,3664,3259],{"class":3204},[3063,3666,3208],{"class":3097},[3063,3668,3670,3672,3674,3676,3678,3681],{"class":3065,"line":3669},45,[3063,3671,3596],{"class":3158},[3063,3673,3010],{"class":3097},[3063,3675,3601],{"class":3283},[3063,3677,3287],{"class":3097},[3063,3679,3680],{"class":3360},"\"Category is required.\"",[3063,3682,3364],{"class":3097},[3063,3684,3686],{"class":3065,"line":3685},46,[3063,3687,3083],{"emptyLinePlaceholder":3082},[3063,3689,3691,3694,3696],{"class":3065,"line":3690},47,[3063,3692,3693],{"class":3307},"        return",[3063,3695,3549],{"class":3158},[3063,3697,3108],{"class":3097},[3063,3699,3701],{"class":3065,"line":3700},48,[3063,3702,3387],{"class":3097},[3063,3704,3706],{"class":3065,"line":3705},49,[3063,3707,3083],{"emptyLinePlaceholder":3082},[3063,3709,3711],{"class":3065,"line":3710},50,[3063,3712,3713],{"class":3069},"    // 5. Override ToString для кращого відлагодження\n",[3063,3715,3717,3719,3722,3724,3727],{"class":3065,"line":3716},51,[3063,3718,3152],{"class":3089},[3063,3720,3721],{"class":3089}," override",[3063,3723,3456],{"class":3089},[3063,3725,3726],{"class":3283}," ToString",[3063,3728,3729],{"class":3097},"() =>\n",[3063,3731,3733,3736,3738,3741,3743,3746,3748,3750,3752,3755,3757,3759,3762,3765,3767,3770,3772,3774,3776,3778],{"class":3065,"line":3732},52,[3063,3734,3735],{"class":3360},"        $\"Product[",[3063,3737,3468],{"class":3467},[3063,3739,3740],{"class":3158},"Id",[3063,3742,3474],{"class":3467},[3063,3744,3745],{"class":3360},"]: ",[3063,3747,3468],{"class":3467},[3063,3749,3471],{"class":3158},[3063,3751,3474],{"class":3467},[3063,3753,3754],{"class":3360}," @ ",[3063,3756,3468],{"class":3467},[3063,3758,3182],{"class":3158},[3063,3760,3761],{"class":3097},":",[3063,3763,3764],{"class":3158},"C",[3063,3766,3474],{"class":3467},[3063,3768,3769],{"class":3360}," (Active: ",[3063,3771,3468],{"class":3467},[3063,3773,3408],{"class":3158},[3063,3775,3474],{"class":3467},[3063,3777,3499],{"class":3360},[3063,3779,3108],{"class":3097},[3063,3781,3783],{"class":3065,"line":3782},53,[3063,3784,3785],{"class":3097},"}\n",[3041,3787,3789],{"id":3788},"розширення-dbcontext","Розширення DbContext",[3021,3791,3793],{"className":3057,"code":3792,"language":3059,"meta":3029,"style":3029},"// Infrastructure/ShopDbContextExtensions.cs\nnamespace YourApp.Infrastructure;\n\npublic partial class ShopDbContext\n{\n    // Зручні методи запитів\n    public IQueryable\u003CProduct> ActiveProducts =>\n        Set\u003CProduct>().Where(p => p.IsActive);\n\n    public IQueryable\u003CProduct> ProductsByCategory(int categoryId) =>\n        Set\u003CProduct>().Where(p => p.CategoryId == categoryId && p.IsActive);\n\n    // Методи для складних операцій\n    public async Task\u003Cint> SoftDeleteProductAsync(int productId)\n    {\n        return await Set\u003CProduct>()\n            .Where(p => p.Id == productId)\n            .ExecuteUpdateAsync(s => s.SetProperty(p => p.IsActive, false));\n    }\n\n    // Кастомні події (аудит, доменні події)\n    public override async Task\u003Cint> SaveChangesAsync(CancellationToken ct = default)\n    {\n        // Автоматично встановити UpdatedAt для Modified entities\n        var modifiedEntries = ChangeTracker.Entries\u003CIHasTimestamps>()\n            .Where(e => e.State == EntityState.Modified);\n\n        foreach (var entry in modifiedEntries)\n            entry.Entity.UpdatedAt = DateTime.UtcNow;\n\n        return await base.SaveChangesAsync(ct);\n    }\n}\n",[2984,3794,3795,3800,3812,3816,3827,3831,3836,3854,3883,3887,3913,3953,3957,3962,3990,3994,4011,4037,4078,4082,4086,4091,4125,4129,4134,4158,4190,4194,4214,4241,4245,4265,4269],{"__ignoreMap":3029},[3063,3796,3797],{"class":3065,"line":3066},[3063,3798,3799],{"class":3069},"// Infrastructure/ShopDbContextExtensions.cs\n",[3063,3801,3802,3804,3806,3808,3810],{"class":3065,"line":3073},[3063,3803,3090],{"class":3089},[3063,3805,3094],{"class":3093},[3063,3807,3010],{"class":3097},[3063,3809,3100],{"class":3093},[3063,3811,3108],{"class":3097},[3063,3813,3814],{"class":3065,"line":3079},[3063,3815,3083],{"emptyLinePlaceholder":3082},[3063,3817,3818,3820,3822,3824],{"class":3065,"line":3086},[3063,3819,3125],{"class":3089},[3063,3821,3128],{"class":3089},[3063,3823,3131],{"class":3089},[3063,3825,3826],{"class":3093}," ShopDbContext\n",[3063,3828,3829],{"class":3065,"line":3111},[3063,3830,3140],{"class":3097},[3063,3832,3833],{"class":3065,"line":3116},[3063,3834,3835],{"class":3069},"    // Зручні методи запитів\n",[3063,3837,3838,3840,3843,3845,3847,3849,3852],{"class":3065,"line":3122},[3063,3839,3152],{"class":3089},[3063,3841,3842],{"class":3093}," IQueryable",[3063,3844,3523],{"class":3097},[3063,3846,3039],{"class":3093},[3063,3848,3529],{"class":3097},[3063,3850,3851],{"class":3158},"ActiveProducts",[3063,3853,3162],{"class":3097},[3063,3855,3856,3859,3861,3863,3866,3869,3871,3873,3875,3877,3879,3881],{"class":3065,"line":3137},[3063,3857,3858],{"class":3283},"        Set",[3063,3860,3523],{"class":3097},[3063,3862,3039],{"class":3093},[3063,3864,3865],{"class":3097},">().",[3063,3867,3868],{"class":3283},"Where",[3063,3870,3287],{"class":3097},[3063,3872,2963],{"class":3158},[3063,3874,3237],{"class":3097},[3063,3876,2963],{"class":3158},[3063,3878,3010],{"class":3097},[3063,3880,3408],{"class":3158},[3063,3882,3364],{"class":3097},[3063,3884,3885],{"class":3065,"line":3143},[3063,3886,3083],{"emptyLinePlaceholder":3082},[3063,3888,3889,3891,3893,3895,3897,3899,3902,3904,3907,3910],{"class":3065,"line":3149},[3063,3890,3152],{"class":3089},[3063,3892,3842],{"class":3093},[3063,3894,3523],{"class":3097},[3063,3896,3039],{"class":3093},[3063,3898,3529],{"class":3097},[3063,3900,3901],{"class":3283},"ProductsByCategory",[3063,3903,3287],{"class":3097},[3063,3905,3906],{"class":3089},"int",[3063,3908,3909],{"class":3158}," categoryId",[3063,3911,3912],{"class":3097},") =>\n",[3063,3914,3915,3917,3919,3921,3923,3925,3927,3929,3931,3933,3935,3937,3940,3943,3945,3947,3949,3951],{"class":3065,"line":3165},[3063,3916,3858],{"class":3283},[3063,3918,3523],{"class":3097},[3063,3920,3039],{"class":3093},[3063,3922,3865],{"class":3097},[3063,3924,3868],{"class":3283},[3063,3926,3287],{"class":3097},[3063,3928,2963],{"class":3158},[3063,3930,3237],{"class":3097},[3063,3932,2963],{"class":3158},[3063,3934,3010],{"class":3097},[3063,3936,3660],{"class":3158},[3063,3938,3939],{"class":3097}," == ",[3063,3941,3942],{"class":3158},"categoryId",[3063,3944,3247],{"class":3097},[3063,3946,2963],{"class":3158},[3063,3948,3010],{"class":3097},[3063,3950,3408],{"class":3158},[3063,3952,3364],{"class":3097},[3063,3954,3955],{"class":3065,"line":3176},[3063,3956,3083],{"emptyLinePlaceholder":3082},[3063,3958,3959],{"class":3065,"line":3211},[3063,3960,3961],{"class":3069},"    // Методи для складних операцій\n",[3063,3963,3964,3966,3969,3972,3974,3976,3978,3981,3983,3985,3988],{"class":3065,"line":3221},[3063,3965,3152],{"class":3089},[3063,3967,3968],{"class":3089}," async",[3063,3970,3971],{"class":3093}," Task",[3063,3973,3523],{"class":3097},[3063,3975,3906],{"class":3089},[3063,3977,3529],{"class":3097},[3063,3979,3980],{"class":3283},"SoftDeleteProductAsync",[3063,3982,3287],{"class":3097},[3063,3984,3906],{"class":3089},[3063,3986,3987],{"class":3158}," productId",[3063,3989,3208],{"class":3097},[3063,3991,3992],{"class":3065,"line":3226},[3063,3993,3301],{"class":3097},[3063,3995,3996,3998,4001,4004,4006,4008],{"class":3065,"line":3264},[3063,3997,3693],{"class":3307},[3063,3999,4000],{"class":3089}," await",[3063,4002,4003],{"class":3283}," Set",[3063,4005,3523],{"class":3097},[3063,4007,3039],{"class":3093},[3063,4009,4010],{"class":3097},">()\n",[3063,4012,4013,4016,4018,4020,4022,4024,4026,4028,4030,4032,4035],{"class":3065,"line":3269},[3063,4014,4015],{"class":3097},"            .",[3063,4017,3868],{"class":3283},[3063,4019,3287],{"class":3097},[3063,4021,2963],{"class":3158},[3063,4023,3237],{"class":3097},[3063,4025,2963],{"class":3158},[3063,4027,3010],{"class":3097},[3063,4029,3740],{"class":3158},[3063,4031,3939],{"class":3097},[3063,4033,4034],{"class":3158},"productId",[3063,4036,3208],{"class":3097},[3063,4038,4039,4041,4044,4046,4049,4051,4053,4055,4058,4060,4062,4064,4066,4068,4070,4073,4075],{"class":3065,"line":3275},[3063,4040,4015],{"class":3097},[3063,4042,4043],{"class":3283},"ExecuteUpdateAsync",[3063,4045,3287],{"class":3097},[3063,4047,4048],{"class":3158},"s",[3063,4050,3237],{"class":3097},[3063,4052,4048],{"class":3158},[3063,4054,3010],{"class":3097},[3063,4056,4057],{"class":3283},"SetProperty",[3063,4059,3287],{"class":3097},[3063,4061,2963],{"class":3158},[3063,4063,3237],{"class":3097},[3063,4065,2963],{"class":3158},[3063,4067,3010],{"class":3097},[3063,4069,3408],{"class":3158},[3063,4071,4072],{"class":3097},", ",[3063,4074,3435],{"class":3089},[3063,4076,4077],{"class":3097},"));\n",[3063,4079,4080],{"class":3065,"line":3298},[3063,4081,3387],{"class":3097},[3063,4083,4084],{"class":3065,"line":3304},[3063,4085,3083],{"emptyLinePlaceholder":3082},[3063,4087,4088],{"class":3065,"line":3333},[3063,4089,4090],{"class":3069},"    // Кастомні події (аудит, доменні події)\n",[3063,4092,4093,4095,4097,4099,4101,4103,4105,4107,4110,4112,4115,4118,4120,4123],{"class":3065,"line":3357},[3063,4094,3152],{"class":3089},[3063,4096,3721],{"class":3089},[3063,4098,3968],{"class":3089},[3063,4100,3971],{"class":3093},[3063,4102,3523],{"class":3097},[3063,4104,3906],{"class":3089},[3063,4106,3529],{"class":3097},[3063,4108,4109],{"class":3283},"SaveChangesAsync",[3063,4111,3287],{"class":3097},[3063,4113,4114],{"class":3093},"CancellationToken",[3063,4116,4117],{"class":3158}," ct",[3063,4119,3377],{"class":3097},[3063,4121,4122],{"class":3089},"default",[3063,4124,3208],{"class":3097},[3063,4126,4127],{"class":3065,"line":3367},[3063,4128,3301],{"class":3097},[3063,4130,4131],{"class":3065,"line":3372},[3063,4132,4133],{"class":3069},"        // Автоматично встановити UpdatedAt для Modified entities\n",[3063,4135,4136,4138,4141,4143,4146,4148,4151,4153,4156],{"class":3065,"line":3384},[3063,4137,3546],{"class":3089},[3063,4139,4140],{"class":3158}," modifiedEntries",[3063,4142,3377],{"class":3097},[3063,4144,4145],{"class":3158},"ChangeTracker",[3063,4147,3010],{"class":3097},[3063,4149,4150],{"class":3283},"Entries",[3063,4152,3523],{"class":3097},[3063,4154,4155],{"class":3093},"IHasTimestamps",[3063,4157,4010],{"class":3097},[3063,4159,4160,4162,4164,4166,4169,4171,4173,4175,4178,4180,4183,4185,4188],{"class":3065,"line":3390},[3063,4161,4015],{"class":3097},[3063,4163,3868],{"class":3283},[3063,4165,3287],{"class":3097},[3063,4167,4168],{"class":3158},"e",[3063,4170,3237],{"class":3097},[3063,4172,4168],{"class":3158},[3063,4174,3010],{"class":3097},[3063,4176,4177],{"class":3158},"State",[3063,4179,3939],{"class":3097},[3063,4181,4182],{"class":3158},"EntityState",[3063,4184,3010],{"class":3097},[3063,4186,4187],{"class":3158},"Modified",[3063,4189,3364],{"class":3097},[3063,4191,4192],{"class":3065,"line":3395},[3063,4193,3083],{"emptyLinePlaceholder":3082},[3063,4195,4196,4199,4201,4204,4207,4210,4212],{"class":3065,"line":3418},[3063,4197,4198],{"class":3307},"        foreach",[3063,4200,3311],{"class":3097},[3063,4202,4203],{"class":3089},"var",[3063,4205,4206],{"class":3158}," entry",[3063,4208,4209],{"class":3307}," in",[3063,4211,4140],{"class":3158},[3063,4213,3208],{"class":3097},[3063,4215,4216,4219,4221,4224,4226,4229,4231,4234,4236,4239],{"class":3065,"line":3440},[3063,4217,4218],{"class":3158},"            entry",[3063,4220,3010],{"class":3097},[3063,4222,4223],{"class":3158},"Entity",[3063,4225,3010],{"class":3097},[3063,4227,4228],{"class":3158},"UpdatedAt",[3063,4230,3377],{"class":3097},[3063,4232,4233],{"class":3158},"DateTime",[3063,4235,3010],{"class":3097},[3063,4237,4238],{"class":3158},"UtcNow",[3063,4240,3108],{"class":3097},[3063,4242,4243],{"class":3065,"line":3445},[3063,4244,3083],{"emptyLinePlaceholder":3082},[3063,4246,4247,4249,4251,4254,4256,4258,4260,4263],{"class":3065,"line":3451},[3063,4248,3693],{"class":3307},[3063,4250,4000],{"class":3089},[3063,4252,4253],{"class":3089}," base",[3063,4255,3010],{"class":3097},[3063,4257,4109],{"class":3283},[3063,4259,3287],{"class":3097},[3063,4261,4262],{"class":3158},"ct",[3063,4264,3364],{"class":3097},[3063,4266,4267],{"class":3065,"line":3504},[3063,4268,3387],{"class":3097},[3063,4270,4271],{"class":3065,"line":3509},[3063,4272,3785],{"class":3097},[3041,4274,4276],{"id":4275},"interface-based-розширення","Interface-based розширення",[2963,4278,4279],{},"Ще потужніший підхід — додати інтерфейси до entity через partial class:",[3021,4281,4283],{"className":3057,"code":4282,"language":3059,"meta":3029,"style":3029},"// Core/Interfaces/IAuditableEntity.cs\npublic interface IAuditableEntity\n{\n    DateTime CreatedAt { get; set; }\n    DateTime? UpdatedAt { get; set; }\n}\n\n// Infrastructure/Entities/Extensions/Product.Extensions.cs\npublic partial class Product : IAuditableEntity, ISoftDeletable\n{\n    // Product вже має CreatedAt і UpdatedAt (від scaffold)\n    // Тепер вони виконують контракт IAuditableEntity\n\n    // ISoftDeletable: реалізація через IsActive поле\n    bool ISoftDeletable.IsDeleted => !IsActive;\n    void ISoftDeletable.SoftDelete() => IsActive = false;\n    void ISoftDeletable.Restore()    => IsActive = true;\n}\n",[2984,4284,4285,4290,4300,4304,4327,4346,4350,4354,4358,4380,4384,4389,4394,4398,4403,4423,4445,4467],{"__ignoreMap":3029},[3063,4286,4287],{"class":3065,"line":3066},[3063,4288,4289],{"class":3069},"// Core/Interfaces/IAuditableEntity.cs\n",[3063,4291,4292,4294,4297],{"class":3065,"line":3073},[3063,4293,3125],{"class":3089},[3063,4295,4296],{"class":3089}," interface",[3063,4298,4299],{"class":3093}," IAuditableEntity\n",[3063,4301,4302],{"class":3065,"line":3079},[3063,4303,3140],{"class":3097},[3063,4305,4306,4309,4312,4315,4318,4321,4324],{"class":3065,"line":3086},[3063,4307,4308],{"class":3093},"    DateTime",[3063,4310,4311],{"class":3158}," CreatedAt",[3063,4313,4314],{"class":3097}," { ",[3063,4316,4317],{"class":3089},"get",[3063,4319,4320],{"class":3097},"; ",[3063,4322,4323],{"class":3089},"set",[3063,4325,4326],{"class":3097},"; }\n",[3063,4328,4329,4331,4334,4336,4338,4340,4342,4344],{"class":3065,"line":3111},[3063,4330,4308],{"class":3093},[3063,4332,4333],{"class":3097},"? ",[3063,4335,4228],{"class":3158},[3063,4337,4314],{"class":3097},[3063,4339,4317],{"class":3089},[3063,4341,4320],{"class":3097},[3063,4343,4323],{"class":3089},[3063,4345,4326],{"class":3097},[3063,4347,4348],{"class":3065,"line":3116},[3063,4349,3785],{"class":3097},[3063,4351,4352],{"class":3065,"line":3122},[3063,4353,3083],{"emptyLinePlaceholder":3082},[3063,4355,4356],{"class":3065,"line":3137},[3063,4357,3070],{"class":3069},[3063,4359,4360,4362,4364,4366,4369,4372,4375,4377],{"class":3065,"line":3143},[3063,4361,3125],{"class":3089},[3063,4363,3128],{"class":3089},[3063,4365,3131],{"class":3089},[3063,4367,4368],{"class":3093}," Product",[3063,4370,4371],{"class":3097}," : ",[3063,4373,4374],{"class":3093},"IAuditableEntity",[3063,4376,4072],{"class":3097},[3063,4378,4379],{"class":3093},"ISoftDeletable\n",[3063,4381,4382],{"class":3065,"line":3149},[3063,4383,3140],{"class":3097},[3063,4385,4386],{"class":3065,"line":3165},[3063,4387,4388],{"class":3069},"    // Product вже має CreatedAt і UpdatedAt (від scaffold)\n",[3063,4390,4391],{"class":3065,"line":3176},[3063,4392,4393],{"class":3069},"    // Тепер вони виконують контракт IAuditableEntity\n",[3063,4395,4396],{"class":3065,"line":3211},[3063,4397,3083],{"emptyLinePlaceholder":3082},[3063,4399,4400],{"class":3065,"line":3221},[3063,4401,4402],{"class":3069},"    // ISoftDeletable: реалізація через IsActive поле\n",[3063,4404,4405,4408,4411,4413,4416,4419,4421],{"class":3065,"line":3226},[3063,4406,4407],{"class":3089},"    bool",[3063,4409,4410],{"class":3093}," ISoftDeletable",[3063,4412,3010],{"class":3097},[3063,4414,4415],{"class":3158},"IsDeleted",[3063,4417,4418],{"class":3097}," => !",[3063,4420,3408],{"class":3158},[3063,4422,3108],{"class":3097},[3063,4424,4425,4428,4430,4432,4435,4437,4439,4441,4443],{"class":3065,"line":3264},[3063,4426,4427],{"class":3089},"    void",[3063,4429,4410],{"class":3093},[3063,4431,3010],{"class":3097},[3063,4433,4434],{"class":3283},"SoftDelete",[3063,4436,3428],{"class":3097},[3063,4438,3408],{"class":3158},[3063,4440,3377],{"class":3097},[3063,4442,3435],{"class":3089},[3063,4444,3108],{"class":3097},[3063,4446,4447,4449,4451,4453,4456,4459,4461,4463,4465],{"class":3065,"line":3269},[3063,4448,4427],{"class":3089},[3063,4450,4410],{"class":3093},[3063,4452,3010],{"class":3097},[3063,4454,4455],{"class":3283},"Restore",[3063,4457,4458],{"class":3097},"()    => ",[3063,4460,3408],{"class":3158},[3063,4462,3377],{"class":3097},[3063,4464,3413],{"class":3089},[3063,4466,3108],{"class":3097},[3063,4468,4469],{"class":3065,"line":3275},[3063,4470,3785],{"class":3097},[3021,4472,4474],{"className":3057,"code":4473,"language":3059,"meta":3029,"style":3029},"// Global Query Filter у DbContext extensions:\npublic partial class ShopDbContext\n{\n    protected override void OnModelCreating(ModelBuilder modelBuilder)\n    {\n        // Викликаємо базовий OnModelCreating (згенерований)\n        base.OnModelCreating(modelBuilder);\n\n        // Додаємо Global Query Filter для soft delete\n        foreach (var entityType in modelBuilder.Model.GetEntityTypes())\n        {\n            if (typeof(ISoftDeletable).IsAssignableFrom(entityType.ClrType))\n            {\n                var method = typeof(ShopDbContextExtensions)\n                    .GetMethod(nameof(AddSoftDeleteFilter))!\n                    .MakeGenericMethod(entityType.ClrType);\n\n                method.Invoke(null, [modelBuilder]);\n            }\n        }\n    }\n\n    private static void AddSoftDeleteFilter\u003CT>(ModelBuilder builder)\n        where T : class, ISoftDeletable\n    {\n        builder.Entity\u003CT>().HasQueryFilter(e => !e.IsDeleted);\n    }\n}\n",[2984,4475,4476,4481,4491,4495,4517,4521,4526,4543,4547,4552,4580,4585,4618,4623,4642,4662,4679,4683,4706,4711,4716,4720,4724,4752,4769,4773,4805,4809],{"__ignoreMap":3029},[3063,4477,4478],{"class":3065,"line":3066},[3063,4479,4480],{"class":3069},"// Global Query Filter у DbContext extensions:\n",[3063,4482,4483,4485,4487,4489],{"class":3065,"line":3073},[3063,4484,3125],{"class":3089},[3063,4486,3128],{"class":3089},[3063,4488,3131],{"class":3089},[3063,4490,3826],{"class":3093},[3063,4492,4493],{"class":3065,"line":3079},[3063,4494,3140],{"class":3097},[3063,4496,4497,4500,4502,4504,4507,4509,4512,4515],{"class":3065,"line":3086},[3063,4498,4499],{"class":3089},"    protected",[3063,4501,3721],{"class":3089},[3063,4503,3280],{"class":3089},[3063,4505,4506],{"class":3283}," OnModelCreating",[3063,4508,3287],{"class":3097},[3063,4510,4511],{"class":3093},"ModelBuilder",[3063,4513,4514],{"class":3158}," modelBuilder",[3063,4516,3208],{"class":3097},[3063,4518,4519],{"class":3065,"line":3111},[3063,4520,3301],{"class":3097},[3063,4522,4523],{"class":3065,"line":3116},[3063,4524,4525],{"class":3069},"        // Викликаємо базовий OnModelCreating (згенерований)\n",[3063,4527,4528,4531,4533,4536,4538,4541],{"class":3065,"line":3122},[3063,4529,4530],{"class":3089},"        base",[3063,4532,3010],{"class":3097},[3063,4534,4535],{"class":3283},"OnModelCreating",[3063,4537,3287],{"class":3097},[3063,4539,4540],{"class":3158},"modelBuilder",[3063,4542,3364],{"class":3097},[3063,4544,4545],{"class":3065,"line":3137},[3063,4546,3083],{"emptyLinePlaceholder":3082},[3063,4548,4549],{"class":3065,"line":3143},[3063,4550,4551],{"class":3069},"        // Додаємо Global Query Filter для soft delete\n",[3063,4553,4554,4556,4558,4560,4563,4565,4567,4569,4572,4574,4577],{"class":3065,"line":3149},[3063,4555,4198],{"class":3307},[3063,4557,3311],{"class":3097},[3063,4559,4203],{"class":3089},[3063,4561,4562],{"class":3158}," entityType",[3063,4564,4209],{"class":3307},[3063,4566,4514],{"class":3158},[3063,4568,3010],{"class":3097},[3063,4570,4571],{"class":3158},"Model",[3063,4573,3010],{"class":3097},[3063,4575,4576],{"class":3283},"GetEntityTypes",[3063,4578,4579],{"class":3097},"())\n",[3063,4581,4582],{"class":3065,"line":3165},[3063,4583,4584],{"class":3097},"        {\n",[3063,4586,4587,4590,4592,4595,4597,4600,4603,4606,4608,4611,4613,4616],{"class":3065,"line":3176},[3063,4588,4589],{"class":3307},"            if",[3063,4591,3311],{"class":3097},[3063,4593,4594],{"class":3089},"typeof",[3063,4596,3287],{"class":3097},[3063,4598,4599],{"class":3093},"ISoftDeletable",[3063,4601,4602],{"class":3097},").",[3063,4604,4605],{"class":3283},"IsAssignableFrom",[3063,4607,3287],{"class":3097},[3063,4609,4610],{"class":3158},"entityType",[3063,4612,3010],{"class":3097},[3063,4614,4615],{"class":3158},"ClrType",[3063,4617,3590],{"class":3097},[3063,4619,4620],{"class":3065,"line":3211},[3063,4621,4622],{"class":3097},"            {\n",[3063,4624,4625,4628,4631,4633,4635,4637,4640],{"class":3065,"line":3221},[3063,4626,4627],{"class":3089},"                var",[3063,4629,4630],{"class":3158}," method",[3063,4632,3377],{"class":3097},[3063,4634,4594],{"class":3089},[3063,4636,3287],{"class":3097},[3063,4638,4639],{"class":3093},"ShopDbContextExtensions",[3063,4641,3208],{"class":3097},[3063,4643,4644,4647,4650,4652,4654,4656,4659],{"class":3065,"line":3226},[3063,4645,4646],{"class":3097},"                    .",[3063,4648,4649],{"class":3283},"GetMethod",[3063,4651,3287],{"class":3097},[3063,4653,3347],{"class":3089},[3063,4655,3287],{"class":3097},[3063,4657,4658],{"class":3158},"AddSoftDeleteFilter",[3063,4660,4661],{"class":3097},"))!\n",[3063,4663,4664,4666,4669,4671,4673,4675,4677],{"class":3065,"line":3264},[3063,4665,4646],{"class":3097},[3063,4667,4668],{"class":3283},"MakeGenericMethod",[3063,4670,3287],{"class":3097},[3063,4672,4610],{"class":3158},[3063,4674,3010],{"class":3097},[3063,4676,4615],{"class":3158},[3063,4678,3364],{"class":3097},[3063,4680,4681],{"class":3065,"line":3269},[3063,4682,3083],{"emptyLinePlaceholder":3082},[3063,4684,4685,4688,4690,4693,4695,4698,4701,4703],{"class":3065,"line":3275},[3063,4686,4687],{"class":3158},"                method",[3063,4689,3010],{"class":3097},[3063,4691,4692],{"class":3283},"Invoke",[3063,4694,3287],{"class":3097},[3063,4696,4697],{"class":3089},"null",[3063,4699,4700],{"class":3097},", [",[3063,4702,4540],{"class":3158},[3063,4704,4705],{"class":3097},"]);\n",[3063,4707,4708],{"class":3065,"line":3298},[3063,4709,4710],{"class":3097},"            }\n",[3063,4712,4713],{"class":3065,"line":3304},[3063,4714,4715],{"class":3097},"        }\n",[3063,4717,4718],{"class":3065,"line":3333},[3063,4719,3387],{"class":3097},[3063,4721,4722],{"class":3065,"line":3357},[3063,4723,3083],{"emptyLinePlaceholder":3082},[3063,4725,4726,4729,4732,4734,4737,4739,4742,4745,4747,4750],{"class":3065,"line":3367},[3063,4727,4728],{"class":3089},"    private",[3063,4730,4731],{"class":3089}," static",[3063,4733,3280],{"class":3089},[3063,4735,4736],{"class":3283}," AddSoftDeleteFilter",[3063,4738,3523],{"class":3097},[3063,4740,4741],{"class":3093},"T",[3063,4743,4744],{"class":3097},">(",[3063,4746,4511],{"class":3093},[3063,4748,4749],{"class":3158}," builder",[3063,4751,3208],{"class":3097},[3063,4753,4754,4757,4760,4762,4765,4767],{"class":3065,"line":3372},[3063,4755,4756],{"class":3089},"        where",[3063,4758,4759],{"class":3093}," T",[3063,4761,4371],{"class":3097},[3063,4763,4764],{"class":3089},"class",[3063,4766,4072],{"class":3097},[3063,4768,4379],{"class":3093},[3063,4770,4771],{"class":3065,"line":3384},[3063,4772,3301],{"class":3097},[3063,4774,4775,4778,4780,4782,4784,4786,4788,4791,4793,4795,4797,4799,4801,4803],{"class":3065,"line":3390},[3063,4776,4777],{"class":3158},"        builder",[3063,4779,3010],{"class":3097},[3063,4781,4223],{"class":3283},[3063,4783,3523],{"class":3097},[3063,4785,4741],{"class":3093},[3063,4787,3865],{"class":3097},[3063,4789,4790],{"class":3283},"HasQueryFilter",[3063,4792,3287],{"class":3097},[3063,4794,4168],{"class":3158},[3063,4796,4418],{"class":3097},[3063,4798,4168],{"class":3158},[3063,4800,3010],{"class":3097},[3063,4802,4415],{"class":3158},[3063,4804,3364],{"class":3097},[3063,4806,4807],{"class":3065,"line":3395},[3063,4808,3387],{"class":3097},[3063,4810,4811],{"class":3065,"line":3418},[3063,4812,3785],{"class":3097},[4814,4815,4816,4817,4819,4820,4823],"warning",{},"При override ",[2984,4818,4535],{}," у partial ShopDbContext — обов'язково викликайте ",[2984,4821,4822],{},"base.OnModelCreating(modelBuilder)",". Без цього — конфігурація зі згенерованого файлу не застосується і застосунок впаде з некоректним маппінгом.",[2973,4825],{},[2976,4827,4829],{"id":4828},"re-scaffolding-workflow-коли-бд-змінилась","Re-scaffolding Workflow: коли БД змінилась",[2963,4831,4832],{},"Re-scaffolding — регенерація entity при зміні схеми БД. Це повторюваний процес у Database-First підході і важливо виробити правильний workflow.",[3041,4834,4836],{"id":4835},"коли-потрібен-re-scaffold","Коли потрібен re-scaffold",[4838,4839,4840,4844,4855,4858],"ul",{},[4841,4842,4843],"li",{},"DBA додав нову таблицю або нові стовпці до існуючої",[4841,4845,4846,4847,4850,4851,4854],{},"Змінились типи стовпців (наприклад ",[2984,4848,4849],{},"NVARCHAR(100)"," → ",[2984,4852,4853],{},"NVARCHAR(500)",")",[4841,4856,4857],{},"Додано нові Foreign Keys або змінено ON DELETE поведінку",[4841,4859,4860],{},"Видалено або перейменовано стовпці",[3041,4862,4864],{"id":4863},"базова-команда-re-scaffold","Базова команда re-scaffold",[3021,4866,4870],{"className":4867,"code":4868,"language":4869,"meta":3029,"style":3029},"language-bash shiki shiki-themes light-plus dark-plus dark-plus","# --force: перезаписати наявні ЗГЕНЕРОВАНІ файли\ndotnet ef dbcontext scaffold \\\n    \"Name=ConnectionStrings:DefaultConnection\" \\\n    Microsoft.EntityFrameworkCore.SqlServer \\\n    --output-dir Infrastructure/Entities \\\n    --context-dir Infrastructure \\\n    --context ShopDbContext \\\n    --no-onconfiguring \\\n    --force   # ← ключовий флаг: перезаписати\n","bash",[2984,4871,4872,4877,4895,4902,4909,4919,4929,4939,4946],{"__ignoreMap":3029},[3063,4873,4874],{"class":3065,"line":3066},[3063,4875,4876],{"class":3069},"# --force: перезаписати наявні ЗГЕНЕРОВАНІ файли\n",[3063,4878,4879,4882,4885,4888,4891],{"class":3065,"line":3073},[3063,4880,4881],{"class":3283},"dotnet",[3063,4883,4884],{"class":3360}," ef",[3063,4886,4887],{"class":3360}," dbcontext",[3063,4889,4890],{"class":3360}," scaffold",[3063,4892,4894],{"class":4893},"sjcCO"," \\\n",[3063,4896,4897,4900],{"class":3065,"line":3079},[3063,4898,4899],{"class":3360},"    \"Name=ConnectionStrings:DefaultConnection\"",[3063,4901,4894],{"class":4893},[3063,4903,4904,4907],{"class":3065,"line":3086},[3063,4905,4906],{"class":3360},"    Microsoft.EntityFrameworkCore.SqlServer",[3063,4908,4894],{"class":4893},[3063,4910,4911,4914,4917],{"class":3065,"line":3111},[3063,4912,4913],{"class":3089},"    --output-dir",[3063,4915,4916],{"class":3360}," Infrastructure/Entities",[3063,4918,4894],{"class":4893},[3063,4920,4921,4924,4927],{"class":3065,"line":3116},[3063,4922,4923],{"class":3089},"    --context-dir",[3063,4925,4926],{"class":3360}," Infrastructure",[3063,4928,4894],{"class":4893},[3063,4930,4931,4934,4937],{"class":3065,"line":3122},[3063,4932,4933],{"class":3089},"    --context",[3063,4935,4936],{"class":3360}," ShopDbContext",[3063,4938,4894],{"class":4893},[3063,4940,4941,4944],{"class":3065,"line":3137},[3063,4942,4943],{"class":3089},"    --no-onconfiguring",[3063,4945,4894],{"class":4893},[3063,4947,4948,4951],{"class":3065,"line":3143},[3063,4949,4950],{"class":3089},"    --force",[3063,4952,4953],{"class":3069},"   # ← ключовий флаг: перезаписати\n",[4955,4956,4957,4959,4960,4963,4964,4967,4968,4971,4972,4975,4976,4978,4979,4982],"caution",{},[2984,4958,3005],{}," перезаписує ",[2991,4961,4962],{},"всі"," файли у ",[2984,4965,4966],{},"--output-dir"," і ",[2984,4969,4970],{},"--context-dir",". Якщо ваш ",[2984,4973,4974],{},"Product.Extensions.cs"," знаходиться у тій самій директорії що і ",[2984,4977,2997],{}," — він ",[2991,4980,4981],{},"буде перезаписаний","! Завжди тримайте Extensions у окремій піддиректорії або окремому проєкті.",[3041,4984,4986],{"id":4985},"безпечна-структура-директорій","Безпечна структура директорій",[3021,4988,4991],{"className":4989,"code":4990,"language":3026},[3024],"Infrastructure/\n├── Generated/           ← Тут живуть автогенеровані файли\n│   ├── Entities/        ← --output-dir Infrastructure/Generated/Entities\n│   │   ├── Product.cs\n│   │   ├── Category.cs\n│   │   └── ...\n│   └── ShopDbContext.cs ← --context-dir Infrastructure/Generated\n└── Extensions/          ← Тут живуть ваші розширення (НЕ торкається --force)\n    ├── Product.Extensions.cs\n    ├── Category.Extensions.cs\n    └── ShopDbContextExtensions.cs\n",[2984,4992,4990],{"__ignoreMap":3029},[3041,4994,4996],{"id":4995},"автоматизований-re-scaffold-скрипт","Автоматизований re-scaffold скрипт",[3021,4998,5000],{"className":4867,"code":4999,"language":4869,"meta":3029,"style":3029},"#!/bin/bash\n# scripts/rescaffold.sh\n\nset -e  # зупинити при помилці\n\necho \"🔄 Starting re-scaffold...\"\n\n# Параметри\nCONNECTION=\"Name=ConnectionStrings:DefaultConnection\"\nPROVIDER=\"Microsoft.EntityFrameworkCore.SqlServer\"\nPROJECT=\"src/Infrastructure\"\nSTARTUP=\"src/Api\"\n\n# Виконати scaffold\ndotnet ef dbcontext scaffold \"$CONNECTION\" \"$PROVIDER\" \\\n    --project \"$PROJECT\" \\\n    --startup-project \"$STARTUP\" \\\n    --output-dir \"Generated/Entities\" \\\n    --context-dir \"Generated\" \\\n    --context \"ShopDbContext\" \\\n    --namespace \"YourApp.Infrastructure.Generated.Entities\" \\\n    --context-namespace \"YourApp.Infrastructure.Generated\" \\\n    --no-onconfiguring \\\n    --force\n\necho \"✅ Scaffold completed!\"\necho \"\"\necho \"⚠️  Post-scaffold checklist:\"\necho \"   1. Review git diff for any breaking changes\"\necho \"   2. Check if new entities need Extensions classes\"\necho \"   3. Check if removed columns affect Extensions code\"\necho \"   4. Run tests: dotnet test\"\n",[2984,5001,5002,5007,5012,5016,5026,5030,5038,5042,5047,5058,5068,5078,5088,5092,5097,5125,5139,5153,5162,5171,5180,5190,5200,5206,5211,5215,5222,5229,5236,5243,5250,5257],{"__ignoreMap":3029},[3063,5003,5004],{"class":3065,"line":3066},[3063,5005,5006],{"class":3069},"#!/bin/bash\n",[3063,5008,5009],{"class":3065,"line":3073},[3063,5010,5011],{"class":3069},"# scripts/rescaffold.sh\n",[3063,5013,5014],{"class":3065,"line":3079},[3063,5015,3083],{"emptyLinePlaceholder":3082},[3063,5017,5018,5020,5023],{"class":3065,"line":3086},[3063,5019,4323],{"class":3283},[3063,5021,5022],{"class":3089}," -e",[3063,5024,5025],{"class":3069},"  # зупинити при помилці\n",[3063,5027,5028],{"class":3065,"line":3111},[3063,5029,3083],{"emptyLinePlaceholder":3082},[3063,5031,5032,5035],{"class":3065,"line":3116},[3063,5033,5034],{"class":3283},"echo",[3063,5036,5037],{"class":3360}," \"🔄 Starting re-scaffold...\"\n",[3063,5039,5040],{"class":3065,"line":3122},[3063,5041,3083],{"emptyLinePlaceholder":3082},[3063,5043,5044],{"class":3065,"line":3137},[3063,5045,5046],{"class":3069},"# Параметри\n",[3063,5048,5049,5052,5055],{"class":3065,"line":3143},[3063,5050,5051],{"class":3158},"CONNECTION",[3063,5053,5054],{"class":3097},"=",[3063,5056,5057],{"class":3360},"\"Name=ConnectionStrings:DefaultConnection\"\n",[3063,5059,5060,5063,5065],{"class":3065,"line":3149},[3063,5061,5062],{"class":3158},"PROVIDER",[3063,5064,5054],{"class":3097},[3063,5066,5067],{"class":3360},"\"Microsoft.EntityFrameworkCore.SqlServer\"\n",[3063,5069,5070,5073,5075],{"class":3065,"line":3165},[3063,5071,5072],{"class":3158},"PROJECT",[3063,5074,5054],{"class":3097},[3063,5076,5077],{"class":3360},"\"src/Infrastructure\"\n",[3063,5079,5080,5083,5085],{"class":3065,"line":3176},[3063,5081,5082],{"class":3158},"STARTUP",[3063,5084,5054],{"class":3097},[3063,5086,5087],{"class":3360},"\"src/Api\"\n",[3063,5089,5090],{"class":3065,"line":3211},[3063,5091,3083],{"emptyLinePlaceholder":3082},[3063,5093,5094],{"class":3065,"line":3221},[3063,5095,5096],{"class":3069},"# Виконати scaffold\n",[3063,5098,5099,5101,5103,5105,5107,5110,5113,5116,5118,5121,5123],{"class":3065,"line":3226},[3063,5100,4881],{"class":3283},[3063,5102,4884],{"class":3360},[3063,5104,4887],{"class":3360},[3063,5106,4890],{"class":3360},[3063,5108,5109],{"class":3360}," \"",[3063,5111,5112],{"class":3158},"$CONNECTION",[3063,5114,5115],{"class":3360},"\"",[3063,5117,5109],{"class":3360},[3063,5119,5120],{"class":3158},"$PROVIDER",[3063,5122,5115],{"class":3360},[3063,5124,4894],{"class":4893},[3063,5126,5127,5130,5132,5135,5137],{"class":3065,"line":3264},[3063,5128,5129],{"class":3089},"    --project",[3063,5131,5109],{"class":3360},[3063,5133,5134],{"class":3158},"$PROJECT",[3063,5136,5115],{"class":3360},[3063,5138,4894],{"class":4893},[3063,5140,5141,5144,5146,5149,5151],{"class":3065,"line":3269},[3063,5142,5143],{"class":3089},"    --startup-project",[3063,5145,5109],{"class":3360},[3063,5147,5148],{"class":3158},"$STARTUP",[3063,5150,5115],{"class":3360},[3063,5152,4894],{"class":4893},[3063,5154,5155,5157,5160],{"class":3065,"line":3275},[3063,5156,4913],{"class":3089},[3063,5158,5159],{"class":3360}," \"Generated/Entities\"",[3063,5161,4894],{"class":4893},[3063,5163,5164,5166,5169],{"class":3065,"line":3298},[3063,5165,4923],{"class":3089},[3063,5167,5168],{"class":3360}," \"Generated\"",[3063,5170,4894],{"class":4893},[3063,5172,5173,5175,5178],{"class":3065,"line":3304},[3063,5174,4933],{"class":3089},[3063,5176,5177],{"class":3360}," \"ShopDbContext\"",[3063,5179,4894],{"class":4893},[3063,5181,5182,5185,5188],{"class":3065,"line":3333},[3063,5183,5184],{"class":3089},"    --namespace",[3063,5186,5187],{"class":3360}," \"YourApp.Infrastructure.Generated.Entities\"",[3063,5189,4894],{"class":4893},[3063,5191,5192,5195,5198],{"class":3065,"line":3357},[3063,5193,5194],{"class":3089},"    --context-namespace",[3063,5196,5197],{"class":3360}," \"YourApp.Infrastructure.Generated\"",[3063,5199,4894],{"class":4893},[3063,5201,5202,5204],{"class":3065,"line":3367},[3063,5203,4943],{"class":3089},[3063,5205,4894],{"class":4893},[3063,5207,5208],{"class":3065,"line":3372},[3063,5209,5210],{"class":3089},"    --force\n",[3063,5212,5213],{"class":3065,"line":3384},[3063,5214,3083],{"emptyLinePlaceholder":3082},[3063,5216,5217,5219],{"class":3065,"line":3390},[3063,5218,5034],{"class":3283},[3063,5220,5221],{"class":3360}," \"✅ Scaffold completed!\"\n",[3063,5223,5224,5226],{"class":3065,"line":3395},[3063,5225,5034],{"class":3283},[3063,5227,5228],{"class":3360}," \"\"\n",[3063,5230,5231,5233],{"class":3065,"line":3418},[3063,5232,5034],{"class":3283},[3063,5234,5235],{"class":3360}," \"⚠️  Post-scaffold checklist:\"\n",[3063,5237,5238,5240],{"class":3065,"line":3440},[3063,5239,5034],{"class":3283},[3063,5241,5242],{"class":3360}," \"   1. Review git diff for any breaking changes\"\n",[3063,5244,5245,5247],{"class":3065,"line":3445},[3063,5246,5034],{"class":3283},[3063,5248,5249],{"class":3360}," \"   2. Check if new entities need Extensions classes\"\n",[3063,5251,5252,5254],{"class":3065,"line":3451},[3063,5253,5034],{"class":3283},[3063,5255,5256],{"class":3360}," \"   3. Check if removed columns affect Extensions code\"\n",[3063,5258,5259,5261],{"class":3065,"line":3504},[3063,5260,5034],{"class":3283},[3063,5262,5263],{"class":3360}," \"   4. Run tests: dotnet test\"\n",[3041,5265,5267],{"id":5266},"diff-driven-re-scaffold-workflow","Diff-driven re-scaffold workflow",[3021,5269,5271],{"className":4867,"code":5270,"language":4869,"meta":3029,"style":3029},"# 1. Зберегти поточний стан у Git перед scaffold\ngit add .\ngit commit -m \"chore: pre-rescaffold checkpoint\"\n\n# 2. Виконати re-scaffold\n./scripts/rescaffold.sh\n\n# 3. Перевірити що змінилось\ngit diff --stat\n# Показує які файли змінились і скільки рядків\n\n# 4. Детальний diff для review\ngit diff Infrastructure/Generated/\n\n# 5. Перевірити що Extensions не зачіпнуті\ngit diff Infrastructure/Extensions/\n# Повинно бути: нічого (порожньо)\n\n# 6. Запустити тести\ndotnet test\n\n# 7. Закомітити якщо все в порядку\ngit add .\ngit commit -m \"chore: re-scaffold after DB schema update v3.2\"\n",[2984,5272,5273,5278,5289,5302,5306,5311,5316,5320,5325,5335,5340,5344,5349,5358,5362,5367,5376,5381,5385,5390,5397,5401,5406,5414],{"__ignoreMap":3029},[3063,5274,5275],{"class":3065,"line":3066},[3063,5276,5277],{"class":3069},"# 1. Зберегти поточний стан у Git перед scaffold\n",[3063,5279,5280,5283,5286],{"class":3065,"line":3073},[3063,5281,5282],{"class":3283},"git",[3063,5284,5285],{"class":3360}," add",[3063,5287,5288],{"class":3360}," .\n",[3063,5290,5291,5293,5296,5299],{"class":3065,"line":3079},[3063,5292,5282],{"class":3283},[3063,5294,5295],{"class":3360}," commit",[3063,5297,5298],{"class":3089}," -m",[3063,5300,5301],{"class":3360}," \"chore: pre-rescaffold checkpoint\"\n",[3063,5303,5304],{"class":3065,"line":3086},[3063,5305,3083],{"emptyLinePlaceholder":3082},[3063,5307,5308],{"class":3065,"line":3111},[3063,5309,5310],{"class":3069},"# 2. Виконати re-scaffold\n",[3063,5312,5313],{"class":3065,"line":3116},[3063,5314,5315],{"class":3283},"./scripts/rescaffold.sh\n",[3063,5317,5318],{"class":3065,"line":3122},[3063,5319,3083],{"emptyLinePlaceholder":3082},[3063,5321,5322],{"class":3065,"line":3137},[3063,5323,5324],{"class":3069},"# 3. Перевірити що змінилось\n",[3063,5326,5327,5329,5332],{"class":3065,"line":3143},[3063,5328,5282],{"class":3283},[3063,5330,5331],{"class":3360}," diff",[3063,5333,5334],{"class":3089}," --stat\n",[3063,5336,5337],{"class":3065,"line":3149},[3063,5338,5339],{"class":3069},"# Показує які файли змінились і скільки рядків\n",[3063,5341,5342],{"class":3065,"line":3165},[3063,5343,3083],{"emptyLinePlaceholder":3082},[3063,5345,5346],{"class":3065,"line":3176},[3063,5347,5348],{"class":3069},"# 4. Детальний diff для review\n",[3063,5350,5351,5353,5355],{"class":3065,"line":3211},[3063,5352,5282],{"class":3283},[3063,5354,5331],{"class":3360},[3063,5356,5357],{"class":3360}," Infrastructure/Generated/\n",[3063,5359,5360],{"class":3065,"line":3221},[3063,5361,3083],{"emptyLinePlaceholder":3082},[3063,5363,5364],{"class":3065,"line":3226},[3063,5365,5366],{"class":3069},"# 5. Перевірити що Extensions не зачіпнуті\n",[3063,5368,5369,5371,5373],{"class":3065,"line":3264},[3063,5370,5282],{"class":3283},[3063,5372,5331],{"class":3360},[3063,5374,5375],{"class":3360}," Infrastructure/Extensions/\n",[3063,5377,5378],{"class":3065,"line":3269},[3063,5379,5380],{"class":3069},"# Повинно бути: нічого (порожньо)\n",[3063,5382,5383],{"class":3065,"line":3275},[3063,5384,3083],{"emptyLinePlaceholder":3082},[3063,5386,5387],{"class":3065,"line":3298},[3063,5388,5389],{"class":3069},"# 6. Запустити тести\n",[3063,5391,5392,5394],{"class":3065,"line":3304},[3063,5393,4881],{"class":3283},[3063,5395,5396],{"class":3360}," test\n",[3063,5398,5399],{"class":3065,"line":3333},[3063,5400,3083],{"emptyLinePlaceholder":3082},[3063,5402,5403],{"class":3065,"line":3357},[3063,5404,5405],{"class":3069},"# 7. Закомітити якщо все в порядку\n",[3063,5407,5408,5410,5412],{"class":3065,"line":3367},[3063,5409,5282],{"class":3283},[3063,5411,5285],{"class":3360},[3063,5413,5288],{"class":3360},[3063,5415,5416,5418,5420,5422],{"class":3065,"line":3372},[3063,5417,5282],{"class":3283},[3063,5419,5295],{"class":3360},[3063,5421,5298],{"class":3089},[3063,5423,5424],{"class":3360}," \"chore: re-scaffold after DB schema update v3.2\"\n",[3041,5426,5428],{"id":5427},"conflict-resolution-extensions-vs-generated","Conflict resolution: Extensions vs Generated",[2963,5430,5431],{},"Після re-scaffold, Extensions файл може звертатись до частин entity що були видалені:",[3021,5433,5435],{"className":3057,"code":5434,"language":3059,"meta":3029,"style":3029},"// Product.Extensions.cs — ПІСЛЯ re-scaffold Product більше не має поля Discount!\npublic partial class Product\n{\n    // ❌ Compilation Error: Product не має Discount після re-scaffold\n    public bool IsDiscounted => Discount.HasValue && Discount.Value > 0;\n}\n",[2984,5436,5437,5442,5452,5456,5461,5491],{"__ignoreMap":3029},[3063,5438,5439],{"class":3065,"line":3066},[3063,5440,5441],{"class":3069},"// Product.Extensions.cs — ПІСЛЯ re-scaffold Product більше не має поля Discount!\n",[3063,5443,5444,5446,5448,5450],{"class":3065,"line":3073},[3063,5445,3125],{"class":3089},[3063,5447,3128],{"class":3089},[3063,5449,3131],{"class":3089},[3063,5451,3134],{"class":3093},[3063,5453,5454],{"class":3065,"line":3079},[3063,5455,3140],{"class":3097},[3063,5457,5458],{"class":3065,"line":3086},[3063,5459,5460],{"class":3069},"    // ❌ Compilation Error: Product не має Discount після re-scaffold\n",[3063,5462,5463,5465,5467,5469,5471,5473,5475,5477,5479,5481,5483,5485,5487,5489],{"class":3065,"line":3111},[3063,5464,3152],{"class":3089},[3063,5466,3231],{"class":3089},[3063,5468,3234],{"class":3158},[3063,5470,3237],{"class":3097},[3063,5472,3193],{"class":3158},[3063,5474,3010],{"class":3097},[3063,5476,3244],{"class":3158},[3063,5478,3247],{"class":3097},[3063,5480,3193],{"class":3158},[3063,5482,3010],{"class":3097},[3063,5484,3198],{"class":3158},[3063,5486,3256],{"class":3097},[3063,5488,3259],{"class":3204},[3063,5490,3108],{"class":3097},[3063,5492,5493],{"class":3065,"line":3116},[3063,5494,3785],{"class":3097},[2963,5496,5497],{},"Це компіляційна помилка — компілятор одразу сигналізує про проблему. Ви бачите що треба виправити у Extensions. Це правильна поведінка — краще, ніж мовчки мати застарілу логіку.",[2973,5499],{},[2976,5501,5503],{"id":5502},"database-schema-comparison-tools","Database Schema Comparison Tools",[2963,5505,5506,5507,5510,5511],{},"При Database-First підході часто виникає питання: ",[2991,5508,5509],{},"яка різниця між поточною схемою у БД і тим що описано в EF Core моделі?"," Або: ",[2991,5512,5513],{},"яка різниця між dev і prod базами?",[2963,5515,5516],{},"Для цього існують спеціалізовані інструменти:",[3041,5518,5520],{"id":5519},"dotnet-ef-dbcontext-script-sql-для-поточної-моделі","dotnet ef dbcontext script: SQL для поточної моделі",[3021,5522,5524],{"className":4867,"code":5523,"language":4869,"meta":3029,"style":3029},"# Генерує SQL DDL для поточної EF Core моделі (без міграцій!)\ndotnet ef dbcontext script -o current-model.sql\n",[2984,5525,5526,5531],{"__ignoreMap":3029},[3063,5527,5528],{"class":3065,"line":3066},[3063,5529,5530],{"class":3069},"# Генерує SQL DDL для поточної EF Core моделі (без міграцій!)\n",[3063,5532,5533,5535,5537,5539,5542,5545],{"class":3065,"line":3073},[3063,5534,4881],{"class":3283},[3063,5536,4884],{"class":3360},[3063,5538,4887],{"class":3360},[3063,5540,5541],{"class":3360}," script",[3063,5543,5544],{"class":3089}," -o",[3063,5546,5547],{"class":3360}," current-model.sql\n",[2963,5549,5550,5551,5554,5555,4967,5558,5561],{},"Цей скрипт відображає ",[2991,5552,5553],{},"як EF Core бачить вашу модель"," — всі ",[2984,5556,5557],{},"CREATE TABLE",[2984,5559,5560],{},"CREATE INDEX",". Корисно порівняти з реальним станом БД.",[3041,5563,5565],{"id":5564},"ef-core-dbcontextdatabasecompareschemas-відсутній-у-stdlib","EF Core DbContext.Database.CompareSchemas (відсутній у stdlib!)",[2963,5567,5568,5569,5572],{},"EF Core ",[2991,5570,5571],{},"не має"," вбудованого diff-інструменту між моделлю і БД. Але є сторонні рішення:",[3041,5574,5576],{"id":5575},"efcoreschemacompare-open-source","EFCore.SchemaCompare (open source)",[3021,5578,5580],{"className":4867,"code":5579,"language":4869,"meta":3029,"style":3029},"dotnet add package EFCore.SchemaCompare\n",[2984,5581,5582],{"__ignoreMap":3029},[3063,5583,5584,5586,5588,5591],{"class":3065,"line":3066},[3063,5585,4881],{"class":3283},[3063,5587,5285],{"class":3360},[3063,5589,5590],{"class":3360}," package",[3063,5592,5593],{"class":3360}," EFCore.SchemaCompare\n",[3021,5595,5597],{"className":3057,"code":5596,"language":3059,"meta":3029,"style":3029},"using EfSchemaCompare;\n\n// Порівнює C# модель (DbContext) з реальною схемою БД\nvar comparer = new CompareEfSql();\nvar hasErrors = comparer.CompareEfWithDb(context);\n\nif (hasErrors)\n{\n    // Виводить детальний звіт\n    Console.WriteLine(comparer.GetAllErrors);\n    // Example:\n    // DIFFERENT: Entity 'Product', column 'Price': column type. Expected = decimal(18,2), found = decimal(10,2)\n    // MISSING: Entity 'OrderLineItem', index 'IX_OrderLineItems_OrderId' not found in database.\n    // EXTRA: Table 'Tmp_Products' in database, not in entity classes.\n}\nelse\n{\n    Console.WriteLine(\"EF Core model matches database schema perfectly.\");\n}\n",[2984,5598,5599,5609,5613,5618,5635,5659,5663,5675,5679,5684,5705,5710,5715,5720,5725,5729,5734,5738,5753],{"__ignoreMap":3029},[3063,5600,5601,5604,5607],{"class":3065,"line":3066},[3063,5602,5603],{"class":3307},"using",[3063,5605,5606],{"class":3093}," EfSchemaCompare",[3063,5608,3108],{"class":3097},[3063,5610,5611],{"class":3065,"line":3073},[3063,5612,3083],{"emptyLinePlaceholder":3082},[3063,5614,5615],{"class":3065,"line":3079},[3063,5616,5617],{"class":3069},"// Порівнює C# модель (DbContext) з реальною схемою БД\n",[3063,5619,5620,5622,5625,5627,5629,5632],{"class":3065,"line":3086},[3063,5621,4203],{"class":3089},[3063,5623,5624],{"class":3158}," comparer",[3063,5626,3377],{"class":3097},[3063,5628,3554],{"class":3089},[3063,5630,5631],{"class":3093}," CompareEfSql",[3063,5633,5634],{"class":3097},"();\n",[3063,5636,5637,5639,5642,5644,5647,5649,5652,5654,5657],{"class":3065,"line":3111},[3063,5638,4203],{"class":3089},[3063,5640,5641],{"class":3158}," hasErrors",[3063,5643,3377],{"class":3097},[3063,5645,5646],{"class":3158},"comparer",[3063,5648,3010],{"class":3097},[3063,5650,5651],{"class":3283},"CompareEfWithDb",[3063,5653,3287],{"class":3097},[3063,5655,5656],{"class":3158},"context",[3063,5658,3364],{"class":3097},[3063,5660,5661],{"class":3065,"line":3116},[3063,5662,3083],{"emptyLinePlaceholder":3082},[3063,5664,5665,5668,5670,5673],{"class":3065,"line":3122},[3063,5666,5667],{"class":3307},"if",[3063,5669,3311],{"class":3097},[3063,5671,5672],{"class":3158},"hasErrors",[3063,5674,3208],{"class":3097},[3063,5676,5677],{"class":3065,"line":3137},[3063,5678,3140],{"class":3097},[3063,5680,5681],{"class":3065,"line":3143},[3063,5682,5683],{"class":3069},"    // Виводить детальний звіт\n",[3063,5685,5686,5689,5691,5694,5696,5698,5700,5703],{"class":3065,"line":3149},[3063,5687,5688],{"class":3158},"    Console",[3063,5690,3010],{"class":3097},[3063,5692,5693],{"class":3283},"WriteLine",[3063,5695,3287],{"class":3097},[3063,5697,5646],{"class":3158},[3063,5699,3010],{"class":3097},[3063,5701,5702],{"class":3158},"GetAllErrors",[3063,5704,3364],{"class":3097},[3063,5706,5707],{"class":3065,"line":3165},[3063,5708,5709],{"class":3069},"    // Example:\n",[3063,5711,5712],{"class":3065,"line":3176},[3063,5713,5714],{"class":3069},"    // DIFFERENT: Entity 'Product', column 'Price': column type. Expected = decimal(18,2), found = decimal(10,2)\n",[3063,5716,5717],{"class":3065,"line":3211},[3063,5718,5719],{"class":3069},"    // MISSING: Entity 'OrderLineItem', index 'IX_OrderLineItems_OrderId' not found in database.\n",[3063,5721,5722],{"class":3065,"line":3221},[3063,5723,5724],{"class":3069},"    // EXTRA: Table 'Tmp_Products' in database, not in entity classes.\n",[3063,5726,5727],{"class":3065,"line":3226},[3063,5728,3785],{"class":3097},[3063,5730,5731],{"class":3065,"line":3264},[3063,5732,5733],{"class":3307},"else\n",[3063,5735,5736],{"class":3065,"line":3269},[3063,5737,3140],{"class":3097},[3063,5739,5740,5742,5744,5746,5748,5751],{"class":3065,"line":3275},[3063,5741,5688],{"class":3158},[3063,5743,3010],{"class":3097},[3063,5745,5693],{"class":3283},[3063,5747,3287],{"class":3097},[3063,5749,5750],{"class":3360},"\"EF Core model matches database schema perfectly.\"",[3063,5752,3364],{"class":3097},[3063,5754,5755],{"class":3065,"line":3298},[3063,5756,3785],{"class":3097},[2963,5758,5759],{},"Інтеграція у тести:",[3021,5761,5763],{"className":3057,"code":5762,"language":3059,"meta":3029,"style":3029},"[Fact]\npublic void EfCoreModel_ShouldMatchDatabaseSchema()\n{\n    using var context = GetTestContext(); // реальна тестова БД\n\n    var comparer  = new CompareEfSql();\n    var hasErrors = comparer.CompareEfWithDb(context);\n\n    Assert.False(hasErrors, $\"Schema differences:\\n{comparer.GetAllErrors}\");\n}\n",[2984,5764,5765,5776,5787,5791,5813,5817,5833,5853,5857,5893],{"__ignoreMap":3029},[3063,5766,5767,5770,5773],{"class":3065,"line":3066},[3063,5768,5769],{"class":3097},"[",[3063,5771,5772],{"class":3093},"Fact",[3063,5774,5775],{"class":3097},"]\n",[3063,5777,5778,5780,5782,5785],{"class":3065,"line":3073},[3063,5779,3125],{"class":3089},[3063,5781,3280],{"class":3089},[3063,5783,5784],{"class":3283}," EfCoreModel_ShouldMatchDatabaseSchema",[3063,5786,3535],{"class":3097},[3063,5788,5789],{"class":3065,"line":3079},[3063,5790,3140],{"class":3097},[3063,5792,5793,5796,5799,5802,5804,5807,5810],{"class":3065,"line":3086},[3063,5794,5795],{"class":3307},"    using",[3063,5797,5798],{"class":3089}," var",[3063,5800,5801],{"class":3158}," context",[3063,5803,3377],{"class":3097},[3063,5805,5806],{"class":3283},"GetTestContext",[3063,5808,5809],{"class":3097},"(); ",[3063,5811,5812],{"class":3069},"// реальна тестова БД\n",[3063,5814,5815],{"class":3065,"line":3111},[3063,5816,3083],{"emptyLinePlaceholder":3082},[3063,5818,5819,5822,5824,5827,5829,5831],{"class":3065,"line":3116},[3063,5820,5821],{"class":3089},"    var",[3063,5823,5624],{"class":3158},[3063,5825,5826],{"class":3097},"  = ",[3063,5828,3554],{"class":3089},[3063,5830,5631],{"class":3093},[3063,5832,5634],{"class":3097},[3063,5834,5835,5837,5839,5841,5843,5845,5847,5849,5851],{"class":3065,"line":3122},[3063,5836,5821],{"class":3089},[3063,5838,5641],{"class":3158},[3063,5840,3377],{"class":3097},[3063,5842,5646],{"class":3158},[3063,5844,3010],{"class":3097},[3063,5846,5651],{"class":3283},[3063,5848,3287],{"class":3097},[3063,5850,5656],{"class":3158},[3063,5852,3364],{"class":3097},[3063,5854,5855],{"class":3065,"line":3137},[3063,5856,3083],{"emptyLinePlaceholder":3082},[3063,5858,5859,5862,5864,5867,5869,5871,5873,5876,5879,5881,5883,5885,5887,5889,5891],{"class":3065,"line":3143},[3063,5860,5861],{"class":3158},"    Assert",[3063,5863,3010],{"class":3097},[3063,5865,5866],{"class":3283},"False",[3063,5868,3287],{"class":3097},[3063,5870,5672],{"class":3158},[3063,5872,4072],{"class":3097},[3063,5874,5875],{"class":3360},"$\"Schema differences:",[3063,5877,5878],{"class":4893},"\\n",[3063,5880,3468],{"class":3467},[3063,5882,5646],{"class":3158},[3063,5884,3010],{"class":3467},[3063,5886,5702],{"class":3158},[3063,5888,3474],{"class":3467},[3063,5890,5115],{"class":3360},[3063,5892,3364],{"class":3097},[3063,5894,5895],{"class":3065,"line":3149},[3063,5896,3785],{"class":3097},[3041,5898,5900],{"id":5899},"sql-server-schema-compare-visual-studio","SQL Server Schema Compare (Visual Studio)",[2963,5902,5903,5904,5907],{},"Visual Studio Enterprise має вбудований ",[2991,5905,5906],{},"Schema Compare"," для SQL Server:",[5909,5910,5911,5915,5918,5922,5925,5929,5932,5936],"steps",{},[3041,5912,5914],{"id":5913},"відкрити-schema-compare","Відкрити Schema Compare",[2963,5916,5917],{},"SQL Server Object Explorer → ПКМ на базі → Schema Compare...",[3041,5919,5921],{"id":5920},"вказати-джерело-та-ціль","Вказати джерело та ціль",[2963,5923,5924],{},"Source: Dev база (або EF Core generated script)\nTarget: Production база (або Staging)",[3041,5926,5928],{"id":5927},"аналізувати-різниці","Аналізувати різниці",[2963,5930,5931],{},"Schema Compare покаже таблиці, стовпці, індекси що відрізняються — з кольоровим кодуванням (Added/Changed/Deleted).",[3041,5933,5935],{"id":5934},"згенерувати-скрипт-або-застосувати","Згенерувати скрипт або застосувати",[2963,5937,5938],{},"\"Generate Script\" — SQL ALTER TABLE скрипт. \"Update\" — застосувати напряму.",[3041,5940,5942],{"id":5941},"redgate-sql-compare-комерційне-але-лідер-ринку","Redgate SQL Compare (комерційне, але лідер ринку)",[3021,5944,5947],{"className":5945,"code":5946,"language":3026},[3024],"Redgate SQL Compare:\n- Порівнює схеми: Dev ↔ Staging ↔ Production\n- Генерує delta SQL скрипти\n- Інтеграція з SQL Server, MySQL, PostgreSQL\n- CLI версія для CI/CD pipeline\n",[2984,5948,5946],{"__ignoreMap":3029},[3041,5950,5952],{"id":5951},"pgadmin-dbeaver-schema-diff-для-postgresql","pgAdmin / DBeaver Schema Diff (для PostgreSQL)",[3021,5954,5958],{"className":5955,"code":5956,"language":5957,"meta":3029,"style":3029},"language-sql shiki shiki-themes light-plus dark-plus dark-plus","-- PostgreSQL: вбудований information_schema для ручного порівняння\nSELECT table_schema, table_name, column_name, data_type, is_nullable\nFROM information_schema.columns\nWHERE table_schema NOT IN ('information_schema', 'pg_catalog')\nORDER BY table_schema, table_name, ordinal_position;\n\n-- Або через pg_dump + diff:\npg_dump --schema-only prod_db > prod_schema.sql\npg_dump --schema-only dev_db  > dev_schema.sql\ndiff prod_schema.sql dev_schema.sql\n","sql",[2984,5959,5960,5965,5973,5981,6007,6015,6019,6024,6032,6039],{"__ignoreMap":3029},[3063,5961,5962],{"class":3065,"line":3066},[3063,5963,5964],{"class":3069},"-- PostgreSQL: вбудований information_schema для ручного порівняння\n",[3063,5966,5967,5970],{"class":3065,"line":3073},[3063,5968,5969],{"class":3089},"SELECT",[3063,5971,5972],{"class":3097}," table_schema, table_name, column_name, data_type, is_nullable\n",[3063,5974,5975,5978],{"class":3065,"line":3079},[3063,5976,5977],{"class":3089},"FROM",[3063,5979,5980],{"class":3097}," information_schema.columns\n",[3063,5982,5983,5986,5989,5992,5995,5997,6000,6002,6005],{"class":3065,"line":3086},[3063,5984,5985],{"class":3089},"WHERE",[3063,5987,5988],{"class":3097}," table_schema ",[3063,5990,5991],{"class":3089},"NOT",[3063,5993,5994],{"class":3089}," IN",[3063,5996,3311],{"class":3097},[3063,5998,5999],{"class":3360},"'information_schema'",[3063,6001,4072],{"class":3097},[3063,6003,6004],{"class":3360},"'pg_catalog'",[3063,6006,3208],{"class":3097},[3063,6008,6009,6012],{"class":3065,"line":3111},[3063,6010,6011],{"class":3089},"ORDER BY",[3063,6013,6014],{"class":3097}," table_schema, table_name, ordinal_position;\n",[3063,6016,6017],{"class":3065,"line":3116},[3063,6018,3083],{"emptyLinePlaceholder":3082},[3063,6020,6021],{"class":3065,"line":3122},[3063,6022,6023],{"class":3069},"-- Або через pg_dump + diff:\n",[3063,6025,6026,6029],{"class":3065,"line":3137},[3063,6027,6028],{"class":3097},"pg_dump ",[3063,6030,6031],{"class":3069},"--schema-only prod_db > prod_schema.sql\n",[3063,6033,6034,6036],{"class":3065,"line":3143},[3063,6035,6028],{"class":3097},[3063,6037,6038],{"class":3069},"--schema-only dev_db  > dev_schema.sql\n",[3063,6040,6041],{"class":3065,"line":3149},[3063,6042,6043],{"class":3097},"diff prod_schema.sql dev_schema.sql\n",[2973,6045],{},[2976,6047,6049],{"id":6048},"multi-database-scenarios-кілька-бд-в-одному-застосунку","Multi-Database Scenarios: кілька БД в одному застосунку",[2963,6051,6052],{},"Реальні enterprise-системи рідко мають одну БД. Типові сценарії:",[4838,6054,6055,6061,6067,6073],{},[4841,6056,6057,6060],{},[2991,6058,6059],{},"Bounded Contexts",": OrdersDb, InventoryDb, NotificationsDb — кожен сервіс зі своєю БД",[4841,6062,6063,6066],{},[2991,6064,6065],{},"Мультитенантність",": окрема БД на tenant",[4841,6068,6069,6072],{},[2991,6070,6071],{},"Read/Write Separation",": Write на основний SQL Server, Read на Read Replica або Elasticsearch",[4841,6074,6075,6078],{},[2991,6076,6077],{},"Legacy Integration",": нова система читає з legacy OracleDB, пише у нову PostgreSQL",[3041,6080,6082],{"id":6081},"підхід-1-кілька-dbcontext-у-одному-застосунку","Підхід 1: Кілька DbContext у одному застосунку",[3021,6084,6086],{"className":3057,"code":6085,"language":3059,"meta":3029,"style":3029},"// Два DbContext для двох різних БД\npublic class OrdersDbContext : DbContext\n{\n    public OrdersDbContext(DbContextOptions\u003COrdersDbContext> options) : base(options) { }\n\n    public DbSet\u003COrder>       Orders      => Set\u003COrder>();\n    public DbSet\u003COrderItem>   OrderItems  => Set\u003COrderItem>();\n}\n\npublic class InventoryDbContext : DbContext\n{\n    public InventoryDbContext(DbContextOptions\u003CInventoryDbContext> options) : base(options) { }\n\n    public DbSet\u003CProduct>     Products    => Set\u003CProduct>();\n    public DbSet\u003CStockLevel>  StockLevels => Set\u003CStockLevel>();\n}\n",[2984,6087,6088,6093,6107,6111,6145,6149,6179,6207,6211,6215,6228,6232,6261,6265,6292,6319],{"__ignoreMap":3029},[3063,6089,6090],{"class":3065,"line":3066},[3063,6091,6092],{"class":3069},"// Два DbContext для двох різних БД\n",[3063,6094,6095,6097,6099,6102,6104],{"class":3065,"line":3073},[3063,6096,3125],{"class":3089},[3063,6098,3131],{"class":3089},[3063,6100,6101],{"class":3093}," OrdersDbContext",[3063,6103,4371],{"class":3097},[3063,6105,6106],{"class":3093},"DbContext\n",[3063,6108,6109],{"class":3065,"line":3079},[3063,6110,3140],{"class":3097},[3063,6112,6113,6115,6117,6119,6122,6124,6127,6129,6132,6135,6138,6140,6142],{"class":3065,"line":3086},[3063,6114,3152],{"class":3089},[3063,6116,6101],{"class":3283},[3063,6118,3287],{"class":3097},[3063,6120,6121],{"class":3093},"DbContextOptions",[3063,6123,3523],{"class":3097},[3063,6125,6126],{"class":3093},"OrdersDbContext",[3063,6128,3529],{"class":3097},[3063,6130,6131],{"class":3158},"options",[3063,6133,6134],{"class":3097},") : ",[3063,6136,6137],{"class":3089},"base",[3063,6139,3287],{"class":3097},[3063,6141,6131],{"class":3158},[3063,6143,6144],{"class":3097},") { }\n",[3063,6146,6147],{"class":3065,"line":3111},[3063,6148,3083],{"emptyLinePlaceholder":3082},[3063,6150,6151,6153,6156,6158,6161,6164,6167,6170,6173,6175,6177],{"class":3065,"line":3116},[3063,6152,3152],{"class":3089},[3063,6154,6155],{"class":3093}," DbSet",[3063,6157,3523],{"class":3097},[3063,6159,6160],{"class":3093},"Order",[3063,6162,6163],{"class":3097},">       ",[3063,6165,6166],{"class":3158},"Orders",[3063,6168,6169],{"class":3097},"      => ",[3063,6171,6172],{"class":3283},"Set",[3063,6174,3523],{"class":3097},[3063,6176,6160],{"class":3093},[3063,6178,3564],{"class":3097},[3063,6180,6181,6183,6185,6187,6190,6193,6196,6199,6201,6203,6205],{"class":3065,"line":3122},[3063,6182,3152],{"class":3089},[3063,6184,6155],{"class":3093},[3063,6186,3523],{"class":3097},[3063,6188,6189],{"class":3093},"OrderItem",[3063,6191,6192],{"class":3097},">   ",[3063,6194,6195],{"class":3158},"OrderItems",[3063,6197,6198],{"class":3097},"  => ",[3063,6200,6172],{"class":3283},[3063,6202,3523],{"class":3097},[3063,6204,6189],{"class":3093},[3063,6206,3564],{"class":3097},[3063,6208,6209],{"class":3065,"line":3137},[3063,6210,3785],{"class":3097},[3063,6212,6213],{"class":3065,"line":3143},[3063,6214,3083],{"emptyLinePlaceholder":3082},[3063,6216,6217,6219,6221,6224,6226],{"class":3065,"line":3149},[3063,6218,3125],{"class":3089},[3063,6220,3131],{"class":3089},[3063,6222,6223],{"class":3093}," InventoryDbContext",[3063,6225,4371],{"class":3097},[3063,6227,6106],{"class":3093},[3063,6229,6230],{"class":3065,"line":3165},[3063,6231,3140],{"class":3097},[3063,6233,6234,6236,6238,6240,6242,6244,6247,6249,6251,6253,6255,6257,6259],{"class":3065,"line":3176},[3063,6235,3152],{"class":3089},[3063,6237,6223],{"class":3283},[3063,6239,3287],{"class":3097},[3063,6241,6121],{"class":3093},[3063,6243,3523],{"class":3097},[3063,6245,6246],{"class":3093},"InventoryDbContext",[3063,6248,3529],{"class":3097},[3063,6250,6131],{"class":3158},[3063,6252,6134],{"class":3097},[3063,6254,6137],{"class":3089},[3063,6256,3287],{"class":3097},[3063,6258,6131],{"class":3158},[3063,6260,6144],{"class":3097},[3063,6262,6263],{"class":3065,"line":3211},[3063,6264,3083],{"emptyLinePlaceholder":3082},[3063,6266,6267,6269,6271,6273,6275,6278,6281,6284,6286,6288,6290],{"class":3065,"line":3221},[3063,6268,3152],{"class":3089},[3063,6270,6155],{"class":3093},[3063,6272,3523],{"class":3097},[3063,6274,3039],{"class":3093},[3063,6276,6277],{"class":3097},">     ",[3063,6279,6280],{"class":3158},"Products",[3063,6282,6283],{"class":3097},"    => ",[3063,6285,6172],{"class":3283},[3063,6287,3523],{"class":3097},[3063,6289,3039],{"class":3093},[3063,6291,3564],{"class":3097},[3063,6293,6294,6296,6298,6300,6303,6306,6309,6311,6313,6315,6317],{"class":3065,"line":3226},[3063,6295,3152],{"class":3089},[3063,6297,6155],{"class":3093},[3063,6299,3523],{"class":3097},[3063,6301,6302],{"class":3093},"StockLevel",[3063,6304,6305],{"class":3097},">  ",[3063,6307,6308],{"class":3158},"StockLevels",[3063,6310,3237],{"class":3097},[3063,6312,6172],{"class":3283},[3063,6314,3523],{"class":3097},[3063,6316,6302],{"class":3093},[3063,6318,3564],{"class":3097},[3063,6320,6321],{"class":3065,"line":3264},[3063,6322,3785],{"class":3097},[2963,6324,6325],{},"Реєстрація двох DbContext:",[3021,6327,6329],{"className":3057,"code":6328,"language":3059,"meta":3029,"style":3029},"// Program.cs: окремі DbContext для окремих БД\nbuilder.Services.AddDbContext\u003COrdersDbContext>(options =>\n    options.UseSqlServer(\n        builder.Configuration.GetConnectionString(\"OrdersDb\")));\n\nbuilder.Services.AddDbContext\u003CInventoryDbContext>(options =>\n    options.UseNpgsql(\n        builder.Configuration.GetConnectionString(\"InventoryDb\")));\n\n// appsettings.json:\n// {\n//   \"ConnectionStrings\": {\n//     \"OrdersDb\":    \"Server=orders-sql;Database=Orders;...\",\n//     \"InventoryDb\": \"Host=inventory-pg;Database=Inventory;...\"\n//   }\n// }\n",[2984,6330,6331,6336,6361,6374,6396,6400,6422,6433,6452,6456,6461,6466,6471,6476,6481,6486],{"__ignoreMap":3029},[3063,6332,6333],{"class":3065,"line":3066},[3063,6334,6335],{"class":3069},"// Program.cs: окремі DbContext для окремих БД\n",[3063,6337,6338,6341,6343,6346,6348,6351,6353,6355,6357,6359],{"class":3065,"line":3073},[3063,6339,6340],{"class":3158},"builder",[3063,6342,3010],{"class":3097},[3063,6344,6345],{"class":3158},"Services",[3063,6347,3010],{"class":3097},[3063,6349,6350],{"class":3283},"AddDbContext",[3063,6352,3523],{"class":3097},[3063,6354,6126],{"class":3093},[3063,6356,4744],{"class":3097},[3063,6358,6131],{"class":3158},[3063,6360,3162],{"class":3097},[3063,6362,6363,6366,6368,6371],{"class":3065,"line":3079},[3063,6364,6365],{"class":3158},"    options",[3063,6367,3010],{"class":3097},[3063,6369,6370],{"class":3283},"UseSqlServer",[3063,6372,6373],{"class":3097},"(\n",[3063,6375,6376,6378,6380,6383,6385,6388,6390,6393],{"class":3065,"line":3086},[3063,6377,4777],{"class":3158},[3063,6379,3010],{"class":3097},[3063,6381,6382],{"class":3158},"Configuration",[3063,6384,3010],{"class":3097},[3063,6386,6387],{"class":3283},"GetConnectionString",[3063,6389,3287],{"class":3097},[3063,6391,6392],{"class":3360},"\"OrdersDb\"",[3063,6394,6395],{"class":3097},")));\n",[3063,6397,6398],{"class":3065,"line":3111},[3063,6399,3083],{"emptyLinePlaceholder":3082},[3063,6401,6402,6404,6406,6408,6410,6412,6414,6416,6418,6420],{"class":3065,"line":3116},[3063,6403,6340],{"class":3158},[3063,6405,3010],{"class":3097},[3063,6407,6345],{"class":3158},[3063,6409,3010],{"class":3097},[3063,6411,6350],{"class":3283},[3063,6413,3523],{"class":3097},[3063,6415,6246],{"class":3093},[3063,6417,4744],{"class":3097},[3063,6419,6131],{"class":3158},[3063,6421,3162],{"class":3097},[3063,6423,6424,6426,6428,6431],{"class":3065,"line":3122},[3063,6425,6365],{"class":3158},[3063,6427,3010],{"class":3097},[3063,6429,6430],{"class":3283},"UseNpgsql",[3063,6432,6373],{"class":3097},[3063,6434,6435,6437,6439,6441,6443,6445,6447,6450],{"class":3065,"line":3137},[3063,6436,4777],{"class":3158},[3063,6438,3010],{"class":3097},[3063,6440,6382],{"class":3158},[3063,6442,3010],{"class":3097},[3063,6444,6387],{"class":3283},[3063,6446,3287],{"class":3097},[3063,6448,6449],{"class":3360},"\"InventoryDb\"",[3063,6451,6395],{"class":3097},[3063,6453,6454],{"class":3065,"line":3143},[3063,6455,3083],{"emptyLinePlaceholder":3082},[3063,6457,6458],{"class":3065,"line":3149},[3063,6459,6460],{"class":3069},"// appsettings.json:\n",[3063,6462,6463],{"class":3065,"line":3165},[3063,6464,6465],{"class":3069},"// {\n",[3063,6467,6468],{"class":3065,"line":3176},[3063,6469,6470],{"class":3069},"//   \"ConnectionStrings\": {\n",[3063,6472,6473],{"class":3065,"line":3211},[3063,6474,6475],{"class":3069},"//     \"OrdersDb\":    \"Server=orders-sql;Database=Orders;...\",\n",[3063,6477,6478],{"class":3065,"line":3221},[3063,6479,6480],{"class":3069},"//     \"InventoryDb\": \"Host=inventory-pg;Database=Inventory;...\"\n",[3063,6482,6483],{"class":3065,"line":3226},[3063,6484,6485],{"class":3069},"//   }\n",[3063,6487,6488],{"class":3065,"line":3264},[3063,6489,6490],{"class":3069},"// }\n",[2963,6492,6493],{},"Використання у сервісах:",[3021,6495,6497],{"className":3057,"code":6496,"language":3059,"meta":3029,"style":3029},"public class OrderFulfillmentService\n{\n    private readonly OrdersDbContext    _ordersDb;\n    private readonly InventoryDbContext _inventoryDb;\n\n    // Два DbContext — з різних БД, різних провайдерів!\n    public OrderFulfillmentService(\n        OrdersDbContext ordersDb,\n        InventoryDbContext inventoryDb)\n    {\n        _ordersDb    = ordersDb;\n        _inventoryDb = inventoryDb;\n    }\n\n    public async Task FulfillOrderAsync(int orderId)\n    {\n        // Читаємо з OrdersDb\n        var order = await _ordersDb.Orders\n            .Include(o => o.Items)\n            .FirstOrDefaultAsync(o => o.Id == orderId)\n            ?? throw new NotFoundException($\"Order {orderId} not found.\");\n\n        // Резервуємо товар в InventoryDb\n        foreach (var item in order.Items)\n        {\n            var rowsAffected = await _inventoryDb.StockLevels\n                .Where(s => s.ProductId == item.ProductId && s.Available >= item.Quantity)\n                .ExecuteUpdateAsync(s =>\n                    s.SetProperty(sl => sl.Available, sl => sl.Available - item.Quantity));\n\n            if (rowsAffected == 0)\n                throw new InsufficientStockException(item.ProductId);\n        }\n\n        // Оновлюємо статус в OrdersDb\n        order.Status    = \"Fulfilled\";\n        order.FulfilledAt = DateTime.UtcNow;\n        await _ordersDb.SaveChangesAsync();\n    }\n}\n",[2984,6498,6499,6508,6512,6526,6539,6543,6548,6557,6568,6578,6582,6595,6607,6611,6615,6635,6639,6644,6664,6687,6713,6742,6746,6751,6772,6776,6795,6845,6857,6902,6906,6921,6941,6945,6949,6954,6971,6990,7003,7007],{"__ignoreMap":3029},[3063,6500,6501,6503,6505],{"class":3065,"line":3066},[3063,6502,3125],{"class":3089},[3063,6504,3131],{"class":3089},[3063,6506,6507],{"class":3093}," OrderFulfillmentService\n",[3063,6509,6510],{"class":3065,"line":3073},[3063,6511,3140],{"class":3097},[3063,6513,6514,6516,6519,6521,6524],{"class":3065,"line":3079},[3063,6515,4728],{"class":3089},[3063,6517,6518],{"class":3089}," readonly",[3063,6520,6101],{"class":3093},[3063,6522,6523],{"class":3158},"    _ordersDb",[3063,6525,3108],{"class":3097},[3063,6527,6528,6530,6532,6534,6537],{"class":3065,"line":3086},[3063,6529,4728],{"class":3089},[3063,6531,6518],{"class":3089},[3063,6533,6223],{"class":3093},[3063,6535,6536],{"class":3158}," _inventoryDb",[3063,6538,3108],{"class":3097},[3063,6540,6541],{"class":3065,"line":3111},[3063,6542,3083],{"emptyLinePlaceholder":3082},[3063,6544,6545],{"class":3065,"line":3116},[3063,6546,6547],{"class":3069},"    // Два DbContext — з різних БД, різних провайдерів!\n",[3063,6549,6550,6552,6555],{"class":3065,"line":3122},[3063,6551,3152],{"class":3089},[3063,6553,6554],{"class":3283}," OrderFulfillmentService",[3063,6556,6373],{"class":3097},[3063,6558,6559,6562,6565],{"class":3065,"line":3137},[3063,6560,6561],{"class":3093},"        OrdersDbContext",[3063,6563,6564],{"class":3158}," ordersDb",[3063,6566,6567],{"class":3097},",\n",[3063,6569,6570,6573,6576],{"class":3065,"line":3143},[3063,6571,6572],{"class":3093},"        InventoryDbContext",[3063,6574,6575],{"class":3158}," inventoryDb",[3063,6577,3208],{"class":3097},[3063,6579,6580],{"class":3065,"line":3149},[3063,6581,3301],{"class":3097},[3063,6583,6584,6587,6590,6593],{"class":3065,"line":3165},[3063,6585,6586],{"class":3158},"        _ordersDb",[3063,6588,6589],{"class":3097},"    = ",[3063,6591,6592],{"class":3158},"ordersDb",[3063,6594,3108],{"class":3097},[3063,6596,6597,6600,6602,6605],{"class":3065,"line":3176},[3063,6598,6599],{"class":3158},"        _inventoryDb",[3063,6601,3377],{"class":3097},[3063,6603,6604],{"class":3158},"inventoryDb",[3063,6606,3108],{"class":3097},[3063,6608,6609],{"class":3065,"line":3211},[3063,6610,3387],{"class":3097},[3063,6612,6613],{"class":3065,"line":3221},[3063,6614,3083],{"emptyLinePlaceholder":3082},[3063,6616,6617,6619,6621,6623,6626,6628,6630,6633],{"class":3065,"line":3226},[3063,6618,3152],{"class":3089},[3063,6620,3968],{"class":3089},[3063,6622,3971],{"class":3093},[3063,6624,6625],{"class":3283}," FulfillOrderAsync",[3063,6627,3287],{"class":3097},[3063,6629,3906],{"class":3089},[3063,6631,6632],{"class":3158}," orderId",[3063,6634,3208],{"class":3097},[3063,6636,6637],{"class":3065,"line":3264},[3063,6638,3301],{"class":3097},[3063,6640,6641],{"class":3065,"line":3269},[3063,6642,6643],{"class":3069},"        // Читаємо з OrdersDb\n",[3063,6645,6646,6648,6651,6653,6656,6659,6661],{"class":3065,"line":3275},[3063,6647,3546],{"class":3089},[3063,6649,6650],{"class":3158}," order",[3063,6652,3377],{"class":3097},[3063,6654,6655],{"class":3089},"await",[3063,6657,6658],{"class":3158}," _ordersDb",[3063,6660,3010],{"class":3097},[3063,6662,6663],{"class":3158},"Orders\n",[3063,6665,6666,6668,6671,6673,6676,6678,6680,6682,6685],{"class":3065,"line":3298},[3063,6667,4015],{"class":3097},[3063,6669,6670],{"class":3283},"Include",[3063,6672,3287],{"class":3097},[3063,6674,6675],{"class":3158},"o",[3063,6677,3237],{"class":3097},[3063,6679,6675],{"class":3158},[3063,6681,3010],{"class":3097},[3063,6683,6684],{"class":3158},"Items",[3063,6686,3208],{"class":3097},[3063,6688,6689,6691,6694,6696,6698,6700,6702,6704,6706,6708,6711],{"class":3065,"line":3304},[3063,6690,4015],{"class":3097},[3063,6692,6693],{"class":3283},"FirstOrDefaultAsync",[3063,6695,3287],{"class":3097},[3063,6697,6675],{"class":3158},[3063,6699,3237],{"class":3097},[3063,6701,6675],{"class":3158},[3063,6703,3010],{"class":3097},[3063,6705,3740],{"class":3158},[3063,6707,3939],{"class":3097},[3063,6709,6710],{"class":3158},"orderId",[3063,6712,3208],{"class":3097},[3063,6714,6715,6718,6721,6723,6726,6728,6731,6733,6735,6737,6740],{"class":3065,"line":3333},[3063,6716,6717],{"class":3097},"            ?? ",[3063,6719,6720],{"class":3307},"throw",[3063,6722,3339],{"class":3089},[3063,6724,6725],{"class":3093}," NotFoundException",[3063,6727,3287],{"class":3097},[3063,6729,6730],{"class":3360},"$\"Order ",[3063,6732,3468],{"class":3467},[3063,6734,6710],{"class":3158},[3063,6736,3474],{"class":3467},[3063,6738,6739],{"class":3360}," not found.\"",[3063,6741,3364],{"class":3097},[3063,6743,6744],{"class":3065,"line":3357},[3063,6745,3083],{"emptyLinePlaceholder":3082},[3063,6747,6748],{"class":3065,"line":3367},[3063,6749,6750],{"class":3069},"        // Резервуємо товар в InventoryDb\n",[3063,6752,6753,6755,6757,6759,6762,6764,6766,6768,6770],{"class":3065,"line":3372},[3063,6754,4198],{"class":3307},[3063,6756,3311],{"class":3097},[3063,6758,4203],{"class":3089},[3063,6760,6761],{"class":3158}," item",[3063,6763,4209],{"class":3307},[3063,6765,6650],{"class":3158},[3063,6767,3010],{"class":3097},[3063,6769,6684],{"class":3158},[3063,6771,3208],{"class":3097},[3063,6773,6774],{"class":3065,"line":3384},[3063,6775,4584],{"class":3097},[3063,6777,6778,6781,6784,6786,6788,6790,6792],{"class":3065,"line":3390},[3063,6779,6780],{"class":3089},"            var",[3063,6782,6783],{"class":3158}," rowsAffected",[3063,6785,3377],{"class":3097},[3063,6787,6655],{"class":3089},[3063,6789,6536],{"class":3158},[3063,6791,3010],{"class":3097},[3063,6793,6794],{"class":3158},"StockLevels\n",[3063,6796,6797,6800,6802,6804,6806,6808,6810,6812,6815,6817,6820,6822,6824,6826,6828,6830,6833,6836,6838,6840,6843],{"class":3065,"line":3395},[3063,6798,6799],{"class":3097},"                .",[3063,6801,3868],{"class":3283},[3063,6803,3287],{"class":3097},[3063,6805,4048],{"class":3158},[3063,6807,3237],{"class":3097},[3063,6809,4048],{"class":3158},[3063,6811,3010],{"class":3097},[3063,6813,6814],{"class":3158},"ProductId",[3063,6816,3939],{"class":3097},[3063,6818,6819],{"class":3158},"item",[3063,6821,3010],{"class":3097},[3063,6823,6814],{"class":3158},[3063,6825,3247],{"class":3097},[3063,6827,4048],{"class":3158},[3063,6829,3010],{"class":3097},[3063,6831,6832],{"class":3158},"Available",[3063,6834,6835],{"class":3097}," >= ",[3063,6837,6819],{"class":3158},[3063,6839,3010],{"class":3097},[3063,6841,6842],{"class":3158},"Quantity",[3063,6844,3208],{"class":3097},[3063,6846,6847,6849,6851,6853,6855],{"class":3065,"line":3418},[3063,6848,6799],{"class":3097},[3063,6850,4043],{"class":3283},[3063,6852,3287],{"class":3097},[3063,6854,4048],{"class":3158},[3063,6856,3162],{"class":3097},[3063,6858,6859,6862,6864,6866,6868,6871,6873,6875,6877,6879,6881,6883,6885,6887,6889,6891,6894,6896,6898,6900],{"class":3065,"line":3440},[3063,6860,6861],{"class":3158},"                    s",[3063,6863,3010],{"class":3097},[3063,6865,4057],{"class":3283},[3063,6867,3287],{"class":3097},[3063,6869,6870],{"class":3158},"sl",[3063,6872,3237],{"class":3097},[3063,6874,6870],{"class":3158},[3063,6876,3010],{"class":3097},[3063,6878,6832],{"class":3158},[3063,6880,4072],{"class":3097},[3063,6882,6870],{"class":3158},[3063,6884,3237],{"class":3097},[3063,6886,6870],{"class":3158},[3063,6888,3010],{"class":3097},[3063,6890,6832],{"class":3158},[3063,6892,6893],{"class":3097}," - ",[3063,6895,6819],{"class":3158},[3063,6897,3010],{"class":3097},[3063,6899,6842],{"class":3158},[3063,6901,4077],{"class":3097},[3063,6903,6904],{"class":3065,"line":3445},[3063,6905,3083],{"emptyLinePlaceholder":3082},[3063,6907,6908,6910,6912,6915,6917,6919],{"class":3065,"line":3451},[3063,6909,4589],{"class":3307},[3063,6911,3311],{"class":3097},[3063,6913,6914],{"class":3158},"rowsAffected",[3063,6916,3939],{"class":3097},[3063,6918,3259],{"class":3204},[3063,6920,3208],{"class":3097},[3063,6922,6923,6926,6928,6931,6933,6935,6937,6939],{"class":3065,"line":3504},[3063,6924,6925],{"class":3307},"                throw",[3063,6927,3339],{"class":3089},[3063,6929,6930],{"class":3093}," InsufficientStockException",[3063,6932,3287],{"class":3097},[3063,6934,6819],{"class":3158},[3063,6936,3010],{"class":3097},[3063,6938,6814],{"class":3158},[3063,6940,3364],{"class":3097},[3063,6942,6943],{"class":3065,"line":3509},[3063,6944,4715],{"class":3097},[3063,6946,6947],{"class":3065,"line":3515},[3063,6948,3083],{"emptyLinePlaceholder":3082},[3063,6950,6951],{"class":3065,"line":3538},[3063,6952,6953],{"class":3069},"        // Оновлюємо статус в OrdersDb\n",[3063,6955,6956,6959,6961,6964,6966,6969],{"class":3065,"line":3543},[3063,6957,6958],{"class":3158},"        order",[3063,6960,3010],{"class":3097},[3063,6962,6963],{"class":3158},"Status",[3063,6965,6589],{"class":3097},[3063,6967,6968],{"class":3360},"\"Fulfilled\"",[3063,6970,3108],{"class":3097},[3063,6972,6973,6975,6977,6980,6982,6984,6986,6988],{"class":3065,"line":3567},[3063,6974,6958],{"class":3158},[3063,6976,3010],{"class":3097},[3063,6978,6979],{"class":3158},"FulfilledAt",[3063,6981,3377],{"class":3097},[3063,6983,4233],{"class":3158},[3063,6985,3010],{"class":3097},[3063,6987,4238],{"class":3158},[3063,6989,3108],{"class":3097},[3063,6991,6992,6995,6997,6999,7001],{"class":3065,"line":3572},[3063,6993,6994],{"class":3089},"        await",[3063,6996,6658],{"class":3158},[3063,6998,3010],{"class":3097},[3063,7000,4109],{"class":3283},[3063,7002,5634],{"class":3097},[3063,7004,7005],{"class":3065,"line":3593},[3063,7006,3387],{"class":3097},[3063,7008,7009],{"class":3065,"line":3611},[3063,7010,3785],{"class":3097},[3041,7012,7014],{"id":7013},"міграції-для-кількох-dbcontext","Міграції для кількох DbContext",[3021,7016,7018],{"className":4867,"code":7017,"language":4869,"meta":3029,"style":3029},"# Запускаємо міграції окремо для кожного DbContext!\ndotnet ef migrations add InitOrders --context OrdersDbContext\ndotnet ef migrations add InitInventory --context InventoryDbContext\n\n# Database update для кожного теж окремо:\ndotnet ef database update --context OrdersDbContext\ndotnet ef database update --context InventoryDbContext\n",[2984,7019,7020,7025,7045,7063,7067,7072,7088],{"__ignoreMap":3029},[3063,7021,7022],{"class":3065,"line":3066},[3063,7023,7024],{"class":3069},"# Запускаємо міграції окремо для кожного DbContext!\n",[3063,7026,7027,7029,7031,7034,7036,7039,7042],{"class":3065,"line":3073},[3063,7028,4881],{"class":3283},[3063,7030,4884],{"class":3360},[3063,7032,7033],{"class":3360}," migrations",[3063,7035,5285],{"class":3360},[3063,7037,7038],{"class":3360}," InitOrders",[3063,7040,7041],{"class":3089}," --context",[3063,7043,7044],{"class":3360}," OrdersDbContext\n",[3063,7046,7047,7049,7051,7053,7055,7058,7060],{"class":3065,"line":3079},[3063,7048,4881],{"class":3283},[3063,7050,4884],{"class":3360},[3063,7052,7033],{"class":3360},[3063,7054,5285],{"class":3360},[3063,7056,7057],{"class":3360}," InitInventory",[3063,7059,7041],{"class":3089},[3063,7061,7062],{"class":3360}," InventoryDbContext\n",[3063,7064,7065],{"class":3065,"line":3086},[3063,7066,3083],{"emptyLinePlaceholder":3082},[3063,7068,7069],{"class":3065,"line":3111},[3063,7070,7071],{"class":3069},"# Database update для кожного теж окремо:\n",[3063,7073,7074,7076,7078,7081,7084,7086],{"class":3065,"line":3116},[3063,7075,4881],{"class":3283},[3063,7077,4884],{"class":3360},[3063,7079,7080],{"class":3360}," database",[3063,7082,7083],{"class":3360}," update",[3063,7085,7041],{"class":3089},[3063,7087,7044],{"class":3360},[3063,7089,7090,7092,7094,7096,7098,7100],{"class":3065,"line":3122},[3063,7091,4881],{"class":3283},[3063,7093,4884],{"class":3360},[3063,7095,7080],{"class":3360},[3063,7097,7083],{"class":3360},[3063,7099,7041],{"class":3089},[3063,7101,7062],{"class":3360},[2963,7103,7104,7105,7108],{},"Для уникнення конфліктів між міграціями двох contexts — ",[2991,7106,7107],{},"папка міграцій"," має бути явно вказана:",[3021,7110,7112],{"className":3057,"code":7111,"language":3059,"meta":3029,"style":3029},"// OrdersDbContext: міграції у Migrations/Orders/\nbuilder.Services.AddDbContext\u003COrdersDbContext>(options =>\n    options.UseSqlServer(connectionString, sqlOptions =>\n        sqlOptions.MigrationsAssembly(\"YourApp.Infrastructure\")\n                  // Або через налаштування MigrationsHistoryTable зі своїм prefix\n    ));\n",[2984,7113,7114,7119,7141,7161,7178,7183],{"__ignoreMap":3029},[3063,7115,7116],{"class":3065,"line":3066},[3063,7117,7118],{"class":3069},"// OrdersDbContext: міграції у Migrations/Orders/\n",[3063,7120,7121,7123,7125,7127,7129,7131,7133,7135,7137,7139],{"class":3065,"line":3073},[3063,7122,6340],{"class":3158},[3063,7124,3010],{"class":3097},[3063,7126,6345],{"class":3158},[3063,7128,3010],{"class":3097},[3063,7130,6350],{"class":3283},[3063,7132,3523],{"class":3097},[3063,7134,6126],{"class":3093},[3063,7136,4744],{"class":3097},[3063,7138,6131],{"class":3158},[3063,7140,3162],{"class":3097},[3063,7142,7143,7145,7147,7149,7151,7154,7156,7159],{"class":3065,"line":3079},[3063,7144,6365],{"class":3158},[3063,7146,3010],{"class":3097},[3063,7148,6370],{"class":3283},[3063,7150,3287],{"class":3097},[3063,7152,7153],{"class":3158},"connectionString",[3063,7155,4072],{"class":3097},[3063,7157,7158],{"class":3158},"sqlOptions",[3063,7160,3162],{"class":3097},[3063,7162,7163,7166,7168,7171,7173,7176],{"class":3065,"line":3086},[3063,7164,7165],{"class":3158},"        sqlOptions",[3063,7167,3010],{"class":3097},[3063,7169,7170],{"class":3283},"MigrationsAssembly",[3063,7172,3287],{"class":3097},[3063,7174,7175],{"class":3360},"\"YourApp.Infrastructure\"",[3063,7177,3208],{"class":3097},[3063,7179,7180],{"class":3065,"line":3111},[3063,7181,7182],{"class":3069},"                  // Або через налаштування MigrationsHistoryTable зі своїм prefix\n",[3063,7184,7185],{"class":3065,"line":3116},[3063,7186,7187],{"class":3097},"    ));\n",[3021,7189,7191],{"className":3057,"code":7190,"language":3059,"meta":3029,"style":3029},"// Або через конфігурацію контексту:\npublic class OrdersDbContext : DbContext\n{\n    protected override void OnModelCreating(ModelBuilder modelBuilder)\n    {\n        modelBuilder.HasDefaultSchema(\"orders\"); // власна схема для ізоляції\n        // Тоді __EFMigrationsHistory буде у схемі \"orders\"\n    }\n}\n",[2984,7192,7193,7198,7210,7214,7232,7236,7257,7262,7266],{"__ignoreMap":3029},[3063,7194,7195],{"class":3065,"line":3066},[3063,7196,7197],{"class":3069},"// Або через конфігурацію контексту:\n",[3063,7199,7200,7202,7204,7206,7208],{"class":3065,"line":3073},[3063,7201,3125],{"class":3089},[3063,7203,3131],{"class":3089},[3063,7205,6101],{"class":3093},[3063,7207,4371],{"class":3097},[3063,7209,6106],{"class":3093},[3063,7211,7212],{"class":3065,"line":3079},[3063,7213,3140],{"class":3097},[3063,7215,7216,7218,7220,7222,7224,7226,7228,7230],{"class":3065,"line":3086},[3063,7217,4499],{"class":3089},[3063,7219,3721],{"class":3089},[3063,7221,3280],{"class":3089},[3063,7223,4506],{"class":3283},[3063,7225,3287],{"class":3097},[3063,7227,4511],{"class":3093},[3063,7229,4514],{"class":3158},[3063,7231,3208],{"class":3097},[3063,7233,7234],{"class":3065,"line":3111},[3063,7235,3301],{"class":3097},[3063,7237,7238,7241,7243,7246,7248,7251,7254],{"class":3065,"line":3116},[3063,7239,7240],{"class":3158},"        modelBuilder",[3063,7242,3010],{"class":3097},[3063,7244,7245],{"class":3283},"HasDefaultSchema",[3063,7247,3287],{"class":3097},[3063,7249,7250],{"class":3360},"\"orders\"",[3063,7252,7253],{"class":3097},"); ",[3063,7255,7256],{"class":3069},"// власна схема для ізоляції\n",[3063,7258,7259],{"class":3065,"line":3122},[3063,7260,7261],{"class":3069},"        // Тоді __EFMigrationsHistory буде у схемі \"orders\"\n",[3063,7263,7264],{"class":3065,"line":3137},[3063,7265,3387],{"class":3097},[3063,7267,7268],{"class":3065,"line":3143},[3063,7269,3785],{"class":3097},[3041,7271,7273],{"id":7272},"підхід-2-мультитенантна-архітектура-з-окремою-бд-на-тенант","Підхід 2: Мультитенантна архітектура з окремою БД на тенант",[3021,7275,7277],{"className":3057,"code":7276,"language":3059,"meta":3029,"style":3029},"// Tenant-per-database: DbContext створюється динамічно з правильним connection string\npublic class TenantDbContextFactory\n{\n    private readonly ITenantResolver _tenantResolver;\n    private readonly IConfiguration  _configuration;\n\n    public TenantDbContextFactory(\n        ITenantResolver tenantResolver,\n        IConfiguration  configuration)\n    {\n        _tenantResolver = tenantResolver;\n        _configuration  = configuration;\n    }\n\n    public AppDbContext CreateForCurrentTenant()\n    {\n        var tenantId         = _tenantResolver.GetCurrentTenantId();\n        var connectionString = GetConnectionStringForTenant(tenantId);\n\n        var options = new DbContextOptionsBuilder\u003CAppDbContext>()\n            .UseSqlServer(connectionString)\n            .Options;\n\n        return new AppDbContext(options);\n    }\n\n    private string GetConnectionStringForTenant(string tenantId)\n    {\n        // Читаємо з конфігурації або з \"master\" бази\n        return _configuration[$\"Tenants:{tenantId}:ConnectionString\"]\n            ?? throw new InvalidOperationException($\"No connection string for tenant {tenantId}\");\n    }\n}\n\n// Реєстрація через IDbContextFactory:\nbuilder.Services.AddDbContextFactory\u003CAppDbContext>((serviceProvider, options) =>\n{\n    var tenantResolver   = serviceProvider.GetRequiredService\u003CITenantResolver>();\n    var tenantId         = tenantResolver.GetCurrentTenantId();\n    var configuration    = serviceProvider.GetRequiredService\u003CIConfiguration>();\n    var connectionString = configuration[$\"Tenants:{tenantId}:ConnectionString\"];\n\n    options.UseSqlServer(connectionString);\n});\n",[2984,7278,7279,7284,7293,7297,7311,7325,7329,7338,7348,7358,7362,7374,7386,7390,7394,7406,7410,7430,7449,7453,7474,7486,7495,7499,7513,7517,7521,7538,7542,7547,7570,7596,7600,7604,7608,7613,7642,7646,7669,7685,7707,7732,7736,7750],{"__ignoreMap":3029},[3063,7280,7281],{"class":3065,"line":3066},[3063,7282,7283],{"class":3069},"// Tenant-per-database: DbContext створюється динамічно з правильним connection string\n",[3063,7285,7286,7288,7290],{"class":3065,"line":3073},[3063,7287,3125],{"class":3089},[3063,7289,3131],{"class":3089},[3063,7291,7292],{"class":3093}," TenantDbContextFactory\n",[3063,7294,7295],{"class":3065,"line":3079},[3063,7296,3140],{"class":3097},[3063,7298,7299,7301,7303,7306,7309],{"class":3065,"line":3086},[3063,7300,4728],{"class":3089},[3063,7302,6518],{"class":3089},[3063,7304,7305],{"class":3093}," ITenantResolver",[3063,7307,7308],{"class":3158}," _tenantResolver",[3063,7310,3108],{"class":3097},[3063,7312,7313,7315,7317,7320,7323],{"class":3065,"line":3111},[3063,7314,4728],{"class":3089},[3063,7316,6518],{"class":3089},[3063,7318,7319],{"class":3093}," IConfiguration",[3063,7321,7322],{"class":3158},"  _configuration",[3063,7324,3108],{"class":3097},[3063,7326,7327],{"class":3065,"line":3116},[3063,7328,3083],{"emptyLinePlaceholder":3082},[3063,7330,7331,7333,7336],{"class":3065,"line":3122},[3063,7332,3152],{"class":3089},[3063,7334,7335],{"class":3283}," TenantDbContextFactory",[3063,7337,6373],{"class":3097},[3063,7339,7340,7343,7346],{"class":3065,"line":3137},[3063,7341,7342],{"class":3093},"        ITenantResolver",[3063,7344,7345],{"class":3158}," tenantResolver",[3063,7347,6567],{"class":3097},[3063,7349,7350,7353,7356],{"class":3065,"line":3143},[3063,7351,7352],{"class":3093},"        IConfiguration",[3063,7354,7355],{"class":3158},"  configuration",[3063,7357,3208],{"class":3097},[3063,7359,7360],{"class":3065,"line":3149},[3063,7361,3301],{"class":3097},[3063,7363,7364,7367,7369,7372],{"class":3065,"line":3165},[3063,7365,7366],{"class":3158},"        _tenantResolver",[3063,7368,3377],{"class":3097},[3063,7370,7371],{"class":3158},"tenantResolver",[3063,7373,3108],{"class":3097},[3063,7375,7376,7379,7381,7384],{"class":3065,"line":3176},[3063,7377,7378],{"class":3158},"        _configuration",[3063,7380,5826],{"class":3097},[3063,7382,7383],{"class":3158},"configuration",[3063,7385,3108],{"class":3097},[3063,7387,7388],{"class":3065,"line":3211},[3063,7389,3387],{"class":3097},[3063,7391,7392],{"class":3065,"line":3221},[3063,7393,3083],{"emptyLinePlaceholder":3082},[3063,7395,7396,7398,7401,7404],{"class":3065,"line":3226},[3063,7397,3152],{"class":3089},[3063,7399,7400],{"class":3093}," AppDbContext",[3063,7402,7403],{"class":3283}," CreateForCurrentTenant",[3063,7405,3535],{"class":3097},[3063,7407,7408],{"class":3065,"line":3264},[3063,7409,3301],{"class":3097},[3063,7411,7412,7414,7417,7420,7423,7425,7428],{"class":3065,"line":3269},[3063,7413,3546],{"class":3089},[3063,7415,7416],{"class":3158}," tenantId",[3063,7418,7419],{"class":3097},"         = ",[3063,7421,7422],{"class":3158},"_tenantResolver",[3063,7424,3010],{"class":3097},[3063,7426,7427],{"class":3283},"GetCurrentTenantId",[3063,7429,5634],{"class":3097},[3063,7431,7432,7434,7437,7439,7442,7444,7447],{"class":3065,"line":3275},[3063,7433,3546],{"class":3089},[3063,7435,7436],{"class":3158}," connectionString",[3063,7438,3377],{"class":3097},[3063,7440,7441],{"class":3283},"GetConnectionStringForTenant",[3063,7443,3287],{"class":3097},[3063,7445,7446],{"class":3158},"tenantId",[3063,7448,3364],{"class":3097},[3063,7450,7451],{"class":3065,"line":3298},[3063,7452,3083],{"emptyLinePlaceholder":3082},[3063,7454,7455,7457,7460,7462,7464,7467,7469,7472],{"class":3065,"line":3304},[3063,7456,3546],{"class":3089},[3063,7458,7459],{"class":3158}," options",[3063,7461,3377],{"class":3097},[3063,7463,3554],{"class":3089},[3063,7465,7466],{"class":3093}," DbContextOptionsBuilder",[3063,7468,3523],{"class":3097},[3063,7470,7471],{"class":3093},"AppDbContext",[3063,7473,4010],{"class":3097},[3063,7475,7476,7478,7480,7482,7484],{"class":3065,"line":3333},[3063,7477,4015],{"class":3097},[3063,7479,6370],{"class":3283},[3063,7481,3287],{"class":3097},[3063,7483,7153],{"class":3158},[3063,7485,3208],{"class":3097},[3063,7487,7488,7490,7493],{"class":3065,"line":3357},[3063,7489,4015],{"class":3097},[3063,7491,7492],{"class":3158},"Options",[3063,7494,3108],{"class":3097},[3063,7496,7497],{"class":3065,"line":3367},[3063,7498,3083],{"emptyLinePlaceholder":3082},[3063,7500,7501,7503,7505,7507,7509,7511],{"class":3065,"line":3372},[3063,7502,3693],{"class":3307},[3063,7504,3339],{"class":3089},[3063,7506,7400],{"class":3093},[3063,7508,3287],{"class":3097},[3063,7510,6131],{"class":3158},[3063,7512,3364],{"class":3097},[3063,7514,7515],{"class":3065,"line":3384},[3063,7516,3387],{"class":3097},[3063,7518,7519],{"class":3065,"line":3390},[3063,7520,3083],{"emptyLinePlaceholder":3082},[3063,7522,7523,7525,7527,7530,7532,7534,7536],{"class":3065,"line":3395},[3063,7524,4728],{"class":3089},[3063,7526,3456],{"class":3089},[3063,7528,7529],{"class":3283}," GetConnectionStringForTenant",[3063,7531,3287],{"class":3097},[3063,7533,3526],{"class":3089},[3063,7535,7416],{"class":3158},[3063,7537,3208],{"class":3097},[3063,7539,7540],{"class":3065,"line":3418},[3063,7541,3301],{"class":3097},[3063,7543,7544],{"class":3065,"line":3440},[3063,7545,7546],{"class":3069},"        // Читаємо з конфігурації або з \"master\" бази\n",[3063,7548,7549,7551,7554,7556,7559,7561,7563,7565,7568],{"class":3065,"line":3445},[3063,7550,3693],{"class":3307},[3063,7552,7553],{"class":3158}," _configuration",[3063,7555,5769],{"class":3097},[3063,7557,7558],{"class":3360},"$\"Tenants:",[3063,7560,3468],{"class":3467},[3063,7562,7446],{"class":3158},[3063,7564,3474],{"class":3467},[3063,7566,7567],{"class":3360},":ConnectionString\"",[3063,7569,5775],{"class":3097},[3063,7571,7572,7574,7576,7578,7581,7583,7586,7588,7590,7592,7594],{"class":3065,"line":3451},[3063,7573,6717],{"class":3097},[3063,7575,6720],{"class":3307},[3063,7577,3339],{"class":3089},[3063,7579,7580],{"class":3093}," InvalidOperationException",[3063,7582,3287],{"class":3097},[3063,7584,7585],{"class":3360},"$\"No connection string for tenant ",[3063,7587,3468],{"class":3467},[3063,7589,7446],{"class":3158},[3063,7591,3474],{"class":3467},[3063,7593,5115],{"class":3360},[3063,7595,3364],{"class":3097},[3063,7597,7598],{"class":3065,"line":3504},[3063,7599,3387],{"class":3097},[3063,7601,7602],{"class":3065,"line":3509},[3063,7603,3785],{"class":3097},[3063,7605,7606],{"class":3065,"line":3515},[3063,7607,3083],{"emptyLinePlaceholder":3082},[3063,7609,7610],{"class":3065,"line":3538},[3063,7611,7612],{"class":3069},"// Реєстрація через IDbContextFactory:\n",[3063,7614,7615,7617,7619,7621,7623,7626,7628,7630,7633,7636,7638,7640],{"class":3065,"line":3543},[3063,7616,6340],{"class":3158},[3063,7618,3010],{"class":3097},[3063,7620,6345],{"class":3158},[3063,7622,3010],{"class":3097},[3063,7624,7625],{"class":3283},"AddDbContextFactory",[3063,7627,3523],{"class":3097},[3063,7629,7471],{"class":3093},[3063,7631,7632],{"class":3097},">((",[3063,7634,7635],{"class":3158},"serviceProvider",[3063,7637,4072],{"class":3097},[3063,7639,6131],{"class":3158},[3063,7641,3912],{"class":3097},[3063,7643,7644],{"class":3065,"line":3567},[3063,7645,3140],{"class":3097},[3063,7647,7648,7650,7652,7655,7657,7659,7662,7664,7667],{"class":3065,"line":3572},[3063,7649,5821],{"class":3089},[3063,7651,7345],{"class":3158},[3063,7653,7654],{"class":3097},"   = ",[3063,7656,7635],{"class":3158},[3063,7658,3010],{"class":3097},[3063,7660,7661],{"class":3283},"GetRequiredService",[3063,7663,3523],{"class":3097},[3063,7665,7666],{"class":3093},"ITenantResolver",[3063,7668,3564],{"class":3097},[3063,7670,7671,7673,7675,7677,7679,7681,7683],{"class":3065,"line":3593},[3063,7672,5821],{"class":3089},[3063,7674,7416],{"class":3158},[3063,7676,7419],{"class":3097},[3063,7678,7371],{"class":3158},[3063,7680,3010],{"class":3097},[3063,7682,7427],{"class":3283},[3063,7684,5634],{"class":3097},[3063,7686,7687,7689,7692,7694,7696,7698,7700,7702,7705],{"class":3065,"line":3611},[3063,7688,5821],{"class":3089},[3063,7690,7691],{"class":3158}," configuration",[3063,7693,6589],{"class":3097},[3063,7695,7635],{"class":3158},[3063,7697,3010],{"class":3097},[3063,7699,7661],{"class":3283},[3063,7701,3523],{"class":3097},[3063,7703,7704],{"class":3093},"IConfiguration",[3063,7706,3564],{"class":3097},[3063,7708,7709,7711,7713,7715,7717,7719,7721,7723,7725,7727,7729],{"class":3065,"line":3616},[3063,7710,5821],{"class":3089},[3063,7712,7436],{"class":3158},[3063,7714,3377],{"class":3097},[3063,7716,7383],{"class":3158},[3063,7718,5769],{"class":3097},[3063,7720,7558],{"class":3360},[3063,7722,3468],{"class":3467},[3063,7724,7446],{"class":3158},[3063,7726,3474],{"class":3467},[3063,7728,7567],{"class":3360},[3063,7730,7731],{"class":3097},"];\n",[3063,7733,7734],{"class":3065,"line":3632},[3063,7735,3083],{"emptyLinePlaceholder":3082},[3063,7737,7738,7740,7742,7744,7746,7748],{"class":3065,"line":3648},[3063,7739,6365],{"class":3158},[3063,7741,3010],{"class":3097},[3063,7743,6370],{"class":3283},[3063,7745,3287],{"class":3097},[3063,7747,7153],{"class":3158},[3063,7749,3364],{"class":3097},[3063,7751,7752],{"class":3065,"line":3653},[3063,7753,7754],{"class":3097},"});\n",[3041,7756,7758],{"id":7757},"підхід-3-readwrite-separation","Підхід 3: Read/Write Separation",[3021,7760,7762],{"className":3057,"code":7761,"language":3059,"meta":3029,"style":3029},"// Два DbContext: WriteDbContext → master, ReadDbContext → replica\npublic class WriteDbContext : AppDbContext\n{\n    public WriteDbContext(DbContextOptions\u003CWriteDbContext> options) : base(options) { }\n}\n\npublic class ReadDbContext : AppDbContext\n{\n    public ReadDbContext(DbContextOptions\u003CReadDbContext> options) : base(options) { }\n\n    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)\n    {\n        base.OnConfiguring(optionsBuilder);\n        // Read context: глобально NoTracking (все read-only)\n        optionsBuilder.UseQueryTrackingBehavior(QueryTrackingBehavior.NoTracking);\n    }\n}\n\n// Реєстрація:\nbuilder.Services.AddDbContext\u003CWriteDbContext>(options =>\n    options.UseSqlServer(masterConnectionString));\n\nbuilder.Services.AddDbContext\u003CReadDbContext>(options =>\n    options.UseSqlServer(replicaConnectionString));\n\n// Використання в CQRS:\npublic class GetOrderQueryHandler : IRequestHandler\u003CGetOrderQuery, OrderDto>\n{\n    private readonly ReadDbContext _readDb;  // ← завжди read replica\n\n    public async Task\u003COrderDto> Handle(GetOrderQuery request, CancellationToken ct)\n    {\n        return await _readDb.Orders\n            .Select(o => new OrderDto { ... })\n            .FirstOrDefaultAsync(o => o.Id == request.OrderId, ct)\n            ?? throw new NotFoundException();\n    }\n}\n\npublic class PlaceOrderCommandHandler : IRequestHandler\u003CPlaceOrderCommand, int>\n{\n    private readonly WriteDbContext _writeDb;  // ← завжди master\n\n    public async Task\u003Cint> Handle(PlaceOrderCommand request, CancellationToken ct)\n    {\n        var order = new Order { ... };\n        _writeDb.Orders.Add(order);\n        await _writeDb.SaveChangesAsync(ct);\n        return order.Id;\n    }\n}\n",[2984,7763,7764,7769,7783,7787,7816,7820,7824,7837,7841,7870,7874,7895,7899,7915,7920,7942,7946,7950,7954,7959,7981,7996,8000,8022,8037,8041,8046,8073,8077,8094,8098,8130,8134,8146,8167,8201,8213,8217,8221,8225,8249,8253,8269,8273,8303,8307,8323,8343,8359,8371,8375],{"__ignoreMap":3029},[3063,7765,7766],{"class":3065,"line":3066},[3063,7767,7768],{"class":3069},"// Два DbContext: WriteDbContext → master, ReadDbContext → replica\n",[3063,7770,7771,7773,7775,7778,7780],{"class":3065,"line":3073},[3063,7772,3125],{"class":3089},[3063,7774,3131],{"class":3089},[3063,7776,7777],{"class":3093}," WriteDbContext",[3063,7779,4371],{"class":3097},[3063,7781,7782],{"class":3093},"AppDbContext\n",[3063,7784,7785],{"class":3065,"line":3079},[3063,7786,3140],{"class":3097},[3063,7788,7789,7791,7793,7795,7797,7799,7802,7804,7806,7808,7810,7812,7814],{"class":3065,"line":3086},[3063,7790,3152],{"class":3089},[3063,7792,7777],{"class":3283},[3063,7794,3287],{"class":3097},[3063,7796,6121],{"class":3093},[3063,7798,3523],{"class":3097},[3063,7800,7801],{"class":3093},"WriteDbContext",[3063,7803,3529],{"class":3097},[3063,7805,6131],{"class":3158},[3063,7807,6134],{"class":3097},[3063,7809,6137],{"class":3089},[3063,7811,3287],{"class":3097},[3063,7813,6131],{"class":3158},[3063,7815,6144],{"class":3097},[3063,7817,7818],{"class":3065,"line":3111},[3063,7819,3785],{"class":3097},[3063,7821,7822],{"class":3065,"line":3116},[3063,7823,3083],{"emptyLinePlaceholder":3082},[3063,7825,7826,7828,7830,7833,7835],{"class":3065,"line":3122},[3063,7827,3125],{"class":3089},[3063,7829,3131],{"class":3089},[3063,7831,7832],{"class":3093}," ReadDbContext",[3063,7834,4371],{"class":3097},[3063,7836,7782],{"class":3093},[3063,7838,7839],{"class":3065,"line":3137},[3063,7840,3140],{"class":3097},[3063,7842,7843,7845,7847,7849,7851,7853,7856,7858,7860,7862,7864,7866,7868],{"class":3065,"line":3143},[3063,7844,3152],{"class":3089},[3063,7846,7832],{"class":3283},[3063,7848,3287],{"class":3097},[3063,7850,6121],{"class":3093},[3063,7852,3523],{"class":3097},[3063,7854,7855],{"class":3093},"ReadDbContext",[3063,7857,3529],{"class":3097},[3063,7859,6131],{"class":3158},[3063,7861,6134],{"class":3097},[3063,7863,6137],{"class":3089},[3063,7865,3287],{"class":3097},[3063,7867,6131],{"class":3158},[3063,7869,6144],{"class":3097},[3063,7871,7872],{"class":3065,"line":3149},[3063,7873,3083],{"emptyLinePlaceholder":3082},[3063,7875,7876,7878,7880,7882,7885,7887,7890,7893],{"class":3065,"line":3165},[3063,7877,4499],{"class":3089},[3063,7879,3721],{"class":3089},[3063,7881,3280],{"class":3089},[3063,7883,7884],{"class":3283}," OnConfiguring",[3063,7886,3287],{"class":3097},[3063,7888,7889],{"class":3093},"DbContextOptionsBuilder",[3063,7891,7892],{"class":3158}," optionsBuilder",[3063,7894,3208],{"class":3097},[3063,7896,7897],{"class":3065,"line":3176},[3063,7898,3301],{"class":3097},[3063,7900,7901,7903,7905,7908,7910,7913],{"class":3065,"line":3211},[3063,7902,4530],{"class":3089},[3063,7904,3010],{"class":3097},[3063,7906,7907],{"class":3283},"OnConfiguring",[3063,7909,3287],{"class":3097},[3063,7911,7912],{"class":3158},"optionsBuilder",[3063,7914,3364],{"class":3097},[3063,7916,7917],{"class":3065,"line":3221},[3063,7918,7919],{"class":3069},"        // Read context: глобально NoTracking (все read-only)\n",[3063,7921,7922,7925,7927,7930,7932,7935,7937,7940],{"class":3065,"line":3226},[3063,7923,7924],{"class":3158},"        optionsBuilder",[3063,7926,3010],{"class":3097},[3063,7928,7929],{"class":3283},"UseQueryTrackingBehavior",[3063,7931,3287],{"class":3097},[3063,7933,7934],{"class":3158},"QueryTrackingBehavior",[3063,7936,3010],{"class":3097},[3063,7938,7939],{"class":3158},"NoTracking",[3063,7941,3364],{"class":3097},[3063,7943,7944],{"class":3065,"line":3264},[3063,7945,3387],{"class":3097},[3063,7947,7948],{"class":3065,"line":3269},[3063,7949,3785],{"class":3097},[3063,7951,7952],{"class":3065,"line":3275},[3063,7953,3083],{"emptyLinePlaceholder":3082},[3063,7955,7956],{"class":3065,"line":3298},[3063,7957,7958],{"class":3069},"// Реєстрація:\n",[3063,7960,7961,7963,7965,7967,7969,7971,7973,7975,7977,7979],{"class":3065,"line":3304},[3063,7962,6340],{"class":3158},[3063,7964,3010],{"class":3097},[3063,7966,6345],{"class":3158},[3063,7968,3010],{"class":3097},[3063,7970,6350],{"class":3283},[3063,7972,3523],{"class":3097},[3063,7974,7801],{"class":3093},[3063,7976,4744],{"class":3097},[3063,7978,6131],{"class":3158},[3063,7980,3162],{"class":3097},[3063,7982,7983,7985,7987,7989,7991,7994],{"class":3065,"line":3333},[3063,7984,6365],{"class":3158},[3063,7986,3010],{"class":3097},[3063,7988,6370],{"class":3283},[3063,7990,3287],{"class":3097},[3063,7992,7993],{"class":3158},"masterConnectionString",[3063,7995,4077],{"class":3097},[3063,7997,7998],{"class":3065,"line":3357},[3063,7999,3083],{"emptyLinePlaceholder":3082},[3063,8001,8002,8004,8006,8008,8010,8012,8014,8016,8018,8020],{"class":3065,"line":3367},[3063,8003,6340],{"class":3158},[3063,8005,3010],{"class":3097},[3063,8007,6345],{"class":3158},[3063,8009,3010],{"class":3097},[3063,8011,6350],{"class":3283},[3063,8013,3523],{"class":3097},[3063,8015,7855],{"class":3093},[3063,8017,4744],{"class":3097},[3063,8019,6131],{"class":3158},[3063,8021,3162],{"class":3097},[3063,8023,8024,8026,8028,8030,8032,8035],{"class":3065,"line":3372},[3063,8025,6365],{"class":3158},[3063,8027,3010],{"class":3097},[3063,8029,6370],{"class":3283},[3063,8031,3287],{"class":3097},[3063,8033,8034],{"class":3158},"replicaConnectionString",[3063,8036,4077],{"class":3097},[3063,8038,8039],{"class":3065,"line":3384},[3063,8040,3083],{"emptyLinePlaceholder":3082},[3063,8042,8043],{"class":3065,"line":3390},[3063,8044,8045],{"class":3069},"// Використання в CQRS:\n",[3063,8047,8048,8050,8052,8055,8057,8060,8062,8065,8067,8070],{"class":3065,"line":3395},[3063,8049,3125],{"class":3089},[3063,8051,3131],{"class":3089},[3063,8053,8054],{"class":3093}," GetOrderQueryHandler",[3063,8056,4371],{"class":3097},[3063,8058,8059],{"class":3093},"IRequestHandler",[3063,8061,3523],{"class":3097},[3063,8063,8064],{"class":3093},"GetOrderQuery",[3063,8066,4072],{"class":3097},[3063,8068,8069],{"class":3093},"OrderDto",[3063,8071,8072],{"class":3097},">\n",[3063,8074,8075],{"class":3065,"line":3418},[3063,8076,3140],{"class":3097},[3063,8078,8079,8081,8083,8085,8088,8091],{"class":3065,"line":3440},[3063,8080,4728],{"class":3089},[3063,8082,6518],{"class":3089},[3063,8084,7832],{"class":3093},[3063,8086,8087],{"class":3158}," _readDb",[3063,8089,8090],{"class":3097},";  ",[3063,8092,8093],{"class":3069},"// ← завжди read replica\n",[3063,8095,8096],{"class":3065,"line":3445},[3063,8097,3083],{"emptyLinePlaceholder":3082},[3063,8099,8100,8102,8104,8106,8108,8110,8112,8115,8117,8119,8122,8124,8126,8128],{"class":3065,"line":3451},[3063,8101,3152],{"class":3089},[3063,8103,3968],{"class":3089},[3063,8105,3971],{"class":3093},[3063,8107,3523],{"class":3097},[3063,8109,8069],{"class":3093},[3063,8111,3529],{"class":3097},[3063,8113,8114],{"class":3283},"Handle",[3063,8116,3287],{"class":3097},[3063,8118,8064],{"class":3093},[3063,8120,8121],{"class":3158}," request",[3063,8123,4072],{"class":3097},[3063,8125,4114],{"class":3093},[3063,8127,4117],{"class":3158},[3063,8129,3208],{"class":3097},[3063,8131,8132],{"class":3065,"line":3504},[3063,8133,3301],{"class":3097},[3063,8135,8136,8138,8140,8142,8144],{"class":3065,"line":3509},[3063,8137,3693],{"class":3307},[3063,8139,4000],{"class":3089},[3063,8141,8087],{"class":3158},[3063,8143,3010],{"class":3097},[3063,8145,6663],{"class":3158},[3063,8147,8148,8150,8153,8155,8157,8159,8161,8164],{"class":3065,"line":3515},[3063,8149,4015],{"class":3097},[3063,8151,8152],{"class":3283},"Select",[3063,8154,3287],{"class":3097},[3063,8156,6675],{"class":3158},[3063,8158,3237],{"class":3097},[3063,8160,3554],{"class":3089},[3063,8162,8163],{"class":3093}," OrderDto",[3063,8165,8166],{"class":3097}," { ... })\n",[3063,8168,8169,8171,8173,8175,8177,8179,8181,8183,8185,8187,8190,8192,8195,8197,8199],{"class":3065,"line":3538},[3063,8170,4015],{"class":3097},[3063,8172,6693],{"class":3283},[3063,8174,3287],{"class":3097},[3063,8176,6675],{"class":3158},[3063,8178,3237],{"class":3097},[3063,8180,6675],{"class":3158},[3063,8182,3010],{"class":3097},[3063,8184,3740],{"class":3158},[3063,8186,3939],{"class":3097},[3063,8188,8189],{"class":3158},"request",[3063,8191,3010],{"class":3097},[3063,8193,8194],{"class":3158},"OrderId",[3063,8196,4072],{"class":3097},[3063,8198,4262],{"class":3158},[3063,8200,3208],{"class":3097},[3063,8202,8203,8205,8207,8209,8211],{"class":3065,"line":3543},[3063,8204,6717],{"class":3097},[3063,8206,6720],{"class":3307},[3063,8208,3339],{"class":3089},[3063,8210,6725],{"class":3093},[3063,8212,5634],{"class":3097},[3063,8214,8215],{"class":3065,"line":3567},[3063,8216,3387],{"class":3097},[3063,8218,8219],{"class":3065,"line":3572},[3063,8220,3785],{"class":3097},[3063,8222,8223],{"class":3065,"line":3593},[3063,8224,3083],{"emptyLinePlaceholder":3082},[3063,8226,8227,8229,8231,8234,8236,8238,8240,8243,8245,8247],{"class":3065,"line":3611},[3063,8228,3125],{"class":3089},[3063,8230,3131],{"class":3089},[3063,8232,8233],{"class":3093}," PlaceOrderCommandHandler",[3063,8235,4371],{"class":3097},[3063,8237,8059],{"class":3093},[3063,8239,3523],{"class":3097},[3063,8241,8242],{"class":3093},"PlaceOrderCommand",[3063,8244,4072],{"class":3097},[3063,8246,3906],{"class":3089},[3063,8248,8072],{"class":3097},[3063,8250,8251],{"class":3065,"line":3616},[3063,8252,3140],{"class":3097},[3063,8254,8255,8257,8259,8261,8264,8266],{"class":3065,"line":3632},[3063,8256,4728],{"class":3089},[3063,8258,6518],{"class":3089},[3063,8260,7777],{"class":3093},[3063,8262,8263],{"class":3158}," _writeDb",[3063,8265,8090],{"class":3097},[3063,8267,8268],{"class":3069},"// ← завжди master\n",[3063,8270,8271],{"class":3065,"line":3648},[3063,8272,3083],{"emptyLinePlaceholder":3082},[3063,8274,8275,8277,8279,8281,8283,8285,8287,8289,8291,8293,8295,8297,8299,8301],{"class":3065,"line":3653},[3063,8276,3152],{"class":3089},[3063,8278,3968],{"class":3089},[3063,8280,3971],{"class":3093},[3063,8282,3523],{"class":3097},[3063,8284,3906],{"class":3089},[3063,8286,3529],{"class":3097},[3063,8288,8114],{"class":3283},[3063,8290,3287],{"class":3097},[3063,8292,8242],{"class":3093},[3063,8294,8121],{"class":3158},[3063,8296,4072],{"class":3097},[3063,8298,4114],{"class":3093},[3063,8300,4117],{"class":3158},[3063,8302,3208],{"class":3097},[3063,8304,8305],{"class":3065,"line":3669},[3063,8306,3301],{"class":3097},[3063,8308,8309,8311,8313,8315,8317,8320],{"class":3065,"line":3685},[3063,8310,3546],{"class":3089},[3063,8312,6650],{"class":3158},[3063,8314,3377],{"class":3097},[3063,8316,3554],{"class":3089},[3063,8318,8319],{"class":3093}," Order",[3063,8321,8322],{"class":3097}," { ... };\n",[3063,8324,8325,8328,8330,8332,8334,8336,8338,8341],{"class":3065,"line":3690},[3063,8326,8327],{"class":3158},"        _writeDb",[3063,8329,3010],{"class":3097},[3063,8331,6166],{"class":3158},[3063,8333,3010],{"class":3097},[3063,8335,3601],{"class":3283},[3063,8337,3287],{"class":3097},[3063,8339,8340],{"class":3158},"order",[3063,8342,3364],{"class":3097},[3063,8344,8345,8347,8349,8351,8353,8355,8357],{"class":3065,"line":3700},[3063,8346,6994],{"class":3089},[3063,8348,8263],{"class":3158},[3063,8350,3010],{"class":3097},[3063,8352,4109],{"class":3283},[3063,8354,3287],{"class":3097},[3063,8356,4262],{"class":3158},[3063,8358,3364],{"class":3097},[3063,8360,8361,8363,8365,8367,8369],{"class":3065,"line":3705},[3063,8362,3693],{"class":3307},[3063,8364,6650],{"class":3158},[3063,8366,3010],{"class":3097},[3063,8368,3740],{"class":3158},[3063,8370,3108],{"class":3097},[3063,8372,8373],{"class":3065,"line":3710},[3063,8374,3387],{"class":3097},[3063,8376,8377],{"class":3065,"line":3716},[3063,8378,3785],{"class":3097},[3041,8380,8382],{"id":8381},"mermaid-архітектура-multi-database-сценарію","Mermaid: архітектура Multi-Database сценарію",[8384,8385,8386],"mermaid",{},[3021,8387,8390],{"className":8388,"code":8389,"language":8384,"meta":3029,"style":3029},"language-mermaid shiki shiki-themes light-plus dark-plus dark-plus","graph TB\n    API[\"ASP.NET Core API\u003Cbr/>(один процес)\"]\n\n    ODB[\"OrdersDbContext\u003Cbr/>SQL Server\u003Cbr/>:1433\"]\n    IDB[\"InventoryDbContext\u003Cbr/>PostgreSQL\u003Cbr/>:5432\"]\n    RDB[\"ReadDbContext\u003Cbr/>SQL Server Replica\u003Cbr/>:1433\"]\n    WDB[\"WriteDbContext\u003Cbr/>SQL Server Master\u003Cbr/>:1433\"]\n\n    API -->|\"CRUD замовлень\"| ODB\n    API -->|\"Резервування товарів\"| IDB\n    API -->|\"Query handlers (CQRS)\"| RDB\n    API -->|\"Command handlers (CQRS)\"| WDB\n\n    ODB --> SQLSRV[\"SQL Server\u003Cbr/>OrdersDb\"]\n    IDB --> PG[\"PostgreSQL\u003Cbr/>InventoryDb\"]\n    RDB --> REPLICA[\"SQL Server\u003Cbr/>Read Replica\"]\n    WDB --> MASTER[\"SQL Server\u003Cbr/>Master (R/W)\"]\n\n    MASTER -->|\"Async Replication\"| REPLICA\n\n    style API fill:#3b82f6,stroke:#1d4ed8,color:#fff\n    style SQLSRV fill:#64748b,stroke:#334155,color:#fff\n    style PG fill:#64748b,stroke:#334155,color:#fff\n    style REPLICA fill:#10b981,stroke:#059669,color:#fff\n    style MASTER fill:#f59e0b,stroke:#b45309,color:#fff\n",[2984,8391,8392,8397,8402,8406,8411,8416,8421,8426,8430,8435,8440,8445,8450,8454,8459,8464,8469,8474,8478,8483,8487,8492,8497,8502,8507],{"__ignoreMap":3029},[3063,8393,8394],{"class":3065,"line":3066},[3063,8395,8396],{},"graph TB\n",[3063,8398,8399],{"class":3065,"line":3073},[3063,8400,8401],{},"    API[\"ASP.NET Core API\u003Cbr/>(один процес)\"]\n",[3063,8403,8404],{"class":3065,"line":3079},[3063,8405,3083],{"emptyLinePlaceholder":3082},[3063,8407,8408],{"class":3065,"line":3086},[3063,8409,8410],{},"    ODB[\"OrdersDbContext\u003Cbr/>SQL Server\u003Cbr/>:1433\"]\n",[3063,8412,8413],{"class":3065,"line":3111},[3063,8414,8415],{},"    IDB[\"InventoryDbContext\u003Cbr/>PostgreSQL\u003Cbr/>:5432\"]\n",[3063,8417,8418],{"class":3065,"line":3116},[3063,8419,8420],{},"    RDB[\"ReadDbContext\u003Cbr/>SQL Server Replica\u003Cbr/>:1433\"]\n",[3063,8422,8423],{"class":3065,"line":3122},[3063,8424,8425],{},"    WDB[\"WriteDbContext\u003Cbr/>SQL Server Master\u003Cbr/>:1433\"]\n",[3063,8427,8428],{"class":3065,"line":3137},[3063,8429,3083],{"emptyLinePlaceholder":3082},[3063,8431,8432],{"class":3065,"line":3143},[3063,8433,8434],{},"    API -->|\"CRUD замовлень\"| ODB\n",[3063,8436,8437],{"class":3065,"line":3149},[3063,8438,8439],{},"    API -->|\"Резервування товарів\"| IDB\n",[3063,8441,8442],{"class":3065,"line":3165},[3063,8443,8444],{},"    API -->|\"Query handlers (CQRS)\"| RDB\n",[3063,8446,8447],{"class":3065,"line":3176},[3063,8448,8449],{},"    API -->|\"Command handlers (CQRS)\"| WDB\n",[3063,8451,8452],{"class":3065,"line":3211},[3063,8453,3083],{"emptyLinePlaceholder":3082},[3063,8455,8456],{"class":3065,"line":3221},[3063,8457,8458],{},"    ODB --> SQLSRV[\"SQL Server\u003Cbr/>OrdersDb\"]\n",[3063,8460,8461],{"class":3065,"line":3226},[3063,8462,8463],{},"    IDB --> PG[\"PostgreSQL\u003Cbr/>InventoryDb\"]\n",[3063,8465,8466],{"class":3065,"line":3264},[3063,8467,8468],{},"    RDB --> REPLICA[\"SQL Server\u003Cbr/>Read Replica\"]\n",[3063,8470,8471],{"class":3065,"line":3269},[3063,8472,8473],{},"    WDB --> MASTER[\"SQL Server\u003Cbr/>Master (R/W)\"]\n",[3063,8475,8476],{"class":3065,"line":3275},[3063,8477,3083],{"emptyLinePlaceholder":3082},[3063,8479,8480],{"class":3065,"line":3298},[3063,8481,8482],{},"    MASTER -->|\"Async Replication\"| REPLICA\n",[3063,8484,8485],{"class":3065,"line":3304},[3063,8486,3083],{"emptyLinePlaceholder":3082},[3063,8488,8489],{"class":3065,"line":3333},[3063,8490,8491],{},"    style API fill:#3b82f6,stroke:#1d4ed8,color:#fff\n",[3063,8493,8494],{"class":3065,"line":3357},[3063,8495,8496],{},"    style SQLSRV fill:#64748b,stroke:#334155,color:#fff\n",[3063,8498,8499],{"class":3065,"line":3367},[3063,8500,8501],{},"    style PG fill:#64748b,stroke:#334155,color:#fff\n",[3063,8503,8504],{"class":3065,"line":3372},[3063,8505,8506],{},"    style REPLICA fill:#10b981,stroke:#059669,color:#fff\n",[3063,8508,8509],{"class":3065,"line":3384},[3063,8510,8511],{},"    style MASTER fill:#f59e0b,stroke:#b45309,color:#fff\n",[2973,8513],{},[2976,8515,8517],{"id":8516},"fluent-api-vs-scaffolded-data-annotations-повне-порівняння","Fluent API vs Scaffolded Data Annotations: повне порівняння",[2963,8519,8520],{},"Це питання виникає при Database-First: яку конфігурацію обирати і як її підтримувати довгостроково.",[3041,8522,8524],{"id":8523},"що-генерує-scaffold-за-замовчуванням-fluent-api","Що генерує scaffold за замовчуванням (Fluent API)",[3021,8526,8528],{"className":3057,"code":8527,"language":3059,"meta":3029,"style":3029},"// OnModelCreating — детальна конфігурація\nmodelBuilder.Entity\u003CProduct>(entity =>\n{\n    entity.ToTable(\"Products\");\n    entity.HasKey(e => e.Id);\n    entity.Property(e => e.Name)\n          .IsRequired()\n          .HasMaxLength(200);\n    entity.Property(e => e.Price)\n          .HasColumnType(\"decimal(10,2)\");\n    entity.HasCheckConstraint(\"CK_Price_Positive\", \"[Price] >= 0\");\n\n    entity.HasOne(e => e.Category)\n          .WithMany(e => e.Products)\n          .HasForeignKey(e => e.CategoryId)\n          .HasConstraintName(\"FK_Products_Categories\");\n});\n",[2984,8529,8530,8535,8554,8558,8575,8598,8621,8631,8645,8667,8681,8702,8706,8729,8750,8771,8785],{"__ignoreMap":3029},[3063,8531,8532],{"class":3065,"line":3066},[3063,8533,8534],{"class":3069},"// OnModelCreating — детальна конфігурація\n",[3063,8536,8537,8539,8541,8543,8545,8547,8549,8552],{"class":3065,"line":3073},[3063,8538,4540],{"class":3158},[3063,8540,3010],{"class":3097},[3063,8542,4223],{"class":3283},[3063,8544,3523],{"class":3097},[3063,8546,3039],{"class":3093},[3063,8548,4744],{"class":3097},[3063,8550,8551],{"class":3158},"entity",[3063,8553,3162],{"class":3097},[3063,8555,8556],{"class":3065,"line":3079},[3063,8557,3140],{"class":3097},[3063,8559,8560,8563,8565,8568,8570,8573],{"class":3065,"line":3086},[3063,8561,8562],{"class":3158},"    entity",[3063,8564,3010],{"class":3097},[3063,8566,8567],{"class":3283},"ToTable",[3063,8569,3287],{"class":3097},[3063,8571,8572],{"class":3360},"\"Products\"",[3063,8574,3364],{"class":3097},[3063,8576,8577,8579,8581,8584,8586,8588,8590,8592,8594,8596],{"class":3065,"line":3111},[3063,8578,8562],{"class":3158},[3063,8580,3010],{"class":3097},[3063,8582,8583],{"class":3283},"HasKey",[3063,8585,3287],{"class":3097},[3063,8587,4168],{"class":3158},[3063,8589,3237],{"class":3097},[3063,8591,4168],{"class":3158},[3063,8593,3010],{"class":3097},[3063,8595,3740],{"class":3158},[3063,8597,3364],{"class":3097},[3063,8599,8600,8602,8604,8607,8609,8611,8613,8615,8617,8619],{"class":3065,"line":3116},[3063,8601,8562],{"class":3158},[3063,8603,3010],{"class":3097},[3063,8605,8606],{"class":3283},"Property",[3063,8608,3287],{"class":3097},[3063,8610,4168],{"class":3158},[3063,8612,3237],{"class":3097},[3063,8614,4168],{"class":3158},[3063,8616,3010],{"class":3097},[3063,8618,3471],{"class":3158},[3063,8620,3208],{"class":3097},[3063,8622,8623,8626,8629],{"class":3065,"line":3122},[3063,8624,8625],{"class":3097},"          .",[3063,8627,8628],{"class":3283},"IsRequired",[3063,8630,3535],{"class":3097},[3063,8632,8633,8635,8638,8640,8643],{"class":3065,"line":3137},[3063,8634,8625],{"class":3097},[3063,8636,8637],{"class":3283},"HasMaxLength",[3063,8639,3287],{"class":3097},[3063,8641,8642],{"class":3204},"200",[3063,8644,3364],{"class":3097},[3063,8646,8647,8649,8651,8653,8655,8657,8659,8661,8663,8665],{"class":3065,"line":3143},[3063,8648,8562],{"class":3158},[3063,8650,3010],{"class":3097},[3063,8652,8606],{"class":3283},[3063,8654,3287],{"class":3097},[3063,8656,4168],{"class":3158},[3063,8658,3237],{"class":3097},[3063,8660,4168],{"class":3158},[3063,8662,3010],{"class":3097},[3063,8664,3182],{"class":3158},[3063,8666,3208],{"class":3097},[3063,8668,8669,8671,8674,8676,8679],{"class":3065,"line":3149},[3063,8670,8625],{"class":3097},[3063,8672,8673],{"class":3283},"HasColumnType",[3063,8675,3287],{"class":3097},[3063,8677,8678],{"class":3360},"\"decimal(10,2)\"",[3063,8680,3364],{"class":3097},[3063,8682,8683,8685,8687,8690,8692,8695,8697,8700],{"class":3065,"line":3165},[3063,8684,8562],{"class":3158},[3063,8686,3010],{"class":3097},[3063,8688,8689],{"class":3283},"HasCheckConstraint",[3063,8691,3287],{"class":3097},[3063,8693,8694],{"class":3360},"\"CK_Price_Positive\"",[3063,8696,4072],{"class":3097},[3063,8698,8699],{"class":3360},"\"[Price] >= 0\"",[3063,8701,3364],{"class":3097},[3063,8703,8704],{"class":3065,"line":3176},[3063,8705,3083],{"emptyLinePlaceholder":3082},[3063,8707,8708,8710,8712,8715,8717,8719,8721,8723,8725,8727],{"class":3065,"line":3211},[3063,8709,8562],{"class":3158},[3063,8711,3010],{"class":3097},[3063,8713,8714],{"class":3283},"HasOne",[3063,8716,3287],{"class":3097},[3063,8718,4168],{"class":3158},[3063,8720,3237],{"class":3097},[3063,8722,4168],{"class":3158},[3063,8724,3010],{"class":3097},[3063,8726,3481],{"class":3158},[3063,8728,3208],{"class":3097},[3063,8730,8731,8733,8736,8738,8740,8742,8744,8746,8748],{"class":3065,"line":3221},[3063,8732,8625],{"class":3097},[3063,8734,8735],{"class":3283},"WithMany",[3063,8737,3287],{"class":3097},[3063,8739,4168],{"class":3158},[3063,8741,3237],{"class":3097},[3063,8743,4168],{"class":3158},[3063,8745,3010],{"class":3097},[3063,8747,6280],{"class":3158},[3063,8749,3208],{"class":3097},[3063,8751,8752,8754,8757,8759,8761,8763,8765,8767,8769],{"class":3065,"line":3226},[3063,8753,8625],{"class":3097},[3063,8755,8756],{"class":3283},"HasForeignKey",[3063,8758,3287],{"class":3097},[3063,8760,4168],{"class":3158},[3063,8762,3237],{"class":3097},[3063,8764,4168],{"class":3158},[3063,8766,3010],{"class":3097},[3063,8768,3660],{"class":3158},[3063,8770,3208],{"class":3097},[3063,8772,8773,8775,8778,8780,8783],{"class":3065,"line":3264},[3063,8774,8625],{"class":3097},[3063,8776,8777],{"class":3283},"HasConstraintName",[3063,8779,3287],{"class":3097},[3063,8781,8782],{"class":3360},"\"FK_Products_Categories\"",[3063,8784,3364],{"class":3097},[3063,8786,8787],{"class":3065,"line":3269},[3063,8788,7754],{"class":3097},[3041,8790,8792],{"id":8791},"що-можна-замінити-data-annotations-і-що-ні","Що можна замінити Data Annotations (і що — ні)",[8794,8795,8796,9096],"tabs",{},[8797,8798,8800],"tabs-item",{"label":8799},"Fluent API (рекомендовано)",[3021,8801,8803],{"className":3057,"code":8802,"language":3059,"meta":3029,"style":3029},"// Повна конфігурація — все можливо:\nmodelBuilder.Entity\u003CProduct>(entity =>\n{\n    entity.ToTable(\"Products\", \"catalog\");  // схема\n    entity.HasKey(e => e.Id);\n\n    entity.Property(e => e.Name)\n          .HasMaxLength(200)\n          .IsRequired();\n\n    entity.Property(e => e.Price)\n          .HasPrecision(10, 2);\n\n    // Лише через Fluent API:\n    entity.HasCheckConstraint(\"CK_Price\", \"[Price] >= 0\");\n    entity.Property(e => e.CreatedAt)\n          .HasDefaultValueSql(\"GETUTCDATE()\");\n    entity.HasIndex(e => new { e.Name, e.CategoryId })\n          .IsUnique()\n          .HasDatabaseName(\"UX_Products_Name_Category\");\n});\n",[2984,8804,8805,8810,8828,8832,8855,8877,8881,8903,8915,8923,8927,8949,8968,8972,8977,8996,9019,9033,9069,9078,9092],{"__ignoreMap":3029},[3063,8806,8807],{"class":3065,"line":3066},[3063,8808,8809],{"class":3069},"// Повна конфігурація — все можливо:\n",[3063,8811,8812,8814,8816,8818,8820,8822,8824,8826],{"class":3065,"line":3073},[3063,8813,4540],{"class":3158},[3063,8815,3010],{"class":3097},[3063,8817,4223],{"class":3283},[3063,8819,3523],{"class":3097},[3063,8821,3039],{"class":3093},[3063,8823,4744],{"class":3097},[3063,8825,8551],{"class":3158},[3063,8827,3162],{"class":3097},[3063,8829,8830],{"class":3065,"line":3079},[3063,8831,3140],{"class":3097},[3063,8833,8834,8836,8838,8840,8842,8844,8846,8849,8852],{"class":3065,"line":3086},[3063,8835,8562],{"class":3158},[3063,8837,3010],{"class":3097},[3063,8839,8567],{"class":3283},[3063,8841,3287],{"class":3097},[3063,8843,8572],{"class":3360},[3063,8845,4072],{"class":3097},[3063,8847,8848],{"class":3360},"\"catalog\"",[3063,8850,8851],{"class":3097},");  ",[3063,8853,8854],{"class":3069},"// схема\n",[3063,8856,8857,8859,8861,8863,8865,8867,8869,8871,8873,8875],{"class":3065,"line":3111},[3063,8858,8562],{"class":3158},[3063,8860,3010],{"class":3097},[3063,8862,8583],{"class":3283},[3063,8864,3287],{"class":3097},[3063,8866,4168],{"class":3158},[3063,8868,3237],{"class":3097},[3063,8870,4168],{"class":3158},[3063,8872,3010],{"class":3097},[3063,8874,3740],{"class":3158},[3063,8876,3364],{"class":3097},[3063,8878,8879],{"class":3065,"line":3116},[3063,8880,3083],{"emptyLinePlaceholder":3082},[3063,8882,8883,8885,8887,8889,8891,8893,8895,8897,8899,8901],{"class":3065,"line":3122},[3063,8884,8562],{"class":3158},[3063,8886,3010],{"class":3097},[3063,8888,8606],{"class":3283},[3063,8890,3287],{"class":3097},[3063,8892,4168],{"class":3158},[3063,8894,3237],{"class":3097},[3063,8896,4168],{"class":3158},[3063,8898,3010],{"class":3097},[3063,8900,3471],{"class":3158},[3063,8902,3208],{"class":3097},[3063,8904,8905,8907,8909,8911,8913],{"class":3065,"line":3137},[3063,8906,8625],{"class":3097},[3063,8908,8637],{"class":3283},[3063,8910,3287],{"class":3097},[3063,8912,8642],{"class":3204},[3063,8914,3208],{"class":3097},[3063,8916,8917,8919,8921],{"class":3065,"line":3143},[3063,8918,8625],{"class":3097},[3063,8920,8628],{"class":3283},[3063,8922,5634],{"class":3097},[3063,8924,8925],{"class":3065,"line":3149},[3063,8926,3083],{"emptyLinePlaceholder":3082},[3063,8928,8929,8931,8933,8935,8937,8939,8941,8943,8945,8947],{"class":3065,"line":3165},[3063,8930,8562],{"class":3158},[3063,8932,3010],{"class":3097},[3063,8934,8606],{"class":3283},[3063,8936,3287],{"class":3097},[3063,8938,4168],{"class":3158},[3063,8940,3237],{"class":3097},[3063,8942,4168],{"class":3158},[3063,8944,3010],{"class":3097},[3063,8946,3182],{"class":3158},[3063,8948,3208],{"class":3097},[3063,8950,8951,8953,8956,8958,8961,8963,8966],{"class":3065,"line":3176},[3063,8952,8625],{"class":3097},[3063,8954,8955],{"class":3283},"HasPrecision",[3063,8957,3287],{"class":3097},[3063,8959,8960],{"class":3204},"10",[3063,8962,4072],{"class":3097},[3063,8964,8965],{"class":3204},"2",[3063,8967,3364],{"class":3097},[3063,8969,8970],{"class":3065,"line":3211},[3063,8971,3083],{"emptyLinePlaceholder":3082},[3063,8973,8974],{"class":3065,"line":3221},[3063,8975,8976],{"class":3069},"    // Лише через Fluent API:\n",[3063,8978,8979,8981,8983,8985,8987,8990,8992,8994],{"class":3065,"line":3226},[3063,8980,8562],{"class":3158},[3063,8982,3010],{"class":3097},[3063,8984,8689],{"class":3283},[3063,8986,3287],{"class":3097},[3063,8988,8989],{"class":3360},"\"CK_Price\"",[3063,8991,4072],{"class":3097},[3063,8993,8699],{"class":3360},[3063,8995,3364],{"class":3097},[3063,8997,8998,9000,9002,9004,9006,9008,9010,9012,9014,9017],{"class":3065,"line":3264},[3063,8999,8562],{"class":3158},[3063,9001,3010],{"class":3097},[3063,9003,8606],{"class":3283},[3063,9005,3287],{"class":3097},[3063,9007,4168],{"class":3158},[3063,9009,3237],{"class":3097},[3063,9011,4168],{"class":3158},[3063,9013,3010],{"class":3097},[3063,9015,9016],{"class":3158},"CreatedAt",[3063,9018,3208],{"class":3097},[3063,9020,9021,9023,9026,9028,9031],{"class":3065,"line":3269},[3063,9022,8625],{"class":3097},[3063,9024,9025],{"class":3283},"HasDefaultValueSql",[3063,9027,3287],{"class":3097},[3063,9029,9030],{"class":3360},"\"GETUTCDATE()\"",[3063,9032,3364],{"class":3097},[3063,9034,9035,9037,9039,9042,9044,9046,9048,9050,9052,9054,9056,9058,9060,9062,9064,9066],{"class":3065,"line":3275},[3063,9036,8562],{"class":3158},[3063,9038,3010],{"class":3097},[3063,9040,9041],{"class":3283},"HasIndex",[3063,9043,3287],{"class":3097},[3063,9045,4168],{"class":3158},[3063,9047,3237],{"class":3097},[3063,9049,3554],{"class":3089},[3063,9051,4314],{"class":3097},[3063,9053,4168],{"class":3158},[3063,9055,3010],{"class":3097},[3063,9057,3471],{"class":3158},[3063,9059,4072],{"class":3097},[3063,9061,4168],{"class":3158},[3063,9063,3010],{"class":3097},[3063,9065,3660],{"class":3158},[3063,9067,9068],{"class":3097}," })\n",[3063,9070,9071,9073,9076],{"class":3065,"line":3298},[3063,9072,8625],{"class":3097},[3063,9074,9075],{"class":3283},"IsUnique",[3063,9077,3535],{"class":3097},[3063,9079,9080,9082,9085,9087,9090],{"class":3065,"line":3304},[3063,9081,8625],{"class":3097},[3063,9083,9084],{"class":3283},"HasDatabaseName",[3063,9086,3287],{"class":3097},[3063,9088,9089],{"class":3360},"\"UX_Products_Name_Category\"",[3063,9091,3364],{"class":3097},[3063,9093,9094],{"class":3065,"line":3333},[3063,9095,7754],{"class":3097},[8797,9097,9099],{"label":9098},"Data Annotations (обмежено)",[3021,9100,9102],{"className":3057,"code":9101,"language":3059,"meta":3029,"style":3029},"[Table(\"Products\", Schema = \"catalog\")]\npublic partial class Product\n{\n    [Key]\n    public int Id { get; set; }\n\n    [Required]\n    [MaxLength(200)]\n    public string Name { get; set; } = null!;\n\n    [Precision(10, 2)]\n    public decimal Price { get; set; }\n\n    // ❌ НЕ МОЖЛИВО через атрибути:\n    // Check Constraint, Default SQL, Composite Unique Index,\n    // Custom FK constraint name, HasDefaultSchema\n}\n",[2984,9103,9104,9127,9137,9141,9151,9171,9175,9184,9197,9222,9226,9243,9262,9266,9271,9276,9281],{"__ignoreMap":3029},[3063,9105,9106,9108,9111,9113,9115,9117,9120,9122,9124],{"class":3065,"line":3066},[3063,9107,5769],{"class":3097},[3063,9109,9110],{"class":3093},"Table",[3063,9112,3287],{"class":3097},[3063,9114,8572],{"class":3360},[3063,9116,4072],{"class":3097},[3063,9118,9119],{"class":3158},"Schema",[3063,9121,3377],{"class":3097},[3063,9123,8848],{"class":3360},[3063,9125,9126],{"class":3097},")]\n",[3063,9128,9129,9131,9133,9135],{"class":3065,"line":3073},[3063,9130,3125],{"class":3089},[3063,9132,3128],{"class":3089},[3063,9134,3131],{"class":3089},[3063,9136,3134],{"class":3093},[3063,9138,9139],{"class":3065,"line":3079},[3063,9140,3140],{"class":3097},[3063,9142,9143,9146,9149],{"class":3065,"line":3086},[3063,9144,9145],{"class":3097},"    [",[3063,9147,9148],{"class":3093},"Key",[3063,9150,5775],{"class":3097},[3063,9152,9153,9155,9158,9161,9163,9165,9167,9169],{"class":3065,"line":3111},[3063,9154,3152],{"class":3089},[3063,9156,9157],{"class":3089}," int",[3063,9159,9160],{"class":3158}," Id",[3063,9162,4314],{"class":3097},[3063,9164,4317],{"class":3089},[3063,9166,4320],{"class":3097},[3063,9168,4323],{"class":3089},[3063,9170,4326],{"class":3097},[3063,9172,9173],{"class":3065,"line":3116},[3063,9174,3083],{"emptyLinePlaceholder":3082},[3063,9176,9177,9179,9182],{"class":3065,"line":3122},[3063,9178,9145],{"class":3097},[3063,9180,9181],{"class":3093},"Required",[3063,9183,5775],{"class":3097},[3063,9185,9186,9188,9191,9193,9195],{"class":3065,"line":3137},[3063,9187,9145],{"class":3097},[3063,9189,9190],{"class":3093},"MaxLength",[3063,9192,3287],{"class":3097},[3063,9194,8642],{"class":3204},[3063,9196,9126],{"class":3097},[3063,9198,9199,9201,9203,9206,9208,9210,9212,9214,9217,9219],{"class":3065,"line":3143},[3063,9200,3152],{"class":3089},[3063,9202,3456],{"class":3089},[3063,9204,9205],{"class":3158}," Name",[3063,9207,4314],{"class":3097},[3063,9209,4317],{"class":3089},[3063,9211,4320],{"class":3097},[3063,9213,4323],{"class":3089},[3063,9215,9216],{"class":3097},"; } = ",[3063,9218,4697],{"class":3089},[3063,9220,9221],{"class":3097},"!;\n",[3063,9223,9224],{"class":3065,"line":3149},[3063,9225,3083],{"emptyLinePlaceholder":3082},[3063,9227,9228,9230,9233,9235,9237,9239,9241],{"class":3065,"line":3165},[3063,9229,9145],{"class":3097},[3063,9231,9232],{"class":3093},"Precision",[3063,9234,3287],{"class":3097},[3063,9236,8960],{"class":3204},[3063,9238,4072],{"class":3097},[3063,9240,8965],{"class":3204},[3063,9242,9126],{"class":3097},[3063,9244,9245,9247,9249,9252,9254,9256,9258,9260],{"class":3065,"line":3176},[3063,9246,3152],{"class":3089},[3063,9248,3155],{"class":3089},[3063,9250,9251],{"class":3158}," Price",[3063,9253,4314],{"class":3097},[3063,9255,4317],{"class":3089},[3063,9257,4320],{"class":3097},[3063,9259,4323],{"class":3089},[3063,9261,4326],{"class":3097},[3063,9263,9264],{"class":3065,"line":3211},[3063,9265,3083],{"emptyLinePlaceholder":3082},[3063,9267,9268],{"class":3065,"line":3221},[3063,9269,9270],{"class":3069},"    // ❌ НЕ МОЖЛИВО через атрибути:\n",[3063,9272,9273],{"class":3065,"line":3226},[3063,9274,9275],{"class":3069},"    // Check Constraint, Default SQL, Composite Unique Index,\n",[3063,9277,9278],{"class":3065,"line":3264},[3063,9279,9280],{"class":3069},"    // Custom FK constraint name, HasDefaultSchema\n",[3063,9282,9283],{"class":3065,"line":3269},[3063,9284,3785],{"class":3097},[2963,9286,9287,9290,9291,9298],{},[2991,9288,9289],{},"Висновок",": Fluent API є більш повним і виразним. Data Annotations — зручні для простих сценаріїв і зрозуміліші на перший погляд. У великих Database-First проєктах ",[2991,9292,9293,9294,9297],{},"Fluent API у окремих ",[2984,9295,9296],{},"IEntityTypeConfiguration\u003CT>"," файлах"," — найкращий підхід для підтримки.",[3041,9300,9302],{"id":9301},"fluent-api-у-окремих-configuration-файлах-рекомендовано","Fluent API у окремих Configuration файлах (рекомендовано)",[2963,9304,9305,9306,9308],{},"Навіть у Database-First підході — можна перенести конфігурацію з ",[2984,9307,4535],{}," у окремі файли (partial class + IEntityTypeConfiguration):",[3021,9310,9312],{"className":3057,"code":9311,"language":3059,"meta":3029,"style":3029},"// Infrastructure/Configurations/ProductConfiguration.cs\n// Цей файл НЕ генерується scaffold — пишеться розробником\npublic class ProductConfiguration : IEntityTypeConfiguration\u003CProduct>\n{\n    public void Configure(EntityTypeBuilder\u003CProduct> builder)\n    {\n        // Конфігурація яку НЕ генерує scaffold (тільки наша):\n        builder.HasQueryFilter(p => p.IsActive);  // Global Query Filter\n        builder.Property(p => p.Name).UseCollation(\"SQL_Latin1_General_CP1_CS_AS\");\n    }\n}\n\n// ShopDbContextExtensions.cs (наш файл, не scaffold):\npublic partial class ShopDbContext\n{\n    protected override void OnModelCreating(ModelBuilder modelBuilder)\n    {\n        base.OnModelCreating(modelBuilder); // ← scaffold конфігурація\n\n        // Застосувати наші додаткові конфігурації\n        modelBuilder.ApplyConfigurationsFromAssembly(\n            typeof(ProductConfiguration).Assembly,\n            t => t.Namespace?.Contains(\".Configurations\") == true);  // фільтр за namespace\n    }\n}\n",[2984,9313,9314,9319,9324,9344,9348,9372,9376,9381,9406,9438,9442,9446,9450,9455,9465,9469,9487,9491,9508,9512,9517,9528,9545,9581,9585],{"__ignoreMap":3029},[3063,9315,9316],{"class":3065,"line":3066},[3063,9317,9318],{"class":3069},"// Infrastructure/Configurations/ProductConfiguration.cs\n",[3063,9320,9321],{"class":3065,"line":3073},[3063,9322,9323],{"class":3069},"// Цей файл НЕ генерується scaffold — пишеться розробником\n",[3063,9325,9326,9328,9330,9333,9335,9338,9340,9342],{"class":3065,"line":3079},[3063,9327,3125],{"class":3089},[3063,9329,3131],{"class":3089},[3063,9331,9332],{"class":3093}," ProductConfiguration",[3063,9334,4371],{"class":3097},[3063,9336,9337],{"class":3093},"IEntityTypeConfiguration",[3063,9339,3523],{"class":3097},[3063,9341,3039],{"class":3093},[3063,9343,8072],{"class":3097},[3063,9345,9346],{"class":3065,"line":3086},[3063,9347,3140],{"class":3097},[3063,9349,9350,9352,9354,9357,9359,9362,9364,9366,9368,9370],{"class":3065,"line":3111},[3063,9351,3152],{"class":3089},[3063,9353,3280],{"class":3089},[3063,9355,9356],{"class":3283}," Configure",[3063,9358,3287],{"class":3097},[3063,9360,9361],{"class":3093},"EntityTypeBuilder",[3063,9363,3523],{"class":3097},[3063,9365,3039],{"class":3093},[3063,9367,3529],{"class":3097},[3063,9369,6340],{"class":3158},[3063,9371,3208],{"class":3097},[3063,9373,9374],{"class":3065,"line":3116},[3063,9375,3301],{"class":3097},[3063,9377,9378],{"class":3065,"line":3122},[3063,9379,9380],{"class":3069},"        // Конфігурація яку НЕ генерує scaffold (тільки наша):\n",[3063,9382,9383,9385,9387,9389,9391,9393,9395,9397,9399,9401,9403],{"class":3065,"line":3137},[3063,9384,4777],{"class":3158},[3063,9386,3010],{"class":3097},[3063,9388,4790],{"class":3283},[3063,9390,3287],{"class":3097},[3063,9392,2963],{"class":3158},[3063,9394,3237],{"class":3097},[3063,9396,2963],{"class":3158},[3063,9398,3010],{"class":3097},[3063,9400,3408],{"class":3158},[3063,9402,8851],{"class":3097},[3063,9404,9405],{"class":3069},"// Global Query Filter\n",[3063,9407,9408,9410,9412,9414,9416,9418,9420,9422,9424,9426,9428,9431,9433,9436],{"class":3065,"line":3143},[3063,9409,4777],{"class":3158},[3063,9411,3010],{"class":3097},[3063,9413,8606],{"class":3283},[3063,9415,3287],{"class":3097},[3063,9417,2963],{"class":3158},[3063,9419,3237],{"class":3097},[3063,9421,2963],{"class":3158},[3063,9423,3010],{"class":3097},[3063,9425,3471],{"class":3158},[3063,9427,4602],{"class":3097},[3063,9429,9430],{"class":3283},"UseCollation",[3063,9432,3287],{"class":3097},[3063,9434,9435],{"class":3360},"\"SQL_Latin1_General_CP1_CS_AS\"",[3063,9437,3364],{"class":3097},[3063,9439,9440],{"class":3065,"line":3149},[3063,9441,3387],{"class":3097},[3063,9443,9444],{"class":3065,"line":3165},[3063,9445,3785],{"class":3097},[3063,9447,9448],{"class":3065,"line":3176},[3063,9449,3083],{"emptyLinePlaceholder":3082},[3063,9451,9452],{"class":3065,"line":3211},[3063,9453,9454],{"class":3069},"// ShopDbContextExtensions.cs (наш файл, не scaffold):\n",[3063,9456,9457,9459,9461,9463],{"class":3065,"line":3221},[3063,9458,3125],{"class":3089},[3063,9460,3128],{"class":3089},[3063,9462,3131],{"class":3089},[3063,9464,3826],{"class":3093},[3063,9466,9467],{"class":3065,"line":3226},[3063,9468,3140],{"class":3097},[3063,9470,9471,9473,9475,9477,9479,9481,9483,9485],{"class":3065,"line":3264},[3063,9472,4499],{"class":3089},[3063,9474,3721],{"class":3089},[3063,9476,3280],{"class":3089},[3063,9478,4506],{"class":3283},[3063,9480,3287],{"class":3097},[3063,9482,4511],{"class":3093},[3063,9484,4514],{"class":3158},[3063,9486,3208],{"class":3097},[3063,9488,9489],{"class":3065,"line":3269},[3063,9490,3301],{"class":3097},[3063,9492,9493,9495,9497,9499,9501,9503,9505],{"class":3065,"line":3275},[3063,9494,4530],{"class":3089},[3063,9496,3010],{"class":3097},[3063,9498,4535],{"class":3283},[3063,9500,3287],{"class":3097},[3063,9502,4540],{"class":3158},[3063,9504,7253],{"class":3097},[3063,9506,9507],{"class":3069},"// ← scaffold конфігурація\n",[3063,9509,9510],{"class":3065,"line":3298},[3063,9511,3083],{"emptyLinePlaceholder":3082},[3063,9513,9514],{"class":3065,"line":3304},[3063,9515,9516],{"class":3069},"        // Застосувати наші додаткові конфігурації\n",[3063,9518,9519,9521,9523,9526],{"class":3065,"line":3333},[3063,9520,7240],{"class":3158},[3063,9522,3010],{"class":3097},[3063,9524,9525],{"class":3283},"ApplyConfigurationsFromAssembly",[3063,9527,6373],{"class":3097},[3063,9529,9530,9533,9535,9538,9540,9543],{"class":3065,"line":3357},[3063,9531,9532],{"class":3089},"            typeof",[3063,9534,3287],{"class":3097},[3063,9536,9537],{"class":3093},"ProductConfiguration",[3063,9539,4602],{"class":3097},[3063,9541,9542],{"class":3158},"Assembly",[3063,9544,6567],{"class":3097},[3063,9546,9547,9550,9552,9555,9557,9560,9563,9566,9568,9571,9574,9576,9578],{"class":3065,"line":3367},[3063,9548,9549],{"class":3158},"            t",[3063,9551,3237],{"class":3097},[3063,9553,9554],{"class":3158},"t",[3063,9556,3010],{"class":3097},[3063,9558,9559],{"class":3158},"Namespace",[3063,9561,9562],{"class":3097},"?.",[3063,9564,9565],{"class":3283},"Contains",[3063,9567,3287],{"class":3097},[3063,9569,9570],{"class":3360},"\".Configurations\"",[3063,9572,9573],{"class":3097},") == ",[3063,9575,3413],{"class":3089},[3063,9577,8851],{"class":3097},[3063,9579,9580],{"class":3069},"// фільтр за namespace\n",[3063,9582,9583],{"class":3065,"line":3372},[3063,9584,3387],{"class":3097},[3063,9586,9587],{"class":3065,"line":3384},[3063,9588,3785],{"class":3097},[2973,9590],{},[2976,9592,9594],{"id":9593},"практичні-завдання-частина-2","Практичні завдання (Частина 2)",[3041,9596,9598],{"id":9597},"рівень-1-базовий","Рівень 1 — Базовий",[5909,9600,9601,9605,9608,9643,9647,9679,9683],{},[3041,9602,9604],{"id":9603},"завдання-11-partial-class-розширення","Завдання 1.1: Partial Class розширення",[2963,9606,9607],{},"Після scaffold ShopDb:",[9609,9610,9611,9618,9634,9640],"ol",{},[4841,9612,9613,9614,9617],{},"Створіть ",[2984,9615,9616],{},"Models/Extensions/Product.Extensions.cs"," (separate file, same namespace)",[4841,9619,9620,9621,9624,9625,4072,9628,4072,9631],{},"Додайте: ",[2984,9622,9623],{},"FinalPrice"," (computed), ",[2984,9626,9627],{},"DisplayName",[2984,9629,9630],{},"Validate()",[2984,9632,9633],{},"ToString()",[4841,9635,9636,9637,9639],{},"Виконайте re-scaffold з ",[2984,9638,3005],{}," — перевірте що Extensions файл НЕ перезаписаний",[4841,9641,9642],{},"Перевірте що Extensions методи компілюються і доступні через IntelliSense",[3041,9644,9646],{"id":9645},"завдання-12-schema-comparison-з-efcoreschemacompare","Завдання 1.2: Schema Comparison з EFCore.SchemaCompare",[9609,9648,9649,9656,9673,9676],{},[4841,9650,9651,9652,9655],{},"Встановіть ",[2984,9653,9654],{},"EFCore.SchemaCompare"," пакет",[4841,9657,9658,9659,9662,9663],{},"Напишіть xUnit тест ",[2984,9660,9661],{},"DbSchemaMatchesEfModel",":\n",[4838,9664,9665,9668],{},[4841,9666,9667],{},"Використовує реальну тестову БД (SQLite або PostgreSQL test container)",[4841,9669,9670,9672],{},[2984,9671,5651],{}," → Assert.False(hasErrors)",[4841,9674,9675],{},"Навмисно зробіть розбіжність (видаліть стовпець з DB) → переконайтесь що тест падає",[4841,9677,9678],{},"Додайте тест у CI pipeline (це regression guard!)",[3041,9680,9682],{"id":9681},"завдання-13-multi-context-реєстрація","Завдання 1.3: Multi-Context реєстрація",[9609,9684,9685,9693,9699,9705,9710],{},[4841,9686,9613,9687,4967,9689,9692],{},[2984,9688,6126],{},[2984,9690,9691],{},"CatalogDbContext"," з різними entity",[4841,9694,9695,9696],{},"Зареєструйте обидва у ",[2984,9697,9698],{},"Program.cs",[4841,9700,9701,9702],{},"Запустіть ",[2984,9703,9704],{},"dotnet ef database update --context OrdersDbContext",[4841,9706,9701,9707],{},[2984,9708,9709],{},"dotnet ef database update --context CatalogDbContext",[4841,9711,9712,9713,9716],{},"Перевірте: є дві ",[2984,9714,9715],{},"__EFMigrationsHistory"," таблиці або одна? Чому?",[3041,9718,9720],{"id":9719},"рівень-2-логіка","Рівень 2 — Логіка",[5909,9722,9723,9727,9734,9771,9775,9778],{},[3041,9724,9726],{"id":9725},"завдання-21-re-scaffold-workflow-автоматизація","Завдання 2.1: Re-scaffold Workflow автоматизація",[2963,9728,9729,9730,9733],{},"Напишіть PowerShell/bash скрипт ",[2984,9731,9732],{},"rescaffold.sh"," що:",[9609,9735,9736,9746,9752,9759,9768],{},[4841,9737,9738,9739,9742,9743,4854],{},"Зберігає git checkpoint (",[2984,9740,9741],{},"git stash"," або ",[2984,9744,9745],{},"git add; git commit",[4841,9747,9748,9749],{},"Виконує ",[2984,9750,9751],{},"dotnet ef dbcontext scaffold ... --force",[4841,9753,9754,9755,9758],{},"Виводить ",[2984,9756,9757],{},"git diff --stat"," для перегляду змін",[4841,9760,9761,9762,4967,9765],{},"Запускає ",[2984,9763,9764],{},"dotnet build",[2984,9766,9767],{},"dotnet test",[4841,9769,9770],{},"При помилці тестів — виводить попередження \"Extensions may need updating\"",[3041,9772,9774],{"id":9773},"завдання-22-interface-driven-multi-db","Завдання 2.2: Interface-driven Multi-DB",[2963,9776,9777],{},"Реалізуйте Read/Write Separation:",[9609,9779,9780,9787,9795,9803,9811],{},[4841,9781,9782,4850,9785],{},[2984,9783,9784],{},"IWriteRepository\u003CT>",[2984,9786,7801],{},[4841,9788,9789,4850,9792,9794],{},[2984,9790,9791],{},"IReadRepository\u003CT>",[2984,9793,7855],{}," (NoTracking, replica connection)",[4841,9796,9797,4850,9800],{},[2984,9798,9799],{},"OrderCommandHandler",[2984,9801,9802],{},"IWriteRepository\u003COrder>",[4841,9804,9805,4850,9808],{},[2984,9806,9807],{},"OrderQueryHandler",[2984,9809,9810],{},"IReadRepository\u003COrder>",[4841,9812,9813],{},"Unit тест що перевіряє: Query Handler ніколи не використовує WriteDbContext",[3041,9815,9817],{"id":9816},"рівень-3-архітектура","Рівень 3 — Архітектура",[5909,9819,9820,9824,9827,9855],{},[3041,9821,9823],{"id":9822},"завдання-31-tenant-per-database-система","Завдання 3.1: Tenant-per-database система",[2963,9825,9826],{},"Реалізуйте повну multi-tenant систему:",[9609,9828,9829,9837,9843,9849],{},[4841,9830,9831,9833,9834],{},[2984,9832,7666],{}," → отримує TenantId з HTTP заголовка ",[2984,9835,9836],{},"X-Tenant-Id",[4841,9838,9839,9842],{},[2984,9840,9841],{},"TenantConnectionStringProvider"," → читає connection string для тенанту з config або master DB",[4841,9844,9845,9848],{},[2984,9846,9847],{},"TenantDbContextFactory : IDbContextFactory\u003CAppDbContext>"," → створює контекст з правильним connection string",[4841,9850,9851,9852,9854],{},"Middleware що валідує ",[2984,9853,9836],{}," і повертає 400 якщо відсутній",[2963,9856,9857,9858,9860],{},"Тест: два запити з різними ",[2984,9859,9836],{}," → кожен отримує дані рівно свого тенанту (Cross-tenant isolation test).",[2973,9862],{},[2976,9864,9866],{"id":9865},"підсумок-статті-25","Підсумок статті 25",[2963,9868,9869],{},"Дві частини охопили весь lifecycle Database-First підходу та управління схемою:",[2963,9871,9872],{},[2991,9873,9874],{},"Частина 1:",[4838,9876,9877,9883,9893,9918,9927],{},[4841,9878,9879,9882],{},[2991,9880,9881],{},"Code-First vs Database-First",": не просто технічні підходи, а архітектурні рішення про ownership схеми. DBA-controlled org → Database-First. Developer-centric → Code-First",[4841,9884,9885,9888,9889,9892],{},[2991,9886,9887],{},"EnsureCreated vs MigrateAsync",": принципово несумісні. ",[2984,9890,9891],{},"EnsureCreated"," тільки для тестів",[4841,9894,9895,3016,9898,4072,9900,4072,9903,4072,9906,4072,9909,4072,9912,9914,9915,9917],{},[2991,9896,9897],{},"Scaffold-DbContext",[2984,9899,4966],{},[2984,9901,9902],{},"--context",[2984,9904,9905],{},"--no-onconfiguring",[2984,9907,9908],{},"--table",[2984,9910,9911],{},"--schema",[2984,9913,3005],{},". Генерує ",[2984,9916,2986],{}," entity та DbContext",[4841,9919,9920,9923,9924,9926],{},[2991,9921,9922],{},"Анатомія",": Fluent API конфігурація у ",[2984,9925,4535],{},", nullable mapping, навігаційні властивості",[4841,9928,9929,3016,9932,4967,9935,9938],{},[2991,9930,9931],{},"T4 Templates",[2984,9933,9934],{},"EntityType.t4",[2984,9936,9937],{},"DbContext.t4"," для кастомізації 50+ таблиць",[2963,9940,9941],{},[2991,9942,9943],{},"Частина 2:",[4838,9945,9946,9960,9968,9976,9986],{},[4841,9947,9948,9951,9952,9955,9956,9959],{},[2991,9949,9950],{},"Partial Classes",": ключова техніка для розширення entity без ризику перезапису. ",[2984,9953,9954],{},"Extensions/"," директорія окремо від ",[2984,9957,9958],{},"Generated/",". Interface implementation через partial",[4841,9961,9962,3016,9965,9967],{},[2991,9963,9964],{},"Re-scaffold Workflow",[2984,9966,3005],{},", безпечна структура директорій, git-based diff review, автоматизований скрипт",[4841,9969,9970,3016,9973,9975],{},[2991,9971,9972],{},"Schema Comparison",[2984,9974,9654],{}," для regression тестів. SQL Server Schema Compare, pg_dump diff",[4841,9977,9978,9981,9982,9985],{},[2991,9979,9980],{},"Multi-Database",": кілька DbContext → різні БД, різні провайдери. Tenant-per-database з ",[2984,9983,9984],{},"IDbContextFactory",". Read/Write separation для CQRS",[4841,9987,9988,9991,9992,9994],{},[2991,9989,9990],{},"Fluent API переможець",": повніший, гнучкіший, ",[2984,9993,9296],{}," у окремих файлах",[2963,9996,9997,9998,10002,10003,3010],{},"Наступні статті — ",[2967,9999,10001],{"href":10000},"/csharp/ef-core/26.performance-fundamentals","Продуктивність: Основи (26)"," та ",[2967,10004,10006],{"href":10005},"/csharp/ef-core/27.performance-advanced","Продуктивність: Advanced (27)",[2973,10008],{},[2976,10010,10012],{"id":10011},"додаткові-ресурси","Додаткові ресурси",[4838,10014,10015,10023,10030,10037,10044],{},[4841,10016,10017],{},[2967,10018,10022],{"href":10019,"rel":10020},"https://learn.microsoft.com/en-us/ef/core/managing-schemas/scaffolding/",[10021],"nofollow","Reverse Engineering — офіційна документація",[4841,10024,10025],{},[2967,10026,10029],{"href":10027,"rel":10028},"https://github.com/JonPSmith/EfCore.SchemaCompare",[10021],"EFCore.SchemaCompare на GitHub",[4841,10031,10032],{},[2967,10033,10036],{"href":10034,"rel":10035},"https://learn.microsoft.com/en-us/visualstudio/modeling/code-generation-and-t4-text-templates",[10021],"T4 Text Template Transformation",[4841,10038,10039],{},[2967,10040,10043],{"href":10041,"rel":10042},"https://learn.microsoft.com/en-us/ef/core/miscellaneous/multitenancy",[10021],"Multi-tenancy — EF Core docs",[4841,10045,10046],{},[2967,10047,10050],{"href":10048,"rel":10049},"https://learn.microsoft.com/en-us/dotnet/csharp/programming-guide/classes-and-structs/partial-classes-and-methods",[10021],"Partial Classes — C# Reference",[10052,10053,10054],"style",{},"html pre.shiki code .spJ8K, html code.shiki .spJ8K{--shiki-light:#008000;--shiki-default:#6A9955;--shiki-dark:#6A9955}html pre.shiki code .su1O8, html code.shiki .su1O8{--shiki-light:#0000FF;--shiki-default:#569CD6;--shiki-dark:#569CD6}html pre.shiki code .sN1BT, html code.shiki .sN1BT{--shiki-light:#267F99;--shiki-default:#4EC9B0;--shiki-dark:#4EC9B0}html pre.shiki code .sHH4Y, html code.shiki .sHH4Y{--shiki-light:#000000;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html pre.shiki code .siwwj, html code.shiki .siwwj{--shiki-light:#001080;--shiki-default:#9CDCFE;--shiki-dark:#9CDCFE}html pre.shiki code .sJj4R, html code.shiki .sJj4R{--shiki-light:#098658;--shiki-default:#B5CEA8;--shiki-dark:#B5CEA8}html pre.shiki code .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 .sbdoH, html code.shiki .sbdoH{--shiki-light:#A31515;--shiki-default:#CE9178;--shiki-dark:#CE9178}html pre.shiki code .sD7JJ, html code.shiki .sD7JJ{--shiki-light:#000000FF;--shiki-default:#D4D4D4;--shiki-dark:#D4D4D4}html .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 .sjcCO, html code.shiki .sjcCO{--shiki-light:#EE0000;--shiki-default:#D7BA7D;--shiki-dark:#D7BA7D}",{"title":3029,"searchDepth":3073,"depth":3073,"links":10056},[10057,10063,10071,10083,10090,10095,10106,10107],{"id":2978,"depth":3073,"text":2979,"children":10058},[10059,10060,10061,10062],{"id":3043,"depth":3079,"text":3044},{"id":3053,"depth":3079,"text":3054},{"id":3788,"depth":3079,"text":3789},{"id":4275,"depth":3079,"text":4276},{"id":4828,"depth":3073,"text":4829,"children":10064},[10065,10066,10067,10068,10069,10070],{"id":4835,"depth":3079,"text":4836},{"id":4863,"depth":3079,"text":4864},{"id":4985,"depth":3079,"text":4986},{"id":4995,"depth":3079,"text":4996},{"id":5266,"depth":3079,"text":5267},{"id":5427,"depth":3079,"text":5428},{"id":5502,"depth":3073,"text":5503,"children":10072},[10073,10074,10075,10076,10077,10078,10079,10080,10081,10082],{"id":5519,"depth":3079,"text":5520},{"id":5564,"depth":3079,"text":5565},{"id":5575,"depth":3079,"text":5576},{"id":5899,"depth":3079,"text":5900},{"id":5913,"depth":3079,"text":5914},{"id":5920,"depth":3079,"text":5921},{"id":5927,"depth":3079,"text":5928},{"id":5934,"depth":3079,"text":5935},{"id":5941,"depth":3079,"text":5942},{"id":5951,"depth":3079,"text":5952},{"id":6048,"depth":3073,"text":6049,"children":10084},[10085,10086,10087,10088,10089],{"id":6081,"depth":3079,"text":6082},{"id":7013,"depth":3079,"text":7014},{"id":7272,"depth":3079,"text":7273},{"id":7757,"depth":3079,"text":7758},{"id":8381,"depth":3079,"text":8382},{"id":8516,"depth":3073,"text":8517,"children":10091},[10092,10093,10094],{"id":8523,"depth":3079,"text":8524},{"id":8791,"depth":3079,"text":8792},{"id":9301,"depth":3079,"text":9302},{"id":9593,"depth":3073,"text":9594,"children":10096},[10097,10098,10099,10100,10101,10102,10103,10104,10105],{"id":9597,"depth":3079,"text":9598},{"id":9603,"depth":3079,"text":9604},{"id":9645,"depth":3079,"text":9646},{"id":9681,"depth":3079,"text":9682},{"id":9719,"depth":3079,"text":9720},{"id":9725,"depth":3079,"text":9726},{"id":9773,"depth":3079,"text":9774},{"id":9816,"depth":3079,"text":9817},{"id":9822,"depth":3079,"text":9823},{"id":9865,"depth":3073,"text":9866},{"id":10011,"depth":3073,"text":10012},"Partial Classes для безпечного розширення згенерованих entity — бізнес-логіка без ризику перезапису. Re-scaffolding workflow при зміні БД. Database schema comparison tools. Multi-database scenarios у одному застосунку.","md",null,{},{"title":631,"description":10108},"0nqa9rpk8CNSVObhx_c6IoTmVQkZ1Xi3eJxI6cli_s0",[10115,10117],{"title":627,"path":628,"stem":629,"description":10116,"children":-1},"[object Object]",{"title":635,"path":636,"stem":637,"description":10118,"children":-1},"Чому EF Core може бути катастрофічно повільним і як це виявити. N+1 проблема — найдорожчий антипатерн у ORM. Еволюція рішення від Eager Loading до Split Query. Проєкція як фундаментальна техніка. Logging generated SQL і EF Core Diagnostics.",1777912597504]