@dev-anywhere/proxy 0.0.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 +21 -0
- package/README.md +113 -0
- package/assets/fonts/sarasa-fixed-sc/00c59ca448ea49756112128d6a43b127.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/011decfe3e67ab099af86e86a15e4f12.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0133a2c4604dd809764dc749af72dc79.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/02928e8a3c2bf5ae6677bdf552664108.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/02eb3185bd0fd343bb2ae5a13260e805.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/04248a8ea5cd1204f875bc7661f1096c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/059276218fbb9297f1668b1e37611796.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/05c6ae1e7580a1eff4be47341a155474.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0681e7debe2dd44962f142b5c454676e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/06ec078ae766d56e900c2837556eaa21.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0765562d298aed694539cfcc4e26537c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/090128a865f81baab82dfa2776ef9d39.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/098ff8830b5d779d8656e7ca93253c12.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/09b4eba4363eb15092534724123a78f2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0a32f2f6ed1a99ffa5e833e6891d1f82.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0c1c080e1ac07d6a49dca531de9725a4.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0d022c7c63996b8e5dc70b7fd7b0689d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/0e7c7848d16895e68f79a55fd7b8c29d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1021bfe59ce4f905808dd5c3ea3fecc4.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/103c3dd7a5ee0d3be6cab3d04bd40c96.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/104a138cdd04cdbf3f786d2abeeed70a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/105c39ea073d59b2e2f5e16ca68b2d71.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1113c82b3ef6588e32cf20d7bd723355.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1284d569e10290971bf3200bf4ddf94f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/13401a392cffb9f2f7254e91bf44a106.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1364ce313965ebf440ff2f0311cd4c83.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1641d3bae07fd8d369349f78dd6c9c40.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/187edd679db5112bd2699bf1adad4a32.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1918eaee7987c39db5198d637623a538.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1aa2711726a364e3ea6fb7ca8a95d950.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1ab01fa7bd43e3242f3c9d5b6f73dc90.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1ae20f7adc1aa93e95c1842ce3b0c329.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1c16e221d67a71a270f700e31e406230.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1c40fcc59ed13b4b861844a8feb8205b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1d0e76f87701ab6daecb38b5aff5e35c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1d18a292fa9adaae7261de62ab3d5b83.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/1f99a7c91c71e6303282aa66c99c6670.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/20e0dab161488105b7fa7852470503b1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/210b0efe54084a98b88028a6f3f35b0d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2116632ed1a3bdf3bf0106e1703c0781.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/22b3a91f654e334e2e7fbe3780f67764.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/24c5d226f02d96f03a7d050b0fdfd1d1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/254d3a725f55d7111abbafa05858f2d6.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/26506a3ce2dbbfbfde40d11d65230272.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/268f0d094899c94260e840ac55320e20.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/277fc44cecacaf16cefce4681495bdce.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/27b773ac20ca5fa1bdd09d2109dfd5dc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2803a14ad17e6753704504009dad3333.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2a0d78f9037bc2b5846bb4bcb42bfedc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2ac0bf1c4f9ad6e69bc01259f6e17058.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2b01deb59e13f60e307d370b674b7b3b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2b74162a48979b6282e4893199f1346d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2d11bafb03690fc87494426325d90414.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/2dbf9030d7fa41019c78d5a717726261.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/30ce0c911f54ecd538247018d5289bb6.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/31bc5b0f3eb1e0f0845f37ae39f1a43e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/31cdb9be17f46537a2958ba142c9765f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/31d182ea6704bc50689becec8f417674.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/31f6abd51bf73062085ad82253fc302d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/32d1abf03444175b162eadfef3a8e937.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/3397f69dc93262cf856a7c836fb33cb1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/349035f271962ecacc5fa5cd56930c44.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/3525d0f8fb4e24d09024d35992e9095d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/393876ac692b10f5ab7190590b062745.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/3bf37030a116e7377dcaf47356f08ef1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/3c953d2e03b93436ee5c522ce3a27e1a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/3fe795fd56b36a734a31613b1fed5441.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/401e69f55dc74b2e6fb774dcbba811da.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/404518f4b01a2fe38462bd04bfd76d35.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/404fe750213a60cbd07016ab43aacb71.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/41491053defe7688d8d53bfc313c6974.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/420121d03c1887873aa0eaa37a534e0c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/42dffba83a406a85b208fba09e465c4d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/44b74f10fbe44658d1acb6627b3a8b94.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/450be7e82a27dca52c8c1c582fcf182d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/454ab2db1835c113f7b99f366a627247.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/45ec91cf9684c378adb99130acd6477e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4610b2980f41ab106bfe74b6951b34bb.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/46cad9f32908fa67609b51b74bc5693b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4a51964240b7e2c93d276dfef3794b5b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4b800716e356efb3e4b429085bcdb61a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4d8dad1e6c57ca3629d21230981cbb0e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4e625675fc1cb257e07a9ff2e331a9fe.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/4f2207b0342d6e2dd4290614e53cf0c5.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/50b0854263c0cf842979ce447020051a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/552fe6d8e7144695a8fb38b8760295df.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/56175fa000590397d0ec4074c1f1e9f2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/570f8c343327a818e4f0b0c4962ba4a1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/579fbe9abeb15d0e96ee4770e6719719.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/57c37c54a9699166f03c0e6f871e6735.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/58e7c2324d8d292d58534d9f236f1552.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/59584acacc4c1852a6814f7e389ea191.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/598705446ef817283f04a709e974d473.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/59e93050018a5206bebf980e0f52fa78.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/5c129ed60734fd511937f3e72a07f936.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/5c40c780ef301baacb66efb9750e258b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/5d1a982cb896e55ee96ada379c4a1d0c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/5eba8579b88524026f6f6ec30cb81baa.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/601ad6234d43753b4e3efe8486b315c1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/603b9e361dc9dd05250221356255a176.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6133adaaf70a9467188b7e9e34039480.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6158d9e8e90c6a553485c49e8adc991c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/63c707f20b4af887788bfe53c6eee3ec.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/63c8475e9aa8acc0033fde9f792b8acf.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/64466c4274848cdee185275fb52660df.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/652d2da7b3c1dc0a70efdbb85e820055.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6653b549c4d40a370724f6f32223e8a7.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6734957b7e0bdd9f9e66cf24be7dc058.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/689a7d17cf5f82d7f148d660a21df759.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6a864ec605e3b28cfdebf1d8fab2451e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6b4101e34138d999fd3ba9d2684ddebf.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6b70494a68af284b2f21967109bf0311.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6dde2f1140ce0a20159b7c1625bc3b45.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/6faa59cc7dff0a7751347b5baa8a5015.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/70364e9456886f13e2d5c9674ffb08f4.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/736f91321c58d7a9316ea9cddf4d83a7.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7458762c6210651559b6019d5508ae2c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7460d9b012420e17a4bbfd2cb6796468.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/75a0c446c8bf8e4051d7cd65ed17e114.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/77ee59e7be1d66b59e87880e9c283491.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/782cda0bc3f24b5553f93c3d5b6fe581.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/78e84df88175c20e7f2562e4cdaad257.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/79e46c399f5e7383f853281b499269ee.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7acff1baf69f6628c6372be0554191f1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7c5323aa737b291adacceecedac3e56a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7c9c9097a200464fd8ccec8eafc80178.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7f77666d943dd01e947a2e35ffed945f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/7f7e1463e29e6528378ed51589c4ebc0.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/80d4285084348d043a775221eb1c93b1.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/81258e5d1abdeafe44be69ad32f74c19.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/83a8da8e1de25f9eb39fe9d980e04a5d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/83be0efd62f30539368d37cf7529f95c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/84266b5a67707c745ce2c7101c559735.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/84905ac6e9d149bcacc34e2a71b82a67.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/852ae9183bf83dc2a2414290d3aa0236.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/85f5aa76f5f1e09b6852af36ec781960.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/860a1c6d20d4e631bd8e80735f5f3581.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8afa5d1ab930027046ad09ec6349ab31.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8b81739358772947619c9a58575c863a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8b9212f06a935902e9a4d8ad435c6a6e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8cc210fa91949bb44ea2bf607991b7bf.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8f0b853b5728d58ac137870b3fd7ab02.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8f0cdbad1758d5f68a48effbca6f74f8.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/8f5261cd877891595879479ee2440ae4.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/910a758ea12189f5a6a1e025fca33c5a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/911993a058e817f1a231fbac27b3781c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/91daedcfd8d0745222bf82c1fd310a33.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/943d6560828548692eeb4a533aa89494.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/945d0c206221572e76cc03fdd474685a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/946411ab6f4f655273da613188359c06.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/9514ffa1f75fb1ad995392125c963b1b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/971106de1248ba224fbfe50cf5c8a3dc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/9714a586ded7b76747709b1fbe32c344.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/98b75b8dba788b62ef809337088ef030.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/98b7b436090e7874e2affbfd4ad28fd9.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/9a952ea6da219d92841b5503d43c746e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a0eabe8e58be580b4a5b0db87e3d282b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a11e9313ebf3147b1622c47e8cb1bacd.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a2030c8eca8b7b75cf8d6fbbf29940cb.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a2e0e75587d53e7e6c50da180ad76815.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a3e96d6e1a2bdfe27d35140d1cb84600.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a645858417844eaeb35eec60463a8758.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a691a029b0ffe2f516dc0ba2680a7b24.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a8731b06984003e15944af2045bac0ba.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/a8b5a37da8e22efc24e1d51e3dd133f2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/aa4030d4e6a032ebd37ab080a87f0b15.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ab0f237ab351bf258b3579011285bdb2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ac3a84416815f292543ac24c909b8f53.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ac4188ae92c726da6cce789716706455.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ac9e1d7b7d0e738c0965e0c37a171594.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ad0efa49e97c104bb4f7d2a5d15eabef.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ad3ac003a089fb00191c1e00aa56061e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ae481a90ce6dc650ffb8e777ab9fc089.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/aef606698752748d6b411ab230cd1760.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/aefc6f1da14ea5d5daf751fbf71b29a2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/afe37b0740a4f9cc8f6a8aa5515a7826.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b1a36f37842e3cd3981ee51b5d9b3edc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b3791b3fb0de0352fb1265cb02a021fb.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b37e11130ac85f74432248a20b134622.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b39422dfd00810613d15bd6572b05aa3.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b3e514ad857c99211370e0bc572f776f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b4bd13b2c8086a5320644893d557b747.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b6b23d001929bf0450f2ee16e9236f1f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b78cefc4f6a28eb5b5dd551dd873bcbd.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b7e35bfc1e67312ea4f7399d87101681.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b8493cdc783af55e9c50fc16c9b55bc3.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/b8e187dc66c9864eb348ee91951eacdf.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/bbf72353d229f1e5d6ccea65888451fe.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/bc6f1bd38107e5c152f4d568e75b3cf0.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/bc884e50f2cf284e61c6cc8274bb3cc5.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/bd0f4de744997828af2f5be5ec11428f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/bf85faaa5f8efc2e1023f5e9504c0d4e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/c509336eb0f6e35088064f15ddddc844.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/c5dd4097aac455d055badef188473371.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/c66f535c634eadaf34e0fe041672ae4b.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/c693a3ac340ea557c7ee39ddef0451a2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/c8e0baa6e08346d410255ea827a8be27.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ca8a49ee7846219a65d63ee1ec51c194.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/cd52152b8b48869f0fb1175983d9a505.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/cea8a7b71e661b383a7dc2f15cc2f47a.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d2e284352932a735dda55727c0879153.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d2fd9591ebfd0e779de7682d2745dbbb.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d4a14f2651525d4affbbb753b6dbdf07.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d5caf76d2720e294f4d0b1625f01ec9c.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d5d463ae4788148f67a28de9ec4c1dca.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d6e2c1e4c4b88bc14079e9c4a7549497.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d70ea44ff65a30a175801cb104f83448.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d750b314131793e148b4ebf50ffb91af.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d7ae616c789f29be08782214e0288a0d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/d8ae70630c929796cb0364e5202037e4.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/da5bdab3ae018a4c1207bd67ab854bc2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/dab7b758f386eaae246f675ec7e23f0f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/db1cc9bd868a6ff45d7bbe5b28d4b8d7.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/dc34cae5db8221efc393ea494597b1ec.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/dd32cf24e261a9aa261b7e196da33f2d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/dda618cc791e2c20645ca5b7843c50dc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/df9de632322bb3378c647420760aab61.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e00c9c693f510281ead88131d09c18d2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e02d34cae25f04c920a31db2c80d1d8f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e22efb34fb5fe61f5b45504d76d20781.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e25f5e7e1d5b88a59d981b5f6f669283.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e3bb4be954796cb47ac6851ab724ecaf.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e3bdeda10d9131256fad8417fd1da09f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e73e3950ff542ab97969b032c1f7241f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e78879cfa489561d3ec6eef16a4a0a4d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e811136959f0512a25a18acc520970b5.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/e9a7cf7d3d15fb2e50730469b07f2e39.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/eb514af63c1e9b14ccec7dee432b8edc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ed0b4344be5485ee0b0c207f28762fb8.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/edf89add9339773b26c1c2dd2244a179.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ef8de64aca34834c6f110f4adc7515bd.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f0438b1823af0a0de674c239fa03eaba.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f0604027051cb387c2f85940dfff3755.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f065b514c78eb2344c42e496865369a9.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f0d97a6f8e1880316015bc2283d0f156.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f1834c69e3074e71093d17a68b9d4d7d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f29b465128cba69d5e68235945cd3edd.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f330fd41c2fbf0ae89bb98047c08d516.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f383e78159d65c11032dc8b8d9c360f3.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f647c059126fa5bf24355dcbfed09c4f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f6af810265a209ab307886171ded4113.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f74844d2228da8984fe156b5b9c28acc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f7f2840050c367f26cfbc2b81c3a5fad.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f85435b23bb50124e045f50cfe54dd3e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f865cd6ba4d22d8d61fce2a2d63be7dc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/f8a66c1ab7db90820de697f664ea136d.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fa6cf45901b333e63b3eabc045783c69.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fa827d13030f9366a57edbfc22ce4336.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fb6ac0e1252b3d366009d4f4ffeab4fc.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fbb64faabca9f28743b2d0d5dc448e5f.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fd299dbef627ab6738d61d4e4bac56e2.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/fd8cd455b632147c1758bfb9bd0ba42e.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/ff8cf784cc8ef060bf9d2dc7e698b451.woff2 +0 -0
- package/assets/fonts/sarasa-fixed-sc/result.css +20 -0
- package/dist/chunk-2Q3Z3ICU.js +155 -0
- package/dist/chunk-2Q3Z3ICU.js.map +1 -0
- package/dist/chunk-6O6JTF24.js +318 -0
- package/dist/chunk-6O6JTF24.js.map +1 -0
- package/dist/chunk-OO64L35C.js +34 -0
- package/dist/chunk-OO64L35C.js.map +1 -0
- package/dist/chunk-QJ5CQDK7.js +1038 -0
- package/dist/chunk-QJ5CQDK7.js.map +1 -0
- package/dist/chunk-UGFYGF3Y.js +142 -0
- package/dist/chunk-UGFYGF3Y.js.map +1 -0
- package/dist/chunk-ZUWAB67J.js +55 -0
- package/dist/chunk-ZUWAB67J.js.map +1 -0
- package/dist/index.js +279 -0
- package/dist/index.js.map +1 -0
- package/dist/serve.js +4375 -0
- package/dist/serve.js.map +1 -0
- package/dist/session-worker.js +357 -0
- package/dist/session-worker.js.map +1 -0
- package/dist/terminal-TTRA2VJY.js +671 -0
- package/dist/terminal-TTRA2VJY.js.map +1 -0
- package/package.json +67 -0
|
@@ -0,0 +1,1038 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
|
|
3
|
+
// src/common/paths.ts
|
|
4
|
+
import { cpSync, existsSync, mkdirSync, writeFileSync } from "fs";
|
|
5
|
+
import { homedir } from "os";
|
|
6
|
+
import { dirname, resolve } from "path";
|
|
7
|
+
import { fileURLToPath } from "url";
|
|
8
|
+
var HOME = homedir();
|
|
9
|
+
var APP_DIR = `${HOME}/.dev-anywhere`;
|
|
10
|
+
var MODULE_DIR = dirname(fileURLToPath(import.meta.url));
|
|
11
|
+
var SOURCE_FONT_ASSETS_DIR = resolve(MODULE_DIR, "../../assets/fonts");
|
|
12
|
+
var DIST_FONT_ASSETS_DIR = resolve(MODULE_DIR, "../assets/fonts");
|
|
13
|
+
var DEFAULT_FONT_FAMILY = "sarasa-fixed-sc";
|
|
14
|
+
function tildify(cwd) {
|
|
15
|
+
return HOME ? cwd.replace(HOME, "~") : cwd;
|
|
16
|
+
}
|
|
17
|
+
var CONFIG_PATH = `${APP_DIR}/config.json`;
|
|
18
|
+
var RUN_DIR = `${APP_DIR}/run`;
|
|
19
|
+
var SOCK_PATH = `${RUN_DIR}/dev-anywhere.sock`;
|
|
20
|
+
var PID_PATH = `${RUN_DIR}/dev-anywhere.pid`;
|
|
21
|
+
var STOPPED_PATH = `${RUN_DIR}/stopped`;
|
|
22
|
+
var DESIRED_ENV_PATH = `${RUN_DIR}/desired-env`;
|
|
23
|
+
var STATE_DIR = `${APP_DIR}/state`;
|
|
24
|
+
var SESSIONS_PATH = `${STATE_DIR}/sessions.json`;
|
|
25
|
+
var DATA_DIR = `${APP_DIR}/data`;
|
|
26
|
+
var RELAY_DATA_DIR = `${APP_DIR}/relay-data`;
|
|
27
|
+
var FONT_DIR = `${RELAY_DATA_DIR}/fonts`;
|
|
28
|
+
var LOG_DIR = `${APP_DIR}/logs`;
|
|
29
|
+
var SERVICE_LOG_PATH = `${LOG_DIR}/service.log`;
|
|
30
|
+
function sessionDir(sessionId) {
|
|
31
|
+
return `${DATA_DIR}/${sessionId}`;
|
|
32
|
+
}
|
|
33
|
+
function sessionPaths(sessionId) {
|
|
34
|
+
const dir = sessionDir(sessionId);
|
|
35
|
+
return {
|
|
36
|
+
dir,
|
|
37
|
+
workerSock: `${dir}/worker.sock`
|
|
38
|
+
};
|
|
39
|
+
}
|
|
40
|
+
function isInitialized() {
|
|
41
|
+
return existsSync(CONFIG_PATH);
|
|
42
|
+
}
|
|
43
|
+
var DEFAULT_CONFIG = `{
|
|
44
|
+
"defaultEnv": "local",
|
|
45
|
+
"envs": {
|
|
46
|
+
"local": {
|
|
47
|
+
"relayUrl": "ws://localhost:3100"
|
|
48
|
+
},
|
|
49
|
+
"cloud": {
|
|
50
|
+
"relayUrl": "wss://dev-anywhere.vita-tools.top",
|
|
51
|
+
"relayToken": ""
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
`;
|
|
56
|
+
function copyFontFamilyIfMissing(targetFontsDir, source) {
|
|
57
|
+
const family = source.family ?? DEFAULT_FONT_FAMILY;
|
|
58
|
+
const sourceFamilyDir = `${source.dir}/${family}`;
|
|
59
|
+
const targetFamilyDir = `${targetFontsDir}/${family}`;
|
|
60
|
+
if (existsSync(targetFamilyDir) || !existsSync(sourceFamilyDir)) return false;
|
|
61
|
+
mkdirSync(targetFontsDir, { recursive: true });
|
|
62
|
+
cpSync(sourceFamilyDir, targetFamilyDir, { recursive: true });
|
|
63
|
+
return true;
|
|
64
|
+
}
|
|
65
|
+
function installFontAssetsFromSources(targetFontsDir, sources) {
|
|
66
|
+
for (const source of sources) {
|
|
67
|
+
if (copyFontFamilyIfMissing(targetFontsDir, source)) return true;
|
|
68
|
+
}
|
|
69
|
+
return false;
|
|
70
|
+
}
|
|
71
|
+
function installFontAssets() {
|
|
72
|
+
installFontAssetsFromSources(FONT_DIR, [
|
|
73
|
+
{ dir: SOURCE_FONT_ASSETS_DIR, family: DEFAULT_FONT_FAMILY },
|
|
74
|
+
{ dir: DIST_FONT_ASSETS_DIR, family: DEFAULT_FONT_FAMILY }
|
|
75
|
+
]);
|
|
76
|
+
}
|
|
77
|
+
function initWorkspace() {
|
|
78
|
+
mkdirSync(RUN_DIR, { recursive: true });
|
|
79
|
+
mkdirSync(STATE_DIR, { recursive: true });
|
|
80
|
+
mkdirSync(DATA_DIR, { recursive: true });
|
|
81
|
+
mkdirSync(RELAY_DATA_DIR, { recursive: true });
|
|
82
|
+
mkdirSync(LOG_DIR, { recursive: true });
|
|
83
|
+
installFontAssets();
|
|
84
|
+
if (!existsSync(CONFIG_PATH)) {
|
|
85
|
+
writeFileSync(CONFIG_PATH, DEFAULT_CONFIG);
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// ../../packages/shared/dist/schemas/envelope.js
|
|
90
|
+
import { z as z5 } from "zod";
|
|
91
|
+
|
|
92
|
+
// ../../packages/shared/dist/schemas/chat.js
|
|
93
|
+
import { z } from "zod";
|
|
94
|
+
var UserInputPayloadSchema = z.object({
|
|
95
|
+
text: z.string().min(1)
|
|
96
|
+
});
|
|
97
|
+
var AssistantMessagePayloadSchema = z.object({
|
|
98
|
+
text: z.string(),
|
|
99
|
+
isPartial: z.boolean()
|
|
100
|
+
});
|
|
101
|
+
var ThinkingPayloadSchema = z.object({
|
|
102
|
+
text: z.string()
|
|
103
|
+
});
|
|
104
|
+
|
|
105
|
+
// ../../packages/shared/dist/schemas/tool.js
|
|
106
|
+
import { z as z2 } from "zod";
|
|
107
|
+
var ToolUseRequestPayloadSchema = z2.object({
|
|
108
|
+
toolName: z2.string(),
|
|
109
|
+
toolId: z2.string(),
|
|
110
|
+
parameters: z2.record(z2.string(), z2.unknown())
|
|
111
|
+
});
|
|
112
|
+
var ToolApprovePayloadSchema = z2.object({
|
|
113
|
+
toolId: z2.string(),
|
|
114
|
+
whitelistTool: z2.boolean().optional()
|
|
115
|
+
});
|
|
116
|
+
var ToolDenyPayloadSchema = z2.object({
|
|
117
|
+
toolId: z2.string(),
|
|
118
|
+
reason: z2.string().optional()
|
|
119
|
+
});
|
|
120
|
+
var ToolResultPayloadSchema = z2.object({
|
|
121
|
+
toolId: z2.string(),
|
|
122
|
+
result: z2.unknown(),
|
|
123
|
+
isError: z2.boolean()
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
// ../../packages/shared/dist/schemas/session.js
|
|
127
|
+
import { z as z3 } from "zod";
|
|
128
|
+
var sessionStateValues = ["idle", "working", "waiting_approval", "error", "terminated"];
|
|
129
|
+
var providerValues = ["claude", "codex"];
|
|
130
|
+
var ptyOwnerValues = ["local-terminal", "proxy-hosted"];
|
|
131
|
+
var agentStatusPhaseValues = [
|
|
132
|
+
"idle",
|
|
133
|
+
"thinking",
|
|
134
|
+
"tool_use",
|
|
135
|
+
"outputting",
|
|
136
|
+
"waiting_permission",
|
|
137
|
+
"error"
|
|
138
|
+
];
|
|
139
|
+
var SessionInfoSchema = z3.object({
|
|
140
|
+
sessionId: z3.string(),
|
|
141
|
+
name: z3.string().optional(),
|
|
142
|
+
state: z3.enum(sessionStateValues),
|
|
143
|
+
mode: z3.enum(["pty", "json"]).optional(),
|
|
144
|
+
provider: z3.enum(providerValues),
|
|
145
|
+
// PTY 尺寸所有权:
|
|
146
|
+
// - local-terminal: 本地 terminal 进程持有真实 PTY,Web 只按原始 cols/rows 展示
|
|
147
|
+
// - proxy-hosted: serve 内托管 PTY,Web 可按视口请求 resize
|
|
148
|
+
ptyOwner: z3.enum(ptyOwnerValues).optional(),
|
|
149
|
+
lastActive: z3.number().optional()
|
|
150
|
+
});
|
|
151
|
+
var SessionCreatePayloadSchema = z3.object({
|
|
152
|
+
name: z3.string().optional(),
|
|
153
|
+
cwd: z3.string().optional(),
|
|
154
|
+
streamDelta: z3.boolean().optional()
|
|
155
|
+
});
|
|
156
|
+
var SessionListPayloadSchema = z3.object({
|
|
157
|
+
sessions: z3.array(SessionInfoSchema)
|
|
158
|
+
});
|
|
159
|
+
var SessionSwitchPayloadSchema = z3.object({
|
|
160
|
+
sessionId: z3.string()
|
|
161
|
+
});
|
|
162
|
+
var SessionTerminatePayloadSchema = z3.object({
|
|
163
|
+
sessionId: z3.string()
|
|
164
|
+
});
|
|
165
|
+
var SessionStatusPayloadSchema = z3.object({
|
|
166
|
+
sessionId: z3.string(),
|
|
167
|
+
state: z3.enum(sessionStateValues),
|
|
168
|
+
lastActive: z3.number()
|
|
169
|
+
});
|
|
170
|
+
var PtyStatePayloadSchema = z3.object({
|
|
171
|
+
state: z3.enum(["working", "turn_complete", "approval_wait", "mid_pause"]),
|
|
172
|
+
title: z3.string().optional(),
|
|
173
|
+
tool: z3.string().optional()
|
|
174
|
+
});
|
|
175
|
+
var AgentStatusPayloadSchema = z3.object({
|
|
176
|
+
provider: z3.enum(providerValues),
|
|
177
|
+
phase: z3.enum(agentStatusPhaseValues),
|
|
178
|
+
seq: z3.number().int().nonnegative(),
|
|
179
|
+
updatedAt: z3.number(),
|
|
180
|
+
toolName: z3.string().optional(),
|
|
181
|
+
toolInput: z3.record(z3.string(), z3.unknown()).optional(),
|
|
182
|
+
permissionRequest: z3.object({
|
|
183
|
+
requestId: z3.string(),
|
|
184
|
+
toolName: z3.string(),
|
|
185
|
+
input: z3.record(z3.string(), z3.unknown())
|
|
186
|
+
}).optional(),
|
|
187
|
+
permissionResolution: z3.object({
|
|
188
|
+
requestId: z3.string(),
|
|
189
|
+
outcome: z3.enum(["allow", "deny"])
|
|
190
|
+
}).optional(),
|
|
191
|
+
summary: z3.string().optional()
|
|
192
|
+
});
|
|
193
|
+
|
|
194
|
+
// ../../packages/shared/dist/schemas/system.js
|
|
195
|
+
import { z as z4 } from "zod";
|
|
196
|
+
var HeartbeatPayloadSchema = z4.object({});
|
|
197
|
+
var AuthPayloadSchema = z4.object({
|
|
198
|
+
pairingCode: z4.string().optional(),
|
|
199
|
+
token: z4.string().optional()
|
|
200
|
+
});
|
|
201
|
+
var SyncRequestPayloadSchema = z4.object({
|
|
202
|
+
lastSeq: z4.number().int().nonnegative()
|
|
203
|
+
});
|
|
204
|
+
var SyncResponsePayloadSchema = z4.object({
|
|
205
|
+
messages: z4.array(z4.record(z4.string(), z4.unknown()))
|
|
206
|
+
});
|
|
207
|
+
|
|
208
|
+
// ../../packages/shared/dist/schemas/envelope.js
|
|
209
|
+
var BaseEnvelopeFields = {
|
|
210
|
+
seq: z5.number().int().nonnegative(),
|
|
211
|
+
sessionId: z5.string(),
|
|
212
|
+
timestamp: z5.number(),
|
|
213
|
+
source: z5.enum(["proxy", "client"]),
|
|
214
|
+
version: z5.string()
|
|
215
|
+
};
|
|
216
|
+
var MessageEnvelopeSchema = z5.discriminatedUnion("type", [
|
|
217
|
+
// chat (3)
|
|
218
|
+
z5.object({
|
|
219
|
+
...BaseEnvelopeFields,
|
|
220
|
+
type: z5.literal("user_input"),
|
|
221
|
+
payload: UserInputPayloadSchema
|
|
222
|
+
}),
|
|
223
|
+
z5.object({
|
|
224
|
+
...BaseEnvelopeFields,
|
|
225
|
+
type: z5.literal("assistant_message"),
|
|
226
|
+
payload: AssistantMessagePayloadSchema
|
|
227
|
+
}),
|
|
228
|
+
z5.object({
|
|
229
|
+
...BaseEnvelopeFields,
|
|
230
|
+
type: z5.literal("thinking"),
|
|
231
|
+
payload: ThinkingPayloadSchema
|
|
232
|
+
}),
|
|
233
|
+
// tool (4): 工具审批决策属于 relay control,不进入会话消息信封。
|
|
234
|
+
// tool_use_request: 审批流请求(proxy → client),toolId 是 approval requestId
|
|
235
|
+
z5.object({
|
|
236
|
+
...BaseEnvelopeFields,
|
|
237
|
+
type: z5.literal("tool_use_request"),
|
|
238
|
+
payload: ToolUseRequestPayloadSchema
|
|
239
|
+
}),
|
|
240
|
+
// tool_result: 工具执行结果(proxy → client),toolId 对应 assistant_tool_use / tool_use_request 的 toolId
|
|
241
|
+
z5.object({
|
|
242
|
+
...BaseEnvelopeFields,
|
|
243
|
+
type: z5.literal("tool_result"),
|
|
244
|
+
payload: ToolResultPayloadSchema
|
|
245
|
+
}),
|
|
246
|
+
// assistant_tool_use: 纯展示型工具调用(proxy → client),区别于 tool_use_request 无审批语义
|
|
247
|
+
// payload 结构复用 ToolUseRequestPayloadSchema;toolId 是 Claude 分配的 tool_use id
|
|
248
|
+
z5.object({
|
|
249
|
+
...BaseEnvelopeFields,
|
|
250
|
+
type: z5.literal("assistant_tool_use"),
|
|
251
|
+
payload: ToolUseRequestPayloadSchema
|
|
252
|
+
}),
|
|
253
|
+
// session (5)
|
|
254
|
+
z5.object({
|
|
255
|
+
...BaseEnvelopeFields,
|
|
256
|
+
type: z5.literal("session_create"),
|
|
257
|
+
payload: SessionCreatePayloadSchema
|
|
258
|
+
}),
|
|
259
|
+
z5.object({
|
|
260
|
+
...BaseEnvelopeFields,
|
|
261
|
+
type: z5.literal("session_list"),
|
|
262
|
+
payload: SessionListPayloadSchema
|
|
263
|
+
}),
|
|
264
|
+
z5.object({
|
|
265
|
+
...BaseEnvelopeFields,
|
|
266
|
+
type: z5.literal("session_switch"),
|
|
267
|
+
payload: SessionSwitchPayloadSchema
|
|
268
|
+
}),
|
|
269
|
+
z5.object({
|
|
270
|
+
...BaseEnvelopeFields,
|
|
271
|
+
type: z5.literal("session_terminate"),
|
|
272
|
+
payload: SessionTerminatePayloadSchema
|
|
273
|
+
}),
|
|
274
|
+
z5.object({
|
|
275
|
+
...BaseEnvelopeFields,
|
|
276
|
+
type: z5.literal("session_status"),
|
|
277
|
+
payload: SessionStatusPayloadSchema
|
|
278
|
+
}),
|
|
279
|
+
// system (5)
|
|
280
|
+
z5.object({
|
|
281
|
+
...BaseEnvelopeFields,
|
|
282
|
+
type: z5.literal("heartbeat"),
|
|
283
|
+
payload: HeartbeatPayloadSchema
|
|
284
|
+
}),
|
|
285
|
+
z5.object({
|
|
286
|
+
...BaseEnvelopeFields,
|
|
287
|
+
type: z5.literal("auth"),
|
|
288
|
+
payload: AuthPayloadSchema
|
|
289
|
+
}),
|
|
290
|
+
z5.object({
|
|
291
|
+
...BaseEnvelopeFields,
|
|
292
|
+
type: z5.literal("sync_request"),
|
|
293
|
+
payload: SyncRequestPayloadSchema
|
|
294
|
+
}),
|
|
295
|
+
z5.object({
|
|
296
|
+
...BaseEnvelopeFields,
|
|
297
|
+
type: z5.literal("sync_response"),
|
|
298
|
+
payload: SyncResponsePayloadSchema
|
|
299
|
+
})
|
|
300
|
+
]);
|
|
301
|
+
|
|
302
|
+
// ../../packages/shared/dist/builders/index.js
|
|
303
|
+
function buildMessage(type, sessionId, seq, payload, source) {
|
|
304
|
+
const envelope = {
|
|
305
|
+
seq,
|
|
306
|
+
sessionId,
|
|
307
|
+
type,
|
|
308
|
+
payload,
|
|
309
|
+
timestamp: Date.now(),
|
|
310
|
+
source,
|
|
311
|
+
version: "1.0"
|
|
312
|
+
};
|
|
313
|
+
return MessageEnvelopeSchema.parse(envelope);
|
|
314
|
+
}
|
|
315
|
+
|
|
316
|
+
// ../../packages/shared/dist/schemas/relay-control.js
|
|
317
|
+
import { z as z6 } from "zod";
|
|
318
|
+
|
|
319
|
+
// ../../packages/shared/dist/constants/relay-errors.js
|
|
320
|
+
var RelayErrorCode = {
|
|
321
|
+
NOT_REGISTERED: "NOT_REGISTERED",
|
|
322
|
+
NOT_BOUND: "NOT_BOUND",
|
|
323
|
+
PROXY_OFFLINE: "PROXY_OFFLINE",
|
|
324
|
+
INVALID_MESSAGE: "INVALID_MESSAGE",
|
|
325
|
+
UNSUPPORTED: "UNSUPPORTED",
|
|
326
|
+
INVALID_RANGE: "INVALID_RANGE"
|
|
327
|
+
};
|
|
328
|
+
|
|
329
|
+
// ../../packages/shared/dist/constants/control-errors.js
|
|
330
|
+
var ControlErrorCode = {
|
|
331
|
+
INVALID_PATH: "INVALID_PATH",
|
|
332
|
+
PATH_NOT_FOUND: "PATH_NOT_FOUND",
|
|
333
|
+
PATH_NOT_DIRECTORY: "PATH_NOT_DIRECTORY",
|
|
334
|
+
PATH_ACCESS_DENIED: "PATH_ACCESS_DENIED",
|
|
335
|
+
PROXY_OFFLINE: "PROXY_OFFLINE",
|
|
336
|
+
SESSION_NOT_FOUND: "SESSION_NOT_FOUND",
|
|
337
|
+
PROVIDER_UNSUPPORTED: "PROVIDER_UNSUPPORTED",
|
|
338
|
+
WORKER_START_FAILED: "WORKER_START_FAILED",
|
|
339
|
+
PROCESS_START_FAILED: "PROCESS_START_FAILED",
|
|
340
|
+
UNKNOWN: "UNKNOWN"
|
|
341
|
+
};
|
|
342
|
+
|
|
343
|
+
// ../../packages/shared/dist/schemas/relay-control.js
|
|
344
|
+
var ProxyInfoSchema = z6.object({
|
|
345
|
+
proxyId: z6.string(),
|
|
346
|
+
name: z6.string().optional(),
|
|
347
|
+
online: z6.boolean(),
|
|
348
|
+
sessions: z6.array(z6.string()).optional()
|
|
349
|
+
});
|
|
350
|
+
var AgentCliAvailabilitySchema = z6.object({
|
|
351
|
+
available: z6.boolean(),
|
|
352
|
+
command: z6.string().optional(),
|
|
353
|
+
error: z6.string().optional(),
|
|
354
|
+
suggestions: z6.array(z6.string()).optional()
|
|
355
|
+
});
|
|
356
|
+
var AgentCliStatusSchema = z6.object({
|
|
357
|
+
claude: AgentCliAvailabilitySchema,
|
|
358
|
+
codex: AgentCliAvailabilitySchema
|
|
359
|
+
});
|
|
360
|
+
var DirEntrySchema = z6.object({ name: z6.string(), isDir: z6.boolean() });
|
|
361
|
+
var FileTreeGroupSchema = z6.object({
|
|
362
|
+
path: z6.string(),
|
|
363
|
+
entries: z6.array(DirEntrySchema)
|
|
364
|
+
});
|
|
365
|
+
var CommandEntrySchema = z6.object({
|
|
366
|
+
name: z6.string(),
|
|
367
|
+
description: z6.string(),
|
|
368
|
+
argumentHint: z6.string().optional(),
|
|
369
|
+
source: z6.string()
|
|
370
|
+
});
|
|
371
|
+
var HistorySessionSchema = z6.object({
|
|
372
|
+
id: z6.string(),
|
|
373
|
+
title: z6.string(),
|
|
374
|
+
projectDir: z6.string(),
|
|
375
|
+
updatedAt: z6.number(),
|
|
376
|
+
provider: z6.enum(["claude", "codex"]).optional()
|
|
377
|
+
});
|
|
378
|
+
var RequestIdShape = { requestId: z6.string().min(1).optional() };
|
|
379
|
+
var ControlErrorCodeSchema = z6.enum(Object.values(ControlErrorCode));
|
|
380
|
+
var RequestErrorShape = {
|
|
381
|
+
error: z6.string().optional(),
|
|
382
|
+
errorCode: ControlErrorCodeSchema.optional()
|
|
383
|
+
};
|
|
384
|
+
function control(type, shape, directions) {
|
|
385
|
+
return {
|
|
386
|
+
type,
|
|
387
|
+
directions: new Set(Array.isArray(directions) ? directions : directions ? [directions] : []),
|
|
388
|
+
schema: z6.object({
|
|
389
|
+
type: z6.literal(type),
|
|
390
|
+
...shape ?? {}
|
|
391
|
+
})
|
|
392
|
+
};
|
|
393
|
+
}
|
|
394
|
+
var relayControlDefinitions = [
|
|
395
|
+
control("proxy_register", {
|
|
396
|
+
proxyId: z6.string().min(1),
|
|
397
|
+
name: z6.string().optional()
|
|
398
|
+
}),
|
|
399
|
+
control("proxy_register_response", {
|
|
400
|
+
status: z6.enum(["new", "reconnected"])
|
|
401
|
+
}),
|
|
402
|
+
control("proxy_list_request", RequestIdShape),
|
|
403
|
+
control("proxy_list_response", {
|
|
404
|
+
...RequestIdShape,
|
|
405
|
+
proxies: z6.array(ProxyInfoSchema)
|
|
406
|
+
}),
|
|
407
|
+
control("proxy_select", { ...RequestIdShape, proxyId: z6.string().min(1) }),
|
|
408
|
+
control("proxy_select_response", {
|
|
409
|
+
...RequestIdShape,
|
|
410
|
+
success: z6.boolean(),
|
|
411
|
+
proxyId: z6.string().optional(),
|
|
412
|
+
...RequestErrorShape
|
|
413
|
+
}),
|
|
414
|
+
control("relay_error", {
|
|
415
|
+
code: z6.enum(Object.values(RelayErrorCode)),
|
|
416
|
+
message: z6.string()
|
|
417
|
+
}),
|
|
418
|
+
// 客户端注册协议
|
|
419
|
+
control("client_register", {
|
|
420
|
+
clientId: z6.string().min(1)
|
|
421
|
+
}),
|
|
422
|
+
control("client_register_response", {
|
|
423
|
+
status: z6.enum(["restored", "proxy_offline", "new"]),
|
|
424
|
+
proxyId: z6.string().optional()
|
|
425
|
+
}),
|
|
426
|
+
// Proxy 离线通知
|
|
427
|
+
control("proxy_offline", {
|
|
428
|
+
proxyId: z6.string()
|
|
429
|
+
}),
|
|
430
|
+
// Proxy 主动断开,relay 立即清理资源
|
|
431
|
+
control("proxy_disconnect", {
|
|
432
|
+
proxyId: z6.string().min(1)
|
|
433
|
+
}),
|
|
434
|
+
// Proxy 重连后通知 client 恢复
|
|
435
|
+
control("proxy_online", {
|
|
436
|
+
proxyId: z6.string().min(1)
|
|
437
|
+
}),
|
|
438
|
+
// 目录列表请求与响应
|
|
439
|
+
control("dir_list_request", {
|
|
440
|
+
proxyId: z6.string().min(1).optional(),
|
|
441
|
+
...RequestIdShape,
|
|
442
|
+
path: z6.string()
|
|
443
|
+
}, "client_to_proxy"),
|
|
444
|
+
control("dir_list_response", { ...RequestIdShape, ...RequestErrorShape, entries: z6.array(DirEntrySchema), path: z6.string() }, "proxy_to_client"),
|
|
445
|
+
// 目录创建请求与响应
|
|
446
|
+
control("dir_create_request", { ...RequestIdShape, path: z6.string() }, "client_to_proxy"),
|
|
447
|
+
control("dir_create_response", {
|
|
448
|
+
...RequestIdShape,
|
|
449
|
+
...RequestErrorShape,
|
|
450
|
+
path: z6.string(),
|
|
451
|
+
success: z6.boolean()
|
|
452
|
+
}, "proxy_to_client"),
|
|
453
|
+
// 命令列表推送,proxy 将可用命令列表推给 client
|
|
454
|
+
control("command_list_push", { commands: z6.array(CommandEntrySchema) }, "proxy_to_client"),
|
|
455
|
+
// 文件树推送: 按目录分组, 首组 path 即为 session cwd
|
|
456
|
+
// 前端直接把每组写入 tree[path], 与 dir_list_response 共享 cache slot
|
|
457
|
+
control("file_tree_push", {
|
|
458
|
+
groups: z6.array(FileTreeGroupSchema)
|
|
459
|
+
}, "proxy_to_client"),
|
|
460
|
+
// 会话列表请求与权限模式变更
|
|
461
|
+
control("session_list", void 0, ["client_to_proxy", "proxy_to_client"]),
|
|
462
|
+
control("permission_mode_change", {
|
|
463
|
+
mode: z6.enum(["default", "auto_accept", "plan"]),
|
|
464
|
+
// sessionId 可选:传入时 proxy 按该会话的 mode 分叉(PTY 发 Tab ANSI),未传走全局日志行为
|
|
465
|
+
sessionId: z6.string().optional()
|
|
466
|
+
}, "client_to_proxy"),
|
|
467
|
+
// 会话历史浏览
|
|
468
|
+
control("session_history_request", RequestIdShape, "client_to_proxy"),
|
|
469
|
+
control("session_history_response", { ...RequestIdShape, sessions: z6.array(HistorySessionSchema) }, "proxy_to_client"),
|
|
470
|
+
// PTY 语义状态,从 Envelope 迁移到 Control 层
|
|
471
|
+
control("pty_state", { sessionId: z6.string(), payload: PtyStatePayloadSchema }, "proxy_to_client"),
|
|
472
|
+
// Provider 语义状态,来自 Claude/Codex hook 等结构化事件,不从 PTY 字节推断
|
|
473
|
+
control("agent_status", { sessionId: z6.string(), payload: AgentStatusPayloadSchema }, "proxy_to_client"),
|
|
474
|
+
// 终端标题变化,proxy -> client
|
|
475
|
+
control("terminal_title", { sessionId: z6.string(), title: z6.string() }, "proxy_to_client"),
|
|
476
|
+
// 终端尺寸变化,proxy -> client
|
|
477
|
+
control("terminal_resize", { sessionId: z6.string(), cols: z6.number().int().positive(), rows: z6.number().int().positive() }, "proxy_to_client"),
|
|
478
|
+
control("terminal_resize_request", { sessionId: z6.string(), cols: z6.number().int().positive(), rows: z6.number().int().positive() }, "client_to_proxy"),
|
|
479
|
+
// 远程终止 JSON 会话,client -> proxy
|
|
480
|
+
control("session_terminate", { sessionId: z6.string() }, "client_to_proxy"),
|
|
481
|
+
// 中断当前 turn,client -> proxy,SIGINT 到 worker 进程让 claude CLI abort 当前流
|
|
482
|
+
control("session_worker_abort", { sessionId: z6.string() }, "client_to_proxy"),
|
|
483
|
+
// turn 完成信号,proxy -> client,对应 claude stream-json 的 result 事件
|
|
484
|
+
control("turn_result", {
|
|
485
|
+
sessionId: z6.string(),
|
|
486
|
+
success: z6.boolean(),
|
|
487
|
+
isError: z6.boolean(),
|
|
488
|
+
// stream-json result.result 是本轮最终文本。assistant_message 流丢失或 CLI 未发增量时,
|
|
489
|
+
// Web 用它作为 JSON 模式兜底展示,避免 turn 已结束但界面空白。
|
|
490
|
+
result: z6.string().optional()
|
|
491
|
+
}, "proxy_to_client"),
|
|
492
|
+
// 客户端发送到 PTY 的原始字节(ANSI 序列),不追加换行
|
|
493
|
+
control("remote_input_raw", { sessionId: z6.string().min(1), data: z6.string() }, "client_to_proxy"),
|
|
494
|
+
// 客户端询问 proxy 的环境信息 (home 路径等), client -> proxy -> response
|
|
495
|
+
// FilePathPicker 用 homePath 作为 select 模式下的默认起点, 新建会话时打开即可浏览
|
|
496
|
+
control("proxy_info_request", RequestIdShape, "client_to_proxy"),
|
|
497
|
+
control("proxy_info", { ...RequestIdShape, homePath: z6.string(), agentCli: AgentCliStatusSchema }, "proxy_to_client"),
|
|
498
|
+
control("agent_cli_config_update", { ...RequestIdShape, provider: z6.enum(["claude", "codex"]), path: z6.string().min(1) }, "client_to_proxy"),
|
|
499
|
+
control("agent_cli_config_update_response", {
|
|
500
|
+
...RequestIdShape,
|
|
501
|
+
provider: z6.enum(["claude", "codex"]),
|
|
502
|
+
agentCli: AgentCliStatusSchema.optional(),
|
|
503
|
+
...RequestErrorShape
|
|
504
|
+
}, "proxy_to_client"),
|
|
505
|
+
// 远程创建 JSON 会话,client -> proxy -> response
|
|
506
|
+
control("session_create", {
|
|
507
|
+
...RequestIdShape,
|
|
508
|
+
cwd: z6.string(),
|
|
509
|
+
provider: z6.enum(["claude", "codex"]),
|
|
510
|
+
mode: z6.enum(["json", "pty"]).optional(),
|
|
511
|
+
resumeSessionId: z6.string().optional(),
|
|
512
|
+
// 透传给 claude CLI 的 --permission-mode, undefined 时 proxy 兜底为 "default"
|
|
513
|
+
permissionMode: z6.enum(["default", "auto", "acceptEdits", "plan", "bypassPermissions", "dontAsk"]).optional()
|
|
514
|
+
}, "client_to_proxy"),
|
|
515
|
+
control("session_create_response", {
|
|
516
|
+
...RequestIdShape,
|
|
517
|
+
sessionId: z6.string(),
|
|
518
|
+
mode: z6.enum(["json", "pty"]).optional(),
|
|
519
|
+
provider: z6.enum(["claude", "codex"]).optional(),
|
|
520
|
+
ptyOwner: z6.enum(["local-terminal", "proxy-hosted"]).optional(),
|
|
521
|
+
...RequestErrorShape
|
|
522
|
+
}, "proxy_to_client"),
|
|
523
|
+
// 客户端请求会话历史消息,client -> proxy
|
|
524
|
+
control("session_messages_request", { ...RequestIdShape, sessionId: z6.string() }, "client_to_proxy"),
|
|
525
|
+
// 客户端请求会话资源(命令列表 + 文件树),client -> proxy
|
|
526
|
+
control("session_resources_request", { ...RequestIdShape, sessionId: z6.string() }, "client_to_proxy"),
|
|
527
|
+
control("session_resources_response", {
|
|
528
|
+
...RequestIdShape,
|
|
529
|
+
...RequestErrorShape,
|
|
530
|
+
sessionId: z6.string(),
|
|
531
|
+
commands: z6.array(CommandEntrySchema),
|
|
532
|
+
groups: z6.array(FileTreeGroupSchema)
|
|
533
|
+
}, "proxy_to_client"),
|
|
534
|
+
// 客户端请求当前 provider 语义状态;不经 relay 缓存,由 proxy 返回当前值
|
|
535
|
+
control("agent_status_request", { ...RequestIdShape, sessionId: z6.string().optional() }, "client_to_proxy"),
|
|
536
|
+
control("agent_status_response", {
|
|
537
|
+
...RequestIdShape,
|
|
538
|
+
statuses: z6.array(z6.object({ sessionId: z6.string(), payload: AgentStatusPayloadSchema }))
|
|
539
|
+
}, "proxy_to_client"),
|
|
540
|
+
// 客户端确认已收到审批请求;proxy 只记录送达状态,不把它当成用户决策
|
|
541
|
+
control("permission_request_delivered", { sessionId: z6.string(), requestId: z6.string() }, "client_to_proxy"),
|
|
542
|
+
control("tool_approve", { sessionId: z6.string(), payload: ToolApprovePayloadSchema }, "client_to_proxy"),
|
|
543
|
+
control("tool_deny", { sessionId: z6.string(), payload: ToolDenyPayloadSchema }, "client_to_proxy"),
|
|
544
|
+
// proxy 确认用户决策已进入 provider/worker 路径;web 用它更新审批卡片状态
|
|
545
|
+
control("permission_decision_result", {
|
|
546
|
+
sessionId: z6.string(),
|
|
547
|
+
requestId: z6.string(),
|
|
548
|
+
outcome: z6.enum(["allow", "deny"]),
|
|
549
|
+
delivered: z6.boolean(),
|
|
550
|
+
message: z6.string().optional()
|
|
551
|
+
}, "proxy_to_client"),
|
|
552
|
+
// proxy 推送当前 pending 的工具审批列表,client 据此恢复审批卡片
|
|
553
|
+
control("pending_approvals_push", {
|
|
554
|
+
sessionId: z6.string(),
|
|
555
|
+
approvals: z6.array(z6.object({
|
|
556
|
+
requestId: z6.string(),
|
|
557
|
+
toolName: z6.string(),
|
|
558
|
+
input: z6.record(z6.string(), z6.unknown())
|
|
559
|
+
}))
|
|
560
|
+
}, "proxy_to_client"),
|
|
561
|
+
// 恢复会话时推送历史消息,proxy -> client
|
|
562
|
+
control("session_history_messages", {
|
|
563
|
+
...RequestIdShape,
|
|
564
|
+
sessionId: z6.string(),
|
|
565
|
+
messages: z6.array(z6.object({
|
|
566
|
+
role: z6.enum(["user", "assistant"]),
|
|
567
|
+
text: z6.string(),
|
|
568
|
+
timestamp: z6.number().optional()
|
|
569
|
+
}))
|
|
570
|
+
}, "proxy_to_client"),
|
|
571
|
+
// proxy 重连后同步活跃 session 列表给 relay
|
|
572
|
+
control("session_sync", {
|
|
573
|
+
sessions: z6.array(z6.object({
|
|
574
|
+
id: z6.string(),
|
|
575
|
+
mode: z6.enum(["pty", "json"]),
|
|
576
|
+
provider: z6.enum(["claude", "codex"]),
|
|
577
|
+
ptyOwner: z6.enum(["local-terminal", "proxy-hosted"]).optional(),
|
|
578
|
+
state: z6.string()
|
|
579
|
+
}))
|
|
580
|
+
}),
|
|
581
|
+
// PTY 会话订阅,client -> proxy,触发 terminal serialize() 返回当前状态
|
|
582
|
+
control("session_subscribe", { sessionId: z6.string(), requestId: z6.string().optional() }, "client_to_proxy"),
|
|
583
|
+
// PTY 会话快照,proxy -> client,serialize() 的全量终端状态
|
|
584
|
+
control("session_snapshot", {
|
|
585
|
+
sessionId: z6.string(),
|
|
586
|
+
cols: z6.number().int().positive(),
|
|
587
|
+
rows: z6.number().int().positive(),
|
|
588
|
+
data: z6.string(),
|
|
589
|
+
outputSeq: z6.number().int().nonnegative(),
|
|
590
|
+
requestId: z6.string().optional()
|
|
591
|
+
}, "proxy_to_client")
|
|
592
|
+
];
|
|
593
|
+
var relayControlSchemas = relayControlDefinitions.map((definition) => definition.schema);
|
|
594
|
+
var RelayControlSchema = z6.discriminatedUnion("type", relayControlSchemas);
|
|
595
|
+
var ProxyToClientRelayControlTypes = new Set(relayControlDefinitions.filter((definition) => definition.directions.has("proxy_to_client")).map((definition) => definition.type));
|
|
596
|
+
var ClientToProxyRelayControlTypes = new Set(relayControlDefinitions.filter((definition) => definition.directions.has("client_to_proxy")).map((definition) => definition.type));
|
|
597
|
+
|
|
598
|
+
// ../../packages/shared/dist/constants/session.js
|
|
599
|
+
var SessionState = {
|
|
600
|
+
IDLE: "idle",
|
|
601
|
+
WORKING: "working",
|
|
602
|
+
WAITING_APPROVAL: "waiting_approval",
|
|
603
|
+
ERROR: "error",
|
|
604
|
+
TERMINATED: "terminated"
|
|
605
|
+
};
|
|
606
|
+
|
|
607
|
+
// ../../packages/shared/dist/logger.js
|
|
608
|
+
import { lstatSync, mkdirSync as mkdirSync2, readdirSync, renameSync, statSync, symlinkSync, unlinkSync } from "fs";
|
|
609
|
+
import { homedir as homedir2 } from "os";
|
|
610
|
+
import { basename, join } from "path";
|
|
611
|
+
import pino from "pino";
|
|
612
|
+
var DEFAULT_LOG_DIR = `${homedir2()}/.dev-anywhere/logs`;
|
|
613
|
+
var DEFAULT_LOG_RETENTION = 50;
|
|
614
|
+
var PROCESS_LOG_RUN_ID = sanitizeRunId(process.env.DEV_ANYWHERE_LOG_RUN_ID ?? `${(/* @__PURE__ */ new Date()).toISOString().replace(/[:.]/g, "-")}-${process.pid}`);
|
|
615
|
+
function sanitizeRunId(runId) {
|
|
616
|
+
return runId.replace(/[^a-zA-Z0-9._-]/g, "_");
|
|
617
|
+
}
|
|
618
|
+
function linkLatestLog(logDir, name, filePath, runId) {
|
|
619
|
+
const latestPath = join(logDir, `${name}.log`);
|
|
620
|
+
try {
|
|
621
|
+
const stat = lstatSync(latestPath);
|
|
622
|
+
if (stat.isSymbolicLink()) {
|
|
623
|
+
unlinkSync(latestPath);
|
|
624
|
+
} else {
|
|
625
|
+
renameSync(latestPath, join(logDir, `${name}-legacy-${runId}.log`));
|
|
626
|
+
}
|
|
627
|
+
} catch (err) {
|
|
628
|
+
const code = err.code;
|
|
629
|
+
if (code !== "ENOENT")
|
|
630
|
+
return;
|
|
631
|
+
}
|
|
632
|
+
try {
|
|
633
|
+
symlinkSync(basename(filePath), latestPath);
|
|
634
|
+
} catch {
|
|
635
|
+
}
|
|
636
|
+
}
|
|
637
|
+
function resolveRetention() {
|
|
638
|
+
const raw = process.env.DEV_ANYWHERE_LOG_RETENTION;
|
|
639
|
+
if (!raw)
|
|
640
|
+
return DEFAULT_LOG_RETENTION;
|
|
641
|
+
const parsed = Number.parseInt(raw, 10);
|
|
642
|
+
return Number.isFinite(parsed) && parsed >= 0 ? parsed : DEFAULT_LOG_RETENTION;
|
|
643
|
+
}
|
|
644
|
+
function pruneOldLogs(logDir, name, currentFilePath) {
|
|
645
|
+
const keep = resolveRetention();
|
|
646
|
+
if (keep === 0)
|
|
647
|
+
return;
|
|
648
|
+
const currentFileName = basename(currentFilePath);
|
|
649
|
+
const prefix = `${name}-`;
|
|
650
|
+
const candidates = readdirSync(logDir).filter((entry) => entry.startsWith(prefix) && entry.endsWith(".log") && entry !== currentFileName).map((entry) => {
|
|
651
|
+
const path = join(logDir, entry);
|
|
652
|
+
try {
|
|
653
|
+
return { path, mtimeMs: statSync(path).mtimeMs };
|
|
654
|
+
} catch {
|
|
655
|
+
return null;
|
|
656
|
+
}
|
|
657
|
+
}).filter((entry) => entry !== null).sort((a, b) => b.mtimeMs - a.mtimeMs);
|
|
658
|
+
for (const stale of candidates.slice(Math.max(0, keep - 1))) {
|
|
659
|
+
try {
|
|
660
|
+
unlinkSync(stale.path);
|
|
661
|
+
} catch {
|
|
662
|
+
}
|
|
663
|
+
}
|
|
664
|
+
}
|
|
665
|
+
function createLogger(options) {
|
|
666
|
+
const { name, level = "info", logDir = DEFAULT_LOG_DIR, stdout = false, silent = false } = options;
|
|
667
|
+
if (silent) {
|
|
668
|
+
return pino({ level: "silent" });
|
|
669
|
+
}
|
|
670
|
+
mkdirSync2(logDir, { recursive: true });
|
|
671
|
+
const runId = PROCESS_LOG_RUN_ID;
|
|
672
|
+
const filePath = join(logDir, `${name}-${runId}.log`);
|
|
673
|
+
linkLatestLog(logDir, name, filePath, runId);
|
|
674
|
+
pruneOldLogs(logDir, name, filePath);
|
|
675
|
+
const streams = [{ stream: pino.destination(filePath) }];
|
|
676
|
+
if (stdout) {
|
|
677
|
+
streams.unshift({ stream: process.stdout });
|
|
678
|
+
}
|
|
679
|
+
return pino({ level }, pino.multistream(streams));
|
|
680
|
+
}
|
|
681
|
+
|
|
682
|
+
// src/ipc/ipc-protocol.ts
|
|
683
|
+
import { z as z7 } from "zod";
|
|
684
|
+
|
|
685
|
+
// src/ipc/line-buffer.ts
|
|
686
|
+
import { Transform } from "stream";
|
|
687
|
+
var LineBuffer = class extends Transform {
|
|
688
|
+
buffer = "";
|
|
689
|
+
_transform(chunk, _encoding, callback) {
|
|
690
|
+
this.buffer += typeof chunk === "string" ? chunk : chunk.toString();
|
|
691
|
+
const segments = this.buffer.split("\n");
|
|
692
|
+
this.buffer = segments.pop();
|
|
693
|
+
for (const segment of segments) {
|
|
694
|
+
if (segment.length > 0) {
|
|
695
|
+
this.push(segment);
|
|
696
|
+
}
|
|
697
|
+
}
|
|
698
|
+
callback();
|
|
699
|
+
}
|
|
700
|
+
_flush(callback) {
|
|
701
|
+
if (this.buffer.length > 0) {
|
|
702
|
+
this.push(this.buffer);
|
|
703
|
+
this.buffer = "";
|
|
704
|
+
}
|
|
705
|
+
callback();
|
|
706
|
+
}
|
|
707
|
+
};
|
|
708
|
+
|
|
709
|
+
// src/ipc/ipc-protocol.ts
|
|
710
|
+
var IPC_BINARY_MARKER = 0;
|
|
711
|
+
function encodeBinaryIpcFrame(sessionId, data, outputSeq) {
|
|
712
|
+
const sessionIdBuf = Buffer.from(sessionId, "utf-8");
|
|
713
|
+
const payloadLen = 1 + sessionIdBuf.length + 4 + data.length;
|
|
714
|
+
const frame = Buffer.alloc(1 + 4 + payloadLen);
|
|
715
|
+
let offset = 0;
|
|
716
|
+
frame[offset] = IPC_BINARY_MARKER;
|
|
717
|
+
offset += 1;
|
|
718
|
+
frame.writeUInt32LE(payloadLen, offset);
|
|
719
|
+
offset += 4;
|
|
720
|
+
frame[offset] = sessionIdBuf.length;
|
|
721
|
+
offset += 1;
|
|
722
|
+
sessionIdBuf.copy(frame, offset);
|
|
723
|
+
offset += sessionIdBuf.length;
|
|
724
|
+
frame.writeUInt32LE(outputSeq, offset);
|
|
725
|
+
offset += 4;
|
|
726
|
+
data.copy(frame, offset);
|
|
727
|
+
return frame;
|
|
728
|
+
}
|
|
729
|
+
var sessionStateValues2 = Object.values(SessionState);
|
|
730
|
+
var ProviderHookContextSchema = z7.object({
|
|
731
|
+
provider: z7.enum(["claude", "codex"]),
|
|
732
|
+
sessionId: z7.string(),
|
|
733
|
+
hookUrl: z7.string(),
|
|
734
|
+
marker: z7.string(),
|
|
735
|
+
token: z7.string()
|
|
736
|
+
});
|
|
737
|
+
var IpcMessageSchema = z7.discriminatedUnion("type", [
|
|
738
|
+
// 客户端请求创建新会话,sessionId 可选用于重连时复用
|
|
739
|
+
z7.object({
|
|
740
|
+
type: z7.literal("session_create_request"),
|
|
741
|
+
name: z7.string().optional(),
|
|
742
|
+
mode: z7.enum(["pty", "json"]),
|
|
743
|
+
provider: z7.enum(["claude", "codex"]),
|
|
744
|
+
cwd: z7.string(),
|
|
745
|
+
pid: z7.number(),
|
|
746
|
+
sessionId: z7.string().optional()
|
|
747
|
+
}),
|
|
748
|
+
// 服务端响应创建会话
|
|
749
|
+
z7.object({
|
|
750
|
+
type: z7.literal("session_create_response"),
|
|
751
|
+
sessionId: z7.string(),
|
|
752
|
+
error: z7.string().optional(),
|
|
753
|
+
hook: ProviderHookContextSchema.optional()
|
|
754
|
+
}),
|
|
755
|
+
// 客户端请求终止会话
|
|
756
|
+
z7.object({
|
|
757
|
+
type: z7.literal("session_terminate_request"),
|
|
758
|
+
sessionId: z7.string()
|
|
759
|
+
}),
|
|
760
|
+
// 服务端响应终止会话
|
|
761
|
+
z7.object({
|
|
762
|
+
type: z7.literal("session_terminate_response"),
|
|
763
|
+
sessionId: z7.string(),
|
|
764
|
+
success: z7.boolean()
|
|
765
|
+
}),
|
|
766
|
+
// 客户端向服务端注册 PTY 会话
|
|
767
|
+
z7.object({
|
|
768
|
+
type: z7.literal("pty_register"),
|
|
769
|
+
sessionId: z7.string(),
|
|
770
|
+
pid: z7.number()
|
|
771
|
+
}),
|
|
772
|
+
// 客户端取消注册 PTY 会话
|
|
773
|
+
z7.object({
|
|
774
|
+
type: z7.literal("pty_deregister"),
|
|
775
|
+
sessionId: z7.string()
|
|
776
|
+
}),
|
|
777
|
+
// 输入,从服务端转发到客户端的 PTY stdin(手机远程输入注入)
|
|
778
|
+
z7.object({
|
|
779
|
+
type: z7.literal("pty_input"),
|
|
780
|
+
sessionId: z7.string(),
|
|
781
|
+
data: z7.string()
|
|
782
|
+
}),
|
|
783
|
+
// serve → terminal:Web 端移除本地终端会话时,只断开远程视图,不杀本地 CLI。
|
|
784
|
+
z7.object({
|
|
785
|
+
type: z7.literal("pty_detach"),
|
|
786
|
+
sessionId: z7.string()
|
|
787
|
+
}),
|
|
788
|
+
// 服务端广播会话状态变更
|
|
789
|
+
z7.object({
|
|
790
|
+
type: z7.literal("session_status_update"),
|
|
791
|
+
sessionId: z7.string(),
|
|
792
|
+
state: z7.enum(sessionStateValues2)
|
|
793
|
+
}),
|
|
794
|
+
// 错误响应
|
|
795
|
+
z7.object({
|
|
796
|
+
type: z7.literal("error"),
|
|
797
|
+
message: z7.string(),
|
|
798
|
+
code: z7.string().optional()
|
|
799
|
+
}),
|
|
800
|
+
// 客户端请求服务状态(含 relay 连接信息和 worker 状态)
|
|
801
|
+
z7.object({
|
|
802
|
+
type: z7.literal("service_status_request")
|
|
803
|
+
}),
|
|
804
|
+
// 服务端响应增强版服务状态
|
|
805
|
+
z7.object({
|
|
806
|
+
type: z7.literal("service_status_response"),
|
|
807
|
+
config: z7.object({
|
|
808
|
+
envName: z7.string().optional(),
|
|
809
|
+
envNameSource: z7.enum(["cli", "file", "single", "default", "none"]),
|
|
810
|
+
relayUrl: z7.string().optional(),
|
|
811
|
+
relayUrlSource: z7.enum(["env", "file", "none"]),
|
|
812
|
+
relayTokenSource: z7.enum(["env", "file", "none"]),
|
|
813
|
+
hookPort: z7.number(),
|
|
814
|
+
hookPortSource: z7.enum(["env", "file", "default"])
|
|
815
|
+
}),
|
|
816
|
+
relay: z7.object({
|
|
817
|
+
connected: z7.boolean(),
|
|
818
|
+
proxyId: z7.string(),
|
|
819
|
+
reconnectAttempt: z7.number(),
|
|
820
|
+
queueDepth: z7.number()
|
|
821
|
+
}).nullable(),
|
|
822
|
+
sessions: z7.array(
|
|
823
|
+
z7.object({
|
|
824
|
+
id: z7.string(),
|
|
825
|
+
mode: z7.enum(["pty", "json"]),
|
|
826
|
+
state: z7.enum(sessionStateValues2),
|
|
827
|
+
createdAt: z7.string(),
|
|
828
|
+
name: z7.string().optional(),
|
|
829
|
+
hasWorker: z7.boolean()
|
|
830
|
+
})
|
|
831
|
+
)
|
|
832
|
+
}),
|
|
833
|
+
// terminal → serve:终端标题变化,由 xterm onTitleChange 触发
|
|
834
|
+
z7.object({
|
|
835
|
+
type: z7.literal("pty_title_change"),
|
|
836
|
+
sessionId: z7.string(),
|
|
837
|
+
title: z7.string()
|
|
838
|
+
}),
|
|
839
|
+
// terminal → serve:local runtime 观察到的 PTY 语义事件。
|
|
840
|
+
// 下面 state 枚举必须与 src/common/osc-extractor.ts 的 PtySemanticState 保持一致。
|
|
841
|
+
z7.object({
|
|
842
|
+
type: z7.literal("pty_semantic_event"),
|
|
843
|
+
sessionId: z7.string(),
|
|
844
|
+
state: z7.enum(["working", "turn_complete", "approval_wait", "mid_pause"]),
|
|
845
|
+
title: z7.string().optional(),
|
|
846
|
+
tool: z7.string().optional()
|
|
847
|
+
}),
|
|
848
|
+
// terminal → serve:终端尺寸变化
|
|
849
|
+
z7.object({
|
|
850
|
+
type: z7.literal("pty_resize"),
|
|
851
|
+
sessionId: z7.string(),
|
|
852
|
+
cols: z7.number(),
|
|
853
|
+
rows: z7.number()
|
|
854
|
+
}),
|
|
855
|
+
// serve → terminal:请求 HeadlessTerminal serialize() 快照
|
|
856
|
+
z7.object({
|
|
857
|
+
type: z7.literal("pty_subscribe"),
|
|
858
|
+
sessionId: z7.string(),
|
|
859
|
+
requestId: z7.string().optional()
|
|
860
|
+
}),
|
|
861
|
+
// terminal → serve:serialize() 结果
|
|
862
|
+
z7.object({
|
|
863
|
+
type: z7.literal("pty_snapshot"),
|
|
864
|
+
sessionId: z7.string(),
|
|
865
|
+
cols: z7.number(),
|
|
866
|
+
rows: z7.number(),
|
|
867
|
+
data: z7.string(),
|
|
868
|
+
outputSeq: z7.number().int().nonnegative(),
|
|
869
|
+
requestId: z7.string().optional()
|
|
870
|
+
}),
|
|
871
|
+
// serve → terminal:relay 连接状态变更,供终端给用户显示 remote viewing 是否通畅
|
|
872
|
+
z7.object({
|
|
873
|
+
type: z7.literal("bridge_status"),
|
|
874
|
+
connected: z7.boolean()
|
|
875
|
+
})
|
|
876
|
+
]);
|
|
877
|
+
var WorkerMessageSchema = z7.discriminatedUnion("type", [
|
|
878
|
+
// serve → worker: 发送用户输入给 claude
|
|
879
|
+
z7.object({
|
|
880
|
+
type: z7.literal("worker_input"),
|
|
881
|
+
content: z7.string()
|
|
882
|
+
}),
|
|
883
|
+
// serve → worker: 停止 claude 进程
|
|
884
|
+
z7.object({
|
|
885
|
+
type: z7.literal("worker_stop")
|
|
886
|
+
}),
|
|
887
|
+
// serve → worker: 工具审批响应
|
|
888
|
+
z7.object({
|
|
889
|
+
type: z7.literal("worker_approval_response"),
|
|
890
|
+
requestId: z7.string(),
|
|
891
|
+
behavior: z7.enum(["allow", "deny"]),
|
|
892
|
+
message: z7.string().optional()
|
|
893
|
+
}),
|
|
894
|
+
// worker → serve: claude 输出事件(带序列号)
|
|
895
|
+
z7.object({
|
|
896
|
+
type: z7.literal("worker_event"),
|
|
897
|
+
seq: z7.number(),
|
|
898
|
+
event: z7.record(z7.string(), z7.unknown())
|
|
899
|
+
}),
|
|
900
|
+
// worker → serve: claude 进程退出
|
|
901
|
+
z7.object({
|
|
902
|
+
type: z7.literal("worker_exit"),
|
|
903
|
+
code: z7.number()
|
|
904
|
+
}),
|
|
905
|
+
// worker → serve: 工具审批请求
|
|
906
|
+
z7.object({
|
|
907
|
+
type: z7.literal("worker_approval_request"),
|
|
908
|
+
requestId: z7.string(),
|
|
909
|
+
toolName: z7.string(),
|
|
910
|
+
input: z7.record(z7.string(), z7.unknown())
|
|
911
|
+
}),
|
|
912
|
+
// worker → serve: worker 就绪,claude 已启动
|
|
913
|
+
z7.object({
|
|
914
|
+
type: z7.literal("worker_ready"),
|
|
915
|
+
pid: z7.number()
|
|
916
|
+
}),
|
|
917
|
+
// worker → serve: 从 stream-json 的 system.init 事件捕获 Claude CLI 侧的 session ID
|
|
918
|
+
// proxy 拿它来读 ~/.claude/projects/.../<id>.jsonl 历史或后续 --resume
|
|
919
|
+
z7.object({
|
|
920
|
+
type: z7.literal("worker_claude_session_id"),
|
|
921
|
+
sessionId: z7.string()
|
|
922
|
+
}),
|
|
923
|
+
// serve → worker: 将指定工具加入会话白名单,后续同名工具自动审批
|
|
924
|
+
z7.object({
|
|
925
|
+
type: z7.literal("worker_whitelist_add"),
|
|
926
|
+
toolName: z7.string()
|
|
927
|
+
})
|
|
928
|
+
]);
|
|
929
|
+
function serializeWorkerMsg(msg) {
|
|
930
|
+
return JSON.stringify(msg) + "\n";
|
|
931
|
+
}
|
|
932
|
+
function createWorkerReader(stream, onMessage) {
|
|
933
|
+
const lineBuffer = new LineBuffer();
|
|
934
|
+
lineBuffer.on("data", (line) => {
|
|
935
|
+
const str = typeof line === "string" ? line : line.toString();
|
|
936
|
+
if (str.length === 0) return;
|
|
937
|
+
try {
|
|
938
|
+
const raw = JSON.parse(str);
|
|
939
|
+
const result = WorkerMessageSchema.safeParse(raw);
|
|
940
|
+
if (result.success) {
|
|
941
|
+
onMessage(result.data);
|
|
942
|
+
} else {
|
|
943
|
+
stream.emit(
|
|
944
|
+
"error",
|
|
945
|
+
new Error(`Worker message validation failed: ${result.error.message}`)
|
|
946
|
+
);
|
|
947
|
+
}
|
|
948
|
+
} catch (err) {
|
|
949
|
+
stream.emit("error", new Error("Worker message parse error", { cause: err }));
|
|
950
|
+
}
|
|
951
|
+
});
|
|
952
|
+
stream.pipe(lineBuffer);
|
|
953
|
+
}
|
|
954
|
+
function serializeIpc(msg) {
|
|
955
|
+
return JSON.stringify(msg) + "\n";
|
|
956
|
+
}
|
|
957
|
+
function createIpcReader(stream, onMessage, onBinaryFrame) {
|
|
958
|
+
let buf = Buffer.alloc(0);
|
|
959
|
+
let disposed = false;
|
|
960
|
+
function drain() {
|
|
961
|
+
while (buf.length > 0) {
|
|
962
|
+
if (buf[0] === IPC_BINARY_MARKER) {
|
|
963
|
+
if (buf.length < 5) return;
|
|
964
|
+
const payloadLen = buf.readUInt32LE(1);
|
|
965
|
+
const totalFrameLen = 1 + 4 + payloadLen;
|
|
966
|
+
if (buf.length < totalFrameLen) return;
|
|
967
|
+
const payloadStart = 5;
|
|
968
|
+
const sessionIdLen = buf[payloadStart];
|
|
969
|
+
const sessionId = buf.subarray(payloadStart + 1, payloadStart + 1 + sessionIdLen).toString("utf-8");
|
|
970
|
+
const seqOffset = payloadStart + 1 + sessionIdLen;
|
|
971
|
+
const outputSeq = buf.readUInt32LE(seqOffset);
|
|
972
|
+
const ptyData = Buffer.from(buf.subarray(seqOffset + 4, totalFrameLen));
|
|
973
|
+
if (onBinaryFrame) {
|
|
974
|
+
onBinaryFrame(sessionId, ptyData, outputSeq);
|
|
975
|
+
}
|
|
976
|
+
buf = buf.subarray(totalFrameLen);
|
|
977
|
+
} else {
|
|
978
|
+
const newlineIdx = buf.indexOf(10);
|
|
979
|
+
if (newlineIdx === -1) return;
|
|
980
|
+
const line = buf.subarray(0, newlineIdx).toString("utf-8");
|
|
981
|
+
buf = buf.subarray(newlineIdx + 1);
|
|
982
|
+
if (line.length === 0) continue;
|
|
983
|
+
try {
|
|
984
|
+
const raw = JSON.parse(line);
|
|
985
|
+
const result = IpcMessageSchema.safeParse(raw);
|
|
986
|
+
if (result.success) {
|
|
987
|
+
onMessage(result.data);
|
|
988
|
+
} else {
|
|
989
|
+
stream.emit(
|
|
990
|
+
"error",
|
|
991
|
+
new Error(`IPC message validation failed: ${result.error.message}`)
|
|
992
|
+
);
|
|
993
|
+
}
|
|
994
|
+
} catch (err) {
|
|
995
|
+
stream.emit("error", new Error("IPC message parse error", { cause: err }));
|
|
996
|
+
}
|
|
997
|
+
}
|
|
998
|
+
}
|
|
999
|
+
}
|
|
1000
|
+
function onData(chunk) {
|
|
1001
|
+
if (disposed) return;
|
|
1002
|
+
const incoming = typeof chunk === "string" ? Buffer.from(chunk) : chunk;
|
|
1003
|
+
buf = Buffer.concat([buf, incoming]);
|
|
1004
|
+
drain();
|
|
1005
|
+
}
|
|
1006
|
+
stream.on("data", onData);
|
|
1007
|
+
return () => {
|
|
1008
|
+
disposed = true;
|
|
1009
|
+
stream.off("data", onData);
|
|
1010
|
+
};
|
|
1011
|
+
}
|
|
1012
|
+
|
|
1013
|
+
export {
|
|
1014
|
+
tildify,
|
|
1015
|
+
CONFIG_PATH,
|
|
1016
|
+
SOCK_PATH,
|
|
1017
|
+
PID_PATH,
|
|
1018
|
+
STOPPED_PATH,
|
|
1019
|
+
DESIRED_ENV_PATH,
|
|
1020
|
+
SESSIONS_PATH,
|
|
1021
|
+
DATA_DIR,
|
|
1022
|
+
LOG_DIR,
|
|
1023
|
+
SERVICE_LOG_PATH,
|
|
1024
|
+
sessionPaths,
|
|
1025
|
+
isInitialized,
|
|
1026
|
+
initWorkspace,
|
|
1027
|
+
buildMessage,
|
|
1028
|
+
ControlErrorCode,
|
|
1029
|
+
SessionState,
|
|
1030
|
+
createLogger,
|
|
1031
|
+
LineBuffer,
|
|
1032
|
+
encodeBinaryIpcFrame,
|
|
1033
|
+
serializeWorkerMsg,
|
|
1034
|
+
createWorkerReader,
|
|
1035
|
+
serializeIpc,
|
|
1036
|
+
createIpcReader
|
|
1037
|
+
};
|
|
1038
|
+
//# sourceMappingURL=chunk-QJ5CQDK7.js.map
|