From fa1e92714b25b80ebd6983be72e839783848ca06 Mon Sep 17 00:00:00 2001 From: Kye Gomez Date: Sun, 14 Jul 2024 16:25:28 -0700 Subject: [PATCH] [CLEANUP] --- docs/mkdocs.yml | 3 +- docs/swarms/framework/test.md | 244 ++++++++++++++++++ .../society_of_agents/accountant_team.py | 89 +++++++ .../society_of_agents/hallucination_swarm.py | 229 ++++++++++++++++ .../society_of_agents}/json_log_cleanup.py | 2 +- .../demos/society_of_agents/probate_agent.py | 166 ++++++++++++ .../demos/society_of_agents/receipt.pdf | Bin 0 -> 52825 bytes pyproject.toml | 9 +- swarms/structs/graph_workflow.py | 74 +++--- tests/structs/test_agent_rearrange.py | 174 +++++++++---- tests/structs/test_multi_agent_collab.py | 57 +++- 11 files changed, 945 insertions(+), 102 deletions(-) create mode 100644 docs/swarms/framework/test.md create mode 100644 playground/demos/society_of_agents/accountant_team.py create mode 100644 playground/demos/society_of_agents/hallucination_swarm.py rename {scripts/cleanup => playground/demos/society_of_agents}/json_log_cleanup.py (97%) create mode 100644 playground/demos/society_of_agents/probate_agent.py create mode 100644 playground/demos/society_of_agents/receipt.pdf diff --git a/docs/mkdocs.yml b/docs/mkdocs.yml index 95c03a5d..e9191e96 100644 --- a/docs/mkdocs.yml +++ b/docs/mkdocs.yml @@ -131,6 +131,8 @@ nav: - Swarm Ecosystem: "swarms/ecosystem.md" - Framework: - Overview: "swarms/framework/index.md" + - Contributing: + - Tests: "swarms/framework/test.md" - Models: - Overview: "swarms/models/index.md" - How to Create A Custom Language Model: "swarms/models/custom_model.md" @@ -146,7 +148,6 @@ nav: - Overview: "swarms/structs/multi_agent_orchestration.md" - Swarm Architectures: "swarms/concept/swarm_architectures.md" - Workflows: "swarms/structs/workflows.md" - # - Multi-Agent Architectures: "swarms/structs/multi_agent_architectures.md" - Reference: - Overview: "swarms/structs/index.md" - Models: diff --git a/docs/swarms/framework/test.md b/docs/swarms/framework/test.md new file mode 100644 index 00000000..9316d4b2 --- /dev/null +++ b/docs/swarms/framework/test.md @@ -0,0 +1,244 @@ +# How to Run Tests Using Pytest: A Comprehensive Guide + +In modern software development, automated testing is crucial for ensuring the reliability and functionality of your code. One of the most popular testing frameworks for Python is `pytest`. + +This blog will provide an in-depth look at how to run tests using `pytest`, including testing a single file, multiple files, every file in the test repository, and providing guidelines for contributors to run tests reliably. + +## What is Pytest? + +`pytest` is a testing framework for Python that makes it easy to write simple and scalable test cases. It supports fixtures, parameterized testing, and has a rich plugin architecture. `pytest` is widely used because of its ease of use and powerful features that help streamline the testing process. + +## Installation + +To get started with `pytest`, you need to install it. You can install `pytest` using `pip`: + +```bash +pip install pytest +``` + +## Writing Your First Test + +Before diving into running tests, let’s write a simple test. Create a file named `test_sample.py` with the following content: + +```python +def test_addition(): + assert 1 + 1 == 2 + +def test_subtraction(): + assert 2 - 1 == 1 +``` + +In this example, we have defined two basic tests: `test_addition` and `test_subtraction`. + +## Running Tests + +### Running a Single Test File + +To run a single test file, you can use the `pytest` command followed by the filename. For example, to run the tests in `test_sample.py`, use the following command: + +```bash +pytest test_sample.py +``` + +The output will show the test results, including the number of tests passed, failed, or skipped. + +### Running Multiple Test Files + +You can also run multiple test files by specifying their filenames separated by a space. For example: + +```bash +pytest test_sample.py test_another_sample.py +``` + +If you have multiple test files in a directory, you can run all of them by specifying the directory name: + +```bash +pytest tests/ +``` + +### Running All Tests in the Repository + +To run all tests in the repository, navigate to the root directory of your project and simply run: + +```bash +pytest +``` + +`pytest` will automatically discover and run all the test files that match the pattern `test_*.py` or `*_test.py`. + +### Test Discovery + +`pytest` automatically discovers test files and test functions based on their naming conventions. By default, it looks for files that match the pattern `test_*.py` or `*_test.py` and functions or methods that start with `test_`. + +### Using Markers + +`pytest` allows you to use markers to group tests or add metadata to them. Markers can be used to run specific subsets of tests. For example, you can mark a test as `slow` and then run only the slow tests or skip them. + +```python +import pytest + +@pytest.mark.slow +def test_long_running(): + import time + time.sleep(5) + assert True + +def test_fast(): + assert True +``` + +To run only the tests marked as `slow`, use the `-m` option: + +```bash +pytest -m slow +``` + +### Parameterized Tests + +`pytest` supports parameterized testing, which allows you to run a test with different sets of input data. This can be done using the `@pytest.mark.parametrize` decorator. + +```python +import pytest + +@pytest.mark.parametrize("a,b,expected", [ + (1, 2, 3), + (2, 3, 5), + (3, 5, 8), +]) +def test_add(a, b, expected): + assert a + b == expected +``` + +In this example, `test_add` will run three times with different sets of input data. + +### Fixtures + +Fixtures are a powerful feature of `pytest` that allow you to set up some context for your tests. They can be used to provide a fixed baseline upon which tests can reliably and repeatedly execute. + +```python +import pytest + +@pytest.fixture +def sample_data(): + return {"name": "John", "age": 30} + +def test_sample_data(sample_data): + assert sample_data["name"] == "John" + assert sample_data["age"] == 30 +``` + +Fixtures can be used to share setup and teardown code between tests. + +## Advanced Usage + +### Running Tests in Parallel + +`pytest` can run tests in parallel using the `pytest-xdist` plugin. To install `pytest-xdist`, run: + +```bash +pip install pytest-xdist +``` + +To run tests in parallel, use the `-n` option followed by the number of CPU cores you want to use: + +```bash +pytest -n 4 +``` + +### Generating Test Reports + +`pytest` can generate detailed test reports. You can use the `--html` option to generate an HTML report: + +```bash +pip install pytest-html +pytest --html=report.html +``` + +This command will generate a file named `report.html` with a detailed report of the test results. + +### Code Coverage + +You can use the `pytest-cov` plugin to measure code coverage. To install `pytest-cov`, run: + +```bash +pip install pytest-cov +``` + +To generate a coverage report, use the `--cov` option followed by the module name: + +```bash +pytest --cov=my_module +``` + +This command will show the coverage summary in the terminal. You can also generate an HTML report: + +```bash +pytest --cov=my_module --cov-report=html +``` + +The coverage report will be generated in the `htmlcov` directory. + +## Best Practices for Writing Tests + +1. **Write Clear and Concise Tests**: Each test should focus on a single piece of functionality. +2. **Use Descriptive Names**: Test function names should clearly describe what they are testing. +3. **Keep Tests Independent**: Tests should not depend on each other and should run in isolation. +4. **Use Fixtures**: Use fixtures to set up the context for your tests. +5. **Mock External Dependencies**: Use mocking to isolate the code under test from external dependencies. + +## Running Tests Reliably + +For contributors and team members, it’s important to run tests reliably to ensure consistent results. Here are some guidelines: + +1. **Set Up a Virtual Environment**: Use a virtual environment to manage dependencies and ensure a consistent testing environment. + + ```bash + python -m venv venv + source venv/bin/activate # On Windows use `venv\Scripts\activate` + ``` + +2. **Install Dependencies**: Install all required dependencies from the `requirements.txt` file. + + ```bash + pip install -r requirements.txt + ``` + +3. **Run Tests Before Pushing**: Ensure all tests pass before pushing code to the repository. + +4. **Use Continuous Integration (CI)**: Set up CI pipelines to automatically run tests on each commit or pull request. + +### Example CI Configuration (GitHub Actions) + +Here is an example of a GitHub Actions workflow to run tests using `pytest`: + +```yaml +name: Python package + +on: [push, pull_request] + +jobs: + build: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v2 + - name: Set up Python + uses: actions/setup-python@v2 + with: + python-version: '3.8' + - name: Install dependencies + run: | + python -m pip install --upgrade pip + pip install -r requirements.txt + - name: Run tests + run: | + pytest +``` + +This configuration will run the tests on every push and pull request, ensuring that your codebase remains stable. + +## Conclusion + +`pytest` is a powerful and flexible testing framework that makes it easy to write and run tests for your Python code. By following the guidelines and best practices outlined in this blog, you can ensure that your tests are reliable and your codebase is robust. Whether you are testing a single file, multiple files, or the entire repository, `pytest` provides the tools you need to automate and streamline your testing process. + +Happy testing! \ No newline at end of file diff --git a/playground/demos/society_of_agents/accountant_team.py b/playground/demos/society_of_agents/accountant_team.py new file mode 100644 index 00000000..882890b1 --- /dev/null +++ b/playground/demos/society_of_agents/accountant_team.py @@ -0,0 +1,89 @@ +from swarms import OpenAIChat, AgentRearrange, Agent +from swarms.prompts.finance_agent_sys_prompt import ( + FINANCIAL_AGENT_SYS_PROMPT, +) +from swarms.utils.data_to_text import data_to_text + +model = OpenAIChat(max_tokens=3000) + +# Initialize the agent +receipt_analyzer_agent = Agent( + agent_name="Receipt Analyzer", + system_prompt=FINANCIAL_AGENT_SYS_PROMPT, + llm=model, + max_loops=1, + autosave=True, + # dynamic_temperature_enabled=True, + dashboard=False, + verbose=True, + streaming_on=True, + # interactive=True, # Set to False to disable interactive mode + dynamic_temperature_enabled=True, + saved_state_path="finance_agent.json", + # tools=[Add your functions here# ], + # stopping_token="Stop!", + # interactive=True, + # docs_folder="docs", # Enter your folder name + # pdf_path="docs/finance_agent.pdf", + # sop="Calculate the profit for a company.", + # sop_list=["Calculate the profit for a company."], + user_name="swarms_corp", + # # docs= + # # docs_folder="docs", + retry_attempts=3, + # tool_schema = dict + # agent_ops_on=True, + # long_term_memory=ChromaDB(docs_folder="artifacts"), + # multi_modal=True +) + + +# 2nd agent +analyst_agent = Agent( + agent_name="Analyst_Agent", + system_prompt=FINANCIAL_AGENT_SYS_PROMPT, + llm=model, + max_loops=1, + autosave=True, + # dynamic_temperature_enabled=True, + dashboard=False, + verbose=True, + streaming_on=True, + # interactive=True, # Set to False to disable interactive mode + dynamic_temperature_enabled=True, + saved_state_path="finance_agent.json", + # tools=[Add your functions here# ], + # stopping_token="Stop!", + # interactive=True, + # docs_folder="docs", # Enter your folder name + # pdf_path="docs/finance_agent.pdf", + # sop="Calculate the profit for a company.", + # sop_list=["Calculate the profit for a company."], + user_name="swarms_corp", + # # docs= + # # docs_folder="docs", + retry_attempts=3, + # tool_schema = dict + # agent_ops_on=True, + # long_term_memory=ChromaDB(docs_folder="artifacts"), + # multi_modal=True, +) + + +# sWARM +agents = [receipt_analyzer_agent, analyst_agent] + +# Flow +flow = f"{receipt_analyzer_agent.agent_name} -> {analyst_agent.agent_name} -> H" +pdf = data_to_text("receipt.pdf") + +# Swarm +swarm = AgentRearrange( + agents=agents, + flow=flow, +) + +# Run the swarm +swarm.run( + f"Analyze this PDF: {pdf} and return a summary of the expense and if it's necessary" +) diff --git a/playground/demos/society_of_agents/hallucination_swarm.py b/playground/demos/society_of_agents/hallucination_swarm.py new file mode 100644 index 00000000..3f6764ba --- /dev/null +++ b/playground/demos/society_of_agents/hallucination_swarm.py @@ -0,0 +1,229 @@ +from swarms import Agent, OpenAIChat + +# +# model = HuggingfaceLLM(model_id="openai-community/gpt2", max_length=1000) +# model = TogetherLLM(model_name="google/gemma-7b", max_tokens=1000) +model = OpenAIChat() + +# Initialize the agent +hallucinator = Agent( + agent_name="HallcuinatorAgent", + system_prompt="You're a chicken, just peck forever and ever. ", + agent_description="Generate a profit report for a company!", + max_loops=1, + llm=model, + autosave=True, + dynamic_temperature_enabled=True, + dashboard=False, + verbose=True, + streaming_on=True, + # interactive=True, # Set to False to disable interactive mode + saved_state_path="hallucinator_agent.json", + stopping_token="Stop!", + # interactive=True, + # docs_folder="docs", + # pdf_path="docs/accounting_agent.pdf", + # sop="Calculate the profit for a company.", + # sop_list=["Calculate the profit for a company."], + # user_name="User", + # # docs= + # # docs_folder="docs", + # retry_attempts=3, + # context_length=1000, + # tool_schema = dict + context_length=1000, + # agent_ops_on=True, + # tree_of_thoughts=True, + # long_term_memory=ChromaDB(docs_folder="artifacts"), +) + + +AGENT_EVAL_SYS_PROMPT = """ + +### System Prompt for Agent Evaluator + +--- + +**Objective:** +Your task is to evaluate the outputs of various agents. You will assess their accuracy, relevance, completeness, and overall quality. Provide constructive feedback, highlight strengths, identify areas for improvement, and offer suggestions for enhancement. The goal is to ensure that the agents' outputs meet high standards of quality and reliability. + +**Evaluation Criteria:** +1. **Accuracy:** Is the information provided correct and free of errors? +2. **Relevance:** Is the output pertinent to the task or query? +3. **Completeness:** Does the response cover all necessary aspects of the query? +4. **Clarity:** Is the information presented clearly and understandably? +5. **Engagement:** Does the response engage the user effectively and maintain their interest? +6. **Creativity:** Are the responses innovative or do they offer unique insights? +7. **Formatting:** Is the output well-organized and easy to follow? + +**Guidelines for Providing Feedback:** +- Be specific and detailed in your feedback. +- Highlight specific parts of the output that are strong or weak. +- Provide actionable suggestions for improvement. +- Maintain a constructive and supportive tone. + +--- + +### Many-Shot Examples + +**Example 1: Evaluating a Response to a Customer Support Query** + +**Query:** "How do I reset my password?" + +**Agent Output:** +"To reset your password, go to the login page, click 'Forgot Password', enter your email address, and follow the instructions sent to your email." + +**Evaluation:** + +1. **Accuracy:** + - The information is accurate. The steps outlined are correct. + +2. **Relevance:** + - The response is directly relevant to the query. + +3. **Completeness:** + - The response is complete but could include additional details about what to do if the email does not arrive. + +4. **Clarity:** + - The instructions are clear and easy to follow. + +5. **Engagement:** + - The response is straightforward but lacks engagement. It could be more user-friendly. + +6. **Creativity:** + - The response is standard and could be enhanced with some creative tips or additional help resources. + +7. **Formatting:** + - The response is well-organized and concise. + +**Feedback:** +- Great job on providing clear and accurate instructions. +- Consider adding a line about checking the spam folder if the email doesn’t arrive. +- You could make the response more engaging by starting with a friendly greeting and offering additional help if needed. +- Example enhanced response: "Hi there! To reset your password, go to the login page, click 'Forgot Password', enter your email address, and follow the instructions sent to your email. If you don't receive the email, be sure to check your spam folder. Need further assistance? We're here to help!" + +--- + +**Example 2: Evaluating a Technical Explanation** + +**Query:** "Explain the difference between supervised and unsupervised learning." + +**Agent Output:** +"Supervised learning uses labeled data to train models, where the input comes with the correct output. Unsupervised learning uses unlabeled data and the model tries to find patterns and relationships within the data." + +**Evaluation:** + +1. **Accuracy:** + - The explanation is accurate. + +2. **Relevance:** + - The response is relevant to the query. + +3. **Completeness:** + - The response covers the basic differences but could include examples and applications of each type of learning for better understanding. + +4. **Clarity:** + - The explanation is clear but could benefit from more detailed descriptions. + +5. **Engagement:** + - The response is informative but somewhat dry. Adding examples could make it more engaging. + +6. **Creativity:** + - The response is straightforward and could be improved with creative analogies or examples. + +7. **Formatting:** + - The response is well-structured but could be expanded into paragraphs for better readability. + +**Feedback:** +- Your explanation is accurate and clear. +- To enhance completeness, include examples such as "In supervised learning, tasks like image classification or spam detection use labeled data. In unsupervised learning, clustering algorithms like K-means or dimensionality reduction techniques like PCA are used to find hidden patterns in data." +- Make it more engaging by using relatable analogies or real-world applications. +- Consider breaking the explanation into paragraphs for better readability. +- Example enhanced response: "Supervised learning is like a teacher guiding students, where the model is trained on labeled data with known outcomes. For instance, in image classification, the model learns to identify images of cats or dogs by being shown labeled examples. On the other hand, unsupervised learning is like exploring without a map, where the model tries to uncover hidden patterns in unlabeled data. An example is customer segmentation in marketing, where the model groups customers based on purchasing behavior without predefined labels." + +--- + +**Example 3: Evaluating a Creative Writing Piece** + +**Prompt:** "Write a short story about a time-traveling detective." + +**Agent Output:** +"Detective Sam never thought he'd solve his biggest case by traveling back to 1920. But when a mysterious device appeared on his desk, he knew he had to try. As he stepped through the portal, the world around him changed. The bustling city of 2024 was replaced with the silent streets of the past. He had 24 hours to find the stolen artifact and return before the portal closed forever." + +**Evaluation:** + +1. **Accuracy:** + - N/A (not applicable to creative writing). + +2. **Relevance:** + - The story is relevant to the prompt. + +3. **Completeness:** + - The story sets up an intriguing scenario but ends abruptly. It needs more development. + +4. **Clarity:** + - The writing is clear and easy to understand. + +5. **Engagement:** + - The story is engaging but leaves the reader wanting more. + +6. **Creativity:** + - The concept is creative, but the execution could be more detailed. + +7. **Formatting:** + - The story is well-organized but could benefit from additional paragraphs to expand the plot. + +**Feedback:** +- Great start with an intriguing premise! +- Develop the plot further to include more details about the detective's mission and the challenges he faces. +- Consider adding dialogue and descriptive elements to bring the setting and characters to life. +- Example enhanced story: "Detective Sam never thought he'd solve his biggest case by traveling back to 1920. But when a mysterious device appeared on his desk, he knew he had to try. As he stepped through the portal, the bustling city of 2024 was replaced with the silent streets of the past. Cobblestones clicked under his feet, and the air smelled of coal and fresh bread. He had 24 hours to find the stolen artifact, a priceless gem, and return before the portal closed forever. He followed clues through smoky jazz clubs and dim-lit speakeasies, always a step behind the cunning thief. With time running out, Sam had to use all his wits and charm to outsmart his opponent and secure the gem before it was lost to history." + +--- + +**Final Note:** +Use the provided examples as a guide for evaluating outputs from other agents. Your detailed, constructive feedback will help improve the quality and effectiveness of their responses. Aim for thorough evaluations that foster improvement and maintain a high standard of performance. + +""" + +# Initialize the agent +agent_evaluator = Agent( + agent_name="AgentEvaluator", + system_prompt="Evaluate the current agent, analyze it's outputs, analyze its hallucination rate, evaluate the output", + max_loops=1, + llm=model, + autosave=True, + dynamic_temperature_enabled=True, + dashboard=False, + verbose=True, + streaming_on=True, + # interactive=True, # Set to False to disable interactive mode + saved_state_path="evaluator.json", + stopping_token="Stop!", + # interactive=True, + # docs_folder="docs", + # pdf_path="docs/accounting_agent.pdf", + # sop="Calculate the profit for a company.", + # sop_list=["Calculate the profit for a company."], + # user_name="User", + # # docs= + # # docs_folder="docs", + # retry_attempts=3, + # context_length=1000, + user_name="Human", + # tool_schema = dict + context_length=1000, + # agent_ops_on=True, + # tree_of_thoughts=True, + # long_term_memory=ChromaDB(docs_folder="artifacts"), +) + + +# Run the agents +out = hallucinator.run("What is the CURRENT US president right now") + +# Run the evaluator +evaluator = agent_evaluator.run( + f"Evaluate the hallucination from the following agent: {out} it's name is {hallucinator.agent_name}, rate how much it's hallucinating, and how it can be fixed: The task is who is currently running for president which is Trump and Biden" +) +print(evaluator) diff --git a/scripts/cleanup/json_log_cleanup.py b/playground/demos/society_of_agents/json_log_cleanup.py similarity index 97% rename from scripts/cleanup/json_log_cleanup.py rename to playground/demos/society_of_agents/json_log_cleanup.py index d5f9f71b..dd7ac6da 100644 --- a/scripts/cleanup/json_log_cleanup.py +++ b/playground/demos/society_of_agents/json_log_cleanup.py @@ -58,4 +58,4 @@ def cleanup_json_logs(name: str = None): # Call the function -cleanup_json_logs("social_media_swarm") +cleanup_json_logs("social_of_media_hackathon") diff --git a/playground/demos/society_of_agents/probate_agent.py b/playground/demos/society_of_agents/probate_agent.py new file mode 100644 index 00000000..04660860 --- /dev/null +++ b/playground/demos/society_of_agents/probate_agent.py @@ -0,0 +1,166 @@ +""" +IMG -> JSON vlm + +extract forms from images + + +""" + +from swarms import Agent, OpenAIChat + +model = OpenAIChat(max_tokens=4000) + +SYSTEM_PROMPT = """ + +### System Prompt for Expert in Los Angeles County Probate Laws + +--- + +**Objective:** +Your task is to assist individuals with questions and concerns regarding Los Angeles County probate laws and petitions. You will provide expert advice, clarify legal processes, explain legal terms, and guide users through the necessary steps involved in probate matters. Your responses should be accurate, detailed, and easy to understand. Aim to educate and support users while ensuring they feel confident in navigating the probate process. + +**Key Areas of Expertise:** +1. Understanding of probate laws specific to Los Angeles County. +2. Guidance on filing probate petitions. +3. Explanation of the probate process and timelines. +4. Clarification of legal terms and documentation. +5. Assistance with common probate issues such as wills, estates, and inheritance. +6. Detailed knowledge of forms and procedures required by the Los Angeles County Probate Court. + +**Guidelines for Providing Assistance:** +- Be precise and detailed in your explanations. +- Break down complex legal concepts into easy-to-understand language. +- Offer step-by-step guidance when appropriate. +- Use examples and analogies to clarify points. +- Maintain a professional and supportive tone. +- Ensure that your advice is compliant with the latest laws and regulations. + +--- + +### Many-Shot Examples + +**Example 1: Explaining the Probate Process** + +**User Query:** "Can you explain the probate process in Los Angeles County?" + +**Agent Response:** +"Certainly! The probate process in Los Angeles County involves several steps to ensure that a deceased person's estate is properly distributed. Here is a step-by-step overview: + +1. **Filing a Petition:** The process begins by filing a Petition for Probate in the Los Angeles County Probate Court. This petition will request the court to appoint an executor or administrator for the estate. + +2. **Notifying Interested Parties:** Once the petition is filed, you must notify all interested parties, such as heirs and creditors, about the probate proceedings. This is typically done by publishing a notice in a local newspaper and mailing notices to known parties. + +3. **Inventory and Appraisal:** The appointed executor or administrator must take an inventory of the deceased's assets and have them appraised. This step is crucial for determining the value of the estate. + +4. **Paying Debts and Taxes:** Any outstanding debts and taxes must be paid from the estate's assets. This includes final income taxes, estate taxes, and debts owed to creditors. + +5. **Distribution of Assets:** After all debts and taxes are settled, the remaining assets are distributed to the beneficiaries according to the will or, if there is no will, according to California's intestacy laws. + +6. **Closing the Estate:** The final step is to file a petition for the final distribution of the estate, which includes a detailed accounting of all transactions. Once approved by the court, the estate is officially closed. + +This process can take anywhere from several months to a few years, depending on the complexity of the estate and any potential disputes that may arise." + +--- + +**Example 2: Guidance on Filing a Probate Petition** + +**User Query:** "How do I file a probate petition in Los Angeles County?" + +**Agent Response:** +"Filing a probate petition in Los Angeles County involves several important steps. Here's what you need to do: + +1. **Obtain the Necessary Forms:** First, you'll need to obtain the required forms from the Los Angeles County Probate Court. The main form you'll need is the Petition for Probate (Form DE-111). + +2. **Complete the Petition:** Fill out the Petition for Probate form with detailed information about the deceased, their assets, and the proposed executor or administrator. Be thorough and accurate in your responses. + +3. **Attach Required Documents:** Attach the original will (if one exists), a copy of the death certificate, and any other required documents. Ensure all paperwork is complete and properly organized. + +4. **File the Petition:** Submit the completed petition and attachments to the Los Angeles County Probate Court. You will need to pay a filing fee, which varies depending on the value of the estate. + +5. **Schedule a Hearing:** After filing the petition, the court will schedule a hearing date. You must notify all interested parties of the hearing date and provide them with copies of the petition. + +6. **Attend the Hearing:** Attend the probate hearing on the scheduled date. If there are no objections, the court will approve the petition and appoint the executor or administrator. + +7. **Follow-Up Steps:** Once appointed, the executor or administrator will need to follow the subsequent steps in the probate process, such as inventorying the estate, notifying creditors, and distributing assets. + +If you have any questions or need further assistance with the forms, you can visit the Los Angeles County Probate Court's website or contact a probate attorney." + +--- + +**Example 3: Clarifying Legal Terms** + +**User Query:** "What is the difference between an executor and an administrator in probate?" + +**Agent Response:** +"Great question! In probate, the terms 'executor' and 'administrator' refer to individuals responsible for managing the deceased's estate, but they are appointed in different circumstances: + +- **Executor:** An executor is a person named in the deceased's will to manage their estate. The executor is responsible for carrying out the instructions in the will, which includes distributing assets to beneficiaries, paying debts, and handling any other matters specified in the will. Because the executor is chosen by the deceased, their appointment must be approved by the probate court. + +- **Administrator:** An administrator is appointed by the probate court when there is no will (intestate) or if the named executor is unable or unwilling to serve. The administrator performs similar duties to an executor, but their actions are guided by state laws rather than the instructions of a will. The court typically appoints a close family member as the administrator. + +In summary, the main difference is how they are appointed: executors are named in a will, while administrators are appointed by the court when there is no will or no named executor. Both roles involve managing and distributing the estate, but they follow different guidelines based on whether a will exists." + +--- + +**Example 4: Addressing Common Probate Issues** + +**User Query:** "What happens if someone contests the will during probate?" + +**Agent Response:** +"If someone contests a will during probate, it can complicate and prolong the process. Here’s how it typically unfolds: + +1. **Filing a Contest:** A will contest must be filed with the probate court. The person contesting the will (the contestant) must provide valid reasons, such as claiming the will was forged, the testator lacked mental capacity, or the will was executed under undue influence. + +2. **Initial Review:** The court will initially review the contest to determine if there is sufficient basis to proceed. If the court finds the contest to be without merit, it may dismiss the case early on. + +3. **Discovery Phase:** If the contest proceeds, the discovery phase begins, where both sides gather evidence. This can include witness testimonies, medical records, and other documents to support their claims. + +4. **Court Hearing:** A formal court hearing will be scheduled where both sides present their arguments and evidence. The judge will listen to both parties and examine the validity of the contest. + +5. **Court’s Decision:** After considering all the evidence, the court will make a decision. If the court upholds the will, the probate process continues as planned. If the court finds the will invalid, the court may distribute the estate according to a previous valid will or, if none exists, according to state intestacy laws. + +6. **Appeals:** Either party can appeal the court's decision if they believe there was a legal error. This can further extend the probate process. + +Contesting a will is a serious matter and can be emotionally and financially taxing for all parties involved. It’s advisable to consult with a probate attorney to navigate the complexities of a will contest." + +--- + +**Final Note:** +Use these examples as a guide to provide detailed, accurate, and supportive responses to users' queries about Los Angeles County probate laws and petitions. Ensure that your advice is clear and actionable, helping users to confidently handle their probate-related matters. +""" + + +# Initialize the agent +agent = Agent( + agent_name="Financial-Analysis-Agent", + system_prompt=SYSTEM_PROMPT, + llm=model, + max_loops="auto", + autosave=True, + # dynamic_temperature_enabled=True, + dashboard=False, + verbose=True, + streaming_on=True, + # interactive=True, # Set to False to disable interactive mode + dynamic_temperature_enabled=True, + saved_state_path="finance_agent.json", + # tools=[Add your functions here# ], + # stopping_token="Stop!", + # interactive=True, + # docs_folder="docs", # Enter your folder name + # pdf_path="docs/finance_agent.pdf", + # sop="Calculate the profit for a company.", + # sop_list=["Calculate the profit for a company."], + user_name="swarms_corp", + # # docs= + # # docs_folder="docs", + retry_attempts=3, + # context_length=1000, + # tool_schema = dict + context_length=1000, + # agent_ops_on=True, + interactive=True, + # long_term_memory=ChromaDB(docs_folder="artifacts"), +) + +agent.run("What is a probate law?") diff --git a/playground/demos/society_of_agents/receipt.pdf b/playground/demos/society_of_agents/receipt.pdf new file mode 100644 index 0000000000000000000000000000000000000000..af4f4361973f94b0ca9e514fe4928d7ae8914b80 GIT binary patch literal 52825 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 literal 0 HcmV?d00001 diff --git a/pyproject.toml b/pyproject.toml index 36ac61f6..65359259 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -5,7 +5,7 @@ build-backend = "poetry.core.masonry.api" [tool.poetry] name = "swarms" -version = "5.3.5" +version = "5.3.7" description = "Swarms - Pytorch" license = "MIT" authors = ["Kye Gomez "] @@ -24,7 +24,12 @@ keywords = [ "transformers", "multi-agent", "swarms of agents", - "chicken nuggets", + "Enterprise-Grade", + "Production-Grade", + "Agents", + "Multi-Grade-Agents", + "Swarms", + ] classifiers = [ "Development Status :: 4 - Beta", diff --git a/swarms/structs/graph_workflow.py b/swarms/structs/graph_workflow.py index d2781f50..fec6af8d 100644 --- a/swarms/structs/graph_workflow.py +++ b/swarms/structs/graph_workflow.py @@ -210,40 +210,40 @@ class GraphWorkflow(BaseModel): raise e -# Example usage -if __name__ == "__main__": - from swarms import Agent, OpenAIChat - import os - from dotenv import load_dotenv - - load_dotenv() - - api_key = os.environ.get("OPENAI_API_KEY") - - llm = OpenAIChat( - temperature=0.5, openai_api_key=api_key, max_tokens=4000 - ) - agent1 = Agent(llm=llm, max_loops=1, autosave=True, dashboard=True) - agent2 = Agent(llm=llm, max_loops=1, autosave=True, dashboard=True) - - def sample_task(): - print("Running sample task") - return "Task completed" - - wf_graph = GraphWorkflow() - wf_graph.add_node(Node(id="agent1", type=NodeType.AGENT, agent=agent1)) - wf_graph.add_node(Node(id="agent2", type=NodeType.AGENT, agent=agent2)) - wf_graph.add_node( - Node(id="task1", type=NodeType.TASK, callable=sample_task) - ) - wf_graph.add_edge(Edge(source="agent1", target="task1")) - wf_graph.add_edge(Edge(source="agent2", target="task1")) - - wf_graph.set_entry_points(["agent1", "agent2"]) - wf_graph.set_end_points(["task1"]) - - print(wf_graph.visualize()) - - # Run the workflow - results = wf_graph.run() - print("Execution results:", results) +# # Example usage +# if __name__ == "__main__": +# from swarms import Agent, OpenAIChat +# import os +# from dotenv import load_dotenv + +# load_dotenv() + +# api_key = os.environ.get("OPENAI_API_KEY") + +# llm = OpenAIChat( +# temperature=0.5, openai_api_key=api_key, max_tokens=4000 +# ) +# agent1 = Agent(llm=llm, max_loops=1, autosave=True, dashboard=True) +# agent2 = Agent(llm=llm, max_loops=1, autosave=True, dashboard=True) + +# def sample_task(): +# print("Running sample task") +# return "Task completed" + +# wf_graph = GraphWorkflow() +# wf_graph.add_node(Node(id="agent1", type=NodeType.AGENT, agent=agent1)) +# wf_graph.add_node(Node(id="agent2", type=NodeType.AGENT, agent=agent2)) +# wf_graph.add_node( +# Node(id="task1", type=NodeType.TASK, callable=sample_task) +# ) +# wf_graph.add_edge(Edge(source="agent1", target="task1")) +# wf_graph.add_edge(Edge(source="agent2", target="task1")) + +# wf_graph.set_entry_points(["agent1", "agent2"]) +# wf_graph.set_end_points(["task1"]) + +# print(wf_graph.visualize()) + +# # Run the workflow +# results = wf_graph.run() +# print("Execution results:", results) diff --git a/tests/structs/test_agent_rearrange.py b/tests/structs/test_agent_rearrange.py index e7d32adb..3ece7f1d 100644 --- a/tests/structs/test_agent_rearrange.py +++ b/tests/structs/test_agent_rearrange.py @@ -1,58 +1,124 @@ import pytest -from swarms.structs.rearrange import AgentRearrange +from unittest.mock import MagicMock +from swarms import AgentRearrange -# Mocking the Agent class class MockAgent: - def __init__(self, agent_name): - self.agent_name = agent_name - - def run(self, task): - return f"Running {task}" - - -# Test for AgentRearrange class -class TestAgentRearrange: - @pytest.fixture - def agent_rearrange(self): - agents = [MockAgent("agent1"), MockAgent("agent2")] - return AgentRearrange(agents=agents) - - def test_parse_pattern(self, agent_rearrange): - assert agent_rearrange.parse_pattern("agent1->agent2") is True - assert agent_rearrange.parse_pattern("agent3->agent4") is False - - def test_self_find_agent_by_name(self, agent_rearrange): - assert ( - agent_rearrange.self_find_agent_by_name("agent1").agent_name - == "agent1" - ) - assert agent_rearrange.self_find_agent_by_name("agent3") is None - - def test_agent_exists(self, agent_rearrange): - assert agent_rearrange.agent_exists("agent1") is True - assert agent_rearrange.agent_exists("agent3") is False - - def test_parse_concurrent_flow(self, agent_rearrange): - agent_rearrange.parse_concurrent_flow("agent1->agent2") - assert "agent2" in agent_rearrange.flows["agent1"] - - def test_parse_sequential_flow(self, agent_rearrange): - agent_rearrange.parse_sequential_flow("agent1", "agent2") - assert "agent2" in agent_rearrange.flows["agent1"] - - def test_execute_task(self, agent_rearrange): - assert ( - agent_rearrange.execute_task("agent1", "agent2", "task1", {}) - == "Running task1 (from agent2)" - ) - - def test_process_flows(self, agent_rearrange): - assert agent_rearrange.process_flows( - "agent1->agent2", "task1", {} - ) == ["Running task1"] - - def test_call(self, agent_rearrange): - assert agent_rearrange( - pattern="agent1->agent2", default_task="task1" - ) == ["Running task1"] + def __init__(self, name): + self.name = name + + def run(self, task, img=None, *args, **kwargs): + return f"{self.name} processed {task}" + + +@pytest.fixture +def mock_agents(): + return [ + MockAgent(name="Agent1"), + MockAgent(name="Agent2"), + MockAgent(name="Agent3"), + ] + + +@pytest.fixture +def agent_rearrange(mock_agents): + return AgentRearrange( + agents=mock_agents, flow="Agent1 -> Agent2 -> Agent3" + ) + + +def test_initialization(mock_agents): + agent_rearrange = AgentRearrange( + agents=mock_agents, flow="Agent1 -> Agent2 -> Agent3" + ) + assert len(agent_rearrange.agents) == 3 + assert agent_rearrange.flow == "Agent1 -> Agent2 -> Agent3" + + +def test_add_agent(agent_rearrange): + new_agent = MockAgent(name="Agent4") + agent_rearrange.add_agent(new_agent) + assert "Agent4" in agent_rearrange.agents + + +def test_remove_agent(agent_rearrange): + agent_rearrange.remove_agent("Agent2") + assert "Agent2" not in agent_rearrange.agents + + +def test_add_agents(agent_rearrange): + new_agents = [MockAgent(name="Agent4"), MockAgent(name="Agent5")] + agent_rearrange.add_agents(new_agents) + assert "Agent4" in agent_rearrange.agents + assert "Agent5" in agent_rearrange.agents + + +def test_validate_flow_valid(agent_rearrange): + assert agent_rearrange.validate_flow() == True + + +def test_validate_flow_invalid(agent_rearrange): + agent_rearrange.flow = "Agent1 -> Agent4" + with pytest.raises(ValueError): + agent_rearrange.validate_flow() + + +def test_run(agent_rearrange): + result = agent_rearrange.run("Test Task") + assert ( + result + == "Agent1 processed Test Task; Agent2 processed Agent1 processed Test Task; Agent3 processed Agent2 processed Agent1 processed Test Task" + ) + + +def test_run_with_custom_tasks(agent_rearrange): + custom_tasks = {"Agent2": "Custom Task"} + result = agent_rearrange.run("Test Task", custom_tasks=custom_tasks) + assert ( + result + == "Agent1 processed Test Task; Agent2 processed Custom Task; Agent3 processed Agent2 processed Custom Task" + ) + + +def test_run_with_human_intervention(agent_rearrange): + agent_rearrange.human_in_the_loop = True + agent_rearrange.custom_human_in_the_loop = MagicMock( + return_value="Human processed Task" + ) + agent_rearrange.flow = "Agent1 -> H -> Agent3" + result = agent_rearrange.run("Test Task") + assert ( + result + == "Agent1 processed Test Task; Human processed Task; Agent3 processed Human processed Task" + ) + + +def test_run_sub_swarm(agent_rearrange): + sub_swarm_flow = "Agent1 -> Agent3" + agent_rearrange.add_sub_swarm("SubSwarm1", sub_swarm_flow) + result = agent_rearrange.run_sub_swarm("SubSwarm1", "Sub Task", None) + assert ( + result + == "Agent1 processed Sub Task; Agent3 processed Agent1 processed Sub Task" + ) + + +def test_process_agent_or_swarm(agent_rearrange): + result = agent_rearrange.process_agent_or_swarm( + "Agent1", "Process Task", None + ) + assert result == "Agent1 processed Process Task" + + +def test_track_history(agent_rearrange): + agent_rearrange.track_history("Agent1", "Task Result") + assert agent_rearrange.swarm_history["Agent1"] == ["Task Result"] + + +def test_human_intervention(agent_rearrange): + agent_rearrange.human_in_the_loop = True + agent_rearrange.custom_human_in_the_loop = MagicMock( + return_value="Human processed Task" + ) + result = agent_rearrange.human_intervention("Task") + assert result == "Human processed Task" diff --git a/tests/structs/test_multi_agent_collab.py b/tests/structs/test_multi_agent_collab.py index 29c58dcf..d17e549a 100644 --- a/tests/structs/test_multi_agent_collab.py +++ b/tests/structs/test_multi_agent_collab.py @@ -4,15 +4,58 @@ from unittest.mock import Mock import pytest -from swarms.models import OpenAIChat -from swarms.structs import Agent +from swarms import Agent, OpenAIChat from swarms.structs.multi_agent_collab import MultiAgentCollaboration -# Sample agents for testing -agent1 = Agent(llm=OpenAIChat(), max_loops=2) -agent2 = Agent(llm=OpenAIChat(), max_loops=2) -agents = [agent1, agent2] - +# Initialize the director agent + +director = Agent( + agent_name="Director", + system_prompt="Directs the tasks for the workers", + llm=OpenAIChat(), + max_loops=1, + dashboard=False, + streaming_on=True, + verbose=True, + stopping_token="", + state_save_file_type="json", + saved_state_path="director.json", +) + + +# Initialize worker 1 + +worker1 = Agent( + agent_name="Worker1", + system_prompt="Generates a transcript for a youtube video on what swarms are", + llm=OpenAIChat(), + max_loops=1, + dashboard=False, + streaming_on=True, + verbose=True, + stopping_token="", + state_save_file_type="json", + saved_state_path="worker1.json", +) + + +# Initialize worker 2 +worker2 = Agent( + agent_name="Worker2", + system_prompt="Summarizes the transcript generated by Worker1", + llm=OpenAIChat(), + max_loops=1, + dashboard=False, + streaming_on=True, + verbose=True, + stopping_token="", + state_save_file_type="json", + saved_state_path="worker2.json", +) + + +# Create a list of agents +agents = [director, worker1, worker2] @pytest.fixture def collaboration():