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.
Files changed (571) hide show
  1. package/AGENTS.md +258 -0
  2. package/LICENSE +21 -0
  3. package/PRIVACY.md +38 -0
  4. package/README.md +234 -0
  5. package/agents.json +72 -0
  6. package/dist/bin/pigeongov.d.ts +2 -0
  7. package/dist/bin/pigeongov.js +4 -0
  8. package/dist/bin/pigeongov.js.map +1 -0
  9. package/dist/src/advisory/decision-support/contribution-optimizer.d.ts +36 -0
  10. package/dist/src/advisory/decision-support/contribution-optimizer.js +88 -0
  11. package/dist/src/advisory/decision-support/contribution-optimizer.js.map +1 -0
  12. package/dist/src/advisory/decision-support/deduction-optimizer.d.ts +31 -0
  13. package/dist/src/advisory/decision-support/deduction-optimizer.js +96 -0
  14. package/dist/src/advisory/decision-support/deduction-optimizer.js.map +1 -0
  15. package/dist/src/advisory/life-events/events.d.ts +4 -0
  16. package/dist/src/advisory/life-events/events.js +156 -0
  17. package/dist/src/advisory/life-events/events.js.map +1 -0
  18. package/dist/src/advisory/life-events/planner.d.ts +19 -0
  19. package/dist/src/advisory/life-events/planner.js +72 -0
  20. package/dist/src/advisory/life-events/planner.js.map +1 -0
  21. package/dist/src/advisory/screener/engine.d.ts +4 -0
  22. package/dist/src/advisory/screener/engine.js +368 -0
  23. package/dist/src/advisory/screener/engine.js.map +1 -0
  24. package/dist/src/advisory/screener/intake.d.ts +68 -0
  25. package/dist/src/advisory/screener/intake.js +93 -0
  26. package/dist/src/advisory/screener/intake.js.map +1 -0
  27. package/dist/src/analytics/stats.d.ts +18 -0
  28. package/dist/src/analytics/stats.js +158 -0
  29. package/dist/src/analytics/stats.js.map +1 -0
  30. package/dist/src/api/handlers/deadlines.d.ts +2 -0
  31. package/dist/src/api/handlers/deadlines.js +7 -0
  32. package/dist/src/api/handlers/deadlines.js.map +1 -0
  33. package/dist/src/api/handlers/openapi.d.ts +2 -0
  34. package/dist/src/api/handlers/openapi.js +7 -0
  35. package/dist/src/api/handlers/openapi.js.map +1 -0
  36. package/dist/src/api/handlers/workflows.d.ts +5 -0
  37. package/dist/src/api/handlers/workflows.js +73 -0
  38. package/dist/src/api/handlers/workflows.js.map +1 -0
  39. package/dist/src/api/index.d.ts +20 -0
  40. package/dist/src/api/index.js +70 -0
  41. package/dist/src/api/index.js.map +1 -0
  42. package/dist/src/api/openapi.d.ts +5 -0
  43. package/dist/src/api/openapi.js +384 -0
  44. package/dist/src/api/openapi.js.map +1 -0
  45. package/dist/src/api/router.d.ts +19 -0
  46. package/dist/src/api/router.js +57 -0
  47. package/dist/src/api/router.js.map +1 -0
  48. package/dist/src/cli/commands/completions.d.ts +2 -0
  49. package/dist/src/cli/commands/completions.js +396 -0
  50. package/dist/src/cli/commands/completions.js.map +1 -0
  51. package/dist/src/cli/commands/deadlines.d.ts +2 -0
  52. package/dist/src/cli/commands/deadlines.js +47 -0
  53. package/dist/src/cli/commands/deadlines.js.map +1 -0
  54. package/dist/src/cli/commands/doctor.d.ts +2 -0
  55. package/dist/src/cli/commands/doctor.js +50 -0
  56. package/dist/src/cli/commands/doctor.js.map +1 -0
  57. package/dist/src/cli/commands/drafts.d.ts +2 -0
  58. package/dist/src/cli/commands/drafts.js +75 -0
  59. package/dist/src/cli/commands/drafts.js.map +1 -0
  60. package/dist/src/cli/commands/extract.d.ts +2 -0
  61. package/dist/src/cli/commands/extract.js +17 -0
  62. package/dist/src/cli/commands/extract.js.map +1 -0
  63. package/dist/src/cli/commands/fees.d.ts +2 -0
  64. package/dist/src/cli/commands/fees.js +40 -0
  65. package/dist/src/cli/commands/fees.js.map +1 -0
  66. package/dist/src/cli/commands/fill.d.ts +25 -0
  67. package/dist/src/cli/commands/fill.js +354 -0
  68. package/dist/src/cli/commands/fill.js.map +1 -0
  69. package/dist/src/cli/commands/glossary.d.ts +2 -0
  70. package/dist/src/cli/commands/glossary.js +90 -0
  71. package/dist/src/cli/commands/glossary.js.map +1 -0
  72. package/dist/src/cli/commands/life-event.d.ts +2 -0
  73. package/dist/src/cli/commands/life-event.js +41 -0
  74. package/dist/src/cli/commands/life-event.js.map +1 -0
  75. package/dist/src/cli/commands/list.d.ts +2 -0
  76. package/dist/src/cli/commands/list.js +19 -0
  77. package/dist/src/cli/commands/list.js.map +1 -0
  78. package/dist/src/cli/commands/machine.d.ts +2 -0
  79. package/dist/src/cli/commands/machine.js +100 -0
  80. package/dist/src/cli/commands/machine.js.map +1 -0
  81. package/dist/src/cli/commands/merge.d.ts +2 -0
  82. package/dist/src/cli/commands/merge.js +24 -0
  83. package/dist/src/cli/commands/merge.js.map +1 -0
  84. package/dist/src/cli/commands/plugin.d.ts +2 -0
  85. package/dist/src/cli/commands/plugin.js +119 -0
  86. package/dist/src/cli/commands/plugin.js.map +1 -0
  87. package/dist/src/cli/commands/profile.d.ts +2 -0
  88. package/dist/src/cli/commands/profile.js +125 -0
  89. package/dist/src/cli/commands/profile.js.map +1 -0
  90. package/dist/src/cli/commands/review.d.ts +2 -0
  91. package/dist/src/cli/commands/review.js +25 -0
  92. package/dist/src/cli/commands/review.js.map +1 -0
  93. package/dist/src/cli/commands/scaffold.d.ts +2 -0
  94. package/dist/src/cli/commands/scaffold.js +304 -0
  95. package/dist/src/cli/commands/scaffold.js.map +1 -0
  96. package/dist/src/cli/commands/schemas.d.ts +2 -0
  97. package/dist/src/cli/commands/schemas.js +14 -0
  98. package/dist/src/cli/commands/schemas.js.map +1 -0
  99. package/dist/src/cli/commands/screen.d.ts +2 -0
  100. package/dist/src/cli/commands/screen.js +89 -0
  101. package/dist/src/cli/commands/screen.js.map +1 -0
  102. package/dist/src/cli/commands/serve.d.ts +2 -0
  103. package/dist/src/cli/commands/serve.js +46 -0
  104. package/dist/src/cli/commands/serve.js.map +1 -0
  105. package/dist/src/cli/commands/start.d.ts +2 -0
  106. package/dist/src/cli/commands/start.js +16 -0
  107. package/dist/src/cli/commands/start.js.map +1 -0
  108. package/dist/src/cli/commands/stats.d.ts +2 -0
  109. package/dist/src/cli/commands/stats.js +17 -0
  110. package/dist/src/cli/commands/stats.js.map +1 -0
  111. package/dist/src/cli/commands/testdata.d.ts +2 -0
  112. package/dist/src/cli/commands/testdata.js +60 -0
  113. package/dist/src/cli/commands/testdata.js.map +1 -0
  114. package/dist/src/cli/commands/tui.d.ts +2 -0
  115. package/dist/src/cli/commands/tui.js +26 -0
  116. package/dist/src/cli/commands/tui.js.map +1 -0
  117. package/dist/src/cli/commands/validate.d.ts +2 -0
  118. package/dist/src/cli/commands/validate.js +22 -0
  119. package/dist/src/cli/commands/validate.js.map +1 -0
  120. package/dist/src/cli/commands/vault.d.ts +2 -0
  121. package/dist/src/cli/commands/vault.js +73 -0
  122. package/dist/src/cli/commands/vault.js.map +1 -0
  123. package/dist/src/cli/commands/workflows.d.ts +2 -0
  124. package/dist/src/cli/commands/workflows.js +23 -0
  125. package/dist/src/cli/commands/workflows.js.map +1 -0
  126. package/dist/src/cli/display/review.d.ts +2 -0
  127. package/dist/src/cli/display/review.js +4 -0
  128. package/dist/src/cli/display/review.js.map +1 -0
  129. package/dist/src/cli/display/summary.d.ts +2 -0
  130. package/dist/src/cli/display/summary.js +19 -0
  131. package/dist/src/cli/display/summary.js.map +1 -0
  132. package/dist/src/cli/display/validation.d.ts +2 -0
  133. package/dist/src/cli/display/validation.js +16 -0
  134. package/dist/src/cli/display/validation.js.map +1 -0
  135. package/dist/src/cli/index.d.ts +1 -0
  136. package/dist/src/cli/index.js +83 -0
  137. package/dist/src/cli/index.js.map +1 -0
  138. package/dist/src/cli/output.d.ts +21 -0
  139. package/dist/src/cli/output.js +107 -0
  140. package/dist/src/cli/output.js.map +1 -0
  141. package/dist/src/cli/prompts/common.d.ts +21 -0
  142. package/dist/src/cli/prompts/common.js +53 -0
  143. package/dist/src/cli/prompts/common.js.map +1 -0
  144. package/dist/src/cli/prompts/credits.d.ts +3 -0
  145. package/dist/src/cli/prompts/credits.js +23 -0
  146. package/dist/src/cli/prompts/credits.js.map +1 -0
  147. package/dist/src/cli/prompts/deductions.d.ts +12 -0
  148. package/dist/src/cli/prompts/deductions.js +26 -0
  149. package/dist/src/cli/prompts/deductions.js.map +1 -0
  150. package/dist/src/cli/prompts/identity.d.ts +8 -0
  151. package/dist/src/cli/prompts/identity.js +36 -0
  152. package/dist/src/cli/prompts/identity.js.map +1 -0
  153. package/dist/src/cli/prompts/income.d.ts +15 -0
  154. package/dist/src/cli/prompts/income.js +61 -0
  155. package/dist/src/cli/prompts/income.js.map +1 -0
  156. package/dist/src/cli/prompts/workflow.d.ts +3 -0
  157. package/dist/src/cli/prompts/workflow.js +74 -0
  158. package/dist/src/cli/prompts/workflow.js.map +1 -0
  159. package/dist/src/cli/support.d.ts +53 -0
  160. package/dist/src/cli/support.js +80 -0
  161. package/dist/src/cli/support.js.map +1 -0
  162. package/dist/src/cli/tui.d.ts +22 -0
  163. package/dist/src/cli/tui.js +122 -0
  164. package/dist/src/cli/tui.js.map +1 -0
  165. package/dist/src/engine/audit-risk/scorer.d.ts +36 -0
  166. package/dist/src/engine/audit-risk/scorer.js +132 -0
  167. package/dist/src/engine/audit-risk/scorer.js.map +1 -0
  168. package/dist/src/engine/crypto/cost-basis.d.ts +19 -0
  169. package/dist/src/engine/crypto/cost-basis.js +115 -0
  170. package/dist/src/engine/crypto/cost-basis.js.map +1 -0
  171. package/dist/src/engine/crypto/transaction-parser.d.ts +36 -0
  172. package/dist/src/engine/crypto/transaction-parser.js +124 -0
  173. package/dist/src/engine/crypto/transaction-parser.js.map +1 -0
  174. package/dist/src/engine/crypto/wash-sale.d.ts +25 -0
  175. package/dist/src/engine/crypto/wash-sale.js +51 -0
  176. package/dist/src/engine/crypto/wash-sale.js.map +1 -0
  177. package/dist/src/engine/equity/espp.d.ts +41 -0
  178. package/dist/src/engine/equity/espp.js +90 -0
  179. package/dist/src/engine/equity/espp.js.map +1 -0
  180. package/dist/src/engine/equity/iso.d.ts +42 -0
  181. package/dist/src/engine/equity/iso.js +94 -0
  182. package/dist/src/engine/equity/iso.js.map +1 -0
  183. package/dist/src/engine/equity/nso.d.ts +35 -0
  184. package/dist/src/engine/equity/nso.js +63 -0
  185. package/dist/src/engine/equity/nso.js.map +1 -0
  186. package/dist/src/engine/equity/rsu.d.ts +33 -0
  187. package/dist/src/engine/equity/rsu.js +64 -0
  188. package/dist/src/engine/equity/rsu.js.map +1 -0
  189. package/dist/src/engine/field-mapper.d.ts +26 -0
  190. package/dist/src/engine/field-mapper.js +119 -0
  191. package/dist/src/engine/field-mapper.js.map +1 -0
  192. package/dist/src/engine/forms/core-1040.d.ts +2 -0
  193. package/dist/src/engine/forms/core-1040.js +124 -0
  194. package/dist/src/engine/forms/core-1040.js.map +1 -0
  195. package/dist/src/engine/forms/form-8949.d.ts +21 -0
  196. package/dist/src/engine/forms/form-8949.js +96 -0
  197. package/dist/src/engine/forms/form-8949.js.map +1 -0
  198. package/dist/src/engine/forms/index.d.ts +7 -0
  199. package/dist/src/engine/forms/index.js +6 -0
  200. package/dist/src/engine/forms/index.js.map +1 -0
  201. package/dist/src/engine/forms/schedule-b.d.ts +15 -0
  202. package/dist/src/engine/forms/schedule-b.js +78 -0
  203. package/dist/src/engine/forms/schedule-b.js.map +1 -0
  204. package/dist/src/engine/forms/schedule-d.d.ts +9 -0
  205. package/dist/src/engine/forms/schedule-d.js +86 -0
  206. package/dist/src/engine/forms/schedule-d.js.map +1 -0
  207. package/dist/src/engine/fpl.d.ts +70 -0
  208. package/dist/src/engine/fpl.js +127 -0
  209. package/dist/src/engine/fpl.js.map +1 -0
  210. package/dist/src/engine/gig/home-office.d.ts +34 -0
  211. package/dist/src/engine/gig/home-office.js +53 -0
  212. package/dist/src/engine/gig/home-office.js.map +1 -0
  213. package/dist/src/engine/gig/mileage.d.ts +30 -0
  214. package/dist/src/engine/gig/mileage.js +45 -0
  215. package/dist/src/engine/gig/mileage.js.map +1 -0
  216. package/dist/src/engine/gig/multi-1099.d.ts +19 -0
  217. package/dist/src/engine/gig/multi-1099.js +28 -0
  218. package/dist/src/engine/gig/multi-1099.js.map +1 -0
  219. package/dist/src/engine/multiyear/carryforward.d.ts +31 -0
  220. package/dist/src/engine/multiyear/carryforward.js +77 -0
  221. package/dist/src/engine/multiyear/carryforward.js.map +1 -0
  222. package/dist/src/engine/multiyear/compare.d.ts +25 -0
  223. package/dist/src/engine/multiyear/compare.js +94 -0
  224. package/dist/src/engine/multiyear/compare.js.map +1 -0
  225. package/dist/src/engine/multiyear/import.d.ts +23 -0
  226. package/dist/src/engine/multiyear/import.js +193 -0
  227. package/dist/src/engine/multiyear/import.js.map +1 -0
  228. package/dist/src/engine/optimizer/filing-status.d.ts +29 -0
  229. package/dist/src/engine/optimizer/filing-status.js +66 -0
  230. package/dist/src/engine/optimizer/filing-status.js.map +1 -0
  231. package/dist/src/engine/orchestrator.d.ts +14 -0
  232. package/dist/src/engine/orchestrator.js +135 -0
  233. package/dist/src/engine/orchestrator.js.map +1 -0
  234. package/dist/src/engine/scenarios/whatif.d.ts +34 -0
  235. package/dist/src/engine/scenarios/whatif.js +70 -0
  236. package/dist/src/engine/scenarios/whatif.js.map +1 -0
  237. package/dist/src/engine/state/ca/calculator.d.ts +3 -0
  238. package/dist/src/engine/state/ca/calculator.js +107 -0
  239. package/dist/src/engine/state/ca/calculator.js.map +1 -0
  240. package/dist/src/engine/state/common.d.ts +39 -0
  241. package/dist/src/engine/state/common.js +115 -0
  242. package/dist/src/engine/state/common.js.map +1 -0
  243. package/dist/src/engine/state/ga/calculator.d.ts +3 -0
  244. package/dist/src/engine/state/ga/calculator.js +82 -0
  245. package/dist/src/engine/state/ga/calculator.js.map +1 -0
  246. package/dist/src/engine/state/il/calculator.d.ts +3 -0
  247. package/dist/src/engine/state/il/calculator.js +49 -0
  248. package/dist/src/engine/state/il/calculator.js.map +1 -0
  249. package/dist/src/engine/state/mi/calculator.d.ts +3 -0
  250. package/dist/src/engine/state/mi/calculator.js +60 -0
  251. package/dist/src/engine/state/mi/calculator.js.map +1 -0
  252. package/dist/src/engine/state/nc/calculator.d.ts +3 -0
  253. package/dist/src/engine/state/nc/calculator.js +54 -0
  254. package/dist/src/engine/state/nc/calculator.js.map +1 -0
  255. package/dist/src/engine/state/nj/calculator.d.ts +3 -0
  256. package/dist/src/engine/state/nj/calculator.js +90 -0
  257. package/dist/src/engine/state/nj/calculator.js.map +1 -0
  258. package/dist/src/engine/state/ny/calculator.d.ts +3 -0
  259. package/dist/src/engine/state/ny/calculator.js +103 -0
  260. package/dist/src/engine/state/ny/calculator.js.map +1 -0
  261. package/dist/src/engine/state/oh/calculator.d.ts +3 -0
  262. package/dist/src/engine/state/oh/calculator.js +79 -0
  263. package/dist/src/engine/state/oh/calculator.js.map +1 -0
  264. package/dist/src/engine/state/pa/calculator.d.ts +3 -0
  265. package/dist/src/engine/state/pa/calculator.js +58 -0
  266. package/dist/src/engine/state/pa/calculator.js.map +1 -0
  267. package/dist/src/engine/state/registry.d.ts +9 -0
  268. package/dist/src/engine/state/registry.js +67 -0
  269. package/dist/src/engine/state/registry.js.map +1 -0
  270. package/dist/src/engine/state/types.d.ts +57 -0
  271. package/dist/src/engine/state/types.js +2 -0
  272. package/dist/src/engine/state/types.js.map +1 -0
  273. package/dist/src/engine/state/va/calculator.d.ts +3 -0
  274. package/dist/src/engine/state/va/calculator.js +76 -0
  275. package/dist/src/engine/state/va/calculator.js.map +1 -0
  276. package/dist/src/engine/suggestions/missed-deductions.d.ts +43 -0
  277. package/dist/src/engine/suggestions/missed-deductions.js +128 -0
  278. package/dist/src/engine/suggestions/missed-deductions.js.map +1 -0
  279. package/dist/src/engine/tax-calculator.d.ts +55 -0
  280. package/dist/src/engine/tax-calculator.js +156 -0
  281. package/dist/src/engine/tax-calculator.js.map +1 -0
  282. package/dist/src/engine/tax-constants-2025.d.ts +56 -0
  283. package/dist/src/engine/tax-constants-2025.js +138 -0
  284. package/dist/src/engine/tax-constants-2025.js.map +1 -0
  285. package/dist/src/engine/types.d.ts +77 -0
  286. package/dist/src/engine/types.js +2 -0
  287. package/dist/src/engine/types.js.map +1 -0
  288. package/dist/src/engine/va-math.d.ts +24 -0
  289. package/dist/src/engine/va-math.js +90 -0
  290. package/dist/src/engine/va-math.js.map +1 -0
  291. package/dist/src/engine/validator.d.ts +7 -0
  292. package/dist/src/engine/validator.js +108 -0
  293. package/dist/src/engine/validator.js.map +1 -0
  294. package/dist/src/glossary/entries/healthcare.d.ts +2 -0
  295. package/dist/src/glossary/entries/healthcare.js +78 -0
  296. package/dist/src/glossary/entries/healthcare.js.map +1 -0
  297. package/dist/src/glossary/entries/immigration.d.ts +2 -0
  298. package/dist/src/glossary/entries/immigration.js +112 -0
  299. package/dist/src/glossary/entries/immigration.js.map +1 -0
  300. package/dist/src/glossary/entries/tax.d.ts +2 -0
  301. package/dist/src/glossary/entries/tax.js +149 -0
  302. package/dist/src/glossary/entries/tax.js.map +1 -0
  303. package/dist/src/glossary/entries/unemployment.d.ts +2 -0
  304. package/dist/src/glossary/entries/unemployment.js +52 -0
  305. package/dist/src/glossary/entries/unemployment.js.map +1 -0
  306. package/dist/src/glossary/index.d.ts +19 -0
  307. package/dist/src/glossary/index.js +39 -0
  308. package/dist/src/glossary/index.js.map +1 -0
  309. package/dist/src/i18n/index.d.ts +21 -0
  310. package/dist/src/i18n/index.js +92 -0
  311. package/dist/src/i18n/index.js.map +1 -0
  312. package/dist/src/mcp/result.d.ts +3 -0
  313. package/dist/src/mcp/result.js +7 -0
  314. package/dist/src/mcp/result.js.map +1 -0
  315. package/dist/src/mcp/return-bundle.d.ts +19 -0
  316. package/dist/src/mcp/return-bundle.js +49 -0
  317. package/dist/src/mcp/return-bundle.js.map +1 -0
  318. package/dist/src/mcp/shared.d.ts +49102 -0
  319. package/dist/src/mcp/shared.js +354 -0
  320. package/dist/src/mcp/shared.js.map +1 -0
  321. package/dist/src/mcp/tools/build-packet.d.ts +7 -0
  322. package/dist/src/mcp/tools/build-packet.js +23 -0
  323. package/dist/src/mcp/tools/build-packet.js.map +1 -0
  324. package/dist/src/mcp/tools/calculate-tax.d.ts +40 -0
  325. package/dist/src/mcp/tools/calculate-tax.js +17 -0
  326. package/dist/src/mcp/tools/calculate-tax.js.map +1 -0
  327. package/dist/src/mcp/tools/deadlines.d.ts +7 -0
  328. package/dist/src/mcp/tools/deadlines.js +23 -0
  329. package/dist/src/mcp/tools/deadlines.js.map +1 -0
  330. package/dist/src/mcp/tools/describe-form.d.ts +14 -0
  331. package/dist/src/mcp/tools/describe-form.js +23 -0
  332. package/dist/src/mcp/tools/describe-form.js.map +1 -0
  333. package/dist/src/mcp/tools/describe-workflow.d.ts +6 -0
  334. package/dist/src/mcp/tools/describe-workflow.js +16 -0
  335. package/dist/src/mcp/tools/describe-workflow.js.map +1 -0
  336. package/dist/src/mcp/tools/explain-flag.d.ts +7 -0
  337. package/dist/src/mcp/tools/explain-flag.js +19 -0
  338. package/dist/src/mcp/tools/explain-flag.js.map +1 -0
  339. package/dist/src/mcp/tools/extract-document.d.ts +13 -0
  340. package/dist/src/mcp/tools/extract-document.js +20 -0
  341. package/dist/src/mcp/tools/extract-document.js.map +1 -0
  342. package/dist/src/mcp/tools/fees.d.ts +7 -0
  343. package/dist/src/mcp/tools/fees.js +24 -0
  344. package/dist/src/mcp/tools/fees.js.map +1 -0
  345. package/dist/src/mcp/tools/fill-form.d.ts +56 -0
  346. package/dist/src/mcp/tools/fill-form.js +47 -0
  347. package/dist/src/mcp/tools/fill-form.js.map +1 -0
  348. package/dist/src/mcp/tools/fill-workflow.d.ts +7 -0
  349. package/dist/src/mcp/tools/fill-workflow.js +24 -0
  350. package/dist/src/mcp/tools/fill-workflow.js.map +1 -0
  351. package/dist/src/mcp/tools/glossary.d.ts +8 -0
  352. package/dist/src/mcp/tools/glossary.js +54 -0
  353. package/dist/src/mcp/tools/glossary.js.map +1 -0
  354. package/dist/src/mcp/tools/list-forms.d.ts +6 -0
  355. package/dist/src/mcp/tools/list-forms.js +15 -0
  356. package/dist/src/mcp/tools/list-forms.js.map +1 -0
  357. package/dist/src/mcp/tools/list-workflows.d.ts +6 -0
  358. package/dist/src/mcp/tools/list-workflows.js +16 -0
  359. package/dist/src/mcp/tools/list-workflows.js.map +1 -0
  360. package/dist/src/mcp/tools/plan-life-event.d.ts +59 -0
  361. package/dist/src/mcp/tools/plan-life-event.js +41 -0
  362. package/dist/src/mcp/tools/plan-life-event.js.map +1 -0
  363. package/dist/src/mcp/tools/review-form.d.ts +40 -0
  364. package/dist/src/mcp/tools/review-form.js +26 -0
  365. package/dist/src/mcp/tools/review-form.js.map +1 -0
  366. package/dist/src/mcp/tools/review-workflow.d.ts +7 -0
  367. package/dist/src/mcp/tools/review-workflow.js +27 -0
  368. package/dist/src/mcp/tools/review-workflow.js.map +1 -0
  369. package/dist/src/mcp/tools/screen-eligibility.d.ts +53 -0
  370. package/dist/src/mcp/tools/screen-eligibility.js +44 -0
  371. package/dist/src/mcp/tools/screen-eligibility.js.map +1 -0
  372. package/dist/src/mcp/tools/start-workflow.d.ts +6 -0
  373. package/dist/src/mcp/tools/start-workflow.js +17 -0
  374. package/dist/src/mcp/tools/start-workflow.js.map +1 -0
  375. package/dist/src/mcp/tools/validate-form.d.ts +48 -0
  376. package/dist/src/mcp/tools/validate-form.js +30 -0
  377. package/dist/src/mcp/tools/validate-form.js.map +1 -0
  378. package/dist/src/mcp/tools/validate-workflow.d.ts +7 -0
  379. package/dist/src/mcp/tools/validate-workflow.js +24 -0
  380. package/dist/src/mcp/tools/validate-workflow.js.map +1 -0
  381. package/dist/src/pdf/batch.d.ts +8 -0
  382. package/dist/src/pdf/batch.js +15 -0
  383. package/dist/src/pdf/batch.js.map +1 -0
  384. package/dist/src/pdf/merge.d.ts +10 -0
  385. package/dist/src/pdf/merge.js +62 -0
  386. package/dist/src/pdf/merge.js.map +1 -0
  387. package/dist/src/pdf/ocr.d.ts +22 -0
  388. package/dist/src/pdf/ocr.js +52 -0
  389. package/dist/src/pdf/ocr.js.map +1 -0
  390. package/dist/src/pdf/reader.d.ts +2 -0
  391. package/dist/src/pdf/reader.js +354 -0
  392. package/dist/src/pdf/reader.js.map +1 -0
  393. package/dist/src/pdf/shared.d.ts +89 -0
  394. package/dist/src/pdf/shared.js +137 -0
  395. package/dist/src/pdf/shared.js.map +1 -0
  396. package/dist/src/pdf/writer.d.ts +7 -0
  397. package/dist/src/pdf/writer.js +311 -0
  398. package/dist/src/pdf/writer.js.map +1 -0
  399. package/dist/src/plugins/loader.d.ts +15 -0
  400. package/dist/src/plugins/loader.js +86 -0
  401. package/dist/src/plugins/loader.js.map +1 -0
  402. package/dist/src/plugins/sandbox.d.ts +14 -0
  403. package/dist/src/plugins/sandbox.js +88 -0
  404. package/dist/src/plugins/sandbox.js.map +1 -0
  405. package/dist/src/plugins/types.d.ts +17 -0
  406. package/dist/src/plugins/types.js +13 -0
  407. package/dist/src/plugins/types.js.map +1 -0
  408. package/dist/src/schemas/2025/f1040.d.ts +127 -0
  409. package/dist/src/schemas/2025/f1040.js +104 -0
  410. package/dist/src/schemas/2025/f1040.js.map +1 -0
  411. package/dist/src/schemas/2025/f1099-int.d.ts +53 -0
  412. package/dist/src/schemas/2025/f1099-int.js +51 -0
  413. package/dist/src/schemas/2025/f1099-int.js.map +1 -0
  414. package/dist/src/schemas/2025/f1099-nec.d.ts +47 -0
  415. package/dist/src/schemas/2025/f1099-nec.js +45 -0
  416. package/dist/src/schemas/2025/f1099-nec.js.map +1 -0
  417. package/dist/src/schemas/2025/form-8949.d.ts +46 -0
  418. package/dist/src/schemas/2025/form-8949.js +71 -0
  419. package/dist/src/schemas/2025/form-8949.js.map +1 -0
  420. package/dist/src/schemas/2025/index.d.ts +1004 -0
  421. package/dist/src/schemas/2025/index.js +60 -0
  422. package/dist/src/schemas/2025/index.js.map +1 -0
  423. package/dist/src/schemas/2025/schedule-1.d.ts +44 -0
  424. package/dist/src/schemas/2025/schedule-1.js +56 -0
  425. package/dist/src/schemas/2025/schedule-1.js.map +1 -0
  426. package/dist/src/schemas/2025/schedule-b.d.ts +32 -0
  427. package/dist/src/schemas/2025/schedule-b.js +53 -0
  428. package/dist/src/schemas/2025/schedule-b.js.map +1 -0
  429. package/dist/src/schemas/2025/schedule-c.d.ts +58 -0
  430. package/dist/src/schemas/2025/schedule-c.js +46 -0
  431. package/dist/src/schemas/2025/schedule-c.js.map +1 -0
  432. package/dist/src/schemas/2025/schedule-d.d.ts +36 -0
  433. package/dist/src/schemas/2025/schedule-d.js +52 -0
  434. package/dist/src/schemas/2025/schedule-d.js.map +1 -0
  435. package/dist/src/schemas/2025/shared.d.ts +98 -0
  436. package/dist/src/schemas/2025/shared.js +161 -0
  437. package/dist/src/schemas/2025/shared.js.map +1 -0
  438. package/dist/src/schemas/2025/w2.d.ts +57 -0
  439. package/dist/src/schemas/2025/w2.js +62 -0
  440. package/dist/src/schemas/2025/w2.js.map +1 -0
  441. package/dist/src/storage/drafts.d.ts +25 -0
  442. package/dist/src/storage/drafts.js +117 -0
  443. package/dist/src/storage/drafts.js.map +1 -0
  444. package/dist/src/storage/paths.d.ts +15 -0
  445. package/dist/src/storage/paths.js +33 -0
  446. package/dist/src/storage/paths.js.map +1 -0
  447. package/dist/src/storage/profile.d.ts +17 -0
  448. package/dist/src/storage/profile.js +59 -0
  449. package/dist/src/storage/profile.js.map +1 -0
  450. package/dist/src/storage/vault.d.ts +38 -0
  451. package/dist/src/storage/vault.js +187 -0
  452. package/dist/src/storage/vault.js.map +1 -0
  453. package/dist/src/testing/synthetic.d.ts +13 -0
  454. package/dist/src/testing/synthetic.js +164 -0
  455. package/dist/src/testing/synthetic.js.map +1 -0
  456. package/dist/src/types.d.ts +362 -0
  457. package/dist/src/types.js +2 -0
  458. package/dist/src/types.js.map +1 -0
  459. package/dist/src/workflows/deadlines.d.ts +13 -0
  460. package/dist/src/workflows/deadlines.js +88 -0
  461. package/dist/src/workflows/deadlines.js.map +1 -0
  462. package/dist/src/workflows/domains/benefits.d.ts +409 -0
  463. package/dist/src/workflows/domains/benefits.js +744 -0
  464. package/dist/src/workflows/domains/benefits.js.map +1 -0
  465. package/dist/src/workflows/domains/business.d.ts +65 -0
  466. package/dist/src/workflows/domains/business.js +76 -0
  467. package/dist/src/workflows/domains/business.js.map +1 -0
  468. package/dist/src/workflows/domains/education.d.ts +251 -0
  469. package/dist/src/workflows/domains/education.js +504 -0
  470. package/dist/src/workflows/domains/education.js.map +1 -0
  471. package/dist/src/workflows/domains/estate.d.ts +217 -0
  472. package/dist/src/workflows/domains/estate.js +407 -0
  473. package/dist/src/workflows/domains/estate.js.map +1 -0
  474. package/dist/src/workflows/domains/healthcare-ext.d.ts +88 -0
  475. package/dist/src/workflows/domains/healthcare-ext.js +150 -0
  476. package/dist/src/workflows/domains/healthcare-ext.js.map +1 -0
  477. package/dist/src/workflows/domains/healthcare.d.ts +92 -0
  478. package/dist/src/workflows/domains/healthcare.js +126 -0
  479. package/dist/src/workflows/domains/healthcare.js.map +1 -0
  480. package/dist/src/workflows/domains/identity-domain.d.ts +291 -0
  481. package/dist/src/workflows/domains/identity-domain.js +541 -0
  482. package/dist/src/workflows/domains/identity-domain.js.map +1 -0
  483. package/dist/src/workflows/domains/immigration-ext.d.ts +282 -0
  484. package/dist/src/workflows/domains/immigration-ext.js +590 -0
  485. package/dist/src/workflows/domains/immigration-ext.js.map +1 -0
  486. package/dist/src/workflows/domains/immigration.d.ts +121 -0
  487. package/dist/src/workflows/domains/immigration.js +152 -0
  488. package/dist/src/workflows/domains/immigration.js.map +1 -0
  489. package/dist/src/workflows/domains/legal.d.ts +235 -0
  490. package/dist/src/workflows/domains/legal.js +380 -0
  491. package/dist/src/workflows/domains/legal.js.map +1 -0
  492. package/dist/src/workflows/domains/permits.d.ts +65 -0
  493. package/dist/src/workflows/domains/permits.js +74 -0
  494. package/dist/src/workflows/domains/permits.js.map +1 -0
  495. package/dist/src/workflows/domains/retirement.d.ts +59 -0
  496. package/dist/src/workflows/domains/retirement.js +155 -0
  497. package/dist/src/workflows/domains/retirement.js.map +1 -0
  498. package/dist/src/workflows/domains/tax.d.ts +133 -0
  499. package/dist/src/workflows/domains/tax.js +187 -0
  500. package/dist/src/workflows/domains/tax.js.map +1 -0
  501. package/dist/src/workflows/domains/unemployment.d.ts +105 -0
  502. package/dist/src/workflows/domains/unemployment.js +122 -0
  503. package/dist/src/workflows/domains/unemployment.js.map +1 -0
  504. package/dist/src/workflows/domains/veterans.d.ts +189 -0
  505. package/dist/src/workflows/domains/veterans.js +380 -0
  506. package/dist/src/workflows/domains/veterans.js.map +1 -0
  507. package/dist/src/workflows/fees.d.ts +15 -0
  508. package/dist/src/workflows/fees.js +93 -0
  509. package/dist/src/workflows/fees.js.map +1 -0
  510. package/dist/src/workflows/helpers.d.ts +8 -0
  511. package/dist/src/workflows/helpers.js +66 -0
  512. package/dist/src/workflows/helpers.js.map +1 -0
  513. package/dist/src/workflows/io.d.ts +3 -0
  514. package/dist/src/workflows/io.js +38 -0
  515. package/dist/src/workflows/io.js.map +1 -0
  516. package/dist/src/workflows/registry.d.ts +51355 -0
  517. package/dist/src/workflows/registry.js +140 -0
  518. package/dist/src/workflows/registry.js.map +1 -0
  519. package/dist/src/workflows/schemas/benefits.d.ts +78 -0
  520. package/dist/src/workflows/schemas/benefits.js +91 -0
  521. package/dist/src/workflows/schemas/benefits.js.map +1 -0
  522. package/dist/src/workflows/schemas/common.d.ts +27 -0
  523. package/dist/src/workflows/schemas/common.js +28 -0
  524. package/dist/src/workflows/schemas/common.js.map +1 -0
  525. package/dist/src/workflows/schemas/education.d.ts +70 -0
  526. package/dist/src/workflows/schemas/education.js +85 -0
  527. package/dist/src/workflows/schemas/education.js.map +1 -0
  528. package/dist/src/workflows/schemas/estate.d.ts +64 -0
  529. package/dist/src/workflows/schemas/estate.js +85 -0
  530. package/dist/src/workflows/schemas/estate.js.map +1 -0
  531. package/dist/src/workflows/schemas/healthcare-ext.d.ts +15 -0
  532. package/dist/src/workflows/schemas/healthcare-ext.js +17 -0
  533. package/dist/src/workflows/schemas/healthcare-ext.js.map +1 -0
  534. package/dist/src/workflows/schemas/healthcare.d.ts +33 -0
  535. package/dist/src/workflows/schemas/healthcare.js +20 -0
  536. package/dist/src/workflows/schemas/healthcare.js.map +1 -0
  537. package/dist/src/workflows/schemas/identity-domain.d.ts +64 -0
  538. package/dist/src/workflows/schemas/identity-domain.js +87 -0
  539. package/dist/src/workflows/schemas/identity-domain.js.map +1 -0
  540. package/dist/src/workflows/schemas/immigration-ext.d.ts +63 -0
  541. package/dist/src/workflows/schemas/immigration-ext.js +58 -0
  542. package/dist/src/workflows/schemas/immigration-ext.js.map +1 -0
  543. package/dist/src/workflows/schemas/immigration.d.ts +48 -0
  544. package/dist/src/workflows/schemas/immigration.js +27 -0
  545. package/dist/src/workflows/schemas/immigration.js.map +1 -0
  546. package/dist/src/workflows/schemas/legal.d.ts +45 -0
  547. package/dist/src/workflows/schemas/legal.js +45 -0
  548. package/dist/src/workflows/schemas/legal.js.map +1 -0
  549. package/dist/src/workflows/schemas/planning.d.ts +23 -0
  550. package/dist/src/workflows/schemas/planning.js +15 -0
  551. package/dist/src/workflows/schemas/planning.js.map +1 -0
  552. package/dist/src/workflows/schemas/retirement.d.ts +14 -0
  553. package/dist/src/workflows/schemas/retirement.js +22 -0
  554. package/dist/src/workflows/schemas/retirement.js.map +1 -0
  555. package/dist/src/workflows/schemas/tax.d.ts +58 -0
  556. package/dist/src/workflows/schemas/tax.js +51 -0
  557. package/dist/src/workflows/schemas/tax.js.map +1 -0
  558. package/dist/src/workflows/schemas/unemployment.d.ts +33 -0
  559. package/dist/src/workflows/schemas/unemployment.js +19 -0
  560. package/dist/src/workflows/schemas/unemployment.js.map +1 -0
  561. package/dist/src/workflows/schemas/veterans.d.ts +47 -0
  562. package/dist/src/workflows/schemas/veterans.js +51 -0
  563. package/dist/src/workflows/schemas/veterans.js.map +1 -0
  564. package/dist/src/workflows/types.d.ts +9 -0
  565. package/dist/src/workflows/types.js +2 -0
  566. package/dist/src/workflows/types.js.map +1 -0
  567. package/dist/xmcp.config.d.ts +3 -0
  568. package/dist/xmcp.config.js +29 -0
  569. package/dist/xmcp.config.js.map +1 -0
  570. package/llms.txt +48 -0
  571. 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":"AAiCA,MAAM,UAAU,oBAAoB,CAAC,GAAW,EAAE,QAAgB;IAChE,MAAM,KAAK,GAAG,GAAG,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;IACrC,IAAI,KAAK,CAAC,MAAM,GAAG,CAAC;QAAE,OAAO,EAAE,CAAC;IAEhC,MAAM,OAAO,GAAG,KAAK,CAAC,CAAC,CAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC;IACxE,MAAM,YAAY,GAA2B,EAAE,CAAC;IAEhD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;QACtC,MAAM,MAAM,GAAG,KAAK,CAAC,CAAC,CAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC,CAAC;QAC/E,MAAM,GAAG,GAA2B,EAAE,CAAC;QACvC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YACxC,GAAG,CAAC,OAAO,CAAC,CAAC,CAAE,CAAC,GAAG,MAAM,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;QACrC,CAAC;QAED,MAAM,MAAM,GAAG,cAAc,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;QAC7C,IAAI,MAAM,EAAE,CAAC;YACX,YAAY,CAAC,IAAI,CAAC,EAAE,GAAG,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC;IAED,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,SAAS,cAAc,CAAC,GAA2B,EAAE,QAAgB;IACnE,QAAQ,QAAQ,CAAC,WAAW,EAAE,EAAE,CAAC;QAC/B,KAAK,UAAU,CAAC,CAAC,OAAO,cAAc,CAAC,GAAG,CAAC,CAAC;QAC5C,KAAK,QAAQ,CAAC,CAAC,OAAO,YAAY,CAAC,GAAG,CAAC,CAAC;QACxC,KAAK,SAAS,CAAC,CAAC,OAAO,aAAa,CAAC,GAAG,CAAC,CAAC;QAC1C,OAAO,CAAC,CAAC,OAAO,aAAa,CAAC,GAAG,CAAC,CAAC;IACrC,CAAC;AACH,CAAC;AAED,SAAS,cAAc,CAAC,GAA2B;IACjD,MAAM,IAAI,GAAG,kBAAkB,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,kBAAkB,CAAC,IAAI,EAAE,CAAC,CAAC;IAC9E,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IAEvB,OAAO;QACL,IAAI,EAAE,GAAG,CAAC,WAAW,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE;QAC3C,IAAI;QACJ,KAAK,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE;QAC5C,MAAM,EAAE,UAAU,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC;QAC3D,YAAY,EAAE,UAAU,CAAC,GAAG,CAAC,2BAA2B,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC;QACjF,UAAU,EAAE,UAAU,CAAC,GAAG,CAAC,yCAAyC,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC;QAC7F,GAAG,EAAE,UAAU,CAAC,GAAG,CAAC,oBAAoB,CAAC,IAAI,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,IAAI,SAAS;KAC7E,CAAC;AACJ,CAAC;AAED,SAAS,YAAY,CAAC,GAA2B;IAC/C,MAAM,IAAI,GAAG,kBAAkB,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC;IACnD,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IAEvB,OAAO;QACL,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE;QACtC,IAAI;QACJ,KAAK,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE;QACxC,MAAM,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,CAAC;QAChE,YAAY,EAAE,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC;QAC7C,UAAU,EAAE,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,CAAC;QACpE,GAAG,EAAE,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,IAAI,SAAS;KAChD,CAAC;AACJ,CAAC;AAED,SAAS,aAAa,CAAC,GAA2B;IAChD,MAAM,IAAI,GAAG,kBAAkB,CAAC,GAAG,CAAC,WAAW,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC;IACvE,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IAEvB,OAAO;QACL,IAAI,EAAE,GAAG,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE;QAC1C,IAAI;QACJ,KAAK,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,OAAO,CAAC,IAAI,EAAE;QACxC,MAAM,EAAE,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC;QACzD,YAAY,EAAE,CAAC,EAAE,+DAA+D;QAChF,UAAU,EAAE,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC;KAC5C,CAAC;AACJ,CAAC;AAED,SAAS,aAAa,CAAC,GAA2B;IAChD,MAAM,IAAI,GAAG,kBAAkB,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,kBAAkB,CAAC,IAAI,EAAE,CAAC,CAAC;IAC9E,IAAI,CAAC,IAAI;QAAE,OAAO,IAAI,CAAC;IAEvB,OAAO;QACL,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,GAAG,CAAC,WAAW,CAAC,IAAI,EAAE;QAC3C,IAAI;QACJ,KAAK,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,IAAI,EAAE;QAC3D,MAAM,EAAE,UAAU,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,UAAU,CAAC,IAAI,GAAG,CAAC;QAC3D,YAAY,EAAE,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,gBAAgB,CAAC,IAAI,GAAG,CAAC;QACtE,UAAU,EAAE,UAAU,CAAC,GAAG,CAAC,OAAO,CAAC,IAAI,GAAG,CAAC,aAAa,CAAC,IAAI,GAAG,CAAC;QACjE,GAAG,EAAE,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,GAAG,CAAC,IAAI,SAAS;KAChD,CAAC;AACJ,CAAC;AAED,SAAS,kBAAkB,CAAC,GAAW;IACrC,MAAM,UAAU,GAAG,GAAG,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC;IAC5C,IAAI,CAAC,KAAK,EAAE,UAAU,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,KAAK,CAAC;IAC3D,IAAI,CAAC,MAAM,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,MAAM,CAAC;IACjD,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,SAAS,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,OAAO,CAAC;IACtE,IAAI,CAAC,gBAAgB,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,gBAAgB,CAAC;IAClG,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,QAAQ,CAAC;IAC9D,IAAI,CAAC,SAAS,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,SAAS,CAAC;IACvD,IAAI,CAAC,MAAM,EAAE,SAAS,EAAE,UAAU,EAAE,SAAS,EAAE,YAAY,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC;QAAE,OAAO,UAAU,CAAC;IACrG,OAAO,IAAI,CAAC;AACd,CAAC;AAED,MAAM,UAAU,yBAAyB,CAAC,YAAoC;IAC5E,MAAM,IAAI,GAAsB,EAAE,CAAC;IACnC,IAAI,UAAU,GAAG,CAAC,CAAC;IAEnB,KAAK,MAAM,EAAE,IAAI,YAAY,EAAE,CAAC;QAC9B,IAAI,EAAE,CAAC,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC,IAAI,KAAK,OAAO,IAAI,EAAE,CAAC,IAAI,KAAK,gBAAgB,IAAI,EAAE,CAAC,IAAI,KAAK,QAAQ,IAAI,EAAE,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;YAC9H,UAAU,EAAE,CAAC;YACb,MAAM,MAAM,GACV,EAAE,CAAC,IAAI,KAAK,gBAAgB,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;gBAC1C,EAAE,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;oBACjC,EAAE,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;wBACnC,EAAE,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,UAAU,CAAC;YAE7C,IAAI,CAAC,IAAI,CAAC;gBACR,EAAE,EAAE,OAAO,UAAU,EAAE;gBACvB,KAAK,EAAE,EAAE,CAAC,KAAK;gBACf,YAAY,EAAE,EAAE,CAAC,IAAI;gBACrB,SAAS,EAAE,EAAE,CAAC,UAAU,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,CAAC;gBACxC,MAAM,EAAE,EAAE,CAAC,MAAM;gBACjB,MAAM;gBACN,QAAQ,EAAE,EAAE,CAAC,QAAQ;aACtB,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC"}
@@ -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;