import re import json import autogen from autogen.cache import Cache # from finrobot.utils import create_inner_assistant from functools import partial from textwrap import dedent llm_config = { "config_list": autogen.config_list_from_json( "../OAI_CONFIG_LIST", filter_dict={ "model": ["gpt-4-0125-preview"], }, ), "cache_seed": 42, "temperature": 0, } leader_system_message = dedent( """ You are the leader of the following group members: {group_desc} As a group leader, you are responsible for coordinating the team's efforts to achieve the project's objectives. You must ensure that the team is working together effectively and efficiently. Summarize the status of the whole project progess every time you respond, and assign task to one of the group members to progress the project. Orders should follow the format: \"[] \" and appear at the end of your response. After receiving feedback from the team members, check the progress of the task, and make sure the task is well completed before proceding to th next order. Reply "TERMINATE" in the end when everything is done. """ ) role_system_message = dedent( """ As a {title}, your reponsibilities are as follows: {responsibilities} Reply "TERMINATE" in the end when everything is done. """ ) order_template = dedent( """ Follow leader's order and complete the following task with your group members: {order} For coding tasks, provide python scripts and executor will run it for you. Save your results or any intermediate data locally and let group leader know how to read them. DO NOT include "TERMINATE" in your response until you have received the results from the execution of the Python scripts. If the task cannot be done currently or need assistance from other members, report the reasons or requirements to group leader ended with TERMINATE. """ ) def order_trigger(sender, name, pattern): # print(pattern) # print(sender.name) return sender.name == name and pattern in sender.last_message()["content"] def order_message(pattern, recipient, messages, sender, config): full_order = recipient.chat_messages_for_summary(sender)[-1]["content"] pattern = rf"\[{pattern}\](?::)?\s*(.+?)(?=\n\[|$)" match = re.search(pattern, full_order, re.DOTALL) if match: order = match.group(1).strip() else: order = full_order return order_template.format(order=order) group_config = json.load(open("investment_group.json")) user_proxy = autogen.UserProxyAgent( name="User", # human_input_mode="ALWAYS", human_input_mode="NEVER", is_termination_msg=lambda x: x.get("content", "") and "TERMINATE" in x.get("content", ""), code_execution_config={ "last_n_messages": 3, "work_dir": "quant", "use_docker": False, }, ) def concat(responsiblities): return "\n".join([f" - {r}" for r in responsiblities]) with_leader_config = { "Market Sentiment Analysts": True, "Risk Assessment Analysts": True, "Fundamental Analysts": True, } representatives = [] for group_name, single_group_config in group_config["groups"].items(): with_leader = with_leader_config.get(group_name) if with_leader: group_members = single_group_config["with_leader"] leader_config = group_members["leader"] else: group_members = single_group_config["without_leader"] leader_config = None employees = group_members["employees"] need_suffix = len(set([c["title"] for c in employees])) == 1 employee_agents = {} for i, c in enumerate(employees): name = c["title"] + (f"_{i+1}" if need_suffix else "") employee_agents[name] = autogen.AssistantAgent( name=name, system_message=role_system_message.format( title=c["title"], responsibilities=concat(c["responsibilities"]) ), llm_config=llm_config, ) if leader_config: group_descs = "\n\n".join( [ "Name: {}\nResponsibility:\n{}".format( c["title"] + (f"_{i+1}" if need_suffix else ""), concat(c["responsibilities"]), ) for i, c in enumerate(employees) ] ) role_prompt = role_system_message.format( title=leader_config["title"], responsibilities=concat(leader_config["responsibilities"]), ) leader_prompt = leader_system_message.format(group_desc=group_descs) leader = autogen.AssistantAgent( name=leader_config["title"], system_message=role_prompt + "\n\n" + leader_prompt, llm_config=llm_config, ) for name, agent in employee_agents.items(): user_proxy.register_nested_chats( [ { "sender": user_proxy, "recipient": agent, "message": partial(order_message, name), "summary_method": "reflection_with_llm", "max_turns": 10, "max_consecutive_auto_reply": 3, } ], trigger=partial(order_trigger, name=leader.name, pattern=f"[{name}]"), ) representatives.append( { "name": leader_config["title"], "responsibilities": concat(single_group_config["responsibilities"]), "agent": leader, } ) else: group_chat = autogen.GroupChat( list(employee_agents.values()) + [user_proxy], messages=[], speaker_selection_method="round_robin", ) manager = autogen.GroupChatManager( group_chat, name=group_name, llm_config=llm_config ) representatives.append( { "name": group_name, "responsibilities": concat(single_group_config["responsibilities"]), "agent": manager, } ) cio_config = group_config["CIO"] role_prompt = role_system_message.format( title=cio_config["role"], responsibilities=concat(cio_config["responsibilities"]) ) group_descs = "\n\n".join( [ "Name: {}\nResponsibility:\n{}".format( c["name"], c["responsibilities"], ) for c in representatives ] ) leader_prompt = leader_system_message.format(group_desc=group_descs) cio = autogen.AssistantAgent( name=cio_config["role"], system_message=role_prompt + "\n\n" + leader_prompt, llm_config=llm_config, ) for agent_config in representatives: name = agent_config["name"] user_proxy.register_nested_chats( [ { "sender": user_proxy, "recipient": agent_config["agent"], "message": partial(order_message, name), "summary_method": "reflection_with_llm", "max_turns": 10, "max_consecutive_auto_reply": 3, } ], trigger=partial(order_trigger, name=cio.name, pattern=f"[{name}]"), ) task = dedent( """ As the Chief Investment Officer, your task is to optimize our current investment portfolio based on the latest annual reports. This will involve coordinating with the Market Sentiment Analysts, Risk Assessment Analysts, and Fundamental Analysts to gather and analyze the relevant data. The goal is to ensure our portfolio is well-positioned for growth while managing risks effectively. Specific Steps: [Review New Annual Reports]: Collect the latest annual reports from companies within our portfolio and potential new investments. [Coordinate with Market Sentiment Analysts]: Task: Analyze the market sentiment for these companies based on recent trends and news. Deliverable: Provide a comprehensive report on market sentiment and investor behavior for each company. [Coordinate with Risk Assessment Analysts]: Task: Evaluate the risks associated with each company and their sectors. Deliverable: Develop a risk assessment report highlighting potential risks and proposed mitigation strategies. [Coordinate with Fundamental Analysts]: Task: Perform a detailed financial analysis of each company, including key financial metrics, trends, and forecasts. Deliverable: Provide a summary of financial health and growth prospects for each company. [Integrate Findings]: Combine insights from all three groups to get a holistic view of each company’s potential. Evaluate how these insights impact our current portfolio and identify any necessary adjustments. [Optimize Portfolio]: Based on the integrated analysis, recommend adjustments to the portfolio. Ensure the portfolio is balanced, with an optimal mix of high-growth, stable, and low-risk investments. [Report and Implement]: Present your final recommendations and the rationale behind them. Oversee the implementation of the approved portfolio adjustments. [Expected Outcome]: A revised portfolio that leverages the latest financial insights and market sentiment to maximize growth and manage risks effectively. If you have any questions or need additional resources, please let me know. """ ) task = dedent( """ As the Chief Investment Officer, your task is to evaluate the potential investment in Company ABC based on the provided data. You will need to coordinate with the Market Sentiment Analysts, Risk Assessment Analysts, and Fundamental Analysts to gather and analyze the relevant information. Your final deliverable should include a comprehensive evaluation and a recommendation on whether to invest in Company ABC. Specific Instructions: Coordinate with Market Sentiment Analysts: Task: Calculate the sentiment score based on the provided market sentiment data. Data: Positive mentions (80), Negative mentions (20) Formula: Sentiment Score = (Positive Mentions - Negative Mentions) / Total Mentions Expected Output: Sentiment Score (percentage) Coordinate with Risk Assessment Analysts: Task: Calculate the risk score using the provided financial ratios. Data: Debt-to-Equity Ratio: 1.5 Current Ratio: 2.0 Return on Equity (ROE): 0.1 (10%) Weights: Debt-to-Equity (0.5), Current Ratio (0.3), ROE (0.2) Formula: Risk Score = 0.5 * Debt-to-Equity + 0.3 * (1 / Current Ratio) - 0.2 * ROE Expected Output: Risk Score Coordinate with Fundamental Analysts: Task: Calculate the Profit Margin and Return on Assets (ROA) based on the provided financial data. Data: Revenue: $1,000,000 Net Income: $100,000 Total Assets: $500,000 Formulas: Profit Margin = (Net Income / Revenue) * 100 ROA = (Net Income / Total Assets) * 100 Expected Outputs: Profit Margin (percentage) and ROA (percentage) Final Deliverable: Integrate Findings: Compile the insights from all three groups to get a holistic view of Company ABC's potential. Evaluation and Recommendation: Based on the integrated analysis, provide a recommendation on whether to invest in Company ABC, including the rationale behind your decision. """ ) with Cache.disk() as cache: user_proxy.initiate_chat(cio, message=task, cache=cache)