skills/adverse-selection-prior/SKILL.md
Produces a Bayesian prior probability that an offered transaction is +EV for the recipient, given that the counterparty chose to propose it. Applies Akerlof market-for-lemons logic -- if they offered it, they believe it is +EV for them, so the prior that it is +EV for us is materially below 50%. Reusable across trade evaluation, waiver drops (another team dropping a player is also adverse selection), job-offer analysis, M&A, and any "someone offered me this" situation. Use when you receive an unsolicited trade/offer/proposal, analyzing incoming trade prior, evaluating why a counterparty proposed a deal, or when user mentions adverse selection, market for lemons, why did they offer this, incoming trade prior, they proposed it, Bayesian adjustment on received offer.
npx skillsauth add lyndonkl/claude adverse-selection-priorInstall 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.
Scenario: We receive a fantasy trade offer. Opponent (classified active archetype) offers us a star hitter (their side) for two of our mid-tier hitters (our side). On surface the offer looks roughly fair -- our own projection model says the deal is +3% EV for us.
Inputs:
offer_type: tradeproposer_archetype: activeoffer_symmetry_score: 72 (looks fair-to-slightly-favorable for us)proposer_info_asymmetry: 55 (moderate -- we do not have closer news that they might have, but no public bombshell either)Step 1 -- Base prior by offer type:
Step 2 -- Symmetry adjustment:
offer_symmetry_score = 72 > 70 threshold -> +0.07.Step 3 -- Asymmetry adjustment:
proposer_info_asymmetry = 55 (below 60 threshold but non-trivial) -> -0.05.Step 4 -- Archetype adjustment:
active archetype -> they analyzed before offering -> -0.05.prior_ev_probability: 0.37.Step 5 -- Recommended adjustment (EV haircut):
recommended_adjustment = 0.88 (apply a 12% EV haircut to our own projection).Step 6 -- Bayesian rationale: "Active trader proposed this offer. They had the option to propose any trade or no trade, and they selected this one -- strong evidence it is +EV for them. Surface symmetry (72) partially offsets, but moderate info asymmetry (55) and analytical archetype push the prior to 0.37. Apply 12% EV haircut. Our +3% projection becomes -9% after adjustment; this is below the accept threshold."
Step 7 -- Override hints:
Output contract:
{
"prior_ev_probability": 0.37,
"recommended_adjustment": 0.88,
"bayesian_rationale": "Active trader selected this offer from all possible offers they could have made -- strong signal it is +EV for them. Surface symmetry (72) partially offsets, but moderate info asymmetry (55) and analytical archetype push prior below 0.40.",
"override_hints": [
"If closer/injury news broke in last 48h, asymmetry jumps to 85+; re-run.",
"If we can independently confirm the star has no hidden issue, symmetry rises to 85; re-run.",
"If repeated-game partner with cooperation history, base prior rises by +0.05."
]
}
Copy this checklist and track progress:
Adverse Selection Prior Progress:
- [ ] Step 1: Identify offer_type and set base prior
- [ ] Step 2: Collect proposer_archetype from opponent classifier
- [ ] Step 3: Score offer_symmetry (0-100) from surface analysis
- [ ] Step 4: Score proposer_info_asymmetry (0-100)
- [ ] Step 5: Apply adjustments to produce prior_ev_probability
- [ ] Step 6: Convert prior to recommended_adjustment (multiplicative haircut)
- [ ] Step 7: Write bayesian_rationale and override_hints
- [ ] Step 8: Validate against rubric
Step 1: Identify offer type and set base prior
Classify the incoming proposal. Different offer types carry different base priors because the counterparty's selection pressure differs. See resources/template.md for the base-prior table.
trade -> base prior 0.40 (active selection; they wrote the offer)waiver_claim_dropped_by_other -> base prior 0.42 (they chose to drop the player; weaker selection than a trade)free_agent_pickup_available -> base prior 0.48 (no one claimed; may just mean no one noticed, weaker adverse selection)counter_offer -> base prior 0.38 (they rejected our first offer and crafted a counter -- the selection is tighter)generic -> base prior 0.40 (unknown context)Step 2: Collect proposer archetype
Read the counterparty's archetype from the opponent classifier (if available) or estimate it. See resources/methodology.md for the full adjustment table.
active -> analyzed before offering -> archetype_delta = -0.05expert -> strongest selection pressure -> archetype_delta = -0.08dormant -> may be clicking without deep analysis -> archetype_delta = +0.08frustrated -> mixed (panic-selling vs scheming) -> archetype_delta = 0.0 (but flag for case-by-case review)unknown -> neutral -> archetype_delta = 0.0Step 3: Score offer symmetry
Compute offer_symmetry_score (0-100) for how fair the offer looks on surface using our own valuation model.
offer_symmetry_score > 70, add +0.05 to +0.10 (higher for scores above 85)offer_symmetry_score < 30 (and still not obviously predatory), flag as "too good to be true" -- investigateStep 4: Score proposer info asymmetry
Estimate proposer_info_asymmetry (0-100) -- do they plausibly have information we do not?
Step 5: Apply adjustments to produce prior
Combine base prior with all deltas. See resources/methodology.md for the derivation.
base_prior from Step 1symmetry_delta from Step 3asymmetry_delta from Step 4archetype_delta from Step 2prior_ev_probability = clipped sumStep 6: Convert prior to recommended adjustment
Translate the prior into a multiplicative EV haircut that downstream consumers can apply to their own EV calculation. See Quick Reference for the mapping.
recommended_adjustment = 1.00 (no haircut; rare case)recommended_adjustment = 0.95recommended_adjustment = 0.90recommended_adjustment = 0.85recommended_adjustment = 0.80 (deep haircut; strong adverse-selection signal)adjusted_EV = own_EV x recommended_adjustmentStep 7: Write rationale and override hints
Produce the string outputs that make the prior reviewable.
bayesian_rationale: 2-4 sentences naming the base prior, the two largest adjustments, and the conclusionoverride_hints: 2-5 conditional statements of the form "If X, then re-run with Y adjusted to Z"Step 8: Validate against rubric
Score the output against resources/evaluators/rubric_adverse_selection_prior.json. Minimum standard: average score of 3.5 or above.
Pattern 1: Dormant-manager trade offer (prior rises toward 0.50)
dormantPattern 2: Expert-trader offer (prior drops toward 0.25)
expertPattern 3: Frustrated/panic-selling offer (mixed; case-by-case)
frustratedPattern 4: Unsolicited job offer or M&A approach
Pattern 5: Waiver drop by another team
Base prior is always below 0.50 for received offers. This is the core of the market-for-lemons insight. If your prior exceeds 0.50, you have either (a) mis-specified the archetype (e.g., dormant with very high symmetry can approach but rarely cross 0.50) or (b) ignored the selection pressure. Double-check.
"Seems fair" = "probably bad". Counterparties who propose transactions have selected them from the space of possible transactions. A surface-fair offer is the worst-case from an adverse-selection standpoint because they had many fair-looking options and chose the one best for them.
Do not double-count adjustments. If you have already adjusted your own EV for known information (e.g., you penalized the player because you know about an injury), do not also treat that information as asymmetry reducing the prior. Asymmetry captures information they have that you do not.
Clip priors to [0.10, 0.70]. Priors outside this band are almost always specification errors. A 0.05 prior implies near-certainty of a predatory offer, which is rare even from experts. A 0.75 prior implies we should take every offer, contradicting the selection argument.
Archetype signal comes from behavior, not self-report. A counterparty who claims to be a "casual player" but has made 40 waiver claims and won 3 bidding wars is not dormant. Read archetype from observed actions.
Panic-sell signals must be triangulated. "They seem frustrated" is not sufficient to raise the prior. Require at least two of: recent heavy losses, message-tone evidence, visible tilt-trades (overreaching for recently-hot players), public complaint about the league.
Repeated-game reputation adjusts the prior. If you have cooperated with this counterparty 5+ times and they have never sent a -EV offer, the archetype-independent base prior rises by ~0.05. If they have sent a predatory offer before, the base prior drops by ~0.05. Track per-counterparty history.
Override hints must be actionable and specific. "Do more research" is not a hint. "If the team's closer has been pulled from the role this week, re-run with asymmetry=85" is actionable.
The recommended adjustment is multiplicative, not additive. Downstream consumers apply adjusted_EV = own_EV x recommended_adjustment. Do not output an additive haircut -- it breaks when own_EV is near zero or negative.
This skill produces a prior, not a verdict. Downstream decision logic (accept/counter/reject) belongs in the consumer skill (e.g., mlb-trade-analyzer or the M&A decision agent). This skill only outputs the prior and the haircut factor.
Base priors by offer type:
| Offer type | Base prior | Rationale |
|------------|------------|-----------|
| trade | 0.40 | Active selection; they wrote the offer |
| waiver_claim_dropped_by_other | 0.42 | They dropped it; weaker selection than targeted trade |
| free_agent_pickup_available | 0.48 | Weak selection; may be unclaimed due to oversight |
| counter_offer | 0.38 | Tighter selection -- they rejected first offer and crafted this |
| generic | 0.40 | Default when context unknown |
Archetype adjustments:
| Archetype | Delta | Interpretation |
|-----------|-------|----------------|
| expert | -0.08 | Strongest selection; they chose this from many offers |
| active | -0.05 | Analyzed before offering |
| frustrated | 0.0 (flag) | Mixed: panic-sell (+) vs scheme (-); require sub-signal |
| dormant | +0.08 | May be clicking without analysis |
| unknown | 0.0 | Neutral |
Symmetry adjustments (based on offer_symmetry_score):
| Score | Delta | |-------|-------| | 85-100 | +0.10 | | 70-85 | +0.05 to +0.07 | | 30-70 | 0.0 | | < 30 | 0.0 (and flag as "too good to be true") |
Asymmetry adjustments (based on proposer_info_asymmetry):
| Score | Delta | |-------|-------| | 0-30 | 0.0 | | 30-60 | -0.03 to -0.05 | | 60-80 | -0.10 to -0.15 | | 80-100 | -0.15 to -0.20 |
Prior-to-adjustment mapping:
| prior_ev_probability | recommended_adjustment | Interpretation |
|------------------------|--------------------------|----------------|
| >= 0.50 | 1.00 | No haircut; unusual for received offer |
| 0.45 | 0.95 | Light haircut |
| 0.40 | 0.90 | Standard haircut |
| 0.35 | 0.85 | Moderate haircut |
| 0.30 | 0.80 | Deep haircut -- strong adverse selection |
| <= 0.25 | 0.75 | Predatory suspicion -- default to reject |
Output contract:
prior_ev_probability: float in [0.10, 0.70]
recommended_adjustment: float in [0.75, 1.00]
bayesian_rationale: string (2-4 sentences)
override_hints: string[] (2-5 conditional statements)
Key resources:
Inputs required:
offer_type (enum: trade | waiver_claim_dropped_by_other | free_agent_pickup_available | counter_offer | generic)proposer_archetype (enum: active | dormant | frustrated | expert | unknown)offer_symmetry_score (int, 0-100)proposer_info_asymmetry (int, 0-100)Outputs produced:
prior_ev_probability (float, 0-1, baseline < 0.5 for received offers)recommended_adjustment (float, multiplicative factor)bayesian_rationale (string)override_hints (string[])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.