skills/hsa-receipt-vault/SKILL.md
Maintains a ledger of HSA-qualified medical expenses paid out of pocket (not reimbursed from the HSA), each with date, amount, provider, description, and a path to the scanned receipt. Tracks the running unreimbursed total — the amount of HSA balance the household can pull tax-free at any future date — and validates each candidate against IRS-qualified-expense categories. Use when a new medical receipt arrives, when totaling future tax-free HSA reimbursements, when planning a deferred reimbursement, or when user mentions HSA receipt vault, qualified medical expenses, or HSA shoebox strategy.
npx skillsauth add lyndonkl/claude hsa-receipt-vaultInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
The HSA "shoebox" strategy: pay medical bills out of pocket, save the receipts, let the HSA grow tax-free for decades, then reimburse yourself any time later — there is no time limit on HSA reimbursements as long as the expense was incurred after the HSA was opened. This skill maintains the ledger, validates entries, and exposes the running unreimbursed total.
Every dollar in this ledger is a dollar the household can withdraw tax-free from the HSA at any future point — effectively making the HSA function like a Roth IRA for medical-expense-equivalent dollars, with the optionality to reimburse before retirement if cash is needed.
Two modes:
A — single receipt to add:
{
"date": "2026-03-10",
"amount_cents": 18500,
"provider": "Dr. Smith DDS",
"description": "Dental cleaning",
"receipt_path": "receipts/2026/2026-03-10-smith-dds.pdf",
"paid_from_account_id": "acc_chk_001",
"tx_id": "tx_20260310_004"
}
B — bulk reconcile (scan transactions for category: health.*):
{
"transactions": [...],
"existing_vault": [...],
"hsa_open_date": "2018-01-15"
}
In bulk mode, the skill walks category: health.* transactions, proposes vault entries for those not already in the ledger, and asks the user (via output) to confirm each.
IRS Publication 502 governs HSA-qualified expenses. The skill validates against these high-level categories:
| Qualified | Examples | |---|---| | Medical care | Doctor visits, hospital, surgery, ambulance | | Dental | Cleanings, fillings, extractions, orthodontia | | Vision | Eye exams, glasses, contacts, LASIK | | Mental health | Therapy, psychiatry, prescribed inpatient treatment | | Prescription drugs | Rx, insulin (OTC insulin since 2020) | | OTC medicine and menstrual products | Allowed since CARES Act 2020 | | Medical equipment | Crutches, hearing aids, CPAP, blood-pressure monitors | | Long-term care | Premiums (within IRS limits) and qualified care | | Acupuncture, chiropractic | Yes | | Smoking cessation | Yes | | Weight loss for specific diseases | Only if prescribed for diabetes, hypertension, etc. |
Generally not qualified:
If a candidate is ambiguous, mark qualification: unclear with the IRS Pub 502 reference and let the user confirm.
HSA Vault Progress:
- [ ] Step 1: Validate receipt date is after HSA open date
- [ ] Step 2: Map description to a qualified-expense category
- [ ] Step 3: Confirm receipt_path exists or note it's pending
- [ ] Step 4: Confirm amount paid from a non-HSA account (else not a vault entry)
- [ ] Step 5: Generate id and append to ledger
- [ ] Step 6: Update running unreimbursed_total
- [ ] Step 7: Emit warnings for ambiguous qualifications
Receipts dated before the HSA was opened are NOT eligible for tax-free reimbursement, no matter when the HSA grows. Reject with a clear warning.
Use the description and the merchant category to map to a qualified category. If unmappable, mark qualification: unclear and suggest the user attach a Letter of Medical Necessity if applicable.
The receipt must be retained — the IRS can request substantiation up to the statute of limitations after reimbursement. Acceptable receipt formats: PDF, JPG, PNG. Path stored relative to the household-finance root: receipts/YYYY/YYYY-MM-DD-provider.ext.
If receipt_path is missing, accept the entry but mark receipt_status: pending and emit a warning. Do not allow pending entries to count toward the running unreimbursed_total — they will, once the receipt is attached.
A receipt only enters the vault if it was paid from a non-HSA account. Anything paid directly from the HSA debit card or HSA bill-pay is already a tax-free distribution; it doesn't go in the shoebox.
Generate id: rcpt_<year>_<seq>, append to hsa.json.receipt_vault[]. Never mutate prior entries; if a correction is needed, append a correction_of: <prior_id> entry.
unreimbursed_total_cents = Σ amount_cents WHERE receipt_status = "ok" AND reimbursed = false.
This is the headline number — the dollars the household can pull tax-free at any time.
qualification: unclear — surface to user.For mode A (single add):
{
"added": {
"id": "rcpt_2026_0014",
"date": "2026-03-10",
"amount_cents": 18500,
"provider": "Dr. Smith DDS",
"description": "Dental cleaning",
"category": "dental",
"qualification": "ok",
"receipt_path": "receipts/2026/2026-03-10-smith-dds.pdf",
"receipt_status": "ok",
"paid_from_account_id": "acc_chk_001",
"tx_id": "tx_20260310_004",
"reimbursed": false,
"tax_year": 2026
},
"running_totals": {
"unreimbursed_total_cents": 4435500,
"by_year_cents": {
"2024": 1820000,
"2025": 2150000,
"2026": 465500
}
},
"warnings": []
}
For mode B (bulk reconcile):
{
"proposed": [
{
"tx_id": "tx_20260118_005",
"amount_cents": 4500,
"provider": "CVS Pharmacy",
"description_raw": "CVS/PHARMACY #04123",
"suggested_category": "prescription_drugs",
"qualification": "ok",
"needs_receipt_upload": true
},
{
"tx_id": "tx_20260201_002",
"amount_cents": 8500,
"provider": "Equinox",
"description_raw": "EQUINOX MONTHLY DUES",
"suggested_category": null,
"qualification": "not_qualified",
"rationale": "Gym memberships are not qualified unless prescribed for a specific medical condition with a Letter of Medical Necessity."
}
],
"running_totals": {
"unreimbursed_total_cents": 4435500
}
}
receipt_path faithfully; flag missing receipts; do NOT count them toward the unreimbursed total.correction_of to amend.development
--- name: zettel-note description: The note-writing discipline for this vault's evergreen knowledge graph, modeled on a Zettelkasten reading companion and governed by the vault conventions. Enforces declarative-claim titles, one claim per note (atomicity), own-words prose with no block quotes, the piped [[slug|Title]] link form, the labeled link-relationship vocabulary (Confirms/Contradicts/Extends/Context/Prerequisite/Builds-on/Applies/Example-of/Contrasts-with), 3-6 links per note, and search-
development
Plans between-round FIFA World Cup Fantasy transfers — budgets the round's free transfer(s), forces out players whose nation has been eliminated, chases fixture-swing drops, upgrades on value, and decides when a rebuild is large enough to fire the Wildcard instead of spending free transfers one at a time. Ranks candidate in/out pairs by EV gain over each player's remaining survival horizon (delta xEV weighted by progression_carry) MINUS transfer cost (a free transfer is cheap, a points hit is real, churning the squad for marginal swings is a critic flag), and tags forced/fixture/upgrade priority. Emits a `transfer-plan` signal. Use when called by wc-squad-architect (whose transfer work this skill is the engine for) and by the strategists in the populate stage when their candidate is transfer-adjacent rather than a full rebuild.
testing
Reads and updates the FIFA World Cup Fantasy tournament state machine (footballfantasy/context/tournament-state.md) — the temporal backbone tracking phase (pre-tournament → group MD1-3 → R32 → R16 → QF → SF → final), budget ($100m group / $105m knockouts), nation cap (3 group, loosening in knockouts), chips remaining, surviving nations, each owned player's elimination-risk horizon, and deadlines. Validates state on load (count/feasibility checks), applies phase transitions, and appends to the append-only state log (never silent overwrite). Use to load state at the start of a run and to commit state changes after the manager makes a move.
development
Validates and persists FIFA World Cup Fantasy signal files to signals/YYYY-MM-DD-<type>.md. Checks the required frontmatter (type, round, date, emitted_by, confidence, source_urls), range-checks declared numeric signals, confirms every factual claim carries a source URL or "manager-provided", rejects unknown signal types, and refuses to persist a signal that fails validation (logging the failure instead). Keeps the inter-agent signal layer auditable so downstream agents can trust what they read and never re-derive it. Use whenever an agent or skill writes a signal.