mirror of
https://github.com/ltcptgeneral/cs239-caching.git
synced 2025-10-24 04:49:20 +00:00
Compare commits
33 Commits
metrics
...
bb38d1eeb4
Author | SHA1 | Date | |
---|---|---|---|
|
bb38d1eeb4 | ||
|
b865e3e168 | ||
|
db002dbdd0 | ||
|
9cd8331a9d | ||
|
583c8633f7 | ||
|
d2da61fa86 | ||
|
4197ffb0ea | ||
|
474d827de6 | ||
|
839e1b8bd7 | ||
|
31ffbc18ec | ||
|
f737271303 | ||
|
32ac22806c | ||
|
8430009f8c | ||
|
8cb333b305 | ||
|
b28decfc91 | ||
|
d57069856f | ||
|
72b72a949f | ||
|
35ea5a234f | ||
|
50b5ea0acd | ||
|
3d40e4cc6f | ||
|
9699111c11 | ||
|
ebf531f586 | ||
|
6b082e0fea | ||
|
e5d8d3c0cf | ||
|
1c30154aa7 | ||
|
33d9c0b716 | ||
|
c3340e037d | ||
|
f7903f4fea | ||
|
e9b1128826 | ||
|
a276151e0c | ||
|
f475991f50 | ||
|
9dd93d96f2 | ||
|
5dc9a99a3a |
2
.gitignore
vendored
2
.gitignore
vendored
@@ -7,6 +7,7 @@ __pycache__/
|
||||
|
||||
# Ignore database files (TinyDB JSON)
|
||||
database.json
|
||||
temp_DB.json
|
||||
|
||||
# Ignore environment variables file (if used)
|
||||
.env
|
||||
@@ -22,4 +23,3 @@ database.json
|
||||
# Ignore MacOS system files
|
||||
.DS_Store
|
||||
|
||||
|
||||
|
@@ -8,7 +8,7 @@ class Node:
|
||||
self.next = None
|
||||
self.prev = None
|
||||
|
||||
class SeiveCache(Cache):
|
||||
class SieveCache(Cache):
|
||||
def __init__(self, limit: int):
|
||||
super().__init__(limit)
|
||||
self.limit = limit # Fix: Store limit properly
|
||||
@@ -18,13 +18,13 @@ class SeiveCache(Cache):
|
||||
self.hand = None
|
||||
|
||||
def print_cache_state(self):
|
||||
print("Current cache state:")
|
||||
#print("Current cache state:")
|
||||
node = self.head
|
||||
if not node:
|
||||
print("Cache is empty.")
|
||||
#print("Cache is empty.")
|
||||
return
|
||||
for _ in range(len(self.cache)):
|
||||
print(f"Key: {node.key}, Value: {node.value}, Visited: {node.visited}")
|
||||
#print(f"Key: {node.key}, Value: {node.value}, Visited: {node.visited}")
|
||||
node = node.next
|
||||
if node == self.head:
|
||||
break
|
||||
@@ -33,20 +33,17 @@ class SeiveCache(Cache):
|
||||
if key in self.cache:
|
||||
node = self.cache[key]
|
||||
node.visited = True
|
||||
print(f"GET {key}: {node.value}")
|
||||
self.print_cache_state()
|
||||
#self.print_cache_state()
|
||||
return node.value
|
||||
print(f"GET {key}: MISS")
|
||||
self.print_cache_state()
|
||||
return None
|
||||
|
||||
def put(self, key: str, val: str) -> bool:
|
||||
print(f"PUT {key}: {val}")
|
||||
if key in self.cache:
|
||||
node = self.cache[key]
|
||||
node.value = val
|
||||
node.visited = True
|
||||
self.print_cache_state()
|
||||
#self.print_cache_state()
|
||||
return False # No eviction needed
|
||||
|
||||
new_node = Node(key, val)
|
||||
@@ -66,11 +63,10 @@ class SeiveCache(Cache):
|
||||
self.cache[key] = new_node
|
||||
if not self.hand:
|
||||
self.hand = self.head
|
||||
self.print_cache_state()
|
||||
#self.print_cache_state()
|
||||
return False
|
||||
|
||||
def invalidate(self, key: str) -> bool:
|
||||
print(f"INVALIDATE {key}")
|
||||
if key in self.cache:
|
||||
node = self.cache.pop(key)
|
||||
if node == self.head:
|
||||
@@ -81,16 +77,14 @@ class SeiveCache(Cache):
|
||||
node.next.prev = node.prev
|
||||
if node.prev:
|
||||
node.prev.next = node.next
|
||||
self.print_cache_state()
|
||||
#self.print_cache_state()
|
||||
return True
|
||||
print("INVALIDATE FAILED: Key not found")
|
||||
return False
|
||||
|
||||
def next_hand(self):
|
||||
self.hand = self.hand.next if self.hand.next else self.head
|
||||
|
||||
def evict(self):
|
||||
print("EVICTION START")
|
||||
while self.hand.visited:
|
||||
self.hand.visited = False
|
||||
self.next_hand()
|
||||
@@ -107,8 +101,7 @@ class SeiveCache(Cache):
|
||||
obj_to_evict.prev.next = obj_to_evict.next
|
||||
|
||||
del self.cache[obj_to_evict.key]
|
||||
print(f"EVICTED {obj_to_evict.key}")
|
||||
self.print_cache_state()
|
||||
#self.print_cache_state()
|
||||
|
||||
# Basic API demo for future testing
|
||||
if __name__ == "__main__":
|
16
app/cache/idealcache.py
vendored
Normal file
16
app/cache/idealcache.py
vendored
Normal file
@@ -0,0 +1,16 @@
|
||||
from .cache import Cache
|
||||
from database import get_user_profile
|
||||
|
||||
class IdealCache(Cache):
|
||||
|
||||
def __init__(self, limit: int):
|
||||
pass
|
||||
|
||||
def get(self, key):
|
||||
return get_user_profile(key)
|
||||
|
||||
def put(self, key, val):
|
||||
return False
|
||||
|
||||
def invalidate(self, key):
|
||||
return False
|
15
app/cache/nocache.py
vendored
Normal file
15
app/cache/nocache.py
vendored
Normal file
@@ -0,0 +1,15 @@
|
||||
from .cache import Cache
|
||||
|
||||
class NoCache(Cache):
|
||||
|
||||
def __init__(self, limit: int):
|
||||
pass
|
||||
|
||||
def get(self, key):
|
||||
return None
|
||||
|
||||
def put(self, key, val):
|
||||
return False
|
||||
|
||||
def invalidate(self, key):
|
||||
return False
|
52
app/cache/prefetch_cache.py
vendored
52
app/cache/prefetch_cache.py
vendored
@@ -1,11 +1,29 @@
|
||||
from .cache import BaselineCache
|
||||
from .cache import Cache
|
||||
from database import get_user_profile, get_friends
|
||||
from collections import OrderedDict
|
||||
import math
|
||||
|
||||
class PrefetchCache(BaselineCache):
|
||||
key_relations = None
|
||||
class PrefetchCache(Cache):
|
||||
limit = None
|
||||
cache = None
|
||||
|
||||
def __init__(self):
|
||||
def __init__(self, limit):
|
||||
super()
|
||||
self.key_relations = dict()
|
||||
self.limit = limit
|
||||
self.cache = OrderedDict()
|
||||
|
||||
def __eq__(self, other):
|
||||
return self.cache == other
|
||||
|
||||
def __len__(self):
|
||||
return len(self.cache)
|
||||
|
||||
def get(self, key: str) -> str:
|
||||
if key in self.cache:
|
||||
self.cache.move_to_end(key)
|
||||
return self.cache[key]
|
||||
else:
|
||||
return None
|
||||
|
||||
def put(self, key: str, val: str) -> bool:
|
||||
# LRU evict
|
||||
@@ -14,16 +32,26 @@ class PrefetchCache(BaselineCache):
|
||||
self.cache.popitem(last = False)
|
||||
evict = True
|
||||
self.cache[key] = val
|
||||
self.prefetch(key, val)
|
||||
if self.prefetch(val):
|
||||
evict = True
|
||||
|
||||
return evict
|
||||
|
||||
def prefetch(self, key: str, val: str) -> bool:
|
||||
if len(self.cache) >= self.limit and key in self.key_relations:
|
||||
self.cache[self.key_relations[key][0]] = self.key_relations[key][1]
|
||||
def prefetch(self, profile) -> bool:
|
||||
evict = False
|
||||
friends_prof = get_friends(profile["user_id"], math.ceil(self.limit*0.1))
|
||||
for i in friends_prof:
|
||||
if len(self.cache) >= self.limit:
|
||||
self.cache.popitem(last = False)
|
||||
evict = True
|
||||
self.cache[i] = friends_prof[i]
|
||||
return evict
|
||||
|
||||
def invalidate(self, key: str) -> bool:
|
||||
# basic delete invalidation, no (p)refetching
|
||||
if key in self.cache:
|
||||
del self.cache[key]
|
||||
return True
|
||||
else:
|
||||
return False
|
||||
|
||||
def set_relations(self, key: str, related_key: str, related_val: str):
|
||||
self.key_relations[key] = related_key | related_val
|
||||
return
|
20
app/cache/read_after_write_cache.py
vendored
Normal file
20
app/cache/read_after_write_cache.py
vendored
Normal file
@@ -0,0 +1,20 @@
|
||||
from .cache import BaselineCache
|
||||
from database import get_user_profile
|
||||
|
||||
class ReadAfterWriteCache(BaselineCache):
|
||||
|
||||
def __init__(self, limit):
|
||||
super().__init__( limit )
|
||||
|
||||
def invalidate(self, key: str) -> bool:
|
||||
# basic delete invalidation, but after writing, we immediately read the value and add it to the cache
|
||||
invalidated = False
|
||||
if key in self.cache:
|
||||
del self.cache[key]
|
||||
invalidated = True
|
||||
|
||||
newData = get_user_profile( key )
|
||||
self.put( key, newData )
|
||||
|
||||
return invalidated
|
||||
|
5
app/cache/tiered_cache.py
vendored
5
app/cache/tiered_cache.py
vendored
@@ -21,6 +21,11 @@ class TieredCache(BaselineCache):
|
||||
f = open(self.l2_map[key], "r")
|
||||
v = f.read()
|
||||
f.close()
|
||||
|
||||
# we will also preemptively return the value from l1 to l2:
|
||||
del self.l2_map[key]
|
||||
self.put(key, v)
|
||||
|
||||
return v
|
||||
else: # otherwise its a cache miss and return None
|
||||
return None
|
||||
|
@@ -1,7 +1,8 @@
|
||||
import os
|
||||
import yaml
|
||||
import sys
|
||||
|
||||
CONFIG_FILE = "config.yaml"
|
||||
CONFIG_FILE = sys.argv[1]
|
||||
|
||||
def load_config():
|
||||
with open(CONFIG_FILE, "r") as f:
|
||||
|
@@ -1,4 +0,0 @@
|
||||
cache_strategy: "Tiered" # Change this to "Prefetch" or "Tiered" or "Seive"
|
||||
cache_limit: 10
|
||||
l2_cache_limit: 100
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_baseline.yaml
Normal file
4
app/config_baseline.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "Baseline"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_ideal.yaml
Normal file
4
app/config_ideal.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "Ideal"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_nocache.yaml
Normal file
4
app/config_nocache.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "None"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_prefetch.yaml
Normal file
4
app/config_prefetch.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "Prefetch"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_readafterwrite.yaml
Normal file
4
app/config_readafterwrite.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "ReadAfterWrite"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_sieve.yaml
Normal file
4
app/config_sieve.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "Sieve"
|
||||
cache_limit: 50
|
||||
l2_cache_limit: 100 # unused
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
4
app/config_tiered.yaml
Normal file
4
app/config_tiered.yaml
Normal file
@@ -0,0 +1,4 @@
|
||||
cache_strategy: "Tiered"
|
||||
cache_limit: 25
|
||||
l2_cache_limit: 175
|
||||
db_file: "llmData_sns.json" # Change this to the name of any json file within the "database/datastore" folder
|
@@ -1,24 +1,53 @@
|
||||
|
||||
from tinydb import TinyDB, Query
|
||||
from config import DB_FILE
|
||||
import shutil
|
||||
|
||||
import random
|
||||
|
||||
DB_LOCATION = "database/datastore/" + DB_FILE
|
||||
|
||||
# Initialize TinyDB as a NoSQL key-value store
|
||||
db = TinyDB(DB_LOCATION)
|
||||
# We don't want to change our main DB file, so we will make a temp DB file and use that as our DB file
|
||||
shutil.copyfile( DB_LOCATION, "temp_DB.json" )
|
||||
|
||||
db = TinyDB("temp_DB.json")
|
||||
User = Query()
|
||||
|
||||
def get_user_ids():
|
||||
return [x["user_id"] for x in db.all()]
|
||||
|
||||
def get_user_friend_ids():
|
||||
user_friends = {}
|
||||
for x in db.all():
|
||||
user_friends[x["user_id"]] = x["friends"]
|
||||
return user_friends
|
||||
|
||||
def get_friends(user_id, num_friends):
|
||||
friends = {}
|
||||
curr_user = db.search(User.user_id == user_id)
|
||||
random.seed(0)
|
||||
if not curr_user:
|
||||
return {}
|
||||
sample_size = min(num_friends, len(curr_user[0]["friends"]))
|
||||
for f in random.sample(curr_user[0]["friends"], sample_size):
|
||||
friends[f] = db.search(User.user_id == user_id)[0]
|
||||
return friends
|
||||
|
||||
def get_user_profile(user_id):
|
||||
"""Fetch user profile from TinyDB"""
|
||||
result = db.search(User.user_id == user_id)
|
||||
return result[0] if result else None
|
||||
|
||||
def update_user_profile(user_id, name, followers, bio, posts, friends):
|
||||
def update_user_profile( data ):
|
||||
"""Update user profile in TinyDB"""
|
||||
db.upsert({"user_id": user_id, "name": name, "followers": followers, "bio": bio, "posts": posts, "friends": friends}, User.user_id == user_id)
|
||||
user_id = str( data["user_id"] )
|
||||
|
||||
# Basically make sure friends stay the same (for prefetching). Not great implementation, but it works
|
||||
curr_user = db.search(User.user_id == user_id)
|
||||
if( curr_user and data["friends"] == None ):
|
||||
data["friends"] = curr_user[0]["friends"]
|
||||
|
||||
db.upsert( data, User.user_id == user_id )
|
||||
|
||||
def init_db():
|
||||
"""Ensure TinyDB is initialized before FastAPI starts and prepopulate some data"""
|
||||
@@ -28,7 +57,7 @@ def init_db():
|
||||
# Prepopulate database with some sample users if empty
|
||||
if len(db) == 0:
|
||||
db.insert_multiple([
|
||||
{"user_id": "1", "name": "Alice", "followers": 100, "bio": "Love coding!", "posts": "Hello, world!"},
|
||||
{"user_id": "2", "name": "Bob", "followers": 200, "bio": "Tech enthusiast", "posts": "AI is amazing!"},
|
||||
{"user_id": "3", "name": "Charlie", "followers": 50, "bio": "Blogger", "posts": "Check out my latest post!"}
|
||||
{"user_id": "1", "name": "Alice", "followers": 100, "bio": "Love coding!", "posts": "Hello, world!", "friends": ["2"]},
|
||||
{"user_id": "2", "name": "Bob", "followers": 200, "bio": "Tech enthusiast", "posts": "AI is amazing!","friends": ["3", "1"]},
|
||||
{"user_id": "3", "name": "Charlie", "followers": 50, "bio": "Blogger", "posts": "Check out my latest post!", "friends": ["1"]}
|
||||
])
|
||||
|
File diff suppressed because it is too large
Load Diff
@@ -6,6 +6,10 @@ warnings.filterwarnings('ignore')
|
||||
import re
|
||||
import random
|
||||
import json
|
||||
from tinydb import TinyDB
|
||||
from tinydb.storages import JSONStorage
|
||||
from tinydb.middlewares import CachingMiddleware
|
||||
import math
|
||||
|
||||
HUGGINGFACEHUB_API_TOKEN = None
|
||||
os.environ["HUGGINGFACEHUB_API_TOKEN"] = HUGGINGFACEHUB_API_TOKEN
|
||||
@@ -21,13 +25,11 @@ def parse_profile(text, user_id, num_users):
|
||||
match = re.search(r"([A-Za-z ]+)\|([A-Za-z &\-!]+)\|([A-Za-z .',!?&\-]+)", text)
|
||||
name, bio, posts = match.groups()
|
||||
|
||||
# Generate mock followers count (randomized for realism)
|
||||
followers = random.randint(10, 5000)
|
||||
|
||||
# Generate mock friends (users with nearby IDs)
|
||||
friend_ids = [str(fid) for fid in range(num_users) if fid != user_id]
|
||||
friend_ids = [str(fid) for fid in range(user_id) if fid != user_id]
|
||||
random.shuffle(friend_ids)
|
||||
friends = friend_ids[:random.randint(1, num_users-1)] # Each user gets 1-5 friends
|
||||
friends = friend_ids[:random.randint(1, min(100, math.ceil(num_users/3)))]
|
||||
|
||||
return {
|
||||
"user_id": str(user_id),
|
||||
@@ -38,7 +40,7 @@ def parse_profile(text, user_id, num_users):
|
||||
"friends": friends
|
||||
}
|
||||
|
||||
def generate_data(num_users):
|
||||
def generate_data(base_id, num_users):
|
||||
system_message = """You are a data generator creating user profiles for a social media app.
|
||||
Always provide user profiles in this format: Name | Interest | Recent Activity.
|
||||
Do not include numbers, IDs, or assistant labels. Only return a properly formatted response.
|
||||
@@ -60,25 +62,30 @@ def generate_data(num_users):
|
||||
huggingfacehub_api_token=HUGGINGFACEHUB_API_TOKEN,
|
||||
)
|
||||
llm_chain = prompt | llm
|
||||
data = {}
|
||||
i = 0
|
||||
data = []
|
||||
i = base_id
|
||||
user_id = 0
|
||||
while user_id < num_users:
|
||||
raw_text = llm_chain.invoke({"user_id": i})
|
||||
while not valid_data(raw_text):
|
||||
i = i + 1
|
||||
raw_text = llm_chain.invoke({"user_id": i})
|
||||
user_profile = parse_profile(raw_text, user_id, num_users)
|
||||
user_profile = parse_profile(raw_text, base_id + user_id, num_users)
|
||||
user_id = user_id + 1
|
||||
data[user_id] = user_profile
|
||||
i = i + 1
|
||||
data.append(user_profile)
|
||||
|
||||
return data
|
||||
|
||||
if __name__ == "__main__":
|
||||
data = generate_data(100)
|
||||
base_id = input("Enter base id (check db to find the next consecutive user_id): ")
|
||||
num_users = input("Enter number of users to generate: ")
|
||||
data = generate_data(int(base_id), int(num_users))
|
||||
|
||||
# Create json file
|
||||
json_object = json.dumps( {"_default": data}, indent=4 )
|
||||
with open( "datastore/llmData_sns.json", "w" ) as f:
|
||||
f.write( json_object )
|
||||
file_path = "datastore/llmData_sns.json"
|
||||
global db
|
||||
db = TinyDB(file_path, storage=CachingMiddleware(JSONStorage), indent=4)
|
||||
db.insert_multiple(data)
|
||||
db.close()
|
||||
|
33
app/database/generate_data_basic.py
Normal file
33
app/database/generate_data_basic.py
Normal file
@@ -0,0 +1,33 @@
|
||||
import random, string
|
||||
from tinydb import TinyDB
|
||||
from tinydb.storages import JSONStorage
|
||||
from tinydb.middlewares import CachingMiddleware
|
||||
|
||||
def randomstring(length):
|
||||
letters = string.ascii_lowercase
|
||||
return ''.join(random.choice(letters) for i in range(length))
|
||||
|
||||
def generate_data(base_id, num_users):
|
||||
data = []
|
||||
for i in range(num_users):
|
||||
data.append({
|
||||
"user_id": str(base_id + i),
|
||||
"name": randomstring(10),
|
||||
"followers": random.randint(1, 10000),
|
||||
"bio": randomstring(250),
|
||||
"posts": randomstring(100),
|
||||
"friends": random.choices(range(base_id + i - 1), k=random.randint(1, 50))
|
||||
})
|
||||
|
||||
return data
|
||||
|
||||
base_id = input("Enter base id (check db to find the next consecutive user_id): ")
|
||||
num_users = input("Enter number of users to generate: ")
|
||||
data = generate_data(int(base_id), int(num_users))
|
||||
|
||||
# Create json file
|
||||
file_path = "datastore/llmData_sns.json"
|
||||
global db
|
||||
db = TinyDB(file_path, storage=CachingMiddleware(JSONStorage), indent=4)
|
||||
db.insert_multiple(data)
|
||||
db.close()
|
37
app/main.py
37
app/main.py
@@ -1,23 +1,40 @@
|
||||
from fastapi import FastAPI, HTTPException
|
||||
from database import get_user_ids, get_user_profile, update_user_profile
|
||||
from database import get_user_ids, get_user_profile, update_user_profile, get_user_friend_ids
|
||||
from cache.cache import BaselineCache
|
||||
from cache.prefetch_cache import PrefetchCache
|
||||
from cache.tiered_cache import TieredCache
|
||||
from cache.eviction_seive import SeiveCache
|
||||
from cache.eviction_sieve import SieveCache
|
||||
from cache.nocache import NoCache
|
||||
from cache.idealcache import IdealCache
|
||||
from cache.read_after_write_cache import ReadAfterWriteCache
|
||||
from config import CACHE_STRATEGY, CACHE_LIMIT, L2_CACHE_LIMIT
|
||||
from models.models import User
|
||||
import time
|
||||
|
||||
app = FastAPI()
|
||||
|
||||
# Initialize cache based on strategy from config.yaml or environment variable
|
||||
if CACHE_STRATEGY == "Baseline":
|
||||
print("Using baseline cache strategy")
|
||||
cache = BaselineCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "Prefetch":
|
||||
print("Using prefetch cache strategy")
|
||||
cache = PrefetchCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "Tiered":
|
||||
print("Using tiered cache strategy")
|
||||
cache = TieredCache(limit=CACHE_LIMIT, l2_limit=L2_CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "Seive":
|
||||
cache = SeiveCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "Sieve":
|
||||
print("Using sieve cache strategy")
|
||||
cache = SieveCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "None":
|
||||
print("Using no cache strategy")
|
||||
cache = NoCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "Ideal":
|
||||
print("Using ideal cache strategy")
|
||||
cache = IdealCache(limit=CACHE_LIMIT)
|
||||
elif CACHE_STRATEGY == "ReadAfterWrite":
|
||||
print("Using read-after-write cache strategy")
|
||||
cache = ReadAfterWriteCache(limit=CACHE_LIMIT)
|
||||
else:
|
||||
raise ValueError(f"Invalid CACHE_STRATEGY: {CACHE_STRATEGY}")
|
||||
|
||||
@@ -25,6 +42,10 @@ else:
|
||||
def fetch_user_ids():
|
||||
return {"ids": get_user_ids()}
|
||||
|
||||
@app.get("/users_and_friends")
|
||||
def fetch_user_and_friends():
|
||||
return get_user_friend_ids()
|
||||
|
||||
@app.get("/user/{user_id}")
|
||||
def fetch_user_profile(user_id: str):
|
||||
"""Fetch user profile with caching"""
|
||||
@@ -34,6 +55,7 @@ def fetch_user_profile(user_id: str):
|
||||
return {"user_id": user_id, "profile": cached_profile, "source": "cache", "time_ms": (time.time() - start) * 1000}
|
||||
|
||||
profile = get_user_profile(user_id)
|
||||
time.sleep(10 / 1000) # simulate 10 ms db delay, we do this here instead of the actual db in the ideal cache case
|
||||
if profile is None:
|
||||
raise HTTPException(status_code=404, detail="User not found")
|
||||
|
||||
@@ -41,8 +63,11 @@ def fetch_user_profile(user_id: str):
|
||||
return {"user_id": user_id, "profile": profile, "source": "database", "time_ms": (time.time() - start) * 1000}
|
||||
|
||||
@app.post("/update_user/")
|
||||
def modify_user_profile(user_id: str, name: str, followers: int, bio: str, posts: str, friends: list[str]):
|
||||
async def modify_user_profile(user_data : User):
|
||||
"""Update user profile and refresh cache"""
|
||||
update_user_profile(user_id, name, followers, bio, posts, friends)
|
||||
user_id=user_data.user_id
|
||||
user_dict = user_data.dict()
|
||||
|
||||
update_user_profile(user_dict)
|
||||
cache.invalidate(user_id) # Invalidate old cache
|
||||
return {"message": "User profile updated successfully"}
|
||||
|
9
app/models/models.py
Normal file
9
app/models/models.py
Normal file
@@ -0,0 +1,9 @@
|
||||
from pydantic import BaseModel
|
||||
|
||||
class User(BaseModel):
|
||||
user_id: str
|
||||
name: str | None = None
|
||||
followers: int | None = None
|
||||
bio: str | None = None
|
||||
posts: str | None = None
|
||||
friends: list | None = None
|
@@ -3,3 +3,4 @@ uvicorn
|
||||
tinydb
|
||||
pyyaml
|
||||
requests
|
||||
tqdm
|
71
tests/frequent_after_write.py
Normal file
71
tests/frequent_after_write.py
Normal file
@@ -0,0 +1,71 @@
|
||||
import requests
|
||||
import random
|
||||
import json
|
||||
from tqdm import tqdm
|
||||
import time
|
||||
from collections import deque
|
||||
from utils import print_report
|
||||
|
||||
baseurl = "http://localhost:8000"
|
||||
|
||||
endpoints = {
|
||||
"/user/{user_id}": 0.5, # 50% read operations
|
||||
"/update_user/?user_id={user_id}&name=Test&followers=100&bio=Updated&posts=Updated": 0.5 # 50% write operations
|
||||
}
|
||||
|
||||
# Fetch all user IDs
|
||||
user_ids = json.loads(requests.get(baseurl + "/users").content)["ids"]
|
||||
|
||||
random.seed(0)
|
||||
|
||||
prev_updated_users = deque()
|
||||
def generate_random():
|
||||
"""Randomly generate a read or write request, favoring cache hits."""
|
||||
endpoint = random.choices(list(endpoints.keys()), list(endpoints.values()))[0]
|
||||
# Reads
|
||||
if endpoint == "/user/{user_id}":
|
||||
# Favor frequently accessed user IDs to increase hit ratio
|
||||
if( prev_updated_users ):
|
||||
random_user = str(random.choice(prev_updated_users)) if random.random() < 0.7 else str(random.choice(user_ids))
|
||||
else:
|
||||
random_user = str(random.choice(user_ids))
|
||||
return baseurl + endpoint.replace("{user_id}", random_user)
|
||||
# Writes
|
||||
else:
|
||||
random_user = str(random.choice(user_ids))
|
||||
prev_updated_users.append( random_user )
|
||||
if( len( prev_updated_users ) > 10 ):
|
||||
prev_updated_users.popleft()
|
||||
return random_user
|
||||
|
||||
times = []
|
||||
hits = []
|
||||
|
||||
start = time.time()
|
||||
for i in tqdm(range(10000)):
|
||||
url = generate_random()
|
||||
|
||||
if( "user" not in url ):
|
||||
write_obj = { "user_id":url,"name": "Test", "followers":"100","bio":"updated","posts":"updated"}
|
||||
response = requests.post("http://localhost:8000/update_user/", json = write_obj)
|
||||
else:
|
||||
response = requests.get(url)
|
||||
|
||||
try:
|
||||
content = json.loads(response.content)
|
||||
|
||||
if "time_ms" in content: # Only process if "time_ms" exists
|
||||
times.append(content["time_ms"])
|
||||
hits.append(content["source"] == "cache")
|
||||
|
||||
except json.JSONDecodeError:
|
||||
print(f"Error decoding JSON: {response.content}")
|
||||
exit(1)
|
||||
except KeyError:
|
||||
print(f"Unexpected response format: {content}")
|
||||
exit(1)
|
||||
|
||||
end = time.time()
|
||||
|
||||
print(f"\n--- Results ---")
|
||||
print_report(hits, times, end - start)
|
@@ -5,6 +5,7 @@ import random
|
||||
import json
|
||||
from tqdm import tqdm
|
||||
import time
|
||||
from utils import print_report
|
||||
|
||||
baseurl = "http://localhost:8000"
|
||||
|
||||
@@ -36,21 +37,4 @@ for i in tqdm(range(10000)):
|
||||
hits.append(content["source"] == "cache")
|
||||
end = time.time()
|
||||
|
||||
hits_count = sum(hits)
|
||||
miss_count = len(hits) - hits_count
|
||||
|
||||
hits_time = 0
|
||||
miss_time = 0
|
||||
for i in range(len(times)):
|
||||
if hits[i]:
|
||||
hits_time += times[i]
|
||||
else:
|
||||
miss_time += times[i]
|
||||
total_time = hits_time + miss_time
|
||||
|
||||
print(f"hits: {hits_count} misses: {miss_count} ratio: { hits_count / (hits_count + miss_count)}")
|
||||
print(f"average response time (ms) : {total_time / len(times)}")
|
||||
print(f"average cache hit response time (ms) : {hits_time / hits_count}")
|
||||
print(f"average cache miss response time (ms): {miss_time / miss_count}")
|
||||
print(f"cache throughput (requests / ms) : { len(times) / total_time}")
|
||||
print(f"real throughput (requests / ms) : { len(times) / (end - start) / 1000}")
|
||||
print_report(hits, times, end - start)
|
72
tests/results_baseline
Normal file
72
tests/results_baseline
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 507 misses: 9493 ratio: 0.0507
|
||||
average response time (ms) : 12.171702599525451
|
||||
average cache hit response time (ms) : 0.0013355202458549063
|
||||
average cache miss response time (ms): 12.821694815810583
|
||||
cache throughput (requests / s) : 82.15777470926605
|
||||
real throughput (requests / s) : 75.03700407782604
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 394 misses: 7631 ratio: 0.04909657320872274
|
||||
average response time (ms) : 12.201453324790313
|
||||
average cache hit response time (ms) : 0.0013572915556467125
|
||||
average cache miss response time (ms): 12.831362620701
|
||||
cache throughput (requests / s) : 81.95744993493923
|
||||
real throughput (requests / s) : 73.58096764800857
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 112 misses: 1906 ratio: 0.05550049554013875
|
||||
average response time (ms) : 12.256505940666992
|
||||
average cache hit response time (ms) : 0.0013560056686401367
|
||||
average cache miss response time (ms): 12.976640669271303
|
||||
cache throughput (requests / s) : 81.58932120140436
|
||||
real throughput (requests / s) : 58.17744310065741
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 7096 misses: 2904 ratio: 0.7096
|
||||
average response time (ms) : 3.7240190267562867
|
||||
average cache hit response time (ms) : 0.000715490930378773
|
||||
average cache miss response time (ms): 12.822008658374966
|
||||
cache throughput (requests / s) : 268.52709205168185
|
||||
real throughput (requests / s) : 211.60014253369587
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 1068 misses: 3862 ratio: 0.21663286004056795
|
||||
average response time (ms) : 9.787649653746197
|
||||
average cache hit response time (ms) : 0.0015231553981366675
|
||||
average cache miss response time (ms): 12.493911461160938
|
||||
cache throughput (requests / s) : 102.16957445113012
|
||||
real throughput (requests / s) : 48.726267135940624
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 634 misses: 9366 ratio: 0.0634
|
||||
average response time (ms) : 12.046342158317566
|
||||
average cache hit response time (ms) : 0.0014511944743737062
|
||||
average cache miss response time (ms): 12.861680709574943
|
||||
cache throughput (requests / s) : 83.01275082988872
|
||||
real throughput (requests / s) : 75.60957264263929
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 1069 misses: 8931 ratio: 0.1069
|
||||
average response time (ms) : 11.471518325805665
|
||||
average cache hit response time (ms) : 0.0013089603748803278
|
||||
average cache miss response time (ms): 12.844450115263228
|
||||
cache throughput (requests / s) : 87.1724188201363
|
||||
real throughput (requests / s) : 79.16835820770777
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 2255 misses: 7745 ratio: 0.2255
|
||||
average response time (ms) : 9.954997634887695
|
||||
average cache hit response time (ms) : 0.0012764645257175894
|
||||
average cache miss response time (ms): 12.853079137685146
|
||||
cache throughput (requests / s) : 100.4520580191259
|
||||
real throughput (requests / s) : 90.12648776249651
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 9992 misses: 8 ratio: 0.9992
|
||||
average response time (ms) : 0.011115074157714844
|
||||
average cache hit response time (ms) : 0.0008463239173492114
|
||||
average cache miss response time (ms): 12.83678412437439
|
||||
cache throughput (requests / s) : 89967.91076791077
|
||||
real throughput (requests / s) : 1050.3783705054095
|
72
tests/results_ideal
Normal file
72
tests/results_ideal
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.7374683618545532
|
||||
average cache hit response time (ms) : 2.7374683618545532
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 365.3010255514076
|
||||
real throughput (requests / s) : 264.62084114988136
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 8025 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.3531507034539434
|
||||
average cache hit response time (ms) : 2.3531507034539434
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 424.96215755846185
|
||||
real throughput (requests / s) : 174.59831308637462
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 2018 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.5427952275876367
|
||||
average cache hit response time (ms) : 2.5427952275876367
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 393.26800253149185
|
||||
real throughput (requests / s) : 24.773972578001025
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 1.3468331575393677
|
||||
average cache hit response time (ms) : 1.3468331575393677
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 742.4824629555278
|
||||
real throughput (requests / s) : 424.7398305196224
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 4930 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.266260528177567
|
||||
average cache hit response time (ms) : 2.266260528177567
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 441.25553420116205
|
||||
real throughput (requests / s) : 77.73575713810418
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.762448763847351
|
||||
average cache hit response time (ms) : 2.762448763847351
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 361.9976642054595
|
||||
real throughput (requests / s) : 264.15171149841484
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.631771683692932
|
||||
average cache hit response time (ms) : 2.631771683692932
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 379.97217091293743
|
||||
real throughput (requests / s) : 273.36179952669863
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 2.3204839706420897
|
||||
average cache hit response time (ms) : 2.3204839706420897
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 430.94458425553995
|
||||
real throughput (requests / s) : 297.5298540772206
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 10000 misses: 0 ratio: 1.0
|
||||
average response time (ms) : 0.007048773765563965
|
||||
average cache hit response time (ms) : 0.007048773765563965
|
||||
average cache miss response time (ms): N/A
|
||||
cache throughput (requests / s) : 141868.64740721197
|
||||
real throughput (requests / s) : 1066.3182833103006
|
72
tests/results_nocache
Normal file
72
tests/results_nocache
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 12.805271649360657
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.805271649360657
|
||||
cache throughput (requests / s) : 78.09283765174385
|
||||
real throughput (requests / s) : 71.53643320734811
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 0 misses: 8025 ratio: 0.0
|
||||
average response time (ms) : 12.849865197392639
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.849865197392639
|
||||
cache throughput (requests / s) : 77.82182806111534
|
||||
real throughput (requests / s) : 70.04929255901189
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 0 misses: 2018 ratio: 0.0
|
||||
average response time (ms) : 12.95277124826452
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.95277124826452
|
||||
cache throughput (requests / s) : 77.20355596752974
|
||||
real throughput (requests / s) : 55.53823196963498
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 11.695257019996642
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 11.695257019996642
|
||||
cache throughput (requests / s) : 85.50474763318087
|
||||
real throughput (requests / s) : 77.81244515584922
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 0 misses: 4930 ratio: 0.0
|
||||
average response time (ms) : 12.357432702008415
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.357432702008415
|
||||
cache throughput (requests / s) : 80.92295739045157
|
||||
real throughput (requests / s) : 43.34797667618127
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 12.735849499702454
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.735849499702454
|
||||
cache throughput (requests / s) : 78.51851578674534
|
||||
real throughput (requests / s) : 71.97159807797185
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 12.631608128547668
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.631608128547668
|
||||
cache throughput (requests / s) : 79.16648377810118
|
||||
real throughput (requests / s) : 72.52127479757786
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 12.301788067817688
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 12.301788067817688
|
||||
cache throughput (requests / s) : 81.28899591564806
|
||||
real throughput (requests / s) : 74.36930521285352
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 0 misses: 10000 ratio: 0.0
|
||||
average response time (ms) : 10.0695650100708
|
||||
average cache hit response time (ms) : N/A
|
||||
average cache miss response time (ms): 10.0695650100708
|
||||
cache throughput (requests / s) : 99.3091557579575
|
||||
real throughput (requests / s) : 89.69346607705592
|
72
tests/results_prefetch
Normal file
72
tests/results_prefetch
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 499 misses: 9501 ratio: 0.0499
|
||||
average response time (ms) : 12.209891486167908
|
||||
average cache hit response time (ms) : 0.0013277860347159162
|
||||
average cache miss response time (ms): 12.851094863324677
|
||||
cache throughput (requests / s) : 81.90080977647176
|
||||
real throughput (requests / s) : 74.83126189192693
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 405 misses: 7620 ratio: 0.05046728971962617
|
||||
average response time (ms) : 11.846880363155377
|
||||
average cache hit response time (ms) : 0.0013816503830897955
|
||||
average cache miss response time (ms): 12.476463956157053
|
||||
cache throughput (requests / s) : 84.41040757954049
|
||||
real throughput (requests / s) : 65.28853446272649
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 95 misses: 1923 ratio: 0.04707631318136769
|
||||
average response time (ms) : 12.044375786341808
|
||||
average cache hit response time (ms) : 0.0017768458316200658
|
||||
average cache miss response time (ms): 12.639303971130403
|
||||
cache throughput (requests / s) : 83.02630354110914
|
||||
real throughput (requests / s) : 20.054516595218445
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 5788 misses: 4212 ratio: 0.5788
|
||||
average response time (ms) : 5.223854732513428
|
||||
average cache hit response time (ms) : 0.0010156268981216525
|
||||
average cache miss response time (ms): 12.400918536716038
|
||||
cache throughput (requests / s) : 191.42951923528236
|
||||
real throughput (requests / s) : 159.80528851176524
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 1047 misses: 3883 ratio: 0.2123732251521298
|
||||
average response time (ms) : 9.824660583635373
|
||||
average cache hit response time (ms) : 0.0011850337016662189
|
||||
average cache miss response time (ms): 12.47343186892525
|
||||
cache throughput (requests / s) : 101.78468675708436
|
||||
real throughput (requests / s) : 48.92688928507726
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 1609 misses: 8391 ratio: 0.1609
|
||||
average response time (ms) : 10.815512228012086
|
||||
average cache hit response time (ms) : 0.0012467706181233976
|
||||
average cache miss response time (ms): 12.889180815897543
|
||||
cache throughput (requests / s) : 92.45979098521182
|
||||
real throughput (requests / s) : 83.71821192504922
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 2792 misses: 7208 ratio: 0.2792
|
||||
average response time (ms) : 9.297505378723145
|
||||
average cache hit response time (ms) : 0.0012284705153850566
|
||||
average cache miss response time (ms): 12.89839399244624
|
||||
cache throughput (requests / s) : 107.55573234606003
|
||||
real throughput (requests / s) : 96.20708400138871
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 4381 misses: 5619 ratio: 0.4381
|
||||
average response time (ms) : 7.246287417411804
|
||||
average cache hit response time (ms) : 0.0011703244792465423
|
||||
average cache miss response time (ms): 12.89513204886536
|
||||
cache throughput (requests / s) : 138.00170244381164
|
||||
real throughput (requests / s) : 120.23923821573455
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 9997 misses: 3 ratio: 0.9997
|
||||
average response time (ms) : 0.004515457153320313
|
||||
average cache hit response time (ms) : 0.0006464036378882415
|
||||
average cache miss response time (ms): 12.897491455078125
|
||||
cache throughput (requests / s) : 221461.51896595422
|
||||
real throughput (requests / s) : 1070.4467798603782
|
72
tests/results_readafterwrite
Normal file
72
tests/results_readafterwrite
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 505 misses: 9495 ratio: 0.0505
|
||||
average response time (ms) : 12.217395067214966
|
||||
average cache hit response time (ms) : 0.0012926536031288675
|
||||
average cache miss response time (ms): 12.867119313541872
|
||||
cache throughput (requests / s) : 81.85050859847135
|
||||
real throughput (requests / s) : 74.74830604127537
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 394 misses: 7631 ratio: 0.04909657320872274
|
||||
average response time (ms) : 12.254692267777392
|
||||
average cache hit response time (ms) : 0.001333086623758229
|
||||
average cache miss response time (ms): 12.887351620073884
|
||||
cache throughput (requests / s) : 81.6013962773598
|
||||
real throughput (requests / s) : 73.22107615138934
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 112 misses: 1906 ratio: 0.05550049554013875
|
||||
average response time (ms) : 12.277225436729529
|
||||
average cache hit response time (ms) : 0.001330460820879255
|
||||
average cache miss response time (ms): 12.998579181378936
|
||||
cache throughput (requests / s) : 81.45162806967119
|
||||
real throughput (requests / s) : 57.99762893791816
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 7096 misses: 2904 ratio: 0.7096
|
||||
average response time (ms) : 3.7444978952407837
|
||||
average cache hit response time (ms) : 0.0007732812285826467
|
||||
average cache miss response time (ms): 12.8923869658436
|
||||
cache throughput (requests / s) : 267.05850236182243
|
||||
real throughput (requests / s) : 210.51762371076964
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 3509 misses: 1421 ratio: 0.711764705882353
|
||||
average response time (ms) : 3.600564883156674
|
||||
average cache hit response time (ms) : 0.0013591693520511987
|
||||
average cache miss response time (ms): 12.488399400919109
|
||||
cache throughput (requests / s) : 277.7341979526512
|
||||
real throughput (requests / s) : 60.02808337161936
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 2163 misses: 7837 ratio: 0.2163
|
||||
average response time (ms) : 10.187475109100342
|
||||
average cache hit response time (ms) : 0.08817761792003474
|
||||
average cache miss response time (ms): 12.974865752640344
|
||||
cache throughput (requests / s) : 98.15974903405778
|
||||
real throughput (requests / s) : 88.35527295489469
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 2766 misses: 7234 ratio: 0.2766
|
||||
average response time (ms) : 9.404127931594848
|
||||
average cache hit response time (ms) : 0.07571936170863484
|
||||
average cache miss response time (ms): 12.97094823907415
|
||||
cache throughput (requests / s) : 106.33628203209798
|
||||
real throughput (requests / s) : 95.07378087157565
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 4087 misses: 5913 ratio: 0.4087
|
||||
average response time (ms) : 7.68894898891449
|
||||
average cache hit response time (ms) : 0.05545212465887814
|
||||
average cache miss response time (ms): 12.965137333952995
|
||||
cache throughput (requests / s) : 130.05678688228338
|
||||
real throughput (requests / s) : 114.06672701630119
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 9992 misses: 8 ratio: 0.9992
|
||||
average response time (ms) : 0.011892271041870118
|
||||
average cache hit response time (ms) : 0.0006238444506215324
|
||||
average cache miss response time (ms): 14.086157083511353
|
||||
cache throughput (requests / s) : 84088.22810035326
|
||||
real throughput (requests / s) : 1066.9048383247582
|
72
tests/results_tiered
Normal file
72
tests/results_tiered
Normal file
@@ -0,0 +1,72 @@
|
||||
|
||||
--- random_read Results ---
|
||||
hits: 1966 misses: 8034 ratio: 0.1966
|
||||
average response time (ms) : 10.438107824325561
|
||||
average cache hit response time (ms) : 0.09135385010759076
|
||||
average cache miss response time (ms): 12.97006180905453
|
||||
cache throughput (requests / s) : 95.80280418923658
|
||||
real throughput (requests / s) : 86.41953468377949
|
||||
|
||||
--- read_heavy Results ---
|
||||
hits: 1563 misses: 6462 ratio: 0.19476635514018692
|
||||
average response time (ms) : 10.474605560302734
|
||||
average cache hit response time (ms) : 0.0937643984686619
|
||||
average cache miss response time (ms): 12.98547754048637
|
||||
cache throughput (requests / s) : 95.46898871207694
|
||||
real throughput (requests / s) : 84.39105440656928
|
||||
|
||||
--- write_heavy Results ---
|
||||
hits: 415 misses: 1603 ratio: 0.20564915758176414
|
||||
average response time (ms) : 10.430269364204822
|
||||
average cache hit response time (ms) : 0.0991062945630177
|
||||
average cache miss response time (ms): 13.10489985322625
|
||||
cache throughput (requests / s) : 95.87480103168338
|
||||
real throughput (requests / s) : 64.8966729031306
|
||||
|
||||
--- frequent_users Results ---
|
||||
hits: 7557 misses: 2443 ratio: 0.7557
|
||||
average response time (ms) : 3.1802247285842897
|
||||
average cache hit response time (ms) : 0.011482802872243217
|
||||
average cache miss response time (ms): 12.982182457854012
|
||||
cache throughput (requests / s) : 314.44318730430115
|
||||
real throughput (requests / s) : 240.3280431399857
|
||||
|
||||
--- frequent_after_write Results ---
|
||||
hits: 1284 misses: 3646 ratio: 0.26044624746450307
|
||||
average response time (ms) : 9.321854902812966
|
||||
average cache hit response time (ms) : 0.022931448024381358
|
||||
average cache miss response time (ms): 12.596626629622769
|
||||
cache throughput (requests / s) : 107.27478709180934
|
||||
real throughput (requests / s) : 49.79781897651969
|
||||
|
||||
--- weighted_friend_readonly 0.25 Results ---
|
||||
hits: 635 misses: 9365 ratio: 0.0635
|
||||
average response time (ms) : 12.026190519332886
|
||||
average cache hit response time (ms) : 0.0014263813889871433
|
||||
average cache miss response time (ms): 12.841537580474837
|
||||
cache throughput (requests / s) : 83.15185082029382
|
||||
real throughput (requests / s) : 75.84771942953694
|
||||
|
||||
--- weighted_friend_readonly 0.50 Results ---
|
||||
hits: 1072 misses: 8928 ratio: 0.1072
|
||||
average response time (ms) : 11.465663266181945
|
||||
average cache hit response time (ms) : 0.001328649805552924
|
||||
average cache miss response time (ms): 12.842205236248645
|
||||
cache throughput (requests / s) : 87.21693431809628
|
||||
real throughput (requests / s) : 79.27228928100207
|
||||
|
||||
--- weighted_friend_readonly 0.75 Results ---
|
||||
hits: 2253 misses: 7747 ratio: 0.2253
|
||||
average response time (ms) : 9.959305834770202
|
||||
average cache hit response time (ms) : 0.0012218290786980208
|
||||
average cache miss response time (ms): 12.855338268592709
|
||||
cache throughput (requests / s) : 100.40860443393278
|
||||
real throughput (requests / s) : 90.2328882960655
|
||||
|
||||
--- weighted_friend_readonly 1 Results ---
|
||||
hits: 9992 misses: 8 ratio: 0.9992
|
||||
average response time (ms) : 0.01112067699432373
|
||||
average cache hit response time (ms) : 0.0008874601894039646
|
||||
average cache miss response time (ms): 12.792408466339111
|
||||
cache throughput (requests / s) : 89922.582996559
|
||||
real throughput (requests / s) : 1052.9166753169109
|
26
tests/utils.py
Normal file
26
tests/utils.py
Normal file
@@ -0,0 +1,26 @@
|
||||
def print_report(hits, request_times, real_time):
|
||||
|
||||
hits_count = sum(hits)
|
||||
miss_count = len(hits) - hits_count
|
||||
|
||||
hits_time = 0
|
||||
miss_time = 0
|
||||
for i in range(len(request_times)):
|
||||
if hits[i]:
|
||||
hits_time += request_times[i]
|
||||
else:
|
||||
miss_time += request_times[i]
|
||||
total_time = hits_time + miss_time
|
||||
|
||||
print(f"hits: {hits_count} misses: {miss_count} ratio: { hits_count / (hits_count + miss_count)}")
|
||||
print(f"average response time (ms) : {total_time / len(request_times)}")
|
||||
if hits_count > 0:
|
||||
print(f"average cache hit response time (ms) : {hits_time / hits_count}")
|
||||
else :
|
||||
print(f"average cache hit response time (ms) : N/A")
|
||||
if miss_count > 0:
|
||||
print(f"average cache miss response time (ms): {miss_time / miss_count}")
|
||||
else:
|
||||
print(f"average cache miss response time (ms): N/A")
|
||||
print(f"cache throughput (requests / s) : { len(request_times) / total_time * 1000}")
|
||||
print(f"real throughput (requests / s) : { len(request_times) / (real_time)}")
|
82
tests/varied_workload.py
Normal file
82
tests/varied_workload.py
Normal file
@@ -0,0 +1,82 @@
|
||||
import requests
|
||||
import random
|
||||
import json
|
||||
import time
|
||||
from tqdm import tqdm
|
||||
from utils import print_report
|
||||
|
||||
baseurl = "http://localhost:8000"
|
||||
|
||||
# Fetch all user IDs and friends list
|
||||
user_ids = json.loads(requests.get(baseurl + "/users").content)["ids"]
|
||||
user_friends = json.loads(requests.get(baseurl + "/users_and_friends").content)
|
||||
|
||||
random.seed(0)
|
||||
|
||||
# Workload Configurations
|
||||
workloads = {
|
||||
"random_read": {"read": 1.0, "write": 0.0},
|
||||
"read_heavy": {"read": 0.8, "write": 0.2},
|
||||
"write_heavy": {"read": 0.2, "write": 0.8},
|
||||
"frequent_users": {"read": 1.0, "write": 0.0, "frequent_percent": 0.7},
|
||||
#"frequent_after_write": {"read": 0.7, "write": 0.3},
|
||||
#"friend_based": {"read": 1.0, "write": 0.0, "friend_access": 0.7},
|
||||
}
|
||||
|
||||
# Frequent users for workload testing (top 10 users)
|
||||
frequent_users = user_ids[:10]
|
||||
|
||||
def generate_request(workload, last_updated=None):
|
||||
"""Generate read or write requests based on workload type"""
|
||||
if random.random() < workload["read"]:
|
||||
user_id = select_user(workload, last_updated)
|
||||
return baseurl + f"/user/{user_id}", None, "GET"
|
||||
|
||||
# Write operation (updates user profile)
|
||||
user_id = select_user(workload, last_updated)
|
||||
write_obj = { "user_id":user_id,"name": "UpdatedUser", "followers":"500","bio":"Updated","posts":"UpdatedPost"}
|
||||
url = baseurl + f"/update_user/"
|
||||
return url, write_obj, "POST"
|
||||
|
||||
def select_user(workload, last_updated):
|
||||
"""Selects a user based on workload type"""
|
||||
if "frequent_percent" in workload and random.random() < workload["frequent_percent"]:
|
||||
return random.choice(frequent_users)
|
||||
|
||||
if "friend_access" in workload and random.random() < workload["friend_access"]:
|
||||
return random.choice(user_friends.get(str(last_updated), user_ids))
|
||||
|
||||
return random.choice(user_ids)
|
||||
|
||||
def run_workload(name, workload):
|
||||
"""Run a workload and log performance metrics"""
|
||||
times = []
|
||||
hits = []
|
||||
start = time.time()
|
||||
last_updated = None
|
||||
|
||||
for _ in tqdm(range(10000), desc=f"Running {name}"):
|
||||
url, data, method = generate_request(workload, last_updated)
|
||||
|
||||
if( method == "GET" ):
|
||||
response = requests.request(method, url)
|
||||
else:
|
||||
response = requests.post(url, json = data)
|
||||
|
||||
try:
|
||||
content = json.loads(response.content)
|
||||
if "time_ms" in content:
|
||||
times.append(content["time_ms"])
|
||||
hits.append(content["source"] == "cache")
|
||||
if method == "POST":
|
||||
last_updated = content.get("user_id", last_updated)
|
||||
except (json.JSONDecodeError, KeyError):
|
||||
print(f"Error processing response: {response.content}")
|
||||
|
||||
end = time.time()
|
||||
print(f"\n--- {name} Results ---")
|
||||
print_report(hits, times, end - start)
|
||||
|
||||
# Run all workloads
|
||||
for workload_name, workload_config in workloads.items():
|
||||
run_workload(workload_name, workload_config)
|
65
tests/weighted_frequentuser_readwrite.py
Normal file
65
tests/weighted_frequentuser_readwrite.py
Normal file
@@ -0,0 +1,65 @@
|
||||
import requests
|
||||
import random
|
||||
import json
|
||||
from tqdm import tqdm
|
||||
import time
|
||||
from utils import print_report
|
||||
|
||||
baseurl = "http://localhost:8000"
|
||||
|
||||
endpoints = {
|
||||
"/user/{user_id}": 0.8, # 80% read operations
|
||||
"/update_user/?user_id={user_id}&name=Test&followers=100&bio=Updated&posts=Updated": 0.2 # 20% write operations
|
||||
}
|
||||
|
||||
# Fetch all user IDs
|
||||
user_ids = json.loads(requests.get(baseurl + "/users").content)["ids"]
|
||||
|
||||
random.seed(0)
|
||||
|
||||
def generate_random():
|
||||
"""Randomly generate a read or write request, favoring cache hits."""
|
||||
endpoint = random.choices(list(endpoints.keys()), list(endpoints.values()))[0]
|
||||
if endpoint == "/user/{user_id}":
|
||||
# Favor frequently accessed user IDs to increase hit ratio
|
||||
if len(user_ids) > 0:
|
||||
# Sample from a subset of user IDs to simulate frequent access
|
||||
frequent_users = user_ids[:int(len(user_ids) * 0.2)] # 20% frequent users
|
||||
random_user = str(random.choice(frequent_users)) if random.random() < 0.7 else str(random.choice(user_ids))
|
||||
else:
|
||||
random_user = str(random.choice(user_ids))
|
||||
else:
|
||||
random_user = str(random.choice(user_ids))
|
||||
return baseurl + endpoint.replace("{user_id}", random_user)
|
||||
|
||||
times = []
|
||||
hits = []
|
||||
|
||||
# Warm-up phase
|
||||
# for _ in tqdm(range(1000)):
|
||||
# url = generate_random()
|
||||
# requests.get(url)
|
||||
|
||||
# Main testing phase
|
||||
start = time.time()
|
||||
for i in tqdm(range(10000)):
|
||||
url = generate_random()
|
||||
response = requests.get(url)
|
||||
|
||||
try:
|
||||
content = json.loads(response.content)
|
||||
|
||||
if "time_ms" in content: # Only process if "time_ms" exists
|
||||
times.append(content["time_ms"])
|
||||
hits.append(content["source"] == "cache")
|
||||
|
||||
except json.JSONDecodeError:
|
||||
print(f"Error decoding JSON: {response.content}")
|
||||
exit(1)
|
||||
except KeyError:
|
||||
print(f"Unexpected response format: {content}")
|
||||
exit(1)
|
||||
|
||||
end = time.time()
|
||||
|
||||
print_report(hits, times, end - start)
|
49
tests/weighted_friend_readonly.py
Normal file
49
tests/weighted_friend_readonly.py
Normal file
@@ -0,0 +1,49 @@
|
||||
# Tests latency and hit rate of endpoints. Can be configured with weighted averages for various endpoints.
|
||||
|
||||
import requests
|
||||
import random
|
||||
import json
|
||||
from tqdm import tqdm
|
||||
import time
|
||||
from utils import print_report
|
||||
|
||||
baseurl = "http://localhost:8000"
|
||||
|
||||
chance_of_selecting_friend = 1
|
||||
|
||||
user_friends = json.loads(requests.get(baseurl + "/users_and_friends").content)
|
||||
user_ids = json.loads(requests.get(baseurl + "/users").content)["ids"]
|
||||
|
||||
random.seed(0)
|
||||
|
||||
def fetch_friend(prob):
|
||||
return random.random() < prob
|
||||
|
||||
def generate_random():
|
||||
random_user = str(random.choice(user_ids))
|
||||
return random_user
|
||||
|
||||
def generate_random_friend(user):
|
||||
next_user = str(random.choice(user_friends[user]))
|
||||
return next_user
|
||||
|
||||
times = []
|
||||
hits = []
|
||||
|
||||
start = time.time()
|
||||
curr_user = generate_random()
|
||||
last_user = curr_user
|
||||
for i in tqdm(range(10000)):
|
||||
url = baseurl + "/user/" + curr_user
|
||||
response = requests.get(url)
|
||||
content = json.loads(response.content)
|
||||
times.append(content["time_ms"])
|
||||
hits.append(content["source"] == "cache")
|
||||
if fetch_friend(chance_of_selecting_friend):
|
||||
curr_user = generate_random_friend(last_user)
|
||||
else:
|
||||
curr_user = generate_random()
|
||||
last_user = curr_user
|
||||
end = time.time()
|
||||
|
||||
print_report(hits, times, end - start)
|
Reference in New Issue
Block a user