Stop scattering LLM SDK/API calls across your codebase. Here is the 2-file rule that fixed mine Here is a 2-3 sentence factual summary of the article: The author argues that scattering direct LLM SDK calls across a codebase creates maintenance nightmares, particularly during SDK upgrades, and proposes a "2-file rule" based on hexagonal architecture. This rule restricts all SDK imports to just two files—an adapter and a provider registry—while the rest of the application interacts with a typed, provider-agnostic interface. The author also introduces "capability factories" to consolidate repeated cognitive operations (like classification or drafting) into reusable, centrally-managed functions, which they have released as an open-source library. I upgraded an LLM SDK and expected a routine version bump. Instead I had to touch 15+ files, fix breaking changes across four providers, and spend the rest of the day hoping I had not missed one. That was the second time it happened. I knew there would be a third. If you have ever shipped a production LLM system, you probably recognize the smell: - An SDK minor version renames maxTokens to maxOutputTokens and now 15 files break at runtime, not compile time. - Switching one classification task from Claude to a cheaper model means editing import paths and type signatures in business logic. - You have written classifyEmail , scoreLead , triageTicket , and categorizeRequest , and they are all the same function with a different prompt string. This is not an SDK problem. It is an architecture problem. Here is how I fixed it, and the open-source library that came out of it. The 2-file rule I made one rule: only two files in the entire codebase are allowed to import the LLM SDK. One adapter that translates my interface into SDK calls, and one provider registry that creates clients from config. Everything else talks to a typed interface and has no idea which provider, model, or SDK is in play. This is just hexagonal architecture ports and adapters, per Alistair Cockburn applied to LLMs. You already do this for databases and message queues. Nobody scatters raw SQL across business logic. LLM providers belong in the same category. They are infrastructure, not application logic. The dependency flow goes from this: Application code ├─ direct SDK call ├─ direct SDK call └─ model router leaking SDK types To this: Application code ↓ llmClassify , llmDraft , llmScore ... Capabilities ↓ LLM Port TypeScript interface, zero SDK imports ↓ Adapters + Provider Registry the only 2 files that touch the SDK ↓ OpenAI / Anthropic / Gemini / Ollama / Vercel AI SDK The caller says what it wants taskType: "triage" . The infrastructure decides how . No model name parameter. No provider parameter. Policy is deferred to config. The proof: an SDK upgrade that did not hurt The real test came during a major SDK version jump with breaking changes maxTokens to maxOutputTokens , CoreMessage to ModelMessage , and more . Here is what the migration commit looked like: - 2 files changed the adapter and the agent runtime , plus 1 minor fix. - All 18 activity files unchanged. - All 10 agent files unchanged. - The final migration deleted more code than it added: 192 insertions, 688 deletions. 28 out of 31 files did not change, because they do not know the SDK exists. If a core dependency upgrade touches your business logic, your boundaries are wrong. The part that surprised me: the same 7 operations, everywhere I started this to isolate the SDK. Then I noticed the bigger problem. I was not calling LLMs in 21 different places. I was reimplementing the same seven cognitive operations with slight variations: | Capability | What you give it | What you get back | |---|---|---| Classify | content + rubric | one label from an enum + reasoning | Score | content + rubric + axes | numeric ratings per axis | Draft | persona + situation | longer text in a chosen tone | Summarize | long content + length target | shorter content, key points kept | Extract | unstructured text + schema | a typed structured object | Plan | goal + constraints | an ordered list of steps | Analyze | evidence + question | recommendation with caveats | Five activities classified content with five different prompt structures. Nine drafted messages with nine different tone injections. Same operation, no shared implementation. When I improved one classification prompt, I had to remember to update four other places. I usually forgot. You are not writing 47 prompts. You are writing 7 prompts, 47 times, with slightly different ingredients. So I extracted them into capability factories. A factory takes the invariant parts schema, rubric, model routing, observability hooks and returns a function that takes only the varying part the content : js import { createClassifier } from "@llm-ports/capabilities"; import { z } from "zod"; const IntentSchema = z.object { intent: z.enum "question", "request", "complaint", "feedback", "other" , urgency: z.enum "low", "normal", "high" , reasoning: z.string , } ; export const classifyIntent = createClassifier { port: llm, // your provider-agnostic port schema: IntentSchema, schemaName: "user-intent", rubric: question: asking for information request: wants something done complaint: reports a problem feedback: opinion only other: anything else , } ; Then every call site, across all your files, is the same shape: js const result = await classifyIntent { content: userMessage } ; // { intent: "request", urgency: "high", reasoning: "..." } fully typed Improve the rubric once, and every classifier in the system gets better. Prompt engineering stops being scattered strings and becomes a reusable system asset. llm-ports I pulled this pattern out of my production system and shipped it as an open-source, MIT-licensed TypeScript library: llm-ports . 60 second setup Configure providers in .env : LLM PROVIDER FAST=anthropic|