@dev.sail.money/sailor 0.0.2-12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/AGENTS.md +111 -0
- package/LICENSE +21 -0
- package/README.md +337 -0
- package/docs/PERMISSION_MODEL.md +93 -0
- package/examples/permissions/BoundedBet_Limitless_Base.sol +96 -0
- package/examples/permissions/BoundedBorrow_AaveV3_Arbitrum.sol +94 -0
- package/examples/permissions/BoundedPerp_GMXv2_Arbitrum.sol +143 -0
- package/examples/permissions/BoundedSwap_UniswapV3_Base.sol +113 -0
- package/examples/permissions/BoundedSwap_UniswapV4_Unichain.sol +144 -0
- package/examples/permissions/BoundedTransfer_ERC20_Ethereum.sol +73 -0
- package/examples/permissions/README.md +52 -0
- package/examples/permissions/foundry.toml +10 -0
- package/examples/permissions/interfaces/IPermission.sol +18 -0
- package/package.json +39 -0
- package/packages/cli/README.md +34 -0
- package/packages/cli/dist/index.cjs +42801 -0
- package/packages/cli/dist/server.cjs +51201 -0
- package/packages/sdk/README.md +65 -0
- package/packages/sdk/dist/abis/MandateFactory.d.ts +140 -0
- package/packages/sdk/dist/abis/MandateFactory.d.ts.map +1 -0
- package/packages/sdk/dist/abis/MandateFactory.js +93 -0
- package/packages/sdk/dist/abis/MandateFactory.js.map +1 -0
- package/packages/sdk/dist/abis/SailGovernance.d.ts +57 -0
- package/packages/sdk/dist/abis/SailGovernance.d.ts.map +1 -0
- package/packages/sdk/dist/abis/SailGovernance.js +52 -0
- package/packages/sdk/dist/abis/SailGovernance.js.map +1 -0
- package/packages/sdk/dist/abis/SailKernel.d.ts +466 -0
- package/packages/sdk/dist/abis/SailKernel.d.ts.map +1 -0
- package/packages/sdk/dist/abis/SailKernel.js +311 -0
- package/packages/sdk/dist/abis/SailKernel.js.map +1 -0
- package/packages/sdk/dist/abis/index.d.ts +4 -0
- package/packages/sdk/dist/abis/index.d.ts.map +1 -0
- package/packages/sdk/dist/abis/index.js +7 -0
- package/packages/sdk/dist/abis/index.js.map +1 -0
- package/packages/sdk/dist/capabilities.d.ts +77 -0
- package/packages/sdk/dist/capabilities.d.ts.map +1 -0
- package/packages/sdk/dist/capabilities.js +158 -0
- package/packages/sdk/dist/capabilities.js.map +1 -0
- package/packages/sdk/dist/client.d.ts +37 -0
- package/packages/sdk/dist/client.d.ts.map +1 -0
- package/packages/sdk/dist/client.js +646 -0
- package/packages/sdk/dist/client.js.map +1 -0
- package/packages/sdk/dist/deployments.d.ts +82 -0
- package/packages/sdk/dist/deployments.d.ts.map +1 -0
- package/packages/sdk/dist/deployments.js +197 -0
- package/packages/sdk/dist/deployments.js.map +1 -0
- package/packages/sdk/dist/discovery.d.ts +10 -0
- package/packages/sdk/dist/discovery.d.ts.map +1 -0
- package/packages/sdk/dist/discovery.js +41 -0
- package/packages/sdk/dist/discovery.js.map +1 -0
- package/packages/sdk/dist/eip712.d.ts +257 -0
- package/packages/sdk/dist/eip712.d.ts.map +1 -0
- package/packages/sdk/dist/eip712.js +255 -0
- package/packages/sdk/dist/eip712.js.map +1 -0
- package/packages/sdk/dist/errors.d.ts +47 -0
- package/packages/sdk/dist/errors.d.ts.map +1 -0
- package/packages/sdk/dist/errors.js +168 -0
- package/packages/sdk/dist/errors.js.map +1 -0
- package/packages/sdk/dist/fees.d.ts +16 -0
- package/packages/sdk/dist/fees.d.ts.map +1 -0
- package/packages/sdk/dist/fees.js +51 -0
- package/packages/sdk/dist/fees.js.map +1 -0
- package/packages/sdk/dist/index.d.ts +23 -0
- package/packages/sdk/dist/index.d.ts.map +1 -0
- package/packages/sdk/dist/index.js +16 -0
- package/packages/sdk/dist/index.js.map +1 -0
- package/packages/sdk/dist/intelligence.d.ts +923 -0
- package/packages/sdk/dist/intelligence.d.ts.map +1 -0
- package/packages/sdk/dist/intelligence.js +357 -0
- package/packages/sdk/dist/intelligence.js.map +1 -0
- package/packages/sdk/dist/keyring.d.ts +60 -0
- package/packages/sdk/dist/keyring.d.ts.map +1 -0
- package/packages/sdk/dist/keyring.js +159 -0
- package/packages/sdk/dist/keyring.js.map +1 -0
- package/packages/sdk/dist/lifi.d.ts +83 -0
- package/packages/sdk/dist/lifi.d.ts.map +1 -0
- package/packages/sdk/dist/lifi.js +69 -0
- package/packages/sdk/dist/lifi.js.map +1 -0
- package/packages/sdk/dist/safe.d.ts +158 -0
- package/packages/sdk/dist/safe.d.ts.map +1 -0
- package/packages/sdk/dist/safe.js +163 -0
- package/packages/sdk/dist/safe.js.map +1 -0
- package/packages/sdk/dist/signing.d.ts +106 -0
- package/packages/sdk/dist/signing.d.ts.map +1 -0
- package/packages/sdk/dist/signing.js +2 -0
- package/packages/sdk/dist/signing.js.map +1 -0
- package/packages/sdk/dist/templates/ammLiquidity.d.ts +16 -0
- package/packages/sdk/dist/templates/ammLiquidity.d.ts.map +1 -0
- package/packages/sdk/dist/templates/ammLiquidity.js +60 -0
- package/packages/sdk/dist/templates/ammLiquidity.js.map +1 -0
- package/packages/sdk/dist/templates/approveAndCallBatch.d.ts +14 -0
- package/packages/sdk/dist/templates/approveAndCallBatch.d.ts.map +1 -0
- package/packages/sdk/dist/templates/approveAndCallBatch.js +53 -0
- package/packages/sdk/dist/templates/approveAndCallBatch.js.map +1 -0
- package/packages/sdk/dist/templates/boundedBorrow.d.ts +16 -0
- package/packages/sdk/dist/templates/boundedBorrow.d.ts.map +1 -0
- package/packages/sdk/dist/templates/boundedBorrow.js +56 -0
- package/packages/sdk/dist/templates/boundedBorrow.js.map +1 -0
- package/packages/sdk/dist/templates/boundedSwap.d.ts +16 -0
- package/packages/sdk/dist/templates/boundedSwap.d.ts.map +1 -0
- package/packages/sdk/dist/templates/boundedSwap.js +56 -0
- package/packages/sdk/dist/templates/boundedSwap.js.map +1 -0
- package/packages/sdk/dist/templates/defiBundle.d.ts +14 -0
- package/packages/sdk/dist/templates/defiBundle.d.ts.map +1 -0
- package/packages/sdk/dist/templates/defiBundle.js +52 -0
- package/packages/sdk/dist/templates/defiBundle.js.map +1 -0
- package/packages/sdk/dist/templates/index.d.ts +15 -0
- package/packages/sdk/dist/templates/index.d.ts.map +1 -0
- package/packages/sdk/dist/templates/index.js +8 -0
- package/packages/sdk/dist/templates/index.js.map +1 -0
- package/packages/sdk/dist/templates/pendle.d.ts +14 -0
- package/packages/sdk/dist/templates/pendle.d.ts.map +1 -0
- package/packages/sdk/dist/templates/pendle.js +52 -0
- package/packages/sdk/dist/templates/pendle.js.map +1 -0
- package/packages/sdk/dist/templates/transferTarget.d.ts +10 -0
- package/packages/sdk/dist/templates/transferTarget.d.ts.map +1 -0
- package/packages/sdk/dist/templates/transferTarget.js +42 -0
- package/packages/sdk/dist/templates/transferTarget.js.map +1 -0
- package/packages/sdk/dist/types.d.ts +382 -0
- package/packages/sdk/dist/types.d.ts.map +1 -0
- package/packages/sdk/dist/types.js +2 -0
- package/packages/sdk/dist/types.js.map +1 -0
- package/packages/sdk/package.json +52 -0
- package/packages/ui/dist/assets/Arc-VDBY7LNS-BChRXCXW.js +1 -0
- package/packages/ui/dist/assets/Brave-BRAKJXDS-mq-Xo37j.js +1 -0
- package/packages/ui/dist/assets/Browser-76IHF3Y2-BMhRaC5Z.js +1 -0
- package/packages/ui/dist/assets/Chrome-65Q5P54Y-DR9MQEVr.js +1 -0
- package/packages/ui/dist/assets/Edge-XSPUTORV-DEoZslQE.js +1 -0
- package/packages/ui/dist/assets/Firefox-AAHGJQIP-Bp_Hm04m.js +1 -0
- package/packages/ui/dist/assets/Linux-OO4TNCLJ-B0aw93n9.js +1 -0
- package/packages/ui/dist/assets/Macos-MW4AE7LN-Vvm8Drw3.js +1 -0
- package/packages/ui/dist/assets/Opera-KQZLSACL-Cwv5MDFy.js +1 -0
- package/packages/ui/dist/assets/Safari-ZPL37GXR-C4Ggg6rz.js +1 -0
- package/packages/ui/dist/assets/Windows-PPTHQER6-BlyV2p7Y.js +1 -0
- package/packages/ui/dist/assets/add-DaJhwIBV.js +15 -0
- package/packages/ui/dist/assets/all-wallets-BUxsqWXi.js +6 -0
- package/packages/ui/dist/assets/apechain-SX5YFU6N-q5qBv-mp.js +1 -0
- package/packages/ui/dist/assets/app-store-DkltwTqE.js +17 -0
- package/packages/ui/dist/assets/apple-owVOeaIT.js +18 -0
- package/packages/ui/dist/assets/ar_AR-LIPSOZP5-BQrIDibT.js +1519 -0
- package/packages/ui/dist/assets/arbitrum-WURIBY6W-CqVkHBr5.js +1 -0
- package/packages/ui/dist/assets/arrow-bottom-D2mmNJve.js +8 -0
- package/packages/ui/dist/assets/arrow-bottom-circle-CbNYijx-.js +11 -0
- package/packages/ui/dist/assets/arrow-left-DJB61s4C.js +8 -0
- package/packages/ui/dist/assets/arrow-right-BBrsQ9R4.js +8 -0
- package/packages/ui/dist/assets/arrow-top-Cil6bOc8.js +8 -0
- package/packages/ui/dist/assets/assets-Q6ZU7ZJ5-P8HioiAD.js +1 -0
- package/packages/ui/dist/assets/avalanche-KOMJD3XY-Dsn_JPR4.js +1 -0
- package/packages/ui/dist/assets/bank-CbwEmRo3.js +14 -0
- package/packages/ui/dist/assets/base-OAXLRA4F-CoYTVIiL.js +1 -0
- package/packages/ui/dist/assets/base-QS6CYWIN-CsjdbWCf.js +1 -0
- package/packages/ui/dist/assets/basic-CLNfjw3m.js +2128 -0
- package/packages/ui/dist/assets/berachain-NJECWIVC-DumxnFvf.js +1 -0
- package/packages/ui/dist/assets/blast-V555OVXZ-BbhJh1tj.js +1 -0
- package/packages/ui/dist/assets/browser-B5TtF4Pb.js +14 -0
- package/packages/ui/dist/assets/bsc-N647EYR2-B2nLKXWV.js +1 -0
- package/packages/ui/dist/assets/card-CO7BVB-C.js +14 -0
- package/packages/ui/dist/assets/ccip-2W7K3_J3.js +1 -0
- package/packages/ui/dist/assets/celo-GEP4TUHG-CenIBYLU.js +1 -0
- package/packages/ui/dist/assets/checkmark-BEtSHq9m.js +11 -0
- package/packages/ui/dist/assets/checkmark-bold-D9xGHzPE.js +8 -0
- package/packages/ui/dist/assets/chevron-bottom-BDztht6i.js +8 -0
- package/packages/ui/dist/assets/chevron-left-EV4GFNbc.js +8 -0
- package/packages/ui/dist/assets/chevron-right-B4_bB9oR.js +8 -0
- package/packages/ui/dist/assets/chevron-top-D54xPNzF.js +8 -0
- package/packages/ui/dist/assets/chrome-store-DYUpAJJq.js +61 -0
- package/packages/ui/dist/assets/clock-Ca1T1Soz.js +8 -0
- package/packages/ui/dist/assets/close-BZqWjurK.js +8 -0
- package/packages/ui/dist/assets/coinPlaceholder-e6fl2XDo.js +8 -0
- package/packages/ui/dist/assets/compass-DCLC7zIh.js +8 -0
- package/packages/ui/dist/assets/connect-UA7M4XW6-IY3X6Bmr.js +1 -0
- package/packages/ui/dist/assets/copy-Th2AaD-O.js +15 -0
- package/packages/ui/dist/assets/core-Ckx_cyuH.js +907 -0
- package/packages/ui/dist/assets/create-FASO7PVG-D_rvSpre.js +1 -0
- package/packages/ui/dist/assets/cronos-HJPAQTAE-BEOvlOC4.js +1 -0
- package/packages/ui/dist/assets/cursor-DV7rOqbJ.js +3 -0
- package/packages/ui/dist/assets/cursor-transparent-BKHeABKB.js +12 -0
- package/packages/ui/dist/assets/de_DE-YE3KOFHU-BRt5ztUe.js +1519 -0
- package/packages/ui/dist/assets/degen-FQQ4XGHB-CeHTs88l.js +1 -0
- package/packages/ui/dist/assets/desktop-CBjY8t6F.js +9 -0
- package/packages/ui/dist/assets/disconnect-DbSs2cli.js +8 -0
- package/packages/ui/dist/assets/discord-ZlLOAUkM.js +17 -0
- package/packages/ui/dist/assets/es_419-7LMPU7G4-DH7rM0yQ.js +1519 -0
- package/packages/ui/dist/assets/ethereum-RGGVA4PY-SWGOlkuk.js +1 -0
- package/packages/ui/dist/assets/etherscan-CKUrqWYN.js +6 -0
- package/packages/ui/dist/assets/events-CiKP71cK.js +1 -0
- package/packages/ui/dist/assets/exclamation-triangle-DA1QzFiO.js +4 -0
- package/packages/ui/dist/assets/extension-BVJkmvpJ.js +8 -0
- package/packages/ui/dist/assets/external-link-D_bsR7B2.js +8 -0
- package/packages/ui/dist/assets/facebook-CmFmhojx.js +26 -0
- package/packages/ui/dist/assets/fallback-Ofl6uSnB.js +1 -0
- package/packages/ui/dist/assets/farcaster-Co-M3Ss8.js +12 -0
- package/packages/ui/dist/assets/filters-B1WwNaFU.js +8 -0
- package/packages/ui/dist/assets/flow-5FQJFCTK-CUie2reO.js +1 -0
- package/packages/ui/dist/assets/fr_FR-VBJP3ZLL-B-_ocunw.js +1519 -0
- package/packages/ui/dist/assets/github-CP4fP6gn.js +18 -0
- package/packages/ui/dist/assets/gnosis-37ZC4RBL-B137OtHZ.js +1 -0
- package/packages/ui/dist/assets/google-CsOIXJ6V.js +18 -0
- package/packages/ui/dist/assets/gravity-J5YQHTYH-Bj6B0uod.js +1 -0
- package/packages/ui/dist/assets/hardhat-TX56IT5N-CV1FY-wE.js +1 -0
- package/packages/ui/dist/assets/help-circle-DiMkomdF.js +12 -0
- package/packages/ui/dist/assets/hi_IN-WBVD5XYI-D73g2UFs.js +1519 -0
- package/packages/ui/dist/assets/hyperevm-VKPAA4SA-CHwraEsx.js +1 -0
- package/packages/ui/dist/assets/id-lmscL5LX.js +12 -0
- package/packages/ui/dist/assets/id_ID-SBYANJ7G-Cjpa4ay6.js +1519 -0
- package/packages/ui/dist/assets/image-B-ubJrY5.js +4 -0
- package/packages/ui/dist/assets/index-BaukYv-x.js +16 -0
- package/packages/ui/dist/assets/index-CF0KMmke.js +395 -0
- package/packages/ui/dist/assets/index-CKxgNxS9.css +1 -0
- package/packages/ui/dist/assets/index-CZR1Qjhs.js +1 -0
- package/packages/ui/dist/assets/index-DVgfCzCo.js +1 -0
- package/packages/ui/dist/assets/index-Dbh5V1Z0.js +1 -0
- package/packages/ui/dist/assets/index-Q2Yai4Fe.js +2103 -0
- package/packages/ui/dist/assets/index.es-C78cE5SI.js +26 -0
- package/packages/ui/dist/assets/info-Cqg57EVo.js +3 -0
- package/packages/ui/dist/assets/info-circle-DkeSWNKV.js +12 -0
- package/packages/ui/dist/assets/ink-FZMYZWHG-62p-5IK5.js +1 -0
- package/packages/ui/dist/assets/ja_JP-ZRMWJV3I-DXbifiMm.js +1519 -0
- package/packages/ui/dist/assets/kaia-65D2U3PU-JmuLQ4gC.js +1 -0
- package/packages/ui/dist/assets/ko_KR-FR54RFUG-upinSHjQ.js +1519 -0
- package/packages/ui/dist/assets/lightbulb-DNlO4qKh.js +3 -0
- package/packages/ui/dist/assets/linea-QRMVQ5DY-DuI3vv0d.js +1 -0
- package/packages/ui/dist/assets/login-UP3DZBGS-Db_wM5oQ.js +1 -0
- package/packages/ui/dist/assets/mail-kVQ8Jb9Y.js +8 -0
- package/packages/ui/dist/assets/manta-SI27YFEJ-CpVOKa06.js +1 -0
- package/packages/ui/dist/assets/mantle-CKIUT334-DR2WgqzU.js +1 -0
- package/packages/ui/dist/assets/metaMaskWallet-EI6MED72-D5HFOsnz.js +1 -0
- package/packages/ui/dist/assets/metamask-sdk-CBalSvz7.js +542 -0
- package/packages/ui/dist/assets/mobile-BEteuhF7.js +9 -0
- package/packages/ui/dist/assets/monad-4KWC6TSS-DVXSkpiz.js +1 -0
- package/packages/ui/dist/assets/more-DBWmXQli.js +11 -0
- package/packages/ui/dist/assets/ms_MY-EZSGYYYQ-4cPLK-3L.js +1519 -0
- package/packages/ui/dist/assets/native-CJ5et6AR.js +1 -0
- package/packages/ui/dist/assets/network-placeholder-Dg1uUHiL.js +14 -0
- package/packages/ui/dist/assets/nftPlaceholder-i3AHSiD9.js +8 -0
- package/packages/ui/dist/assets/off-BtMm0fi2.js +8 -0
- package/packages/ui/dist/assets/optimism-HAF2GUT7-ec6Nqxs9.js +1 -0
- package/packages/ui/dist/assets/parseSignature-Cb5FlWWg.js +1 -0
- package/packages/ui/dist/assets/play-store-iKKkXa6a.js +32 -0
- package/packages/ui/dist/assets/plus-CA5NaRtb.js +13 -0
- package/packages/ui/dist/assets/polygon-WW6ZI7PM-DXlmm4L1.js +1 -0
- package/packages/ui/dist/assets/pt_BR-JQFQ3P4L-DOHfdcA2.js +1519 -0
- package/packages/ui/dist/assets/qr-code-D2kiqR7h.js +6 -0
- package/packages/ui/dist/assets/rainbowWallet-O26YNBMX-DUhYus-9.js +1 -0
- package/packages/ui/dist/assets/recycle-horizontal-Dcme7R03.js +9 -0
- package/packages/ui/dist/assets/refresh-Dega3sDp.js +8 -0
- package/packages/ui/dist/assets/refresh-S4T5V5GX-CwqIaaxK.js +1 -0
- package/packages/ui/dist/assets/reown-logo-xNkksyWJ.js +12 -0
- package/packages/ui/dist/assets/ronin-EMCPYXZT-N-QBHZdV.js +1 -0
- package/packages/ui/dist/assets/ru_RU-Z42UEJBP-Cvb2oWxQ.js +1519 -0
- package/packages/ui/dist/assets/safeWallet-5MNKTR5Z-D-5imDLD.js +1 -0
- package/packages/ui/dist/assets/sanko-RHQYXGM5-OX010CbN.js +1 -0
- package/packages/ui/dist/assets/scan-4UYSQ56Q-CjMz6-XC.js +1 -0
- package/packages/ui/dist/assets/scroll-5OBGQVOV-DJFECiai.js +1 -0
- package/packages/ui/dist/assets/search-HYl7NO8x.js +8 -0
- package/packages/ui/dist/assets/secp256k1-Cxd6_SiH.js +1 -0
- package/packages/ui/dist/assets/send-CJU8CUAo.js +15 -0
- package/packages/ui/dist/assets/sign-A7IJEUT5-CGsRnPrd.js +1 -0
- package/packages/ui/dist/assets/superposition-HG6MMR2Y-bRkgatRO.js +1 -0
- package/packages/ui/dist/assets/swapHorizontal-IMUKiUre.js +8 -0
- package/packages/ui/dist/assets/swapHorizontalBold-CNYnNJ9-.js +8 -0
- package/packages/ui/dist/assets/swapHorizontalMedium-B9VxEYsT.js +16 -0
- package/packages/ui/dist/assets/swapHorizontalRoundedBold-Dz33l_Jh.js +8 -0
- package/packages/ui/dist/assets/swapVertical-CHUmjVJ0.js +8 -0
- package/packages/ui/dist/assets/telegram-kl9S2mbU.js +16 -0
- package/packages/ui/dist/assets/th_TH-4YB4VSB2-BUipNP-V.js +1519 -0
- package/packages/ui/dist/assets/three-dots-U5lhA1Am.js +5 -0
- package/packages/ui/dist/assets/tr_TR-5FKHPPIO-D5jTpIm9.js +1519 -0
- package/packages/ui/dist/assets/twitch-KTEUWXEp.js +18 -0
- package/packages/ui/dist/assets/twitterIcon-BHiq8mRg.js +6 -0
- package/packages/ui/dist/assets/uk_UA-ZD4IBC52-DgnQrpzl.js +1519 -0
- package/packages/ui/dist/assets/unichain-C5BWO2ZY-BfguYsnu.js +1 -0
- package/packages/ui/dist/assets/verify-CfN-BXNd.js +8 -0
- package/packages/ui/dist/assets/verify-filled-DwZccetj.js +8 -0
- package/packages/ui/dist/assets/vi_VN-5EVRZKLY-x078672g.js +1519 -0
- package/packages/ui/dist/assets/w3m-modal-CS-PFqPE.js +642 -0
- package/packages/ui/dist/assets/wallet-DVlGkhOY.js +8 -0
- package/packages/ui/dist/assets/wallet-placeholder-CvR_iEWX.js +14 -0
- package/packages/ui/dist/assets/walletConnectWallet-YHWKVTDY-D3lyiczV.js +1 -0
- package/packages/ui/dist/assets/walletconnect-8pZBDvVI.js +30 -0
- package/packages/ui/dist/assets/warning-circle-ylLEE0Yp.js +12 -0
- package/packages/ui/dist/assets/x-C_TBsTMj.js +12 -0
- package/packages/ui/dist/assets/xdc-KJ3TDBYO-DNV6zchh.js +1 -0
- package/packages/ui/dist/assets/zetachain-TLDS5IPW-Udhyw16T.js +1 -0
- package/packages/ui/dist/assets/zh_CN-4XK5YJPR-Bt6Yz5Ek.js +1519 -0
- package/packages/ui/dist/assets/zh_HK-N4YN2WSI-Cvzl1V16.js +1519 -0
- package/packages/ui/dist/assets/zh_TW-CNCRXH6Z-BNelatfN.js +1519 -0
- package/packages/ui/dist/assets/zksync-DH7HK5U4-Dt4usFw6.js +1 -0
- package/packages/ui/dist/assets/zora-FYL5H3IO-iB4wygST.js +1 -0
- package/packages/ui/dist/index.html +14 -0
- package/scripts/check-docs.mjs +262 -0
- package/scripts/check-init.mjs +109 -0
- package/scripts/postinstall.js +366 -0
- package/templates/custom-mandate/.sail/contracts/interfaces/IPermission.sol +18 -0
- package/templates/custom-mandate/README.md +85 -0
- package/templates/custom-mandate/foundry.toml +8 -0
- package/templates/custom-mandate/mandates/BoundedCallPermission.sol +35 -0
- package/templates/custom-mandate/mandates/README.md +16 -0
- package/templates/dca-rebalancer/.cursor/rules +25 -0
- package/templates/dca-rebalancer/.env.example +6 -0
- package/templates/dca-rebalancer/.github/workflows/agent-tick.yml +32 -0
- package/templates/dca-rebalancer/.sail/.gitkeep +0 -0
- package/templates/dca-rebalancer/.sail/README.md +13 -0
- package/templates/dca-rebalancer/.sail/config.json +10 -0
- package/templates/dca-rebalancer/AGENTS.md +246 -0
- package/templates/dca-rebalancer/AGENT_PLAYBOOK.md +110 -0
- package/templates/dca-rebalancer/CLAUDE.md +2 -0
- package/templates/dca-rebalancer/README.md +16 -0
- package/templates/dca-rebalancer/_gitignore +13 -0
- package/templates/dca-rebalancer/docs/PERMISSION_MODEL.md +93 -0
- package/templates/dca-rebalancer/package.json +17 -0
- package/templates/dca-rebalancer/src/agent.ts +253 -0
- package/templates/dca-rebalancer/src/config.ts +27 -0
- package/templates/dca-rebalancer/src/mandate.ts +67 -0
- package/templates/dca-rebalancer/tsconfig.json +8 -0
- package/templates/dca-rebalancer/ui/README.md +3 -0
- package/templates/lifi-permissions/LifiBoundedApprovePermissionCloneable.sol +84 -0
- package/templates/lifi-permissions/LifiDiamondSwapPermissionCloneable.sol +97 -0
- package/templates/lifi-permissions/README.md +53 -0
|
@@ -0,0 +1 @@
|
|
|
1
|
+
var l="data:image/svg+xml,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20fill%3D%22none%22%20viewBox%3D%220%200%2028%2028%22%3E%3Cg%20transform%3D%22translate(0%2C0)%20scale(0.7)%22%3E%3Cg%20clip-path%3D%22url(%23a)%22%3E%3Cpath%20fill%3D%22%23000%22%20d%3D%22M0%200h40v40H0z%22%2F%3E%3Cpath%20fill%3D%22url(%23b)%22%20fill-opacity%3D%22.1%22%20d%3D%22M0%200h40v40H0z%22%2F%3E%3Cpath%20fill%3D%22%23fff%22%20fill-rule%3D%22evenodd%22%20d%3D%22m32.25%2020.005-6.954-6.922v5.066l-6.901%205.078h6.901v3.7l6.954-6.922ZM7.75%2020.005l6.954%206.923v-5.036l6.901-5.119h-6.901v-3.7L7.75%2020.004Z%22%20clip-rule%3D%22evenodd%22%2F%3E%3C%2Fg%3E%3Cdefs%3E%3ClinearGradient%20id%3D%22b%22%20x1%3D%220%22%20x2%3D%2220%22%20y1%3D%220%22%20y2%3D%2240%22%20gradientUnits%3D%22userSpaceOnUse%22%3E%3Cstop%20stop-color%3D%22%23fff%22%2F%3E%3Cstop%20offset%3D%221%22%20stop-color%3D%22%23fff%22%20stop-opacity%3D%220%22%2F%3E%3C%2FlinearGradient%3E%3CclipPath%20id%3D%22a%22%3E%3Cpath%20fill%3D%22%23fff%22%20d%3D%22M0%200h40v40H0z%22%2F%3E%3C%2FclipPath%3E%3C%2Fdefs%3E%3C%2Fg%3E%3C%2Fsvg%3E";export{l as default};
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
var o="data:image/svg+xml,%3Csvg%20xmlns%3D%22http%3A%2F%2Fwww.w3.org%2F2000%2Fsvg%22%20width%3D%2228%22%20height%3D%2228%22%20fill%3D%22none%22%3E%3Cg%20clip-path%3D%22url(%23a)%22%3E%3Cpath%20fill%3D%22url(%23b)%22%20d%3D%22M.943%2013.754c0%207.586%205.944%2013.755%2013.252%2013.755%207.308%200%2013.252-6.17%2013.252-13.755C27.44%206.17%2021.497%200%2014.195%200%206.887%200%20.943%206.17.943%2013.754Z%22%2F%3E%3Cpath%20fill%3D%22url(%23c)%22%20d%3D%22M.943%2013.754c0%207.586%205.944%2013.755%2013.252%2013.755%207.308%200%2013.252-6.17%2013.252-13.755C27.44%206.17%2021.497%200%2014.195%200%206.887%200%20.943%206.17.943%2013.754Z%22%2F%3E%3Cpath%20fill%3D%22url(%23d)%22%20d%3D%22M.943%2013.754c0%207.586%205.944%2013.755%2013.252%2013.755%207.308%200%2013.252-6.17%2013.252-13.755C27.44%206.17%2021.497%200%2014.195%200%206.887%200%20.943%206.17.943%2013.754Z%22%2F%3E%3C%2Fg%3E%3Cdefs%3E%3CradialGradient%20id%3D%22b%22%20cx%3D%220%22%20cy%3D%220%22%20r%3D%221%22%20gradientTransform%3D%22matrix(19.9547%200%200%2020.7113%2018.16%206.7)%22%20gradientUnits%3D%22userSpaceOnUse%22%3E%3Cstop%20offset%3D%22.005%22%20stop-color%3D%22%23fff%22%2F%3E%3Cstop%20offset%3D%22.458%22%20stop-color%3D%22%23B7D8C8%22%2F%3E%3Cstop%20offset%3D%22.656%22%20stop-color%3D%22%236D9487%22%2F%3E%3Cstop%20offset%3D%221%22%20stop-color%3D%22%234B4C3C%22%2F%3E%3C%2FradialGradient%3E%3CradialGradient%20id%3D%22c%22%20cx%3D%220%22%20cy%3D%220%22%20r%3D%221%22%20gradientTransform%3D%22matrix(19.9547%200%200%2020.7113%2018.16%206.7)%22%20gradientUnits%3D%22userSpaceOnUse%22%3E%3Cstop%20offset%3D%22.005%22%20stop-color%3D%22%23fff%22%2F%3E%3Cstop%20offset%3D%22.458%22%20stop-color%3D%22%23B5B4C6%22%2F%3E%3Cstop%20offset%3D%22.656%22%20stop-color%3D%22%239B8F8F%22%2F%3E%3Cstop%20offset%3D%221%22%20stop-color%3D%22%234B4C3C%22%2F%3E%3C%2FradialGradient%3E%3CradialGradient%20id%3D%22d%22%20cx%3D%220%22%20cy%3D%220%22%20r%3D%221%22%20gradientTransform%3D%22matrix(19.9547%200%200%2020.7113%2018.16%206.7)%22%20gradientUnits%3D%22userSpaceOnUse%22%3E%3Cstop%20offset%3D%22.156%22%20stop-color%3D%22%23DCC8D0%22%2F%3E%3Cstop%20offset%3D%22.302%22%20stop-color%3D%22%2378C8CF%22%2F%3E%3Cstop%20offset%3D%22.427%22%20stop-color%3D%22%234D959E%22%2F%3E%3Cstop%20offset%3D%22.557%22%20stop-color%3D%22%23305EB9%22%2F%3E%3Cstop%20offset%3D%22.797%22%20stop-color%3D%22%23311F12%22%2F%3E%3Cstop%20offset%3D%22.906%22%20stop-color%3D%22%23684232%22%2F%3E%3Cstop%20offset%3D%221%22%20stop-color%3D%22%232D1C13%22%2F%3E%3C%2FradialGradient%3E%3CclipPath%20id%3D%22a%22%3E%3Cpath%20fill%3D%22%23fff%22%20d%3D%22M0%200h28v28H0z%22%2F%3E%3C%2FclipPath%3E%3C%2Fdefs%3E%3C%2Fsvg%3E";export{o as default};
|
|
@@ -0,0 +1,14 @@
|
|
|
1
|
+
<!DOCTYPE html>
|
|
2
|
+
<html lang="en" style="color-scheme: dark">
|
|
3
|
+
<head>
|
|
4
|
+
<meta charset="UTF-8" />
|
|
5
|
+
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
6
|
+
<meta name="theme-color" content="#040b16" />
|
|
7
|
+
<title>Sail - Unlocking Personalized Money</title>
|
|
8
|
+
<script type="module" crossorigin src="/assets/index-Q2Yai4Fe.js"></script>
|
|
9
|
+
<link rel="stylesheet" crossorigin href="/assets/index-CKxgNxS9.css">
|
|
10
|
+
</head>
|
|
11
|
+
<body>
|
|
12
|
+
<div id="root"></div>
|
|
13
|
+
</body>
|
|
14
|
+
</html>
|
|
@@ -0,0 +1,262 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Doc-drift gate.
|
|
4
|
+
*
|
|
5
|
+
* The agent-facing docs (CLAUDE.md, AGENTS.md, AGENT_PLAYBOOK.md, the scaffolded
|
|
6
|
+
* template, package READMEs) tell an LLM agent which `sailor` commands and which
|
|
7
|
+
* `client.<ns>.<method>(...)` SDK calls to use. If a doc names a command or method
|
|
8
|
+
* that no longer exists, the agent confidently does the wrong thing. This script
|
|
9
|
+
* is the regression net: it derives the *real* surface from source and fails if a
|
|
10
|
+
* doc references something that isn't there.
|
|
11
|
+
*
|
|
12
|
+
* - CLI commands ← parsed from packages/cli/src/index.ts (commander tree)
|
|
13
|
+
* - SDK methods ← parsed from packages/sdk/src/client.ts (namespace classes)
|
|
14
|
+
*
|
|
15
|
+
* Only `sailor …` is validated, never the SailFramework `sail …` binary that some
|
|
16
|
+
* docs reference. Only references inside `inline code` or ```fenced``` blocks are
|
|
17
|
+
* checked, so prose ("the sailor CLI") never trips it.
|
|
18
|
+
*
|
|
19
|
+
* Run: `node scripts/check-docs.mjs` (or `pnpm docs:check`). Exit 1 on any miss.
|
|
20
|
+
*/
|
|
21
|
+
|
|
22
|
+
import { readFileSync, readdirSync, statSync } from "node:fs";
|
|
23
|
+
import { dirname, join, relative } from "node:path";
|
|
24
|
+
import { fileURLToPath } from "node:url";
|
|
25
|
+
|
|
26
|
+
const ROOT = join(dirname(fileURLToPath(import.meta.url)), "..");
|
|
27
|
+
const rel = (p) => relative(ROOT, p);
|
|
28
|
+
|
|
29
|
+
// ── 1. Derive the real CLI command surface ──────────────────────────────────
|
|
30
|
+
|
|
31
|
+
/**
|
|
32
|
+
* Parse the commander tree from index.ts into:
|
|
33
|
+
* leaves — Set of full top-level invocations ("init", "doctor", "dispatch preview")
|
|
34
|
+
* groups — Map<groupName, Set<subcommand>> (e.g. "mandate" → {prepare, sign, …})
|
|
35
|
+
*/
|
|
36
|
+
function parseCliSurface() {
|
|
37
|
+
const src = readFileSync(join(ROOT, "packages/cli/src/index.ts"), "utf-8");
|
|
38
|
+
const leaves = new Set();
|
|
39
|
+
const groups = new Map();
|
|
40
|
+
const varToGroup = new Map(); // commander variable name → group command name
|
|
41
|
+
|
|
42
|
+
// `const ui = program.command("ui")` — a group declared on a variable.
|
|
43
|
+
for (const m of src.matchAll(/const\s+(\w+)\s*=\s*program\s*\.command\(\s*"([^"]+)"/g)) {
|
|
44
|
+
const groupName = m[2].split(/\s+/)[0];
|
|
45
|
+
varToGroup.set(m[1], groupName);
|
|
46
|
+
if (!groups.has(groupName)) groups.set(groupName, new Set());
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
// `program.command("init [dir]")` — a top-level leaf (not assigned to a var).
|
|
50
|
+
for (const m of src.matchAll(/program\s*\.command\(\s*"([^"]+)"/g)) {
|
|
51
|
+
const name = m[1].split(/\s+/)[0];
|
|
52
|
+
// Multi-word leaf names (e.g. the "dispatch preview" stub) keep their full form too.
|
|
53
|
+
const full = m[1]
|
|
54
|
+
.replace(/\s*\[.*$/, "")
|
|
55
|
+
.replace(/\s*<.*$/, "")
|
|
56
|
+
.trim();
|
|
57
|
+
if (!groups.has(name)) {
|
|
58
|
+
leaves.add(name);
|
|
59
|
+
if (full.includes(" ")) leaves.add(full);
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
// `stub("setup", …)` / `stub("dispatch preview", …)` — registered top-level commands.
|
|
64
|
+
for (const m of src.matchAll(/\bstub\(\s*"([^"]+)"/g)) {
|
|
65
|
+
leaves.add(m[1]);
|
|
66
|
+
leaves.add(m[1].split(/\s+/)[0]);
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
// `ui.command("start")` / `mandate.command("prepare")` — subcommands of a group.
|
|
70
|
+
for (const m of src.matchAll(/(\w+)\s*\.command\(\s*"([^"]+)"/g)) {
|
|
71
|
+
const groupName = varToGroup.get(m[1]);
|
|
72
|
+
if (!groupName) continue; // not a known group variable (e.g. program.command handled above)
|
|
73
|
+
const sub = m[2].split(/\s+/)[0];
|
|
74
|
+
groups.get(groupName).add(sub);
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
return { leaves, groups };
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// ── 2. Derive the real SDK surface ───────────────────────────────────────────
|
|
81
|
+
|
|
82
|
+
/**
|
|
83
|
+
* Parse client.ts into:
|
|
84
|
+
* namespaces — Map<propertyName, Set<methodName>> (account → {create, get, …})
|
|
85
|
+
* clientMethods — Set of direct SailorClient methods (capabilities, withSigner, …)
|
|
86
|
+
*/
|
|
87
|
+
function parseSdkSurface() {
|
|
88
|
+
const src = readFileSync(join(ROOT, "packages/sdk/src/client.ts"), "utf-8");
|
|
89
|
+
const lines = src.split("\n");
|
|
90
|
+
|
|
91
|
+
// Map namespace *class* → set of method names, by slicing each class body.
|
|
92
|
+
const classMethods = new Map();
|
|
93
|
+
let current = null;
|
|
94
|
+
let depth = 0;
|
|
95
|
+
const reserved = new Set([
|
|
96
|
+
"if",
|
|
97
|
+
"for",
|
|
98
|
+
"while",
|
|
99
|
+
"switch",
|
|
100
|
+
"catch",
|
|
101
|
+
"return",
|
|
102
|
+
"super",
|
|
103
|
+
"constructor",
|
|
104
|
+
"function",
|
|
105
|
+
"await",
|
|
106
|
+
"typeof",
|
|
107
|
+
"new",
|
|
108
|
+
]);
|
|
109
|
+
for (const line of lines) {
|
|
110
|
+
const classMatch = line.match(/^\s*(?:export\s+)?(?:abstract\s+)?class\s+(\w+)/);
|
|
111
|
+
if (classMatch) {
|
|
112
|
+
current = classMatch[1];
|
|
113
|
+
classMethods.set(current, new Set());
|
|
114
|
+
depth = 0;
|
|
115
|
+
}
|
|
116
|
+
if (current) {
|
|
117
|
+
// Method declarations live at class-body depth 1.
|
|
118
|
+
const methodMatch = line.match(
|
|
119
|
+
/^\s{2}(?:public\s+|private\s+|protected\s+)?(?:async\s+)?(?:get\s+)?(\w+)\s*[(<]/,
|
|
120
|
+
);
|
|
121
|
+
if (methodMatch && !reserved.has(methodMatch[1])) {
|
|
122
|
+
classMethods.get(current).add(methodMatch[1]);
|
|
123
|
+
}
|
|
124
|
+
depth += (line.match(/{/g)?.length ?? 0) - (line.match(/}/g)?.length ?? 0);
|
|
125
|
+
if (depth <= 0 && line.includes("}")) current = null;
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// Map runtime property name → class: `this.account = new AccountNamespace(`
|
|
130
|
+
const namespaces = new Map();
|
|
131
|
+
for (const m of src.matchAll(/this\.(\w+)\s*=\s*new\s+(\w+)\s*\(/g)) {
|
|
132
|
+
const [, prop, cls] = m;
|
|
133
|
+
if (classMethods.has(cls)) namespaces.set(prop, classMethods.get(cls));
|
|
134
|
+
}
|
|
135
|
+
// `this.dispatch = dispatch;` — assigned from a local built earlier in the ctor.
|
|
136
|
+
for (const m of src.matchAll(/this\.(\w+)\s*=\s*(\w+);/g)) {
|
|
137
|
+
const [, prop, localVar] = m;
|
|
138
|
+
if (namespaces.has(prop)) continue;
|
|
139
|
+
// Resolve the local: `const dispatch = new DispatchNamespace(`
|
|
140
|
+
const localDecl = new RegExp(`(?:const|let)\\s+${localVar}\\s*=\\s*new\\s+(\\w+)\\s*\\(`);
|
|
141
|
+
const lm = src.match(localDecl);
|
|
142
|
+
if (lm && classMethods.has(lm[1])) namespaces.set(prop, classMethods.get(lm[1]));
|
|
143
|
+
}
|
|
144
|
+
|
|
145
|
+
// Direct methods on the SailorClient class itself.
|
|
146
|
+
const clientMethods = classMethods.get("SailorClient") ?? new Set();
|
|
147
|
+
|
|
148
|
+
return { namespaces, clientMethods };
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
// ── 3. Collect doc files + extract code references ───────────────────────────
|
|
152
|
+
|
|
153
|
+
const DOC_GLOBS = [
|
|
154
|
+
"CLAUDE.md",
|
|
155
|
+
"AGENTS.md",
|
|
156
|
+
"AGENT_PLAYBOOK.md",
|
|
157
|
+
"README.md",
|
|
158
|
+
"docs",
|
|
159
|
+
"templates",
|
|
160
|
+
"packages/cli/README.md",
|
|
161
|
+
"packages/sdk/README.md",
|
|
162
|
+
"packages/chains/README.md",
|
|
163
|
+
"packages/create-app/README.md",
|
|
164
|
+
];
|
|
165
|
+
|
|
166
|
+
function collectDocs() {
|
|
167
|
+
const files = [];
|
|
168
|
+
const walk = (p) => {
|
|
169
|
+
const st = statSync(p, { throwIfNoEntry: false });
|
|
170
|
+
if (!st) return;
|
|
171
|
+
if (st.isDirectory()) {
|
|
172
|
+
for (const e of readdirSync(p)) {
|
|
173
|
+
if (e === "node_modules" || e === "dist" || e.startsWith(".git")) continue;
|
|
174
|
+
walk(join(p, e));
|
|
175
|
+
}
|
|
176
|
+
} else if (p.endsWith(".md")) {
|
|
177
|
+
files.push(p);
|
|
178
|
+
}
|
|
179
|
+
};
|
|
180
|
+
for (const g of DOC_GLOBS) walk(join(ROOT, g));
|
|
181
|
+
return [...new Set(files)];
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
/** Pull the text of every `inline` and ```fenced``` code region from markdown. */
|
|
185
|
+
function codeRegions(md) {
|
|
186
|
+
const regions = [];
|
|
187
|
+
// Fenced blocks first, then strip them so inline matching doesn't double-count.
|
|
188
|
+
const rest = md.replace(/```[\s\S]*?```/g, (block) => {
|
|
189
|
+
regions.push(block.replace(/```/g, ""));
|
|
190
|
+
return "\n";
|
|
191
|
+
});
|
|
192
|
+
for (const m of rest.matchAll(/`([^`\n]+)`/g)) regions.push(m[1]);
|
|
193
|
+
return regions;
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
// ── 4. Validate ──────────────────────────────────────────────────────────────
|
|
197
|
+
|
|
198
|
+
function main() {
|
|
199
|
+
const cli = parseCliSurface();
|
|
200
|
+
const sdk = parseSdkSurface();
|
|
201
|
+
const docs = collectDocs();
|
|
202
|
+
const errors = [];
|
|
203
|
+
|
|
204
|
+
for (const file of docs) {
|
|
205
|
+
const md = readFileSync(file, "utf-8");
|
|
206
|
+
for (const code of codeRegions(md)) {
|
|
207
|
+
for (const lineRaw of code.split("\n")) {
|
|
208
|
+
const line = lineRaw.trim();
|
|
209
|
+
|
|
210
|
+
// ── CLI: `sailor <word1> [<word2>]` ──────────────────────────────────
|
|
211
|
+
for (const m of line.matchAll(/\bsailor\s+([a-z][\w-]*)(?:\s+([a-z][\w-]*))?/g)) {
|
|
212
|
+
const [, w1, w2] = m;
|
|
213
|
+
const two = w2 ? `${w1} ${w2}` : null;
|
|
214
|
+
if (cli.leaves.has(w1) || (two && cli.leaves.has(two))) continue; // top-level leaf
|
|
215
|
+
if (cli.groups.has(w1)) {
|
|
216
|
+
// Group: if a subcommand word follows (and isn't a flag), it must exist.
|
|
217
|
+
if (!w2 || w2.startsWith("-")) continue; // bare group invocation, or a flag
|
|
218
|
+
if (cli.groups.get(w1).has(w2)) continue;
|
|
219
|
+
errors.push(
|
|
220
|
+
`${rel(file)}: \`sailor ${w1} ${w2}\` — "${w2}" is not a subcommand of "${w1}" (have: ${[...cli.groups.get(w1)].join(", ")})`,
|
|
221
|
+
);
|
|
222
|
+
continue;
|
|
223
|
+
}
|
|
224
|
+
errors.push(`${rel(file)}: \`sailor ${w1}\` — unknown command`);
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
// ── SDK: `client.<ns>[.<method>]` ────────────────────────────────────
|
|
228
|
+
for (const m of line.matchAll(/\bclient\.(\w+)(?:\.(\w+))?/g)) {
|
|
229
|
+
const [, ns, method] = m;
|
|
230
|
+
if (sdk.namespaces.has(ns)) {
|
|
231
|
+
if (!method) continue;
|
|
232
|
+
if (sdk.namespaces.get(ns).has(method)) continue;
|
|
233
|
+
errors.push(
|
|
234
|
+
`${rel(file)}: \`client.${ns}.${method}\` — "${method}" is not a method of the "${ns}" namespace (have: ${[...sdk.namespaces.get(ns)].join(", ")})`,
|
|
235
|
+
);
|
|
236
|
+
continue;
|
|
237
|
+
}
|
|
238
|
+
if (sdk.clientMethods.has(ns)) continue; // direct client method, e.g. capabilities/withSigner
|
|
239
|
+
errors.push(`${rel(file)}: \`client.${ns}\` — unknown SDK namespace or method`);
|
|
240
|
+
}
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
// ── Report ───────────────────────────────────────────────────────────────
|
|
246
|
+
const cliCount = cli.leaves.size + [...cli.groups.values()].reduce((n, s) => n + s.size, 0);
|
|
247
|
+
const sdkCount =
|
|
248
|
+
[...sdk.namespaces.values()].reduce((n, s) => n + s.size, 0) + sdk.clientMethods.size;
|
|
249
|
+
console.log(
|
|
250
|
+
`Doc-drift check: ${docs.length} docs vs ${cliCount} CLI commands, ${sdkCount} SDK methods`,
|
|
251
|
+
);
|
|
252
|
+
|
|
253
|
+
if (errors.length > 0) {
|
|
254
|
+
console.error(`\n✗ ${errors.length} stale reference(s):\n`);
|
|
255
|
+
for (const e of [...new Set(errors)].sort()) console.error(` ${e}`);
|
|
256
|
+
console.error("\nFix the doc, or update the CLI/SDK so the referenced surface exists.");
|
|
257
|
+
process.exit(1);
|
|
258
|
+
}
|
|
259
|
+
console.log("✓ Every sailor command and client.* method referenced in docs exists.");
|
|
260
|
+
}
|
|
261
|
+
|
|
262
|
+
main();
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* `sailor init` smoke test.
|
|
4
|
+
*
|
|
5
|
+
* Scaffolds a fresh project from the in-tree CLI bundle into a temp dir and
|
|
6
|
+
* asserts the scaffold succeeded. This exists to catch the class of regression
|
|
7
|
+
* the doc-drift gate structurally cannot — e.g. `packageRoot()` resolving to a
|
|
8
|
+
* `bin.sailor` package that ships no `templates/`, which made `init` fail from a
|
|
9
|
+
* monorepo checkout with "Template ... not found. Available: none".
|
|
10
|
+
*
|
|
11
|
+
* It runs the REAL built bundle from a monorepo layout, which is exactly the
|
|
12
|
+
* in-tree path that broke before. Pure Node + child_process; the only build
|
|
13
|
+
* dependency is the CLI bundle (`pnpm --filter sailor build`).
|
|
14
|
+
*
|
|
15
|
+
* Run: node scripts/check-init.mjs (CI builds the CLI first)
|
|
16
|
+
* Exit: 0 = scaffold OK, 1 = failure (prints what was missing).
|
|
17
|
+
*/
|
|
18
|
+
|
|
19
|
+
import { execFileSync } from "node:child_process";
|
|
20
|
+
import fs from "node:fs";
|
|
21
|
+
import os from "node:os";
|
|
22
|
+
import path from "node:path";
|
|
23
|
+
import { fileURLToPath } from "node:url";
|
|
24
|
+
|
|
25
|
+
const ROOT = path.join(path.dirname(fileURLToPath(import.meta.url)), "..");
|
|
26
|
+
const BUNDLE = path.join(ROOT, "packages/cli/dist/index.cjs");
|
|
27
|
+
const PROJECT = "smoke-agent";
|
|
28
|
+
|
|
29
|
+
function fail(msg) {
|
|
30
|
+
console.error(`✗ init smoke test FAILED: ${msg}`);
|
|
31
|
+
process.exit(1);
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
if (!fs.existsSync(BUNDLE)) {
|
|
35
|
+
fail(`CLI bundle not found at ${BUNDLE}.\n Build it first: pnpm --filter sailor build`);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
// Scaffold into a temp dir. `init` requires the destination to live inside the
|
|
39
|
+
// process cwd, so we run the bundle with cwd set to a fresh temp root.
|
|
40
|
+
const tmpRoot = fs.mkdtempSync(path.join(os.tmpdir(), "sailor-init-smoke-"));
|
|
41
|
+
const dest = path.join(tmpRoot, PROJECT);
|
|
42
|
+
|
|
43
|
+
try {
|
|
44
|
+
let stdout = "";
|
|
45
|
+
try {
|
|
46
|
+
stdout = execFileSync(process.execPath, [BUNDLE, "init", PROJECT], {
|
|
47
|
+
cwd: tmpRoot,
|
|
48
|
+
encoding: "utf-8",
|
|
49
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
50
|
+
});
|
|
51
|
+
} catch (err) {
|
|
52
|
+
const out = `${err.stdout ?? ""}${err.stderr ?? ""}`.trim();
|
|
53
|
+
fail(`\`sailor init ${PROJECT}\` exited non-zero.\n ${out || err.message}`);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
// A successful fresh init prints the welcome + next steps, ending with the
|
|
57
|
+
// `Say: "start"` call-to-action. (Older builds printed "Done!".)
|
|
58
|
+
if (!/Say: "start"/.test(stdout)) {
|
|
59
|
+
fail(`init did not report success.\n stdout: ${stdout.trim()}`);
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
// Assert the scaffold landed.
|
|
63
|
+
const mustExist = [
|
|
64
|
+
".sail/config.json",
|
|
65
|
+
"package.json",
|
|
66
|
+
"foundry.toml",
|
|
67
|
+
"mandates",
|
|
68
|
+
"AGENTS.md",
|
|
69
|
+
];
|
|
70
|
+
for (const rel of mustExist) {
|
|
71
|
+
if (!fs.existsSync(path.join(dest, rel))) fail(`expected scaffolded "${rel}" — not found`);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
// config.json is valid JSON named after the project.
|
|
75
|
+
const config = JSON.parse(fs.readFileSync(path.join(dest, ".sail/config.json"), "utf-8"));
|
|
76
|
+
if (config.name !== PROJECT) fail(`config.json name is "${config.name}", expected "${PROJECT}"`);
|
|
77
|
+
|
|
78
|
+
// package.json is valid, renamed, and the workspace protocol was resolved away
|
|
79
|
+
// (a leftover "workspace:*" would make the scaffold un-installable for users).
|
|
80
|
+
const pkg = JSON.parse(fs.readFileSync(path.join(dest, "package.json"), "utf-8"));
|
|
81
|
+
if (pkg.name !== PROJECT) fail(`package.json name is "${pkg.name}", expected "${PROJECT}"`);
|
|
82
|
+
if (pkg.dependencies?.["@sail/sdk"] === "workspace:*") {
|
|
83
|
+
fail(`package.json still has "@sail/sdk": "workspace:*" — init did not resolve it`);
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
// Regression guard: an absolute path outside the cwd must be REJECTED, not
|
|
87
|
+
// silently nested into `<cwd>/<abs path>`. (Pre-fix, `path.join` swallowed the
|
|
88
|
+
// leading slash and scaffolded a bogus nested tree while printing success.)
|
|
89
|
+
const outside = path.join(os.tmpdir(), "sailor-init-outside", "agent");
|
|
90
|
+
let rejected = false;
|
|
91
|
+
try {
|
|
92
|
+
execFileSync(process.execPath, [BUNDLE, "init", outside], {
|
|
93
|
+
cwd: tmpRoot,
|
|
94
|
+
stdio: ["ignore", "pipe", "pipe"],
|
|
95
|
+
});
|
|
96
|
+
} catch {
|
|
97
|
+
rejected = true; // non-zero exit = correctly refused
|
|
98
|
+
}
|
|
99
|
+
if (!rejected) fail(`an absolute path outside cwd ("${outside}") was accepted — should be rejected`);
|
|
100
|
+
if (fs.existsSync(path.join(outside, ".sail/config.json"))) {
|
|
101
|
+
fail(`init scaffolded into an out-of-cwd absolute path "${outside}"`);
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
console.log(`✓ init smoke test passed — scaffolded ${PROJECT}/ from the in-tree bundle`);
|
|
105
|
+
console.log("✓ init guard passed — absolute path outside cwd rejected, not silently nested");
|
|
106
|
+
} finally {
|
|
107
|
+
fs.rmSync(path.join(os.tmpdir(), "sailor-init-outside"), { recursive: true, force: true });
|
|
108
|
+
fs.rmSync(tmpRoot, { recursive: true, force: true });
|
|
109
|
+
}
|