gokhanmeteerturk/adaptive-shots
Github Repo | Documentation (in progress)
Adaptive Shots is a Python package that turns your prompts into few-shot prompts on the fly.
It automatically selects the most relevant prompt&answer pairs from the database based on what worked well before. Adaptive-shots package uses reinforcement learning algorithm UCB1-Tuned and learns from user feedback to improve its selections over time by using the numerical feedback as a reward.
This package implements UCB1-Tuned, but with a contextual cosine similarity of prompts to dynamically select and rank the most relevant and well performing old prompts for a given prompt text.
Initially designed for my Intelligent Tutoring System (ITS) project, the package is generalizable for any scenario requiring adaptive, reinforcement-learning-based prompt management.
- Sentence transformers for generating vector representations.
Defaults toall-MiniLM-L6-v2
(384 dimensional dense vector space) but can be configured easily. - SQLite Vector Search with sqlite-vec. (Cosine distance is used)
- Exploration vs Exploitation. Adaptive-shots keeps experimenting with less-tried but still relevant old prompts, while focusing on maximizing the performance of generated few-shot prompts.
pip install adaptive-shots
Core feature, few-shots prompt generation, looks like this:
shot_list = db.get_best_shots(
prompt='Who is the author of Twelfth Night?',
domain='geography',limit=2
)
You can then use the returned list for message generation:
shot_list.to_messages()
// Output:
[
{"role": "user", "content": "Who wrote Hamlet?"},
{"role": "assistant", "content": "William Shakespeare"},
{"role": "user", "content": "Who wrote Oedipus Rex?"},
{"role": "assistant", "content": "Sophocles"}
]
from adaptive_shots import initialize_adaptive_shot_db
from openai import OpenAI
# Initialize database and OpenAI client
db = initialize_adaptive_shot_db('./shots.db')
client = OpenAI()
# Register initial prompts
db.register_prompt(
prompt='What is the capital of France?',
answer='Paris',
rating=9.5,
domain='geography'
)
# Generate few-shots prompt
user_query = 'What is the capital of Germany?'
few_shots_prompt, shot_list = db.create_few_shots_prompt(
prompt=user_query,
domain='geography',
limit=1
)
print(few_shots_prompt)
# This will print the following:
"""
Prompt: What is the capital of France?
Answer: Paris
Prompt: What is the capital of Germany?
Answer:
"""
# You can give this directly to your LLM, or
# alternatively, use to_messages for chat completions like this:
response = client.chat.completions.create(
model='gpt-4',
messages=[
*shot_list.to_messages(),
{"role": "user", "content": user_query},
]
)
answer = response.choices[0].message.content
# Register feedback so all used shots get rewarded:
db.register_prompt(
prompt=user_query,
answer=answer,
rating=9.0, # ideally you should receive this from the user
domain='geography',
used_shots=shot_list
)
register_prompt(prompt: str, answer: str, rating: float, domain: str, used_shots: Optional[ShotPromptsList] = None) -> None
Registers a new prompt-answer pair and updates the ratings of used shots.
Generates a few-shot prompt using the best-performing relevant examples.
Generates a one-shot prompt using the single best-matching example.
Retrieves the optimal set of shots using the UCB algorithm.
This work is licensed under the GPLv3 - see the LICENSE file for details.
If you use this package in your research, please cite:
@software{erturk2024adaptive,
author = {Ertürk, Gökhan Mete},
title = {Adaptive-Shots: A Contextual Combinatorial Bandit Approach to Few-Shot Prompt Selection},
year = {2024},
url = {https://github.com/gokhanmeteerturk/adaptive-shots}
}