-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathSite.py
129 lines (107 loc) · 3.64 KB
/
Site.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
from typing import Optional, Any
from data_models import DataLog
def extract_num(key: str) -> int:
"""
Author(s):
- Rishav Roy
"""
return int(key[1:])
class Site:
def __init__(self, site_id: int):
"""
Author(s):
- Rishav Roy
- Akash Kumar Shrivastva
"""
self.site_id = site_id
# main data storage for the committed values -> Dict[str: list[DataLog]]
# Format: data_id -> list[DataLog]
self.data_store = {}
# temporary storage for any write instruction -> Dict[str: list[DataLog]]
# Format: data_id -> history of data_values
self.data_history = {}
self.initialize_site_data()
def initialize_site_data(self):
"""
Author(s):
- Rishav Roy
- Akash Kumar Shrivastva
"""
# initialize data items with their corresponding initial values
# even indexed items are replicated at all sites
# odd indexed items are replicated only at site (i % 10) + 1
for i in range(1, 21):
data_id = f"x{i}"
if i % 2 == 0 or (i % 10) + 1 == self.site_id:
self.data_store[data_id] = []
self.data_store[data_id].append(DataLog(
value=10 * i,
timestamp=-1,
transaction_id="T_init",
committed=True
))
self.data_history[data_id] = []
def get_value_using_snapshot_isolation(self, data_id: str, timestamp: int) -> Any:
"""
Author(s):
- Akash Kumar Shrivastva
"""
committed_writes = reversed(self.data_store[data_id])
for write in committed_writes:
if write.timestamp < timestamp:
return write.value
return None
def read(self, data_id: str, timestamp: int) -> Optional[int]:
"""
Author(s):
- Rishav Roy
- Akash Kumar Shrivastva
"""
# Data is read from the committed data_store
if data_id in self.data_store:
value = self.get_value_using_snapshot_isolation(data_id, timestamp)
return value
return None
def write(self, t_id: str, data_id: str, value: int, timestamp: int) -> bool:
"""
Author(s):
- Akash Kumar Shrivastva
"""
if data_id not in self.data_store:
return False
self.data_history[data_id].append(DataLog(
value=value,
timestamp=timestamp,
transaction_id=t_id,
committed=False
))
return True
def persist(self, t_id: str, data_id: str, timestamp: int):
"""
Author(s):
- Rishav Roy
- Akash Kumar Shrivastva
"""
data_history = self.data_history[data_id]
valid_logs = [log for log in reversed(data_history) if log.transaction_id == t_id]
if not valid_logs:
return
last_log = valid_logs[0]
commit_value = last_log.value
self.data_store[data_id].append(DataLog(
value=commit_value,
timestamp=timestamp,
transaction_id=t_id,
committed=True
))
def dump(self) -> str:
"""
Author(s):
- Rishav Roy
- Akash Kumar Shrivastva
"""
status = f"site {self.site_id} - "
ordered_data = sorted(self.data_store.keys(), key=extract_num)
data_status = [f"{data_id}: {self.data_store[data_id][-1].value}" for data_id in ordered_data]
status += ", ".join(data_status)
return status