Skill

milan-jovanovic-blog

Search Milan Jovanovic's .NET blog for Clean Architecture, DDD, CQRS, EF Core, and ASP.NET Core patterns. Use for finding applicable patterns, code examples, and architecture guidance. Invoke when working with .NET projects that could benefit from proven architectural patterns.

From milan-jovanovic
Install
1
Run in your terminal
$
npx claudepluginhub melodic-software/claude-code-plugins --plugin milan-jovanovic
Tool Access

This skill is limited to using the following tools:

ReadGlobGrepBash
Supporting Assets
View in Repository
canonical/index.json
canonical/milanjovanovic-tech/blog/a-practical-demo-of-zero-downtime-migrations-using-password-hashing.md
canonical/milanjovanovic-tech/blog/containerize-your-dotnet-applications-without-a-dockerfile.md
canonical/milanjovanovic-tech/blog/dbcontext-is-not-thread-safe-parallelizing-ef-core-queries-the-right-way.md
canonical/milanjovanovic-tech/blog/exploring-csharp-file-based-apps-in-dotnet-10.md
canonical/milanjovanovic-tech/blog/how-to-build-a-high-performance-cache-without-external-libraries.md
canonical/milanjovanovic-tech/blog/how-to-extract-structured-data-from-images-using-ollama-in-dotnet.md
canonical/milanjovanovic-tech/blog/integrate-keycloak-with-aspnetcore-using-oauth-2.md
canonical/milanjovanovic-tech/blog/server-sent-events-in-aspnetcore-and-dotnet-10.md
canonical/milanjovanovic-tech/blog/solving-message-ordering-from-first-principles.md
canonical/milanjovanovic-tech/blog/solving-the-distributed-cache-invalidation-problem-with-redis-and-hybridcache.md
canonical/milanjovanovic-tech/blog/the-false-comfort-of-the-happy-path-decoupling-your-services.md
canonical/milanjovanovic-tech/blog/the-idempotent-consumer-pattern-in-dotnet-and-why-you-need-it.md
canonical/milanjovanovic-tech/blog/the-new-slnx-solution-format-migration-guide.md
canonical/milanjovanovic-tech/blog/the-urge-to-build-something.md
canonical/milanjovanovic-tech/blog/vertical-slice-architecture-where-does-the-shared-logic-live.md
canonical/milanjovanovic-tech/blog/whats-new-in-ef-core-10-leftjoin-and-rightjoin-operators-in-linq.md
config/defaults.yaml
milan_jovanovic_api.py
references/sources.json
Skill Content

Milan Jovanovic Blog Skill

Overview

This skill provides access to Milan Jovanovic's curated .NET blog content, filtered to November 2025+ (aligned with .NET 10 GA). It helps developers discover and apply proven patterns for Clean Architecture, Domain-Driven Design, CQRS, Entity Framework Core, and ASP.NET Core.

Content scope: Articles published November 2025 and later, with promotional content stripped.

When to Use This Skill

Use this skill when:

  • Working with .NET projects that could benefit from architectural patterns
  • Implementing Clean Architecture, DDD, CQRS, or Vertical Slice patterns
  • Configuring Entity Framework Core or ASP.NET Core
  • Looking for proven .NET code patterns and examples
  • Researching .NET 10/Aspire features

Quick Start

Search by Keywords

python scripts/core/find_articles.py search clean architecture cqrs

Filter by Tag

python scripts/core/find_articles.py tag ef-core

Resolve doc_id to Path

python scripts/core/find_articles.py resolve milanjovanovic-tech-blog-{slug}

Natural Language Query

python scripts/core/find_articles.py query "how to implement CQRS"

List Recent Articles

python scripts/core/find_articles.py list --sort date --limit 10

Tag Taxonomy

TagDescription
clean-architectureClean/Onion/Hexagonal Architecture patterns
dddDomain-Driven Design (aggregates, value objects, domain events)
cqrsCommand Query Responsibility Segregation
mediatrMediatR library usage patterns
ef-coreEntity Framework Core optimization and patterns
aspnet-coreASP.NET Core patterns and Minimal APIs
modular-monolithModular Monolith architecture
vertical-sliceVertical Slice Architecture
dotnet-10.NET 10 features and patterns
aspire.NET Aspire patterns
result-patternResult pattern for error handling
outbox-patternTransactional outbox pattern
specification-patternSpecification pattern
repository-patternRepository pattern
validationFluentValidation patterns
testingUnit/integration testing patterns

Path Resolution

Windows PowerShell users: Avoid cd && python chains - use absolute paths or run from repo root to prevent path doubling issues.

Script location: All scripts are in scripts/ relative to skill root.

Canonical content: Scraped articles are stored in canonical/milanjovanovic-tech/blog/.

Index Management

Check Index Stats

python scripts/management/manage_index.py stats

Verify Index Integrity

python scripts/management/manage_index.py verify

Refresh Index (After Scraping)

python scripts/management/refresh_index.py

Scraping (Use /scrape-posts Command)

For scraping operations, use the /milan-jovanovic:scrape-posts command which handles:

  • Pre-filtering optimization - Parses dates from listing page before scraping individual articles
  • Date filtering (November 2025+)
  • Content cleanup (removes promotional sections)
  • Idempotent updates (skip unchanged articles via content hash comparison)

Efficiency Optimizations

The scraping workflow uses pre-filtering to minimize firecrawl API calls:

ScenarioWithout OptimizationWith OptimizationSavings
No new articles10+ requests1-2 requests80-90%
1 new article10+ requests2-3 requests70-80%
Force (unchanged)10+ requests10+ requests (skips writes)I/O savings

How it works:

  1. Scrape listing page only (1-2 requests)
  2. Parse dates from listing markdown (no network)
  3. Compare against index to identify new articles
  4. Scrape ONLY articles not already indexed

Check for New Articles (Pre-Filter)

Before scraping, check what needs updating:

# Check for new articles since November 2025
python scripts/core/check_new_articles.py .claude/temp/listing.md --json --since 2025-11-01

# Force mode - include existing articles for re-check
python scripts/core/check_new_articles.py .claude/temp/listing.md --json --force

# URLs only output
python scripts/core/check_new_articles.py .claude/temp/listing.md --urls-only

Output includes to_scrape list with in_index and content_hash for smart handling.

Python API

For programmatic access, use the public API:

from milan_jovanovic_api import (
    search_articles,
    get_by_tag,
    resolve_doc_id,
    get_article_content,
)

# Search by keywords
results = search_articles(['clean-architecture', 'cqrs'])

# Get articles by tag
ef_articles = get_by_tag('ef-core')

# Resolve doc_id to path
path = resolve_doc_id('milanjovanovic-tech-blog-some-slug')

# Get article content
content = get_article_content('milanjovanovic-tech-blog-some-slug')

Content Cleanup

Scraped articles have promotional content removed:

  • Sponsor sections (between title and first H2)
  • Promotional footer ("Whenever you're ready, there are X ways...")
  • Newsletter signup sections
  • Course CTAs
  • Reading time metadata

All educational content, code blocks, and internal links are preserved.

Related Components

  • Command: /milan-jovanovic:scrape-posts - Scrape new articles
  • Agent: blog-advisor - Proactive project analysis and recommendations

Troubleshooting

No Results Found

  1. Check if index exists: python scripts/management/manage_index.py count
  2. If count is 0, run scrape: /milan-jovanovic:scrape-posts
  3. Verify tags with: python scripts/core/find_articles.py tag --list

Path Issues on Windows

Use PowerShell or prefix Git Bash with MSYS_NO_PATHCONV=1:

MSYS_NO_PATHCONV=1 python scripts/core/find_articles.py search cqrs

Version History

  • v1.1.0 (2025-12-27): Pre-filtering optimization - 80-90% reduction in firecrawl API calls
  • v1.0.0 (2025-12-22): Initial release - November 2025+ articles, full tag taxonomy

Last Updated: 2025-12-27

Similar Skills
cache-components

Expert guidance for Next.js Cache Components and Partial Prerendering (PPR). **PROACTIVE ACTIVATION**: Use this skill automatically when working in Next.js projects that have `cacheComponents: true` in their next.config.ts/next.config.js. When this config is detected, proactively apply Cache Components patterns and best practices to all React Server Component implementations. **DETECTION**: At the start of a session in a Next.js project, check for `cacheComponents: true` in next.config. If enabled, this skill's patterns should guide all component authoring, data fetching, and caching decisions. **USE CASES**: Implementing 'use cache' directive, configuring cache lifetimes with cacheLife(), tagging cached data with cacheTag(), invalidating caches with updateTag()/revalidateTag(), optimizing static vs dynamic content boundaries, debugging cache issues, and reviewing Cache Component implementations.

138.5k
Stats
Parent Repo Stars40
Parent Repo Forks6
Last CommitDec 27, 2025