agentshire 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +36 -0
- package/LICENSE +21 -0
- package/README.md +437 -0
- package/README.zh-CN.md +441 -0
- package/ROADMAP.md +207 -0
- package/THIRD_PARTY_NOTICES.md +49 -0
- package/VISION.md +101 -0
- package/index.ts +386 -0
- package/openclaw.plugin.json +27 -0
- package/package.json +100 -0
- package/src/bridge/AGENTS.md +136 -0
- package/src/bridge/ActivityStream.ts +184 -0
- package/src/bridge/CitizenManager.ts +280 -0
- package/src/bridge/DirectorBridge.ts +1076 -0
- package/src/bridge/EventTranslator.ts +142 -0
- package/src/bridge/NpcEventQueue.ts +61 -0
- package/src/bridge/ReconnectManager.ts +42 -0
- package/src/bridge/RouteManager.ts +270 -0
- package/src/bridge/StateTracker.ts +100 -0
- package/src/bridge/ToolVfxMapper.ts +94 -0
- package/src/bridge/__tests__/ActivityStream.test.ts +156 -0
- package/src/bridge/__tests__/CitizenManager.test.ts +174 -0
- package/src/bridge/__tests__/EventTranslator.test.ts +292 -0
- package/src/bridge/__tests__/NpcEventQueue.test.ts +92 -0
- package/src/bridge/__tests__/RouteManager.test.ts +106 -0
- package/src/bridge/data/route-config.json +36 -0
- package/src/bridge/data/route-config.ts +19 -0
- package/src/bridge/implicit-chat.ts +192 -0
- package/src/bridge/index.ts +12 -0
- package/src/contracts/agent-state.ts +87 -0
- package/src/contracts/agui.ts +212 -0
- package/src/contracts/chat.ts +67 -0
- package/src/contracts/events.ts +170 -0
- package/src/contracts/index.ts +71 -0
- package/src/contracts/media.ts +77 -0
- package/src/contracts/registry.ts +32 -0
- package/src/plugin/AGENTS.md +192 -0
- package/src/plugin/__tests__/editor-serve-megapack.test.ts +150 -0
- package/src/plugin/__tests__/hook-translator.test.ts +197 -0
- package/src/plugin/auto-config.ts +131 -0
- package/src/plugin/channel.ts +399 -0
- package/src/plugin/chat-asset-resolver.ts +75 -0
- package/src/plugin/chat-session-watcher.ts +98 -0
- package/src/plugin/citizen-agent-manager.ts +178 -0
- package/src/plugin/citizen-chat-router.ts +83 -0
- package/src/plugin/citizen-workshop-manager.ts +96 -0
- package/src/plugin/custom-asset-manager.ts +250 -0
- package/src/plugin/editor-serve.ts +1368 -0
- package/src/plugin/group-discussion.ts +223 -0
- package/src/plugin/hook-translator.ts +202 -0
- package/src/plugin/llm-agent-proxy.ts +193 -0
- package/src/plugin/outbound-adapter.ts +187 -0
- package/src/plugin/paths.ts +33 -0
- package/src/plugin/plan-manager.ts +594 -0
- package/src/plugin/runtime.ts +9 -0
- package/src/plugin/session-history.ts +838 -0
- package/src/plugin/session-log-watcher.ts +221 -0
- package/src/plugin/soul-prompt-template.ts +290 -0
- package/src/plugin/subagent-tracker.ts +205 -0
- package/src/plugin/tools.ts +493 -0
- package/src/plugin/town-session.ts +40 -0
- package/src/plugin/ws-server.ts +680 -0
- package/src/town-souls.ts +134 -0
- package/town-frontend/dist/assets/CustomAssetStore-oi8aIurt.js +2 -0
- package/town-frontend/dist/assets/GLTFLoader-BA5RqSME.js +3804 -0
- package/town-frontend/dist/assets/OrbitControls-ZmySp9sQ.js +2 -0
- package/town-frontend/dist/assets/SettingsPanel-BO52reJe.js +2 -0
- package/town-frontend/dist/assets/SkeletonUtils-BCVmgslc.js +2 -0
- package/town-frontend/dist/assets/SkillLearnCard-Dk38iDpy.js +6 -0
- package/town-frontend/dist/assets/TopicSetupPanel-DLaLHB_Z.js +2 -0
- package/town-frontend/dist/assets/Trap-Bold-CT0JBE39.woff2 +0 -0
- package/town-frontend/dist/assets/Trap-SemiBold-R4_-Ld0j.woff2 +0 -0
- package/town-frontend/dist/assets/WeatherSystem-Cb3BvHes.js +550 -0
- package/town-frontend/dist/assets/avatars/char-female-a.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-female-b.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-female-c.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-female-d.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-female-e.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-female-f.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-a.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-b.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-c.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-d.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-e.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-male-f.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-beaver.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-bee.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-bunny.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-cat.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-caterpillar.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-chick.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-cow.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-crab.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-deer.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-dog.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-elephant.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-fish.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-fox.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-giraffe.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-hog.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-koala.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-lion.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-monkey.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-panda.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-parrot.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-penguin.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-pig.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-polar.webp +0 -0
- package/town-frontend/dist/assets/avatars/char-pet-tiger.webp +0 -0
- package/town-frontend/dist/assets/character-catalog-DSmLtlNC.js +2 -0
- package/town-frontend/dist/assets/citizenEditor-DubGSJOQ.js +296 -0
- package/town-frontend/dist/assets/command-parser-BUd15Bmv.js +12 -0
- package/town-frontend/dist/assets/editor-B5QO0OtX.js +258 -0
- package/town-frontend/dist/assets/editor-Bk8g1NCD.css +1 -0
- package/town-frontend/dist/assets/index-BWfrufil.js +2 -0
- package/town-frontend/dist/assets/index-faS20RJk.js +6 -0
- package/town-frontend/dist/assets/logo-DJI6EtST.png +0 -0
- package/town-frontend/dist/assets/logo-title-AdKPZX5E.png +0 -0
- package/town-frontend/dist/assets/main-CqsN43aT.js +224 -0
- package/town-frontend/dist/assets/main-D7neuy3w.css +1 -0
- package/town-frontend/dist/assets/models/buildings/base.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/base.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/bench.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/bench.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/box_A.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/box_A.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/box_B.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/box_B.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_A.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_A.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_A_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_A_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_B.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_B.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_B_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_B_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_C.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_C.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_C_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_C_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_D.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_D.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_D_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_D_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_E.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_E.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_E_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_E_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_F.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_F.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_F_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_F_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_G.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_G.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_G_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_G_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_H.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_H.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/building_H_withoutBase.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/building_H_withoutBase.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/bush.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/bush.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/car_hatchback.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/car_hatchback.gltf +442 -0
- package/town-frontend/dist/assets/models/buildings/car_police.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/car_police.gltf +442 -0
- package/town-frontend/dist/assets/models/buildings/car_sedan.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/car_sedan.gltf +442 -0
- package/town-frontend/dist/assets/models/buildings/car_stationwagon.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/car_stationwagon.gltf +442 -0
- package/town-frontend/dist/assets/models/buildings/car_taxi.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/car_taxi.gltf +442 -0
- package/town-frontend/dist/assets/models/buildings/citybits_texture.png +0 -0
- package/town-frontend/dist/assets/models/buildings/dumpster.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/dumpster.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/firehydrant.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/firehydrant.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_corner.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_corner.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_corner_curved.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_corner_curved.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_junction.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_junction.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_straight.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_straight.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_straight_crossing.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_straight_crossing.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/road_tsplit.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/road_tsplit.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/streetlight.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/streetlight.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_A.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_A.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_B.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_B.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_C.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/trafficlight_C.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/trash_A.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/trash_A.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/trash_B.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/trash_B.gltf +136 -0
- package/town-frontend/dist/assets/models/buildings/watertower.bin +0 -0
- package/town-frontend/dist/assets/models/buildings/watertower.gltf +136 -0
- package/town-frontend/dist/assets/models/characters/Textures/colormap.png +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-a.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-b.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-c.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-d.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-e.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-female-f.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-a.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-b.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-c.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-d.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-e.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-male-f.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-beaver.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-bee.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-bunny.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-cat.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-caterpillar.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-chick.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-cow.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-crab.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-deer.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-dog.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-elephant.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-fish.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-fox.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-giraffe.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-hog.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-koala.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-lion.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-monkey.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-panda.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-parrot.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-penguin.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-pig.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-polar.glb +0 -0
- package/town-frontend/dist/assets/models/characters/character-pet-tiger.glb +0 -0
- package/town-frontend/dist/assets/models/characters/colormap.png +0 -0
- package/town-frontend/dist/assets/models/furniture/armchair.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/armchair.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/armchair_pillows.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/armchair_pillows.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/bed_double_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/bed_double_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/bed_double_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/bed_double_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/bed_single_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/bed_single_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/bed_single_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/bed_single_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/book_set.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/book_set.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/book_single.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/book_single.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_medium.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_medium.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_medium_decorated.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_medium_decorated.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_small.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_small.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_small_decorated.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cabinet_small_decorated.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cactus_medium_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cactus_medium_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cactus_medium_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cactus_medium_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cactus_small_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cactus_small_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/cactus_small_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/cactus_small_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_A_wood.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_A_wood.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_B_wood.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_B_wood.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_C.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_C.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_stool.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_stool.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/chair_stool_wood.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/chair_stool_wood.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/couch.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/couch.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/couch_pillows.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/couch_pillows.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/furniturebits_texture.png +0 -0
- package/town-frontend/dist/assets/models/furniture/lamp_standing.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/lamp_standing.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/lamp_table.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/lamp_table.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_large_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_large_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_large_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_large_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_medium.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_medium.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_C.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_small_C.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_standing_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_standing_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_standing_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pictureframe_standing_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pillow_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pillow_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/pillow_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/pillow_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_oval_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_oval_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_oval_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_oval_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_stripes_A.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_stripes_A.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_stripes_B.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/rug_rectangle_stripes_B.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_A_big.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_A_big.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_A_small.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_A_small.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_large.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_large.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_large_decorated.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_large_decorated.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_small.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_small.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_small_decorated.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/shelf_B_small_decorated.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/table_low.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/table_low.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/table_medium.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/table_medium.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/table_medium_long.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/table_medium_long.gltf +136 -0
- package/town-frontend/dist/assets/models/furniture/table_small.bin +0 -0
- package/town-frontend/dist/assets/models/furniture/table_small.gltf +136 -0
- package/town-frontend/dist/assets/models/props/bench.bin +0 -0
- package/town-frontend/dist/assets/models/props/bench.gltf +136 -0
- package/town-frontend/dist/assets/models/props/bush.bin +0 -0
- package/town-frontend/dist/assets/models/props/bush.gltf +136 -0
- package/town-frontend/dist/assets/models/props/capybara.glb +0 -0
- package/town-frontend/dist/assets/models/props/car_hatchback.bin +0 -0
- package/town-frontend/dist/assets/models/props/car_hatchback.gltf +442 -0
- package/town-frontend/dist/assets/models/props/car_sedan.bin +0 -0
- package/town-frontend/dist/assets/models/props/car_sedan.gltf +442 -0
- package/town-frontend/dist/assets/models/props/car_taxi.bin +0 -0
- package/town-frontend/dist/assets/models/props/car_taxi.gltf +442 -0
- package/town-frontend/dist/assets/models/props/citybits_texture.png +0 -0
- package/town-frontend/dist/assets/models/props/dumpster.bin +0 -0
- package/town-frontend/dist/assets/models/props/dumpster.gltf +136 -0
- package/town-frontend/dist/assets/models/props/firehydrant.bin +0 -0
- package/town-frontend/dist/assets/models/props/firehydrant.gltf +136 -0
- package/town-frontend/dist/assets/models/props/streetlight.bin +0 -0
- package/town-frontend/dist/assets/models/props/streetlight.gltf +136 -0
- package/town-frontend/dist/assets/models/props/trafficlight_A.bin +0 -0
- package/town-frontend/dist/assets/models/props/trafficlight_A.gltf +136 -0
- package/town-frontend/dist/assets/models/props/trash_A.bin +0 -0
- package/town-frontend/dist/assets/models/props/trash_A.gltf +136 -0
- package/town-frontend/dist/assets/models/props/watertower.bin +0 -0
- package/town-frontend/dist/assets/models/props/watertower.gltf +136 -0
- package/town-frontend/dist/assets/models/stage-deco/Flowers_1_D.glb +0 -0
- package/town-frontend/dist/assets/models/stage-deco/Flowers_2_B.glb +0 -0
- package/town-frontend/dist/assets/models/stage-deco/Grass_A_1.glb +0 -0
- package/town-frontend/dist/assets/models/stage-deco/Park_GrassHill_A.glb +0 -0
- package/town-frontend/dist/assets/models/stage-deco/Pebles_1_A_2.glb +0 -0
- package/town-frontend/dist/assets/music/bgm_day.mp3 +0 -0
- package/town-frontend/dist/assets/music/bgm_dusk.mp3 +0 -0
- package/town-frontend/dist/assets/music/bgm_night.mp3 +0 -0
- package/town-frontend/dist/assets/music/bgm_work.mp3 +0 -0
- package/town-frontend/dist/assets/office-whiteboard-idle-CyEwBrq_.webp +0 -0
- package/town-frontend/dist/assets/preview-B6hYEQij.js +2 -0
- package/town-frontend/dist/assets/town-BKesnERP.js +8888 -0
- package/town-frontend/dist/assets/town-BnswKsjF.css +1 -0
- package/town-frontend/dist/citizen-editor.html +166 -0
- package/town-frontend/dist/editor.html +227 -0
- package/town-frontend/dist/index.html +22 -0
- package/town-frontend/dist/preview.html +56 -0
- package/town-frontend/dist/town.html +165 -0
- package/town-frontend/dist/viewer.html +91 -0
- package/town-souls/CHEN.md +130 -0
- package/town-souls/CHENGZI.md +92 -0
- package/town-souls/CITIZEN_tpl.md +16 -0
- package/town-souls/DIANDIAN.md +92 -0
- package/town-souls/HAITANG.md +94 -0
- package/town-souls/QIQI.md +119 -0
- package/town-souls/SOUL.md +141 -0
- package/town-souls/SOUL_tpl.md +135 -0
- package/town-souls/XIAOLIE.md +107 -0
- package/town-souls/YAN.md +107 -0
- package/town-workspace/IDENTITY.md +5 -0
- package/town-workspace/SOUL.md +141 -0
- package/town-workspace/project-workflow.md +81 -0
- package/town-workspace/town-defaults.json +92 -0
- package/town-workspace/town-guide.md +45 -0
|
@@ -0,0 +1,292 @@
|
|
|
1
|
+
// @desc Tests for EventTranslator: AgentEvent → GameEvent mapping
|
|
2
|
+
import { describe, it, expect, beforeEach } from 'vitest'
|
|
3
|
+
import { EventTranslator } from '../EventTranslator.js'
|
|
4
|
+
import { StateTracker } from '../StateTracker.js'
|
|
5
|
+
import type { AgentEvent } from '../../contracts/events.js'
|
|
6
|
+
|
|
7
|
+
describe('EventTranslator', () => {
|
|
8
|
+
let tracker: StateTracker
|
|
9
|
+
let translator: EventTranslator
|
|
10
|
+
|
|
11
|
+
beforeEach(() => {
|
|
12
|
+
tracker = new StateTracker()
|
|
13
|
+
translator = new EventTranslator(tracker)
|
|
14
|
+
})
|
|
15
|
+
|
|
16
|
+
describe('system.init', () => {
|
|
17
|
+
it('should return world_init GameEvent', () => {
|
|
18
|
+
const event: AgentEvent = { type: 'system', subtype: 'init', sessionId: 'sess-1', model: 'test' }
|
|
19
|
+
const result = translator.translate(event)
|
|
20
|
+
|
|
21
|
+
expect(result).toHaveLength(1)
|
|
22
|
+
expect(result[0]).toMatchObject({
|
|
23
|
+
type: 'world_init',
|
|
24
|
+
config: { townName: '夏尔', stewardName: 'OpenClaw', citizenCount: 5 },
|
|
25
|
+
})
|
|
26
|
+
})
|
|
27
|
+
})
|
|
28
|
+
|
|
29
|
+
describe('sub_agent.started', () => {
|
|
30
|
+
it('should return npc_spawn and workstation_assign when station available', () => {
|
|
31
|
+
const event: AgentEvent = {
|
|
32
|
+
type: 'sub_agent',
|
|
33
|
+
subtype: 'started',
|
|
34
|
+
agentId: 'agent_alice',
|
|
35
|
+
agentType: 'sub',
|
|
36
|
+
parentToolUseId: 'tool-1',
|
|
37
|
+
task: 'Build the frontend',
|
|
38
|
+
model: 'test',
|
|
39
|
+
displayName: 'Alice',
|
|
40
|
+
}
|
|
41
|
+
const result = translator.translate(event)
|
|
42
|
+
|
|
43
|
+
expect(result.length).toBeGreaterThanOrEqual(2)
|
|
44
|
+
expect(result[0]).toMatchObject({
|
|
45
|
+
type: 'npc_spawn',
|
|
46
|
+
npcId: 'alice',
|
|
47
|
+
name: 'Alice',
|
|
48
|
+
role: 'programming',
|
|
49
|
+
category: 'citizen',
|
|
50
|
+
task: 'Build the frontend',
|
|
51
|
+
})
|
|
52
|
+
expect(result[1]).toMatchObject({
|
|
53
|
+
type: 'workstation_assign',
|
|
54
|
+
npcId: 'alice',
|
|
55
|
+
})
|
|
56
|
+
expect((result[1] as any).stationId).toBeTruthy()
|
|
57
|
+
})
|
|
58
|
+
|
|
59
|
+
it('should register agent→NPC mapping in StateTracker', () => {
|
|
60
|
+
const event: AgentEvent = {
|
|
61
|
+
type: 'sub_agent',
|
|
62
|
+
subtype: 'started',
|
|
63
|
+
agentId: 'agent_bob',
|
|
64
|
+
agentType: 'sub',
|
|
65
|
+
parentToolUseId: 'tool-2',
|
|
66
|
+
task: 'Write tests',
|
|
67
|
+
model: 'test',
|
|
68
|
+
}
|
|
69
|
+
translator.translate(event)
|
|
70
|
+
|
|
71
|
+
expect(tracker.resolveNpcId('agent_bob')).toBe('bob')
|
|
72
|
+
})
|
|
73
|
+
|
|
74
|
+
it('should use agentId as displayName when displayName is absent', () => {
|
|
75
|
+
const event: AgentEvent = {
|
|
76
|
+
type: 'sub_agent',
|
|
77
|
+
subtype: 'started',
|
|
78
|
+
agentId: 'agent_charlie',
|
|
79
|
+
agentType: 'sub',
|
|
80
|
+
parentToolUseId: 'tool-3',
|
|
81
|
+
task: 'Deploy',
|
|
82
|
+
model: 'test',
|
|
83
|
+
}
|
|
84
|
+
const result = translator.translate(event)
|
|
85
|
+
|
|
86
|
+
expect(result[0]).toMatchObject({ type: 'npc_spawn', name: 'agent_charlie' })
|
|
87
|
+
})
|
|
88
|
+
|
|
89
|
+
it('should allocate all 10 office workstations before running out', () => {
|
|
90
|
+
const assignedStations = new Set<string>()
|
|
91
|
+
|
|
92
|
+
for (let i = 0; i < 10; i++) {
|
|
93
|
+
const event: AgentEvent = {
|
|
94
|
+
type: 'sub_agent',
|
|
95
|
+
subtype: 'started',
|
|
96
|
+
agentId: `agent_worker_${i}`,
|
|
97
|
+
agentType: 'sub',
|
|
98
|
+
parentToolUseId: `tool-${i}`,
|
|
99
|
+
task: 'Test allocation',
|
|
100
|
+
model: 'test',
|
|
101
|
+
}
|
|
102
|
+
const result = translator.translate(event)
|
|
103
|
+
const workstationAssign = result.find(e => e.type === 'workstation_assign')
|
|
104
|
+
expect(workstationAssign).toBeTruthy()
|
|
105
|
+
assignedStations.add((workstationAssign as { stationId: string }).stationId)
|
|
106
|
+
}
|
|
107
|
+
|
|
108
|
+
expect(assignedStations.size).toBe(10)
|
|
109
|
+
|
|
110
|
+
const overflowEvent: AgentEvent = {
|
|
111
|
+
type: 'sub_agent',
|
|
112
|
+
subtype: 'started',
|
|
113
|
+
agentId: 'agent_worker_overflow',
|
|
114
|
+
agentType: 'sub',
|
|
115
|
+
parentToolUseId: 'tool-overflow',
|
|
116
|
+
task: 'Overflow allocation',
|
|
117
|
+
model: 'test',
|
|
118
|
+
}
|
|
119
|
+
const overflowResult = translator.translate(overflowEvent)
|
|
120
|
+
|
|
121
|
+
expect(overflowResult.some(e => e.type === 'workstation_assign')).toBe(false)
|
|
122
|
+
})
|
|
123
|
+
})
|
|
124
|
+
|
|
125
|
+
describe('sub_agent.done (success)', () => {
|
|
126
|
+
it('should return npc_phase(done) + npc_glow(green) + fx(completion_stars)', () => {
|
|
127
|
+
tracker.registerMapping('agent_alice', 'alice')
|
|
128
|
+
|
|
129
|
+
const event: AgentEvent = {
|
|
130
|
+
type: 'sub_agent',
|
|
131
|
+
subtype: 'done',
|
|
132
|
+
agentId: 'agent_alice',
|
|
133
|
+
result: 'All done',
|
|
134
|
+
toolCalls: 5,
|
|
135
|
+
status: 'completed',
|
|
136
|
+
}
|
|
137
|
+
const result = translator.translate(event)
|
|
138
|
+
|
|
139
|
+
expect(result).toHaveLength(3)
|
|
140
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'alice', phase: 'done' })
|
|
141
|
+
expect(result[1]).toMatchObject({ type: 'npc_glow', npcId: 'alice', color: 'green' })
|
|
142
|
+
expect(result[2]).toMatchObject({ type: 'fx', effect: 'completion_stars', params: { npcId: 'alice' } })
|
|
143
|
+
})
|
|
144
|
+
})
|
|
145
|
+
|
|
146
|
+
describe('sub_agent.done (failed)', () => {
|
|
147
|
+
it('should return npc_phase(error) + npc_glow(red) + fx(error_sparks)', () => {
|
|
148
|
+
tracker.registerMapping('agent_bob', 'bob')
|
|
149
|
+
|
|
150
|
+
const event: AgentEvent = {
|
|
151
|
+
type: 'sub_agent',
|
|
152
|
+
subtype: 'done',
|
|
153
|
+
agentId: 'agent_bob',
|
|
154
|
+
result: 'Crashed',
|
|
155
|
+
toolCalls: 2,
|
|
156
|
+
status: 'failed',
|
|
157
|
+
}
|
|
158
|
+
const result = translator.translate(event)
|
|
159
|
+
|
|
160
|
+
expect(result).toHaveLength(3)
|
|
161
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'bob', phase: 'error' })
|
|
162
|
+
expect(result[1]).toMatchObject({ type: 'npc_glow', npcId: 'bob', color: 'red' })
|
|
163
|
+
expect(result[2]).toMatchObject({ type: 'fx', effect: 'error_sparks', params: { npcId: 'bob' } })
|
|
164
|
+
})
|
|
165
|
+
|
|
166
|
+
it('should return empty array when agentId has no mapping', () => {
|
|
167
|
+
const event: AgentEvent = {
|
|
168
|
+
type: 'sub_agent',
|
|
169
|
+
subtype: 'done',
|
|
170
|
+
agentId: 'unknown_agent',
|
|
171
|
+
result: '',
|
|
172
|
+
toolCalls: 0,
|
|
173
|
+
status: 'completed',
|
|
174
|
+
}
|
|
175
|
+
const result = translator.translate(event)
|
|
176
|
+
|
|
177
|
+
expect(result).toEqual([])
|
|
178
|
+
})
|
|
179
|
+
})
|
|
180
|
+
|
|
181
|
+
describe('text_delta', () => {
|
|
182
|
+
it('should return dialog_message with isStreaming: true', () => {
|
|
183
|
+
const event: AgentEvent = { type: 'text_delta', delta: 'Hello world' }
|
|
184
|
+
const result = translator.translate(event)
|
|
185
|
+
|
|
186
|
+
expect(result).toHaveLength(1)
|
|
187
|
+
expect(result[0]).toMatchObject({
|
|
188
|
+
type: 'dialog_message',
|
|
189
|
+
npcId: 'steward',
|
|
190
|
+
text: 'Hello world',
|
|
191
|
+
isStreaming: true,
|
|
192
|
+
})
|
|
193
|
+
})
|
|
194
|
+
|
|
195
|
+
it('should use contextNpcId when provided', () => {
|
|
196
|
+
const event: AgentEvent = { type: 'text_delta', delta: 'Working...' }
|
|
197
|
+
const result = translator.translate(event, 'alice')
|
|
198
|
+
|
|
199
|
+
expect(result[0]).toMatchObject({ type: 'dialog_message', npcId: 'alice', isStreaming: true })
|
|
200
|
+
})
|
|
201
|
+
})
|
|
202
|
+
|
|
203
|
+
describe('text', () => {
|
|
204
|
+
it('should return dialog_message with isStreaming: false', () => {
|
|
205
|
+
const event: AgentEvent = { type: 'text', content: 'Final answer' }
|
|
206
|
+
const result = translator.translate(event)
|
|
207
|
+
|
|
208
|
+
expect(result).toHaveLength(1)
|
|
209
|
+
expect(result[0]).toMatchObject({
|
|
210
|
+
type: 'dialog_message',
|
|
211
|
+
npcId: 'steward',
|
|
212
|
+
text: 'Final answer',
|
|
213
|
+
isStreaming: false,
|
|
214
|
+
})
|
|
215
|
+
})
|
|
216
|
+
})
|
|
217
|
+
|
|
218
|
+
describe('tool_use', () => {
|
|
219
|
+
it('should return npc_phase(working)', () => {
|
|
220
|
+
const event: AgentEvent = {
|
|
221
|
+
type: 'tool_use',
|
|
222
|
+
toolUseId: 'tu-1',
|
|
223
|
+
name: 'bash',
|
|
224
|
+
input: { command: 'echo hello' },
|
|
225
|
+
}
|
|
226
|
+
const result = translator.translate(event)
|
|
227
|
+
|
|
228
|
+
expect(result.length).toBeGreaterThanOrEqual(1)
|
|
229
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'steward', phase: 'working' })
|
|
230
|
+
})
|
|
231
|
+
|
|
232
|
+
it('should include workstation_screen when tool targets a file', () => {
|
|
233
|
+
const event: AgentEvent = {
|
|
234
|
+
type: 'tool_use',
|
|
235
|
+
toolUseId: 'tu-2',
|
|
236
|
+
name: 'read_file',
|
|
237
|
+
input: { path: '/src/index.ts' },
|
|
238
|
+
}
|
|
239
|
+
const result = translator.translate(event)
|
|
240
|
+
|
|
241
|
+
expect(result).toHaveLength(2)
|
|
242
|
+
expect(result[1]).toMatchObject({
|
|
243
|
+
type: 'workstation_screen',
|
|
244
|
+
state: { mode: 'coding', fileName: 'index.ts' },
|
|
245
|
+
})
|
|
246
|
+
})
|
|
247
|
+
})
|
|
248
|
+
|
|
249
|
+
describe('thinking_delta', () => {
|
|
250
|
+
it('should return npc_phase(thinking) for steward when no contextNpcId', () => {
|
|
251
|
+
const event: AgentEvent = { type: 'thinking_delta', delta: 'hmm...' }
|
|
252
|
+
const result = translator.translate(event)
|
|
253
|
+
|
|
254
|
+
expect(result).toHaveLength(1)
|
|
255
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'steward', phase: 'thinking' })
|
|
256
|
+
})
|
|
257
|
+
|
|
258
|
+
it('should use contextNpcId when provided', () => {
|
|
259
|
+
const event: AgentEvent = { type: 'thinking_delta', delta: 'considering...' }
|
|
260
|
+
const result = translator.translate(event, 'bob')
|
|
261
|
+
|
|
262
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'bob', phase: 'thinking' })
|
|
263
|
+
})
|
|
264
|
+
})
|
|
265
|
+
|
|
266
|
+
describe('error', () => {
|
|
267
|
+
it('should return npc_phase(error) + npc_emote(frustrated) when contextNpcId provided', () => {
|
|
268
|
+
const event: AgentEvent = { type: 'error', message: 'Something broke', recoverable: false }
|
|
269
|
+
const result = translator.translate(event, 'alice')
|
|
270
|
+
|
|
271
|
+
expect(result).toHaveLength(2)
|
|
272
|
+
expect(result[0]).toMatchObject({ type: 'npc_phase', npcId: 'alice', phase: 'error' })
|
|
273
|
+
expect(result[1]).toMatchObject({ type: 'npc_emote', npcId: 'alice', emote: 'frustrated' })
|
|
274
|
+
})
|
|
275
|
+
|
|
276
|
+
it('should return empty array when no contextNpcId', () => {
|
|
277
|
+
const event: AgentEvent = { type: 'error', message: 'fail', recoverable: true }
|
|
278
|
+
const result = translator.translate(event)
|
|
279
|
+
|
|
280
|
+
expect(result).toEqual([])
|
|
281
|
+
})
|
|
282
|
+
})
|
|
283
|
+
|
|
284
|
+
describe('unknown event type', () => {
|
|
285
|
+
it('should return empty array', () => {
|
|
286
|
+
const event = { type: 'debug', category: 'test', message: 'blah' } as AgentEvent
|
|
287
|
+
const result = translator.translate(event)
|
|
288
|
+
|
|
289
|
+
expect(result).toEqual([])
|
|
290
|
+
})
|
|
291
|
+
})
|
|
292
|
+
})
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
// @desc Tests for NpcEventQueue: dialog protection and phase event buffering
|
|
2
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest'
|
|
3
|
+
import { NpcEventQueue, calcDialogDuration } from '../NpcEventQueue.js'
|
|
4
|
+
|
|
5
|
+
describe('calcDialogDuration', () => {
|
|
6
|
+
it('returns minimum 1500 for very short text', () => {
|
|
7
|
+
expect(calcDialogDuration(0)).toBe(1500)
|
|
8
|
+
expect(calcDialogDuration(5)).toBe(1500)
|
|
9
|
+
})
|
|
10
|
+
|
|
11
|
+
it('scales linearly with text length', () => {
|
|
12
|
+
expect(calcDialogDuration(20)).toBe(2400) // 20 * 120
|
|
13
|
+
})
|
|
14
|
+
|
|
15
|
+
it('caps at 8000 for long text', () => {
|
|
16
|
+
expect(calcDialogDuration(200)).toBe(8000)
|
|
17
|
+
expect(calcDialogDuration(1000)).toBe(8000)
|
|
18
|
+
})
|
|
19
|
+
})
|
|
20
|
+
|
|
21
|
+
describe('NpcEventQueue', () => {
|
|
22
|
+
let emitFn: ReturnType<typeof vi.fn>
|
|
23
|
+
let queue: NpcEventQueue
|
|
24
|
+
|
|
25
|
+
beforeEach(() => {
|
|
26
|
+
vi.useFakeTimers()
|
|
27
|
+
emitFn = vi.fn()
|
|
28
|
+
queue = new NpcEventQueue(emitFn)
|
|
29
|
+
})
|
|
30
|
+
|
|
31
|
+
describe('enqueueDialog', () => {
|
|
32
|
+
it('emits events immediately', () => {
|
|
33
|
+
const events = [{ type: 'dialog_message', npcId: 'npc1', text: 'hi' }] as any
|
|
34
|
+
queue.enqueueDialog(events, 5)
|
|
35
|
+
expect(emitFn).toHaveBeenCalledWith(events)
|
|
36
|
+
})
|
|
37
|
+
|
|
38
|
+
it('starts protection timer that unblocks pending phases after duration', () => {
|
|
39
|
+
queue.enqueueDialog([{ type: 'dialog_message' }] as any, 20)
|
|
40
|
+
|
|
41
|
+
const phaseEvents = [{ type: 'npc_phase' }] as any
|
|
42
|
+
queue.enqueuePhase(phaseEvents)
|
|
43
|
+
expect(emitFn).toHaveBeenCalledTimes(1) // only dialog emitted
|
|
44
|
+
|
|
45
|
+
vi.advanceTimersByTime(calcDialogDuration(20))
|
|
46
|
+
expect(emitFn).toHaveBeenCalledTimes(2)
|
|
47
|
+
expect(emitFn).toHaveBeenLastCalledWith(phaseEvents)
|
|
48
|
+
})
|
|
49
|
+
})
|
|
50
|
+
|
|
51
|
+
describe('enqueuePhase', () => {
|
|
52
|
+
it('buffers events during dialog protection', () => {
|
|
53
|
+
queue.enqueueDialog([{ type: 'dialog_message' }] as any, 10)
|
|
54
|
+
emitFn.mockClear()
|
|
55
|
+
|
|
56
|
+
queue.enqueuePhase([{ type: 'phase_a' }] as any)
|
|
57
|
+
queue.enqueuePhase([{ type: 'phase_b' }] as any)
|
|
58
|
+
expect(emitFn).not.toHaveBeenCalled()
|
|
59
|
+
})
|
|
60
|
+
|
|
61
|
+
it('emits immediately when no dialog protection active', () => {
|
|
62
|
+
const events = [{ type: 'npc_phase' }] as any
|
|
63
|
+
queue.enqueuePhase(events)
|
|
64
|
+
expect(emitFn).toHaveBeenCalledWith(events)
|
|
65
|
+
})
|
|
66
|
+
})
|
|
67
|
+
|
|
68
|
+
describe('flush', () => {
|
|
69
|
+
it('drains all pending events immediately and cancels protection', () => {
|
|
70
|
+
queue.enqueueDialog([{ type: 'dialog_message' }] as any, 30)
|
|
71
|
+
emitFn.mockClear()
|
|
72
|
+
|
|
73
|
+
queue.enqueuePhase([{ type: 'phase_a' }] as any)
|
|
74
|
+
queue.enqueuePhase([{ type: 'phase_b' }] as any)
|
|
75
|
+
|
|
76
|
+
queue.flush()
|
|
77
|
+
|
|
78
|
+
expect(emitFn).toHaveBeenCalledTimes(2)
|
|
79
|
+
expect(emitFn).toHaveBeenNthCalledWith(1, [{ type: 'phase_a' }])
|
|
80
|
+
expect(emitFn).toHaveBeenNthCalledWith(2, [{ type: 'phase_b' }])
|
|
81
|
+
})
|
|
82
|
+
|
|
83
|
+
it('allows subsequent enqueuePhase to emit immediately after flush', () => {
|
|
84
|
+
queue.enqueueDialog([{ type: 'dialog_message' }] as any, 10)
|
|
85
|
+
queue.flush()
|
|
86
|
+
emitFn.mockClear()
|
|
87
|
+
|
|
88
|
+
queue.enqueuePhase([{ type: 'phase_c' }] as any)
|
|
89
|
+
expect(emitFn).toHaveBeenCalledWith([{ type: 'phase_c' }])
|
|
90
|
+
})
|
|
91
|
+
})
|
|
92
|
+
})
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
// @desc Tests for RouteManager: A* pathfinding and NPC movement
|
|
2
|
+
import { describe, it, expect, vi, beforeEach } from 'vitest'
|
|
3
|
+
import { RouteManager } from '../RouteManager.js'
|
|
4
|
+
|
|
5
|
+
vi.mock('../data/route-config.js', () => ({
|
|
6
|
+
CITIZEN_DESTINATION_POINTS: [
|
|
7
|
+
{ id: 'dest_a', x: 30, z: 20 },
|
|
8
|
+
{ id: 'dest_b', x: 35, z: 25 },
|
|
9
|
+
{ id: 'dest_c', x: 40, z: 30 },
|
|
10
|
+
],
|
|
11
|
+
ROUTE_GRAPHS: {
|
|
12
|
+
town: {
|
|
13
|
+
node_a: { x: 10, z: 10, neighbors: ['node_b'] },
|
|
14
|
+
node_b: { x: 15, z: 10, neighbors: ['node_a', 'node_c'] },
|
|
15
|
+
node_c: { x: 20, z: 10, neighbors: ['node_b', 'node_d'] },
|
|
16
|
+
node_d: { x: 20, z: 15, neighbors: ['node_c'] },
|
|
17
|
+
},
|
|
18
|
+
office: {
|
|
19
|
+
off_a: { x: 5, z: 3, neighbors: ['off_b'] },
|
|
20
|
+
off_b: { x: 8, z: 3, neighbors: ['off_a'] },
|
|
21
|
+
},
|
|
22
|
+
},
|
|
23
|
+
}))
|
|
24
|
+
|
|
25
|
+
describe('RouteManager', () => {
|
|
26
|
+
let emitFn: ReturnType<typeof vi.fn>
|
|
27
|
+
let rm: RouteManager
|
|
28
|
+
|
|
29
|
+
beforeEach(() => {
|
|
30
|
+
emitFn = vi.fn()
|
|
31
|
+
rm = new RouteManager(emitFn)
|
|
32
|
+
})
|
|
33
|
+
|
|
34
|
+
describe('distance2D', () => {
|
|
35
|
+
it('computes euclidean distance', () => {
|
|
36
|
+
expect(rm.distance2D({ x: 0, z: 0 }, { x: 3, z: 4 })).toBeCloseTo(5)
|
|
37
|
+
})
|
|
38
|
+
|
|
39
|
+
it('returns 0 for same point', () => {
|
|
40
|
+
expect(rm.distance2D({ x: 5, z: 5 }, { x: 5, z: 5 })).toBe(0)
|
|
41
|
+
})
|
|
42
|
+
})
|
|
43
|
+
|
|
44
|
+
describe('planRouteNodePath', () => {
|
|
45
|
+
it('returns [startId] when start equals end', () => {
|
|
46
|
+
expect(rm.planRouteNodePath('node_a', 'node_a', 'town')).toEqual(['node_a'])
|
|
47
|
+
})
|
|
48
|
+
|
|
49
|
+
it('returns correct path through graph', () => {
|
|
50
|
+
const path = rm.planRouteNodePath('node_a', 'node_d', 'town')
|
|
51
|
+
expect(path[0]).toBe('node_a')
|
|
52
|
+
expect(path[path.length - 1]).toBe('node_d')
|
|
53
|
+
expect(path.length).toBeGreaterThanOrEqual(3)
|
|
54
|
+
})
|
|
55
|
+
|
|
56
|
+
it('returns direct [start, end] for disconnected nodes', () => {
|
|
57
|
+
const path = rm.planRouteNodePath('off_a', 'off_b', 'office')
|
|
58
|
+
expect(path).toEqual(['off_a', 'off_b'])
|
|
59
|
+
})
|
|
60
|
+
})
|
|
61
|
+
|
|
62
|
+
describe('chooseCitizenDestination', () => {
|
|
63
|
+
it('returns a destination object with id, x, z, score', () => {
|
|
64
|
+
const dest = rm.chooseCitizenDestination('npc1', { x: 10, z: 10 })
|
|
65
|
+
expect(dest).toHaveProperty('id')
|
|
66
|
+
expect(dest).toHaveProperty('x')
|
|
67
|
+
expect(dest).toHaveProperty('z')
|
|
68
|
+
expect(dest).toHaveProperty('score')
|
|
69
|
+
})
|
|
70
|
+
|
|
71
|
+
it('avoids claimed destinations when unclaimed alternatives exist', () => {
|
|
72
|
+
rm.claimDestinationForNpc('other_npc', 'dest_a')
|
|
73
|
+
const results = new Set<string>()
|
|
74
|
+
for (let i = 0; i < 20; i++) {
|
|
75
|
+
results.add(rm.chooseCitizenDestination('npc1', { x: 10, z: 10 }).id)
|
|
76
|
+
}
|
|
77
|
+
expect(results.has('dest_a')).toBe(false)
|
|
78
|
+
})
|
|
79
|
+
})
|
|
80
|
+
|
|
81
|
+
describe('moveNpcAndWait', () => {
|
|
82
|
+
beforeEach(() => {
|
|
83
|
+
vi.useFakeTimers()
|
|
84
|
+
})
|
|
85
|
+
|
|
86
|
+
it('resolves as interrupted on timeout', async () => {
|
|
87
|
+
const promise = rm.moveNpcAndWait('npc1', { x: 0, y: 0, z: 0 }, 5, 1000)
|
|
88
|
+
|
|
89
|
+
expect(emitFn).toHaveBeenCalledWith([
|
|
90
|
+
expect.objectContaining({ type: 'npc_move_to', npcId: 'npc1' }),
|
|
91
|
+
])
|
|
92
|
+
|
|
93
|
+
vi.advanceTimersByTime(1000)
|
|
94
|
+
await expect(promise).resolves.toBe('interrupted')
|
|
95
|
+
})
|
|
96
|
+
|
|
97
|
+
it('resolves as arrived when resolveMoveRequest is called', async () => {
|
|
98
|
+
const promise = rm.moveNpcAndWait('npc1', { x: 0, y: 0, z: 0 }, 5, 5000)
|
|
99
|
+
|
|
100
|
+
const requestId = emitFn.mock.calls[0][0][0].requestId
|
|
101
|
+
rm.resolveMoveRequest(requestId, 'npc1', 'arrived')
|
|
102
|
+
|
|
103
|
+
await expect(promise).resolves.toBe('arrived')
|
|
104
|
+
})
|
|
105
|
+
})
|
|
106
|
+
})
|
|
@@ -0,0 +1,36 @@
|
|
|
1
|
+
{
|
|
2
|
+
"citizenDestinationPoints": [
|
|
3
|
+
{ "id": "house_a_door", "x": 5.5, "z": 7.2 },
|
|
4
|
+
{ "id": "house_b_door", "x": 5.4, "z": 12.3 },
|
|
5
|
+
{ "id": "house_c_door", "x": 5.5, "z": 17.1 },
|
|
6
|
+
{ "id": "market_door", "x": 29.2, "z": 6.2 },
|
|
7
|
+
{ "id": "cafe_door", "x": 29.1, "z": 12.0 },
|
|
8
|
+
{ "id": "museum_door", "x": 29.0, "z": 18.0 },
|
|
9
|
+
{ "id": "user_home_door", "x": 10.2, "z": 18.2 }
|
|
10
|
+
],
|
|
11
|
+
"routeGraphs": {
|
|
12
|
+
"town": {
|
|
13
|
+
"plaza_n": { "x": 18, "z": 17, "neighbors": ["plaza_c", "west_n", "east_n"] },
|
|
14
|
+
"plaza_c": { "x": 18, "z": 13, "neighbors": ["plaza_n", "plaza_s", "west_c", "east_c"] },
|
|
15
|
+
"plaza_s": { "x": 18, "z": 9, "neighbors": ["plaza_c", "west_s", "east_s"] },
|
|
16
|
+
"west_n": { "x": 12, "z": 17, "neighbors": ["plaza_n", "west_c", "west_far_n"] },
|
|
17
|
+
"west_c": { "x": 12, "z": 13, "neighbors": ["west_n", "plaza_c", "west_s", "west_far_c"] },
|
|
18
|
+
"west_s": { "x": 12, "z": 9, "neighbors": ["west_c", "plaza_s", "west_far_s"] },
|
|
19
|
+
"east_n": { "x": 24, "z": 17, "neighbors": ["plaza_n", "east_c", "east_far_n"] },
|
|
20
|
+
"east_c": { "x": 24, "z": 13, "neighbors": ["east_n", "plaza_c", "east_s", "east_far_c"] },
|
|
21
|
+
"east_s": { "x": 24, "z": 9, "neighbors": ["east_c", "plaza_s", "east_far_s"] },
|
|
22
|
+
"west_far_n": { "x": 8, "z": 17, "neighbors": ["west_n", "west_far_c"] },
|
|
23
|
+
"west_far_c": { "x": 8, "z": 13, "neighbors": ["west_far_n", "west_c", "west_far_s"] },
|
|
24
|
+
"west_far_s": { "x": 8, "z": 9, "neighbors": ["west_far_c", "west_s"] },
|
|
25
|
+
"east_far_n": { "x": 28, "z": 17, "neighbors": ["east_n", "east_far_c"] },
|
|
26
|
+
"east_far_c": { "x": 28, "z": 13, "neighbors": ["east_far_n", "east_c", "east_far_s"] },
|
|
27
|
+
"east_far_s": { "x": 28, "z": 9, "neighbors": ["east_far_c", "east_s"] }
|
|
28
|
+
},
|
|
29
|
+
"office": {
|
|
30
|
+
"office_entry_l": { "x": 14, "z": 5, "neighbors": ["office_center"] },
|
|
31
|
+
"office_entry_c": { "x": 17, "z": 5, "neighbors": ["office_center"] },
|
|
32
|
+
"office_entry_r": { "x": 20, "z": 5, "neighbors": ["office_center"] },
|
|
33
|
+
"office_center": { "x": 17, "z": 12, "neighbors": ["office_entry_l", "office_entry_c", "office_entry_r"] }
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
@@ -0,0 +1,19 @@
|
|
|
1
|
+
import config from './route-config.json'
|
|
2
|
+
|
|
3
|
+
export type RouteSceneId = 'town' | 'office'
|
|
4
|
+
|
|
5
|
+
export type RouteNode = {
|
|
6
|
+
x: number
|
|
7
|
+
z: number
|
|
8
|
+
neighbors: string[]
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
type RouteConfigSchema = {
|
|
12
|
+
citizenDestinationPoints: Array<{ id: string; x: number; z: number }>
|
|
13
|
+
routeGraphs: Record<RouteSceneId, Record<string, RouteNode>>
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
const typed = config as RouteConfigSchema
|
|
17
|
+
|
|
18
|
+
export const CITIZEN_DESTINATION_POINTS = typed.citizenDestinationPoints
|
|
19
|
+
export const ROUTE_GRAPHS = typed.routeGraphs
|