pycityagent 2.0.0a48__cp312-cp312-macosx_11_0_arm64.whl → 2.0.0a50__cp312-cp312-macosx_11_0_arm64.whl

Sign up to get free protection for your applications and to get access to all the features.
@@ -1,33 +1,29 @@
1
1
  from __future__ import annotations
2
+
2
3
  import asyncio
3
4
  import inspect
4
5
  import json
5
6
  import logging
6
- import random
7
7
  import uuid
8
8
  from abc import ABC, abstractmethod
9
- from copy import deepcopy
10
9
  from datetime import datetime, timezone
11
10
  from enum import Enum
12
- from typing import Any, List, Optional, Type, get_type_hints
13
- from uuid import UUID
11
+ from typing import Any, Optional, Union, get_type_hints
14
12
 
15
13
  import fastavro
16
- from pyparsing import Dict
17
14
  import ray
18
- from mosstool.util.format_converter import dict2pb
19
15
  from pycityproto.city.person.v2 import person_pb2 as person_pb2
16
+ from pyparsing import Dict
20
17
 
21
- from pycityagent.workflow import Block
22
-
23
- from .economy import EconomyClient
24
- from .environment import Simulator
25
- from .environment.sim.person_service import PersonService
26
- from .llm import LLM
27
- from .memory import Memory
28
- from .message.messager import Messager
29
- from .metrics import MlflowClient
30
- from .utils import DIALOG_SCHEMA, SURVEY_SCHEMA, process_survey_for_llm
18
+ from ..economy import EconomyClient
19
+ from ..environment import Simulator
20
+ from ..environment.sim.person_service import PersonService
21
+ from ..llm import LLM
22
+ from ..memory import Memory
23
+ from ..message.messager import Messager
24
+ from ..metrics import MlflowClient
25
+ from ..utils import DIALOG_SCHEMA, SURVEY_SCHEMA, process_survey_for_llm
26
+ from ..workflow import Block
31
27
 
32
28
  logger = logging.getLogger("pycityagent")
33
29
 
@@ -49,7 +45,8 @@ class Agent(ABC):
49
45
  """
50
46
  Agent base class
51
47
  """
52
- configurable_fields: List[str] = []
48
+
49
+ configurable_fields: list[str] = []
53
50
  default_values: dict[str, Any] = {}
54
51
 
55
52
  def __init__(
@@ -107,12 +104,8 @@ class Agent(ABC):
107
104
  return state
108
105
 
109
106
  @classmethod
110
- def export_class_config(cls) -> Dict[str, Dict]:
111
- result = {
112
- "agent_name": cls.__name__,
113
- "config": {},
114
- "blocks": []
115
- }
107
+ def export_class_config(cls) -> dict[str, Dict]:
108
+ result = {"agent_name": cls.__name__, "config": {}, "blocks": []}
116
109
  config = {
117
110
  field: cls.default_values.get(field, "default_value")
118
111
  for field in cls.configurable_fields
@@ -123,25 +116,29 @@ class Agent(ABC):
123
116
  for attr_name, attr_type in hints.items():
124
117
  if inspect.isclass(attr_type) and issubclass(attr_type, Block):
125
118
  block_config = attr_type.export_class_config()
126
- result["blocks"].append({
127
- "name": attr_name,
128
- "config": block_config,
129
- "children": cls._export_subblocks(attr_type)
130
- })
119
+ result["blocks"].append(
120
+ {
121
+ "name": attr_name,
122
+ "config": block_config,
123
+ "children": cls._export_subblocks(attr_type),
124
+ }
125
+ )
131
126
  return result
132
127
 
133
128
  @classmethod
134
- def _export_subblocks(cls, block_cls: Type[Block]) -> List[Dict]:
129
+ def _export_subblocks(cls, block_cls: type[Block]) -> list[Dict]:
135
130
  children = []
136
131
  hints = get_type_hints(block_cls) # 获取类的注解
137
132
  for attr_name, attr_type in hints.items():
138
133
  if inspect.isclass(attr_type) and issubclass(attr_type, Block):
139
134
  block_config = attr_type.export_class_config()
140
- children.append({
141
- "name": attr_name,
142
- "config": block_config,
143
- "children": cls._export_subblocks(attr_type)
144
- })
135
+ children.append(
136
+ {
137
+ "name": attr_name,
138
+ "config": block_config,
139
+ "children": cls._export_subblocks(attr_type),
140
+ }
141
+ )
145
142
  return children
146
143
 
147
144
  @classmethod
@@ -151,28 +148,29 @@ class Agent(ABC):
151
148
  json.dump(config, f, indent=4)
152
149
 
153
150
  @classmethod
154
- def import_block_config(cls, config: Dict[str, List[Dict]]) -> "Agent":
155
- agent = cls(name=config["agent_name"])
151
+ def import_block_config(cls, config: dict[str, Union[list[dict], str]]) -> Agent:
152
+ agent = cls(name=config["agent_name"]) # type:ignore
156
153
 
157
- def build_block(block_data: Dict) -> Block:
154
+ def build_block(block_data: dict[str, Any]) -> Block:
158
155
  block_cls = globals()[block_data["name"]]
159
156
  block_instance = block_cls.import_config(block_data)
160
157
  return block_instance
161
158
 
162
159
  # 创建顶层Block
163
160
  for block_data in config["blocks"]:
161
+ assert isinstance(block_data, dict)
164
162
  block = build_block(block_data)
165
163
  setattr(agent, block.name.lower(), block)
166
164
 
167
165
  return agent
168
166
 
169
167
  @classmethod
170
- def import_from_file(cls, filepath: str) -> "Agent":
168
+ def import_from_file(cls, filepath: str) -> Agent:
171
169
  with open(filepath, "r") as f:
172
170
  config = json.load(f)
173
171
  return cls.import_block_config(config)
174
-
175
- def load_from_config(self, config: Dict[str, List[Dict]]) -> None:
172
+
173
+ def load_from_config(self, config: dict[str, list[dict]]) -> None:
176
174
  """
177
175
  使用配置更新当前Agent实例的Block层次结构。
178
176
  """
@@ -185,13 +183,15 @@ class Agent(ABC):
185
183
  # 递归更新或创建顶层Block
186
184
  for block_data in config.get("blocks", []):
187
185
  block_name = block_data["name"]
188
- existing_block = getattr(self, block_name, None)
186
+ existing_block = getattr(self, block_name, None) # type:ignore
189
187
 
190
188
  if existing_block:
191
189
  # 如果Block已经存在,则递归更新
192
190
  existing_block.load_from_config(block_data)
193
191
  else:
194
- raise KeyError(f"Block '{block_name}' not found in agent '{self.__class__.__name__}'")
192
+ raise KeyError(
193
+ f"Block '{block_name}' not found in agent '{self.__class__.__name__}'"
194
+ )
195
195
 
196
196
  def load_from_file(self, filepath: str) -> None:
197
197
  with open(filepath, "r") as f:
@@ -315,7 +315,7 @@ class Agent(ABC):
315
315
  f"Copy Writer access before assignment, please `set_pgsql_writer` first!"
316
316
  )
317
317
  return self._pgsql_writer
318
-
318
+
319
319
  async def messager_ping(self):
320
320
  if self._messager is None:
321
321
  raise RuntimeError("Messager is not set")
@@ -633,307 +633,3 @@ class Agent(ABC):
633
633
  await self._messager.ping.remote()
634
634
  if not self._blocked:
635
635
  await self.forward()
636
-
637
-
638
- class CitizenAgent(Agent):
639
- """
640
- CitizenAgent: 城市居民智能体类及其定义
641
- """
642
-
643
- def __init__(
644
- self,
645
- name: str,
646
- llm_client: Optional[LLM] = None,
647
- simulator: Optional[Simulator] = None,
648
- mlflow_client: Optional[MlflowClient] = None,
649
- memory: Optional[Memory] = None,
650
- economy_client: Optional[EconomyClient] = None,
651
- messager: Optional[Messager] = None, # type:ignore
652
- avro_file: Optional[dict] = None,
653
- ) -> None:
654
- super().__init__(
655
- name=name,
656
- type=AgentType.Citizen,
657
- llm_client=llm_client,
658
- economy_client=economy_client,
659
- messager=messager,
660
- simulator=simulator,
661
- mlflow_client=mlflow_client,
662
- memory=memory,
663
- avro_file=avro_file,
664
- )
665
-
666
- async def bind_to_simulator(self):
667
- await self._bind_to_simulator()
668
- await self._bind_to_economy()
669
-
670
- async def _bind_to_simulator(self):
671
- """
672
- Bind Agent to Simulator
673
-
674
- Args:
675
- person_template (dict, optional): The person template in dict format. Defaults to PersonService.default_dict_person().
676
- """
677
- if self._simulator is None:
678
- logger.warning("Simulator is not set")
679
- return
680
- if not self._has_bound_to_simulator:
681
- FROM_MEMORY_KEYS = {
682
- "attribute",
683
- "home",
684
- "work",
685
- "vehicle_attribute",
686
- "bus_attribute",
687
- "pedestrian_attribute",
688
- "bike_attribute",
689
- }
690
- simulator = self.simulator
691
- memory = self.memory
692
- person_id = await memory.get("id")
693
- # ATTENTION:模拟器分配的id从0开始
694
- if person_id >= 0:
695
- await simulator.get_person(person_id)
696
- logger.debug(f"Binding to Person `{person_id}` already in Simulator")
697
- else:
698
- dict_person = deepcopy(self._person_template)
699
- for _key in FROM_MEMORY_KEYS:
700
- try:
701
- _value = await memory.get(_key)
702
- if _value:
703
- dict_person[_key] = _value
704
- except KeyError as e:
705
- continue
706
- resp = await simulator.add_person(
707
- dict2pb(dict_person, person_pb2.Person())
708
- )
709
- person_id = resp["person_id"]
710
- await memory.update("id", person_id, protect_llm_read_only_fields=False)
711
- logger.debug(f"Binding to Person `{person_id}` just added to Simulator")
712
- # 防止模拟器还没有到prepare阶段导致get_person出错
713
- self._has_bound_to_simulator = True
714
- self._agent_id = person_id
715
- self.memory.set_agent_id(person_id)
716
-
717
- async def _bind_to_economy(self):
718
- if self._economy_client is None:
719
- logger.warning("Economy client is not set")
720
- return
721
- if not self._has_bound_to_economy:
722
- if self._has_bound_to_simulator:
723
- try:
724
- await self._economy_client.remove_agents([self._agent_id])
725
- except:
726
- pass
727
- person_id = await self.memory.get("id")
728
- currency = await self.memory.get("currency")
729
- await self._economy_client.add_agents(
730
- {
731
- "id": person_id,
732
- "currency": currency,
733
- }
734
- )
735
- self._has_bound_to_economy = True
736
- else:
737
- logger.debug(
738
- f"Binding to Economy before binding to Simulator, skip binding to Economy Simulator"
739
- )
740
-
741
- async def handle_gather_message(self, payload: dict):
742
- """处理收到的消息,识别发送者"""
743
- # 从消息中解析发送者 ID 和消息内容
744
- target = payload["target"]
745
- sender_id = payload["from"]
746
- content = await self.memory.get(f"{target}")
747
- payload = {
748
- "from": self._uuid,
749
- "content": content,
750
- }
751
- await self._send_message(sender_id, payload, "gather")
752
-
753
-
754
- class InstitutionAgent(Agent):
755
- """
756
- InstitutionAgent: 机构智能体类及其定义
757
- """
758
-
759
- def __init__(
760
- self,
761
- name: str,
762
- llm_client: Optional[LLM] = None,
763
- simulator: Optional[Simulator] = None,
764
- mlflow_client: Optional[MlflowClient] = None,
765
- memory: Optional[Memory] = None,
766
- economy_client: Optional[EconomyClient] = None,
767
- messager: Optional[Messager] = None, # type:ignore
768
- avro_file: Optional[dict] = None,
769
- ) -> None:
770
- super().__init__(
771
- name=name,
772
- type=AgentType.Institution,
773
- llm_client=llm_client,
774
- economy_client=economy_client,
775
- mlflow_client=mlflow_client,
776
- messager=messager,
777
- simulator=simulator,
778
- memory=memory,
779
- avro_file=avro_file,
780
- )
781
- # 添加响应收集器
782
- self._gather_responses: dict[str, asyncio.Future] = {}
783
-
784
- async def bind_to_simulator(self):
785
- await self._bind_to_economy()
786
-
787
- async def _bind_to_economy(self):
788
- print("Debug:", self._economy_client, self._has_bound_to_economy)
789
- if self._economy_client is None:
790
- logger.debug("Economy client is not set")
791
- return
792
- if not self._has_bound_to_economy:
793
- # TODO: More general id generation
794
- _id = random.randint(100000, 999999)
795
- self._agent_id = _id
796
- self.memory.set_agent_id(_id)
797
- map_header = self.simulator.map.header
798
- # TODO: remove random position assignment
799
- await self.memory.update(
800
- "position",
801
- {
802
- "xy_position": {
803
- "x": float(
804
- random.randrange(
805
- start=int(map_header["west"]),
806
- stop=int(map_header["east"]),
807
- )
808
- ),
809
- "y": float(
810
- random.randrange(
811
- start=int(map_header["south"]),
812
- stop=int(map_header["north"]),
813
- )
814
- ),
815
- }
816
- },
817
- protect_llm_read_only_fields=False,
818
- )
819
- await self.memory.update("id", _id, protect_llm_read_only_fields=False)
820
- try:
821
- await self._economy_client.remove_orgs([self._agent_id])
822
- except:
823
- pass
824
- try:
825
- _memory = self.memory
826
- _id = await _memory.get("id")
827
- _type = await _memory.get("type")
828
- try:
829
- nominal_gdp = await _memory.get("nominal_gdp")
830
- except:
831
- nominal_gdp = []
832
- try:
833
- real_gdp = await _memory.get("real_gdp")
834
- except:
835
- real_gdp = []
836
- try:
837
- unemployment = await _memory.get("unemployment")
838
- except:
839
- unemployment = []
840
- try:
841
- wages = await _memory.get("wages")
842
- except:
843
- wages = []
844
- try:
845
- prices = await _memory.get("prices")
846
- except:
847
- prices = []
848
- try:
849
- inventory = await _memory.get("inventory")
850
- except:
851
- inventory = 0
852
- try:
853
- price = await _memory.get("price")
854
- except:
855
- price = 0
856
- try:
857
- currency = await _memory.get("currency")
858
- except:
859
- currency = 0.0
860
- try:
861
- interest_rate = await _memory.get("interest_rate")
862
- except:
863
- interest_rate = 0.0
864
- try:
865
- bracket_cutoffs = await _memory.get("bracket_cutoffs")
866
- except:
867
- bracket_cutoffs = []
868
- try:
869
- bracket_rates = await _memory.get("bracket_rates")
870
- except:
871
- bracket_rates = []
872
- await self._economy_client.add_orgs(
873
- {
874
- "id": _id,
875
- "type": _type,
876
- "nominal_gdp": nominal_gdp,
877
- "real_gdp": real_gdp,
878
- "unemployment": unemployment,
879
- "wages": wages,
880
- "prices": prices,
881
- "inventory": inventory,
882
- "price": price,
883
- "currency": currency,
884
- "interest_rate": interest_rate,
885
- "bracket_cutoffs": bracket_cutoffs,
886
- "bracket_rates": bracket_rates,
887
- }
888
- )
889
- except Exception as e:
890
- logger.error(f"Failed to bind to Economy: {e}")
891
- self._has_bound_to_economy = True
892
-
893
- async def handle_gather_message(self, payload: dict):
894
- """处理收到的消息,识别发送者"""
895
- content = payload["content"]
896
- sender_id = payload["from"]
897
-
898
- # 将响应存储到对应的Future中
899
- response_key = str(sender_id)
900
- if response_key in self._gather_responses:
901
- self._gather_responses[response_key].set_result(
902
- {
903
- "from": sender_id,
904
- "content": content,
905
- }
906
- )
907
-
908
- async def gather_messages(self, agent_uuids: list[str], target: str) -> list[dict]:
909
- """从多个智能体收集消息
910
-
911
- Args:
912
- agent_uuids: 目标智能体UUID列表
913
- target: 要收集的信息类型
914
-
915
- Returns:
916
- list[dict]: 收集到的所有响应
917
- """
918
- # 为每个agent创建Future
919
- futures = {}
920
- for agent_uuid in agent_uuids:
921
- futures[agent_uuid] = asyncio.Future()
922
- self._gather_responses[agent_uuid] = futures[agent_uuid]
923
-
924
- # 发送gather请求
925
- payload = {
926
- "from": self._uuid,
927
- "target": target,
928
- }
929
- for agent_uuid in agent_uuids:
930
- await self._send_message(agent_uuid, payload, "gather")
931
-
932
- try:
933
- # 等待所有响应
934
- responses = await asyncio.gather(*futures.values())
935
- return responses
936
- finally:
937
- # 清理Future
938
- for key in futures:
939
- self._gather_responses.pop(key, None)
@@ -11,22 +11,32 @@ import logging
11
11
 
12
12
  logger = logging.getLogger("pycityagent")
13
13
 
14
- class BankAgent(InstitutionAgent):
15
- def __init__(self,
16
- name: str,
17
- llm_client: Optional[LLM] = None,
18
- simulator: Optional[Simulator] = None,
19
- memory: Optional[Memory] = None,
20
- economy_client: Optional[EconomyClient] = None,
21
- messager: Optional[Messager] = None,
22
- avro_file: Optional[dict] = None,
23
- ) -> None:
24
- super().__init__(name=name, llm_client=llm_client, simulator=simulator, memory=memory, economy_client=economy_client, messager=messager, avro_file=avro_file)
14
+
15
+ class BankAgent(InstitutionAgent):
16
+ def __init__(
17
+ self,
18
+ name: str,
19
+ llm_client: Optional[LLM] = None,
20
+ simulator: Optional[Simulator] = None,
21
+ memory: Optional[Memory] = None,
22
+ economy_client: Optional[EconomyClient] = None,
23
+ messager: Optional[Messager] = None,
24
+ avro_file: Optional[dict] = None,
25
+ ) -> None:
26
+ super().__init__(
27
+ name=name,
28
+ llm_client=llm_client,
29
+ simulator=simulator,
30
+ memory=memory,
31
+ economy_client=economy_client,
32
+ messager=messager,
33
+ avro_file=avro_file,
34
+ )
25
35
  self.initailzed = False
26
36
  self.last_time_trigger = None
27
37
  self.time_diff = 30 * 24 * 60 * 60
28
38
  self.forward_times = 0
29
-
39
+
30
40
  async def month_trigger(self):
31
41
  now_time = await self.simulator.get_time()
32
42
  if self.last_time_trigger is None:
@@ -36,19 +46,21 @@ class BankAgent(InstitutionAgent):
36
46
  self.last_time_trigger = now_time
37
47
  return True
38
48
  return False
39
-
49
+
40
50
  async def gather_messages(self, agent_ids, content):
41
51
  infos = await super().gather_messages(agent_ids, content)
42
- return [info['content'] for info in infos]
52
+ return [info["content"] for info in infos]
43
53
 
44
54
  async def forward(self):
45
55
  if await self.month_trigger():
46
56
  citizens = await self.memory.get("citizens")
47
57
  while True:
48
- agents_forward = await self.gather_messages(citizens, 'forward')
58
+ agents_forward = await self.gather_messages(citizens, "forward")
49
59
  if np.all(np.array(agents_forward) > self.forward_times):
50
60
  break
51
61
  await asyncio.sleep(1)
52
62
  self.forward_times += 1
53
63
  for uuid in citizens:
54
- await self.send_message_to_agent(uuid, f"bank_forward@{self.forward_times}")
64
+ await self.send_message_to_agent(
65
+ uuid, f"bank_forward@{self.forward_times}"
66
+ )
@@ -11,7 +11,8 @@ import logging
11
11
 
12
12
  logger = logging.getLogger("pycityagent")
13
13
 
14
- class FirmAgent(InstitutionAgent):
14
+
15
+ class FirmAgent(InstitutionAgent):
15
16
  configurable_fields = ["time_diff", "max_price_inflation", "max_wage_inflation"]
16
17
  default_values = {
17
18
  "time_diff": 30 * 24 * 60 * 60,
@@ -19,16 +20,25 @@ class FirmAgent(InstitutionAgent):
19
20
  "max_wage_inflation": 0.05,
20
21
  }
21
22
 
22
- def __init__(self,
23
- name: str,
24
- llm_client: Optional[LLM] = None,
25
- simulator: Optional[Simulator] = None,
26
- memory: Optional[Memory] = None,
27
- economy_client: Optional[EconomyClient] = None,
28
- messager: Optional[Messager] = None,
29
- avro_file: Optional[dict] = None,
30
- ) -> None:
31
- super().__init__(name=name, llm_client=llm_client, simulator=simulator, memory=memory, economy_client=economy_client, messager=messager, avro_file=avro_file)
23
+ def __init__(
24
+ self,
25
+ name: str,
26
+ llm_client: Optional[LLM] = None,
27
+ simulator: Optional[Simulator] = None,
28
+ memory: Optional[Memory] = None,
29
+ economy_client: Optional[EconomyClient] = None,
30
+ messager: Optional[Messager] = None,
31
+ avro_file: Optional[dict] = None,
32
+ ) -> None:
33
+ super().__init__(
34
+ name=name,
35
+ llm_client=llm_client,
36
+ simulator=simulator,
37
+ memory=memory,
38
+ economy_client=economy_client,
39
+ messager=messager,
40
+ avro_file=avro_file,
41
+ )
32
42
  self.initailzed = False
33
43
  self.last_time_trigger = None
34
44
  self.forward_times = 0
@@ -45,31 +55,59 @@ class FirmAgent(InstitutionAgent):
45
55
  self.last_time_trigger = now_time
46
56
  return True
47
57
  return False
48
-
58
+
49
59
  async def gather_messages(self, agent_ids, content):
50
60
  infos = await super().gather_messages(agent_ids, content)
51
- return [info['content'] for info in infos]
61
+ return [info["content"] for info in infos]
52
62
 
53
63
  async def forward(self):
54
64
  if await self.month_trigger():
55
65
  employees = await self.memory.get("employees")
56
66
  while True:
57
- agents_forward = await self.gather_messages(employees, 'forward')
67
+ agents_forward = await self.gather_messages(employees, "forward")
58
68
  if np.all(np.array(agents_forward) > self.forward_times):
59
69
  break
60
70
  await asyncio.sleep(1)
61
- goods_demand = await self.gather_messages(employees, 'goods_demand')
62
- goods_consumption = await self.gather_messages(employees, 'goods_consumption')
63
- print(f'goods_demand: {goods_demand}, goods_consumption: {goods_consumption}')
71
+ goods_demand = await self.gather_messages(employees, "goods_demand")
72
+ goods_consumption = await self.gather_messages(
73
+ employees, "goods_consumption"
74
+ )
75
+ print(
76
+ f"goods_demand: {goods_demand}, goods_consumption: {goods_consumption}"
77
+ )
64
78
  total_demand = sum(goods_demand)
65
- last_inventory = sum(goods_consumption) + await self.economy_client.get(self._agent_id, "inventory")
66
- print(f'total_demand: {total_demand}, last_inventory: {last_inventory}, goods_contumption: {sum(goods_consumption)}')
67
- max_change_rate = (total_demand - last_inventory)/(max(total_demand, last_inventory)+1e-8)
68
- skills = await self.gather_messages(employees, 'work_skill')
79
+ last_inventory = sum(goods_consumption) + await self.economy_client.get(
80
+ self._agent_id, "inventory"
81
+ )
82
+ print(
83
+ f"total_demand: {total_demand}, last_inventory: {last_inventory}, goods_contumption: {sum(goods_consumption)}"
84
+ )
85
+ max_change_rate = (total_demand - last_inventory) / (
86
+ max(total_demand, last_inventory) + 1e-8
87
+ )
88
+ skills = await self.gather_messages(employees, "work_skill")
69
89
  for skill, uuid in zip(skills, employees):
70
- await self.send_message_to_agent(uuid, f"work_skill@{max(skill*(1 + np.random.uniform(0, max_change_rate*self.max_wage_inflation)), 1)}")
71
- price = await self.economy_client.get(self._agent_id, 'price')
72
- await self.economy_client.update(self._agent_id, 'price', max(price*(1 + np.random.uniform(0, max_change_rate*self.max_price_inflation)), 1))
90
+ await self.send_message_to_agent(
91
+ uuid,
92
+ f"work_skill@{max(skill*(1 + np.random.uniform(0, max_change_rate*self.max_wage_inflation)), 1)}",
93
+ )
94
+ price = await self.economy_client.get(self._agent_id, "price")
95
+ await self.economy_client.update(
96
+ self._agent_id,
97
+ "price",
98
+ max(
99
+ price
100
+ * (
101
+ 1
102
+ + np.random.uniform(
103
+ 0, max_change_rate * self.max_price_inflation
104
+ )
105
+ ),
106
+ 1,
107
+ ),
108
+ )
73
109
  self.forward_times += 1
74
110
  for uuid in employees:
75
- await self.send_message_to_agent(uuid, f"firm_forward@{self.forward_times}")
111
+ await self.send_message_to_agent(
112
+ uuid, f"firm_forward@{self.forward_times}"
113
+ )