surv-ai


Namesurv-ai JSON
Version 0.2.0 PyPI version JSON
download
home_pagehttps://github.com/DanielBalsam/surv_ai
SummaryA framework for multi-agent modeling using large language models
upload_time2023-06-13 03:25:29
maintainer
docs_urlNone
authorDaniel Balsam
requires_python>=3.9,<4.0
licenseMIT
keywords ai artificial intelligence data science statistics models llm agents survai surv_ai survey multi-agent large language model artificial intelligence machine learning natural language processing nlp sentiment analysis
VCS
bugtrack_url
requirements No requirements were recorded.
Travis-CI No Travis.
coveralls test coverage No coveralls.
            # ๐Ÿ•ต Multi-agent modeling with large language models 
<!-- [![PyPi](https://img.shields.io/badge/Official%20Website-agpt.co-blue?style=flat&logo=world&logoColor=white)](https://agpt.co) -->
[![PyPi](https://shields.io/pypi/v/surv-ai)](https://pypi.org/project/surv-ai/)
[![PyPi](https://shields.io/pypi/pyversions/surv-ai)](https://pypi.org/project/surv-ai/)
[![Unit Tests](https://shields.io/github/actions/workflow/status/DanielBalsam/surv_ai/.github/workflows/ci.yaml?branch=main)](https://github.com/DanielBalsam/surv_ai/actions/workflows/ci.yaml)
[![GitHub Repo stars](https://img.shields.io/github/stars/DanielBalsam/surv_ai?style=social)](https://github.com/DanielBalsam/surv_ai/stargazers)


**`surv_ai` is a large language model framework designed for multi-agent modeling. This allows large-language models to be used as engines to power research into predictive modeling, bias analysis, and other forms of comparative analysis.**

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/midterms_2022.png)

**Multi-agent modeling** involves generating statistical models based on the actions of multiple agents. In our case, these models are crafted by agents querying and processing text from a comprehensive data corpus. The agents then reason with the information extracted to produce a data point.

Given the stochastic nature of large language models, individual data points may vary. However, if a sufficient number of agents are employed, the generated models can effectively support comparative analysis.

They key idea here was inspired by the [technique of bagging (bootstrap aggregating) in classical ensemble classifiers.](https://en.wikipedia.org/wiki/Bootstrap_aggregating) The basic idea is that if you have a lot of weak learners with only limited information - in aggregate those weak learners can produce much higher quality results than a single weak learner with maximum information (or a bunch a weak learners with maximum information).

This technique appears to work better when each agent has access to a limited subset of the total information even when using language models with larger context windows.

The abstraction in this repository capable of producing a multi-agent data point is known as a `Survey`. A `Survey` takes a statement as input and returns the percentage of agents that agreed with it.

A more complex abstraction, called a `Model`, enables the variation of input parameters into a `Survey` to create a nuanced multi-agent model.

The data points produced serve as sentiment analysis against a text corpus, making them susceptible to the biases of both the large language models and the corpus itself. However, if the data source provided is quality - then the model produced can be high quality as well.

As advances in large language models and AI continue, multi-agent modeling may remain a valuable framework for classification and regression models. It also has the potential to become a useful data point for researchers investigating complex issues with numerous underlying variables.

## ๐Ÿ“ฒ Installation 

Package is available on [PyPi](https://pypi.org/project/surv-ai/): 

```
pip install surv-ai
```

## ๐Ÿ“ป Contact 

For all inquiries, contact me at: daniel.balsam@survai.org

## โœ… Responsible use 

The examples provided below are intended merely to illustrate potential applications of this framework; they are not to be considered scientifically rigorous.

Approaches like this have potential in guiding research and decision-making. However, it is crucial to rely on a diverse range of data points and to interpret each model cautiously.

I am always open to suggestions for further enhancing this approach.

## ๐Ÿ“ Basic usage 

The two key abstractions in this repository are a `Survey` and a `Model`. 

Executing the method `Survey.conduct` with a hypothesis will spin up a number of agents and seed them with some base knowledge. The agents are then asked to to assign a true or false value to the hypothesis provided. 

```
class Survey:
    def __init__(
        self,
        client: LargeLanguageModelClientInterface,
        tool_belt: ToolBeltInterface,
        n_agents=10,
        max_concurrency=10,
        max_knowledge_per_agent=3,
    ):
        ...

    async def conduct(self, hypothesis: str) -> SurveyResponse:
        ...
```

A `Model` is an abstraction that allows one to conduct many surveys while changing some independent variable.

For example: one could could change the publish date of news articles that should be considered in the survey. 

```
class Model:
    def __init__(
        self,
        survey_class: type[SurveyInterface],
        parameters: list[SurveyParameter],
        max_concurrency: int = 1,
    ):
        ...

    async def build(self, hypothesis: str) -> list[DataPoint]:
        ...
```

All abstractions implemented in this repository adhere to simple abstract interfaces - so you can easily build your own agents, surveys, and models.

## ๐ŸŽ“ Examples 

All the below examples will be conducted with either GPT or Claude, and Google Custom Search. Links to relevant docs to get your own API keys:

[**OpenAI Docs**](https://platform.openai.com/) / 
[**Anthropic Docs**](https://console.anthropic.com/docs) / 
[**Google Custom Search Docs**](https://developers.google.com/custom-search/v1/overview)

### Comparing against a ground truth

Let's start by establishing the system's ability to figure out if information is true.

```
from surv_ai import (
    GPTClient,
    Survey,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge
)

client = GPTClient(os.environ["OPEN_AI_API_KEY"])

tool_belt = ToolBelt(
    tools=[
        GoogleCustomSearchTool(
            google_api_key=os.environ["GOOGLE_API_KEY"],
            google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
            start_date="2023-01-01",
            end_date="2023-05-01",
            n_pages=10,
        )
    ]
)
base_knowledge = [
    Knowledge(
        text="It is currently 2023/05/01, all the articles are from 2023.",
        source="Additional context",
    ),
]

survey = Survey(
    client=client,
    tool_belt=tool_belt,
    base_knowledge=base_knowledge,
    max_knowledge_per_agent=3,
    n_agents=10,
)

await survey.conduct(
    "California experienced a significant amount of rainfall this winter.",
)  # This should always returns high a confidence agreement.
```

Executing this code should yield an output similar to: `SurveyResponse(in_favor=10, against=0, undecided=0, error=0, percent_in_favor=1.0, uncertainty=0.0)`

It's important to recognize that occasionally, one or two agents might incorrectly respond to a straightforward question like this. That's why we employ multiple agents in these systems - to counteract the randomness inherent in Large Language Models (LLMs).

To further test our hypothesis, we could also assert the contrary position and ensure we obtain the opposite value.

```
from surv_ai import (
    GPTClient,
    Survey,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge
)

client = GPTClient(os.environ["OPEN_AI_API_KEY"])

tool_belt = ToolBelt(
    tools=[
        GoogleCustomSearchTool(
            google_api_key=os.environ["GOOGLE_API_KEY"],
            google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
            start_date="2023-01-01",
            end_date="2023-05-01",
            n_pages=10,
        )
    ]
)
base_knowledge = [
    Knowledge(
        text="It is currently 2023/05/01, all the articles are from 2023.",
        source="Additional context",
    ),
]

survey = Survey(
    client=client,
    tool_belt=tool_belt,
    base_knowledge=base_knowledge,
    max_knowledge_per_agent=3,
    n_agents=10,
)

await survey.conduct(
    "California experienced little rainfall this winter.",
)  # This should always returns a high confidence disagreement.
```

Executing this code should result in an output similar to: `SurveyResponse(in_favor=0, against=10, undecided=0, error=0, percent_in_favor=0.0, uncertainty=0.0)`

### Comparing changes in sentiment over time

One thing we can use this tool for is measuring changes in sentiment overtime. GPT's training data ends in late 2021, so one way we can test our models with GPT is by looking at events that happened after GPT's training cutoff.

For instance, we can plot how sentiment regarding the United States' 2022 Midterm Elections evolved in the months leading up to it:

```
from surv_ai import (
    GPTClient,
    Model,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge,
    Survey,
    SurveyParameter
)


client = GPTClient(os.environ["OPEN_AI_API_KEY"])

def build_parameter(date_range: tuple[str, str]):
    tool_belt = ToolBelt(
        tools=[
            GoogleCustomSearchTool(
                google_api_key=os.environ["GOOGLE_API_KEY"],
                google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
                n_pages=30,
                start_date=date_range[0],
                end_date=date_range[1],
                max_concurrency=20,
            ),
        ],
    )
    base_knowledge = [
        Knowledge(
            text=f"It is currently {date_range[0]}. The included articles were published between {date_range[0]} and {date_range[1]}",
            source="Additional context",
        ),
    ]
    return SurveyParameter(
        independent_variable=date_range[1],
        kwargs={
            "client": client,
            "n_agents": 100,
            "max_knowledge_per_agent":5,
            "max_concurrency": 20,
            "tool_belt": tool_belt,
            "base_knowledge": base_knowledge,
        },
    )

date_ranges = [
    ('2022-05-01', '2022-06-01'),
    ('2022-06-01', '2022-07-01'),
    ('2022-07-01', '2022-08-01'),
    ('2022-08-01', '2022-09-01'),
    ('2022-09-01', '2022-10-01'),
    ('2022-10-01', '2022-11-05'),
]

model = Model(
    Survey,
    parameters=[build_parameter(date_range) for date_range in date_ranges],
)

democrat_results = await model.build(
    "Democrats are favored to maintain control of the Senate in the 2022 November Midterm elections.",
)
```

We can also plot the inverted statement and observe opposite trend lines:

```
republican_results = await model.build(
    "Republicans are favored to maintain control of the Senate in the 2022 November Midterm elections.",
)
```

When compared with a leading model in political opinion polling, our model is presented as follows:

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/midterms_2022.png)

*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com. [FiveThirtyEight data can be found here.](https://projects.fivethirtyeight.com/2022-election-forecast/senate/)*

Pretty cool! Note that the outputs don't represent complementary probabilities due to nuances in how the models reason, and biases of the models and data sources, but we can observe trend lines that do mirror each other quite well.

Another example could involve plotting sentiments about the economy and using fluctuations in the S&P 500 as a benchmark for accuracy.

```
from surv_ai import (
    GPTClient,
    Model,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge,
    Survey,
    SurveyParameter
)
client = GPTClient(os.environ["OPEN_AI_API_KEY"])

def build_parameter(date_range: tuple[str, str]):
    tool_belt = ToolBelt(
        tools=[
            GoogleCustomSearchTool(
                google_api_key=os.environ["GOOGLE_API_KEY"],
                google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
                n_pages=20,
                start_date=date_range[0],
                end_date=date_range[1]
            ),
        ],
    )
    base_knowledge = [
        Knowledge(
            text=f"It is currently {date_range[0]}. The included articles were published between {date_range[0]} and {date_range[1]}",
            source="Additional context",
        ),
    ]
    return SurveyParameter(
        independent_variable=date_range[1],
        kwargs={
            "client": client,
            "n_agents": 100,
            "max_knowledge_per_agent":5,
            "max_concurrency": 10,
            "tool_belt": tool_belt,
            "base_knowledge": base_knowledge,
        },
    )

date_ranges = [
    ('2021-09-01', '2022-01-01'),
    ('2022-01-01', '2022-03-01'),
    ('2022-03-01', '2022-06-01'),
    ('2022-06-01', '2022-09-01'),
    ('2022-09-01', '2023-01-01'),
    ('2023-01-01', '2023-03-01'),
    ('2023-03-01', '2023-06-01'),
]

model = Model(
    Survey,
    parameters=[build_parameter(date_range) for date_range in date_ranges],
)

results = await model.build(
    "The United States economy looks like it is heading for a recession.",
)
```

This gives us the following graph:

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/S&P.png)

*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com. Please note that it is the complement of multi-agent model that is plotted. [S&P data can be found here.](https://www.google.com/finance/quote/.INX:INDEXSP?sa=X&ved=2ahUKEwi2hJWyg6D_AhXiGFkFHdqHCIYQ3ecFegQIIRAf)*

### Measuring bias in a data corpus

A promising application of this technique is observing bias within a text corpus. For instance, we could create a model that uses different news sites as its independent variable to explore how the agents' conclusions might vary based on the data source utilized.

```
from surv_ai import (
    GPTClient,
    Model,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge,
    Survey,
    SurveyParameter
)

client = GPTClient(os.environ["OPEN_AI_API_KEY"])

def build_parameter(news_source: str):
    tool_belt = ToolBelt(
        tools=[
            GoogleCustomSearchTool(
                google_api_key=os.environ["GOOGLE_API_KEY"],
                google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
                n_pages=20,
                start_date="2023-05-01",
                end_date="2023-06-01",
                only_include_sources=[news_source]
            ),
        ],
    )
    base_knowledge = [
        Knowledge(
            text=f"It is currently 2023-06-01. The included articles were published between 2023-05-01 and 2023-06-01",
            source="Additional context",
        ),
    ]
    return SurveyParameter(
        independent_variable=news_source,
        kwargs={
            "client": client,
            "n_agents": 100,
            "max_knowledge_per_agent":10,
            "max_concurrency": 10,
            "tool_belt": tool_belt,
            "base_knowledge": base_knowledge,
        },
    )

news_sources = [
    "nytimes.com",
    "cnn.com",
    "wsj.com",
    "foxnews.com",
]

model = Model(
    Survey,
    parameters=[build_parameter(news_source) for news_source in news_sources],
)
results = await model.build(
    "Republicans are responsible for the impending debt ceiling crisis."
)
```

This provides us with a scatter plot representing the above-mentioned news sources:

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/debt_ceiling.png)

*In this example, for each news site the agents looked only at articles published in May of 2023. Omitted publications did not have enough articles on the topic published to get reliable results.*

### Measuring biases in different large language models

Another promising method for measuring bias involves comparing biases across various large language models, provided they all receive identical input parameters.

```
from surv_ai import (
    GPTClient,
    AnthropicClient,
    Model,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge,
    Survey,
    SurveyParameter,
    LargeLanguageModelClientInterface
)

clients = [AnthropicClient(os.environ["ANTHROPIC_API_KEY"]), GPTClient(os.environ["OPEN_AI_API_KEY"])]

def build_parameter(client: LargeLanguageModelClientInterface):
    tool_belt = ToolBelt(
        tools=[
            GoogleCustomSearchTool(
                google_api_key=os.environ["GOOGLE_API_KEY"],
                google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
                n_pages=20,
                start_date="2023-01-01",
                end_date="2024-05-01",
                max_concurrency=3,
            )
        ],
    )
    base_knowledge = [
        Knowledge(
            text=f"It is currently 2023-06-01. The included articles were published between 2023-01-01 and 2023-06-01",
            source="Additional context",
        ),
    ]
    return SurveyParameter(
        independent_variable=client.__class__.__name__,
        kwargs={
            "client": client,
            "n_agents": 100,
            "max_knowledge_per_agent":3,
            "max_concurrency": 3,
            "tool_belt": tool_belt,
            "base_knowledge": base_knowledge,
        },
    )

model = Model(
    Survey,
    parameters=[build_parameter(client) for client in clients],
)
results = await model.build(
    "OpenAI has been irresponsible in their handling of AI technology."
)
```

When we compare the results between Anthropic and OpenAI's models, the scatter plot appears as follows:

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/openai.png)

*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com for articles published in the first half of 2023.*


### Measuring sentiment on Twitter

In addition to querying news sources we can also query social media websites for sentiment. There are many uses for this, but one simple and fun one would be to try and see if we can predict who will win the Academy Award for best picture.

```
from surv_ai import (
    GPTClient,
    ToolBelt,
    TwitterTool,
    Knowledge,
    Survey,
)
client = GPTClient(os.environ["OPEN_AI_API_KEY"])

films = [
    "Everything Everywhere All At Once",
    "All Quiet On the Western Front",
    "The Banshees of Inisherin",
    "The Fablemans",
    "Tรกr",
    "Top Gun: Maverick",
    "Triangle of Sadness",
    "Women Talking",
]

survey = Survey(
    client,
    n_agents=200,
    tool_belt=ToolBelt(
        tools=[
            TwitterTool(
                start_date="2023-01-01",
                end_date="2023-03-11",
                n_tweets=1000,
            )
        ]
    ),
    max_knowledge_per_agent=10,
    base_knowledge=[
        Knowledge(text="It is currently March 11th, 2023 and the Oscar's are tomorrow.", source="Additional context")
    ]
)

results = {}
for film in films:
    results[film] = await survey.conduct(f"{film} is likely to win the Academy Award for Best Picture.")
```

Placing our results on a scatter plot, we get the correct result!

![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/best_picture.png)
*In this example, the agents were allowed to read Tweets up between January 1st 2023, and March 11th 2023 (the day before the Oscar's).*

Sentiment on Twitter will not always be perfectly predictive of who will win an Academy Award, but it is an interesting way to try and predict outcomes.

## ๐Ÿง  Tips 

Ultimately, a `Survey` is powered by a Large Language Model (LLM), which means that the survey hypothesis might require tuning, much like the general need to tune prompts. Here are some insights on crafting hypotheses.

In these systems, any ambiguity in the original hypothesis can lead to unexpected results. This often happens because the agents interpret the statement too literally, thus rejecting the precise phrasing of the statement.

Another useful tactic involves seeding base knowledge to the agents, which provides extra context to the problem. To revisit a previous example:

```
from surv_ai import (
    GPTClient,
    Survey,
    ToolBelt,
    GoogleCustomSearchTool,
    Knowledge
)

client = GPTClient(os.environ["OPEN_AI_API_KEY"])

tool_belt = ToolBelt(
    tools=[
        GoogleCustomSearchTool(
            google_api_key=os.environ["GOOGLE_API_KEY"],
            google_search_engine_id=os.environ["GOOGLE_SEARCH_ENGINE_ID"],
            start_date="2023-01-01",
            end_date="2023-05-01",
            n_pages=10,
        )
    ]
)
base_knowledge = [
    Knowledge(
        text="It is currently 2023/05/01, all the articles are from 2023.",
        source="Additional context",
    ),
]

survey = Survey(
    client=client,
    tool_belt=tool_belt,
    base_knowledge=base_knowledge,
    max_knowledge_per_agent=3,
    n_agents=10,
)

await survey.conduct(
    "California experienced a significant amount of rainfall this winter.",
)  # This should always returns high a confidence agreement.
```

The base knowledge assists in keeping the agents on track. As GPT's training data concludes in 2021, without an additional prompt, GPT might assume it's still 2021, which would prevent it from evaluating the statement's validity accurately.

A bit of trial and error is often necessary, but you can debug more efficiently by reading the output of the agents. This is made possible by setting the log level:

```
from lib.log import logger, AgentLogLevel

logger.set_log_level(AgentLogLevel.OUTPUT)  # Output from the agents will be logged
logger.set_log_level(AgentLogLevel.INTERNAL)  # Agent internal "thoughts" will be logged
```

You may also prompt agents directly:

```
from surv_ai import (
    GPTClient,
    ReasoningAgent,
    Knowledge
)

client = GPTClient(os.environ["OPEN_AI_API_KEY"])
agent = ReasoningAgent(client)

agent.teach_text("You can provide text information to an agent using `teach_text`.", "Context")
agent.teach_knowledge(
    Knowledge(
        text="You can similarly pass a knowledge object directly using `teach_knowledge`",
        source="Context"
    )
)

agent.prompt("There are multiple ways to teach an agent knowledge.")
```

If you are noticing a large number of errors, you may be hitting rate limits in your LLM API - you can get around this by adjusting the `max_concurrency` parameter in both your tools and `Survey`.

## ๐Ÿคฉ Inspiration 

This project was inspired by numerous innovative projects and recent papers. Some of the inspirations for this project include:

1. [Generative Agents: Interactive Simulacra of Human Behavior](https://arxiv.org/abs/2304.03442).
2. [Large Language Models are Zero-Shot Reasoners](https://arxiv.org/abs/2205.11916)
3. [AutoGPT](https://github.com/Significant-Gravitas/Auto-GPT)

Additionally, thanks go to the multitude of researchers and engineers out there who are contributing to unlocking the power of these models!

## ๐Ÿ“ˆ Next steps 

A few directions I plan to explore with this project include:

1. Use the AmbiFC fact-checking data set to benchmark the approach of this framework and compare all future changes against this benchmark: https://paperswithcode.com/paper/evidence-based-verification-for-real-world
2. More documentation and use guides!
3. Kicking off a development blog.
4. More integrations with various helpful tools.
5. More examples and use cases.
6. Experimentation to further optimize performance.

## ๐Ÿค Contribute 

If you'd like to contribute then please reach out!

            

Raw data

            {
    "_id": null,
    "home_page": "https://github.com/DanielBalsam/surv_ai",
    "name": "surv-ai",
    "maintainer": "",
    "docs_url": null,
    "requires_python": ">=3.9,<4.0",
    "maintainer_email": "",
    "keywords": "ai,artificial intelligence,data science,statistics,models,llm,agents,survai,surv_ai,survey,multi-agent,large language model,artificial intelligence,machine learning,natural language processing,nlp,sentiment analysis",
    "author": "Daniel Balsam",
    "author_email": "daniel.balsam@survai.org",
    "download_url": "https://files.pythonhosted.org/packages/20/f1/a0ab7c7feb5de5c880dabd439c53a1665360c333aa62ae923fc4c6bd62dd/surv_ai-0.2.0.tar.gz",
    "platform": null,
    "description": "# \ud83d\udd75 Multi-agent modeling with large language models \n<!-- [![PyPi](https://img.shields.io/badge/Official%20Website-agpt.co-blue?style=flat&logo=world&logoColor=white)](https://agpt.co) -->\n[![PyPi](https://shields.io/pypi/v/surv-ai)](https://pypi.org/project/surv-ai/)\n[![PyPi](https://shields.io/pypi/pyversions/surv-ai)](https://pypi.org/project/surv-ai/)\n[![Unit Tests](https://shields.io/github/actions/workflow/status/DanielBalsam/surv_ai/.github/workflows/ci.yaml?branch=main)](https://github.com/DanielBalsam/surv_ai/actions/workflows/ci.yaml)\n[![GitHub Repo stars](https://img.shields.io/github/stars/DanielBalsam/surv_ai?style=social)](https://github.com/DanielBalsam/surv_ai/stargazers)\n\n\n**`surv_ai` is a large language model framework designed for multi-agent modeling. This allows large-language models to be used as engines to power research into predictive modeling, bias analysis, and other forms of comparative analysis.**\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/midterms_2022.png)\n\n**Multi-agent modeling** involves generating statistical models based on the actions of multiple agents. In our case, these models are crafted by agents querying and processing text from a comprehensive data corpus. The agents then reason with the information extracted to produce a data point.\n\nGiven the stochastic nature of large language models, individual data points may vary. However, if a sufficient number of agents are employed, the generated models can effectively support comparative analysis.\n\nThey key idea here was inspired by the [technique of bagging (bootstrap aggregating) in classical ensemble classifiers.](https://en.wikipedia.org/wiki/Bootstrap_aggregating) The basic idea is that if you have a lot of weak learners with only limited information - in aggregate those weak learners can produce much higher quality results than a single weak learner with maximum information (or a bunch a weak learners with maximum information).\n\nThis technique appears to work better when each agent has access to a limited subset of the total information even when using language models with larger context windows.\n\nThe abstraction in this repository capable of producing a multi-agent data point is known as a `Survey`. A `Survey` takes a statement as input and returns the percentage of agents that agreed with it.\n\nA more complex abstraction, called a `Model`, enables the variation of input parameters into a `Survey` to create a nuanced multi-agent model.\n\nThe data points produced serve as sentiment analysis against a text corpus, making them susceptible to the biases of both the large language models and the corpus itself. However, if the data source provided is quality - then the model produced can be high quality as well.\n\nAs advances in large language models and AI continue, multi-agent modeling may remain a valuable framework for classification and regression models. It also has the potential to become a useful data point for researchers investigating complex issues with numerous underlying variables.\n\n## \ud83d\udcf2 Installation \n\nPackage is available on [PyPi](https://pypi.org/project/surv-ai/): \n\n```\npip install surv-ai\n```\n\n## \ud83d\udcfb Contact \n\nFor all inquiries, contact me at: daniel.balsam@survai.org\n\n## \u2705 Responsible use \n\nThe examples provided below are intended merely to illustrate potential applications of this framework; they are not to be considered scientifically rigorous.\n\nApproaches like this have potential in guiding research and decision-making. However, it is crucial to rely on a diverse range of data points and to interpret each model cautiously.\n\nI am always open to suggestions for further enhancing this approach.\n\n## \ud83d\udcdd Basic usage \n\nThe two key abstractions in this repository are a `Survey` and a `Model`. \n\nExecuting the method `Survey.conduct` with a hypothesis will spin up a number of agents and seed them with some base knowledge. The agents are then asked to to assign a true or false value to the hypothesis provided. \n\n```\nclass Survey:\n    def __init__(\n        self,\n        client: LargeLanguageModelClientInterface,\n        tool_belt: ToolBeltInterface,\n        n_agents=10,\n        max_concurrency=10,\n        max_knowledge_per_agent=3,\n    ):\n        ...\n\n    async def conduct(self, hypothesis: str) -> SurveyResponse:\n        ...\n```\n\nA `Model` is an abstraction that allows one to conduct many surveys while changing some independent variable.\n\nFor example: one could could change the publish date of news articles that should be considered in the survey. \n\n```\nclass Model:\n    def __init__(\n        self,\n        survey_class: type[SurveyInterface],\n        parameters: list[SurveyParameter],\n        max_concurrency: int = 1,\n    ):\n        ...\n\n    async def build(self, hypothesis: str) -> list[DataPoint]:\n        ...\n```\n\nAll abstractions implemented in this repository adhere to simple abstract interfaces - so you can easily build your own agents, surveys, and models.\n\n## \ud83c\udf93 Examples \n\nAll the below examples will be conducted with either GPT or Claude, and Google Custom Search. Links to relevant docs to get your own API keys:\n\n[**OpenAI Docs**](https://platform.openai.com/) / \n[**Anthropic Docs**](https://console.anthropic.com/docs) / \n[**Google Custom Search Docs**](https://developers.google.com/custom-search/v1/overview)\n\n### Comparing against a ground truth\n\nLet's start by establishing the system's ability to figure out if information is true.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Survey,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge\n)\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ntool_belt = ToolBelt(\n    tools=[\n        GoogleCustomSearchTool(\n            google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n            google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n            start_date=\"2023-01-01\",\n            end_date=\"2023-05-01\",\n            n_pages=10,\n        )\n    ]\n)\nbase_knowledge = [\n    Knowledge(\n        text=\"It is currently 2023/05/01, all the articles are from 2023.\",\n        source=\"Additional context\",\n    ),\n]\n\nsurvey = Survey(\n    client=client,\n    tool_belt=tool_belt,\n    base_knowledge=base_knowledge,\n    max_knowledge_per_agent=3,\n    n_agents=10,\n)\n\nawait survey.conduct(\n    \"California experienced a significant amount of rainfall this winter.\",\n)  # This should always returns high a confidence agreement.\n```\n\nExecuting this code should yield an output similar to: `SurveyResponse(in_favor=10, against=0, undecided=0, error=0, percent_in_favor=1.0, uncertainty=0.0)`\n\nIt's important to recognize that occasionally, one or two agents might incorrectly respond to a straightforward question like this. That's why we employ multiple agents in these systems - to counteract the randomness inherent in Large Language Models (LLMs).\n\nTo further test our hypothesis, we could also assert the contrary position and ensure we obtain the opposite value.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Survey,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge\n)\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ntool_belt = ToolBelt(\n    tools=[\n        GoogleCustomSearchTool(\n            google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n            google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n            start_date=\"2023-01-01\",\n            end_date=\"2023-05-01\",\n            n_pages=10,\n        )\n    ]\n)\nbase_knowledge = [\n    Knowledge(\n        text=\"It is currently 2023/05/01, all the articles are from 2023.\",\n        source=\"Additional context\",\n    ),\n]\n\nsurvey = Survey(\n    client=client,\n    tool_belt=tool_belt,\n    base_knowledge=base_knowledge,\n    max_knowledge_per_agent=3,\n    n_agents=10,\n)\n\nawait survey.conduct(\n    \"California experienced little rainfall this winter.\",\n)  # This should always returns a high confidence disagreement.\n```\n\nExecuting this code should result in an output similar to: `SurveyResponse(in_favor=0, against=10, undecided=0, error=0, percent_in_favor=0.0, uncertainty=0.0)`\n\n### Comparing changes in sentiment over time\n\nOne thing we can use this tool for is measuring changes in sentiment overtime. GPT's training data ends in late 2021, so one way we can test our models with GPT is by looking at events that happened after GPT's training cutoff.\n\nFor instance, we can plot how sentiment regarding the United States' 2022 Midterm Elections evolved in the months leading up to it:\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Model,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge,\n    Survey,\n    SurveyParameter\n)\n\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ndef build_parameter(date_range: tuple[str, str]):\n    tool_belt = ToolBelt(\n        tools=[\n            GoogleCustomSearchTool(\n                google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n                google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n                n_pages=30,\n                start_date=date_range[0],\n                end_date=date_range[1],\n                max_concurrency=20,\n            ),\n        ],\n    )\n    base_knowledge = [\n        Knowledge(\n            text=f\"It is currently {date_range[0]}. The included articles were published between {date_range[0]} and {date_range[1]}\",\n            source=\"Additional context\",\n        ),\n    ]\n    return SurveyParameter(\n        independent_variable=date_range[1],\n        kwargs={\n            \"client\": client,\n            \"n_agents\": 100,\n            \"max_knowledge_per_agent\":5,\n            \"max_concurrency\": 20,\n            \"tool_belt\": tool_belt,\n            \"base_knowledge\": base_knowledge,\n        },\n    )\n\ndate_ranges = [\n    ('2022-05-01', '2022-06-01'),\n    ('2022-06-01', '2022-07-01'),\n    ('2022-07-01', '2022-08-01'),\n    ('2022-08-01', '2022-09-01'),\n    ('2022-09-01', '2022-10-01'),\n    ('2022-10-01', '2022-11-05'),\n]\n\nmodel = Model(\n    Survey,\n    parameters=[build_parameter(date_range) for date_range in date_ranges],\n)\n\ndemocrat_results = await model.build(\n    \"Democrats are favored to maintain control of the Senate in the 2022 November Midterm elections.\",\n)\n```\n\nWe can also plot the inverted statement and observe opposite trend lines:\n\n```\nrepublican_results = await model.build(\n    \"Republicans are favored to maintain control of the Senate in the 2022 November Midterm elections.\",\n)\n```\n\nWhen compared with a leading model in political opinion polling, our model is presented as follows:\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/midterms_2022.png)\n\n*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com. [FiveThirtyEight data can be found here.](https://projects.fivethirtyeight.com/2022-election-forecast/senate/)*\n\nPretty cool! Note that the outputs don't represent complementary probabilities due to nuances in how the models reason, and biases of the models and data sources, but we can observe trend lines that do mirror each other quite well.\n\nAnother example could involve plotting sentiments about the economy and using fluctuations in the S&P 500 as a benchmark for accuracy.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Model,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge,\n    Survey,\n    SurveyParameter\n)\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ndef build_parameter(date_range: tuple[str, str]):\n    tool_belt = ToolBelt(\n        tools=[\n            GoogleCustomSearchTool(\n                google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n                google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n                n_pages=20,\n                start_date=date_range[0],\n                end_date=date_range[1]\n            ),\n        ],\n    )\n    base_knowledge = [\n        Knowledge(\n            text=f\"It is currently {date_range[0]}. The included articles were published between {date_range[0]} and {date_range[1]}\",\n            source=\"Additional context\",\n        ),\n    ]\n    return SurveyParameter(\n        independent_variable=date_range[1],\n        kwargs={\n            \"client\": client,\n            \"n_agents\": 100,\n            \"max_knowledge_per_agent\":5,\n            \"max_concurrency\": 10,\n            \"tool_belt\": tool_belt,\n            \"base_knowledge\": base_knowledge,\n        },\n    )\n\ndate_ranges = [\n    ('2021-09-01', '2022-01-01'),\n    ('2022-01-01', '2022-03-01'),\n    ('2022-03-01', '2022-06-01'),\n    ('2022-06-01', '2022-09-01'),\n    ('2022-09-01', '2023-01-01'),\n    ('2023-01-01', '2023-03-01'),\n    ('2023-03-01', '2023-06-01'),\n]\n\nmodel = Model(\n    Survey,\n    parameters=[build_parameter(date_range) for date_range in date_ranges],\n)\n\nresults = await model.build(\n    \"The United States economy looks like it is heading for a recession.\",\n)\n```\n\nThis gives us the following graph:\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/S&P.png)\n\n*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com. Please note that it is the complement of multi-agent model that is plotted. [S&P data can be found here.](https://www.google.com/finance/quote/.INX:INDEXSP?sa=X&ved=2ahUKEwi2hJWyg6D_AhXiGFkFHdqHCIYQ3ecFegQIIRAf)*\n\n### Measuring bias in a data corpus\n\nA promising application of this technique is observing bias within a text corpus. For instance, we could create a model that uses different news sites as its independent variable to explore how the agents' conclusions might vary based on the data source utilized.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Model,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge,\n    Survey,\n    SurveyParameter\n)\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ndef build_parameter(news_source: str):\n    tool_belt = ToolBelt(\n        tools=[\n            GoogleCustomSearchTool(\n                google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n                google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n                n_pages=20,\n                start_date=\"2023-05-01\",\n                end_date=\"2023-06-01\",\n                only_include_sources=[news_source]\n            ),\n        ],\n    )\n    base_knowledge = [\n        Knowledge(\n            text=f\"It is currently 2023-06-01. The included articles were published between 2023-05-01 and 2023-06-01\",\n            source=\"Additional context\",\n        ),\n    ]\n    return SurveyParameter(\n        independent_variable=news_source,\n        kwargs={\n            \"client\": client,\n            \"n_agents\": 100,\n            \"max_knowledge_per_agent\":10,\n            \"max_concurrency\": 10,\n            \"tool_belt\": tool_belt,\n            \"base_knowledge\": base_knowledge,\n        },\n    )\n\nnews_sources = [\n    \"nytimes.com\",\n    \"cnn.com\",\n    \"wsj.com\",\n    \"foxnews.com\",\n]\n\nmodel = Model(\n    Survey,\n    parameters=[build_parameter(news_source) for news_source in news_sources],\n)\nresults = await model.build(\n    \"Republicans are responsible for the impending debt ceiling crisis.\"\n)\n```\n\nThis provides us with a scatter plot representing the above-mentioned news sources:\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/debt_ceiling.png)\n\n*In this example, for each news site the agents looked only at articles published in May of 2023. Omitted publications did not have enough articles on the topic published to get reliable results.*\n\n### Measuring biases in different large language models\n\nAnother promising method for measuring bias involves comparing biases across various large language models, provided they all receive identical input parameters.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    AnthropicClient,\n    Model,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge,\n    Survey,\n    SurveyParameter,\n    LargeLanguageModelClientInterface\n)\n\nclients = [AnthropicClient(os.environ[\"ANTHROPIC_API_KEY\"]), GPTClient(os.environ[\"OPEN_AI_API_KEY\"])]\n\ndef build_parameter(client: LargeLanguageModelClientInterface):\n    tool_belt = ToolBelt(\n        tools=[\n            GoogleCustomSearchTool(\n                google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n                google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n                n_pages=20,\n                start_date=\"2023-01-01\",\n                end_date=\"2024-05-01\",\n                max_concurrency=3,\n            )\n        ],\n    )\n    base_knowledge = [\n        Knowledge(\n            text=f\"It is currently 2023-06-01. The included articles were published between 2023-01-01 and 2023-06-01\",\n            source=\"Additional context\",\n        ),\n    ]\n    return SurveyParameter(\n        independent_variable=client.__class__.__name__,\n        kwargs={\n            \"client\": client,\n            \"n_agents\": 100,\n            \"max_knowledge_per_agent\":3,\n            \"max_concurrency\": 3,\n            \"tool_belt\": tool_belt,\n            \"base_knowledge\": base_knowledge,\n        },\n    )\n\nmodel = Model(\n    Survey,\n    parameters=[build_parameter(client) for client in clients],\n)\nresults = await model.build(\n    \"OpenAI has been irresponsible in their handling of AI technology.\"\n)\n```\n\nWhen we compare the results between Anthropic and OpenAI's models, the scatter plot appears as follows:\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/openai.png)\n\n*In this example, the agents crawled websites such as nytimes.com, wsj.com, abcnews.com, cnn.com, bloomberg.com, foxnews.com, economist.com, washingtonpost.com, and nbcnews.com for articles published in the first half of 2023.*\n\n\n### Measuring sentiment on Twitter\n\nIn addition to querying news sources we can also query social media websites for sentiment. There are many uses for this, but one simple and fun one would be to try and see if we can predict who will win the Academy Award for best picture.\n\n```\nfrom surv_ai import (\n    GPTClient,\n    ToolBelt,\n    TwitterTool,\n    Knowledge,\n    Survey,\n)\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\nfilms = [\n    \"Everything Everywhere All At Once\",\n    \"All Quiet On the Western Front\",\n    \"The Banshees of Inisherin\",\n    \"The Fablemans\",\n    \"T\u00e1r\",\n    \"Top Gun: Maverick\",\n    \"Triangle of Sadness\",\n    \"Women Talking\",\n]\n\nsurvey = Survey(\n    client,\n    n_agents=200,\n    tool_belt=ToolBelt(\n        tools=[\n            TwitterTool(\n                start_date=\"2023-01-01\",\n                end_date=\"2023-03-11\",\n                n_tweets=1000,\n            )\n        ]\n    ),\n    max_knowledge_per_agent=10,\n    base_knowledge=[\n        Knowledge(text=\"It is currently March 11th, 2023 and the Oscar's are tomorrow.\", source=\"Additional context\")\n    ]\n)\n\nresults = {}\nfor film in films:\n    results[film] = await survey.conduct(f\"{film} is likely to win the Academy Award for Best Picture.\")\n```\n\nPlacing our results on a scatter plot, we get the correct result!\n\n![](https://raw.githubusercontent.com/DanielBalsam/surv_ai/main/examples/best_picture.png)\n*In this example, the agents were allowed to read Tweets up between January 1st 2023, and March 11th 2023 (the day before the Oscar's).*\n\nSentiment on Twitter will not always be perfectly predictive of who will win an Academy Award, but it is an interesting way to try and predict outcomes.\n\n## \ud83e\udde0 Tips \n\nUltimately, a `Survey` is powered by a Large Language Model (LLM), which means that the survey hypothesis might require tuning, much like the general need to tune prompts. Here are some insights on crafting hypotheses.\n\nIn these systems, any ambiguity in the original hypothesis can lead to unexpected results. This often happens because the agents interpret the statement too literally, thus rejecting the precise phrasing of the statement.\n\nAnother useful tactic involves seeding base knowledge to the agents, which provides extra context to the problem. To revisit a previous example:\n\n```\nfrom surv_ai import (\n    GPTClient,\n    Survey,\n    ToolBelt,\n    GoogleCustomSearchTool,\n    Knowledge\n)\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\n\ntool_belt = ToolBelt(\n    tools=[\n        GoogleCustomSearchTool(\n            google_api_key=os.environ[\"GOOGLE_API_KEY\"],\n            google_search_engine_id=os.environ[\"GOOGLE_SEARCH_ENGINE_ID\"],\n            start_date=\"2023-01-01\",\n            end_date=\"2023-05-01\",\n            n_pages=10,\n        )\n    ]\n)\nbase_knowledge = [\n    Knowledge(\n        text=\"It is currently 2023/05/01, all the articles are from 2023.\",\n        source=\"Additional context\",\n    ),\n]\n\nsurvey = Survey(\n    client=client,\n    tool_belt=tool_belt,\n    base_knowledge=base_knowledge,\n    max_knowledge_per_agent=3,\n    n_agents=10,\n)\n\nawait survey.conduct(\n    \"California experienced a significant amount of rainfall this winter.\",\n)  # This should always returns high a confidence agreement.\n```\n\nThe base knowledge assists in keeping the agents on track. As GPT's training data concludes in 2021, without an additional prompt, GPT might assume it's still 2021, which would prevent it from evaluating the statement's validity accurately.\n\nA bit of trial and error is often necessary, but you can debug more efficiently by reading the output of the agents. This is made possible by setting the log level:\n\n```\nfrom lib.log import logger, AgentLogLevel\n\nlogger.set_log_level(AgentLogLevel.OUTPUT)  # Output from the agents will be logged\nlogger.set_log_level(AgentLogLevel.INTERNAL)  # Agent internal \"thoughts\" will be logged\n```\n\nYou may also prompt agents directly:\n\n```\nfrom surv_ai import (\n    GPTClient,\n    ReasoningAgent,\n    Knowledge\n)\n\nclient = GPTClient(os.environ[\"OPEN_AI_API_KEY\"])\nagent = ReasoningAgent(client)\n\nagent.teach_text(\"You can provide text information to an agent using `teach_text`.\", \"Context\")\nagent.teach_knowledge(\n    Knowledge(\n        text=\"You can similarly pass a knowledge object directly using `teach_knowledge`\",\n        source=\"Context\"\n    )\n)\n\nagent.prompt(\"There are multiple ways to teach an agent knowledge.\")\n```\n\nIf you are noticing a large number of errors, you may be hitting rate limits in your LLM API - you can get around this by adjusting the `max_concurrency` parameter in both your tools and `Survey`.\n\n## \ud83e\udd29 Inspiration \n\nThis project was inspired by numerous innovative projects and recent papers. Some of the inspirations for this project include:\n\n1. [Generative Agents: Interactive Simulacra of Human Behavior](https://arxiv.org/abs/2304.03442).\n2. [Large Language Models are Zero-Shot Reasoners](https://arxiv.org/abs/2205.11916)\n3. [AutoGPT](https://github.com/Significant-Gravitas/Auto-GPT)\n\nAdditionally, thanks go to the multitude of researchers and engineers out there who are contributing to unlocking the power of these models!\n\n## \ud83d\udcc8 Next steps \n\nA few directions I plan to explore with this project include:\n\n1. Use the AmbiFC fact-checking data set to benchmark the approach of this framework and compare all future changes against this benchmark: https://paperswithcode.com/paper/evidence-based-verification-for-real-world\n2. More documentation and use guides!\n3. Kicking off a development blog.\n4. More integrations with various helpful tools.\n5. More examples and use cases.\n6. Experimentation to further optimize performance.\n\n## \ud83e\udd1d Contribute \n\nIf you'd like to contribute then please reach out!\n",
    "bugtrack_url": null,
    "license": "MIT",
    "summary": "A framework for multi-agent modeling using large language models",
    "version": "0.2.0",
    "project_urls": {
        "Documentation": "https://github.com/DanielBalsam/surv_ai/blob/main/README.md",
        "Homepage": "https://github.com/DanielBalsam/surv_ai",
        "Repository": "https://github.com/DanielBalsam/surv_ai"
    },
    "split_keywords": [
        "ai",
        "artificial intelligence",
        "data science",
        "statistics",
        "models",
        "llm",
        "agents",
        "survai",
        "surv_ai",
        "survey",
        "multi-agent",
        "large language model",
        "artificial intelligence",
        "machine learning",
        "natural language processing",
        "nlp",
        "sentiment analysis"
    ],
    "urls": [
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "c1cb4c2968e2f0691aa5a60b109a6431a7d5f11c3f5dde1900410d26814fe95b",
                "md5": "2526f077c070dc2aeb79d67baa3d2998",
                "sha256": "2c1cb9768fd7a7058ab8810b22081e0cff7db41fd73bd2547e7566db823fc087"
            },
            "downloads": -1,
            "filename": "surv_ai-0.2.0-py3-none-any.whl",
            "has_sig": false,
            "md5_digest": "2526f077c070dc2aeb79d67baa3d2998",
            "packagetype": "bdist_wheel",
            "python_version": "py3",
            "requires_python": ">=3.9,<4.0",
            "size": 30744,
            "upload_time": "2023-06-13T03:25:27",
            "upload_time_iso_8601": "2023-06-13T03:25:27.512455Z",
            "url": "https://files.pythonhosted.org/packages/c1/cb/4c2968e2f0691aa5a60b109a6431a7d5f11c3f5dde1900410d26814fe95b/surv_ai-0.2.0-py3-none-any.whl",
            "yanked": false,
            "yanked_reason": null
        },
        {
            "comment_text": "",
            "digests": {
                "blake2b_256": "20f1a0ab7c7feb5de5c880dabd439c53a1665360c333aa62ae923fc4c6bd62dd",
                "md5": "fcad92f3f8cc39b6df4a54740a7e7d63",
                "sha256": "ec737921852158de82a5adce542d021fe93092072022dfd077276e00b638b0c0"
            },
            "downloads": -1,
            "filename": "surv_ai-0.2.0.tar.gz",
            "has_sig": false,
            "md5_digest": "fcad92f3f8cc39b6df4a54740a7e7d63",
            "packagetype": "sdist",
            "python_version": "source",
            "requires_python": ">=3.9,<4.0",
            "size": 27130,
            "upload_time": "2023-06-13T03:25:29",
            "upload_time_iso_8601": "2023-06-13T03:25:29.584035Z",
            "url": "https://files.pythonhosted.org/packages/20/f1/a0ab7c7feb5de5c880dabd439c53a1665360c333aa62ae923fc4c6bd62dd/surv_ai-0.2.0.tar.gz",
            "yanked": false,
            "yanked_reason": null
        }
    ],
    "upload_time": "2023-06-13 03:25:29",
    "github": true,
    "gitlab": false,
    "bitbucket": false,
    "codeberg": false,
    "github_user": "DanielBalsam",
    "github_project": "surv_ai",
    "travis_ci": false,
    "coveralls": false,
    "github_actions": true,
    "lcname": "surv-ai"
}
        
Elapsed time: 0.09252s