Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Feat/corcel+llama+gemini #252

Merged
merged 13 commits into from
Oct 4, 2024
Merged
Show file tree
Hide file tree
Changes from 5 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
19 changes: 19 additions & 0 deletions packages/dvilela/customs/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2024 Valory AG
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# ------------------------------------------------------------------------------
dvilelaf marked this conversation as resolved.
Show resolved Hide resolved
20 changes: 20 additions & 0 deletions packages/dvilela/customs/corcel_request/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2023-2024 Valory AG
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# ------------------------------------------------------------------------------

"""This module contains the corcel request tool."""
16 changes: 16 additions & 0 deletions packages/dvilela/customs/corcel_request/component.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
name: corcel_request
author: dvilela
version: 0.1.0
type: custom
description: A tool to make LLM requests to Corcel.
license: Apache-2.0
aea_version: '>=1.0.0, <2.0.0'
fingerprint:
__init__.py: bafybeicuohbwuyxqyv5ovgimdz326ivvvzatnw7mczil5n3tktur56cw6u
corcel_request.py: bafybeidvrt7sice2wuubmolzpjfitnzgminxznrz3wjz7yv47kszgrt4be
fingerprint_ignore_patterns: []
entry_point: corcel_request.py
callable: run
dependencies:
requests:
version: ==2.28.1
138 changes: 138 additions & 0 deletions packages/dvilela/customs/corcel_request/corcel_request.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,138 @@
import requests
0xArdi marked this conversation as resolved.
Show resolved Hide resolved
import re
import json
from typing import Optional, Dict, Any, Tuple

PREDICTION_OFFLINE_PROMPT = """
You are an LLM inside a multi-agent system that takes in a prompt of a user requesting a probability estimation
for a given event. You are provided with an input under the label "USER_PROMPT". You must follow the instructions
under the label "INSTRUCTIONS". You must provide your response in the format specified under "OUTPUT_FORMAT".

INSTRUCTIONS
* Read the input under the label "USER_PROMPT" delimited by three backticks.
* The "USER_PROMPT" specifies an event.
* The event will only have two possible outcomes: either the event will happen or the event will not happen.
* If the event has more than two possible outcomes, you must ignore the rest of the instructions and output the response "Error".
* You must provide a probability estimation of the event happening, based on your training data.
* You are provided an itemized list of information under the label "ADDITIONAL_INFORMATION" delimited by three backticks.
* You can use any item in "ADDITIONAL_INFORMATION" in addition to your training data.
* If an item in "ADDITIONAL_INFORMATION" is not relevant, you must ignore that item for the estimation.
* You must provide your response in the format specified under "OUTPUT_FORMAT".
* Do not include any other contents in your response.

USER_PROMPT:
```
{user_prompt}
```

ADDITIONAL_INFORMATION:
```
```

OUTPUT_FORMAT
* Your output response must be only a single JSON object to be parsed by Python's "json.loads()".
* The JSON must contain four fields: "p_yes", "p_no", "confidence", and "info_utility".
* Each item in the JSON must have a value between 0 and 1.
- "p_yes": Estimated probability that the event in the "USER_PROMPT" occurs.
- "p_no": Estimated probability that the event in the "USER_PROMPT" does not occur.
- "confidence": A value between 0 and 1 indicating the confidence in the prediction. 0 indicates lowest
confidence value; 1 maximum confidence value.
- "info_utility": Utility of the information provided in "ADDITIONAL_INFORMATION" to help you make the prediction.
0 indicates lowest utility; 1 maximum utility.
* The sum of "p_yes" and "p_no" must equal 1.
* Output only the JSON object. Do not include any other contents in your response.
* This is incorrect:"```json{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}```"
* This is incorrect:```json"{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}"```
* This is correct:"{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}"
"""

CORCEL_URL = "https://api.corcel.io/v1/chat/completions"

AVAILABLE_TOOLS = ["corcel-prediction", "corcel-completion"]

DEFAULT_VALUES = {
"model": "llama-3",
"temperature": 0.1,
"max_tokens": 500
}


def send_corcel_request(api_key: str, prompt: str, **kwargs) -> str:
"""Makes a request to Corcel API"""

payload = {key: kwargs.get(key, default_value) for key, default_value in DEFAULT_VALUES.items()}

payload["messages"] = [
{
"role": "system",
"content": prompt
}
]

headers = {
"accept": "application/json",
"content-type": "application/json",
"Authorization": api_key
}

response = requests.post(CORCEL_URL, json=payload, headers=headers, timeout=60)
return response.text


def error_response(msg: str) -> Tuple[str, None, None, None]:
"""Return an error mech response."""
return msg, None, None, None


def response_post_process(response: str, tool_name: str) -> str:
"""Loads the prediction into a json object"""

# Join all the response pieces of content
content_regex = r'\{"content":\s"(.*?)(?:",\s|"})'
matches = re.findall(content_regex, response, re.DOTALL)
joined = ("").join(matches)

# Clean and parse the response
clean_response = joined.replace('\\n', '').replace('"', '').replace('\\', '')

if tool_name != "corcel-prediction":
return clean_response

# Add missing quotes to the json keys
clean_response = re.sub(r'(\w+):', r'"\1":', clean_response)

# Load the response
try:
json_response = json.loads(clean_response)
return json.dumps(json_response)
except json.JSONDecodeError:
return f"Error: response could not be properly postprocessed: {clean_response}"


def run(**kwargs) -> Tuple[Optional[str], Optional[Dict[str, Any]], Any, Any]:
dvilelaf marked this conversation as resolved.
Show resolved Hide resolved
"""Run the task"""

api_key = kwargs.get("api_keys", {}).get("corcel", None)
tool_name = kwargs.get("tool", None)
prompt = kwargs.get("prompt", None)

if api_key is None:
return error_response("Corcel API key is not available.")

if tool_name is None:
return error_response("No tool name has been specified.")

if tool_name not in AVAILABLE_TOOLS:
return error_response(f"Tool {tool_name} is not an available tool [{AVAILABLE_TOOLS}].")

if prompt is None:
return error_response("No prompt has been given.")

if tool_name == "corcel-prediction":
kwargs["prompt"] = PREDICTION_OFFLINE_PROMPT.format(user_prompt=prompt)

response = send_corcel_request(api_key=api_key, **kwargs)

response = response_post_process(response, tool_name)

return response, prompt, None, None
20 changes: 20 additions & 0 deletions packages/dvilela/customs/gemini_prediction/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2023-2024 Valory AG
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# ------------------------------------------------------------------------------

"""This module contains the Gemini prediction tool."""
16 changes: 16 additions & 0 deletions packages/dvilela/customs/gemini_prediction/component.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
name: gemini_prediction
author: dvilela
version: 0.1.0
type: custom
description: A tool to make LLM prediction requests to Gemini.
license: Apache-2.0
aea_version: '>=1.0.0, <2.0.0'
fingerprint:
__init__.py: bafybeictllsxej6cl3hhrsfyroyiqcp5ian2wayajwpesbyjwnf5gwinxq
gemini_prediction.py: bafybeichwqlsx4ttthccdxe7zobjy3oexz5l4btkjxod7clin7avy5yf4i
fingerprint_ignore_patterns: []
entry_point: gemini_prediction.py
callable: run
dependencies:
google-generativeai:
version: ==0.5.3
102 changes: 102 additions & 0 deletions packages/dvilela/customs/gemini_prediction/gemini_prediction.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,102 @@
import re
import json
from typing import Optional, Dict, Any, Tuple
import google.generativeai as genai

PREDICTION_OFFLINE_PROMPT = """
You are an LLM inside a multi-agent system that takes in a prompt of a user requesting a probability estimation
for a given event. You are provided with an input under the label "USER_PROMPT". You must follow the instructions
under the label "INSTRUCTIONS". You must provide your response in the format specified under "OUTPUT_FORMAT".

INSTRUCTIONS
* Read the input under the label "USER_PROMPT" delimited by three backticks.
* The "USER_PROMPT" specifies an event.
* The event will only have two possible outcomes: either the event will happen or the event will not happen.
* If the event has more than two possible outcomes, you must ignore the rest of the instructions and output the response "Error".
* You must provide a probability estimation of the event happening, based on your training data.
* You are provided an itemized list of information under the label "ADDITIONAL_INFORMATION" delimited by three backticks.
* You can use any item in "ADDITIONAL_INFORMATION" in addition to your training data.
* If an item in "ADDITIONAL_INFORMATION" is not relevant, you must ignore that item for the estimation.
* You must provide your response in the format specified under "OUTPUT_FORMAT".
* Do not include any other contents in your response.

USER_PROMPT:
```
{user_prompt}
```

ADDITIONAL_INFORMATION:
```
```

OUTPUT_FORMAT
* Your output response must be only a single JSON object to be parsed by Python's "json.loads()".
* The JSON must contain four fields: "p_yes", "p_no", "confidence", and "info_utility".
* Each item in the JSON must have a value between 0 and 1.
- "p_yes": Estimated probability that the event in the "USER_PROMPT" occurs.
- "p_no": Estimated probability that the event in the "USER_PROMPT" does not occur.
- "confidence": A value between 0 and 1 indicating the confidence in the prediction. 0 indicates lowest
confidence value; 1 maximum confidence value.
- "info_utility": Utility of the information provided in "ADDITIONAL_INFORMATION" to help you make the prediction.
0 indicates lowest utility; 1 maximum utility.
* The sum of "p_yes" and "p_no" must equal 1.
* Output only the JSON object. Do not include any other contents in your response.
* This is incorrect:"```json{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}```"
* This is incorrect:```json"{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}"```
* This is correct:"{{\n \"p_yes\": 0.2,\n \"p_no\": 0.8,\n \"confidence\": 0.7,\n \"info_utility\": 0.5\n}}"
"""

AVAILABLE_TOOLS = ["gemini-prediction", "gemini-completion"]
AVAILABLE_MODELS = ["gemini-1.5-flash", "gemini-1.5-pro"]


def error_response(msg: str) -> Tuple[str, None, None, None]:
"""Return an error mech response."""
return msg, None, None, None


def response_post_process(response: str) -> str:
"""Loads the prediction into a json object"""

try:
json_response = json.loads(response)
return json.dumps(json_response)
except json.JSONDecodeError:
return f"Error: response could not be properly postprocessed: {response}"


def run(**kwargs) -> Tuple[Optional[str], Optional[Dict[str, Any]], Any, Any]:
dvilelaf marked this conversation as resolved.
Show resolved Hide resolved
"""Run the task"""

api_key = kwargs.get("api_keys", {}).get("gemini", None)
tool_name = kwargs.get("tool", None)
prompt = kwargs.get("prompt", None)
model = kwargs.get("model", "gemini-1.5-flash")

if api_key is None:
return error_response("Gemini API key is not available.")

if tool_name is None:
return error_response("No tool name has been specified.")

if tool_name not in AVAILABLE_TOOLS:
return error_response(f"Tool {tool_name} is not an available tool [{AVAILABLE_TOOLS}].")

if prompt is None:
return error_response("No prompt has been given.")

if model not in AVAILABLE_MODELS:
return error_response(f"Model {model} is not an avaliable model: {AVAILABLE_MODELS}")

if tool_name == "gemini-prediction":
prompt = PREDICTION_OFFLINE_PROMPT.format(user_prompt=prompt)

genai.configure(api_key=api_key)
model = genai.GenerativeModel(model)
response = model.generate_content(prompt)
response = response.text

if tool_name == "gemini-prediction":
response = response_post_process(response)

return response, prompt, None, None
20 changes: 20 additions & 0 deletions packages/dvilela/customs/llama_request/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
# -*- coding: utf-8 -*-
# ------------------------------------------------------------------------------
#
# Copyright 2023-2024 Valory AG
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# ------------------------------------------------------------------------------

"""This module contains the llama request tool."""
16 changes: 16 additions & 0 deletions packages/dvilela/customs/llama_request/component.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
name: llama_request
author: dvilela
version: 0.1.0
type: custom
description: A tool to run local inference through llama-cpp.
license: Apache-2.0
aea_version: '>=1.0.0, <2.0.0'
fingerprint:
__init__.py: bafybeiger6zvjsg356hyirgjadryvqlvz22wdkzqjsivkeqe3jvr33gcde
llama_request.py: bafybeie26qv3vo3yv4fpfalpaluoyexatxtn2p5is57jhsxpkmf2fhftaa
fingerprint_ignore_patterns: []
entry_point: llama_request.py
callable: run
dependencies:
llama-cpp-python:
version: ==0.2.56
Loading
Loading