Skills migrate-to-shoehorn
Migrate test files from `as` type assertions to @total-typescript/shoehorn. Use when user mentions shoehorn, wants to replace `as` in tests, or needs partial test data.
install
source · Clone the upstream repo
git clone https://github.com/mattpocock/skills
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/mattpocock/skills "$T" && mkdir -p ~/.claude/skills && cp -r "$T/migrate-to-shoehorn" ~/.claude/skills/mattpocock-skills-migrate-to-shoehorn && rm -rf "$T"
manifest:
migrate-to-shoehorn/SKILL.mdsource content
Migrate to Shoehorn
Why shoehorn?
shoehorn lets you pass partial data in tests while keeping TypeScript happy. It replaces as assertions with type-safe alternatives.
Test code only. Never use shoehorn in production code.
Problems with
as in tests:
- Trained not to use it
- Must manually specify target type
- Double-as (
) for intentionally wrong dataas unknown as Type
Install
npm i @total-typescript/shoehorn
Migration patterns
Large objects with few needed properties
Before:
type Request = { body: { id: string }; headers: Record<string, string>; cookies: Record<string, string>; // ...20 more properties }; it("gets user by id", () => { // Only care about body.id but must fake entire Request getUser({ body: { id: "123" }, headers: {}, cookies: {}, // ...fake all 20 properties }); });
After:
import { fromPartial } from "@total-typescript/shoehorn"; it("gets user by id", () => { getUser( fromPartial({ body: { id: "123" }, }), ); });
as Type
→ fromPartial()
as TypefromPartial()Before:
getUser({ body: { id: "123" } } as Request);
After:
import { fromPartial } from "@total-typescript/shoehorn"; getUser(fromPartial({ body: { id: "123" } }));
as unknown as Type
→ fromAny()
as unknown as TypefromAny()Before:
getUser({ body: { id: 123 } } as unknown as Request); // wrong type on purpose
After:
import { fromAny } from "@total-typescript/shoehorn"; getUser(fromAny({ body: { id: 123 } }));
When to use each
| Function | Use case |
|---|---|
| Pass partial data that still type-checks |
| Pass intentionally wrong data (keeps autocomplete) |
| Force full object (swap with fromPartial later) |
Workflow
-
Gather requirements - ask user:
- What test files have
assertions causing problems?as - Are they dealing with large objects where only some properties matter?
- Do they need to pass intentionally wrong data for error testing?
- What test files have
-
Install and migrate:
- Install:
npm i @total-typescript/shoehorn - Find test files with
assertions:asgrep -r " as [A-Z]" --include="*.test.ts" --include="*.spec.ts" - Replace
withas TypefromPartial() - Replace
withas unknown as TypefromAny() - Add imports from
@total-typescript/shoehorn - Run type check to verify
- Install: