mirror of
https://github.com/NovaOSS/nova-api.git
synced 2024-11-25 18:53:58 +01:00
Update keys.py & add full support for rate limits per key, different base urls per key and provider name
This commit is contained in:
parent
23533b4aa3
commit
41a47cdbdc
55
api/keys.py
55
api/keys.py
|
@ -1,6 +1,7 @@
|
|||
import json
|
||||
import os
|
||||
import random
|
||||
import time
|
||||
from pymongo import MongoClient
|
||||
|
||||
with open('./config.json', 'r') as file:
|
||||
|
@ -19,18 +20,26 @@ class Keys:
|
|||
db = client.get_database('keys_db')
|
||||
collection = db['keys']
|
||||
|
||||
def __init__(self, key: str, model: str):
|
||||
def __init__(self, key: str, model: str, provider_name: str, ratelimit: int, url: str):
|
||||
self.key = key
|
||||
self.model = model
|
||||
self.provider_name = provider_name
|
||||
self.ratelimit = ratelimit
|
||||
self.url = url
|
||||
if not Keys.cache:
|
||||
self._load_keys()
|
||||
|
||||
def _load_keys(self) -> None:
|
||||
cursor = Keys.collection.find({}, {'_id': 0, 'key_value': 1, 'model': 1})
|
||||
cursor = Keys.collection.find({}, {'_id': 0, 'key_value': 1, 'model': 1, 'provider_name': 1, 'ratelimit': 1, 'url': 1, 'last_used': 1})
|
||||
for doc in cursor:
|
||||
key_value = doc['key_value']
|
||||
model = doc['model']
|
||||
Keys.cache.setdefault(model, set()).add(key_value)
|
||||
provider_name = doc['provider_name']
|
||||
ratelimit = doc['ratelimit']
|
||||
url = doc['url']
|
||||
last_used = doc.get('last_used', 0)
|
||||
key_data = {'provider_name': provider_name, 'ratelimit': ratelimit, 'url': url, 'last_used': last_used}
|
||||
Keys.cache.setdefault(model, {}).setdefault(key_value, key_data)
|
||||
|
||||
def lock(self) -> None:
|
||||
self.locked_keys.add(self.key)
|
||||
|
@ -43,15 +52,27 @@ class Keys:
|
|||
|
||||
@staticmethod
|
||||
def get(model: str) -> str:
|
||||
key_candidates = list(Keys.cache.get(model, set()))
|
||||
key_candidates = list(Keys.cache.get(model, {}).keys())
|
||||
random.shuffle(key_candidates)
|
||||
|
||||
for key_candidate in key_candidates:
|
||||
key = Keys(key_candidate, model)
|
||||
current_time = time.time()
|
||||
|
||||
if not key.is_locked():
|
||||
for key_candidate in key_candidates:
|
||||
key_data = Keys.cache[model][key_candidate]
|
||||
key = Keys(key_candidate, model, key_data['provider_name'], key_data['ratelimit'], key_data['url'])
|
||||
time_since_last_used = current_time - key_data.get('last_used', 0)
|
||||
|
||||
if not key.is_locked() and time_since_last_used >= key.ratelimit:
|
||||
key.lock()
|
||||
return key.key
|
||||
key_data['last_used'] = current_time # Update last_used in the cache
|
||||
Keys.collection.update_one(
|
||||
{'key_value': key.key, 'model': key.model},
|
||||
{'$set': {'last_used': current_time}} # Update last_used in the database
|
||||
)
|
||||
return {
|
||||
'url': key.url,
|
||||
'key_value': key.key
|
||||
}
|
||||
|
||||
print(f"[WARN] No unlocked keys found for model '{model}' in get keys request!")
|
||||
|
||||
|
@ -59,17 +80,25 @@ class Keys:
|
|||
Keys.collection.delete_one({'key_value': self.key, 'model': self.model})
|
||||
# Update cache
|
||||
try:
|
||||
Keys.cache[self.model].remove(self.key)
|
||||
del Keys.cache[self.model][self.key]
|
||||
except KeyError:
|
||||
print(f"[WARN] Tried to remove a key from cache which was not present: {self.key}")
|
||||
|
||||
def save(self) -> None:
|
||||
Keys.collection.insert_one({'key_value': self.key, 'model': self.model})
|
||||
key_data = {
|
||||
'provider_name': self.provider_name,
|
||||
'ratelimit': self.ratelimit,
|
||||
'url': self.url,
|
||||
'last_used': 0 # Initialize last_used to 0 when saving a new key
|
||||
}
|
||||
Keys.collection.insert_one({'key_value': self.key, 'model': self.model, **key_data})
|
||||
# Update cache
|
||||
Keys.cache.setdefault(self.model, set()).add(self.key)
|
||||
Keys.cache.setdefault(self.model, {}).setdefault(self.key, key_data)
|
||||
|
||||
# Usage example:
|
||||
# os.environ['MONGO_URI'] = "mongodb://localhost:27017"
|
||||
# key_instance = Keys("example_key", "example_model")
|
||||
# key_instance = Keys("example_key", "gpt-4", "openai", "10", "https://whatever-openai-thing-is/chat/completions/")
|
||||
# key_instance.save()
|
||||
# key_value = Keys.get("example_model")
|
||||
# key_value = Keys.get("gpt-4")
|
||||
|
||||
|
||||
|
|
Loading…
Reference in a new issue