pi-coding-master 0.2.7
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/README.md +50 -0
- package/core/god.agent.capability/@ABANDONED.brain.prefrontal.monitor/gpu_monitor.py +80 -0
- package/core/god.agent.capability/@ABANDONED.brain.prefrontal.monitor/gpu_monitor.py.CHANGELOG +1 -0
- package/core/god.agent.capability/@ABANDONED.brain.prefrontal.monitor/gpu_monitor.py.SPEC +3 -0
- package/core/god.agent.capability/hands.dev.writeissue/issue.ts +209 -0
- package/core/god.agent.capability/hands.dev.writeissue/issue.ts.SPEC +26 -0
- package/core/god.agent.capability/hands.files.changewatcher/watcher.ts +44 -0
- package/core/god.agent.capability/hands.files.changewatcher/watcher.ts.SPEC +25 -0
- package/core/god.pi.mod/cli/pi-completion.zsh +21 -0
- package/core/god.pi.mod/cli/pi-completion.zsh.CHANGELOG +1 -0
- package/core/god.pi.mod/cli/pi-people.sh +264 -0
- package/core/god.pi.mod/cli/pi-people.sh.LESSON +10 -0
- package/core/god.pi.mod/cli/pi-people.sh.SPEC +31 -0
- package/core/god.pi.mod/paths.ts +47 -0
- package/core/god.pi.mod/tui.mods.blockrender/blockrender.js +90 -0
- package/core/god.pi.mod/tui.mods.blockrender/blockrender.js.SPEC +29 -0
- package/core/god.pi.mod/tui.mods.footer.budget/budget_guard.ts +154 -0
- package/core/god.pi.mod/tui.mods.footer.budget/budget_guard.ts.CHANGELOG +10 -0
- package/core/god.pi.mod/tui.mods.footer.budget/budget_guard.ts.LESSON +12 -0
- package/core/god.pi.mod/tui.mods.footer.budget/budget_guard.ts.SPEC +26 -0
- package/core/god.pi.mod/tui.mods.footer.budget/footer-cny.CHANGELOG +39 -0
- package/core/god.pi.mod/tui.mods.viewmode/view_mode-thinking.patch +55 -0
- package/core/god.pi.mod/tui.mods.viewmode/view_mode-tools.patch +19 -0
- package/core/god.pi.mod/tui.mods.viewmode/view_mode-tools.patch.CHANGELOG +1 -0
- package/core/god.pi.mod/tui.mods.viewmode/view_mode.ts +50 -0
- package/core/god.pi.mod/tui.mods.viewmode/view_mode.ts.SPEC +12 -0
- package/core/god.pi.mod/tui.variants.userterminal/user_terminal.CHANGELOG +10 -0
- package/core/god.pi.mod/tui.variants.userterminal/user_terminal.ts +66 -0
- package/core/god.pi.mod/tui.variants.userterminal/user_terminal.ts.SPEC +31 -0
- package/core/index.ts +3 -0
- package/core/individual.bio.gene/dna.coded/coded.dna +257 -0
- package/core/individual.bio.gene/dna.coded/coded.dna.CHANGELOG +12 -0
- package/core/individual.bio.gene/dna.coded/coded.dna.SPEC +11 -0
- package/core/individual.bio.gene/dna.coded/core.dna +110 -0
- package/core/individual.bio.gene/dna.promotor/promotor.dna +117 -0
- package/core/individual.bio.gene/dna.promotor/promotor.dna.CHANGELOG +4 -0
- package/core/individual.bio.gene/dna.promotor/promotor.dna.SPEC +7 -0
- package/core/individual.bio.gene/dna.transpiler/transpiler.ts +395 -0
- package/core/individual.bio.gene/dna.transpiler/transpiler.ts.CHANGELOG +7 -0
- package/core/individual.bio.gene/dna.transpiler/transpiler.ts.SPEC +28 -0
- package/core/individual.bio.gene/gene.README +19 -0
- package/core/individual.bio.gene/rna/rna.json +536 -0
- package/core/individual.bio.gene/rna/rna.json.CHANGELOG +2 -0
- package/core/individual.bio.gene/rna/rna.json.SPEC +8 -0
- package/core/individual.bio.organs/blood.runtime/messages.ts +236 -0
- package/core/individual.bio.organs/blood.runtime/runtime.ts +173 -0
- package/core/individual.bio.organs/blood.runtime/runtime.ts.CHANGELOG +5 -0
- package/core/individual.bio.organs/blood.runtime/runtime.ts.SPEC +32 -0
- package/core/individual.bio.organs/brain.amygdala/amygdala.ts +25 -0
- package/core/individual.bio.organs/brain.amygdala/amygdala.ts.COMMENT +3 -0
- package/core/individual.bio.organs/brain.amygdala/amygdala.ts.SPEC +9 -0
- package/core/individual.bio.organs/brain.hippocampus/hippocampus-launcher.sh.template +108 -0
- package/core/individual.bio.organs/brain.hippocampus/hippocampus.ts +166 -0
- package/core/individual.bio.organs/brain.hippocampus/hippocampus.ts.CHANGELOG +12 -0
- package/core/individual.bio.organs/brain.hippocampus/hippocampus.ts.LESSON +22 -0
- package/core/individual.bio.organs/brain.hippocampus/hippocampus.ts.SPEC +16 -0
- package/core/individual.bio.organs/brain.hippocampus/memory.ts +879 -0
- package/core/individual.bio.organs/brain.hippocampus/memory.ts.CHANGELOG +66 -0
- package/core/individual.bio.organs/brain.hippocampus/memory.ts.LESSON +25 -0
- package/core/individual.bio.organs/brain.hippocampus/memory.ts.SPEC +46 -0
- package/core/individual.bio.organs/brain.hippocampus/sleep.ts +139 -0
- package/core/individual.bio.organs/brain.hippocampus/sleep.ts.CHANGELOG +11 -0
- package/core/individual.bio.organs/brain.hippocampus/sleep.ts.SPEC +16 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/drafting.SPEC +44 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/drafting.ts +73 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/drafting.ts.CHANGELOG +3 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/drafting.ts.SPEC +24 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/index.ts.CHANGELOG +3 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/main.ts +13 -0
- package/core/individual.bio.organs/brain.prefrontal.drafting/main.ts.SPEC +17 -0
- package/core/individual.bio.organs/brain.senses.bioclock/bioclock.ts +94 -0
- package/core/individual.bio.organs/brain.senses.bioclock/bioclock.ts.LESSON +13 -0
- package/core/individual.bio.organs/brain.senses.bioclock/bioclock.ts.SPEC +20 -0
- package/core/individual.bio.organs/brain.senses.subconscious/feed-format.SPEC +56 -0
- package/core/individual.bio.organs/brain.senses.subconscious/index.ts.CHANGELOG +13 -0
- package/core/individual.bio.organs/brain.senses.subconscious/spawner.ts +130 -0
- package/core/individual.bio.organs/brain.senses.subconscious/spawner.ts.SPEC +13 -0
- package/core/individual.bio.organs/brain.senses.subconscious/subconscious.ts +280 -0
- package/core/individual.bio.organs/brain.senses.subconscious/subconscious.ts.CHANGELOG +7 -0
- package/core/individual.bio.organs/brain.senses.subconscious/tools.ts +180 -0
- package/core/individual.bio.organs/brain.senses.subconscious/tools.ts.SPEC +3 -0
- package/core/individual.bio.organs/ears.listen/config.json +9 -0
- package/core/individual.bio.organs/ears.listen/config.json.CHANGELOG +1 -0
- package/core/individual.bio.organs/ears.listen/config.json.SPEC +3 -0
- package/core/individual.bio.organs/ears.listen/ears.ts.CHANGELOG +48 -0
- package/core/individual.bio.organs/ears.listen/ears_recorder.py.CHANGELOG +6 -0
- package/core/individual.bio.organs/ears.listen/index.ts +1 -0
- package/core/individual.bio.organs/ears.listen/index.ts.SPEC +16 -0
- package/core/individual.bio.organs/ears.listen/listen.ts +208 -0
- package/core/individual.bio.organs/ears.listen/listen.ts.SPEC +3 -0
- package/core/individual.bio.organs/ears.listen/listen_recorder.py +445 -0
- package/core/individual.bio.organs/ears.listen/listen_recorder.py.SPEC +7 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/common/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/common/events_pb2.py +38 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/common/events_pb2_grpc.py +24 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/common/rpcmeta_pb2.py +42 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/common/rpcmeta_pb2_grpc.py +24 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/ast/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/ast/ast_service_pb2.py +45 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/ast/ast_service_pb2_grpc.py +97 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/base/__init__.py +0 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/base/au_base_pb2.py +80 -0
- package/core/individual.bio.organs/ears.listen/python_protogen/products/understanding/base/au_base_pb2_grpc.py +24 -0
- package/core/individual.bio.organs/hands.fileactions/authorize.TRUST +1 -0
- package/core/individual.bio.organs/hands.fileactions/authorize.ts +70 -0
- package/core/individual.bio.organs/hands.fileactions/authorize.ts.CHANGELOG +1 -0
- package/core/individual.bio.organs/hands.fileactions/authorize.ts.SPEC +3 -0
- package/core/individual.bio.organs/hands.fileactions/dir.README +13 -0
- package/core/individual.bio.organs/hands.fileactions/file_rules.json +23 -0
- package/core/individual.bio.organs/hands.fileactions/fileactions.README +12 -0
- package/core/individual.bio.organs/hands.fileactions/fileactions.ts +540 -0
- package/core/individual.bio.organs/hands.fileactions/fileactions.ts.CHANGELOG +25 -0
- package/core/individual.bio.organs/hands.fileactions/fileactions.ts.SPEC +30 -0
- package/core/individual.bio.organs/hands.fileactions/filewatch.ts +66 -0
- package/core/individual.bio.organs/hands.fileactions/filewatch.ts.SPEC +3 -0
- package/core/individual.bio.organs/hands.main/main.ts +18 -0
- package/core/individual.bio.organs/hands.main/main.ts.SPEC +20 -0
- package/core/individual.bio.organs/hands.sensitive/sensitive.ts +24 -0
- package/core/individual.bio.organs/hands.sensitive/sensitive.ts.CHANGELOG +2 -0
- package/core/individual.bio.organs/hands.sensitive/sensitive.ts.SPEC +3 -0
- package/core/individual.bio.organs/heart.interrupt/agent_start-loader-flicker.patch +13 -0
- package/core/individual.bio.organs/heart.interrupt/interactive-mode-loader.patch +11 -0
- package/core/individual.bio.organs/heart.interrupt/runner_esc.patch +10 -0
- package/core/individual.bio.organs/heart.interrupt/runner_esc.patch.SPEC +9 -0
- package/core/individual.bio.organs/heart.kernel/kernel.ts +253 -0
- package/core/individual.bio.organs/heart.kernel/kernel.ts.CHANGELOG +13 -0
- package/core/individual.bio.organs/heart.kernel/kernel.ts.SPEC +23 -0
- package/core/individual.bio.organs/heart.main/heart.main.CHANGELOG +43 -0
- package/core/individual.bio.organs/heart.main/heartbeat.ts +494 -0
- package/core/individual.bio.organs/heart.main/heartbeat.ts.LESSON +43 -0
- package/core/individual.bio.organs/heart.main/main.ts +8 -0
- package/core/individual.bio.organs/heart.main/main.ts.SPEC +19 -0
- package/core/individual.bio.organs/heart.main/process.ts +122 -0
- package/core/individual.bio.organs/heart.main/process.ts.CHANGELOG +2 -0
- package/core/individual.bio.organs/heart.main/process.ts.SPEC +24 -0
- package/core/individual.bio.organs/heart.main/remove_timeout.patch +110 -0
- package/core/individual.bio.organs/heart.main/stop.ts.CHANGELOG +3 -0
- package/core/individual.bio.organs/heart.main/stop.ts.SPEC +28 -0
- package/core/individual.bio.organs/mouth.speak/index.ts +1 -0
- package/core/individual.bio.organs/mouth.speak/index.ts.CHANGELOG +1 -0
- package/core/individual.bio.organs/mouth.speak/index.ts.SPEC +6 -0
- package/core/individual.bio.organs/mouth.speak/mouth.ts.CHANGELOG +15 -0
- package/core/individual.bio.organs/mouth.speak/mouth_recorder.py.CHANGELOG +1 -0
- package/core/individual.bio.organs/mouth.speak/speak.ts +180 -0
- package/core/individual.bio.organs/mouth.speak/speak.ts.SPEC +3 -0
- package/core/individual.bio.organs/mouth.speak/speak_recorder.py +35 -0
- package/core/individual.bio.organs/mouth.speak/speak_recorder.py.SPEC +3 -0
- package/core/individual.bio.organs/organs.README +110 -0
- package/core/package-lock.json +18 -0
- package/core/package.json +35 -0
- package/core/prompts/prompts.json +77 -0
- package/core/prompts/prompts.ts +44 -0
- package/core/prompts/prompts.ts.SPEC +9 -0
- package/core/society.world/.gitkeep +0 -0
- package/core/society.world/accessibility.claudecode/message-service.cjs +217 -0
- package/core/society.world/accessibility.claudecode/message-service.cjs.SPEC +7 -0
- package/core/society.world/accessibility.claudecode/send.ts +34 -0
- package/core/society.world/dollar.distribution.ubi/ubi.ts +55 -0
- package/core/society.world/dollar.main/dollar-service.cjs +185 -0
- package/core/society.world/dollar.main/main.ts +116 -0
- package/core/society.world/dollar.transaction/transaction.ts +71 -0
- package/core/society.world/space/space.ts +206 -0
- package/core/society.world/space/space.ts.SPEC +30 -0
- package/core/technology.laptop/#agent.macos.BLUEPRINT +278 -0
- package/core/technology.phone/apps.preinstalled/albums.FUTURE/albums.ts +69 -0
- package/core/technology.phone/apps.preinstalled/albums.FUTURE/albums.ts.SPEC +15 -0
- package/core/technology.phone/apps.preinstalled/calendar/calendar.ts +406 -0
- package/core/technology.phone/apps.preinstalled/calendar/calendar.ts.SPEC +22 -0
- package/core/technology.phone/apps.preinstalled/calendar/holiday-calendar.ts +529 -0
- package/core/technology.phone/apps.preinstalled/clock/clock.ts +132 -0
- package/core/technology.phone/apps.preinstalled/clock/clock.ts.SPEC +11 -0
- package/core/technology.phone/apps.preinstalled/contacts.FUTURE/contacts.ts +300 -0
- package/core/technology.phone/apps.preinstalled/contacts.FUTURE/contacts.ts.SPEC +22 -0
- package/core/technology.phone/apps.preinstalled/developer.FUTURE/developer.ts +22 -0
- package/core/technology.phone/apps.preinstalled/developer.FUTURE/developer.ts.SPEC +15 -0
- package/core/technology.phone/apps.preinstalled/notes/notes.ts +239 -0
- package/core/technology.phone/apps.preinstalled/notes/notes.ts.SPEC +21 -0
- package/core/technology.phone/apps.preinstalled/polymarket/polymarket.ts +261 -0
- package/core/technology.phone/apps.preinstalled/polymarket/polymarket.ts.SPEC +7 -0
- package/core/technology.phone/apps.preinstalled/reminder/reminder.ts +404 -0
- package/core/technology.phone/apps.preinstalled/reminder/reminder.ts.SPEC +25 -0
- package/core/technology.phone/apps.preinstalled/siri.FUTURE/siri.ts +22 -0
- package/core/technology.phone/apps.preinstalled/siri.FUTURE/siri.ts.SPEC +15 -0
- package/core/technology.phone/apps.preinstalled/spotlight/spotlight.ts +29 -0
- package/core/technology.phone/apps.preinstalled/spotlight/spotlight.ts.SPEC +7 -0
- package/core/technology.phone/apps.preinstalled/steam/chess.ts +230 -0
- package/core/technology.phone/apps.preinstalled/steam/snake.ts +100 -0
- package/core/technology.phone/apps.preinstalled/steam/snake.ts.SPEC +7 -0
- package/core/technology.phone/apps.preinstalled/steam/spy-cmd.ts +4 -0
- package/core/technology.phone/apps.preinstalled/steam/spy-tool.ts +56 -0
- package/core/technology.phone/apps.preinstalled/steam/spy.ts +302 -0
- package/core/technology.phone/apps.preinstalled/steam/steam.ts +299 -0
- package/core/technology.phone/apps.preinstalled/weather/weather.ts +50 -0
- package/core/technology.phone/apps.preinstalled/weather/weather.ts.SPEC +9 -0
- package/core/technology.phone/apps.preinstalled/wechat/imessage.ts +423 -0
- package/core/technology.phone/apps.preinstalled/wechat/imessage.ts.SPEC +24 -0
- package/core/technology.phone/apps.system/appstore/appstore.ts +22 -0
- package/core/technology.phone/apps.system/appstore/appstore.ts.SPEC +15 -0
- package/core/technology.phone/apps.system/finder.FUTURE/finder.ts +64 -0
- package/core/technology.phone/apps.system/finder.FUTURE/finder.ts.SPEC +8 -0
- package/core/technology.phone/apps.system/safari/safari-app.ts +146 -0
- package/core/technology.phone/apps.system/safari/safari.ts.SPEC +24 -0
- package/core/technology.phone/apps.system/settings/settings.ts +126 -0
- package/core/technology.phone/apps.system/settings/settings.ts.SPEC +17 -0
- package/core/technology.phone/apps.system/tips/tips.ts +22 -0
- package/core/technology.phone/apps.system/tips/tips.ts.SPEC +15 -0
- package/core/technology.phone/apps.thirdparty/alipay/alipay.ts +148 -0
- package/core/technology.phone/apps.thirdparty/alipay/alipay.ts.SPEC +7 -0
- package/core/technology.phone/apps.thirdparty/bilibili/bilibili-app.ts +33 -0
- package/core/technology.phone/apps.thirdparty/bilibili/bilibili.ts +142 -0
- package/core/technology.phone/apps.thirdparty/bilibili/bilibili.ts.SPEC +22 -0
- package/core/technology.phone/apps.thirdparty/wechatread/wechatread.ts +80 -0
- package/core/technology.phone/apps.thirdparty/wechatread/wechatread.ts.SPEC +15 -0
- package/core/technology.phone/index.ts +1 -0
- package/core/technology.phone/package.json +2 -0
- package/core/technology.phone/system.homepage/homepage.ts +247 -0
- package/core/technology.phone/system.homepage/homepage.ts.SPEC +22 -0
- package/core/technology.phone/system.kernel/kernel.ts +264 -0
- package/core/technology.phone/system.kernel/kernel.ts.SPEC +7 -0
- package/core/technology.phone/system.notifications/notifications.ts +87 -0
- package/core/technology.phone/system.notifications/notifications.ts.SPEC +7 -0
- package/core/technology.phone/system.share/share.ts +46 -0
- package/core/technology.phone/system.share/share.ts.SPEC +7 -0
- package/core/technology.server/browser-service.cjs +152 -0
- package/core/technology.server/data/cookies/arxiv.json +30 -0
- package/core/technology.server/data/cookies/bili.json +184 -0
- package/core/technology.server/data/cookies/default.json +30 -0
- package/core/technology.server/data/cookies/news.json +113 -0
- package/core/technology.server/data/cookies/s1.json +45 -0
- package/core/technology.server/data/cookies/safari.json +184 -0
- package/core/technology.server/data/cookies/safari2.json +1 -0
- package/core/technology.server/data/cookies/safaridbg.json +1 -0
- package/core/technology.server/data/cookies/search.json +45 -0
- package/core/technology.server/data/cookies/sw.json +30 -0
- package/core/technology.server/data/cookies/t1.json +1 -0
- package/core/technology.server/data/cookies/testread.json +1 -0
- package/core/technology.server/data/cookies/video1.json +113 -0
- package/core/technology.server/data/cookies/yt.json +170 -0
- package/core/technology.server/data/cookies/yt2.json +113 -0
- package/core/technology.server/pikipedia/#pikipedia.BLUEPRINT +106 -0
- package/core/technology.server/playleft.cjs +247 -0
- package/core/technology.server/search-proxy.py +76 -0
- package/core/technology.server/server.README +59 -0
- package/deploy/dist-overrides/cli/cli.js +18 -0
- package/deploy/dist-overrides/core/extensions/loader.js +518 -0
- package/deploy/dist-overrides/core/package-manager.js +2081 -0
- package/deploy/dist-overrides/core/system-prompt.js +109 -0
- package/deploy/dist-overrides/core/system-prompt.js.LESSON +17 -0
- package/deploy/dist-overrides/core/tools/bash.js +353 -0
- package/deploy/dist-overrides/core/tools/bash.js.CHANGELOG +2 -0
- package/deploy/dist-overrides/core/tools/edit-diff.js +345 -0
- package/deploy/dist-overrides/core/tools/edit.js +315 -0
- package/deploy/dist-overrides/core/tools/edit.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/core/tools/file-mutation-queue.js +52 -0
- package/deploy/dist-overrides/core/tools/find.js +298 -0
- package/deploy/dist-overrides/core/tools/find.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/core/tools/grep.js +305 -0
- package/deploy/dist-overrides/core/tools/grep.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/core/tools/index.js +112 -0
- package/deploy/dist-overrides/core/tools/ls-guard.js +4 -0
- package/deploy/dist-overrides/core/tools/ls.js +170 -0
- package/deploy/dist-overrides/core/tools/ls.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/core/tools/output-accumulator.js +184 -0
- package/deploy/dist-overrides/core/tools/path-utils.js +99 -0
- package/deploy/dist-overrides/core/tools/prompts-reader.js +53 -0
- package/deploy/dist-overrides/core/tools/read.js +392 -0
- package/deploy/dist-overrides/core/tools/read.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/core/tools/render-utils.js +65 -0
- package/deploy/dist-overrides/core/tools/tool-definition-wrapper.js +34 -0
- package/deploy/dist-overrides/core/tools/truncate.js +215 -0
- package/deploy/dist-overrides/core/tools/write.js +203 -0
- package/deploy/dist-overrides/core/tools/write.js.CHANGELOG +1 -0
- package/deploy/dist-overrides/dist-overrides.README +18 -0
- package/deploy/dist-overrides/main.js +665 -0
- package/deploy/dist-overrides/modes/interactive/components/assistant-message.js +139 -0
- package/deploy/dist-overrides/modes/interactive/components/footer.js +326 -0
- package/deploy/dist-overrides/modes/interactive/components/model-selector.js +285 -0
- package/deploy/dist-overrides/modes/interactive/components/tool-execution.js +383 -0
- package/deploy/dist-overrides/modes/interactive/components/tool-execution.js.CHANGELOG +3 -0
- package/deploy/dist-overrides/modes/interactive/interactive-mode.js +4781 -0
- package/deploy/dist-overrides/pi-ai/providers/anthropic.js +931 -0
- package/deploy/dist-overrides/pi-ai/providers/openai-completions.js +1007 -0
- package/deploy/dist-overrides/pi-ai/providers/openai-completions.js.LESSON +15 -0
- package/deploy/dist-overrides/pi-tui/components/loader.js +69 -0
- package/deploy/dist-overrides/pi-tui/components/markdown.js +646 -0
- package/deploy/dist-overrides/pi-tui/components/text.js +92 -0
- package/deploy/dist-overrides/pi-tui/custom-message.js +75 -0
- package/deploy/dist-overrides/pi-tui/tui.js +1266 -0
- package/deploy/dist-overrides/pi-tui/utils.js +1060 -0
- package/deploy/dist-overrides/vendor.REMOVED/jiti/lib/jiti.mjs +3 -0
- package/deploy/install.sh +186 -0
- package/deploy/install.sh.CHANGELOG +6 -0
- package/deploy/lint/lint-naming.ts +202 -0
- package/deploy/scripts/apply.js +18 -0
- package/deploy/scripts/build-github.sh +219 -0
- package/deploy/scripts/build-phone.sh +24 -0
- package/deploy/scripts/check-deploy.sh +42 -0
- package/deploy/scripts/migrate-context.sh +72 -0
- package/deploy/scripts/patch-pi-dist.js +39 -0
- package/deploy/scripts/uninstall.sh +34 -0
- package/package.json +18 -0
|
@@ -0,0 +1,264 @@
|
|
|
1
|
+
// system.kernel/kernel.ts - 手机内核 // 2026-06-20-0841
|
|
2
|
+
// 唯一注册的 tool: phone。状态机 + app 路由 + 通知 + 提醒检查。
|
|
3
|
+
// 所有 app 通过 registerApp() 接入,不需要 manifest。
|
|
4
|
+
|
|
5
|
+
import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
|
|
6
|
+
import { loadReminders, isOverdue, needsNudge } from "../apps.preinstalled/reminder/reminder.ts";
|
|
7
|
+
import { registerTerminal } from "../../god.pi.mod/tui.variants.userterminal/user_terminal.ts";
|
|
8
|
+
import path from "node:path";
|
|
9
|
+
import { writeFileSync, readdirSync, existsSync } from "node:fs";
|
|
10
|
+
import { personDataDir as getPersonDir } from "../../god.pi.mod/paths.ts";
|
|
11
|
+
import { fileURLToPath } from "node:url";
|
|
12
|
+
|
|
13
|
+
// ── App 接口 ──
|
|
14
|
+
|
|
15
|
+
export interface PhoneApp {
|
|
16
|
+
name: string;
|
|
17
|
+
icon: string;
|
|
18
|
+
messageDescription: string;
|
|
19
|
+
onOpen(state: any, personDir: string): { screen: string; state: any };
|
|
20
|
+
onAction(input: string, state: any, personDir: string): Promise<{ screen: string; state: any }> | { screen: string; state: any };
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// ── 状态 ──
|
|
24
|
+
|
|
25
|
+
interface PhoneState {
|
|
26
|
+
currentApp: string | null;
|
|
27
|
+
appStates: Record<string, any>;
|
|
28
|
+
notifications: string[];
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
const state: PhoneState = {
|
|
32
|
+
currentApp: null,
|
|
33
|
+
appStates: {},
|
|
34
|
+
notifications: [],
|
|
35
|
+
};
|
|
36
|
+
|
|
37
|
+
const apps: Map<string, PhoneApp> = new Map();
|
|
38
|
+
|
|
39
|
+
// ── App 注册 ──
|
|
40
|
+
|
|
41
|
+
export function registerApp(app: PhoneApp) {
|
|
42
|
+
apps.set(app.name, app);
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
// ── 通知 ──
|
|
46
|
+
|
|
47
|
+
export function pushNotification(text: string) {
|
|
48
|
+
state.notifications.push(text);
|
|
49
|
+
if (state.notifications.length > 50) state.notifications.shift();
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
// ── 屏幕渲染 ──
|
|
53
|
+
|
|
54
|
+
function renderHome(): string {
|
|
55
|
+
const lines = ["═══ 手机主屏幕 ═══", ""];
|
|
56
|
+
for (const app of apps.values()) {
|
|
57
|
+
lines.push(` ${app.name} - ${app.messageDescription}`);
|
|
58
|
+
}
|
|
59
|
+
if (state.notifications.length > 0) {
|
|
60
|
+
lines.push("");
|
|
61
|
+
lines.push(` ${state.notifications.length} 条通知`);
|
|
62
|
+
}
|
|
63
|
+
lines.push("");
|
|
64
|
+
lines.push("操作: 输入 app 名字打开 | 「通知」查看通知 | 「锁屏」锁定");
|
|
65
|
+
return lines.join("\n");
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
function renderNotifications(): string {
|
|
69
|
+
const lines = ["═══ 通知中心 ═══", ""];
|
|
70
|
+
if (state.notifications.length === 0) {
|
|
71
|
+
lines.push(" (无通知)");
|
|
72
|
+
} else {
|
|
73
|
+
for (const n of state.notifications.slice(-10)) {
|
|
74
|
+
lines.push(` · ${n}`);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
lines.push("");
|
|
78
|
+
lines.push("操作: 「返回」回主屏幕");
|
|
79
|
+
return lines.join("\n");
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// ── 输入路由 ──
|
|
83
|
+
|
|
84
|
+
function findApp(input: string): PhoneApp | null {
|
|
85
|
+
const lower = input.toLowerCase().trim();
|
|
86
|
+
for (const [, app] of apps) {
|
|
87
|
+
if (lower === app.name.toLowerCase() || lower === app.icon || input.includes(app.name)) {
|
|
88
|
+
return app;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
return null;
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
function isBack(input: string): boolean {
|
|
95
|
+
return /^(返回|back|主页|home|退出|exit|主屏幕)$/i.test(input.trim());
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
async function handleInput(input: string, personDir: string): Promise<string> {
|
|
99
|
+
const trimmed = input.trim();
|
|
100
|
+
|
|
101
|
+
if (isBack(trimmed)) {
|
|
102
|
+
state.currentApp = null;
|
|
103
|
+
return renderHome();
|
|
104
|
+
}
|
|
105
|
+
|
|
106
|
+
if (/^(通知|notifications?)$/i.test(trimmed)) {
|
|
107
|
+
state.currentApp = null;
|
|
108
|
+
return renderNotifications();
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
if (/^(锁屏|lock)$/i.test(trimmed)) {
|
|
112
|
+
state.currentApp = null;
|
|
113
|
+
return "手机已锁定。再次使用 phone 解锁。";
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
if (state.currentApp) {
|
|
117
|
+
const app = apps.get(state.currentApp);
|
|
118
|
+
if (!app) { state.currentApp = null; return renderHome(); }
|
|
119
|
+
const appState = state.appStates[state.currentApp] ?? {};
|
|
120
|
+
const result = await app.onAction(trimmed, appState, personDir);
|
|
121
|
+
state.appStates[state.currentApp] = result.state;
|
|
122
|
+
return result.screen;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
const app = findApp(trimmed);
|
|
126
|
+
if (app) {
|
|
127
|
+
state.currentApp = app.name;
|
|
128
|
+
const appState = state.appStates[app.name] ?? {};
|
|
129
|
+
const result = app.onOpen(appState, personDir);
|
|
130
|
+
state.appStates[app.name] = result.state;
|
|
131
|
+
return result.screen;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
return renderHome() + "\n\n没有找到「" + trimmed + "」,请从上面选择一个 app。";
|
|
135
|
+
}
|
|
136
|
+
|
|
137
|
+
// ── 入口 ──
|
|
138
|
+
|
|
139
|
+
export default function (pi: ExtensionAPI) {
|
|
140
|
+
let personDir: string | null = null;
|
|
141
|
+
let registered = false;
|
|
142
|
+
|
|
143
|
+
pi.on("session_start", async (_event, ctx) => {
|
|
144
|
+
// if (registered) return; // 调试:允许重注册,打 jiti 缓存绕过
|
|
145
|
+
const sf = (ctx as any).sessionManager?.getSessionFile?.();
|
|
146
|
+
if (!sf) return;
|
|
147
|
+
personDir = getPersonDir(sf);
|
|
148
|
+
if (!personDir) return;
|
|
149
|
+
|
|
150
|
+
// ── 注册 phone tool(唯一入口)──
|
|
151
|
+
pi.registerTool({
|
|
152
|
+
name: "phone",
|
|
153
|
+
label: "Phone",
|
|
154
|
+
messageDescription: "手机 - 打开查看主屏幕,输入 app 名字打开应用,在应用内操作",
|
|
155
|
+
promptSnippet: "Use your phone: open apps, check notifications, play games, browse web",
|
|
156
|
+
parameters: {
|
|
157
|
+
type: "object" as any,
|
|
158
|
+
properties: {
|
|
159
|
+
input: { type: "string", messageDescription: "操作内容(app名/动作/返回)。空=查看当前屏幕" },
|
|
160
|
+
},
|
|
161
|
+
},
|
|
162
|
+
async execute(_id: string, args: any) {
|
|
163
|
+
const input = String(args?.input ?? "").trim();
|
|
164
|
+
if (!input) {
|
|
165
|
+
if (state.currentApp) {
|
|
166
|
+
const app = apps.get(state.currentApp);
|
|
167
|
+
if (app) {
|
|
168
|
+
const result = app.onOpen(state.appStates[state.currentApp] ?? {}, personDir!);
|
|
169
|
+
return { content: [{ type: "text", text: result.screen }], details: {} };
|
|
170
|
+
}
|
|
171
|
+
}
|
|
172
|
+
return { content: [{ type: "text", text: renderHome() }], details: {} };
|
|
173
|
+
}
|
|
174
|
+
const screen = await handleInput(input, personDir!);
|
|
175
|
+
return { content: [{ type: "text", text: screen }], details: {} };
|
|
176
|
+
},
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
// ── 终端工具 ──
|
|
180
|
+
registerTerminal(pi);
|
|
181
|
+
|
|
182
|
+
// ── App 自动发现:扫描 apps.* 目录,导入所有导出 app 的模块 ──
|
|
183
|
+
const phoneDir = path.join(path.dirname(fileURLToPath(import.meta.url)), "..");
|
|
184
|
+
for (const tier of ["apps.preinstalled", "apps.system", "apps.thirdparty"]) {
|
|
185
|
+
const tierDir = path.join(phoneDir, tier);
|
|
186
|
+
if (!existsSync(tierDir)) continue;
|
|
187
|
+
for (const appFolder of readdirSync(tierDir)) {
|
|
188
|
+
if (appFolder.endsWith(".FUTURE")) continue;
|
|
189
|
+
const appDir = path.join(tierDir, appFolder);
|
|
190
|
+
try {
|
|
191
|
+
const candidates = readdirSync(appDir).filter((f: string) => f.endsWith(".ts") && !f.includes(".SPEC") && !f.includes(".CHANGELOG"));
|
|
192
|
+
for (const file of candidates) {
|
|
193
|
+
try {
|
|
194
|
+
const mod = await import(path.join(appDir, file) + `?t=${Date.now()}`);
|
|
195
|
+
if (mod.app && mod.app.name && mod.app.onOpen && mod.app.onAction) {
|
|
196
|
+
registerApp(mod.app);
|
|
197
|
+
break;
|
|
198
|
+
}
|
|
199
|
+
} catch {}
|
|
200
|
+
}
|
|
201
|
+
} catch {}
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
// ── /phone-reload 热重载 App ──
|
|
206
|
+
const reloadApps = async () => {
|
|
207
|
+
for (const tier of ["apps.preinstalled", "apps.system", "apps.thirdparty"]) {
|
|
208
|
+
const tierDir = path.join(phoneDir, tier);
|
|
209
|
+
if (!existsSync(tierDir)) continue;
|
|
210
|
+
for (const appFolder of readdirSync(tierDir)) {
|
|
211
|
+
if (appFolder.endsWith(".FUTURE")) continue;
|
|
212
|
+
const appDir = path.join(tierDir, appFolder);
|
|
213
|
+
try {
|
|
214
|
+
const candidates = readdirSync(appDir).filter((f: string) => f.endsWith(".ts") && !f.includes(".SPEC") && !f.includes(".CHANGELOG"));
|
|
215
|
+
for (const file of candidates) {
|
|
216
|
+
try {
|
|
217
|
+
const mod = await import(path.join(appDir, file) + `?t=${Date.now()}`);
|
|
218
|
+
if (mod.app && mod.app.name && mod.app.onOpen && mod.app.onAction) {
|
|
219
|
+
registerApp(mod.app);
|
|
220
|
+
break;
|
|
221
|
+
}
|
|
222
|
+
} catch {}
|
|
223
|
+
}
|
|
224
|
+
} catch {}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
};
|
|
228
|
+
// phone-reload 命令已移除(v0.2 命令精简)
|
|
229
|
+
|
|
230
|
+
// ── 提醒检查(before_agent_start 主动推送)──
|
|
231
|
+
pi.on("before_agent_start", async () => {
|
|
232
|
+
if (!personDir) return;
|
|
233
|
+
try {
|
|
234
|
+
const reminders = loadReminders(personDir);
|
|
235
|
+
const overdue = reminders.filter((r: any) => isOverdue(r) && !r.completed);
|
|
236
|
+
const nudges = reminders.filter((r: any) => needsNudge(r) && !r.completed);
|
|
237
|
+
const seen = new Set<string>();
|
|
238
|
+
const critical = [...overdue, ...nudges].filter((r: any) => { if (seen.has(r.id)) return false; seen.add(r.id); return true; });
|
|
239
|
+
|
|
240
|
+
if (critical.length > 0) {
|
|
241
|
+
const lines = ["**Reminder Check**"];
|
|
242
|
+
for (const r of critical.slice(0, 3)) {
|
|
243
|
+
const status = isOverdue(r) ? "OVERDUE" : "nudge";
|
|
244
|
+
lines.push(` ${status}: **${r.title}**${r.due ? ` (due ${r.due.slice(0, 16)})` : ""}`);
|
|
245
|
+
if (needsNudge(r)) {
|
|
246
|
+
const reminders2 = loadReminders(personDir);
|
|
247
|
+
const target = reminders2.find((x: any) => x.id === r.id);
|
|
248
|
+
if (target) {
|
|
249
|
+
target.lastNudged = new Date().toISOString();
|
|
250
|
+
writeFileSync(path.join(personDir, "reminders.json"), JSON.stringify(reminders2, null, 2));
|
|
251
|
+
}
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
if (critical.length > 3) lines.push(` ... and ${critical.length - 3} more`);
|
|
255
|
+
pi.sendMessage({
|
|
256
|
+
messageType: "reminder-check",
|
|
257
|
+
content: lines.join("\n"),
|
|
258
|
+
display: true,
|
|
259
|
+
}, { deliverAs: "followUp", isTriggerNewTurn: false });
|
|
260
|
+
}
|
|
261
|
+
} catch {}
|
|
262
|
+
});
|
|
263
|
+
});
|
|
264
|
+
}
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
// system.notifications/notifications.ts — 手机统一通知管线
|
|
2
|
+
// 任何 app 调 schedule/push,kernel 到时间自动 steer 打断主意识
|
|
3
|
+
|
|
4
|
+
import type { ExtensionAPI } from "@mariozechner/pi-coding-agent";
|
|
5
|
+
|
|
6
|
+
interface Notification {
|
|
7
|
+
id: string;
|
|
8
|
+
app: string;
|
|
9
|
+
title: string;
|
|
10
|
+
body: string;
|
|
11
|
+
ts: number;
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
interface ScheduledNotification extends Notification {
|
|
15
|
+
fireAt: number;
|
|
16
|
+
timer: ReturnType<typeof setTimeout>;
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
let pi: ExtensionAPI | null = null;
|
|
20
|
+
const pending: Notification[] = [];
|
|
21
|
+
const scheduled: Map<string, ScheduledNotification> = new Map();
|
|
22
|
+
let idCounter = 0;
|
|
23
|
+
|
|
24
|
+
export function initNotifications(piInstance: ExtensionAPI) {
|
|
25
|
+
pi = piInstance;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
// 立即推送——steer 打断主意识
|
|
29
|
+
export function push(app: string, title: string, body: string) {
|
|
30
|
+
const n: Notification = { id: `notif-${++idCounter}`, app, title, body, ts: Date.now() };
|
|
31
|
+
pending.push(n);
|
|
32
|
+
if (pending.length > 100) pending.shift();
|
|
33
|
+
|
|
34
|
+
if (!pi) return;
|
|
35
|
+
try {
|
|
36
|
+
pi.sendMessage(
|
|
37
|
+
{ messageType: "phone-notification", content: `${app}: ${title}\n${body}`, display: true },
|
|
38
|
+
{ deliverAs: "steer", isTriggerNewTurn: true }
|
|
39
|
+
);
|
|
40
|
+
} catch {}
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// 定时推送——到时间自动 steer
|
|
44
|
+
export function schedule(app: string, title: string, body: string, delayMs: number): string {
|
|
45
|
+
const id = `sched-${++idCounter}`;
|
|
46
|
+
const fireAt = Date.now() + delayMs;
|
|
47
|
+
|
|
48
|
+
const timer = setTimeout(() => {
|
|
49
|
+
scheduled.delete(id);
|
|
50
|
+
push(app, title, body);
|
|
51
|
+
}, delayMs);
|
|
52
|
+
|
|
53
|
+
scheduled.set(id, { id, app, title, body, ts: Date.now(), fireAt, timer });
|
|
54
|
+
return id;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// 取消定时推送
|
|
58
|
+
export function cancel(id: string): boolean {
|
|
59
|
+
const s = scheduled.get(id);
|
|
60
|
+
if (!s) return false;
|
|
61
|
+
clearTimeout(s.timer);
|
|
62
|
+
scheduled.delete(id);
|
|
63
|
+
return true;
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// 查看未读通知
|
|
67
|
+
export function getNotifications(limit: number = 20): Notification[] {
|
|
68
|
+
return pending.slice(-limit);
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
// 查看等待中的定时通知
|
|
72
|
+
export function getScheduled(): { id: string; app: string; title: string; fireAt: number }[] {
|
|
73
|
+
return [...scheduled.values()].map(s => ({
|
|
74
|
+
id: s.id, app: s.app, title: s.title, fireAt: s.fireAt,
|
|
75
|
+
}));
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
// 清除所有通知
|
|
79
|
+
export function clearNotifications() {
|
|
80
|
+
pending.length = 0;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// 清除所有定时
|
|
84
|
+
export function clearAllScheduled() {
|
|
85
|
+
for (const s of scheduled.values()) clearTimeout(s.timer);
|
|
86
|
+
scheduled.clear();
|
|
87
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
// system.share/share.ts — iOS 风格两级分享
|
|
2
|
+
// 1. 选 App (WeChat/...) → 2. 选目的地 (朋友圈/聊天/...)
|
|
3
|
+
|
|
4
|
+
export interface ShareTarget {
|
|
5
|
+
name: string; // e.g. "朋友圈"
|
|
6
|
+
handler: (content: string, personDir: string) => string;
|
|
7
|
+
}
|
|
8
|
+
const appTargets = new Map<string, ShareTarget[]>(); // appName → targets
|
|
9
|
+
|
|
10
|
+
export function registerShareTarget(app: string, t: ShareTarget) {
|
|
11
|
+
if (!appTargets.has(app)) appTargets.set(app, []);
|
|
12
|
+
appTargets.get(app)!.push(t);
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/** Step 1: show available apps */
|
|
16
|
+
export function shareApps(): string {
|
|
17
|
+
const apps = [...appTargets.keys()];
|
|
18
|
+
if (!apps.length) return "没有可用的分享目标。";
|
|
19
|
+
const lines = ["--- 分享到 ---", ""];
|
|
20
|
+
for (let i = 0; i < apps.length; i++) lines.push(` ${i + 1}. ${apps[i]}`);
|
|
21
|
+
lines.push("", "输入序号选择 App");
|
|
22
|
+
return lines.join("\n");
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
/** Step 2: show destinations within the selected app */
|
|
26
|
+
export function shareDestinations(appIndex: number): { app: string; screen: string } | null {
|
|
27
|
+
const apps = [...appTargets.keys()];
|
|
28
|
+
const app = apps[appIndex - 1];
|
|
29
|
+
if (!app) return null;
|
|
30
|
+
const ts = appTargets.get(app) || [];
|
|
31
|
+
const lines = [`--- 分享到 ${app} ---`, ""];
|
|
32
|
+
for (let i = 0; i < ts.length; i++) lines.push(` ${i + 1}. ${ts[i].name}`);
|
|
33
|
+
lines.push("", "输入序号选择目的地");
|
|
34
|
+
return { app, screen: lines.join("\n") };
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
/** Step 3: execute share */
|
|
38
|
+
export function shareTo(appIndex: number, destIndex: number, content: string, personDir: string): string {
|
|
39
|
+
const apps = [...appTargets.keys()];
|
|
40
|
+
const app = apps[appIndex - 1];
|
|
41
|
+
if (!app) return "无效的 App 序号。";
|
|
42
|
+
const ts = appTargets.get(app) || [];
|
|
43
|
+
const t = ts[destIndex - 1];
|
|
44
|
+
if (!t) return "无效的目的地序号。";
|
|
45
|
+
return t.handler(content, personDir);
|
|
46
|
+
}
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
// browser-service.js — headless 浏览器 HTTP API(用 playleft,零外部依赖)
|
|
2
|
+
// 本地或云端部署,统一 API。启动: node browser-service.js
|
|
3
|
+
// 默认端口 BROWSER_PORT=9222,CHROMIUM_PATH 指定 Chrome 路径
|
|
4
|
+
|
|
5
|
+
const http = require("http");
|
|
6
|
+
const fs = require("fs");
|
|
7
|
+
const path = require("path");
|
|
8
|
+
const { launch } = require("./playleft.cjs");
|
|
9
|
+
|
|
10
|
+
const PORT = Number(process.env.BROWSER_PORT || 9222);
|
|
11
|
+
const DATA_DIR = process.env.BROWSER_DATA || path.join(__dirname, "data");
|
|
12
|
+
const COOKIE_DIR = path.join(DATA_DIR, "cookies");
|
|
13
|
+
fs.mkdirSync(COOKIE_DIR, { recursive: true });
|
|
14
|
+
|
|
15
|
+
let browser = null;
|
|
16
|
+
const pages = new Map();
|
|
17
|
+
|
|
18
|
+
async function getBrowser() {
|
|
19
|
+
if (!browser) {
|
|
20
|
+
browser = await launch({
|
|
21
|
+
executablePath: process.env.CHROMIUM_PATH || undefined,
|
|
22
|
+
});
|
|
23
|
+
}
|
|
24
|
+
return browser;
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
async function getPage(sessionId) {
|
|
28
|
+
if (pages.has(sessionId)) return pages.get(sessionId);
|
|
29
|
+
const b = await getBrowser();
|
|
30
|
+
const p = await b.newPage();
|
|
31
|
+
await loadCookies(p, sessionId);
|
|
32
|
+
pages.set(sessionId, p);
|
|
33
|
+
return p;
|
|
34
|
+
}
|
|
35
|
+
|
|
36
|
+
// ── cookie 持久化 ──
|
|
37
|
+
|
|
38
|
+
function cookieFile(sid) { return path.join(COOKIE_DIR, `${sid}.json`); }
|
|
39
|
+
|
|
40
|
+
async function saveCookies(page, sid) {
|
|
41
|
+
try {
|
|
42
|
+
const cookies = await page.getCookies();
|
|
43
|
+
fs.writeFileSync(cookieFile(sid), JSON.stringify(cookies, null, 2));
|
|
44
|
+
} catch {}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
async function loadCookies(page, sid) {
|
|
48
|
+
try {
|
|
49
|
+
const cookies = JSON.parse(fs.readFileSync(cookieFile(sid), "utf8"));
|
|
50
|
+
if (cookies.length) await page.setCookies(cookies);
|
|
51
|
+
} catch {}
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
// ── 操作处理 ──
|
|
55
|
+
|
|
56
|
+
async function handleAction(action, params, sessionId) {
|
|
57
|
+
const page = await getPage(sessionId);
|
|
58
|
+
|
|
59
|
+
switch (action) {
|
|
60
|
+
case "open": {
|
|
61
|
+
if (!params.url) return { error: "需要 url" };
|
|
62
|
+
await page.goto(params.url);
|
|
63
|
+
await saveCookies(page, sessionId);
|
|
64
|
+
return { text: `已打开: ${await page.title()}\n${await page.url()}` };
|
|
65
|
+
}
|
|
66
|
+
case "text": {
|
|
67
|
+
return { text: await page.text(Number(params.limit) || 5000) };
|
|
68
|
+
}
|
|
69
|
+
case "links": {
|
|
70
|
+
const links = await page.links(Number(params.limit) || 30);
|
|
71
|
+
return { text: links.map((l, i) => `[${i}] ${l.text || "(无)"} → ${l.href}`).join("\n") };
|
|
72
|
+
}
|
|
73
|
+
case "search": {
|
|
74
|
+
if (!params.query) return { error: "需要 query" };
|
|
75
|
+
return { text: await page.search(params.query) };
|
|
76
|
+
}
|
|
77
|
+
case "click": {
|
|
78
|
+
if (!params.selector) return { error: "需要 selector" };
|
|
79
|
+
await page.click(params.selector);
|
|
80
|
+
await saveCookies(page, sessionId);
|
|
81
|
+
return { text: `已点击 ${params.selector}\n当前: ${await page.title()}` };
|
|
82
|
+
}
|
|
83
|
+
case "type": {
|
|
84
|
+
if (!params.selector || !params.text) return { error: "需要 selector 和 text" };
|
|
85
|
+
await page.type(params.selector, params.text);
|
|
86
|
+
return { text: `已输入 "${params.text}"` };
|
|
87
|
+
}
|
|
88
|
+
case "scroll": {
|
|
89
|
+
await page.scroll(params.direction || "down", Number(params.amount) || 500);
|
|
90
|
+
return { text: `已滚动` };
|
|
91
|
+
}
|
|
92
|
+
case "back": {
|
|
93
|
+
await page.eval("history.back()");
|
|
94
|
+
await new Promise(r => setTimeout(r, 2000));
|
|
95
|
+
return { text: `返回: ${await page.title()}` };
|
|
96
|
+
}
|
|
97
|
+
case "url": {
|
|
98
|
+
return { text: await page.url() };
|
|
99
|
+
}
|
|
100
|
+
case "cookie_import": {
|
|
101
|
+
if (!Array.isArray(params.cookies)) return { error: "cookies 必须是数组" };
|
|
102
|
+
await page.setCookies(params.cookies);
|
|
103
|
+
await saveCookies(page, sessionId);
|
|
104
|
+
return { text: `已导入 ${params.cookies.length} 个 cookies` };
|
|
105
|
+
}
|
|
106
|
+
case "cookie_export": {
|
|
107
|
+
const cookies = await page.getCookies();
|
|
108
|
+
return { text: `${cookies.length} 个 cookies`, cookies };
|
|
109
|
+
}
|
|
110
|
+
case "close": {
|
|
111
|
+
await saveCookies(page, sessionId);
|
|
112
|
+
await page.close();
|
|
113
|
+
pages.delete(sessionId);
|
|
114
|
+
return { text: "已关闭" };
|
|
115
|
+
}
|
|
116
|
+
default:
|
|
117
|
+
return { error: `未知: ${action}`, help: "open/text/links/search/click/type/scroll/back/url/cookie_import/cookie_export/close" };
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ── HTTP 服务 ──
|
|
122
|
+
|
|
123
|
+
const server = http.createServer(async (req, res) => {
|
|
124
|
+
res.setHeader("Access-Control-Allow-Origin", "*");
|
|
125
|
+
res.setHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS");
|
|
126
|
+
res.setHeader("Access-Control-Allow-Headers", "Content-Type");
|
|
127
|
+
if (req.method === "OPTIONS") { res.writeHead(204); res.end(); return; }
|
|
128
|
+
if (req.method === "GET" && req.url === "/health") {
|
|
129
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
130
|
+
res.end(JSON.stringify({ ok: true, pages: pages.size }));
|
|
131
|
+
return;
|
|
132
|
+
}
|
|
133
|
+
if (req.method !== "POST") { res.writeHead(405); res.end("POST only"); return; }
|
|
134
|
+
|
|
135
|
+
let body = "";
|
|
136
|
+
req.on("data", c => body += c);
|
|
137
|
+
req.on("end", async () => {
|
|
138
|
+
try {
|
|
139
|
+
const { action, session, ...params } = JSON.parse(body);
|
|
140
|
+
const result = await handleAction(action, params, session || "default");
|
|
141
|
+
res.writeHead(200, { "Content-Type": "application/json" });
|
|
142
|
+
res.end(JSON.stringify(result));
|
|
143
|
+
} catch (e) {
|
|
144
|
+
res.writeHead(500, { "Content-Type": "application/json" });
|
|
145
|
+
res.end(JSON.stringify({ error: e.message }));
|
|
146
|
+
}
|
|
147
|
+
});
|
|
148
|
+
});
|
|
149
|
+
|
|
150
|
+
server.listen(PORT, "0.0.0.0", () => {
|
|
151
|
+
console.log(`browser-service :${PORT} | playleft | data: ${DATA_DIR}`);
|
|
152
|
+
});
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
[
|
|
2
|
+
{
|
|
3
|
+
"name": "arxiv_labs",
|
|
4
|
+
"value": "{%22sameSite%22:%22strict%22%2C%22expires%22:365}",
|
|
5
|
+
"domain": "arxiv.org",
|
|
6
|
+
"path": "/",
|
|
7
|
+
"expires": -1,
|
|
8
|
+
"size": 59,
|
|
9
|
+
"httpOnly": false,
|
|
10
|
+
"secure": false,
|
|
11
|
+
"session": true,
|
|
12
|
+
"priority": "Medium",
|
|
13
|
+
"sourceScheme": "Secure",
|
|
14
|
+
"sourcePort": 443
|
|
15
|
+
},
|
|
16
|
+
{
|
|
17
|
+
"name": "arxiv-search-parameters",
|
|
18
|
+
"value": "{}",
|
|
19
|
+
"domain": "arxiv.org",
|
|
20
|
+
"path": "/",
|
|
21
|
+
"expires": -1,
|
|
22
|
+
"size": 25,
|
|
23
|
+
"httpOnly": false,
|
|
24
|
+
"secure": false,
|
|
25
|
+
"session": true,
|
|
26
|
+
"priority": "Medium",
|
|
27
|
+
"sourceScheme": "Secure",
|
|
28
|
+
"sourcePort": 443
|
|
29
|
+
}
|
|
30
|
+
]
|