pigeongov 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +258 -0
- package/LICENSE +21 -0
- package/PRIVACY.md +38 -0
- package/README.md +234 -0
- package/agents.json +72 -0
- package/dist/bin/pigeongov.d.ts +2 -0
- package/dist/bin/pigeongov.js +4 -0
- package/dist/bin/pigeongov.js.map +1 -0
- package/dist/src/advisory/decision-support/contribution-optimizer.d.ts +36 -0
- package/dist/src/advisory/decision-support/contribution-optimizer.js +88 -0
- package/dist/src/advisory/decision-support/contribution-optimizer.js.map +1 -0
- package/dist/src/advisory/decision-support/deduction-optimizer.d.ts +31 -0
- package/dist/src/advisory/decision-support/deduction-optimizer.js +96 -0
- package/dist/src/advisory/decision-support/deduction-optimizer.js.map +1 -0
- package/dist/src/advisory/life-events/events.d.ts +4 -0
- package/dist/src/advisory/life-events/events.js +156 -0
- package/dist/src/advisory/life-events/events.js.map +1 -0
- package/dist/src/advisory/life-events/planner.d.ts +19 -0
- package/dist/src/advisory/life-events/planner.js +72 -0
- package/dist/src/advisory/life-events/planner.js.map +1 -0
- package/dist/src/advisory/screener/engine.d.ts +4 -0
- package/dist/src/advisory/screener/engine.js +368 -0
- package/dist/src/advisory/screener/engine.js.map +1 -0
- package/dist/src/advisory/screener/intake.d.ts +68 -0
- package/dist/src/advisory/screener/intake.js +93 -0
- package/dist/src/advisory/screener/intake.js.map +1 -0
- package/dist/src/analytics/stats.d.ts +18 -0
- package/dist/src/analytics/stats.js +158 -0
- package/dist/src/analytics/stats.js.map +1 -0
- package/dist/src/api/handlers/deadlines.d.ts +2 -0
- package/dist/src/api/handlers/deadlines.js +7 -0
- package/dist/src/api/handlers/deadlines.js.map +1 -0
- package/dist/src/api/handlers/openapi.d.ts +2 -0
- package/dist/src/api/handlers/openapi.js +7 -0
- package/dist/src/api/handlers/openapi.js.map +1 -0
- package/dist/src/api/handlers/workflows.d.ts +5 -0
- package/dist/src/api/handlers/workflows.js +73 -0
- package/dist/src/api/handlers/workflows.js.map +1 -0
- package/dist/src/api/index.d.ts +20 -0
- package/dist/src/api/index.js +70 -0
- package/dist/src/api/index.js.map +1 -0
- package/dist/src/api/openapi.d.ts +5 -0
- package/dist/src/api/openapi.js +384 -0
- package/dist/src/api/openapi.js.map +1 -0
- package/dist/src/api/router.d.ts +19 -0
- package/dist/src/api/router.js +57 -0
- package/dist/src/api/router.js.map +1 -0
- package/dist/src/cli/commands/completions.d.ts +2 -0
- package/dist/src/cli/commands/completions.js +396 -0
- package/dist/src/cli/commands/completions.js.map +1 -0
- package/dist/src/cli/commands/deadlines.d.ts +2 -0
- package/dist/src/cli/commands/deadlines.js +47 -0
- package/dist/src/cli/commands/deadlines.js.map +1 -0
- package/dist/src/cli/commands/doctor.d.ts +2 -0
- package/dist/src/cli/commands/doctor.js +50 -0
- package/dist/src/cli/commands/doctor.js.map +1 -0
- package/dist/src/cli/commands/drafts.d.ts +2 -0
- package/dist/src/cli/commands/drafts.js +75 -0
- package/dist/src/cli/commands/drafts.js.map +1 -0
- package/dist/src/cli/commands/extract.d.ts +2 -0
- package/dist/src/cli/commands/extract.js +17 -0
- package/dist/src/cli/commands/extract.js.map +1 -0
- package/dist/src/cli/commands/fees.d.ts +2 -0
- package/dist/src/cli/commands/fees.js +40 -0
- package/dist/src/cli/commands/fees.js.map +1 -0
- package/dist/src/cli/commands/fill.d.ts +25 -0
- package/dist/src/cli/commands/fill.js +354 -0
- package/dist/src/cli/commands/fill.js.map +1 -0
- package/dist/src/cli/commands/glossary.d.ts +2 -0
- package/dist/src/cli/commands/glossary.js +90 -0
- package/dist/src/cli/commands/glossary.js.map +1 -0
- package/dist/src/cli/commands/life-event.d.ts +2 -0
- package/dist/src/cli/commands/life-event.js +41 -0
- package/dist/src/cli/commands/life-event.js.map +1 -0
- package/dist/src/cli/commands/list.d.ts +2 -0
- package/dist/src/cli/commands/list.js +19 -0
- package/dist/src/cli/commands/list.js.map +1 -0
- package/dist/src/cli/commands/machine.d.ts +2 -0
- package/dist/src/cli/commands/machine.js +100 -0
- package/dist/src/cli/commands/machine.js.map +1 -0
- package/dist/src/cli/commands/merge.d.ts +2 -0
- package/dist/src/cli/commands/merge.js +24 -0
- package/dist/src/cli/commands/merge.js.map +1 -0
- package/dist/src/cli/commands/plugin.d.ts +2 -0
- package/dist/src/cli/commands/plugin.js +119 -0
- package/dist/src/cli/commands/plugin.js.map +1 -0
- package/dist/src/cli/commands/profile.d.ts +2 -0
- package/dist/src/cli/commands/profile.js +125 -0
- package/dist/src/cli/commands/profile.js.map +1 -0
- package/dist/src/cli/commands/review.d.ts +2 -0
- package/dist/src/cli/commands/review.js +25 -0
- package/dist/src/cli/commands/review.js.map +1 -0
- package/dist/src/cli/commands/scaffold.d.ts +2 -0
- package/dist/src/cli/commands/scaffold.js +304 -0
- package/dist/src/cli/commands/scaffold.js.map +1 -0
- package/dist/src/cli/commands/schemas.d.ts +2 -0
- package/dist/src/cli/commands/schemas.js +14 -0
- package/dist/src/cli/commands/schemas.js.map +1 -0
- package/dist/src/cli/commands/screen.d.ts +2 -0
- package/dist/src/cli/commands/screen.js +89 -0
- package/dist/src/cli/commands/screen.js.map +1 -0
- package/dist/src/cli/commands/serve.d.ts +2 -0
- package/dist/src/cli/commands/serve.js +46 -0
- package/dist/src/cli/commands/serve.js.map +1 -0
- package/dist/src/cli/commands/start.d.ts +2 -0
- package/dist/src/cli/commands/start.js +16 -0
- package/dist/src/cli/commands/start.js.map +1 -0
- package/dist/src/cli/commands/stats.d.ts +2 -0
- package/dist/src/cli/commands/stats.js +17 -0
- package/dist/src/cli/commands/stats.js.map +1 -0
- package/dist/src/cli/commands/testdata.d.ts +2 -0
- package/dist/src/cli/commands/testdata.js +60 -0
- package/dist/src/cli/commands/testdata.js.map +1 -0
- package/dist/src/cli/commands/tui.d.ts +2 -0
- package/dist/src/cli/commands/tui.js +26 -0
- package/dist/src/cli/commands/tui.js.map +1 -0
- package/dist/src/cli/commands/validate.d.ts +2 -0
- package/dist/src/cli/commands/validate.js +22 -0
- package/dist/src/cli/commands/validate.js.map +1 -0
- package/dist/src/cli/commands/vault.d.ts +2 -0
- package/dist/src/cli/commands/vault.js +73 -0
- package/dist/src/cli/commands/vault.js.map +1 -0
- package/dist/src/cli/commands/workflows.d.ts +2 -0
- package/dist/src/cli/commands/workflows.js +23 -0
- package/dist/src/cli/commands/workflows.js.map +1 -0
- package/dist/src/cli/display/review.d.ts +2 -0
- package/dist/src/cli/display/review.js +4 -0
- package/dist/src/cli/display/review.js.map +1 -0
- package/dist/src/cli/display/summary.d.ts +2 -0
- package/dist/src/cli/display/summary.js +19 -0
- package/dist/src/cli/display/summary.js.map +1 -0
- package/dist/src/cli/display/validation.d.ts +2 -0
- package/dist/src/cli/display/validation.js +16 -0
- package/dist/src/cli/display/validation.js.map +1 -0
- package/dist/src/cli/index.d.ts +1 -0
- package/dist/src/cli/index.js +83 -0
- package/dist/src/cli/index.js.map +1 -0
- package/dist/src/cli/output.d.ts +21 -0
- package/dist/src/cli/output.js +107 -0
- package/dist/src/cli/output.js.map +1 -0
- package/dist/src/cli/prompts/common.d.ts +21 -0
- package/dist/src/cli/prompts/common.js +53 -0
- package/dist/src/cli/prompts/common.js.map +1 -0
- package/dist/src/cli/prompts/credits.d.ts +3 -0
- package/dist/src/cli/prompts/credits.js +23 -0
- package/dist/src/cli/prompts/credits.js.map +1 -0
- package/dist/src/cli/prompts/deductions.d.ts +12 -0
- package/dist/src/cli/prompts/deductions.js +26 -0
- package/dist/src/cli/prompts/deductions.js.map +1 -0
- package/dist/src/cli/prompts/identity.d.ts +8 -0
- package/dist/src/cli/prompts/identity.js +36 -0
- package/dist/src/cli/prompts/identity.js.map +1 -0
- package/dist/src/cli/prompts/income.d.ts +15 -0
- package/dist/src/cli/prompts/income.js +61 -0
- package/dist/src/cli/prompts/income.js.map +1 -0
- package/dist/src/cli/prompts/workflow.d.ts +3 -0
- package/dist/src/cli/prompts/workflow.js +74 -0
- package/dist/src/cli/prompts/workflow.js.map +1 -0
- package/dist/src/cli/support.d.ts +53 -0
- package/dist/src/cli/support.js +80 -0
- package/dist/src/cli/support.js.map +1 -0
- package/dist/src/cli/tui.d.ts +22 -0
- package/dist/src/cli/tui.js +122 -0
- package/dist/src/cli/tui.js.map +1 -0
- package/dist/src/engine/audit-risk/scorer.d.ts +36 -0
- package/dist/src/engine/audit-risk/scorer.js +132 -0
- package/dist/src/engine/audit-risk/scorer.js.map +1 -0
- package/dist/src/engine/crypto/cost-basis.d.ts +19 -0
- package/dist/src/engine/crypto/cost-basis.js +115 -0
- package/dist/src/engine/crypto/cost-basis.js.map +1 -0
- package/dist/src/engine/crypto/transaction-parser.d.ts +36 -0
- package/dist/src/engine/crypto/transaction-parser.js +124 -0
- package/dist/src/engine/crypto/transaction-parser.js.map +1 -0
- package/dist/src/engine/crypto/wash-sale.d.ts +25 -0
- package/dist/src/engine/crypto/wash-sale.js +51 -0
- package/dist/src/engine/crypto/wash-sale.js.map +1 -0
- package/dist/src/engine/equity/espp.d.ts +41 -0
- package/dist/src/engine/equity/espp.js +90 -0
- package/dist/src/engine/equity/espp.js.map +1 -0
- package/dist/src/engine/equity/iso.d.ts +42 -0
- package/dist/src/engine/equity/iso.js +94 -0
- package/dist/src/engine/equity/iso.js.map +1 -0
- package/dist/src/engine/equity/nso.d.ts +35 -0
- package/dist/src/engine/equity/nso.js +63 -0
- package/dist/src/engine/equity/nso.js.map +1 -0
- package/dist/src/engine/equity/rsu.d.ts +33 -0
- package/dist/src/engine/equity/rsu.js +64 -0
- package/dist/src/engine/equity/rsu.js.map +1 -0
- package/dist/src/engine/field-mapper.d.ts +26 -0
- package/dist/src/engine/field-mapper.js +119 -0
- package/dist/src/engine/field-mapper.js.map +1 -0
- package/dist/src/engine/forms/core-1040.d.ts +2 -0
- package/dist/src/engine/forms/core-1040.js +124 -0
- package/dist/src/engine/forms/core-1040.js.map +1 -0
- package/dist/src/engine/forms/form-8949.d.ts +21 -0
- package/dist/src/engine/forms/form-8949.js +96 -0
- package/dist/src/engine/forms/form-8949.js.map +1 -0
- package/dist/src/engine/forms/index.d.ts +7 -0
- package/dist/src/engine/forms/index.js +6 -0
- package/dist/src/engine/forms/index.js.map +1 -0
- package/dist/src/engine/forms/schedule-b.d.ts +15 -0
- package/dist/src/engine/forms/schedule-b.js +78 -0
- package/dist/src/engine/forms/schedule-b.js.map +1 -0
- package/dist/src/engine/forms/schedule-d.d.ts +9 -0
- package/dist/src/engine/forms/schedule-d.js +86 -0
- package/dist/src/engine/forms/schedule-d.js.map +1 -0
- package/dist/src/engine/fpl.d.ts +70 -0
- package/dist/src/engine/fpl.js +127 -0
- package/dist/src/engine/fpl.js.map +1 -0
- package/dist/src/engine/gig/home-office.d.ts +34 -0
- package/dist/src/engine/gig/home-office.js +53 -0
- package/dist/src/engine/gig/home-office.js.map +1 -0
- package/dist/src/engine/gig/mileage.d.ts +30 -0
- package/dist/src/engine/gig/mileage.js +45 -0
- package/dist/src/engine/gig/mileage.js.map +1 -0
- package/dist/src/engine/gig/multi-1099.d.ts +19 -0
- package/dist/src/engine/gig/multi-1099.js +28 -0
- package/dist/src/engine/gig/multi-1099.js.map +1 -0
- package/dist/src/engine/multiyear/carryforward.d.ts +31 -0
- package/dist/src/engine/multiyear/carryforward.js +77 -0
- package/dist/src/engine/multiyear/carryforward.js.map +1 -0
- package/dist/src/engine/multiyear/compare.d.ts +25 -0
- package/dist/src/engine/multiyear/compare.js +94 -0
- package/dist/src/engine/multiyear/compare.js.map +1 -0
- package/dist/src/engine/multiyear/import.d.ts +23 -0
- package/dist/src/engine/multiyear/import.js +193 -0
- package/dist/src/engine/multiyear/import.js.map +1 -0
- package/dist/src/engine/optimizer/filing-status.d.ts +29 -0
- package/dist/src/engine/optimizer/filing-status.js +66 -0
- package/dist/src/engine/optimizer/filing-status.js.map +1 -0
- package/dist/src/engine/orchestrator.d.ts +14 -0
- package/dist/src/engine/orchestrator.js +135 -0
- package/dist/src/engine/orchestrator.js.map +1 -0
- package/dist/src/engine/scenarios/whatif.d.ts +34 -0
- package/dist/src/engine/scenarios/whatif.js +70 -0
- package/dist/src/engine/scenarios/whatif.js.map +1 -0
- package/dist/src/engine/state/ca/calculator.d.ts +3 -0
- package/dist/src/engine/state/ca/calculator.js +107 -0
- package/dist/src/engine/state/ca/calculator.js.map +1 -0
- package/dist/src/engine/state/common.d.ts +39 -0
- package/dist/src/engine/state/common.js +115 -0
- package/dist/src/engine/state/common.js.map +1 -0
- package/dist/src/engine/state/ga/calculator.d.ts +3 -0
- package/dist/src/engine/state/ga/calculator.js +82 -0
- package/dist/src/engine/state/ga/calculator.js.map +1 -0
- package/dist/src/engine/state/il/calculator.d.ts +3 -0
- package/dist/src/engine/state/il/calculator.js +49 -0
- package/dist/src/engine/state/il/calculator.js.map +1 -0
- package/dist/src/engine/state/mi/calculator.d.ts +3 -0
- package/dist/src/engine/state/mi/calculator.js +60 -0
- package/dist/src/engine/state/mi/calculator.js.map +1 -0
- package/dist/src/engine/state/nc/calculator.d.ts +3 -0
- package/dist/src/engine/state/nc/calculator.js +54 -0
- package/dist/src/engine/state/nc/calculator.js.map +1 -0
- package/dist/src/engine/state/nj/calculator.d.ts +3 -0
- package/dist/src/engine/state/nj/calculator.js +90 -0
- package/dist/src/engine/state/nj/calculator.js.map +1 -0
- package/dist/src/engine/state/ny/calculator.d.ts +3 -0
- package/dist/src/engine/state/ny/calculator.js +103 -0
- package/dist/src/engine/state/ny/calculator.js.map +1 -0
- package/dist/src/engine/state/oh/calculator.d.ts +3 -0
- package/dist/src/engine/state/oh/calculator.js +79 -0
- package/dist/src/engine/state/oh/calculator.js.map +1 -0
- package/dist/src/engine/state/pa/calculator.d.ts +3 -0
- package/dist/src/engine/state/pa/calculator.js +58 -0
- package/dist/src/engine/state/pa/calculator.js.map +1 -0
- package/dist/src/engine/state/registry.d.ts +9 -0
- package/dist/src/engine/state/registry.js +67 -0
- package/dist/src/engine/state/registry.js.map +1 -0
- package/dist/src/engine/state/types.d.ts +57 -0
- package/dist/src/engine/state/types.js +2 -0
- package/dist/src/engine/state/types.js.map +1 -0
- package/dist/src/engine/state/va/calculator.d.ts +3 -0
- package/dist/src/engine/state/va/calculator.js +76 -0
- package/dist/src/engine/state/va/calculator.js.map +1 -0
- package/dist/src/engine/suggestions/missed-deductions.d.ts +43 -0
- package/dist/src/engine/suggestions/missed-deductions.js +128 -0
- package/dist/src/engine/suggestions/missed-deductions.js.map +1 -0
- package/dist/src/engine/tax-calculator.d.ts +55 -0
- package/dist/src/engine/tax-calculator.js +156 -0
- package/dist/src/engine/tax-calculator.js.map +1 -0
- package/dist/src/engine/tax-constants-2025.d.ts +56 -0
- package/dist/src/engine/tax-constants-2025.js +138 -0
- package/dist/src/engine/tax-constants-2025.js.map +1 -0
- package/dist/src/engine/types.d.ts +77 -0
- package/dist/src/engine/types.js +2 -0
- package/dist/src/engine/types.js.map +1 -0
- package/dist/src/engine/va-math.d.ts +24 -0
- package/dist/src/engine/va-math.js +90 -0
- package/dist/src/engine/va-math.js.map +1 -0
- package/dist/src/engine/validator.d.ts +7 -0
- package/dist/src/engine/validator.js +108 -0
- package/dist/src/engine/validator.js.map +1 -0
- package/dist/src/glossary/entries/healthcare.d.ts +2 -0
- package/dist/src/glossary/entries/healthcare.js +78 -0
- package/dist/src/glossary/entries/healthcare.js.map +1 -0
- package/dist/src/glossary/entries/immigration.d.ts +2 -0
- package/dist/src/glossary/entries/immigration.js +112 -0
- package/dist/src/glossary/entries/immigration.js.map +1 -0
- package/dist/src/glossary/entries/tax.d.ts +2 -0
- package/dist/src/glossary/entries/tax.js +149 -0
- package/dist/src/glossary/entries/tax.js.map +1 -0
- package/dist/src/glossary/entries/unemployment.d.ts +2 -0
- package/dist/src/glossary/entries/unemployment.js +52 -0
- package/dist/src/glossary/entries/unemployment.js.map +1 -0
- package/dist/src/glossary/index.d.ts +19 -0
- package/dist/src/glossary/index.js +39 -0
- package/dist/src/glossary/index.js.map +1 -0
- package/dist/src/i18n/index.d.ts +21 -0
- package/dist/src/i18n/index.js +92 -0
- package/dist/src/i18n/index.js.map +1 -0
- package/dist/src/mcp/result.d.ts +3 -0
- package/dist/src/mcp/result.js +7 -0
- package/dist/src/mcp/result.js.map +1 -0
- package/dist/src/mcp/return-bundle.d.ts +19 -0
- package/dist/src/mcp/return-bundle.js +49 -0
- package/dist/src/mcp/return-bundle.js.map +1 -0
- package/dist/src/mcp/shared.d.ts +49102 -0
- package/dist/src/mcp/shared.js +354 -0
- package/dist/src/mcp/shared.js.map +1 -0
- package/dist/src/mcp/tools/build-packet.d.ts +7 -0
- package/dist/src/mcp/tools/build-packet.js +23 -0
- package/dist/src/mcp/tools/build-packet.js.map +1 -0
- package/dist/src/mcp/tools/calculate-tax.d.ts +40 -0
- package/dist/src/mcp/tools/calculate-tax.js +17 -0
- package/dist/src/mcp/tools/calculate-tax.js.map +1 -0
- package/dist/src/mcp/tools/deadlines.d.ts +7 -0
- package/dist/src/mcp/tools/deadlines.js +23 -0
- package/dist/src/mcp/tools/deadlines.js.map +1 -0
- package/dist/src/mcp/tools/describe-form.d.ts +14 -0
- package/dist/src/mcp/tools/describe-form.js +23 -0
- package/dist/src/mcp/tools/describe-form.js.map +1 -0
- package/dist/src/mcp/tools/describe-workflow.d.ts +6 -0
- package/dist/src/mcp/tools/describe-workflow.js +16 -0
- package/dist/src/mcp/tools/describe-workflow.js.map +1 -0
- package/dist/src/mcp/tools/explain-flag.d.ts +7 -0
- package/dist/src/mcp/tools/explain-flag.js +19 -0
- package/dist/src/mcp/tools/explain-flag.js.map +1 -0
- package/dist/src/mcp/tools/extract-document.d.ts +13 -0
- package/dist/src/mcp/tools/extract-document.js +20 -0
- package/dist/src/mcp/tools/extract-document.js.map +1 -0
- package/dist/src/mcp/tools/fees.d.ts +7 -0
- package/dist/src/mcp/tools/fees.js +24 -0
- package/dist/src/mcp/tools/fees.js.map +1 -0
- package/dist/src/mcp/tools/fill-form.d.ts +56 -0
- package/dist/src/mcp/tools/fill-form.js +47 -0
- package/dist/src/mcp/tools/fill-form.js.map +1 -0
- package/dist/src/mcp/tools/fill-workflow.d.ts +7 -0
- package/dist/src/mcp/tools/fill-workflow.js +24 -0
- package/dist/src/mcp/tools/fill-workflow.js.map +1 -0
- package/dist/src/mcp/tools/glossary.d.ts +8 -0
- package/dist/src/mcp/tools/glossary.js +54 -0
- package/dist/src/mcp/tools/glossary.js.map +1 -0
- package/dist/src/mcp/tools/list-forms.d.ts +6 -0
- package/dist/src/mcp/tools/list-forms.js +15 -0
- package/dist/src/mcp/tools/list-forms.js.map +1 -0
- package/dist/src/mcp/tools/list-workflows.d.ts +6 -0
- package/dist/src/mcp/tools/list-workflows.js +16 -0
- package/dist/src/mcp/tools/list-workflows.js.map +1 -0
- package/dist/src/mcp/tools/plan-life-event.d.ts +59 -0
- package/dist/src/mcp/tools/plan-life-event.js +41 -0
- package/dist/src/mcp/tools/plan-life-event.js.map +1 -0
- package/dist/src/mcp/tools/review-form.d.ts +40 -0
- package/dist/src/mcp/tools/review-form.js +26 -0
- package/dist/src/mcp/tools/review-form.js.map +1 -0
- package/dist/src/mcp/tools/review-workflow.d.ts +7 -0
- package/dist/src/mcp/tools/review-workflow.js +27 -0
- package/dist/src/mcp/tools/review-workflow.js.map +1 -0
- package/dist/src/mcp/tools/screen-eligibility.d.ts +53 -0
- package/dist/src/mcp/tools/screen-eligibility.js +44 -0
- package/dist/src/mcp/tools/screen-eligibility.js.map +1 -0
- package/dist/src/mcp/tools/start-workflow.d.ts +6 -0
- package/dist/src/mcp/tools/start-workflow.js +17 -0
- package/dist/src/mcp/tools/start-workflow.js.map +1 -0
- package/dist/src/mcp/tools/validate-form.d.ts +48 -0
- package/dist/src/mcp/tools/validate-form.js +30 -0
- package/dist/src/mcp/tools/validate-form.js.map +1 -0
- package/dist/src/mcp/tools/validate-workflow.d.ts +7 -0
- package/dist/src/mcp/tools/validate-workflow.js +24 -0
- package/dist/src/mcp/tools/validate-workflow.js.map +1 -0
- package/dist/src/pdf/batch.d.ts +8 -0
- package/dist/src/pdf/batch.js +15 -0
- package/dist/src/pdf/batch.js.map +1 -0
- package/dist/src/pdf/merge.d.ts +10 -0
- package/dist/src/pdf/merge.js +62 -0
- package/dist/src/pdf/merge.js.map +1 -0
- package/dist/src/pdf/ocr.d.ts +22 -0
- package/dist/src/pdf/ocr.js +52 -0
- package/dist/src/pdf/ocr.js.map +1 -0
- package/dist/src/pdf/reader.d.ts +2 -0
- package/dist/src/pdf/reader.js +354 -0
- package/dist/src/pdf/reader.js.map +1 -0
- package/dist/src/pdf/shared.d.ts +89 -0
- package/dist/src/pdf/shared.js +137 -0
- package/dist/src/pdf/shared.js.map +1 -0
- package/dist/src/pdf/writer.d.ts +7 -0
- package/dist/src/pdf/writer.js +311 -0
- package/dist/src/pdf/writer.js.map +1 -0
- package/dist/src/plugins/loader.d.ts +15 -0
- package/dist/src/plugins/loader.js +86 -0
- package/dist/src/plugins/loader.js.map +1 -0
- package/dist/src/plugins/sandbox.d.ts +14 -0
- package/dist/src/plugins/sandbox.js +88 -0
- package/dist/src/plugins/sandbox.js.map +1 -0
- package/dist/src/plugins/types.d.ts +17 -0
- package/dist/src/plugins/types.js +13 -0
- package/dist/src/plugins/types.js.map +1 -0
- package/dist/src/schemas/2025/f1040.d.ts +127 -0
- package/dist/src/schemas/2025/f1040.js +104 -0
- package/dist/src/schemas/2025/f1040.js.map +1 -0
- package/dist/src/schemas/2025/f1099-int.d.ts +53 -0
- package/dist/src/schemas/2025/f1099-int.js +51 -0
- package/dist/src/schemas/2025/f1099-int.js.map +1 -0
- package/dist/src/schemas/2025/f1099-nec.d.ts +47 -0
- package/dist/src/schemas/2025/f1099-nec.js +45 -0
- package/dist/src/schemas/2025/f1099-nec.js.map +1 -0
- package/dist/src/schemas/2025/form-8949.d.ts +46 -0
- package/dist/src/schemas/2025/form-8949.js +71 -0
- package/dist/src/schemas/2025/form-8949.js.map +1 -0
- package/dist/src/schemas/2025/index.d.ts +1004 -0
- package/dist/src/schemas/2025/index.js +60 -0
- package/dist/src/schemas/2025/index.js.map +1 -0
- package/dist/src/schemas/2025/schedule-1.d.ts +44 -0
- package/dist/src/schemas/2025/schedule-1.js +56 -0
- package/dist/src/schemas/2025/schedule-1.js.map +1 -0
- package/dist/src/schemas/2025/schedule-b.d.ts +32 -0
- package/dist/src/schemas/2025/schedule-b.js +53 -0
- package/dist/src/schemas/2025/schedule-b.js.map +1 -0
- package/dist/src/schemas/2025/schedule-c.d.ts +58 -0
- package/dist/src/schemas/2025/schedule-c.js +46 -0
- package/dist/src/schemas/2025/schedule-c.js.map +1 -0
- package/dist/src/schemas/2025/schedule-d.d.ts +36 -0
- package/dist/src/schemas/2025/schedule-d.js +52 -0
- package/dist/src/schemas/2025/schedule-d.js.map +1 -0
- package/dist/src/schemas/2025/shared.d.ts +98 -0
- package/dist/src/schemas/2025/shared.js +161 -0
- package/dist/src/schemas/2025/shared.js.map +1 -0
- package/dist/src/schemas/2025/w2.d.ts +57 -0
- package/dist/src/schemas/2025/w2.js +62 -0
- package/dist/src/schemas/2025/w2.js.map +1 -0
- package/dist/src/storage/drafts.d.ts +25 -0
- package/dist/src/storage/drafts.js +117 -0
- package/dist/src/storage/drafts.js.map +1 -0
- package/dist/src/storage/paths.d.ts +15 -0
- package/dist/src/storage/paths.js +33 -0
- package/dist/src/storage/paths.js.map +1 -0
- package/dist/src/storage/profile.d.ts +17 -0
- package/dist/src/storage/profile.js +59 -0
- package/dist/src/storage/profile.js.map +1 -0
- package/dist/src/storage/vault.d.ts +38 -0
- package/dist/src/storage/vault.js +187 -0
- package/dist/src/storage/vault.js.map +1 -0
- package/dist/src/testing/synthetic.d.ts +13 -0
- package/dist/src/testing/synthetic.js +164 -0
- package/dist/src/testing/synthetic.js.map +1 -0
- package/dist/src/types.d.ts +362 -0
- package/dist/src/types.js +2 -0
- package/dist/src/types.js.map +1 -0
- package/dist/src/workflows/deadlines.d.ts +13 -0
- package/dist/src/workflows/deadlines.js +88 -0
- package/dist/src/workflows/deadlines.js.map +1 -0
- package/dist/src/workflows/domains/benefits.d.ts +409 -0
- package/dist/src/workflows/domains/benefits.js +744 -0
- package/dist/src/workflows/domains/benefits.js.map +1 -0
- package/dist/src/workflows/domains/business.d.ts +65 -0
- package/dist/src/workflows/domains/business.js +76 -0
- package/dist/src/workflows/domains/business.js.map +1 -0
- package/dist/src/workflows/domains/education.d.ts +251 -0
- package/dist/src/workflows/domains/education.js +504 -0
- package/dist/src/workflows/domains/education.js.map +1 -0
- package/dist/src/workflows/domains/estate.d.ts +217 -0
- package/dist/src/workflows/domains/estate.js +407 -0
- package/dist/src/workflows/domains/estate.js.map +1 -0
- package/dist/src/workflows/domains/healthcare-ext.d.ts +88 -0
- package/dist/src/workflows/domains/healthcare-ext.js +150 -0
- package/dist/src/workflows/domains/healthcare-ext.js.map +1 -0
- package/dist/src/workflows/domains/healthcare.d.ts +92 -0
- package/dist/src/workflows/domains/healthcare.js +126 -0
- package/dist/src/workflows/domains/healthcare.js.map +1 -0
- package/dist/src/workflows/domains/identity-domain.d.ts +291 -0
- package/dist/src/workflows/domains/identity-domain.js +541 -0
- package/dist/src/workflows/domains/identity-domain.js.map +1 -0
- package/dist/src/workflows/domains/immigration-ext.d.ts +282 -0
- package/dist/src/workflows/domains/immigration-ext.js +590 -0
- package/dist/src/workflows/domains/immigration-ext.js.map +1 -0
- package/dist/src/workflows/domains/immigration.d.ts +121 -0
- package/dist/src/workflows/domains/immigration.js +152 -0
- package/dist/src/workflows/domains/immigration.js.map +1 -0
- package/dist/src/workflows/domains/legal.d.ts +235 -0
- package/dist/src/workflows/domains/legal.js +380 -0
- package/dist/src/workflows/domains/legal.js.map +1 -0
- package/dist/src/workflows/domains/permits.d.ts +65 -0
- package/dist/src/workflows/domains/permits.js +74 -0
- package/dist/src/workflows/domains/permits.js.map +1 -0
- package/dist/src/workflows/domains/retirement.d.ts +59 -0
- package/dist/src/workflows/domains/retirement.js +155 -0
- package/dist/src/workflows/domains/retirement.js.map +1 -0
- package/dist/src/workflows/domains/tax.d.ts +133 -0
- package/dist/src/workflows/domains/tax.js +187 -0
- package/dist/src/workflows/domains/tax.js.map +1 -0
- package/dist/src/workflows/domains/unemployment.d.ts +105 -0
- package/dist/src/workflows/domains/unemployment.js +122 -0
- package/dist/src/workflows/domains/unemployment.js.map +1 -0
- package/dist/src/workflows/domains/veterans.d.ts +189 -0
- package/dist/src/workflows/domains/veterans.js +380 -0
- package/dist/src/workflows/domains/veterans.js.map +1 -0
- package/dist/src/workflows/fees.d.ts +15 -0
- package/dist/src/workflows/fees.js +93 -0
- package/dist/src/workflows/fees.js.map +1 -0
- package/dist/src/workflows/helpers.d.ts +8 -0
- package/dist/src/workflows/helpers.js +66 -0
- package/dist/src/workflows/helpers.js.map +1 -0
- package/dist/src/workflows/io.d.ts +3 -0
- package/dist/src/workflows/io.js +38 -0
- package/dist/src/workflows/io.js.map +1 -0
- package/dist/src/workflows/registry.d.ts +51355 -0
- package/dist/src/workflows/registry.js +140 -0
- package/dist/src/workflows/registry.js.map +1 -0
- package/dist/src/workflows/schemas/benefits.d.ts +78 -0
- package/dist/src/workflows/schemas/benefits.js +91 -0
- package/dist/src/workflows/schemas/benefits.js.map +1 -0
- package/dist/src/workflows/schemas/common.d.ts +27 -0
- package/dist/src/workflows/schemas/common.js +28 -0
- package/dist/src/workflows/schemas/common.js.map +1 -0
- package/dist/src/workflows/schemas/education.d.ts +70 -0
- package/dist/src/workflows/schemas/education.js +85 -0
- package/dist/src/workflows/schemas/education.js.map +1 -0
- package/dist/src/workflows/schemas/estate.d.ts +64 -0
- package/dist/src/workflows/schemas/estate.js +85 -0
- package/dist/src/workflows/schemas/estate.js.map +1 -0
- package/dist/src/workflows/schemas/healthcare-ext.d.ts +15 -0
- package/dist/src/workflows/schemas/healthcare-ext.js +17 -0
- package/dist/src/workflows/schemas/healthcare-ext.js.map +1 -0
- package/dist/src/workflows/schemas/healthcare.d.ts +33 -0
- package/dist/src/workflows/schemas/healthcare.js +20 -0
- package/dist/src/workflows/schemas/healthcare.js.map +1 -0
- package/dist/src/workflows/schemas/identity-domain.d.ts +64 -0
- package/dist/src/workflows/schemas/identity-domain.js +87 -0
- package/dist/src/workflows/schemas/identity-domain.js.map +1 -0
- package/dist/src/workflows/schemas/immigration-ext.d.ts +63 -0
- package/dist/src/workflows/schemas/immigration-ext.js +58 -0
- package/dist/src/workflows/schemas/immigration-ext.js.map +1 -0
- package/dist/src/workflows/schemas/immigration.d.ts +48 -0
- package/dist/src/workflows/schemas/immigration.js +27 -0
- package/dist/src/workflows/schemas/immigration.js.map +1 -0
- package/dist/src/workflows/schemas/legal.d.ts +45 -0
- package/dist/src/workflows/schemas/legal.js +45 -0
- package/dist/src/workflows/schemas/legal.js.map +1 -0
- package/dist/src/workflows/schemas/planning.d.ts +23 -0
- package/dist/src/workflows/schemas/planning.js +15 -0
- package/dist/src/workflows/schemas/planning.js.map +1 -0
- package/dist/src/workflows/schemas/retirement.d.ts +14 -0
- package/dist/src/workflows/schemas/retirement.js +22 -0
- package/dist/src/workflows/schemas/retirement.js.map +1 -0
- package/dist/src/workflows/schemas/tax.d.ts +58 -0
- package/dist/src/workflows/schemas/tax.js +51 -0
- package/dist/src/workflows/schemas/tax.js.map +1 -0
- package/dist/src/workflows/schemas/unemployment.d.ts +33 -0
- package/dist/src/workflows/schemas/unemployment.js +19 -0
- package/dist/src/workflows/schemas/unemployment.js.map +1 -0
- package/dist/src/workflows/schemas/veterans.d.ts +47 -0
- package/dist/src/workflows/schemas/veterans.js +51 -0
- package/dist/src/workflows/schemas/veterans.js.map +1 -0
- package/dist/src/workflows/types.d.ts +9 -0
- package/dist/src/workflows/types.js +2 -0
- package/dist/src/workflows/types.js.map +1 -0
- package/dist/xmcp.config.d.ts +3 -0
- package/dist/xmcp.config.js +29 -0
- package/dist/xmcp.config.js.map +1 -0
- package/llms.txt +48 -0
- package/package.json +82 -0
|
@@ -0,0 +1,132 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Estimate audit risk based on publicly available IRS audit rate data and known patterns.
|
|
3
|
+
*
|
|
4
|
+
* THIS IS NOT LEGAL ADVICE. This is a statistical estimate based on published IRS data.
|
|
5
|
+
* Actual audit selection involves many factors not captured here.
|
|
6
|
+
*/
|
|
7
|
+
export function assessAuditRisk(profile) {
|
|
8
|
+
const factors = [];
|
|
9
|
+
let baseScore = 1; // Everyone starts at 1
|
|
10
|
+
// Factor 1: Income level (IRS audits high earners disproportionately)
|
|
11
|
+
if (profile.agi > 1_000_000) {
|
|
12
|
+
factors.push({
|
|
13
|
+
id: "high_income",
|
|
14
|
+
label: "High income",
|
|
15
|
+
contribution: 2,
|
|
16
|
+
description: "AGI over $1M has historically higher audit rates (~2-4% vs ~0.4% average)",
|
|
17
|
+
severity: "concern",
|
|
18
|
+
});
|
|
19
|
+
baseScore += 2;
|
|
20
|
+
}
|
|
21
|
+
else if (profile.agi > 500_000) {
|
|
22
|
+
factors.push({
|
|
23
|
+
id: "elevated_income",
|
|
24
|
+
label: "Elevated income",
|
|
25
|
+
contribution: 1,
|
|
26
|
+
description: "AGI $500K-$1M has moderately elevated audit rates",
|
|
27
|
+
severity: "warning",
|
|
28
|
+
});
|
|
29
|
+
baseScore += 1;
|
|
30
|
+
}
|
|
31
|
+
// Factor 2: Schedule C with high expenses
|
|
32
|
+
if (profile.scheduleCNet !== undefined && profile.scheduleCExpenses !== undefined) {
|
|
33
|
+
const expenseRatio = profile.scheduleCNet !== 0
|
|
34
|
+
? profile.scheduleCExpenses / Math.abs(profile.scheduleCNet + profile.scheduleCExpenses)
|
|
35
|
+
: 0;
|
|
36
|
+
if (profile.scheduleCNet < 0) {
|
|
37
|
+
factors.push({
|
|
38
|
+
id: "schedule_c_loss",
|
|
39
|
+
label: "Schedule C loss",
|
|
40
|
+
contribution: 2,
|
|
41
|
+
description: "Reporting a business loss increases scrutiny, especially if recurring",
|
|
42
|
+
severity: "concern",
|
|
43
|
+
});
|
|
44
|
+
baseScore += 2;
|
|
45
|
+
}
|
|
46
|
+
else if (expenseRatio > 0.75) {
|
|
47
|
+
factors.push({
|
|
48
|
+
id: "high_expense_ratio",
|
|
49
|
+
label: "High expense-to-income ratio",
|
|
50
|
+
contribution: 1,
|
|
51
|
+
description: `Schedule C expenses are ${(expenseRatio * 100).toFixed(0)}% of gross receipts`,
|
|
52
|
+
severity: "warning",
|
|
53
|
+
});
|
|
54
|
+
baseScore += 1;
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
// Factor 3: Home office deduction
|
|
58
|
+
if (profile.hasHomeOffice && profile.scheduleCNet !== undefined) {
|
|
59
|
+
factors.push({
|
|
60
|
+
id: "home_office",
|
|
61
|
+
label: "Home office deduction",
|
|
62
|
+
contribution: 1,
|
|
63
|
+
description: "Home office deductions receive additional scrutiny — ensure exclusive and regular use",
|
|
64
|
+
severity: "warning",
|
|
65
|
+
});
|
|
66
|
+
baseScore += 1;
|
|
67
|
+
}
|
|
68
|
+
// Factor 4: Large charitable deductions relative to income
|
|
69
|
+
if (profile.charitableDeductions !== undefined && profile.agi > 0) {
|
|
70
|
+
const charitableRatio = profile.charitableDeductions / profile.agi;
|
|
71
|
+
if (charitableRatio > 0.3) {
|
|
72
|
+
factors.push({
|
|
73
|
+
id: "high_charitable",
|
|
74
|
+
label: "Large charitable deductions",
|
|
75
|
+
contribution: 2,
|
|
76
|
+
description: `Charitable deductions are ${(charitableRatio * 100).toFixed(0)}% of AGI (above typical levels)`,
|
|
77
|
+
severity: "concern",
|
|
78
|
+
});
|
|
79
|
+
baseScore += 2;
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
// Factor 5: Crypto activity
|
|
83
|
+
if (profile.hasCryptoActivity) {
|
|
84
|
+
factors.push({
|
|
85
|
+
id: "crypto_activity",
|
|
86
|
+
label: "Cryptocurrency activity",
|
|
87
|
+
contribution: 1,
|
|
88
|
+
description: "IRS has increased focus on crypto reporting compliance",
|
|
89
|
+
severity: "info",
|
|
90
|
+
});
|
|
91
|
+
baseScore += 1;
|
|
92
|
+
}
|
|
93
|
+
// Factor 6: Very low income with high deductions
|
|
94
|
+
if (profile.agi > 0 && profile.agi < 25_000 && profile.useItemized && profile.totalItemized !== undefined) {
|
|
95
|
+
if (profile.totalItemized > profile.agi * 0.5) {
|
|
96
|
+
factors.push({
|
|
97
|
+
id: "low_income_high_deductions",
|
|
98
|
+
label: "Low income with high itemized deductions",
|
|
99
|
+
contribution: 1,
|
|
100
|
+
description: "Itemized deductions exceeding 50% of AGI on low income may draw attention",
|
|
101
|
+
severity: "warning",
|
|
102
|
+
});
|
|
103
|
+
baseScore += 1;
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
// Factor 7: EITC (IRS audits EITC returns at elevated rates — unfortunately)
|
|
107
|
+
if (profile.agi > 0 && profile.agi < 60_000 && profile.filingStatus !== "married_filing_separately") {
|
|
108
|
+
// We note this but don't add to score — the elevated rate is an IRS enforcement choice,
|
|
109
|
+
// not an indicator the taxpayer did anything wrong.
|
|
110
|
+
factors.push({
|
|
111
|
+
id: "eitc_eligible_range",
|
|
112
|
+
label: "EITC income range",
|
|
113
|
+
contribution: 0,
|
|
114
|
+
description: "EITC-eligible returns are audited at higher-than-average rates (IRS enforcement priority)",
|
|
115
|
+
severity: "info",
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
// Cap at 10
|
|
119
|
+
const score = Math.min(10, Math.max(1, baseScore));
|
|
120
|
+
const riskLevel = score <= 2 ? "low" :
|
|
121
|
+
score <= 4 ? "moderate" :
|
|
122
|
+
score <= 6 ? "elevated" : "high";
|
|
123
|
+
return {
|
|
124
|
+
score,
|
|
125
|
+
riskLevel,
|
|
126
|
+
factors,
|
|
127
|
+
disclaimer: "This is a statistical estimate based on published IRS audit rate data, not legal or tax advice. " +
|
|
128
|
+
"Actual audit selection involves many factors not captured here. Always report income accurately " +
|
|
129
|
+
"and keep documentation for all deductions.",
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
//# sourceMappingURL=scorer.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"scorer.js","sourceRoot":"","sources":["../../../../src/engine/audit-risk/scorer.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import type { ParsedCryptoLot, CryptoDisposal, RawCryptoTransaction } from "./transaction-parser.js";
|
|
2
|
+
export type CostBasisMethod = "fifo" | "lifo" | "specific-id";
|
|
3
|
+
export declare function calculateDisposals(lots: ParsedCryptoLot[], sales: RawCryptoTransaction[], method?: CostBasisMethod): CryptoDisposal[];
|
|
4
|
+
export interface CryptoTaxSummary {
|
|
5
|
+
totalProceeds: number;
|
|
6
|
+
totalCostBasis: number;
|
|
7
|
+
totalGainOrLoss: number;
|
|
8
|
+
shortTermGain: number;
|
|
9
|
+
shortTermLoss: number;
|
|
10
|
+
longTermGain: number;
|
|
11
|
+
longTermLoss: number;
|
|
12
|
+
netShortTerm: number;
|
|
13
|
+
netLongTerm: number;
|
|
14
|
+
disposalCount: number;
|
|
15
|
+
stakingIncome: number;
|
|
16
|
+
miningIncome: number;
|
|
17
|
+
airdropIncome: number;
|
|
18
|
+
}
|
|
19
|
+
export declare function summarizeCryptoTax(disposals: CryptoDisposal[], transactions: RawCryptoTransaction[]): CryptoTaxSummary;
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
const ONE_YEAR_MS = 365.25 * 24 * 60 * 60 * 1000;
|
|
2
|
+
export function calculateDisposals(lots, sales, method = "fifo") {
|
|
3
|
+
// Clone lots so we can mutate remaining amounts
|
|
4
|
+
const availableLots = lots.map((lot) => ({
|
|
5
|
+
...lot,
|
|
6
|
+
remaining: lot.amount,
|
|
7
|
+
costBasisPerUnit: lot.amount > 0 ? lot.costBasis / lot.amount : 0,
|
|
8
|
+
}));
|
|
9
|
+
const disposals = [];
|
|
10
|
+
let disposalCounter = 0;
|
|
11
|
+
for (const sale of sales.filter((tx) => tx.type === "sell" || tx.type === "trade")) {
|
|
12
|
+
let amountToSell = sale.amount;
|
|
13
|
+
const proceedsPerUnit = sale.amount > 0 ? sale.totalValue / sale.amount : 0;
|
|
14
|
+
const saleDate = new Date(sale.date);
|
|
15
|
+
const lotAllocations = [];
|
|
16
|
+
let totalCostBasis = 0;
|
|
17
|
+
// Filter lots for the same asset
|
|
18
|
+
const assetLots = availableLots.filter((lot) => lot.asset === sale.asset && lot.remaining > 0);
|
|
19
|
+
// Sort by method
|
|
20
|
+
if (method === "fifo") {
|
|
21
|
+
assetLots.sort((a, b) => new Date(a.dateAcquired).getTime() - new Date(b.dateAcquired).getTime());
|
|
22
|
+
}
|
|
23
|
+
else if (method === "lifo") {
|
|
24
|
+
assetLots.sort((a, b) => new Date(b.dateAcquired).getTime() - new Date(a.dateAcquired).getTime());
|
|
25
|
+
}
|
|
26
|
+
// specific-id would require explicit lot matching — not implemented in auto mode
|
|
27
|
+
for (const lot of assetLots) {
|
|
28
|
+
if (amountToSell <= 0)
|
|
29
|
+
break;
|
|
30
|
+
const amountFromLot = Math.min(lot.remaining, amountToSell);
|
|
31
|
+
const costBasisFromLot = amountFromLot * lot.costBasisPerUnit;
|
|
32
|
+
lotAllocations.push({
|
|
33
|
+
lotId: lot.id,
|
|
34
|
+
amountUsed: amountFromLot,
|
|
35
|
+
costBasis: roundCurrency(costBasisFromLot),
|
|
36
|
+
});
|
|
37
|
+
totalCostBasis += costBasisFromLot;
|
|
38
|
+
lot.remaining -= amountFromLot;
|
|
39
|
+
amountToSell -= amountFromLot;
|
|
40
|
+
}
|
|
41
|
+
if (amountToSell > 0.0001) {
|
|
42
|
+
// Sold more than owned — flag this but still create the disposal
|
|
43
|
+
totalCostBasis = 0; // Unknown cost basis for excess
|
|
44
|
+
}
|
|
45
|
+
const proceeds = sale.totalValue - (sale.fee ?? 0);
|
|
46
|
+
const gainOrLoss = proceeds - totalCostBasis;
|
|
47
|
+
// Determine holding period from the earliest lot used
|
|
48
|
+
const earliestLotDate = lotAllocations.length > 0
|
|
49
|
+
? new Date(availableLots.find((l) => l.id === lotAllocations[0].lotId)?.dateAcquired ?? sale.date)
|
|
50
|
+
: saleDate;
|
|
51
|
+
const holdingPeriodMs = saleDate.getTime() - earliestLotDate.getTime();
|
|
52
|
+
const holdingPeriod = holdingPeriodMs > ONE_YEAR_MS ? "long-term" : "short-term";
|
|
53
|
+
disposalCounter++;
|
|
54
|
+
disposals.push({
|
|
55
|
+
id: `disposal-${disposalCounter}`,
|
|
56
|
+
asset: sale.asset,
|
|
57
|
+
dateSold: sale.date,
|
|
58
|
+
proceeds: roundCurrency(proceeds),
|
|
59
|
+
amount: sale.amount,
|
|
60
|
+
costBasis: roundCurrency(totalCostBasis),
|
|
61
|
+
gainOrLoss: roundCurrency(gainOrLoss),
|
|
62
|
+
holdingPeriod,
|
|
63
|
+
lots: lotAllocations,
|
|
64
|
+
});
|
|
65
|
+
}
|
|
66
|
+
return disposals;
|
|
67
|
+
}
|
|
68
|
+
export function summarizeCryptoTax(disposals, transactions) {
|
|
69
|
+
let shortTermGain = 0;
|
|
70
|
+
let shortTermLoss = 0;
|
|
71
|
+
let longTermGain = 0;
|
|
72
|
+
let longTermLoss = 0;
|
|
73
|
+
for (const d of disposals) {
|
|
74
|
+
if (d.holdingPeriod === "short-term") {
|
|
75
|
+
if (d.gainOrLoss >= 0)
|
|
76
|
+
shortTermGain += d.gainOrLoss;
|
|
77
|
+
else
|
|
78
|
+
shortTermLoss += Math.abs(d.gainOrLoss);
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
if (d.gainOrLoss >= 0)
|
|
82
|
+
longTermGain += d.gainOrLoss;
|
|
83
|
+
else
|
|
84
|
+
longTermLoss += Math.abs(d.gainOrLoss);
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
const stakingIncome = transactions
|
|
88
|
+
.filter((tx) => tx.type === "staking_reward")
|
|
89
|
+
.reduce((sum, tx) => sum + tx.totalValue, 0);
|
|
90
|
+
const miningIncome = transactions
|
|
91
|
+
.filter((tx) => tx.type === "mining")
|
|
92
|
+
.reduce((sum, tx) => sum + tx.totalValue, 0);
|
|
93
|
+
const airdropIncome = transactions
|
|
94
|
+
.filter((tx) => tx.type === "airdrop")
|
|
95
|
+
.reduce((sum, tx) => sum + tx.totalValue, 0);
|
|
96
|
+
return {
|
|
97
|
+
totalProceeds: roundCurrency(disposals.reduce((sum, d) => sum + d.proceeds, 0)),
|
|
98
|
+
totalCostBasis: roundCurrency(disposals.reduce((sum, d) => sum + d.costBasis, 0)),
|
|
99
|
+
totalGainOrLoss: roundCurrency(disposals.reduce((sum, d) => sum + d.gainOrLoss, 0)),
|
|
100
|
+
shortTermGain: roundCurrency(shortTermGain),
|
|
101
|
+
shortTermLoss: roundCurrency(shortTermLoss),
|
|
102
|
+
longTermGain: roundCurrency(longTermGain),
|
|
103
|
+
longTermLoss: roundCurrency(longTermLoss),
|
|
104
|
+
netShortTerm: roundCurrency(shortTermGain - shortTermLoss),
|
|
105
|
+
netLongTerm: roundCurrency(longTermGain - longTermLoss),
|
|
106
|
+
disposalCount: disposals.length,
|
|
107
|
+
stakingIncome: roundCurrency(stakingIncome),
|
|
108
|
+
miningIncome: roundCurrency(miningIncome),
|
|
109
|
+
airdropIncome: roundCurrency(airdropIncome),
|
|
110
|
+
};
|
|
111
|
+
}
|
|
112
|
+
function roundCurrency(value) {
|
|
113
|
+
return Number(value.toFixed(2));
|
|
114
|
+
}
|
|
115
|
+
//# sourceMappingURL=cost-basis.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"cost-basis.js","sourceRoot":"","sources":["../../../../src/engine/crypto/cost-basis.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
export interface RawCryptoTransaction {
|
|
2
|
+
date: string;
|
|
3
|
+
type: "buy" | "sell" | "trade" | "staking_reward" | "mining" | "airdrop" | "transfer";
|
|
4
|
+
asset: string;
|
|
5
|
+
amount: number;
|
|
6
|
+
pricePerUnit: number;
|
|
7
|
+
totalValue: number;
|
|
8
|
+
fee?: number | undefined;
|
|
9
|
+
exchange?: string | undefined;
|
|
10
|
+
}
|
|
11
|
+
export interface ParsedCryptoLot {
|
|
12
|
+
id: string;
|
|
13
|
+
asset: string;
|
|
14
|
+
dateAcquired: string;
|
|
15
|
+
costBasis: number;
|
|
16
|
+
amount: number;
|
|
17
|
+
source: "purchase" | "staking" | "mining" | "airdrop" | "trade";
|
|
18
|
+
exchange?: string | undefined;
|
|
19
|
+
}
|
|
20
|
+
export interface CryptoDisposal {
|
|
21
|
+
id: string;
|
|
22
|
+
asset: string;
|
|
23
|
+
dateSold: string;
|
|
24
|
+
proceeds: number;
|
|
25
|
+
amount: number;
|
|
26
|
+
costBasis: number;
|
|
27
|
+
gainOrLoss: number;
|
|
28
|
+
holdingPeriod: "short-term" | "long-term";
|
|
29
|
+
lots: Array<{
|
|
30
|
+
lotId: string;
|
|
31
|
+
amountUsed: number;
|
|
32
|
+
costBasis: number;
|
|
33
|
+
}>;
|
|
34
|
+
}
|
|
35
|
+
export declare function parseCsvTransactions(csv: string, exchange: string): RawCryptoTransaction[];
|
|
36
|
+
export declare function buildLotsFromTransactions(transactions: RawCryptoTransaction[]): ParsedCryptoLot[];
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
export function parseCsvTransactions(csv, exchange) {
|
|
2
|
+
const lines = csv.trim().split("\n");
|
|
3
|
+
if (lines.length < 2)
|
|
4
|
+
return [];
|
|
5
|
+
const headers = lines[0].toLowerCase().split(",").map((h) => h.trim());
|
|
6
|
+
const transactions = [];
|
|
7
|
+
for (let i = 1; i < lines.length; i++) {
|
|
8
|
+
const values = lines[i].split(",").map((v) => v.trim().replace(/^"|"$/g, ""));
|
|
9
|
+
const row = {};
|
|
10
|
+
for (let j = 0; j < headers.length; j++) {
|
|
11
|
+
row[headers[j]] = values[j] ?? "";
|
|
12
|
+
}
|
|
13
|
+
const parsed = mapExchangeRow(row, exchange);
|
|
14
|
+
if (parsed) {
|
|
15
|
+
transactions.push({ ...parsed, exchange });
|
|
16
|
+
}
|
|
17
|
+
}
|
|
18
|
+
return transactions;
|
|
19
|
+
}
|
|
20
|
+
function mapExchangeRow(row, exchange) {
|
|
21
|
+
switch (exchange.toLowerCase()) {
|
|
22
|
+
case "coinbase": return mapCoinbaseRow(row);
|
|
23
|
+
case "kraken": return mapKrakenRow(row);
|
|
24
|
+
case "binance": return mapBinanceRow(row);
|
|
25
|
+
default: return mapGenericRow(row);
|
|
26
|
+
}
|
|
27
|
+
}
|
|
28
|
+
function mapCoinbaseRow(row) {
|
|
29
|
+
const type = mapTransactionType(row["type"] ?? row["transaction type"] ?? "");
|
|
30
|
+
if (!type)
|
|
31
|
+
return null;
|
|
32
|
+
return {
|
|
33
|
+
date: row["timestamp"] ?? row["date"] ?? "",
|
|
34
|
+
type,
|
|
35
|
+
asset: row["asset"] ?? row["currency"] ?? "",
|
|
36
|
+
amount: parseFloat(row["quantity"] ?? row["amount"] ?? "0"),
|
|
37
|
+
pricePerUnit: parseFloat(row["spot price at transaction"] ?? row["price"] ?? "0"),
|
|
38
|
+
totalValue: parseFloat(row["total (inclusive of fees and/or spread)"] ?? row["total"] ?? "0"),
|
|
39
|
+
fee: parseFloat(row["fees and/or spread"] ?? row["fee"] ?? "0") || undefined,
|
|
40
|
+
};
|
|
41
|
+
}
|
|
42
|
+
function mapKrakenRow(row) {
|
|
43
|
+
const type = mapTransactionType(row["type"] ?? "");
|
|
44
|
+
if (!type)
|
|
45
|
+
return null;
|
|
46
|
+
return {
|
|
47
|
+
date: row["time"] ?? row["date"] ?? "",
|
|
48
|
+
type,
|
|
49
|
+
asset: row["pair"] ?? row["asset"] ?? "",
|
|
50
|
+
amount: Math.abs(parseFloat(row["vol"] ?? row["amount"] ?? "0")),
|
|
51
|
+
pricePerUnit: parseFloat(row["price"] ?? "0"),
|
|
52
|
+
totalValue: Math.abs(parseFloat(row["cost"] ?? row["total"] ?? "0")),
|
|
53
|
+
fee: parseFloat(row["fee"] ?? "0") || undefined,
|
|
54
|
+
};
|
|
55
|
+
}
|
|
56
|
+
function mapBinanceRow(row) {
|
|
57
|
+
const type = mapTransactionType(row["operation"] ?? row["type"] ?? "");
|
|
58
|
+
if (!type)
|
|
59
|
+
return null;
|
|
60
|
+
return {
|
|
61
|
+
date: row["utc_time"] ?? row["date"] ?? "",
|
|
62
|
+
type,
|
|
63
|
+
asset: row["coin"] ?? row["asset"] ?? "",
|
|
64
|
+
amount: parseFloat(row["change"] ?? row["amount"] ?? "0"),
|
|
65
|
+
pricePerUnit: 0, // Binance CSV often doesn't include price — caller must enrich
|
|
66
|
+
totalValue: parseFloat(row["total"] ?? "0"),
|
|
67
|
+
};
|
|
68
|
+
}
|
|
69
|
+
function mapGenericRow(row) {
|
|
70
|
+
const type = mapTransactionType(row["type"] ?? row["transaction_type"] ?? "");
|
|
71
|
+
if (!type)
|
|
72
|
+
return null;
|
|
73
|
+
return {
|
|
74
|
+
date: row["date"] ?? row["timestamp"] ?? "",
|
|
75
|
+
type,
|
|
76
|
+
asset: row["asset"] ?? row["currency"] ?? row["coin"] ?? "",
|
|
77
|
+
amount: parseFloat(row["amount"] ?? row["quantity"] ?? "0"),
|
|
78
|
+
pricePerUnit: parseFloat(row["price"] ?? row["price_per_unit"] ?? "0"),
|
|
79
|
+
totalValue: parseFloat(row["total"] ?? row["total_value"] ?? "0"),
|
|
80
|
+
fee: parseFloat(row["fee"] ?? "0") || undefined,
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
function mapTransactionType(raw) {
|
|
84
|
+
const normalized = raw.toLowerCase().trim();
|
|
85
|
+
if (["buy", "purchase"].includes(normalized))
|
|
86
|
+
return "buy";
|
|
87
|
+
if (["sell"].includes(normalized))
|
|
88
|
+
return "sell";
|
|
89
|
+
if (["trade", "swap", "convert"].includes(normalized))
|
|
90
|
+
return "trade";
|
|
91
|
+
if (["staking_reward", "staking", "reward", "earn"].includes(normalized))
|
|
92
|
+
return "staking_reward";
|
|
93
|
+
if (["mining", "mined"].includes(normalized))
|
|
94
|
+
return "mining";
|
|
95
|
+
if (["airdrop"].includes(normalized))
|
|
96
|
+
return "airdrop";
|
|
97
|
+
if (["send", "receive", "transfer", "deposit", "withdrawal"].includes(normalized))
|
|
98
|
+
return "transfer";
|
|
99
|
+
return null;
|
|
100
|
+
}
|
|
101
|
+
export function buildLotsFromTransactions(transactions) {
|
|
102
|
+
const lots = [];
|
|
103
|
+
let lotCounter = 0;
|
|
104
|
+
for (const tx of transactions) {
|
|
105
|
+
if (tx.type === "buy" || tx.type === "trade" || tx.type === "staking_reward" || tx.type === "mining" || tx.type === "airdrop") {
|
|
106
|
+
lotCounter++;
|
|
107
|
+
const source = tx.type === "staking_reward" ? "staking" :
|
|
108
|
+
tx.type === "mining" ? "mining" :
|
|
109
|
+
tx.type === "airdrop" ? "airdrop" :
|
|
110
|
+
tx.type === "trade" ? "trade" : "purchase";
|
|
111
|
+
lots.push({
|
|
112
|
+
id: `lot-${lotCounter}`,
|
|
113
|
+
asset: tx.asset,
|
|
114
|
+
dateAcquired: tx.date,
|
|
115
|
+
costBasis: tx.totalValue + (tx.fee ?? 0),
|
|
116
|
+
amount: tx.amount,
|
|
117
|
+
source,
|
|
118
|
+
exchange: tx.exchange,
|
|
119
|
+
});
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
return lots;
|
|
123
|
+
}
|
|
124
|
+
//# sourceMappingURL=transaction-parser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transaction-parser.js","sourceRoot":"","sources":["../../../../src/engine/crypto/transaction-parser.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import type { CryptoDisposal, ParsedCryptoLot } from "./transaction-parser.js";
|
|
2
|
+
export interface WashSaleResult {
|
|
3
|
+
disallowedLoss: number;
|
|
4
|
+
adjustedDisposals: CryptoDisposal[];
|
|
5
|
+
washSaleEvents: WashSaleEvent[];
|
|
6
|
+
}
|
|
7
|
+
export interface WashSaleEvent {
|
|
8
|
+
disposalId: string;
|
|
9
|
+
asset: string;
|
|
10
|
+
originalLoss: number;
|
|
11
|
+
disallowedAmount: number;
|
|
12
|
+
replacementLotId: string;
|
|
13
|
+
reason: string;
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Detect wash sales in crypto disposals.
|
|
17
|
+
*
|
|
18
|
+
* The IRS hasn't definitively required wash sale rules for crypto (as of 2025),
|
|
19
|
+
* but applying them is the conservative approach and aligns with pending legislation.
|
|
20
|
+
*
|
|
21
|
+
* Wash sale: selling at a loss and buying substantially identical asset within
|
|
22
|
+
* 30 days before or after the sale. The loss is disallowed and added to the
|
|
23
|
+
* replacement lot's cost basis.
|
|
24
|
+
*/
|
|
25
|
+
export declare function detectWashSales(disposals: CryptoDisposal[], lots: ParsedCryptoLot[]): WashSaleResult;
|
|
@@ -0,0 +1,51 @@
|
|
|
1
|
+
const WASH_SALE_WINDOW_MS = 30 * 24 * 60 * 60 * 1000; // 30 days
|
|
2
|
+
/**
|
|
3
|
+
* Detect wash sales in crypto disposals.
|
|
4
|
+
*
|
|
5
|
+
* The IRS hasn't definitively required wash sale rules for crypto (as of 2025),
|
|
6
|
+
* but applying them is the conservative approach and aligns with pending legislation.
|
|
7
|
+
*
|
|
8
|
+
* Wash sale: selling at a loss and buying substantially identical asset within
|
|
9
|
+
* 30 days before or after the sale. The loss is disallowed and added to the
|
|
10
|
+
* replacement lot's cost basis.
|
|
11
|
+
*/
|
|
12
|
+
export function detectWashSales(disposals, lots) {
|
|
13
|
+
const events = [];
|
|
14
|
+
const adjustedDisposals = disposals.map((d) => ({ ...d }));
|
|
15
|
+
let totalDisallowed = 0;
|
|
16
|
+
for (const disposal of adjustedDisposals) {
|
|
17
|
+
// Only check losses
|
|
18
|
+
if (disposal.gainOrLoss >= 0)
|
|
19
|
+
continue;
|
|
20
|
+
const saleDate = new Date(disposal.dateSold).getTime();
|
|
21
|
+
const windowStart = saleDate - WASH_SALE_WINDOW_MS;
|
|
22
|
+
const windowEnd = saleDate + WASH_SALE_WINDOW_MS;
|
|
23
|
+
// Look for replacement purchases of the same asset within the window
|
|
24
|
+
const replacementLot = lots.find((lot) => {
|
|
25
|
+
if (lot.asset !== disposal.asset)
|
|
26
|
+
return false;
|
|
27
|
+
const acquiredDate = new Date(lot.dateAcquired).getTime();
|
|
28
|
+
return acquiredDate >= windowStart && acquiredDate <= windowEnd && acquiredDate !== saleDate;
|
|
29
|
+
});
|
|
30
|
+
if (replacementLot) {
|
|
31
|
+
const disallowedAmount = Math.abs(disposal.gainOrLoss);
|
|
32
|
+
totalDisallowed += disallowedAmount;
|
|
33
|
+
events.push({
|
|
34
|
+
disposalId: disposal.id,
|
|
35
|
+
asset: disposal.asset,
|
|
36
|
+
originalLoss: disposal.gainOrLoss,
|
|
37
|
+
disallowedAmount,
|
|
38
|
+
replacementLotId: replacementLot.id,
|
|
39
|
+
reason: `Substantially identical asset (${disposal.asset}) purchased within 30 days of sale at a loss`,
|
|
40
|
+
});
|
|
41
|
+
// Adjust the disposal: loss is disallowed
|
|
42
|
+
disposal.gainOrLoss = 0;
|
|
43
|
+
}
|
|
44
|
+
}
|
|
45
|
+
return {
|
|
46
|
+
disallowedLoss: Number(totalDisallowed.toFixed(2)),
|
|
47
|
+
adjustedDisposals,
|
|
48
|
+
washSaleEvents: events,
|
|
49
|
+
};
|
|
50
|
+
}
|
|
51
|
+
//# sourceMappingURL=wash-sale.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"wash-sale.js","sourceRoot":"","sources":["../../../../src/engine/crypto/wash-sale.ts"],"names":[],"mappings":"AAiBA,MAAM,mBAAmB,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC,UAAU;AAEhE;;;;;;;;;GASG;AACH,MAAM,UAAU,eAAe,CAC7B,SAA2B,EAC3B,IAAuB;IAEvB,MAAM,MAAM,GAAoB,EAAE,CAAC;IACnC,MAAM,iBAAiB,GAAG,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,EAAE,CAAC,CAAC,CAAC;IAC3D,IAAI,eAAe,GAAG,CAAC,CAAC;IAExB,KAAK,MAAM,QAAQ,IAAI,iBAAiB,EAAE,CAAC;QACzC,oBAAoB;QACpB,IAAI,QAAQ,CAAC,UAAU,IAAI,CAAC;YAAE,SAAS;QAEvC,MAAM,QAAQ,GAAG,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,OAAO,EAAE,CAAC;QACvD,MAAM,WAAW,GAAG,QAAQ,GAAG,mBAAmB,CAAC;QACnD,MAAM,SAAS,GAAG,QAAQ,GAAG,mBAAmB,CAAC;QAEjD,qEAAqE;QACrE,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,EAAE,EAAE;YACvC,IAAI,GAAG,CAAC,KAAK,KAAK,QAAQ,CAAC,KAAK;gBAAE,OAAO,KAAK,CAAC;YAC/C,MAAM,YAAY,GAAG,IAAI,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,OAAO,EAAE,CAAC;YAC1D,OAAO,YAAY,IAAI,WAAW,IAAI,YAAY,IAAI,SAAS,IAAI,YAAY,KAAK,QAAQ,CAAC;QAC/F,CAAC,CAAC,CAAC;QAEH,IAAI,cAAc,EAAE,CAAC;YACnB,MAAM,gBAAgB,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC;YACvD,eAAe,IAAI,gBAAgB,CAAC;YAEpC,MAAM,CAAC,IAAI,CAAC;gBACV,UAAU,EAAE,QAAQ,CAAC,EAAE;gBACvB,KAAK,EAAE,QAAQ,CAAC,KAAK;gBACrB,YAAY,EAAE,QAAQ,CAAC,UAAU;gBACjC,gBAAgB;gBAChB,gBAAgB,EAAE,cAAc,CAAC,EAAE;gBACnC,MAAM,EAAE,kCAAkC,QAAQ,CAAC,KAAK,8CAA8C;aACvG,CAAC,CAAC;YAEH,0CAA0C;YAC1C,QAAQ,CAAC,UAAU,GAAG,CAAC,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,OAAO;QACL,cAAc,EAAE,MAAM,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAClD,iBAAiB;QACjB,cAAc,EAAE,MAAM;KACvB,CAAC;AACJ,CAAC"}
|
|
@@ -0,0 +1,41 @@
|
|
|
1
|
+
export interface EsppPurchaseResult {
|
|
2
|
+
purchaseDate: string;
|
|
3
|
+
shares: number;
|
|
4
|
+
purchasePrice: number;
|
|
5
|
+
fairMarketValue: number;
|
|
6
|
+
discount: number;
|
|
7
|
+
costBasis: number;
|
|
8
|
+
}
|
|
9
|
+
export interface EsppSaleResult {
|
|
10
|
+
dispositionType: "qualifying" | "disqualifying";
|
|
11
|
+
ordinaryIncome: number;
|
|
12
|
+
capitalGain: number;
|
|
13
|
+
totalGain: number;
|
|
14
|
+
proceeds: number;
|
|
15
|
+
adjustedCostBasis: number;
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Calculate the purchase details of an ESPP transaction.
|
|
19
|
+
*
|
|
20
|
+
* Employees typically buy shares at 85% of the lower of (FMV at offering
|
|
21
|
+
* start, FMV at purchase date). The discount is a future tax liability
|
|
22
|
+
* waiting to happen — the IRS always gets its cut.
|
|
23
|
+
*/
|
|
24
|
+
export declare function calculateEsppPurchase(shares: number, purchasePrice: number, fairMarketValue: number, purchaseDate: string): EsppPurchaseResult;
|
|
25
|
+
/**
|
|
26
|
+
* Calculate the tax consequences of selling ESPP shares.
|
|
27
|
+
*
|
|
28
|
+
* Qualifying disposition (both required):
|
|
29
|
+
* - >2 years from offering date
|
|
30
|
+
* - >1 year from purchase date
|
|
31
|
+
* Ordinary income = lesser of:
|
|
32
|
+
* (a) actual gain (sale price - purchase price) × shares
|
|
33
|
+
* (b) offering discount (offering FMV × discount %) × shares
|
|
34
|
+
* For simplicity, we use (FMV at purchase - purchase price) as the
|
|
35
|
+
* discount baseline when offering-date FMV isn't provided.
|
|
36
|
+
*
|
|
37
|
+
* Disqualifying disposition:
|
|
38
|
+
* Ordinary income = (FMV at purchase - purchase price) × shares
|
|
39
|
+
* Capital gain = total gain - ordinary income
|
|
40
|
+
*/
|
|
41
|
+
export declare function calculateEsppSale(purchase: EsppPurchaseResult, salePrice: number, saleDate: string, offeringDate: string): EsppSaleResult;
|