2023-06-30 17:10:48 +08:00
|
|
|
#!/usr/bin/env python
|
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
"""
|
|
|
|
|
@Time : 2023/5/11 22:12
|
|
|
|
|
@Author : alexanderwu
|
|
|
|
|
@File : environment.py
|
|
|
|
|
"""
|
|
|
|
|
import asyncio
|
|
|
|
|
from typing import Iterable
|
2023-11-27 15:36:50 +08:00
|
|
|
from pathlib import Path
|
2023-06-30 17:10:48 +08:00
|
|
|
|
2023-07-22 11:55:57 +08:00
|
|
|
from pydantic import BaseModel, Field
|
2023-07-12 15:46:51 +08:00
|
|
|
|
2023-11-20 11:24:46 +08:00
|
|
|
# from metagpt.document import Document
|
2023-11-27 15:36:50 +08:00
|
|
|
from metagpt.logs import logger
|
2023-11-20 11:24:46 +08:00
|
|
|
from metagpt.document import Repo
|
2023-07-22 11:28:22 +08:00
|
|
|
from metagpt.memory import Memory
|
2023-06-30 17:10:48 +08:00
|
|
|
from metagpt.roles import Role
|
|
|
|
|
from metagpt.schema import Message
|
|
|
|
|
|
|
|
|
|
|
2023-07-12 15:46:51 +08:00
|
|
|
class Environment(BaseModel):
|
2023-11-20 11:24:46 +08:00
|
|
|
"""
|
|
|
|
|
Environment, hosting a batch of roles, roles can publish messages to the environment, and can be observed by other roles
|
2023-08-08 11:54:31 +01:00
|
|
|
"""
|
2023-07-12 15:46:51 +08:00
|
|
|
|
|
|
|
|
roles: dict[str, Role] = Field(default_factory=dict)
|
|
|
|
|
memory: Memory = Field(default_factory=Memory)
|
|
|
|
|
history: str = Field(default='')
|
2023-11-20 11:24:46 +08:00
|
|
|
repo: Repo = Field(default_factory=Repo)
|
2023-11-27 15:36:50 +08:00
|
|
|
kv: dict = Field(default_factory=dict)
|
2023-07-12 15:46:51 +08:00
|
|
|
|
|
|
|
|
class Config:
|
|
|
|
|
arbitrary_types_allowed = True
|
2023-06-30 17:10:48 +08:00
|
|
|
|
|
|
|
|
def add_role(self, role: Role):
|
2023-08-08 11:54:31 +01:00
|
|
|
"""增加一个在当前环境的角色
|
|
|
|
|
Add a role in the current environment
|
|
|
|
|
"""
|
2023-06-30 17:10:48 +08:00
|
|
|
role.set_env(self)
|
|
|
|
|
self.roles[role.profile] = role
|
|
|
|
|
|
|
|
|
|
def add_roles(self, roles: Iterable[Role]):
|
2023-08-08 11:54:31 +01:00
|
|
|
"""增加一批在当前环境的角色
|
|
|
|
|
Add a batch of characters in the current environment
|
|
|
|
|
"""
|
2023-06-30 17:10:48 +08:00
|
|
|
for role in roles:
|
|
|
|
|
self.add_role(role)
|
|
|
|
|
|
|
|
|
|
def publish_message(self, message: Message):
|
2023-08-08 11:54:31 +01:00
|
|
|
"""向当前环境发布信息
|
|
|
|
|
Post information to the current environment
|
|
|
|
|
"""
|
2023-07-07 10:30:53 +08:00
|
|
|
# self.message_queue.put(message)
|
2023-06-30 17:10:48 +08:00
|
|
|
self.memory.add(message)
|
|
|
|
|
self.history += f"\n{message}"
|
|
|
|
|
|
2023-11-27 15:36:50 +08:00
|
|
|
def set_doc(self, content: str, filename: str):
|
2023-11-20 11:24:46 +08:00
|
|
|
"""向当前环境发布文档(包括代码)"""
|
2023-11-27 15:36:50 +08:00
|
|
|
return self.repo.set(content, filename)
|
|
|
|
|
|
|
|
|
|
def get_doc(self, filename: str):
|
|
|
|
|
return self.repo.get(filename)
|
|
|
|
|
|
|
|
|
|
def set(self, k: str, v: str):
|
|
|
|
|
self.kv[k] = v
|
|
|
|
|
|
|
|
|
|
def get(self, k: str):
|
|
|
|
|
return self.kv.get(k, None)
|
|
|
|
|
|
|
|
|
|
def load_existing_repo(self, path: Path, inc: bool):
|
|
|
|
|
self.repo = Repo.from_path(path)
|
|
|
|
|
logger.info(self.repo.eda())
|
|
|
|
|
|
|
|
|
|
# Incremental mode: publish all docs to messages. Then roles can read the docs.
|
|
|
|
|
if inc:
|
|
|
|
|
docs = self.repo.get_text_documents()
|
|
|
|
|
for doc in docs:
|
|
|
|
|
msg = Message(content=doc.content)
|
|
|
|
|
self.publish_message(msg)
|
|
|
|
|
logger.info(f"Message from existing doc {doc.path}: {msg}")
|
|
|
|
|
logger.info(f"Load {len(docs)} docs from existing repo.")
|
|
|
|
|
raise NotImplementedError
|
2023-11-20 11:24:46 +08:00
|
|
|
|
2023-06-30 17:10:48 +08:00
|
|
|
async def run(self, k=1):
|
2023-08-08 11:54:31 +01:00
|
|
|
"""处理一次所有信息的运行
|
|
|
|
|
Process all Role runs at once
|
|
|
|
|
"""
|
2023-06-30 17:10:48 +08:00
|
|
|
# while not self.message_queue.empty():
|
|
|
|
|
# message = self.message_queue.get()
|
|
|
|
|
# rsp = await self.manager.handle(message, self)
|
|
|
|
|
# self.message_queue.put(rsp)
|
|
|
|
|
for _ in range(k):
|
|
|
|
|
futures = []
|
|
|
|
|
for role in self.roles.values():
|
|
|
|
|
future = role.run()
|
|
|
|
|
futures.append(future)
|
|
|
|
|
|
|
|
|
|
await asyncio.gather(*futures)
|
|
|
|
|
|
|
|
|
|
def get_roles(self) -> dict[str, Role]:
|
2023-08-08 11:47:17 +01:00
|
|
|
"""获得环境内的所有角色
|
|
|
|
|
Process all Role runs at once
|
|
|
|
|
"""
|
2023-06-30 17:10:48 +08:00
|
|
|
return self.roles
|
|
|
|
|
|
|
|
|
|
def get_role(self, name: str) -> Role:
|
2023-08-08 11:47:17 +01:00
|
|
|
"""获得环境内的指定角色
|
|
|
|
|
get all the environment roles
|
|
|
|
|
"""
|
2023-06-30 17:10:48 +08:00
|
|
|
return self.roles.get(name, None)
|