@iloom/cli 0.10.2 → 0.11.1
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/LICENSE +1 -1
- package/README.md +4 -2
- package/dist/{BranchNamingService-4OP6LOH6.js → BranchNamingService-XBCO747L.js} +4 -4
- package/dist/ClaudeContextManager-SXDCWDJA.js +14 -0
- package/dist/ClaudeService-6E6MCGJE.js +13 -0
- package/dist/GitHubService-2R5GQG4K.js +12 -0
- package/dist/IssueTrackerFactory-XN6MQ4UN.js +14 -0
- package/dist/{LoomLauncher-FRECYMXS.js → LoomLauncher-5AZU2F5I.js} +15 -12
- package/dist/LoomLauncher-5AZU2F5I.js.map +1 -0
- package/dist/MetadataManager-CMQQTFLQ.js +10 -0
- package/dist/ProjectCapabilityDetector-IC6NAFGY.js +11 -0
- package/dist/{PromptTemplateManager-YOE2SIPG.js → PromptTemplateManager-T5VTLJP3.js} +3 -3
- package/dist/README.md +4 -2
- package/dist/{SettingsManager-FNKCOZMQ.js → SettingsManager-WQ5NSGAH.js} +3 -3
- package/dist/SettingsMigrationManager-S6J7OHUH.js +10 -0
- package/dist/agents/iloom-code-reviewer.md +50 -8
- package/dist/agents/iloom-issue-analyze-and-plan.md +10 -0
- package/dist/agents/iloom-issue-analyzer.md +13 -0
- package/dist/agents/iloom-issue-complexity-evaluator.md +8 -1
- package/dist/agents/iloom-issue-enhancer.md +8 -1
- package/dist/agents/iloom-issue-implementer.md +3 -1
- package/dist/agents/iloom-issue-planner.md +5 -0
- package/dist/build-OLS6J5KZ.js +27 -0
- package/dist/{chunk-WWKOVDWC.js → chunk-3GTUXW26.js} +3 -3
- package/dist/{chunk-4FGEGQW4.js → chunk-3RXYOBME.js} +5 -5
- package/dist/{chunk-HEXKPKCK.js → chunk-5LTID2AF.js} +6 -6
- package/dist/{chunk-G5V75JD5.js → chunk-5PNZBH6V.js} +2 -2
- package/dist/{chunk-SKSYYBCU.js → chunk-5UFGO4ZT.js} +24 -3
- package/dist/{chunk-SKSYYBCU.js.map → chunk-5UFGO4ZT.js.map} +1 -1
- package/dist/{chunk-XE4BDRZD.js → chunk-6YVJVUR4.js} +3 -3
- package/dist/{chunk-RJ3VBUFK.js → chunk-7FIXNAUO.js} +36 -7
- package/dist/chunk-7FIXNAUO.js.map +1 -0
- package/dist/{chunk-QFTDZ5E3.js → chunk-7NFCGKZT.js} +3 -3
- package/dist/{chunk-433MOLAU.js → chunk-7OCGBJLR.js} +2 -2
- package/dist/{chunk-2VEWSM34.js → chunk-ABVMUNCD.js} +8 -8
- package/dist/{chunk-7JDMYTFZ.js → chunk-CV47VCMQ.js} +2 -2
- package/dist/{chunk-7VHJNVLF.js → chunk-ET6A2JR4.js} +8 -6
- package/dist/chunk-ET6A2JR4.js.map +1 -0
- package/dist/{chunk-MORRVYPT.js → chunk-G2MNSPA4.js} +2 -2
- package/dist/{chunk-RSYT7MVI.js → chunk-GMDSYLI6.js} +36 -3
- package/dist/chunk-GMDSYLI6.js.map +1 -0
- package/dist/{chunk-VT4PDUYT.js → chunk-H2SSF24U.js} +284 -209
- package/dist/chunk-H2SSF24U.js.map +1 -0
- package/dist/{chunk-LLHXQS3C.js → chunk-HLDY5S4C.js} +3 -3
- package/dist/{chunk-BYUMEDDD.js → chunk-IDCE26KD.js} +3 -3
- package/dist/{chunk-ZGM2FE2R.js → chunk-IR74O2F6.js} +221 -86
- package/dist/chunk-IR74O2F6.js.map +1 -0
- package/dist/{chunk-O7VL5N6S.js → chunk-K7R5QY6C.js} +2 -2
- package/dist/{chunk-BU53XIGY.js → chunk-KQFIGI37.js} +5 -5
- package/dist/{chunk-I5T677EA.js → chunk-LE2NOUTN.js} +3 -3
- package/dist/{chunk-KXDRI47U.js → chunk-LHDD4JHC.js} +6 -6
- package/dist/{chunk-2YZCWAVZ.js → chunk-LL6TOX3G.js} +9 -9
- package/dist/{chunk-YQ57ORTV.js → chunk-NCPZYQ4B.js} +2 -2
- package/dist/{chunk-EWJFUFPT.js → chunk-NDSGJZI2.js} +2 -2
- package/dist/{chunk-VG45TUYK.js → chunk-NH3QZYE5.js} +2 -2
- package/dist/{chunk-BFLMCE2U.js → chunk-NN5RYWXA.js} +7 -7
- package/dist/chunk-NOMQ5RFG.js +118 -0
- package/dist/chunk-NOMQ5RFG.js.map +1 -0
- package/dist/{chunk-KIK2ZFAL.js → chunk-QNHZM5ZV.js} +3 -3
- package/dist/{chunk-V3SVMFDQ.js → chunk-QR4FU53I.js} +8 -8
- package/dist/{chunk-FXDYIV3K.js → chunk-QVAA5KHK.js} +2 -2
- package/dist/{chunk-NGJZ4TOU.js → chunk-RBYTXYGD.js} +2 -2
- package/dist/{chunk-SWSJWA2S.js → chunk-RMLADZRY.js} +12 -9
- package/dist/chunk-RMLADZRY.js.map +1 -0
- package/dist/{chunk-CVCTIDDK.js → chunk-RVI6C2H5.js} +5 -5
- package/dist/{chunk-Q7POFB5Q.js → chunk-SQYHPBFP.js} +2 -2
- package/dist/{chunk-BFHDVFSK.js → chunk-TEJAGQX2.js} +64 -40
- package/dist/chunk-TEJAGQX2.js.map +1 -0
- package/dist/{chunk-QZWEJVWV.js → chunk-TZNNJLGT.js} +6 -6
- package/dist/{chunk-63QWFWH3.js → chunk-UDCI3QTS.js} +2 -2
- package/dist/{chunk-UKBAJ2QQ.js → chunk-UHIBKD73.js} +7 -7
- package/dist/{chunk-WXIM2WS7.js → chunk-V4STTBQD.js} +10 -10
- package/dist/{chunk-P4O6EH46.js → chunk-VMZG66UV.js} +5 -5
- package/dist/{chunk-C6HNNJIV.js → chunk-VNYWBHKR.js} +34 -4
- package/dist/chunk-VNYWBHKR.js.map +1 -0
- package/dist/{chunk-HYGUPUV5.js → chunk-VUUN3KE4.js} +10 -10
- package/dist/chunk-VUUN3KE4.js.map +1 -0
- package/dist/{chunk-UUEW5KWB.js → chunk-WG4MLJ6J.js} +15 -10
- package/dist/chunk-WG4MLJ6J.js.map +1 -0
- package/dist/{chunk-3F27M7ZD.js → chunk-XFQGI2E3.js} +66 -43
- package/dist/chunk-XFQGI2E3.js.map +1 -0
- package/dist/{chunk-LUKXJSRI.js → chunk-XXFSOVL3.js} +4 -4
- package/dist/{chunk-PZ5WSR5Z.js → chunk-Y3RX7LZT.js} +8 -5
- package/dist/chunk-Y3RX7LZT.js.map +1 -0
- package/dist/{chunk-KB64WNBZ.js → chunk-YRCEOQPX.js} +4 -2
- package/dist/chunk-YRCEOQPX.js.map +1 -0
- package/dist/{chunk-6MLEBAYZ.js → chunk-ZAXRQLK3.js} +2 -2
- package/dist/{claude-LN7OWVNI.js → claude-ONQTDWV3.js} +4 -4
- package/dist/{cleanup-4ZM2AJDC.js → cleanup-YOM6PQCN.js} +35 -34
- package/dist/{cleanup-4ZM2AJDC.js.map → cleanup-YOM6PQCN.js.map} +1 -1
- package/dist/cli.js +227 -152
- package/dist/cli.js.map +1 -1
- package/dist/{color-4TJ4P5EY.js → color-VQD52LOI.js} +3 -3
- package/dist/{commit-4CFLXRZ3.js → commit-DC2Q5CDY.js} +15 -15
- package/dist/{compile-7ALJHZ4N.js → compile-4NCQECKE.js} +11 -11
- package/dist/{contribute-5GKLK3BQ.js → contribute-M5UWXCAV.js} +12 -12
- package/dist/darwin-5BHWRJ7D.js +10 -0
- package/dist/{dev-server-7SMIB7OF.js → dev-server-CYRP6M73.js} +19 -19
- package/dist/{feedback-EZWF5CAL.js → feedback-BMAZGKRW.js} +16 -16
- package/dist/{git-GTLKAZRJ.js → git-BXUD6CL5.js} +6 -6
- package/dist/ignite-IO4LXVXJ.js +35 -0
- package/dist/index.d.ts +39 -65
- package/dist/index.js +634 -107
- package/dist/index.js.map +1 -1
- package/dist/{init-ZB2RITW6.js → init-CI43GJHV.js} +17 -17
- package/dist/{init-ZB2RITW6.js.map → init-CI43GJHV.js.map} +1 -1
- package/dist/{install-deps-RLSGSHH7.js → install-deps-SRTM5U7D.js} +11 -11
- package/dist/{installation-detector-MMFWLJYN.js → installation-detector-HF6QN7KP.js} +3 -3
- package/dist/{issues-4UUAQ5K6.js → issues-DMRQJH7E.js} +15 -15
- package/dist/lint-BSWRMGPZ.js +27 -0
- package/dist/linux-RYLOP2LY.js +103 -0
- package/dist/linux-RYLOP2LY.js.map +1 -0
- package/dist/mcp/chunk-PIIRD4LO.js +373 -0
- package/dist/mcp/chunk-PIIRD4LO.js.map +1 -0
- package/dist/mcp/darwin-3JFFE3W2.js +10 -0
- package/dist/mcp/issue-management-server.js +23 -127
- package/dist/mcp/issue-management-server.js.map +1 -1
- package/dist/mcp/linux-JBVS4R3A.js +103 -0
- package/dist/mcp/linux-JBVS4R3A.js.map +1 -0
- package/dist/mcp/tmux-RYBLEHUZ.js +156 -0
- package/dist/mcp/tmux-RYBLEHUZ.js.map +1 -0
- package/dist/mcp/wsl-4QZIQLLE.js +78 -0
- package/dist/mcp/wsl-4QZIQLLE.js.map +1 -0
- package/dist/neon-helpers-HWIYRKOW.js +11 -0
- package/dist/{open-FXWW3VI4.js → open-2Y7GSUTJ.js} +19 -19
- package/dist/{plan-D3KSN5MU.js → plan-SWFPLNJE.js} +47 -46
- package/dist/{plan-D3KSN5MU.js.map → plan-SWFPLNJE.js.map} +1 -1
- package/dist/{projects-2UOXFLNZ.js → projects-IUSUXD5D.js} +6 -6
- package/dist/{prompt-ONNPSNKM.js → prompt-7LZB4PAT.js} +3 -3
- package/dist/prompts/init-prompt.txt +56 -107
- package/dist/prompts/issue-prompt.txt +58 -12
- package/dist/prompts/pr-prompt.txt +154 -4
- package/dist/prompts/regular-prompt.txt +20 -3
- package/dist/prompts/swarm-orchestrator-prompt.txt +114 -8
- package/dist/{rebase-62FDLIH4.js → rebase-S6OHAOOF.js} +12 -12
- package/dist/{recap-OMBOKJST.js → recap-GGVCG5VH.js} +9 -9
- package/dist/{remote-IJAMOEAP.js → remote-MZTFHHTU.js} +3 -3
- package/dist/remote-MZTFHHTU.js.map +1 -0
- package/dist/{run-BBXLRIZB.js → run-ST3FR75O.js} +19 -19
- package/dist/schema/settings.schema.json +8 -49
- package/dist/{shell-RF7LTND5.js → shell-W4SBQPTE.js} +8 -8
- package/dist/{summary-YZI25KW4.js → summary-P2JCIIJO.js} +17 -17
- package/dist/test-6JH4FE2X.js +27 -0
- package/dist/{test-git-XM4TM65W.js → test-git-2KFFAQ6B.js} +6 -6
- package/dist/{test-jira-LDTOYFSD.js → test-jira-FKDKG6CD.js} +8 -8
- package/dist/{test-prefix-GBO37XCN.js → test-prefix-GP2DAX37.js} +11 -11
- package/dist/test-prefix-GP2DAX37.js.map +1 -0
- package/dist/{test-tabs-D3POYOJ5.js → test-tabs-YDWMWTVA.js} +3 -3
- package/dist/{test-webserver-NZ3JTVLL.js → test-webserver-QI3QQFZ3.js} +8 -8
- package/dist/tmux-7ZTA3BDI.js +156 -0
- package/dist/tmux-7ZTA3BDI.js.map +1 -0
- package/dist/{update-HJKDYA3F.js → update-XLW7R7FL.js} +4 -4
- package/dist/{update-notifier-LBAUOOLM.js → update-notifier-EYLAXZAA.js} +3 -3
- package/dist/update-notifier-EYLAXZAA.js.map +1 -0
- package/dist/{vscode-6XUGHJKL.js → vscode-TOGE5N67.js} +13 -13
- package/dist/{vscode-announcement-EQ2SKK3T.js → vscode-announcement-NIX7O2MG.js} +3 -3
- package/dist/wsl-Y4GUTOQ7.js +78 -0
- package/dist/wsl-Y4GUTOQ7.js.map +1 -0
- package/package.json +4 -2
- package/dist/ClaudeContextManager-ZKTUVQB2.js +0 -14
- package/dist/ClaudeService-TRWOYQ6O.js +0 -13
- package/dist/GitHubService-MEHKHUQP.js +0 -12
- package/dist/IssueTrackerFactory-NG53YX5S.js +0 -14
- package/dist/LoomLauncher-FRECYMXS.js.map +0 -1
- package/dist/MetadataManager-5QZSTKNN.js +0 -10
- package/dist/ProjectCapabilityDetector-5KSYUTBJ.js +0 -11
- package/dist/SettingsMigrationManager-LEBMJP3B.js +0 -10
- package/dist/build-VHGEMXBA.js +0 -27
- package/dist/chunk-3F27M7ZD.js.map +0 -1
- package/dist/chunk-7VHJNVLF.js.map +0 -1
- package/dist/chunk-BFHDVFSK.js.map +0 -1
- package/dist/chunk-C6HNNJIV.js.map +0 -1
- package/dist/chunk-HYGUPUV5.js.map +0 -1
- package/dist/chunk-KB64WNBZ.js.map +0 -1
- package/dist/chunk-PZ5WSR5Z.js.map +0 -1
- package/dist/chunk-RJ3VBUFK.js.map +0 -1
- package/dist/chunk-RSYT7MVI.js.map +0 -1
- package/dist/chunk-SWSJWA2S.js.map +0 -1
- package/dist/chunk-UUEW5KWB.js.map +0 -1
- package/dist/chunk-VT4PDUYT.js.map +0 -1
- package/dist/chunk-ZGM2FE2R.js.map +0 -1
- package/dist/ignite-MQETGFNA.js +0 -34
- package/dist/lint-AAN2NZWG.js +0 -27
- package/dist/neon-helpers-CQN2PB4S.js +0 -11
- package/dist/test-SGO6I5Z7.js +0 -27
- package/dist/test-prefix-GBO37XCN.js.map +0 -1
- /package/dist/{BranchNamingService-4OP6LOH6.js.map → BranchNamingService-XBCO747L.js.map} +0 -0
- /package/dist/{ClaudeContextManager-ZKTUVQB2.js.map → ClaudeContextManager-SXDCWDJA.js.map} +0 -0
- /package/dist/{ClaudeService-TRWOYQ6O.js.map → ClaudeService-6E6MCGJE.js.map} +0 -0
- /package/dist/{GitHubService-MEHKHUQP.js.map → GitHubService-2R5GQG4K.js.map} +0 -0
- /package/dist/{IssueTrackerFactory-NG53YX5S.js.map → IssueTrackerFactory-XN6MQ4UN.js.map} +0 -0
- /package/dist/{MetadataManager-5QZSTKNN.js.map → MetadataManager-CMQQTFLQ.js.map} +0 -0
- /package/dist/{ProjectCapabilityDetector-5KSYUTBJ.js.map → ProjectCapabilityDetector-IC6NAFGY.js.map} +0 -0
- /package/dist/{PromptTemplateManager-YOE2SIPG.js.map → PromptTemplateManager-T5VTLJP3.js.map} +0 -0
- /package/dist/{SettingsManager-FNKCOZMQ.js.map → SettingsManager-WQ5NSGAH.js.map} +0 -0
- /package/dist/{SettingsMigrationManager-LEBMJP3B.js.map → SettingsMigrationManager-S6J7OHUH.js.map} +0 -0
- /package/dist/{build-VHGEMXBA.js.map → build-OLS6J5KZ.js.map} +0 -0
- /package/dist/{chunk-WWKOVDWC.js.map → chunk-3GTUXW26.js.map} +0 -0
- /package/dist/{chunk-4FGEGQW4.js.map → chunk-3RXYOBME.js.map} +0 -0
- /package/dist/{chunk-HEXKPKCK.js.map → chunk-5LTID2AF.js.map} +0 -0
- /package/dist/{chunk-G5V75JD5.js.map → chunk-5PNZBH6V.js.map} +0 -0
- /package/dist/{chunk-XE4BDRZD.js.map → chunk-6YVJVUR4.js.map} +0 -0
- /package/dist/{chunk-QFTDZ5E3.js.map → chunk-7NFCGKZT.js.map} +0 -0
- /package/dist/{chunk-433MOLAU.js.map → chunk-7OCGBJLR.js.map} +0 -0
- /package/dist/{chunk-2VEWSM34.js.map → chunk-ABVMUNCD.js.map} +0 -0
- /package/dist/{chunk-7JDMYTFZ.js.map → chunk-CV47VCMQ.js.map} +0 -0
- /package/dist/{chunk-MORRVYPT.js.map → chunk-G2MNSPA4.js.map} +0 -0
- /package/dist/{chunk-LLHXQS3C.js.map → chunk-HLDY5S4C.js.map} +0 -0
- /package/dist/{chunk-BYUMEDDD.js.map → chunk-IDCE26KD.js.map} +0 -0
- /package/dist/{chunk-O7VL5N6S.js.map → chunk-K7R5QY6C.js.map} +0 -0
- /package/dist/{chunk-BU53XIGY.js.map → chunk-KQFIGI37.js.map} +0 -0
- /package/dist/{chunk-I5T677EA.js.map → chunk-LE2NOUTN.js.map} +0 -0
- /package/dist/{chunk-KXDRI47U.js.map → chunk-LHDD4JHC.js.map} +0 -0
- /package/dist/{chunk-2YZCWAVZ.js.map → chunk-LL6TOX3G.js.map} +0 -0
- /package/dist/{chunk-YQ57ORTV.js.map → chunk-NCPZYQ4B.js.map} +0 -0
- /package/dist/{chunk-EWJFUFPT.js.map → chunk-NDSGJZI2.js.map} +0 -0
- /package/dist/{chunk-VG45TUYK.js.map → chunk-NH3QZYE5.js.map} +0 -0
- /package/dist/{chunk-BFLMCE2U.js.map → chunk-NN5RYWXA.js.map} +0 -0
- /package/dist/{chunk-KIK2ZFAL.js.map → chunk-QNHZM5ZV.js.map} +0 -0
- /package/dist/{chunk-V3SVMFDQ.js.map → chunk-QR4FU53I.js.map} +0 -0
- /package/dist/{chunk-FXDYIV3K.js.map → chunk-QVAA5KHK.js.map} +0 -0
- /package/dist/{chunk-NGJZ4TOU.js.map → chunk-RBYTXYGD.js.map} +0 -0
- /package/dist/{chunk-CVCTIDDK.js.map → chunk-RVI6C2H5.js.map} +0 -0
- /package/dist/{chunk-Q7POFB5Q.js.map → chunk-SQYHPBFP.js.map} +0 -0
- /package/dist/{chunk-QZWEJVWV.js.map → chunk-TZNNJLGT.js.map} +0 -0
- /package/dist/{chunk-63QWFWH3.js.map → chunk-UDCI3QTS.js.map} +0 -0
- /package/dist/{chunk-UKBAJ2QQ.js.map → chunk-UHIBKD73.js.map} +0 -0
- /package/dist/{chunk-WXIM2WS7.js.map → chunk-V4STTBQD.js.map} +0 -0
- /package/dist/{chunk-P4O6EH46.js.map → chunk-VMZG66UV.js.map} +0 -0
- /package/dist/{chunk-LUKXJSRI.js.map → chunk-XXFSOVL3.js.map} +0 -0
- /package/dist/{chunk-6MLEBAYZ.js.map → chunk-ZAXRQLK3.js.map} +0 -0
- /package/dist/{claude-LN7OWVNI.js.map → claude-ONQTDWV3.js.map} +0 -0
- /package/dist/{color-4TJ4P5EY.js.map → color-VQD52LOI.js.map} +0 -0
- /package/dist/{commit-4CFLXRZ3.js.map → commit-DC2Q5CDY.js.map} +0 -0
- /package/dist/{compile-7ALJHZ4N.js.map → compile-4NCQECKE.js.map} +0 -0
- /package/dist/{contribute-5GKLK3BQ.js.map → contribute-M5UWXCAV.js.map} +0 -0
- /package/dist/{git-GTLKAZRJ.js.map → darwin-5BHWRJ7D.js.map} +0 -0
- /package/dist/{dev-server-7SMIB7OF.js.map → dev-server-CYRP6M73.js.map} +0 -0
- /package/dist/{feedback-EZWF5CAL.js.map → feedback-BMAZGKRW.js.map} +0 -0
- /package/dist/{ignite-MQETGFNA.js.map → git-BXUD6CL5.js.map} +0 -0
- /package/dist/{installation-detector-MMFWLJYN.js.map → ignite-IO4LXVXJ.js.map} +0 -0
- /package/dist/{install-deps-RLSGSHH7.js.map → install-deps-SRTM5U7D.js.map} +0 -0
- /package/dist/{neon-helpers-CQN2PB4S.js.map → installation-detector-HF6QN7KP.js.map} +0 -0
- /package/dist/{issues-4UUAQ5K6.js.map → issues-DMRQJH7E.js.map} +0 -0
- /package/dist/{lint-AAN2NZWG.js.map → lint-BSWRMGPZ.js.map} +0 -0
- /package/dist/{prompt-ONNPSNKM.js.map → mcp/darwin-3JFFE3W2.js.map} +0 -0
- /package/dist/{remote-IJAMOEAP.js.map → neon-helpers-HWIYRKOW.js.map} +0 -0
- /package/dist/{open-FXWW3VI4.js.map → open-2Y7GSUTJ.js.map} +0 -0
- /package/dist/{projects-2UOXFLNZ.js.map → projects-IUSUXD5D.js.map} +0 -0
- /package/dist/{update-notifier-LBAUOOLM.js.map → prompt-7LZB4PAT.js.map} +0 -0
- /package/dist/{rebase-62FDLIH4.js.map → rebase-S6OHAOOF.js.map} +0 -0
- /package/dist/{recap-OMBOKJST.js.map → recap-GGVCG5VH.js.map} +0 -0
- /package/dist/{run-BBXLRIZB.js.map → run-ST3FR75O.js.map} +0 -0
- /package/dist/{shell-RF7LTND5.js.map → shell-W4SBQPTE.js.map} +0 -0
- /package/dist/{summary-YZI25KW4.js.map → summary-P2JCIIJO.js.map} +0 -0
- /package/dist/{test-SGO6I5Z7.js.map → test-6JH4FE2X.js.map} +0 -0
- /package/dist/{test-git-XM4TM65W.js.map → test-git-2KFFAQ6B.js.map} +0 -0
- /package/dist/{test-jira-LDTOYFSD.js.map → test-jira-FKDKG6CD.js.map} +0 -0
- /package/dist/{test-tabs-D3POYOJ5.js.map → test-tabs-YDWMWTVA.js.map} +0 -0
- /package/dist/{test-webserver-NZ3JTVLL.js.map → test-webserver-QI3QQFZ3.js.map} +0 -0
- /package/dist/{update-HJKDYA3F.js.map → update-XLW7R7FL.js.map} +0 -0
- /package/dist/{vscode-6XUGHJKL.js.map → vscode-TOGE5N67.js.map} +0 -0
- /package/dist/{vscode-announcement-EQ2SKK3T.js.map → vscode-announcement-NIX7O2MG.js.map} +0 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/utils/claude-trust.ts"],"sourcesContent":["import lockfile from 'proper-lockfile'\nimport fs from 'fs-extra'\nimport path from 'path'\nimport os from 'os'\nimport crypto from 'crypto'\nimport { getLogger } from './logger-context.js'\n\nconst CLAUDE_CONFIG_PATH = path.join(os.homedir(), '.claude.json')\n\nconst LOCK_OPTIONS: lockfile.LockOptions = {\n\tretries: { retries: 5, minTimeout: 100, maxTimeout: 1000 },\n\trealpath: false, // ~/.claude.json may not exist yet; realpath would fail\n}\n\n/**\n * Pre-accept Claude Code trust for a worktree path by writing to ~/.claude.json.\n *\n * Sets `projects.<worktreePath>.hasTrustDialogAccepted = true` so that Claude Code\n * does not show the trust dialog when launching inside the worktree.\n *\n * Uses `proper-lockfile` to safely participate in Claude Code's own concurrency\n * protocol for ~/.claude.json writes.\n *\n * Throws on failure - callers should handle errors appropriately.\n */\nexport async function preAcceptClaudeTrust(worktreePath: string): Promise<void> {\n\tawait modifyClaudeConfig((config) => {\n\t\tif (!config.projects || typeof config.projects !== 'object' || Array.isArray(config.projects)) {\n\t\t\tconfig.projects = {}\n\t\t}\n\t\tconst projects = config.projects as Record<string, Record<string, unknown>>\n\t\tif (!projects[worktreePath] || typeof projects[worktreePath] !== 'object') {\n\t\t\tprojects[worktreePath] = {}\n\t\t}\n\t\tconst entry = projects[worktreePath]\n\t\tif (entry) {\n\t\t\tentry.hasTrustDialogAccepted = true\n\t\t}\n\t})\n}\n\n/**\n * Remove the Claude Code trust entry for a worktree path from ~/.claude.json.\n *\n * Called during worktree cleanup to prevent ~/.claude.json from accumulating\n * stale entries.\n *\n * Throws on failure - callers should handle errors appropriately.\n */\nexport async function removeClaudeTrust(worktreePath: string): Promise<void> {\n\tawait modifyClaudeConfig((config) => {\n\t\tif (!config.projects || typeof config.projects !== 'object') {\n\t\t\treturn\n\t\t}\n\t\tconst projects = config.projects as Record<string, unknown>\n\t\tdelete projects[worktreePath]\n\t})\n}\n\n/**\n * Shared helper that implements Claude Code's concurrency protocol:\n * 1. Ensure the config file exists (create empty JSON if missing)\n * 2. Acquire lock with proper-lockfile (with retries)\n * 3. Re-read file under lock (not from cache)\n * 4. Apply modifier function\n * 5. Atomic write: write to temp file, rename to target, mode 0o600\n * 6. Release lock\n *\n * Falls back to direct write if lock acquisition fails.\n */\nasync function modifyClaudeConfig(\n\tmodifier: (config: Record<string, unknown>) => void\n): Promise<void> {\n\tconst logger = getLogger()\n\n\t// Ensure the file exists before locking (proper-lockfile requires the file to exist)\n\tawait ensureClaudeConfigExists()\n\n\tlet release: (() => Promise<void>) | null = null\n\tlet lockAcquired = false\n\n\ttry {\n\t\trelease = await lockfile.lock(CLAUDE_CONFIG_PATH, LOCK_OPTIONS)\n\t\tlockAcquired = true\n\t} catch (lockError: unknown) {\n\t\tlogger.warn(\n\t\t\t`Could not acquire lock on ${CLAUDE_CONFIG_PATH}, falling back to direct write: ${lockError instanceof Error ? lockError.message : String(lockError)}`\n\t\t)\n\t}\n\n\ttry {\n\t\t// Re-read under lock (or without lock in fallback mode)\n\t\tconst config = await readClaudeConfig()\n\n\t\t// Apply modification\n\t\tmodifier(config)\n\n\t\t// Atomic write: temp file + rename, mode 0o600\n\t\tawait atomicWriteClaudeConfig(config)\n\t} finally {\n\t\tif (lockAcquired && release) {\n\t\t\ttry {\n\t\t\t\tawait release()\n\t\t\t} catch (unlockError: unknown) {\n\t\t\t\tlogger.debug(\n\t\t\t\t\t`Failed to release lock on ${CLAUDE_CONFIG_PATH}: ${unlockError instanceof Error ? unlockError.message : String(unlockError)}`\n\t\t\t\t)\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * Ensure ~/.claude.json exists. Creates an empty JSON object if missing.\n */\nasync function ensureClaudeConfigExists(): Promise<void> {\n\ttry {\n\t\tawait fs.access(CLAUDE_CONFIG_PATH)\n\t} catch (error: unknown) {\n\t\tif (error instanceof Error && 'code' in error && error.code === 'ENOENT') {\n\t\t\t// File doesn't exist - create with empty object and restricted permissions\n\t\t\tawait fs.writeFile(CLAUDE_CONFIG_PATH, '{}', { encoding: 'utf-8', mode: 0o600 })\n\t\t} else {\n\t\t\tthrow error\n\t\t}\n\t}\n}\n\n/**\n * Read and parse ~/.claude.json. Returns empty object for missing files or malformed JSON.\n * Re-throws other errors (e.g., permission denied).\n */\nasync function readClaudeConfig(): Promise<Record<string, unknown>> {\n\tconst logger = getLogger()\n\tlet content: string\n\ttry {\n\t\tcontent = await fs.readFile(CLAUDE_CONFIG_PATH, 'utf-8')\n\t} catch (error: unknown) {\n\t\tif (error instanceof Error && 'code' in error && error.code === 'ENOENT') {\n\t\t\treturn {}\n\t\t}\n\t\tthrow error\n\t}\n\ttry {\n\t\tconst parsed: unknown = JSON.parse(content)\n\t\tif (parsed && typeof parsed === 'object' && !Array.isArray(parsed)) {\n\t\t\treturn parsed as Record<string, unknown>\n\t\t}\n\t\treturn {}\n\t} catch (error: unknown) {\n\t\tif (error instanceof SyntaxError) {\n\t\t\tlogger.warn(`Malformed JSON in ${CLAUDE_CONFIG_PATH}, treating as empty config`)\n\t\t\treturn {}\n\t\t}\n\t\tthrow error\n\t}\n}\n\n/**\n * Atomically write config to ~/.claude.json via temp file + rename.\n * Sets file mode to 0o600 (owner read/write only).\n */\nasync function atomicWriteClaudeConfig(config: Record<string, unknown>): Promise<void> {\n\tconst tmpPath = `${CLAUDE_CONFIG_PATH}.${crypto.randomUUID()}.tmp`\n\tconst content = JSON.stringify(config, null, 2) + '\\n'\n\n\tawait fs.writeFile(tmpPath, content, { encoding: 'utf-8', mode: 0o600 })\n\tawait fs.rename(tmpPath, CLAUDE_CONFIG_PATH)\n}\n\n// Exported for testing only\nexport const _internal = {\n\tCLAUDE_CONFIG_PATH,\n\tmodifyClaudeConfig,\n\tensureClaudeConfigExists,\n\treadClaudeConfig,\n\tatomicWriteClaudeConfig,\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,10 +1,10 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
GitWorktreeManager
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-LE2NOUTN.js";
|
|
5
5
|
import {
|
|
6
6
|
logger
|
|
7
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-H2SSF24U.js";
|
|
8
8
|
|
|
9
9
|
// src/lib/ShellCompletion.ts
|
|
10
10
|
import omelette from "omelette";
|
|
@@ -299,4 +299,4 @@ Please consult your shell's documentation for setting up custom completions.
|
|
|
299
299
|
export {
|
|
300
300
|
ShellCompletion
|
|
301
301
|
};
|
|
302
|
-
//# sourceMappingURL=chunk-
|
|
302
|
+
//# sourceMappingURL=chunk-QNHZM5ZV.js.map
|
|
@@ -4,26 +4,26 @@ import {
|
|
|
4
4
|
} from "./chunk-NXMDEL3F.js";
|
|
5
5
|
import {
|
|
6
6
|
IssueManagementProviderFactory
|
|
7
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-RMLADZRY.js";
|
|
8
8
|
import {
|
|
9
9
|
hasMultipleRemotes
|
|
10
|
-
} from "./chunk-
|
|
10
|
+
} from "./chunk-QVAA5KHK.js";
|
|
11
11
|
import {
|
|
12
12
|
generateDeterministicSessionId,
|
|
13
13
|
launchClaude
|
|
14
|
-
} from "./chunk-
|
|
14
|
+
} from "./chunk-Y3RX7LZT.js";
|
|
15
15
|
import {
|
|
16
16
|
PromptTemplateManager
|
|
17
|
-
} from "./chunk-
|
|
17
|
+
} from "./chunk-WG4MLJ6J.js";
|
|
18
18
|
import {
|
|
19
19
|
SettingsManager
|
|
20
|
-
} from "./chunk-
|
|
20
|
+
} from "./chunk-ET6A2JR4.js";
|
|
21
21
|
import {
|
|
22
22
|
MetadataManager
|
|
23
|
-
} from "./chunk-
|
|
23
|
+
} from "./chunk-YRCEOQPX.js";
|
|
24
24
|
import {
|
|
25
25
|
logger
|
|
26
|
-
} from "./chunk-
|
|
26
|
+
} from "./chunk-H2SSF24U.js";
|
|
27
27
|
|
|
28
28
|
// src/lib/SessionSummaryService.ts
|
|
29
29
|
import path from "path";
|
|
@@ -369,4 +369,4 @@ var SessionSummaryService = class {
|
|
|
369
369
|
export {
|
|
370
370
|
SessionSummaryService
|
|
371
371
|
};
|
|
372
|
-
//# sourceMappingURL=chunk-
|
|
372
|
+
//# sourceMappingURL=chunk-QR4FU53I.js.map
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
logger_default
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-H2SSF24U.js";
|
|
5
5
|
|
|
6
6
|
// src/utils/remote.ts
|
|
7
7
|
import { execa } from "execa";
|
|
@@ -100,4 +100,4 @@ export {
|
|
|
100
100
|
validateConfiguredRemote,
|
|
101
101
|
getEffectivePRTargetRemote
|
|
102
102
|
};
|
|
103
|
-
//# sourceMappingURL=chunk-
|
|
103
|
+
//# sourceMappingURL=chunk-QVAA5KHK.js.map
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
logger
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-H2SSF24U.js";
|
|
5
5
|
|
|
6
6
|
// src/utils/update-notifier.ts
|
|
7
7
|
import os from "os";
|
|
@@ -218,4 +218,4 @@ export {
|
|
|
218
218
|
UpdateNotifier,
|
|
219
219
|
checkAndNotifyUpdate
|
|
220
220
|
};
|
|
221
|
-
//# sourceMappingURL=chunk-
|
|
221
|
+
//# sourceMappingURL=chunk-RBYTXYGD.js.map
|
|
@@ -2,9 +2,6 @@
|
|
|
2
2
|
import {
|
|
3
3
|
escapeJql
|
|
4
4
|
} from "./chunk-4232AHNQ.js";
|
|
5
|
-
import {
|
|
6
|
-
SettingsManager
|
|
7
|
-
} from "./chunk-7VHJNVLF.js";
|
|
8
5
|
import {
|
|
9
6
|
JiraIssueTracker,
|
|
10
7
|
createLinearChildIssue,
|
|
@@ -21,7 +18,7 @@ import {
|
|
|
21
18
|
getLinearIssueDependencies,
|
|
22
19
|
updateLinearComment,
|
|
23
20
|
updateLinearIssueState
|
|
24
|
-
} from "./chunk-
|
|
21
|
+
} from "./chunk-5LTID2AF.js";
|
|
25
22
|
import {
|
|
26
23
|
addSubIssue,
|
|
27
24
|
closeGhIssue,
|
|
@@ -38,10 +35,13 @@ import {
|
|
|
38
35
|
removeIssueDependency,
|
|
39
36
|
reopenGhIssue,
|
|
40
37
|
updateIssueComment
|
|
41
|
-
} from "./chunk-
|
|
38
|
+
} from "./chunk-NH3QZYE5.js";
|
|
39
|
+
import {
|
|
40
|
+
SettingsManager
|
|
41
|
+
} from "./chunk-ET6A2JR4.js";
|
|
42
42
|
import {
|
|
43
43
|
logger
|
|
44
|
-
} from "./chunk-
|
|
44
|
+
} from "./chunk-H2SSF24U.js";
|
|
45
45
|
|
|
46
46
|
// src/utils/image-processor.ts
|
|
47
47
|
import { tmpdir } from "os";
|
|
@@ -295,6 +295,9 @@ function normalizeAuthor(author) {
|
|
|
295
295
|
};
|
|
296
296
|
}
|
|
297
297
|
function extractNumericIdFromUrl(url) {
|
|
298
|
+
if (url == null) {
|
|
299
|
+
return void 0;
|
|
300
|
+
}
|
|
298
301
|
const match = url.match(/#issuecomment-(\d+)$/);
|
|
299
302
|
if (!(match == null ? void 0 : match[1])) {
|
|
300
303
|
throw new Error(`Cannot extract comment ID from URL: ${url}`);
|
|
@@ -352,7 +355,7 @@ var GitHubIssueManagementProvider = class {
|
|
|
352
355
|
};
|
|
353
356
|
if (raw.comments !== void 0) {
|
|
354
357
|
result.comments = raw.comments.map((comment) => ({
|
|
355
|
-
id: extractNumericIdFromUrl(comment.url),
|
|
358
|
+
id: extractNumericIdFromUrl(comment.url) ?? String(comment.id),
|
|
356
359
|
body: comment.body,
|
|
357
360
|
createdAt: comment.createdAt,
|
|
358
361
|
author: normalizeAuthor(comment.author),
|
|
@@ -426,7 +429,7 @@ var GitHubIssueManagementProvider = class {
|
|
|
426
429
|
};
|
|
427
430
|
if (raw.comments !== void 0) {
|
|
428
431
|
result.comments = raw.comments.map((comment) => ({
|
|
429
|
-
id: extractNumericIdFromUrl(comment.url),
|
|
432
|
+
id: extractNumericIdFromUrl(comment.url) ?? String(comment.id),
|
|
430
433
|
body: comment.body,
|
|
431
434
|
createdAt: comment.createdAt,
|
|
432
435
|
author: normalizeAuthor(comment.author),
|
|
@@ -1433,4 +1436,4 @@ var IssueManagementProviderFactory = class {
|
|
|
1433
1436
|
export {
|
|
1434
1437
|
IssueManagementProviderFactory
|
|
1435
1438
|
};
|
|
1436
|
-
//# sourceMappingURL=chunk-
|
|
1439
|
+
//# sourceMappingURL=chunk-RMLADZRY.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/utils/image-processor.ts","../src/mcp/GitHubIssueManagementProvider.ts","../src/utils/linear-markup-converter.ts","../src/mcp/LinearIssueManagementProvider.ts","../src/mcp/JiraIssueManagementProvider.ts","../src/mcp/IssueManagementProviderFactory.ts"],"sourcesContent":["/* global fetch, AbortController, setTimeout, clearTimeout */\nimport { tmpdir } from 'node:os'\nimport { join, extname } from 'node:path'\nimport { existsSync, mkdirSync, createWriteStream, unlinkSync } from 'node:fs'\nimport { pipeline } from 'node:stream/promises'\nimport { Readable } from 'node:stream'\nimport { createHash } from 'node:crypto'\nimport { execa } from 'execa'\nimport { logger } from './logger.js'\nimport type { IssueProvider } from '../mcp/types.js'\n\n/**\n * Represents a matched image in markdown content\n */\nexport interface ImageMatch {\n fullMatch: string\n url: string\n isMarkdown: boolean // true for , false for <img>\n}\n\n/**\n * Supported image extensions\n */\nconst SUPPORTED_EXTENSIONS = ['.png', '.jpg', '.jpeg', '.gif', '.webp', '.svg']\n\n/**\n * Maximum allowed image size (10MB)\n */\nconst MAX_IMAGE_SIZE = 10 * 1024 * 1024\n\n/**\n * Request timeout in milliseconds (30 seconds)\n */\nconst REQUEST_TIMEOUT_MS = 30000\n\n/**\n * Cache directory path for downloaded images\n */\nexport const CACHE_DIR = join(tmpdir(), 'iloom-images')\n\n/**\n * Cached GitHub auth token (module-level to avoid repeated `gh auth token` calls)\n */\nlet cachedGitHubToken: string | undefined\n\n/**\n * Extract all image URLs from markdown content\n * Handles both  and <img src=\"url\"> formats\n *\n * @param content - Markdown content to parse\n * @returns Array of image matches with full match string and URL\n */\nexport function extractMarkdownImageUrls(content: string): ImageMatch[] {\n if (!content) {\n return []\n }\n\n const matches: ImageMatch[] = []\n\n // Regex for markdown images: \n // Captures the entire match and the URL separately\n // Handles parentheses in URLs by matching balanced parens\n // The URL part matches: non-paren chars OR (balanced paren group)*, followed by non-paren/non-space chars\n const markdownRegex = /!\\[([^\\]]*)\\]\\(((?:[^()\\s]|\\((?:[^()\\s]|\\([^()]*\\))*\\))+)\\)/g\n let match: RegExpExecArray | null\n\n while ((match = markdownRegex.exec(content)) !== null) {\n const url = match[2]\n if (url) {\n matches.push({\n fullMatch: match[0],\n url,\n isMarkdown: true\n })\n }\n }\n\n // Regex for HTML img tags: <img ... src=\"url\" ...>\n // Handles both double and single quotes, and self-closing tags\n const htmlImgRegex = /<img\\s+[^>]*src=[\"']([^\"']+)[\"'][^>]*\\/?>/gi\n\n while ((match = htmlImgRegex.exec(content)) !== null) {\n const url = match[1]\n if (url) {\n matches.push({\n fullMatch: match[0],\n url,\n isMarkdown: false\n })\n }\n }\n\n return matches\n}\n\n/**\n * Check if URL requires authentication to download\n * - Linear: uploads.linear.app\n * - GitHub: private-user-images.githubusercontent.com\n *\n * @param url - Image URL to check\n * @returns true if URL requires authentication\n */\nexport function isAuthenticatedImageUrl(url: string): boolean {\n try {\n const parsedUrl = new URL(url)\n const hostname = parsedUrl.hostname.toLowerCase()\n\n // Linear uploads require authentication\n if (hostname === 'uploads.linear.app') {\n return true\n }\n\n // GitHub private user images require authentication\n if (hostname === 'private-user-images.githubusercontent.com') {\n return true\n }\n\n // GitHub user-attachments (uploaded images in issues/PRs) require authentication\n if (hostname === 'github.com' && parsedUrl.pathname.startsWith('/user-attachments/assets/')) {\n return true\n }\n\n return false\n } catch {\n // Invalid URL - treat as not authenticated\n return false\n }\n}\n\n/**\n * Get extension from URL pathname\n *\n * @param url - URL to extract extension from\n * @returns Extension including dot (e.g., '.png') or null if not found\n */\nfunction getExtensionFromUrl(url: string): string | null {\n try {\n const parsedUrl = new URL(url)\n const pathname = parsedUrl.pathname\n const ext = extname(pathname).toLowerCase()\n\n if (SUPPORTED_EXTENSIONS.includes(ext)) {\n return ext\n }\n return null\n } catch {\n return null\n }\n}\n\n/**\n * Generate cache key from URL\n * For GitHub URLs, strips JWT query params to ensure consistent caching\n * Returns hash + original extension\n *\n * @param url - Image URL to generate cache key for\n * @returns Cache key (hash + extension)\n */\nexport function getCacheKey(url: string): string {\n const parsedUrl = new URL(url)\n\n // For GitHub private images, remove jwt query param to get stable cache key\n // The jwt changes each fetch but the base URL is the same for the same image\n if (parsedUrl.hostname === 'private-user-images.githubusercontent.com') {\n parsedUrl.searchParams.delete('jwt')\n }\n\n // Get URL without volatile params for hashing\n const stableUrl = parsedUrl.toString()\n\n // Generate SHA256 hash of the stable URL (first 16 chars for brevity)\n const hash = createHash('sha256').update(stableUrl).digest('hex').slice(0, 16)\n\n // Extract extension from URL pathname, default to .png\n const ext = getExtensionFromUrl(url) ?? '.png'\n\n return `${hash}${ext}`\n}\n\n/**\n * Check if image is already cached\n * Returns file path if exists, undefined otherwise\n *\n * @param url - Image URL to check cache for\n * @returns Cached file path or undefined\n */\nexport function getCachedImagePath(url: string): string | undefined {\n const cacheKey = getCacheKey(url)\n const cachedPath = join(CACHE_DIR, cacheKey)\n\n if (existsSync(cachedPath)) {\n return cachedPath\n }\n return undefined\n}\n\n/**\n * Get authentication token for the given provider\n *\n * @param provider - Provider type ('github' or 'linear')\n * @returns Authentication token or undefined\n */\nasync function getAuthToken(provider: IssueProvider): Promise<string | undefined> {\n if (provider === 'github') {\n // Return cached token if available\n if (cachedGitHubToken !== undefined) {\n return cachedGitHubToken\n }\n\n try {\n // Execute `gh auth token` to get GitHub token\n const result = await execa('gh', ['auth', 'token'])\n cachedGitHubToken = result.stdout.trim()\n return cachedGitHubToken\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n logger.warn(`Failed to get GitHub auth token via gh CLI: ${message}`)\n return undefined\n }\n }\n\n if (provider === 'linear') {\n // Linear token from environment variable\n return process.env.LINEAR_API_TOKEN\n }\n\n return undefined\n}\n\n/**\n * Clear the cached GitHub auth token (for testing purposes)\n */\nexport function clearCachedGitHubToken(): void {\n cachedGitHubToken = undefined\n}\n\n/**\n * Download image from URL and stream it directly to a file\n *\n * @param url - Image URL to download\n * @param destPath - Destination file path\n * @param authHeader - Optional Authorization header value\n * @throws Error if download fails, times out, or exceeds size limit\n */\nexport async function downloadAndSaveImage(\n url: string,\n destPath: string,\n authHeader?: string\n): Promise<void> {\n const headers: Record<string, string> = {}\n if (authHeader) {\n headers['Authorization'] = authHeader\n }\n\n // Set up abort controller for timeout\n const controller = new AbortController()\n const timeoutId = setTimeout(() => controller.abort(), REQUEST_TIMEOUT_MS)\n\n try {\n const response = await fetch(url, { headers, signal: controller.signal })\n\n if (!response.ok) {\n throw new Error(`Failed to download image: ${response.status} ${response.statusText}`)\n }\n\n // Check Content-Length header if available\n const contentLength = response.headers.get('Content-Length')\n if (contentLength && parseInt(contentLength, 10) > MAX_IMAGE_SIZE) {\n throw new Error(`Image too large: ${contentLength} bytes exceeds ${MAX_IMAGE_SIZE} byte limit`)\n }\n\n if (!response.body) {\n throw new Error('Response body is null')\n }\n\n // Convert ReadableStream to Node.js Readable\n const reader = response.body.getReader()\n let bytesWritten = 0\n\n const nodeReadable = new Readable({\n async read(): Promise<void> {\n try {\n const { done, value } = await reader.read()\n if (done) {\n this.push(null)\n return\n }\n\n bytesWritten += value.byteLength\n if (bytesWritten > MAX_IMAGE_SIZE) {\n reader.cancel()\n this.destroy(new Error(`Image too large: ${bytesWritten} bytes exceeds ${MAX_IMAGE_SIZE} byte limit`))\n return\n }\n\n this.push(Buffer.from(value))\n } catch (err) {\n this.destroy(err instanceof Error ? err : new Error(String(err)))\n }\n }\n })\n\n // Ensure cache directory exists\n if (!existsSync(CACHE_DIR)) {\n mkdirSync(CACHE_DIR, { recursive: true })\n }\n\n // Stream to file\n const writeStream = createWriteStream(destPath)\n\n try {\n await pipeline(nodeReadable, writeStream)\n } catch (pipelineError) {\n // Clean up partial file on error\n try {\n if (existsSync(destPath)) {\n unlinkSync(destPath)\n }\n } catch {\n // Ignore cleanup errors\n }\n throw pipelineError\n }\n } catch (error) {\n if (error instanceof Error && error.name === 'AbortError') {\n throw new Error(`Image download timed out after ${REQUEST_TIMEOUT_MS}ms`)\n }\n throw error\n } finally {\n clearTimeout(timeoutId)\n }\n}\n\n/**\n * Get the destination path for caching an image\n *\n * @param url - Original image URL (used to generate cache key)\n * @returns Local file path where image should be saved\n */\nexport function getCacheDestPath(url: string): string {\n // Ensure cache directory exists\n if (!existsSync(CACHE_DIR)) {\n mkdirSync(CACHE_DIR, { recursive: true })\n }\n\n // Generate cache key from URL\n const cacheKey = getCacheKey(url)\n return join(CACHE_DIR, cacheKey)\n}\n\n/**\n * Rewrite image URLs in markdown content\n *\n * @param content - Original markdown content\n * @param urlMap - Map of original URLs to local file paths\n * @returns Content with URLs replaced\n */\nexport function rewriteMarkdownUrls(\n content: string,\n urlMap: Map<string, string>\n): string {\n let result = content\n\n for (const [originalUrl, localPath] of urlMap) {\n // Escape special regex characters in the URL\n const escapedUrl = originalUrl.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n const urlRegex = new RegExp(escapedUrl, 'g')\n result = result.replace(urlRegex, localPath)\n }\n\n return result\n}\n\n/**\n * Main entry point: process all images in markdown content\n * Downloads authenticated images (with caching), saves locally, rewrites URLs\n *\n * @param content - Markdown content to process\n * @param provider - Image provider for authentication ('github' or 'linear')\n * @returns Content with authenticated image URLs replaced with local file paths\n */\nexport async function processMarkdownImages(\n content: string,\n provider: IssueProvider\n): Promise<string> {\n // Early return if empty\n if (!content) {\n return ''\n }\n\n // Extract all image URLs\n const images = extractMarkdownImageUrls(content)\n if (images.length === 0) {\n return content\n }\n\n // Filter to only authenticated URLs\n const authImages = images.filter(img => isAuthenticatedImageUrl(img.url))\n if (authImages.length === 0) {\n return content\n }\n\n // Get auth token for provider\n const authToken = await getAuthToken(provider)\n\n // Deduplicate URLs (same image might appear multiple times)\n const uniqueUrls = [...new Set(authImages.map(img => img.url))]\n\n // Build URL map - process all unique URLs in parallel\n const urlMap = new Map<string, string>()\n\n // Download/cache images in parallel\n const downloadPromises = uniqueUrls.map(async (url) => {\n try {\n // Check cache first\n const cachedPath = getCachedImagePath(url)\n if (cachedPath) {\n logger.debug(`Using cached image: ${cachedPath}`)\n return { url, localPath: cachedPath }\n }\n\n // Cache miss - download and stream directly to file\n logger.debug(`Downloading image: ${url}`)\n const destPath = getCacheDestPath(url)\n await downloadAndSaveImage(\n url,\n destPath,\n authToken ? `Bearer ${authToken}` : undefined\n )\n return { url, localPath: destPath }\n } catch (error) {\n // Graceful degradation - log warning, return null to keep original URL\n const message = error instanceof Error ? error.message : String(error)\n logger.warn(`Failed to download image ${url}: ${message}`)\n return null\n }\n })\n\n const results = await Promise.all(downloadPromises)\n\n // Build URL map from results\n for (const result of results) {\n if (result !== null) {\n urlMap.set(result.url, result.localPath)\n }\n }\n\n // Rewrite and return\n return rewriteMarkdownUrls(content, urlMap)\n}\n","/**\n * GitHub implementation of Issue Management Provider\n * Uses GitHub CLI for all operations\n * Normalizes GitHub-specific fields (login) to provider-agnostic core fields (id, displayName)\n */\n\nimport type {\n\tIssueManagementProvider,\n\tGetIssueInput,\n\tGetPRInput,\n\tGetReviewCommentsInput,\n\tGetCommentInput,\n\tCreateCommentInput,\n\tUpdateCommentInput,\n\tCreateIssueInput,\n\tCreateChildIssueInput,\n\tCreateDependencyInput,\n\tGetDependenciesInput,\n\tRemoveDependencyInput,\n\tGetChildIssuesInput,\n\tCloseIssueInput,\n\tReopenIssueInput,\n\tEditIssueInput,\n\tCreateIssueResult,\n\tIssueResult,\n\tPRResult,\n\tReviewCommentResult,\n\tCommentDetailResult,\n\tCommentResult,\n\tDependenciesResult,\n\tChildIssueResult,\n\tFlexibleAuthor,\n} from './types.js'\nimport {\n\texecuteGhCommand,\n\tcreateIssueComment,\n\tupdateIssueComment,\n\tcreatePRComment,\n\tcreateIssue,\n\tgetIssueNodeId,\n\taddSubIssue,\n\tgetIssueDatabaseId,\n\tgetIssueDependencies,\n\tcreateIssueDependency,\n\tremoveIssueDependency,\n\tgetSubIssues,\n\tcloseGhIssue,\n\treopenGhIssue,\n\teditGhIssue,\n} from '../utils/github.js'\nimport { processMarkdownImages } from '../utils/image-processor.js'\n\n/**\n * GitHub-specific author structure from API\n */\ninterface GitHubAuthor {\n\tlogin: string\n\tid?: number\n\tavatarUrl?: string\n\turl?: string\n}\n\n/**\n * Normalize GitHub author to FlexibleAuthor format\n */\nfunction normalizeAuthor(author: GitHubAuthor | null | undefined): FlexibleAuthor | null {\n\tif (!author) return null\n\n\treturn {\n\t\tid: author.id ? String(author.id) : author.login,\n\t\tdisplayName: author.login, // GitHub uses login as primary identifier\n\t\tlogin: author.login, // Preserve original GitHub field\n\t\t...(author.avatarUrl && { avatarUrl: author.avatarUrl }),\n\t\t...(author.url && { url: author.url }),\n\t}\n}\n\n/**\n * Extract numeric comment ID from GitHub comment URL\n * URL format: https://github.com/owner/repo/issues/123#issuecomment-3615239386\n *\n * Returns undefined if URL is null/undefined (older gh CLI versions don't include url).\n * Throws if URL is present but has no valid issuecomment fragment.\n */\nexport function extractNumericIdFromUrl(url: string | null | undefined): string | undefined {\n\tif (url == null) {\n\t\treturn undefined\n\t}\n\tconst match = url.match(/#issuecomment-(\\d+)$/)\n\tif (!match?.[1]) {\n\t\tthrow new Error(`Cannot extract comment ID from URL: ${url}`)\n\t}\n\treturn match[1]\n}\n\n/**\n * GitHub-specific implementation of IssueManagementProvider\n */\nexport class GitHubIssueManagementProvider implements IssueManagementProvider {\n\treadonly providerName = 'github'\n\treadonly issuePrefix = '#'\n\n\t/**\n\t * Fetch issue details using gh CLI\n\t * Normalizes GitHub-specific fields to provider-agnostic format\n\t */\n\tasync getIssue(input: GetIssueInput): Promise<IssueResult> {\n\t\tconst { number, includeComments = true, repo } = input\n\n\t\t// Convert string ID to number for GitHub CLI\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\t// Build fields list based on whether we need comments\n\t\tconst fields = includeComments\n\t\t\t? 'body,title,comments,labels,assignees,milestone,author,state,number,url'\n\t\t\t: 'body,title,labels,assignees,milestone,author,state,number,url'\n\n\t\t// Use gh issue view to fetch issue details\n\t\tinterface GitHubIssueResponse {\n\t\t\tnumber: number\n\t\t\ttitle: string\n\t\t\tbody: string\n\t\t\tstate: string\n\t\t\turl: string\n\t\t\tauthor?: GitHubAuthor\n\t\t\tlabels?: Array<{ name: string; color?: string; description?: string }>\n\t\t\tassignees?: Array<GitHubAuthor>\n\t\t\tmilestone?: { title: string; number?: number; state?: string }\n\t\t\tcomments?: Array<{\n\t\t\t\tid: number\n\t\t\t\tauthor: GitHubAuthor\n\t\t\t\tbody: string\n\t\t\t\tcreatedAt: string\n\t\t\t\tupdatedAt?: string\n\t\t\t\turl?: string // Optional: older gh CLI versions (e.g. v2.4.0) don't include url\n\t\t\t}>\n\t\t}\n\n\t\tconst args = [\n\t\t\t'issue',\n\t\t\t'view',\n\t\t\tString(issueNumber),\n\t\t\t'--json',\n\t\t\tfields,\n\t\t]\n\n\t\t// Add --repo flag if repo is provided (gh CLI handles both owner/repo and URL formats)\n\t\tif (repo) {\n\t\t\targs.push('--repo', repo)\n\t\t}\n\n\t\tconst raw = await executeGhCommand<GitHubIssueResponse>(args)\n\n\t\t// Normalize to IssueResult with core fields + passthrough\n\t\tconst result: IssueResult = {\n\t\t\t// Core fields\n\t\t\tid: String(raw.number),\n\t\t\ttitle: raw.title,\n\t\t\tbody: raw.body,\n\t\t\tstate: raw.state,\n\t\t\turl: raw.url,\n\t\t\tprovider: 'github',\n\n\t\t\t// Normalized author\n\t\t\tauthor: normalizeAuthor(raw.author),\n\n\t\t\t// Optional flexible fields\n\t\t\t...(raw.assignees && {\n\t\t\t\tassignees: raw.assignees.map(a => normalizeAuthor(a)).filter((a): a is FlexibleAuthor => a !== null),\n\t\t\t}),\n\t\t\t...(raw.labels && {\n\t\t\t\tlabels: raw.labels,\n\t\t\t}),\n\n\t\t\t// GitHub-specific passthrough fields\n\t\t\t...(raw.milestone && {\n\t\t\t\tmilestone: raw.milestone,\n\t\t\t}),\n\t\t}\n\n\t\t// Handle comments with normalized authors\n\t\t// Use extractNumericIdFromUrl to get REST API-compatible numeric IDs from comment URLs\n\t\t// (GitHub CLI returns GraphQL node IDs in the id field, but REST API expects numeric IDs)\n\t\t// Fallback: older gh CLI versions (e.g. v2.4.0) don't include url, so use comment.id directly\n\t\tif (raw.comments !== undefined) {\n\t\t\tresult.comments = raw.comments.map(comment => ({\n\t\t\t\tid: extractNumericIdFromUrl(comment.url) ?? String(comment.id),\n\t\t\t\tbody: comment.body,\n\t\t\t\tcreatedAt: comment.createdAt,\n\t\t\t\tauthor: normalizeAuthor(comment.author),\n\t\t\t\t...(comment.updatedAt && { updatedAt: comment.updatedAt }),\n\t\t\t}))\n\t\t}\n\n\t\t// Process authenticated images in body and comments\n\t\tresult.body = await processMarkdownImages(result.body, 'github')\n\t\tif (result.comments) {\n\t\t\tfor (const comment of result.comments) {\n\t\t\t\tcomment.body = await processMarkdownImages(comment.body, 'github')\n\t\t\t}\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Fetch pull request details using gh CLI\n\t * Normalizes GitHub-specific fields to provider-agnostic format\n\t */\n\tasync getPR(input: GetPRInput): Promise<PRResult> {\n\t\tconst { number, includeComments = true, repo } = input\n\n\t\t// Convert string ID to number for GitHub CLI\n\t\tconst prNumber = parseInt(number, 10)\n\t\tif (isNaN(prNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub PR number: ${number}. GitHub PR IDs must be numeric.`)\n\t\t}\n\n\t\t// Build fields list based on whether we need comments\n\t\tconst baseFields = 'number,title,body,state,url,author,headRefName,baseRefName,files,commits'\n\t\tconst fields = includeComments\n\t\t\t? `${baseFields},comments`\n\t\t\t: baseFields\n\n\t\t// GitHub PR response structure\n\t\tinterface GitHubPRResponse {\n\t\t\tnumber: number\n\t\t\ttitle: string\n\t\t\tbody: string\n\t\t\tstate: string\n\t\t\turl: string\n\t\t\tauthor?: GitHubAuthor\n\t\t\theadRefName: string\n\t\t\tbaseRefName: string\n\t\t\tfiles?: Array<{\n\t\t\t\tpath: string\n\t\t\t\tadditions: number\n\t\t\t\tdeletions: number\n\t\t\t}>\n\t\t\tcommits?: Array<{\n\t\t\t\toid: string\n\t\t\t\tmessageHeadline: string\n\t\t\t\tauthors: Array<{ name: string; email: string }>\n\t\t\t}>\n\t\t\tcomments?: Array<{\n\t\t\t\tid: number\n\t\t\t\tauthor: GitHubAuthor\n\t\t\t\tbody: string\n\t\t\t\tcreatedAt: string\n\t\t\t\tupdatedAt?: string\n\t\t\t\turl?: string // Optional: older gh CLI versions (e.g. v2.4.0) don't include url\n\t\t\t}>\n\t\t}\n\n\t\tconst args = [\n\t\t\t'pr',\n\t\t\t'view',\n\t\t\tString(prNumber),\n\t\t\t'--json',\n\t\t\tfields,\n\t\t]\n\n\t\t// Add --repo flag if repo is provided\n\t\tif (repo) {\n\t\t\targs.push('--repo', repo)\n\t\t}\n\n\t\tconst raw = await executeGhCommand<GitHubPRResponse>(args)\n\n\t\t// Normalize to PRResult with core fields + passthrough\n\t\tconst result: PRResult = {\n\t\t\t// Core fields\n\t\t\tid: String(raw.number),\n\t\t\tnumber: raw.number,\n\t\t\ttitle: raw.title,\n\t\t\tbody: raw.body,\n\t\t\tstate: raw.state,\n\t\t\turl: raw.url,\n\n\t\t\t// Normalized author\n\t\t\tauthor: normalizeAuthor(raw.author),\n\n\t\t\t// PR-specific fields\n\t\t\theadRefName: raw.headRefName,\n\t\t\tbaseRefName: raw.baseRefName,\n\n\t\t\t// Optional files\n\t\t\t...(raw.files && {\n\t\t\t\tfiles: raw.files,\n\t\t\t}),\n\n\t\t\t// Optional commits - normalize author\n\t\t\t...(raw.commits && {\n\t\t\t\tcommits: raw.commits.map(commit => ({\n\t\t\t\t\toid: commit.oid,\n\t\t\t\t\tmessageHeadline: commit.messageHeadline,\n\t\t\t\t\tauthor: commit.authors?.[0]\n\t\t\t\t\t\t? {\n\t\t\t\t\t\t\tid: commit.authors[0].email,\n\t\t\t\t\t\t\tdisplayName: commit.authors[0].name,\n\t\t\t\t\t\t\tname: commit.authors[0].name,\n\t\t\t\t\t\t\temail: commit.authors[0].email,\n\t\t\t\t\t\t}\n\t\t\t\t\t\t: null,\n\t\t\t\t})),\n\t\t\t}),\n\t\t}\n\n\t\t// Handle comments with normalized authors\n\t\t// Use extractNumericIdFromUrl to get REST API-compatible numeric IDs from comment URLs\n\t\t// Fallback: older gh CLI versions (e.g. v2.4.0) don't include url, so use comment.id directly\n\t\tif (raw.comments !== undefined) {\n\t\t\tresult.comments = raw.comments.map(comment => ({\n\t\t\t\tid: extractNumericIdFromUrl(comment.url) ?? String(comment.id),\n\t\t\t\tbody: comment.body,\n\t\t\t\tcreatedAt: comment.createdAt,\n\t\t\t\tauthor: normalizeAuthor(comment.author),\n\t\t\t\t...(comment.updatedAt && { updatedAt: comment.updatedAt }),\n\t\t\t}))\n\t\t}\n\n\t\t// Process authenticated images in body and comments\n\t\tresult.body = await processMarkdownImages(result.body, 'github')\n\t\tif (result.comments) {\n\t\t\tfor (const comment of result.comments) {\n\t\t\t\tcomment.body = await processMarkdownImages(comment.body, 'github')\n\t\t\t}\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Fetch PR review comments (inline code comments on specific files/lines)\n\t * Uses gh api with --paginate to handle PRs with many review comments\n\t * Optionally filters by review ID\n\t */\n\tasync getReviewComments(input: GetReviewCommentsInput): Promise<ReviewCommentResult[]> {\n\t\tconst { number, reviewId, repo } = input\n\n\t\t// Convert string ID to number for GitHub API\n\t\tconst prNumber = parseInt(number, 10)\n\t\tif (isNaN(prNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub PR number: ${number}. GitHub PR IDs must be numeric.`)\n\t\t}\n\n\t\t// Validate reviewId early to avoid unnecessary API call\n\t\tlet numericReviewId: number | undefined\n\t\tif (reviewId) {\n\t\t\tnumericReviewId = parseInt(reviewId, 10)\n\t\t\tif (isNaN(numericReviewId)) {\n\t\t\t\tthrow new Error(`Invalid review ID: ${reviewId}. Review IDs must be numeric.`)\n\t\t\t}\n\t\t}\n\n\t\t// GitHub API response structure for review comments\n\t\tinterface GitHubReviewComment {\n\t\t\tid: number\n\t\t\tbody: string\n\t\t\tpath: string\n\t\t\tline: number | null\n\t\t\tside: string | null\n\t\t\tuser: GitHubAuthor | null\n\t\t\tcreated_at: string\n\t\t\tupdated_at: string | null\n\t\t\tin_reply_to_id: number | null\n\t\t\tpull_request_review_id: number | null\n\t\t}\n\n\t\t// Use explicit repo path if provided, otherwise use :owner/:repo placeholder\n\t\tconst apiPath = repo\n\t\t\t? `repos/${repo}/pulls/${prNumber}/comments`\n\t\t\t: `repos/:owner/:repo/pulls/${prNumber}/comments`\n\n\t\tconst args = [\n\t\t\t'api',\n\t\t\tapiPath,\n\t\t\t'--paginate',\n\t\t\t'--jq',\n\t\t\t'[.[] | {id: .id, body: .body, path: .path, line: .line, side: .side, user: .user, created_at: .created_at, updated_at: .updated_at, in_reply_to_id: .in_reply_to_id, pull_request_review_id: .pull_request_review_id}]',\n\t\t]\n\n\t\tconst raw = await executeGhCommand<GitHubReviewComment[]>(args)\n\n\t\t// Filter by reviewId if provided (already validated above)\n\t\tlet comments = raw\n\t\tif (numericReviewId !== undefined) {\n\t\t\tcomments = comments.filter(c => c.pull_request_review_id === numericReviewId)\n\t\t}\n\n\t\t// Normalize and process each comment\n\t\tconst results: ReviewCommentResult[] = []\n\t\tfor (const comment of comments) {\n\t\t\tconst processedBody = await processMarkdownImages(comment.body, 'github')\n\t\t\tresults.push({\n\t\t\t\tid: String(comment.id),\n\t\t\t\tbody: processedBody,\n\t\t\t\tpath: comment.path,\n\t\t\t\tline: comment.line,\n\t\t\t\tside: comment.side,\n\t\t\t\tauthor: normalizeAuthor(comment.user),\n\t\t\t\tcreatedAt: comment.created_at,\n\t\t\t\tupdatedAt: comment.updated_at ?? null,\n\t\t\t\tinReplyToId: comment.in_reply_to_id ? String(comment.in_reply_to_id) : null,\n\t\t\t\tpullRequestReviewId: comment.pull_request_review_id,\n\t\t\t})\n\t\t}\n\n\t\treturn results\n\t}\n\n\t/**\n\t * Fetch a specific comment by ID using gh API\n\t * Normalizes author to FlexibleAuthor format\n\t */\n\tasync getComment(input: GetCommentInput): Promise<CommentDetailResult> {\n\t\tconst { commentId, repo } = input\n\t\t// Note: GitHub doesn't need the issue number parameter - comment IDs are globally unique\n\t\t// But we accept it for interface compatibility with other providers\n\n\t\t// Convert string ID to number for GitHub API\n\t\tconst numericCommentId = parseInt(commentId, 10)\n\t\tif (isNaN(numericCommentId)) {\n\t\t\tthrow new Error(`Invalid GitHub comment ID: ${commentId}. GitHub comment IDs must be numeric.`)\n\t\t}\n\n\t\t// GitHub API response structure\n\t\tinterface GitHubCommentResponse {\n\t\t\tid: number\n\t\t\tbody: string\n\t\t\tuser: GitHubAuthor\n\t\t\tcreated_at: string\n\t\t\tupdated_at?: string\n\t\t\thtml_url?: string\n\t\t\treactions?: Record<string, unknown>\n\t\t}\n\n\t\t// Use explicit repo path if provided, otherwise use :owner/:repo placeholder\n\t\tconst apiPath = repo\n\t\t\t? `repos/${repo}/issues/comments/${numericCommentId}`\n\t\t\t: `repos/:owner/:repo/issues/comments/${numericCommentId}`\n\n\t\t// Use gh api to fetch specific comment\n\t\tconst raw = await executeGhCommand<GitHubCommentResponse>([\n\t\t\t'api',\n\t\t\tapiPath,\n\t\t\t'--jq',\n\t\t\t'{id: .id, body: .body, user: .user, created_at: .created_at, updated_at: .updated_at, html_url: .html_url, reactions: .reactions}',\n\t\t])\n\n\t\t// Process authenticated images in comment body\n\t\tconst processedBody = await processMarkdownImages(raw.body, 'github')\n\n\t\t// Normalize to CommentDetailResult\n\t\treturn {\n\t\t\tid: String(raw.id),\n\t\t\tbody: processedBody,\n\t\t\tauthor: normalizeAuthor(raw.user),\n\t\t\tcreated_at: raw.created_at,\n\t\t\t...(raw.updated_at && { updated_at: raw.updated_at }),\n\t\t\t// Passthrough GitHub-specific fields\n\t\t\t...(raw.html_url && { html_url: raw.html_url }),\n\t\t\t...(raw.reactions && { reactions: raw.reactions }),\n\t\t}\n\t}\n\n\t/**\n\t * Create a new comment on an issue or PR\n\t */\n\tasync createComment(input: CreateCommentInput): Promise<CommentResult> {\n\t\tconst { number, body, type } = input\n\n\t\t// Convert string ID to number for GitHub utilities\n\t\tconst numericId = parseInt(number, 10)\n\t\tif (isNaN(numericId)) {\n\t\t\tthrow new Error(`Invalid GitHub ${type} number: ${number}. GitHub IDs must be numeric.`)\n\t\t}\n\n\t\t// Delegate to existing GitHub utilities\n\t\tconst result =\n\t\t\ttype === 'issue'\n\t\t\t\t? await createIssueComment(numericId, body)\n\t\t\t\t: await createPRComment(numericId, body)\n\n\t\t// Convert numeric ID to string for the interface\n\t\treturn {\n\t\t\t...result,\n\t\t\tid: String(result.id),\n\t\t}\n\t}\n\n\t/**\n\t * Update an existing comment\n\t */\n\tasync updateComment(input: UpdateCommentInput): Promise<CommentResult> {\n\t\tconst { commentId, body } = input\n\t\t// Note: GitHub doesn't need the issue number parameter - comment IDs are globally unique\n\t\t// But we accept it for interface compatibility with other providers\n\n\t\t// Convert string ID to number for GitHub utility\n\t\tconst numericCommentId = parseInt(commentId, 10)\n\t\tif (isNaN(numericCommentId)) {\n\t\t\tthrow new Error(`Invalid GitHub comment ID: ${commentId}. GitHub comment IDs must be numeric.`)\n\t\t}\n\n\t\t// Delegate to existing GitHub utility\n\t\tconst result = await updateIssueComment(numericCommentId, body)\n\n\t\t// Convert numeric ID to string for the interface\n\t\treturn {\n\t\t\t...result,\n\t\t\tid: String(result.id),\n\t\t}\n\t}\n\n\t/**\n\t * Create a new issue\n\t */\n\tasync createIssue(input: CreateIssueInput): Promise<CreateIssueResult> {\n\t\tconst { title, body, labels, repo } = input\n\t\t// teamKey is ignored for GitHub\n\n\t\tconst result = await createIssue(title, body, { labels, repo })\n\n\t\t// Ensure number is numeric\n\t\tconst issueNumber = typeof result.number === 'number'\n\t\t\t? result.number\n\t\t\t: parseInt(String(result.number), 10)\n\n\t\treturn {\n\t\t\tid: String(issueNumber),\n\t\t\turl: result.url,\n\t\t\tnumber: issueNumber,\n\t\t}\n\t}\n\n\t/**\n\t * Create a child issue linked to a parent issue\n\t * GitHub requires two-step process: create issue, then link via GraphQL\n\t */\n\tasync createChildIssue(input: CreateChildIssueInput): Promise<CreateIssueResult> {\n\t\tconst { parentId, title, body, labels, repo } = input\n\t\t// teamKey is ignored for GitHub\n\n\t\t// Convert parent identifier to number\n\t\tconst parentNumber = parseInt(parentId, 10)\n\t\tif (isNaN(parentNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub parent issue number: ${parentId}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\t// Step 1: Get parent issue's GraphQL node ID\n\t\tconst parentNodeId = await getIssueNodeId(parentNumber, repo)\n\n\t\t// Step 2: Create the child issue\n\t\tconst childResult = await createIssue(title, body, { labels, repo })\n\t\tconst childNumber = typeof childResult.number === 'number'\n\t\t\t? childResult.number\n\t\t\t: parseInt(String(childResult.number), 10)\n\n\t\t// Step 3: Get child issue's GraphQL node ID\n\t\tconst childNodeId = await getIssueNodeId(childNumber, repo)\n\n\t\t// Step 4: Link child to parent via GraphQL mutation\n\t\tawait addSubIssue(parentNodeId, childNodeId)\n\n\t\treturn {\n\t\t\tid: String(childNumber),\n\t\t\turl: childResult.url,\n\t\t\tnumber: childNumber,\n\t\t}\n\t}\n\n\t/**\n\t * Create a blocking dependency between two issues (A blocks B)\n\t * Uses GitHub's sub-issues API: blocking issue becomes parent, blocked issue becomes sub-issue\n\t */\n\tasync createDependency(input: CreateDependencyInput): Promise<void> {\n\t\tconst { blockingIssue, blockedIssue, repo } = input\n\n\t\t// Convert string IDs to numbers\n\t\tconst blockingNumber = parseInt(blockingIssue, 10)\n\t\tif (isNaN(blockingNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${blockingIssue}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\tconst blockedNumber = parseInt(blockedIssue, 10)\n\t\tif (isNaN(blockedNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${blockedIssue}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\t// Get the database ID of the blocking issue\n\t\t// GitHub API: POST /issues/{blocked_issue_number}/dependencies/blocked_by with body issue_id={blocking_database_id}\n\t\tconst blockingDatabaseId = await getIssueDatabaseId(blockingNumber, repo)\n\n\t\t// Create the dependency: path uses blocked issue number, body uses blocking issue DB ID\n\t\tawait createIssueDependency(blockedNumber, blockingDatabaseId, repo)\n\t}\n\n\t/**\n\t * Get dependencies for an issue\n\t */\n\tasync getDependencies(input: GetDependenciesInput): Promise<DependenciesResult> {\n\t\tconst { number, direction, repo } = input\n\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\tconst result: DependenciesResult = {\n\t\t\tblocking: [],\n\t\t\tblockedBy: [],\n\t\t}\n\n\t\t// Fetch dependencies based on direction\n\t\tif (direction === 'blocking' || direction === 'both') {\n\t\t\tresult.blocking = await getIssueDependencies(issueNumber, 'blocking', repo)\n\t\t}\n\n\t\tif (direction === 'blocked_by' || direction === 'both') {\n\t\t\tresult.blockedBy = await getIssueDependencies(issueNumber, 'blocked_by', repo)\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Remove a blocking dependency between two issues (A blocks B)\n\t * Uses GitHub's sub-issues API: blocking issue is parent, blocked issue is sub-issue\n\t */\n\tasync removeDependency(input: RemoveDependencyInput): Promise<void> {\n\t\tconst { blockingIssue, blockedIssue, repo } = input\n\n\t\t// Convert string IDs to numbers\n\t\tconst blockingNumber = parseInt(blockingIssue, 10)\n\t\tif (isNaN(blockingNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${blockingIssue}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\tconst blockedNumber = parseInt(blockedIssue, 10)\n\t\tif (isNaN(blockedNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${blockedIssue}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\t// Get the database ID of the blocking issue\n\t\t// GitHub API: DELETE /issues/{blocked_issue_number}/dependencies/blocked_by with body issue_id={blocking_database_id}\n\t\tconst blockingDatabaseId = await getIssueDatabaseId(blockingNumber, repo)\n\n\t\t// Remove the dependency: path uses blocked issue number, body uses blocking issue DB ID\n\t\tawait removeIssueDependency(blockedNumber, blockingDatabaseId, repo)\n\t}\n\n\t/**\n\t * Get child issues (sub-issues) of a parent issue\n\t */\n\tasync getChildIssues(input: GetChildIssuesInput): Promise<ChildIssueResult[]> {\n\t\tconst { number, repo } = input\n\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\treturn await getSubIssues(issueNumber, repo)\n\t}\n\n\t/**\n\t * Close an issue\n\t */\n\tasync closeIssue(input: CloseIssueInput): Promise<void> {\n\t\tconst { number, repo } = input\n\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\tawait closeGhIssue(issueNumber, repo)\n\t}\n\n\t/**\n\t * Reopen a closed issue\n\t */\n\tasync reopenIssue(input: ReopenIssueInput): Promise<void> {\n\t\tconst { number, repo } = input\n\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\tawait reopenGhIssue(issueNumber, repo)\n\t}\n\n\t/**\n\t * Edit an issue's properties\n\t * State changes are delegated to closeIssue/reopenIssue\n\t */\n\tasync editIssue(input: EditIssueInput): Promise<void> {\n\t\tconst { number, title, body, state, labels, repo } = input\n\n\t\tconst issueNumber = parseInt(number, 10)\n\t\tif (isNaN(issueNumber)) {\n\t\t\tthrow new Error(`Invalid GitHub issue number: ${number}. GitHub issue IDs must be numeric.`)\n\t\t}\n\n\t\t// Handle state changes via close/reopen\n\t\tif (state === 'closed') {\n\t\t\tawait this.closeIssue({ number, repo })\n\t\t} else if (state === 'open') {\n\t\t\tawait this.reopenIssue({ number, repo })\n\t\t}\n\n\t\t// Handle other field updates\n\t\tif (title !== undefined || body !== undefined || labels !== undefined) {\n\t\t\tawait editGhIssue(\n\t\t\t\tissueNumber,\n\t\t\t\t{\n\t\t\t\t\t...(title !== undefined && { title }),\n\t\t\t\t\t...(body !== undefined && { body }),\n\t\t\t\t\t...(labels !== undefined && { labels }),\n\t\t\t\t},\n\t\t\t\trepo\n\t\t\t)\n\t\t}\n\t}\n}\n","import { appendFileSync } from 'node:fs'\nimport { join, dirname, basename, extname } from 'node:path'\n\n/**\n * Utility class for converting HTML details/summary format to Linear's collapsible format\n *\n * Converts:\n * <details>\n * <summary>Header</summary>\n * CONTENT\n * </details>\n *\n * Into Linear format:\n * +++ Header\n *\n * CONTENT\n *\n * +++\n */\nexport class LinearMarkupConverter {\n\t/**\n\t * Convert HTML details/summary blocks to Linear's collapsible format\n\t * Handles nested details blocks recursively\n\t *\n\t * @param text - Text containing HTML details/summary blocks\n\t * @returns Text with details/summary converted to Linear format\n\t */\n\tstatic convertDetailsToLinear(text: string): string {\n\t\tif (!text) {\n\t\t\treturn text\n\t\t}\n\n\t\t// Process from innermost to outermost to handle nesting correctly\n\t\t// Keep converting until no more details blocks are found\n\t\tlet previousText = ''\n\t\tlet currentText = text\n\n\t\twhile (previousText !== currentText) {\n\t\t\tpreviousText = currentText\n\t\t\tcurrentText = this.convertSinglePass(currentText)\n\t\t}\n\n\t\treturn currentText\n\t}\n\n\t/**\n\t * Perform a single pass of details block conversion\n\t * Converts the innermost details blocks first\n\t */\n\tprivate static convertSinglePass(text: string): string {\n\t\t// Match <details> blocks with optional attributes on the details tag\n\t\t// Supports multiline content between tags\n\t\tconst detailsRegex = /<details[^>]*>\\s*<summary[^>]*>(.*?)<\\/summary>\\s*(.*?)\\s*<\\/details>/gis\n\n\t\treturn text.replace(detailsRegex, (_match, summary, content) => {\n\t\t\t// Clean up the summary - trim whitespace and decode HTML entities\n\t\t\tconst cleanSummary = this.cleanText(summary)\n\n\t\t\t// Clean up the content - preserve internal structure but normalize outer whitespace\n\t\t\t// Note: Don't recursively convert here - the while loop handles that\n\t\t\tconst cleanContent = this.cleanContent(content)\n\n\t\t\t// Build Linear collapsible format\n\t\t\t// Always include blank lines around content for readability\n\t\t\tif (cleanContent) {\n\t\t\t\treturn `+++ ${cleanSummary}\\n\\n${cleanContent}\\n\\n+++`\n\t\t\t} else {\n\t\t\t\t// Empty content - use minimal format\n\t\t\t\treturn `+++ ${cleanSummary}\\n\\n+++`\n\t\t\t}\n\t\t})\n\t}\n\n\t/**\n\t * Clean text by trimming whitespace and decoding common HTML entities\n\t */\n\tprivate static cleanText(text: string): string {\n\t\treturn text\n\t\t\t.trim()\n\t\t\t.replace(/</g, '<')\n\t\t\t.replace(/>/g, '>')\n\t\t\t.replace(/&/g, '&')\n\t\t\t.replace(/"/g, '\"')\n\t\t\t.replace(/'/g, \"'\")\n\t}\n\n\t/**\n\t * Clean content while preserving internal structure\n\t * - Removes leading/trailing whitespace\n\t * - Normalizes internal blank lines (max 2 consecutive newlines)\n\t * - Preserves code blocks and other formatting\n\t */\n\tprivate static cleanContent(content: string): string {\n\t\tif (!content) {\n\t\t\treturn ''\n\t\t}\n\n\t\t// Trim outer whitespace\n\t\tlet cleaned = content.trim()\n\n\t\t// Normalize excessive blank lines (3+ newlines -> 2 newlines)\n\t\tcleaned = cleaned.replace(/\\n{3,}/g, '\\n\\n')\n\n\t\treturn cleaned\n\t}\n\n\t/**\n\t * Check if text contains HTML details/summary blocks\n\t * Useful for conditional conversion\n\t */\n\tstatic hasDetailsBlocks(text: string): boolean {\n\t\tif (!text) {\n\t\t\treturn false\n\t\t}\n\n\t\tconst detailsRegex = /<details[^>]*>.*?<summary[^>]*>.*?<\\/summary>.*?<\\/details>/is\n\t\treturn detailsRegex.test(text)\n\t}\n\n\t/**\n\t * Remove wrapper tags from code sample details blocks\n\t * Identifies details blocks where summary contains \"X lines\" pattern\n\t * and removes the details/summary tags while preserving the content\n\t *\n\t * @param text - Text containing potential code sample details blocks\n\t * @returns Text with code sample wrappers removed\n\t */\n\tstatic removeCodeSampleWrappers(text: string): string {\n\t\tif (!text) {\n\t\t\treturn text\n\t\t}\n\n\t\t// Match details blocks where summary contains \"X lines\" (e.g., \"45 lines\", \"120 lines\")\n\t\t// Pattern: <details><summary>...N lines...</summary>CONTENT</details>\n\t\t// Use [^<]* to match summary content without allowing nested tags to interfere\n\t\t// Then use [\\s\\S]*? for the content to allow any characters including newlines\n\t\tconst codeSampleRegex = /<details[^>]*>\\s*<summary[^>]*>([^<]*\\d+\\s+lines[^<]*)<\\/summary>\\s*([\\s\\S]*?)<\\/details>/gi\n\n\t\treturn text.replace(codeSampleRegex, (_match, _summary, content) => {\n\t\t\t// Return just the content, without any wrapper tags\n\t\t\t// Preserve the content exactly as-is\n\t\t\treturn content.trim()\n\t\t})\n\t}\n\n\t/**\n\t * Convert text for Linear - applies all necessary conversions\n\t * Currently only converts details/summary blocks, but can be extended\n\t * for other HTML to Linear markdown conversions\n\t */\n\tstatic convertToLinear(text: string): string {\n\t\tif (!text) {\n\t\t\treturn text\n\t\t}\n\n\t\t// Log input if logging is enabled\n\t\tthis.logConversion('INPUT', text)\n\n\t\t// Apply all conversions\n\t\tlet converted = text\n\n\t\t// First, remove code sample wrappers (details blocks with \"X lines\" pattern)\n\t\t// This prevents them from being converted to Linear's +++ format\n\t\tconverted = this.removeCodeSampleWrappers(converted)\n\n\t\t// Then convert remaining details/summary blocks to Linear format\n\t\tconverted = this.convertDetailsToLinear(converted)\n\n\t\t// Log output if logging is enabled\n\t\tthis.logConversion('OUTPUT', converted)\n\n\t\treturn converted\n\t}\n\n\t/**\n\t * Log conversion input/output if LINEAR_MARKDOWN_LOG_FILE is set\n\t */\n\tprivate static logConversion(label: string, content: string): void {\n\t\tconst logFilePath = process.env.LINEAR_MARKDOWN_LOG_FILE\n\t\tif (!logFilePath) {\n\t\t\treturn\n\t\t}\n\n\t\ttry {\n\t\t\tconst timestampedPath = this.getTimestampedLogPath(logFilePath)\n\t\t\tconst timestamp = new Date().toISOString()\n\t\t\tconst separator = '================================'\n\n\t\t\tconst logEntry = `${separator}\\n[${timestamp}] CONVERSION ${label}\\n${separator}\\n${label}:\\n${content}\\n\\n`\n\n\t\t\tappendFileSync(timestampedPath, logEntry, 'utf-8')\n\t\t} catch {\n\t\t\t// Silently fail - don't crash if logging fails\n\t\t\t// This is a debug feature and shouldn't break the conversion\n\t\t}\n\t}\n\n\t/**\n\t * Generate timestamped log file path\n\t * Example: debug.log -> debug-20231202-161234.log\n\t */\n\tprivate static getTimestampedLogPath(logFilePath: string): string {\n\t\tconst dir = dirname(logFilePath)\n\t\tconst ext = extname(logFilePath)\n\t\tconst base = basename(logFilePath, ext)\n\n\t\t// Generate timestamp: YYYYMMDD-HHMMSS\n\t\tconst now = new Date()\n\t\tconst timestamp = [\n\t\t\tnow.getFullYear(),\n\t\t\tString(now.getMonth() + 1).padStart(2, '0'),\n\t\t\tString(now.getDate()).padStart(2, '0'),\n\t\t].join('') + '-' + [\n\t\t\tString(now.getHours()).padStart(2, '0'),\n\t\t\tString(now.getMinutes()).padStart(2, '0'),\n\t\t\tString(now.getSeconds()).padStart(2, '0'),\n\t\t].join('')\n\n\t\treturn join(dir, `${base}-${timestamp}${ext}`)\n\t}\n}\n","/**\n * Linear implementation of Issue Management Provider\n * Uses @linear/sdk for all operations\n */\n\nimport type {\n\tIssueManagementProvider,\n\tGetIssueInput,\n\tGetPRInput,\n\tGetCommentInput,\n\tCreateCommentInput,\n\tUpdateCommentInput,\n\tCreateIssueInput,\n\tCreateChildIssueInput,\n\tCreateDependencyInput,\n\tGetDependenciesInput,\n\tRemoveDependencyInput,\n\tGetChildIssuesInput,\n\tCloseIssueInput,\n\tReopenIssueInput,\n\tEditIssueInput,\n\tCreateIssueResult,\n\tIssueResult,\n\tPRResult,\n\tCommentDetailResult,\n\tCommentResult,\n\tDependenciesResult,\n\tChildIssueResult,\n} from './types.js'\nimport {\n\tfetchLinearIssue,\n\tcreateLinearComment,\n\tgetLinearComment,\n\tupdateLinearComment,\n\tfetchLinearIssueComments,\n\tcreateLinearIssue,\n\tcreateLinearChildIssue,\n\tcreateLinearIssueRelation,\n\tgetLinearIssueDependencies,\n\tfindLinearIssueRelation,\n\tdeleteLinearIssueRelation,\n\tgetLinearChildIssues,\n\tupdateLinearIssueState,\n\teditLinearIssue,\n} from '../utils/linear.js'\nimport { LinearMarkupConverter } from '../utils/linear-markup-converter.js'\nimport { processMarkdownImages } from '../utils/image-processor.js'\n\n/**\n * Linear-specific implementation of IssueManagementProvider\n */\nexport class LinearIssueManagementProvider implements IssueManagementProvider {\n\treadonly providerName = 'linear'\n\treadonly issuePrefix = ''\n\n\t/**\n\t * Cached team key extracted from issue identifiers (e.g., \"ENG-123\" -> \"ENG\")\n\t * Used as fallback when teamKey is not explicitly provided to createIssue()\n\t */\n\tprivate cachedTeamKey: string | undefined = undefined\n\n\t/**\n\t * Fetch issue details using Linear SDK\n\t */\n\tasync getIssue(input: GetIssueInput): Promise<IssueResult> {\n\t\tconst { number, includeComments = true } = input\n\n\t\t// Extract and cache team key from identifier (e.g., \"ENG-123\" -> \"ENG\")\n\t\t// This enables createIssue() to use the team key as a fallback\n\t\tconst match = number.match(/^([A-Z]{2,})-\\d+$/i)\n\t\tif (match?.[1]) {\n\t\t\tthis.cachedTeamKey = match[1].toUpperCase()\n\t\t}\n\n\t\t// Fetch issue - Linear uses alphanumeric identifiers like \"ENG-123\"\n\t\tconst raw = await fetchLinearIssue(number)\n\n\t\t// Map Linear state name to open/closed\n\t\tconst state = raw.state && (raw.state.toLowerCase().includes('done') || raw.state.toLowerCase().includes('completed') || raw.state.toLowerCase().includes('canceled'))\n\t\t\t? 'closed'\n\t\t\t: 'open'\n\n\t\t// Build result\n\t\tconst result: IssueResult = {\n\t\t\tid: raw.identifier,\n\t\t\ttitle: raw.title,\n\t\t\tbody: raw.description ?? '',\n\t\t\tstate,\n\t\t\turl: raw.url,\n\t\t\tprovider: 'linear',\n\t\t\tauthor: null, // Linear SDK doesn't return author in basic fetch\n\n\t\t\t// Linear-specific fields\n\t\t\tlinearState: raw.state,\n\t\t\tcreatedAt: raw.createdAt,\n\t\t\tupdatedAt: raw.updatedAt,\n\t\t}\n\n\t\t// Fetch comments if requested\n\t\tif (includeComments) {\n\t\t\ttry {\n\t\t\t\tconst comments = await this.fetchIssueComments(number)\n\t\t\t\tif (comments) {\n\t\t\t\t\tresult.comments = comments\n\t\t\t\t}\n\t\t\t} catch {\n\t\t\t\t// If comments fail, continue without them\n\t\t\t}\n\t\t}\n\n\t\t// Process images in body and comments to make them accessible\n\t\tresult.body = await processMarkdownImages(result.body, 'linear')\n\t\tif (result.comments) {\n\t\t\tfor (const comment of result.comments) {\n\t\t\t\tcomment.body = await processMarkdownImages(comment.body, 'linear')\n\t\t\t}\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Fetch pull request details\n\t * Linear does not support PRs - this throws an error directing to use GitHub\n\t */\n\tasync getPR(_input: GetPRInput): Promise<PRResult> {\n\t\tthrow new Error('Linear does not support pull requests. PRs exist only on GitHub. Use the GitHub provider for PR operations.')\n\t}\n\n\t/**\n\t * Fetch comments for an issue\n\t */\n\tprivate async fetchIssueComments(identifier: string): Promise<IssueResult['comments']> {\n\t\ttry {\n\t\t\tconst comments = await fetchLinearIssueComments(identifier)\n\n\t\t\treturn comments.map(comment => ({\n\t\t\t\tid: comment.id,\n\t\t\t\tbody: comment.body,\n\t\t\t\tcreatedAt: comment.createdAt,\n\t\t\t\tauthor: null, // Linear SDK doesn't return comment author info in basic fetch\n\t\t\t\t...(comment.updatedAt && { updatedAt: comment.updatedAt }),\n\t\t\t}))\n\t\t} catch {\n\t\t\treturn []\n\t\t}\n\t}\n\n\t/**\n\t * Fetch a specific comment by ID\n\t */\n\tasync getComment(input: GetCommentInput): Promise<CommentDetailResult> {\n\t\tconst { commentId } = input\n\n\t\tconst raw = await getLinearComment(commentId)\n\n\t\t// Process images to make them accessible\n\t\tconst processedBody = await processMarkdownImages(raw.body, 'linear')\n\n\t\treturn {\n\t\t\tid: raw.id,\n\t\t\tbody: processedBody,\n\t\t\tauthor: null, // Linear SDK doesn't return comment author info in basic fetch\n\t\t\tcreated_at: raw.createdAt,\n\t\t}\n\t}\n\n\t/**\n\t * Create a new comment on an issue\n\t */\n\tasync createComment(input: CreateCommentInput): Promise<CommentResult> {\n\t\tconst { number, body } = input\n\t\t// Note: Linear doesn't distinguish between issue and PR comments\n\t\t// (Linear doesn't have PRs - that's GitHub-specific)\n\n\t\t// Convert HTML details/summary blocks to Linear's collapsible format\n\t\tconst convertedBody = LinearMarkupConverter.convertToLinear(body)\n\n\t\tconst result = await createLinearComment(number, convertedBody)\n\n\t\treturn {\n\t\t\tid: result.id,\n\t\t\turl: result.url,\n\t\t\tcreated_at: result.createdAt,\n\t\t}\n\t}\n\n\t/**\n\t * Update an existing comment\n\t */\n\tasync updateComment(input: UpdateCommentInput): Promise<CommentResult> {\n\t\tconst { commentId, body } = input\n\n\t\t// Convert HTML details/summary blocks to Linear's collapsible format\n\t\tconst convertedBody = LinearMarkupConverter.convertToLinear(body)\n\n\t\tconst result = await updateLinearComment(commentId, convertedBody)\n\n\t\treturn {\n\t\t\tid: result.id,\n\t\t\turl: result.url,\n\t\t\tupdated_at: result.updatedAt,\n\t\t}\n\t}\n\n\t/**\n\t * Create a new issue\n\t */\n\tasync createIssue(input: CreateIssueInput): Promise<CreateIssueResult> {\n\t\tconst { title, body, labels, teamKey } = input\n\n\t\t// Fallback chain: explicit param > settings (via env) > cached key from getIssue()\n\t\tconst effectiveTeamKey = teamKey ?? process.env.LINEAR_TEAM_KEY ?? this.cachedTeamKey\n\n\t\tif (!effectiveTeamKey) {\n\t\t\tthrow new Error('teamKey is required for Linear issue creation. Configure issueManagement.linear.teamId in settings, or call getIssue first to extract the team from an issue identifier.')\n\t\t}\n\n\t\tconst result = await createLinearIssue(title, body, effectiveTeamKey, labels)\n\n\t\treturn {\n\t\t\tid: result.identifier,\n\t\t\turl: result.url,\n\t\t}\n\t}\n\n\t/**\n\t * Create a child issue linked to a parent issue\n\t * Linear supports atomic creation with parentId field\n\t */\n\tasync createChildIssue(input: CreateChildIssueInput): Promise<CreateIssueResult> {\n\t\tconst { parentId, title, body, labels, teamKey } = input\n\n\t\t// Fetch parent issue to get UUID (parentId in input is identifier like \"ENG-123\")\n\t\tconst parentIssue = await fetchLinearIssue(parentId)\n\n\t\t// Extract team key from parent identifier if not provided\n\t\tconst match = parentId.match(/^([A-Z]{2,})-\\d+$/i)\n\t\tconst effectiveTeamKey = teamKey ?? match?.[1]?.toUpperCase() ?? process.env.LINEAR_TEAM_KEY ?? this.cachedTeamKey\n\n\t\tif (!effectiveTeamKey) {\n\t\t\tthrow new Error('teamKey is required for Linear child issue creation. Provide teamKey parameter or use a parent identifier with team prefix.')\n\t\t}\n\n\t\t// Create child issue with parent's UUID\n\t\tconst result = await createLinearChildIssue(\n\t\t\ttitle,\n\t\t\tbody,\n\t\t\teffectiveTeamKey,\n\t\t\tparentIssue.id, // UUID, not identifier\n\t\t\tlabels\n\t\t)\n\n\t\treturn {\n\t\t\tid: result.identifier,\n\t\t\turl: result.url,\n\t\t}\n\t}\n\n\t/**\n\t * Create a blocking dependency between two issues\n\t */\n\tasync createDependency(input: CreateDependencyInput): Promise<void> {\n\t\tconst { blockingIssue, blockedIssue } = input\n\n\t\t// Fetch both issues to get their UUIDs\n\t\tconst [blockingIssueData, blockedIssueData] = await Promise.all([\n\t\t\tfetchLinearIssue(blockingIssue),\n\t\t\tfetchLinearIssue(blockedIssue),\n\t\t])\n\n\t\t// Create the blocking relation (blockingIssue blocks blockedIssue)\n\t\tawait createLinearIssueRelation(blockingIssueData.id, blockedIssueData.id)\n\t}\n\n\t/**\n\t * Get dependencies for an issue\n\t */\n\tasync getDependencies(input: GetDependenciesInput): Promise<DependenciesResult> {\n\t\tconst { number, direction } = input\n\n\t\treturn await getLinearIssueDependencies(number, direction)\n\t}\n\n\t/**\n\t * Remove a blocking dependency between two issues\n\t */\n\tasync removeDependency(input: RemoveDependencyInput): Promise<void> {\n\t\tconst { blockingIssue, blockedIssue } = input\n\n\t\t// Find the relation ID\n\t\tconst relationId = await findLinearIssueRelation(blockingIssue, blockedIssue)\n\n\t\tif (!relationId) {\n\t\t\tthrow new Error(`No blocking dependency found from ${blockingIssue} to ${blockedIssue}`)\n\t\t}\n\n\t\t// Delete the relation\n\t\tawait deleteLinearIssueRelation(relationId)\n\t}\n\n\t/**\n\t * Get child issues of a parent issue\n\t */\n\tasync getChildIssues(input: GetChildIssuesInput): Promise<ChildIssueResult[]> {\n\t\tconst { number } = input\n\t\t// repo is ignored for Linear\n\t\treturn await getLinearChildIssues(number)\n\t}\n\n\t/**\n\t * Close an issue by transitioning to \"Done\" state\n\t */\n\tasync closeIssue(input: CloseIssueInput): Promise<void> {\n\t\tconst { number } = input\n\t\t// repo is ignored for Linear\n\t\tawait updateLinearIssueState(number, 'Done')\n\t}\n\n\t/**\n\t * Reopen a closed issue by transitioning to \"Todo\" state\n\t */\n\tasync reopenIssue(input: ReopenIssueInput): Promise<void> {\n\t\tconst { number } = input\n\t\t// repo is ignored for Linear\n\t\tawait updateLinearIssueState(number, 'Todo')\n\t}\n\n\t/**\n\t * Edit an issue's properties\n\t * State changes are delegated to closeIssue/reopenIssue\n\t */\n\tasync editIssue(input: EditIssueInput): Promise<void> {\n\t\tconst { number, title, body, state } = input\n\t\t// repo and labels are ignored for Linear\n\n\t\t// Handle state changes via close/reopen\n\t\tif (state === 'closed') {\n\t\t\tawait this.closeIssue({ number })\n\t\t} else if (state === 'open') {\n\t\t\tawait this.reopenIssue({ number })\n\t\t}\n\n\t\t// Handle title/body updates\n\t\tif (title !== undefined || body !== undefined) {\n\t\t\tawait editLinearIssue(number, {\n\t\t\t\t...(title !== undefined && { title }),\n\t\t\t\t...(body !== undefined && { description: body }),\n\t\t\t})\n\t\t}\n\t}\n}\n","/**\n * Jira implementation of Issue Management Provider\n * Uses JiraIssueTracker for all operations\n * Normalizes Jira-specific fields to provider-agnostic core fields\n */\n\nimport type {\n\tIssueManagementProvider,\n\tGetIssueInput,\n\tGetPRInput,\n\tPRResult,\n\tGetCommentInput,\n\tCreateCommentInput,\n\tUpdateCommentInput,\n\tCreateIssueInput,\n\tCreateChildIssueInput,\n\tCreateDependencyInput,\n\tGetDependenciesInput,\n\tDependenciesResult,\n\tRemoveDependencyInput,\n\tGetChildIssuesInput,\n\tCloseIssueInput,\n\tReopenIssueInput,\n\tEditIssueInput,\n\tChildIssueResult,\n\tCreateIssueResult,\n\tIssueResult,\n\tCommentDetailResult,\n\tCommentResult,\n\tFlexibleAuthor,\n} from './types.js'\nimport { escapeJql } from '../utils/jira.js'\nimport { JiraIssueTracker } from '../lib/providers/jira/JiraIssueTracker.js'\nimport type { JiraTrackerConfig } from '../lib/providers/jira/JiraIssueTracker.js'\nimport type { Issue } from '../types/index.js'\nimport { SettingsManager } from '../lib/SettingsManager.js'\nimport type { IloomSettings } from '../lib/SettingsManager.js'\n\n/**\n * Normalize Jira author to FlexibleAuthor format\n */\nfunction normalizeAuthor(author: { displayName?: string; emailAddress?: string; accountId?: string } | null | undefined): FlexibleAuthor | null {\n\tif (!author) return null\n\n\treturn {\n\t\tid: author.accountId ?? author.emailAddress ?? 'unknown',\n\t\tdisplayName: author.displayName ?? author.emailAddress ?? 'Unknown',\n\t\t...(author.emailAddress && { email: author.emailAddress }),\n\t\t...(author.accountId && { accountId: author.accountId }),\n\t}\n}\n/**\n * Extract Jira configuration from settings (for cli usage) or environment variables (in mcp server)\n */\nconst getJiraTrackerConfig = (settings: IloomSettings): JiraTrackerConfig => {\n\tconst jiraSettings = settings.issueManagement?.jira\n\n\tif (jiraSettings?.host && jiraSettings?.username && jiraSettings?.apiToken && jiraSettings?.projectKey) {\n\t\t\tconst config: JiraTrackerConfig = {\n\t\t\thost: jiraSettings.host,\n\t\t\tusername: jiraSettings.username,\n\t\t\tapiToken: jiraSettings.apiToken,\n\t\t\tprojectKey: jiraSettings.projectKey,\n\t\t}\n\n\t\tif (jiraSettings.transitionMappings) {\n\t\t\tconfig.transitionMappings = jiraSettings.transitionMappings\n\t\t}\n\t\tif (jiraSettings.defaultIssueType) {\n\t\t\tconfig.defaultIssueType = jiraSettings.defaultIssueType\n\t\t}\n\t\tif (jiraSettings.defaultSubtaskType) {\n\t\t\tconfig.defaultSubtaskType = jiraSettings.defaultSubtaskType\n\t\t}\n\n\t\treturn config;\n\t}\n\n\tif (process.env.JIRA_HOST && process.env.JIRA_USERNAME && process.env.JIRA_API_TOKEN && process.env.JIRA_PROJECT_KEY) {\n\t\tconst config: JiraTrackerConfig = {\n\t\t\thost: process.env.JIRA_HOST,\n\t\t\tusername: process.env.JIRA_USERNAME,\n\t\t\tapiToken: process.env.JIRA_API_TOKEN,\n\t\t\tprojectKey: process.env.JIRA_PROJECT_KEY,\n\t\t}\n\n\t\tif (process.env.JIRA_TRANSITION_MAPPINGS) {\n\t\t\ttry {\n\t\t\t\tconfig.transitionMappings = JSON.parse(process.env.JIRA_TRANSITION_MAPPINGS)\n\t\t\t} catch {\n\t\t\t\tthrow new Error('Invalid JSON in JIRA_TRANSITION_MAPPINGS environment variable')\n\t\t\t}\n\t\t}\n\t\tif (process.env.JIRA_DEFAULT_ISSUE_TYPE) {\n\t\t\tconfig.defaultIssueType = process.env.JIRA_DEFAULT_ISSUE_TYPE\n\t\t}\n\t\tif (process.env.JIRA_DEFAULT_SUBTASK_TYPE) {\n\t\t\tconfig.defaultSubtaskType = process.env.JIRA_DEFAULT_SUBTASK_TYPE\n\t\t}\n\n\t\treturn config\n\t}\n\n\tthrow new Error(\n\t\t'Missing required Jira settings: issueManagement.jira.{host, username, apiToken, projectKey} or corresponding environment variables'\n\t)\t\n}\n\n/**\n * Jira-specific implementation of IssueManagementProvider\n */\nexport class JiraIssueManagementProvider implements IssueManagementProvider {\n\treadonly providerName = 'jira'\n\treadonly issuePrefix = ''\n\tprivate tracker: JiraIssueTracker\n\tprivate projectKey: string\n\n\tconstructor(settings: IloomSettings) {\n\t\tconst config = getJiraTrackerConfig(settings);\n\n\t\tthis.tracker = new JiraIssueTracker(config)\n\t\tthis.projectKey = config.projectKey\n\t}\n\n\t/**\n\t * Static factory for convenience when settings aren't pre-loaded\n\t */\n\tstatic async create(): Promise<JiraIssueManagementProvider> {\n\t\tconst settingsManager = new SettingsManager()\n\t\tconst settings = await settingsManager.loadSettings()\n\t\treturn new JiraIssueManagementProvider(settings)\n\t}\n\n\t/**\n\t * Fetch issue details using JiraIssueTracker\n\t */\n\tasync getIssue(input: GetIssueInput): Promise<IssueResult> {\n\t\tconst { number, includeComments = true } = input\n\n\t\t// Fetch issue from Jira\n\t\tconst issue = await this.tracker.getIssue(number)\n\t\tconst issueExt = issue as Issue & {\n\t\t\tid?: string\n\t\t\tkey?: string\n\t\t\tauthor?: {\n\t\t\t\tdisplayName?: string\n\t\t\t\temailAddress?: string\n\t\t\t\taccountId?: string\n\t\t\t}\n\t\t\tissueType?: string\n\t\t\tpriority?: string\n\t\t\tstatus?: string\n\t\t}\n\n\t\t// Normalize to IssueResult format\n\t\tconst result: IssueResult = {\n\t\t\tid: issueExt.id ?? String(issue.number),\n\t\t\ttitle: issue.title,\n\t\t\tbody: issue.body,\n\t\t\tstate: issue.state,\n\t\t\turl: issue.url,\n\t\t\tprovider: 'jira',\n\t\t\tauthor: normalizeAuthor(issueExt.author),\n\t\t\tnumber: issue.number,\n\t\t\tkey: issueExt.key,\n\t\t\t// Preserve Jira-specific fields\n\t\t\t...(issueExt.issueType && { issueType: issueExt.issueType }),\n\t\t\t...(issueExt.priority && { priority: issueExt.priority }),\n\t\t\t...(issueExt.status && { status: issueExt.status }),\n\t\t}\n\n\t\t// Add labels if present\n\t\tif (issue.labels && issue.labels.length > 0) {\n\t\t\tresult.labels = issue.labels.map(label => ({ name: label }))\n\t\t}\n\n\t\t// Add assignees if present - Issue type uses assignees array of strings\n\t\tif (issue.assignees && issue.assignees.length > 0) {\n\t\t\tresult.assignees = issue.assignees.map(name => ({\n\t\t\t\tid: name,\n\t\t\t\tdisplayName: name,\n\t\t\t}))\n\t\t}\n\n\t\t// Fetch and add comments if requested\n\t\tif (includeComments) {\n\t\t\tconst comments = await this.tracker.getComments(number)\n\t\t\tresult.comments = comments.map((comment: {\n\t\t\t\tid: string\n\t\t\t\tbody: string\n\t\t\t\tauthor: { displayName: string; emailAddress: string; accountId: string }\n\t\t\t\tcreatedAt: string\n\t\t\t\tupdatedAt: string\n\t\t\t}) => ({\n\t\t\t\tid: comment.id,\n\t\t\t\tbody: comment.body,\n\t\t\t\tauthor: normalizeAuthor(comment.author),\n\t\t\t\tcreatedAt: comment.createdAt,\n\t\t\t\tupdatedAt: comment.updatedAt,\n\t\t\t}))\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Fetch a specific comment by ID\n\t */\n\tasync getComment(input: GetCommentInput): Promise<CommentDetailResult> {\n\t\tconst { commentId, number } = input\n\n\t\t// Fetch all comments and find the specific one\n\t\tconst comments = await this.tracker.getComments(number)\n\t\tconst comment = comments.find(c => c.id === commentId)\n\n\t\tif (!comment) {\n\t\t\tthrow new Error(`Comment ${commentId} not found on issue ${number}`)\n\t\t}\n\n\t\treturn {\n\t\t\tid: comment.id,\n\t\t\tbody: comment.body,\n\t\t\tauthor: normalizeAuthor(comment.author),\n\t\t\tcreated_at: comment.createdAt,\n\t\t\tupdated_at: comment.updatedAt,\n\t\t}\n\t}\n\n\t/**\n\t * Create a new comment on an issue\n\t */\n\tasync createComment(input: CreateCommentInput): Promise<CommentResult> {\n\t\tconst { number, body } = input\n\t\tconst normalizedKey = this.tracker.normalizeIdentifier(number)\n\n\t\t// Jira doesn't distinguish between issue and PR comments\n\t\tconst comment = await this.tracker.addComment(normalizedKey, body)\n\n\t\treturn {\n\t\t\tid: comment.id,\n\t\t\turl: `${this.tracker.getConfig().host}/browse/${normalizedKey}?focusedCommentId=${comment.id}`,\n\t\t\tcreated_at: new Date().toISOString(),\n\t\t}\n\t}\n\n\t/**\n\t * Update an existing comment\n\t */\n\tasync updateComment(input: UpdateCommentInput): Promise<CommentResult> {\n\t\tconst { commentId, number, body } = input\n\t\tconst normalizedKey = this.tracker.normalizeIdentifier(number)\n\n\t\t// Update comment via tracker\n\t\tawait this.tracker.updateComment(normalizedKey, commentId, body)\n\n\t\treturn {\n\t\t\tid: commentId,\n\t\t\turl: `${this.tracker.getConfig().host}/browse/${normalizedKey}?focusedCommentId=${commentId}`,\n\t\t\tupdated_at: new Date().toISOString(),\n\t\t}\n\t}\n\n\t/**\n\t * Create a new issue\n\t */\n\tasync createIssue(input: CreateIssueInput): Promise<CreateIssueResult> {\n\t\tconst { title, body } = input\n\n\t\t// Create issue via tracker (labels not supported in current implementation)\n\t\tconst issue = await this.tracker.createIssue(title, body)\n\n\t\tconst result: CreateIssueResult = {\n\t\t\tid: String(issue.number),\n\t\t\turl: issue.url,\n\t\t}\n\n\t\t// Only add number if it's actually a number\n\t\tif (typeof issue.number === 'number') {\n\t\t\tresult.number = issue.number\n\t\t}\n\n\t\treturn result\n\t}\n\n\t/**\n\t * Fetch pull request details\n\t * Jira does not have pull requests - throw like Linear does\n\t */\n\tasync getPR(_input: GetPRInput): Promise<PRResult> {\n\t\tthrow new Error(\n\t\t\t'Jira does not support pull requests. PRs exist only on GitHub. Use the GitHub provider for PR operations.'\n\t\t)\n\t}\n\n\t/**\n\t * Create a child issue linked to a parent issue\n\t * Uses Jira's parent field to create a subtask\n\t */\n\tasync createChildIssue(input: CreateChildIssueInput): Promise<CreateIssueResult> {\n\t\tconst { parentId, title, body } = input\n\t\tconst parentKey = this.tracker.normalizeIdentifier(parentId)\n\n\t\tconst jiraIssue = await this.tracker.getApiClient().createIssueWithParent(\n\t\t\tthis.projectKey,\n\t\t\ttitle,\n\t\t\tbody,\n\t\t\tparentKey,\n\t\t\tthis.tracker.getConfig().defaultSubtaskType\n\t\t)\n\n\t\treturn {\n\t\t\tid: jiraIssue.key,\n\t\t\turl: `${this.tracker.getConfig().host}/browse/${jiraIssue.key}`,\n\t\t}\n\t}\n\n\t/**\n\t * Create a blocking dependency between two issues\n\t * Uses Jira issue links with \"Blocks\" link type\n\t */\n\tasync createDependency(input: CreateDependencyInput): Promise<void> {\n\t\tconst blockingKey = this.tracker.normalizeIdentifier(input.blockingIssue)\n\t\tconst blockedKey = this.tracker.normalizeIdentifier(input.blockedIssue)\n\n\t\t// In Jira \"Blocks\" link type: inwardIssue = blocker, outwardIssue = blocked\n\t\tawait this.tracker.getApiClient().createIssueLink(blockingKey, blockedKey, 'Blocks')\n\t}\n\n\t/**\n\t * Get dependencies for an issue\n\t * Parses issue links of type \"Blocks\"\n\t */\n\tasync getDependencies(input: GetDependenciesInput): Promise<DependenciesResult> {\n\t\tconst issueKey = this.tracker.normalizeIdentifier(input.number)\n\t\tconst host = this.tracker.getConfig().host\n\n\t\tconst issue = await this.tracker.getApiClient().getIssue(issueKey)\n\t\tconst links = issue.fields.issuelinks ?? []\n\n\t\tconst blocking: DependenciesResult['blocking'] = []\n\t\tconst blockedBy: DependenciesResult['blockedBy'] = []\n\n\t\tfor (const link of links) {\n\t\t\tif (link.type.name !== 'Blocks') continue\n\n\t\t\t// inwardIssue present = the other issue is the blocker\n\t\t\t// → that issue blocks this issue → blockedBy\n\t\t\tif (link.inwardIssue) {\n\t\t\t\tblockedBy.push({\n\t\t\t\t\tid: link.inwardIssue.key,\n\t\t\t\t\ttitle: link.inwardIssue.fields.summary,\n\t\t\t\t\turl: `${host}/browse/${link.inwardIssue.key}`,\n\t\t\t\t\tstate: link.inwardIssue.fields.status.name.toLowerCase(),\n\t\t\t\t})\n\t\t\t}\n\n\t\t\t// outwardIssue present = the other issue is blocked by this issue\n\t\t\t// → this issue blocks that issue → blocking\n\t\t\tif (link.outwardIssue) {\n\t\t\t\tblocking.push({\n\t\t\t\t\tid: link.outwardIssue.key,\n\t\t\t\t\ttitle: link.outwardIssue.fields.summary,\n\t\t\t\t\turl: `${host}/browse/${link.outwardIssue.key}`,\n\t\t\t\t\tstate: link.outwardIssue.fields.status.name.toLowerCase(),\n\t\t\t\t})\n\t\t\t}\n\t\t}\n\n\t\tif (input.direction === 'blocking') {\n\t\t\treturn { blocking, blockedBy: [] }\n\t\t}\n\t\tif (input.direction === 'blocked_by') {\n\t\t\treturn { blocking: [], blockedBy }\n\t\t}\n\t\treturn { blocking, blockedBy }\n\t}\n\n\t/**\n\t * Remove a blocking dependency between two issues\n\t * Finds the matching \"Blocks\" link and deletes it\n\t */\n\tasync removeDependency(input: RemoveDependencyInput): Promise<void> {\n\t\tconst blockingKey = this.tracker.normalizeIdentifier(input.blockingIssue)\n\t\tconst blockedKey = this.tracker.normalizeIdentifier(input.blockedIssue)\n\n\t\t// Fetch the blocked issue to find the link\n\t\tconst issue = await this.tracker.getApiClient().getIssue(blockedKey)\n\t\tconst links = issue.fields.issuelinks ?? []\n\n\t\t// When fetching the blocked issue (B), the blocking issue (A) appears as\n\t\t// inwardIssue in the link data\n\t\tconst matchingLink = links.find(link =>\n\t\t\tlink.type.name === 'Blocks' && link.inwardIssue?.key === blockingKey\n\t\t)\n\n\t\tif (!matchingLink) {\n\t\t\tthrow new Error(\n\t\t\t\t`No \"Blocks\" dependency found from ${blockingKey} to ${blockedKey}`\n\t\t\t)\n\t\t}\n\n\t\tawait this.tracker.getApiClient().deleteIssueLink(matchingLink.id)\n\t}\n\n\t/**\n\t * Get child issues of a parent issue\n\t * Uses JQL search: parent = KEY\n\t */\n\tasync getChildIssues(input: GetChildIssuesInput): Promise<ChildIssueResult[]> {\n\t\tconst parentKey = this.tracker.normalizeIdentifier(input.number)\n\t\tconst host = this.tracker.getConfig().host\n\n\t\tconst issues = await this.tracker.getApiClient().searchIssues(`parent = \"${escapeJql(parentKey)}\"`)\n\n\t\treturn issues.map(issue => ({\n\t\t\tid: issue.key,\n\t\t\ttitle: issue.fields.summary,\n\t\t\turl: `${host}/browse/${issue.key}`,\n\t\t\tstate: issue.fields.status.name.toLowerCase(),\n\t\t}))\n\t}\n\n\t/**\n\t * Close an issue by transitioning to \"Done\" state\n\t */\n\tasync closeIssue(input: CloseIssueInput): Promise<void> {\n\t\tconst issueKey = this.tracker.normalizeIdentifier(input.number)\n\t\tawait this.tracker.closeIssue(issueKey)\n\t}\n\n\t/**\n\t * Reopen a closed issue\n\t */\n\tasync reopenIssue(input: ReopenIssueInput): Promise<void> {\n\t\tconst issueKey = this.tracker.normalizeIdentifier(input.number)\n\t\tawait this.tracker.reopenIssue(issueKey)\n\t}\n\n\t/**\n\t * Edit an issue's properties\n\t * State changes are delegated to closeIssue/reopenIssue\n\t */\n\tasync editIssue(input: EditIssueInput): Promise<void> {\n\t\tconst { number, title, body, state } = input\n\n\t\t// Handle state changes via close/reopen\n\t\tif (state === 'closed') {\n\t\t\tawait this.closeIssue({ number })\n\t\t} else if (state === 'open') {\n\t\t\tawait this.reopenIssue({ number })\n\t\t}\n\n\t\t// Handle title/body updates via Jira API\n\t\tif (title !== undefined || body !== undefined) {\n\t\t\tconst issueKey = this.tracker.normalizeIdentifier(number)\n\t\t\tawait this.tracker.getApiClient().updateIssue(issueKey, {\n\t\t\t\t...(title !== undefined && { summary: title }),\n\t\t\t\t...(body !== undefined && { description: body }),\n\t\t\t})\n\t\t}\n\t}\n}\n","/**\n * Factory for creating issue management providers\n */\n\nimport type { IssueManagementProvider, IssueProvider } from './types.js'\nimport { GitHubIssueManagementProvider } from './GitHubIssueManagementProvider.js'\nimport { LinearIssueManagementProvider } from './LinearIssueManagementProvider.js'\nimport { JiraIssueManagementProvider } from './JiraIssueManagementProvider.js'\nimport type { IloomSettings } from '../lib/SettingsManager.js'\n\n/**\n * Factory class for creating issue management providers\n */\nexport class IssueManagementProviderFactory {\n\t/**\n\t * Create an issue management provider based on the provider type\n\t * @param provider - The provider type (github, linear, jira)\n\t * @param settings - Required for Jira provider, optional for others\n\t */\n\tstatic create(provider: IssueProvider, settings?: IloomSettings): IssueManagementProvider {\n\t\tswitch (provider) {\n\t\t\tcase 'github':\n\t\t\t\treturn new GitHubIssueManagementProvider()\n\t\t\tcase 'linear':\n\t\t\t\treturn new LinearIssueManagementProvider()\n\t\t\tcase 'jira':\n\t\t\t\tif (!settings) {\n\t\t\t\t\tthrow new Error('Settings required for Jira provider')\n\t\t\t\t}\n\t\t\t\treturn new JiraIssueManagementProvider(settings)\n\t\t\tdefault:\n\t\t\t\tthrow new Error(`Unsupported issue management provider: ${provider}`)\n\t\t}\n\t}\n}\n"],"mappings":"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","names":["join","extname","normalizeAuthor"]}
|
|
@@ -1,17 +1,17 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
ProcessManager
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-5PNZBH6V.js";
|
|
5
5
|
import {
|
|
6
6
|
detectPackageManager,
|
|
7
7
|
runScript
|
|
8
|
-
} from "./chunk-
|
|
8
|
+
} from "./chunk-3GTUXW26.js";
|
|
9
9
|
import {
|
|
10
10
|
getPackageScripts
|
|
11
|
-
} from "./chunk-
|
|
11
|
+
} from "./chunk-NCPZYQ4B.js";
|
|
12
12
|
import {
|
|
13
13
|
logger
|
|
14
|
-
} from "./chunk-
|
|
14
|
+
} from "./chunk-H2SSF24U.js";
|
|
15
15
|
|
|
16
16
|
// src/lib/DevServerManager.ts
|
|
17
17
|
import { execa } from "execa";
|
|
@@ -217,4 +217,4 @@ var DevServerManager = class {
|
|
|
217
217
|
export {
|
|
218
218
|
DevServerManager
|
|
219
219
|
};
|
|
220
|
-
//# sourceMappingURL=chunk-
|
|
220
|
+
//# sourceMappingURL=chunk-RVI6C2H5.js.map
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
#!/usr/bin/env node
|
|
2
2
|
import {
|
|
3
3
|
logger
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-H2SSF24U.js";
|
|
5
5
|
|
|
6
6
|
// src/utils/FirstRunManager.ts
|
|
7
7
|
import os from "os";
|
|
@@ -140,4 +140,4 @@ var FirstRunManager = class {
|
|
|
140
140
|
export {
|
|
141
141
|
FirstRunManager
|
|
142
142
|
};
|
|
143
|
-
//# sourceMappingURL=chunk-
|
|
143
|
+
//# sourceMappingURL=chunk-SQYHPBFP.js.map
|