cd /news/developer-tools/dev-log-deleting-code-and-breaking-t… · home topics developer-tools article
[ARTICLE · art-27605] src=dev.to ↗ pub= topic=developer-tools verified=true sentiment=· neutral

Dev Log: deleting code and breaking the build on purpose

A developer building Munchausen v1.0, a mock-data generator for .NET, started by deleting the existing prototype and deliberately breaking the build to test new guardrails. The project uses PublicApiAnalyzers to enforce a deliberate public API contract, and a canary class was used to verify the analyzer catches unintended public symbols. The initial milestone (M0) established the project scaffold with zero public API, while future milestones will tackle determinism and cycle detection.

read3 min publishedJun 15, 2026

Day one of building Munchausen v1.0 started with deleting the existing implementation and deliberately breaking its replacement.

Munchausen is a mock-data generator for .NET that infers realistic values from

your types. Before writing v1.0, I spent time designing the API and architecture,

then captured the result in a single design document. At the end, I divided the

implementation into stages: M0 for the repository scaffold, M1 for determinism,

and later milestones for metadata, inference, compilation, and runtime behavior.

The plan gives me a route through the project. Implementation is where I find out

whether those ideas actually hold together, and M0 was the first test.

The repo already had a prototype from years ago, an early Lie

/

ModelGenerator

/ ValuesGenerator

trio and a .travis.yml

pinned to a

dotnet-1.0 preview. My first instinct was to keep it nearby. Some of it might be

useful, and deleting code always feels more dramatic than leaving it alone.

Reading it changed my mind. The prototype reflected an earlier idea of the

library, and keeping its types invited me to preserve decisions I had already

reconsidered. Removing it gave me a clean place to test the new design without

negotiating with the old one. The history is still there if I need it.

One idea from the design was that the public API should be a deliberate contract,

not whatever happens to escape from the assembly. I added PublicApiAnalyzers

,

which tracks every public symbol in two text files. At M0 those files contain no

API entries. The public surface starts at zero and grows only when I consciously

add to it.

Configuring the analyzer was easy. Trusting it was different. I could see that

the package was installed and the build was green, but that did not prove it

would catch the mistake I cared about. So I added a mistake on purpose.

I wrote a throwaway Canary

class with one public property and no XML comments, then committed it. The build went red with exactly the errors I hoped to see:

RS0016

(this public symbol isn't declared in the API files) for the type, its

getter, and even its implicit constructor, plus CS1591

for the missing comments. One little class tripped both wires at once. Then I removed the canary commit,

rebuilt, and watched the build go green again.

I expected a quick configuration check. Instead, I learned that a green build

only shows the current code passes. A deliberate failure proves the build can

protect a boundary I care about. When later milestones introduce public API, I

know accidental additions cannot quietly slip through.

The rest of M0 turned design choices into places where future work can happen:

net8.0

, nullable reference types, warnings treated as errors, required XML

comments, and a Directory.Build.props

that keeps those choices consistent. I

added CI and created six projects for the package, unit tests, acceptance tests,

determinism tests, benchmarks, and shared test models.

The shared models include Car

, Owner

, Customer

/Order

/Item

, a

positional record, an init-only type, and Employee

whose Manager

is another

Employee

. That last model is a question waiting for a later milestone: what

happens when generation encounters a cycle? I do not need the answer yet, but

putting the model in place now means I cannot conveniently forget the problem

when recursion and cycle detection arrive.

Zero features. Zero public API. A green build, a CI pipeline, and a guardrail

I've personally watched fail and recover. More importantly, I finished M0 with a

better understanding of the design than I had when it existed only on paper.

Starting from zero is not empty progress when the boundaries themselves are part

of what you are building.

M1 takes on the determinism core. Munchausen promises that the same seed produces

the same data forever, across machines, OSes, and .NET versions. .NET doesn't

guarantee that System.Random

will preserve its algorithm across versions, so

I'm building an owned PRNG (xoshiro256** seeded by SplitMix64) and validating

it against published reference vectors. If my numbers don't match the ones the

algorithm's authors published, my code is wrong, full stop. That's the kind of

unambiguous test I love. More soon.

── more in #developer-tools 4 stories · sorted by recency
sponsored brought to you by zahid.host 4,200+ EU-deployed projects
reading about agents? ship yours in a single git push.

Run your AI side-project on zahid.host

EU-based hosting, git-push deploys, automatic HTTPS, no cold starts. Free tier with a custom domain — perfect for shipping the agent you just read about.

$git push zahid main
Live at https://your-agent.zahid.host
Get free account → Pricing
from €0/mo · no card required
LIVE [news/dev-log-deleting-cod…] indexed:0 read:3min 2026-06-15 ·