Added simple diagram for lemma in proof
[iotcloud.git] / doc / iotcloud.tex
1 \documentclass[11pt]{article}\r
2 \newcommand{\tuple}[1]{\ensuremath \langle #1 \rangle}\r
3 \usepackage{color}\r
4 \usepackage{amsthm}\r
5 \usepackage{amsmath}\r
6 \usepackage{graphicx}\r
7 \usepackage{mathrsfs}\r
8 \usepackage{algpseudocode}%\r
9 \usepackage[all]{xy}\r
10 \newtheorem{theorem}{Theorem}\r
11 \newtheorem{prop}{Proposition}\r
12 \newtheorem{lem}{Lemma}\r
13 \newtheorem{defn}{Definition}\r
14 \newcommand{\note}[1]{{\color{red} \bf [[#1]]}}\r
15 \newcommand{\push}[1][1]{\hskip\dimexpr #1\algorithmicindent\relax}\r
16 \begin{document}\r
17 \section{Approach}\r
18 \r
19 \subsection{Keys}\r
20 \r
21 Each device has: user id + password\r
22 \r
23 Server login is:\r
24 hash1(user id), hash1(password)\r
25 \r
26 Symmetric Crypto keys is:\r
27 hash2(user id | password)\r
28 \r
29 Server has finite length queue of entries + max\_entry\_identifier +\r
30 server login key\r
31 \r
32 \subsection{Entry layout}\r
33 Each entry has:\r
34 \begin{enumerate}\r
35 \item Sequence identifier\r
36 \item Random IV (if needed by crypto algorithm)\r
37 \item Encrypted payload\r
38 \end{enumerate}\r
39 \r
40 Payload has:\r
41 \begin{enumerate}\r
42 \item Sequence identifier\r
43 \item Machine id (most probably something like a 64-bit random number \r
44 that is self-generated by client)\r
45 \item HMAC of previous slot\r
46 \item Data entries\r
47 \item HMAC of current slot\r
48 \end{enumerate}\r
49 \r
50 A data entry can be one of these:\r
51 \begin{enumerate}\r
52 \item All or part of a Key-value entry\r
53 \item Slot sequence entry: Machine id + last message identifier \r
54 \newline {The purpose of this is to keep the record of the last slot \r
55 from a certain client if a client's update has to expunge that other \r
56 client's last entry from the queue. This is kept in the slot until \r
57 the entry owner inserts a newer update into the queue.}\r
58 \item Queue state entry: Includes queue size \newline {The purpose \r
59 of this is for the client to tell if the server lies about the number \r
60 of slots in the queue, e.g. if there are 2 queue state entry in the queue, \r
61 e.g. 50 and 70, the client knows that when it sees 50, it should expect \r
62 at most 50 slots in the queue and after it sees 70, it should expect \r
63 50 slots before that queue state entry slot 50 and at most 70 slots. \r
64 The queue state entry slot 70 is counted as slot number 51 in the queue.}\r
65 \item Collision resolution entry: message identifier + machine id of a\r
66 collision winner\r
67 \newline {The purpose of this is to keep keep track of the winner of \r
68 all the collisions until all clients have seen the particular entry.}\r
69 \end{enumerate}\r
70 \r
71 \subsection{Live status}\r
72 \r
73 Live status of entries:\r
74 \begin{enumerate}\r
75 \item Key-Value Entry is dead if either (a) there is a newer key-value pair or (b) it is incomplete.\r
76         \r
77 \item Slot sequence number (of either a message version data\r
78 or user-level data) is dead if there is a newer slot from the same machine.\r
79 \r
80 \item Queue state entry is dead if there is a newer queue state entry.\r
81 {In the case of queue state entries 50 and 70, this means that queue state \r
82 entry 50 is dead and 70 is live. However, not until the number of slots reaches \r
83 70 that queue state entry 50 will be expunged from the queue.}\r
84 \r
85 \item Collision resolution entry is dead if this entry has been seen\r
86 by all clients after a collision happens.\r
87 \end{enumerate}\r
88 \r
89 When data is at the end of the queue ready to expunge, if:\r
90 \begin{enumerate}\r
91 \item The key-value entry is not dead, it must be reinserted at the\r
92 beginning of the queue.\r
93 \r
94 \item If the slot sequence number is not dead, then a message sequence\r
95 entry must be inserted.\r
96 \r
97 \item If the queue state entry is not dead, it must be reinserted at the\r
98 beginning of the queue.\r
99 \end{enumerate}\r
100 \r
101 \r
102 \paragraph{Reads:}\r
103 Client sends a sequence number.  Server replies with either: all data\r
104 entries or all newer data entries.\r
105 \r
106 \paragraph{Writes:}\r
107 Client sends slot, server verifies that sequence number is valid,\r
108 checks entry hash, and replies with an accept message if all checks\r
109 pass.  On success, client updates its sequence number.  On failure,\r
110 server sends updates slots to client and client validates those slots.\r
111 \r
112 \paragraph{Local state on each client:}\r
113 A list of machines and the corresponding latest sequence numbers.\r
114 \r
115 \paragraph{Validation procedure on client:}\r
116 \begin{enumerate}\r
117 \item Decrypt each new slot in order.\r
118 \item For each slot:\r
119     (a) check its HMAC, and\r
120     (b) check that the previous entry HMAC field matches the previous\r
121     entry.\r
122 \item Check that the last-message entry for our machine matches the stored HMAC of our last message sent.\r
123 \item For all other machines, check that the latest sequence number is\r
124 at least as large (never goes backwards).\r
125 \item That the queue has a current queue state entry.\r
126 \item That the number of entries received is consistent with the size\r
127 specified in the queue state entry.\r
128 \end{enumerate}\r
129 \r
130 Key-value entries can span multiple slots.  They aren't valid until\r
131 they are complete.\r
132 \r
133 \subsection{Resizing Queue}\r
134 Client can make a request to resize the queue. This is done as a write that combines:\r
135   (a) a slot with the message, and (b) a request to the server. The queue can only be expanded, never contracted; attempting to decrease the size of the queue will cause future clients to throw an error.\r
136 \r
137 \subsection{Server Algorithm}\r
138 $s \in SN$ is a sequence number set\\\r
139 $sv \in SV$ is a slot's value\\\r
140 $slot_s = \tuple{s, sv} \in SL \subseteq SN \times SV$ \\ \\\r
141 \textbf{State} \\\r
142 \textit{SL = set of live slots on server} \\\r
143 \textit{max = maximum number of slots (input only for resize message)} \\\r
144 \textit{n = number of slots} \\ \\\r
145 \textbf{Helper Function} \\\r
146 $MaxSlot(SL_s)= \tuple{s, sv} \mid \tuple{s, sv}\r
147 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s \geq s_s$ \\\r
148 $MinSlot(SL_s)= \tuple{s, sv} \mid \tuple{s, sv} \r
149 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s \leq s_s$ \\\r
150 $SeqN(\tuple{s, sv})=s$ \\\r
151 $SlotVal(\tuple{s, sv})=sv$ \\\r
152 \r
153 \begin{algorithmic}[1]\r
154 \Function{GetSlot}{$s_g$}\r
155 \State \Return{$\{\tuple{s, sv} \in SL \mid s \geq s_g\}$}\r
156 \EndFunction\r
157 \end{algorithmic}\r
158 \r
159 \begin{algorithmic}[1]\r
160 \Function{PutSlot}{$s_p,sv_p,max'$}\r
161 \If{$(max' \neq \emptyset)$}  \Comment{Resize}\r
162 \State $max \gets max'$\r
163 \EndIf\r
164 \State $\tuple{s_n,sv_n} \gets MaxSlot(SL)$\Comment{Last sv}\r
165 \State $s_n \gets SeqN(\tuple{s_n,sv_n})$\r
166 \If{$(s_p = s_n + 1)$}\r
167         \If{$n = max$}\r
168         \State $\tuple{s_m,sv_m} \gets MinSlot(SL)$\Comment{First sv}\r
169                 \State $SL \gets SL - \{\tuple{s_m,sv_m}\}$\r
170         \Else \Comment{$n < max$}\r
171                 \State $n \gets n + 1$\r
172         \EndIf\r
173     \State $SL \gets SL \cup \{\tuple{s_p,sv_p}\}$\r
174         \State \Return{$(true,\emptyset)$}\r
175 \Else\r
176         \State \Return{$(false,\{\tuple{s,sv}\in SL \mid \r
177     s \geq s_p\})$}\r
178 \EndIf\r
179 \EndFunction\r
180 \end{algorithmic}\r
181 \r
182 \subsection{Client Algorithm}\r
183 \subsubsection{Reading Slots}\r
184 \textbf{Data Entry} \\\r
185 $de$ is a data entry \\\r
186 $k$ is key of entry \\\r
187 $v$ is value of entry \\\r
188 $kv$ is a key-value entry $\tuple{k,v}$, $kv \in DE$ \\\r
189 $ss$ is a slot sequence entry $\tuple{id,s_{last}}$, \r
190 id + last s of a machine, $ss \in DE$ \\\r
191 $qs$ is a queue state entry (contains $max$ size of queue), $qs \in DE$ \\\r
192 $cr$ is a collision resolution entry $\tuple{s_{col},id_{col}}$, \r
193 s + id of a machine that wins a collision, $cr \in DE$ \\\r
194 $DE$ is a set of all data entries, possibly of different types, in a single message \\\r
195 $s \in SN$ is a sequence number set \\\r
196 $id$ is a machine ID\\\r
197 $hmac_p$ is the HMAC value of the previous slot \\\r
198 $hmac_c$ is the HMAC value of the current slot \\\r
199 $Dat_s = \tuple{s,id,hmac_p,DE,hmac_c}$ \\\r
200 $sv_s = \tuple{s, E(Dat_s)} = \r
201 \tuple{s, E(\tuple{s,id,hmac_p,DE,hmac_c})}$ \\ \\\r
202 \r
203 \textbf{States} \\\r
204 \textit{$id_{self}$ = machine Id of this client} \\\r
205 \textit{$max_g$ = maximum number of slots (initially $max_g > 0$)} \\\r
206 \textit{m = number of slots stored on client (initially $m = 0$)} \\\r
207 \textit{$sl_{last}$ = info of last slot in queue = \r
208         $\tuple{s_{last},sv_{last},id_{last}}$ (initially $\emptyset$)} \\\r
209 \textit{DT = set of $\tuple{k,v}$ on client} \\\r
210 \textit{MS = associative array of $\tuple{id, s_{last}}$ of all clients on client \r
211 (initially empty)} \\\r
212 \textit{$MS_g$ = set MS to save all $\tuple{id, s_{last}}$ pairs while\r
213 traversing DT after a request to server (initially empty)} \\\r
214 \textit{SK = Secret Key} \\\r
215 \textit{$SM$ = associative array of $\tuple{s, id}$ of all slots in a previous read\r
216 (initially empty)} \\ \\\r
217 \r
218 \textbf{Helper Functions} \\\r
219 $MaxSlot(SL_s)= \tuple{s, sv}$ \textit{such that} $\tuple{s, sv}\r
220 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s \geq s_s$ \\\r
221 $MinSlot(SL_s)= \tuple{s, sv}$ \textit{such that} $\tuple{s, sv} \r
222 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s \leq s_s$ \\\r
223 $Slot(SL_s,s_s)= \tuple{s, sv}$ \textit{such that} $\tuple{s, sv} \r
224 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s = s_s$ \\\r
225 $SeqN(\tuple{s, sv})=s$ \\\r
226 $SlotVal(\tuple{s, sv})=sv$ \\\r
227 $CreateLastSL(s,sv,id)=\tuple{s,sv,id}=sl_{last}$ \\\r
228 $Decrypt(SK_s,sv_s)=Dat_s=\tuple{s,id,hmac_p,DE,hmac_c}$ \\\r
229 $GetSeqN(Dat_s = \tuple{s,id,hmac_p,DE,hmac_c})=s$ \\\r
230 $GetMacId(Dat_s = \tuple{s,id,hmac_p,DE,hmac_c})=id$ \\\r
231 $GetPrevHmac(Dat_s = \tuple{s,id,hmac_p,DE,hmac_c})=hmac_p$ \\\r
232 $GetCurrHmac(Dat_s = \tuple{s,id,hmac_p,DE,hmac_c})=hmac_c$ \\\r
233 $GetDatEnt(Dat_s = \tuple{s,id,hmac_p,DE,hmac_c})=DE$ \\\r
234 $GetKV($key-value data entry$)=\tuple{k_s,v_s}$ \\\r
235 $GetSS($slot-sequence data entry$)=\tuple{id,s_{last}}$ \\\r
236 $GetQS($queue-state data entry$)=qs_s$ \\\r
237 $GetCR($collision-resolution data entry$)=\tuple{s_s,id_s}$ \\\r
238 $GetS(\tuple{s, id})=s$ \\\r
239 $GetId(\tuple{s, id})=id$ \\\r
240 $GetKey(\tuple{k, v})=k$ \\\r
241 $GetVal(\tuple{k, v})=v$ \\\r
242 $GetKeyVal(DT_s,k_s)= \tuple{k, v}$ \textit{such that} $\tuple{k, v} \r
243 \in DT_s \wedge \forall \tuple{k_s, v_s} \in DT_s, k = k_s$ \\\r
244 $MaxLastSeqN(MS_s)= s_{last}$ \textit{such that} $\tuple{id, s_{last}} \in MS_s \r
245 \wedge \forall \tuple{id_s, s_{s_{last}}} \in MS_s, s_{last} \geq s_{s_{last}}$ \\\r
246 $MinLastSeqN(MS_s)= s_{last}$ \textit{such that} $\tuple{id, s_{last}} \in MS_s \r
247 \wedge \forall \tuple{id_s, s_{s_{last}}} \in MS_s, s_{last} \leq s_{s_{last}}$ \\\r
248 \r
249 \begin{algorithmic}[1]\r
250 \Procedure{Error}{$msg$}\r
251 \State $Print(msg)$\r
252 \State $Halt()$\r
253 \EndProcedure\r
254 \end{algorithmic}\r
255 \r
256 \begin{algorithmic}[1]\r
257 \Function{ValidHmac}{$DE_s,SK_s,hmac_{stored}$}\r
258 \State $hmac_{computed} \gets Hmac(DE_s,SK_s)$\r
259 \State \Return {$hmac_{stored} = hmac_{computed}$}\r
260 \EndFunction\r
261 \end{algorithmic}\r
262 \r
263 \begin{algorithmic}[1]\r
264 \Function{ValidPrevHmac}{$DE_s,hmac_{p_s},hmac_{p_{sto}}$}\r
265 \If{$hmac_{p_s} = \emptyset$}\Comment{First slot - no previous HMAC}\r
266         \State \Return $true$\r
267 \Else\r
268         \State \Return {$hmac_{p_{sto}} = hmac_{p_s}$}\r
269 \EndIf\r
270 \EndFunction\r
271 \end{algorithmic}\r
272 \r
273 \note{So if a slot has a null previous hmac, everything is fine?  What if it isn't the first slot?}\r
274 \r
275 \begin{algorithmic}[1]\r
276 \Function{GetQueSta}{$Dat_s$}\r
277 \State $DE_s \gets GetDatEnt(DE_s)$\r
278 \State $de_{qs} \gets de_s$ \textit{such that} $de_s \in DE_s, \r
279         de_s \in D \land de_s = qs$\r
280 \If{$de_{qs} \neq \emptyset$}\r
281         \State $qs_{ret} \gets GetQS(de_{qs})$\r
282 \Else\r
283         \State $qs_{ret} \gets \emptyset$\r
284 \EndIf\r
285 \State \Return{$qs_{ret}$}\r
286 \EndFunction\r
287 \end{algorithmic}\r
288 \r
289 \begin{algorithmic}[1]\r
290 \Function{GetSlotSeq}{$Dat_s$}\r
291 \State $DE_s \gets GetDatEnt(Dat_s)$\r
292 \State $de_{ss} \gets de_s$ \textit{such that} $de_s \in DE_s, \r
293         de_s \in D \land de_s = ss$\r
294 \If{$de_{ss} \neq \emptyset$}\r
295         \State $\tuple{id_{ret},s_{last_{ret}}} \gets GetSS(de_{ss})$\r
296 \Else\r
297         \State $\tuple{id_{ret},s_{last_{ret}}} \gets \emptyset$\r
298 \EndIf\r
299 \State \Return{$\tuple{id_{ret},s_{last_{ret}}}$}\r
300 \EndFunction\r
301 \end{algorithmic}\r
302 \r
303 \begin{algorithmic}[1]\r
304 \Function{GetColRes}{$Dat_s$}\Comment{At most 2 $cr$ entries in a slot}\r
305 \State $DE_s \gets GetDatEnt(Dat_s)$\r
306 \State $de_{cr} \gets de_s$ \textit{such that} $de_s \in DE_s, \r
307         de_s \in D \land de_s = cr$\r
308 \If{$de_{cr} \neq \emptyset$}\r
309         \State $\tuple{s_{ret},id_{ret}} \gets GetCR(de_{cr})$\r
310 \Else\r
311         \State $\tuple{s_{ret},id_{ret}} \r
312         \gets \emptyset$\r
313 \EndIf\r
314 \State $de_{r_{cr}} \gets de_s$ \textit{such that} $de_s \in DE_s, \r
315         de_s \in D \land de_s = cr \land de_s \neq de_{cr}$\r
316 \If{$de_{r_{cr}} \neq \emptyset$}\r
317         \State $\tuple{s_{r_{ret}},id_{r_{ret}}} \gets GetCR(de_{r_{cr}})$\r
318 \Else\r
319         \State $\tuple{s_{r_{ret}},id_{r_{ret}}} \r
320         \gets \emptyset$\r
321 \EndIf\r
322 \State \Return{$\{\tuple{s_{ret},id_{ret}},\tuple{s_{r_{ret}},id_{r_{ret}}}\}$}\r
323 \EndFunction\r
324 \end{algorithmic}\r
325 \r
326 \begin{algorithmic}[1]\r
327 \Function{UpdateLastSeqN}{$id_s,s_s,MS_s$}\r
328 \State $s_t \gets MS_s[id_s]$\r
329 \If{$s_t = \emptyset$}\r
330         \State $MS_s[id_s] = s_s$  \Comment{First occurrence}\r
331 \Else\r
332         \State $MS_S[id_s] \gets max(s_t, s_s)$\r
333 \EndIf\r
334 \State \Return{$MS_s$}\r
335 \EndFunction\r
336 \end{algorithmic}\r
337 \r
338 \begin{algorithmic}[1]\r
339 \Procedure{CheckLastSeqN}{$MS_s,MS_t$}\Comment{Check $MS_t$ based on the newer $MS_s$}\r
340 \For {$\tuple{id, s_t}$ in $MS_t$}\r
341         \State $s_s \gets MS_s[id]$\r
342         \If{$s_s = \emptyset$}\r
343         \Call{Error}{'No $s$ for machine $id$'}\r
344         \ElsIf{$id = id_{self}$ and $s_s \neq s_t$}\r
345                         \State \Call{Error}{'Invalid last $s$ for this machine'}\r
346         \ElsIf{$id \neq id_{self}$ and $s_{s_{last}} < s_{t_{last}}$}\r
347         \State \Call{Error}{'Invalid last $s$ for machine $id$'}\r
348     \Else\r
349                 \State $MS_t[id] \gets s_s$\r
350         \EndIf\r
351 \EndFor\r
352 \EndProcedure\r
353 \end{algorithmic}\r
354 \r
355 \begin{algorithmic}[1]\r
356 \Procedure{CheckCollision}{$MS_s,SM_s,\tuple{s_s,id_s}$}\r
357 \If{$\tuple{s_s,id_s} \neq \emptyset$}\r
358         \State $s_s \gets GetS(\tuple{s_s,id_s})$\r
359         \State $id_s \gets GetId(\tuple{s_s,id_s})$\r
360         \State $s_{s_{last}} \gets GetLastSeqN(MS_s,id_s)$\r
361         \If{$s_{s_{last}} < s_s$}\r
362                 \State $\Call{CheckColRes}{SM_s,\tuple{s_s,id_s}}$\r
363         \EndIf\r
364 \EndIf\r
365 \EndProcedure\r
366 \end{algorithmic}\r
367 \r
368 \begin{algorithmic}[1]\r
369 \Procedure{CheckColRes}{$SM_s,\tuple{s_t,id_t}$}\Comment{Check $id_s$ in $SM_s$}\r
370 \State $id_s \gets SM_s[s_t]$\r
371 \If{$id_s \neq id_t$}\r
372         \State \Call{Error}{'Invalid $id_s$ for this slot update'}\r
373 \EndIf\r
374 \EndProcedure\r
375 \end{algorithmic}\r
376 \r
377 \begin{algorithmic}[1]\r
378 \Function{StoreLastSlot}{$MS_s,sl_l,s_s,sv_s,id_s$}\r
379 \State $s_{min} \gets MinLastSeqN(MS_s)$\r
380 \If{$s_{min} \neq \emptyset \land s_{min} = s_s$}\Comment{$MS$ initially empty}\r
381         \State $sl_l \gets CreateLastSL(s_s,sv_s,id_s)$\r
382 \EndIf\r
383 \State \Return{$sl_l$}\r
384 \EndFunction\r
385 \end{algorithmic}\r
386 \r
387 \begin{algorithmic}[1]\r
388 \Function{UpdateDT}{$DT_s,Dat_s$}\r
389 \State $DE_s \gets GetDatEnt(Dat_s)$\r
390 \ForAll{$de_s \in DE_s$}\r
391         \If{$de_s$ \textit{such that} $de_s \in D \land de_s = kv$}\r
392                 \State $\tuple{k_s,v_s} \gets GetKV(de_s)$\r
393                 \State $\tuple{k_s,v_t} \gets GetKeyVal(DT_s,k_s)$\r
394                 \If{$\tuple{k_s,v_t} = \emptyset$}\r
395                         \State $DT_s \gets DT_s \cup \{\tuple{k_s,v_s}\}$\r
396                 \Else\r
397                 \State $DT_s \gets (DT_s - \{\tuple{k_s,v_t}\}) \cup \r
398                         \{\tuple{k_s,v_s}\}$\r
399                 \EndIf\r
400     \EndIf\r
401 \EndFor\r
402 \State \Return{$DT_s$}\r
403 \EndFunction\r
404 \end{algorithmic}\r
405 \r
406 \begin{algorithmic}[1]\r
407 \Procedure{ProcessSL}{$SL_g$}\r
408 \State $MS_g \gets \emptyset$\r
409 \State $SM_{curr} \gets \emptyset$\r
410 \State $\tuple{s_{g_{max}},sv_{g_{max}}} \gets MaxSlot(SL_g)$\r
411 \State $s_{g_{max}} \gets SeqN(\tuple{s_{g_{max}},sv_{g_{max}}})$\r
412 \State $\tuple{s_{g_{min}},sv_{g_{min}}} \gets MinSlot(SL_g)$\r
413 \State $s_{g_{min}} \gets SeqN(\tuple{s_{g_{min}},sv_{g_{min}}})$\r
414 \For{$s_g \gets s_{g_{min}}$ \textbf{to} $s_{g_{max}}$}\Comment{Process slots \r
415         in $SL_g$ in order}\r
416         \State $\tuple{s_g,sv_g} \gets Slot(SL_g,s_g)$\r
417         \State $SM_{curr} \gets SM_{curr} \cup \{\tuple{s_g,sv_g}\}$\r
418         \State $Dat_g \gets Decrypt(SK,sv_g)$\r
419         \State $s_{g_{in}} \gets GetSeqN(Dat_g)$\r
420     \If{$s_g \neq s_{g_{in}}$}\r
421                 \State \Call{Error}{'Invalid sequence number'}\r
422         \EndIf\r
423         \State $DE_g \gets GetDatEnt(Dat_g)$\r
424         \State $hmac_{p_{stored}} \gets GetPrevHmac(Dat_g)$\r
425         \If{$\neg \Call{ValidPrevHmac}{DE_g,hmac_{p_g},hmac_{p_{stored}}}$}\r
426                 \State \Call{Error}{'Invalid previous HMAC value'}\r
427         \EndIf\r
428         \State $hmac_{c_g} \gets GetCurrHmac(Dat_g)$\r
429         \If{$\neg \Call{ValidHmac}{DE_g,SK,hmac_{c_g}}$}\r
430                 \State \Call{Error}{'Invalid current HMAC value'}\r
431         \EndIf\r
432         \State $hmac_{p_g} \gets Hmac(DE_g,SK)$\Comment{Update $hmac_{p_g}$ for next check}\r
433         \State $qs_g \gets \Call{GetQueSta}{Dat_g}$\Comment{Handle qs}\r
434         \If{$qs_g \neq \emptyset \land qs_g > max_g$}\r
435                 \State $max_g \gets qs_g$\r
436         \EndIf\r
437     %Check for last s in Dat\r
438         \State $id_g \gets GetMacId(Dat_g)$\Comment{Handle last s}\r
439         \State $MS_g \gets \Call{UpdateLastSeqN}{id_g,s_g,MS_g}$\r
440     %Check for last s in DE in Dat\r
441     \State $\tuple{id_d,s_{d_{last}}} \gets \Call{GetSlotSeq}{Dat_g}$\Comment{Handle ss}\r
442         \If{$\tuple{id_d,s_{d_{last}}} \neq \emptyset$}\r
443         \State $MS_g \gets \Call{UpdateLastSeqN}{id_d,s_{d_{last}},MS_g}$\r
444         \EndIf\r
445         \State $\{\tuple{s_e,id_e},\tuple{s_f,id_f}\} \gets \r
446         \Call{GetColRes}{Dat_g}$\Comment{Handle cr}\r
447         \State $\Call{CheckCollision}{MS,SM,\tuple{s_e,id_e}}$\Comment{From normal slot}\r
448         \State $\Call{CheckCollision}{MS,SM,\tuple{s_f,id_f}}$\Comment{From reinsertion}\r
449         \State $sl_{last} \gets \Call{StoreLastSlot}{MS,sl_{last},s_g,sv_g,id_g}$\r
450         \State $DT \gets \Call{UpdateDT}{DT,Dat_g}$\r
451 \EndFor\r
452 \State $SM \gets SM_{curr}$\r
453 \If{$m + |SL_g| \leq max_g$}\Comment{Check actual size against $max_g$}\r
454         \State $m \gets m + |SL_g|$\r
455 \Else\r
456         \State \Call{Error}{'Actual queue size exceeds $max_g$'}\r
457 \EndIf\r
458 \State $\Call{CheckLastSeqN}{MS_g,MS}$\r
459 \EndProcedure\r
460 \end{algorithmic}\r
461 \r
462 \begin{algorithmic}[1]\r
463 \Procedure{GetKVPairs}{}\r
464 \State $s_g \gets GetLastSeqN(MS,id_{self}) + 1$\r
465 \State $SL_c \gets \Call{GetSlot}{s_g}$\r
466 \State $\Call{ProcessSL}{SL_c}$\Comment{Process slots and update DT}\r
467 \EndProcedure\r
468 \end{algorithmic}\r
469 \r
470 \begin{algorithmic}[1]\r
471 \Function{GetValFromKey}{$k_g$}\r
472 \State $\tuple{k_s,v_s} \gets \tuple{k,v}$ \textit{such that} $\tuple{k,v} \r
473         \in DT \land k = k_g$\r
474 \State \Return{$v_s$}\r
475 \EndFunction\r
476 \end{algorithmic}\r
477 \r
478 \subsubsection{Writing Slots}\r
479 \textbf{Data Entry} \\\r
480 $k$ is key of entry \\\r
481 $v$ is value of entry \\\r
482 $kv$ is a key-value entry $\tuple{k,v}$\\\r
483 $D = \{kv,ss,qs,cr\}$ \\\r
484 $DE = \{de \mid de \in D\}$ \\\r
485 $Dat_s = \tuple{s,id,hmac_p,DE,hmac_c}$ \\\r
486 $sv_s = \tuple{s, E(Dat_s)} = \r
487 \tuple{s, E(\tuple{s,id,hmac_p,DE,hmac_c})}$ \\ \\\r
488 \textbf{States} \\\r
489 \textit{$cp$ = data entry $DE$ maximum size/capacity} \\\r
490 \textit{$cr_p$ = saved cr entry $\tuple{s,id}$ on client if there is a collision\r
491 (sent in the following slot)} \\\r
492 \textit{$cr_{p_{last}}$ = saved cr entry $\tuple{s,id}$ on client if there is a \r
493 collision in reinserting the last slot (sent in the following slot)} \\\r
494 \textit{$ck_p$ = counter of $kv \in KV$ for putting pairs (initially 0)} \\\r
495 \textit{$ck_g$ = counter of $kv \in KV$ for getting pairs (initially 0)} \\\r
496 \textit{$hmac_{c_p}$ = the HMAC value of the current slot} \\\r
497 \textit{$hmac_{p_p}$ = the HMAC value of the previous slot \r
498 ($hmac_{p_p} = \emptyset$ for the first slot)} \\\r
499 \textit{$id_{self}$ = machine Id of this client} \\\r
500 \textit{$sl_{last}$ = info of last slot in queue = \r
501         $\tuple{s_{last},sv_{last},id_{last}}$ (initially $\emptyset$)} \\\r
502 \textit{$th_p$ = threshold number of dead slots for a resize to happen} \\\r
503 \textit{$m'_p$ = offset added to $max$ for resize} \\\r
504 \textit{$KV$ = set of $\tuple{ck, \tuple{k,v}}$ of kv entries on client} \\\r
505 \textit{$SL_p$ = set of returned slots on client} \\\r
506 \textit{SK = Secret Key} \\ \\\r
507 \textbf{Helper Functions} \\\r
508 $CreateDat(s,id,hmac_p,DE,hmac_c)=Dat_s=\tuple{s,id,hmac_p,DE,hmac_c}$ \\\r
509 $CreateCR(s,id)=\tuple{s,id}$ \\\r
510 $CreateQS(max')=qs$ \\\r
511 $CreateSS(id,s_{last})=\tuple{id,s_{last}}$ \\\r
512 $Encrypt(Dat_s,SK_s)=sv_s$ \\\r
513 $GetStatus(\tuple{status,SL})=status$ \\\r
514 $GetSL(\tuple{status,SL})=SL$ \\\r
515 $GetLastS(sl = \tuple{s,sv,id})=s$ \\\r
516 $GetSV(sl = \tuple{s,sv,id})=sv$ \\\r
517 $GetID(sl = \tuple{s,sv,id})=id$ \\\r
518 $GetColSeqN(SL_s,s_s)= \tuple{s, sv}$ \textit{such that} $\tuple{s, sv}\r
519 \in SL_s \wedge \forall \tuple{s_s, sv_s} \in SL_s, s = s_s$ \\\r
520 $GetKV(KV_s,k_s)= \tuple{ck,\tuple{k, v}}$ \textit{such that} \r
521 $\tuple{ck,\tuple{k, v}} \in KV_s \wedge\r
522 \forall \tuple{ck_s,\tuple{k_s, v_s}} \in KV_s, k = k_s$ \\\r
523 \r
524 \begin{algorithmic}[1]\r
525 \Function{PutKVPair}{$KV_s,\tuple{k_s,v_s}$}\r
526 \State $\tuple{ck_s,\tuple{k_s,v_t}} \gets GetKV(KV_s,k_s)$\r
527 \If{$\tuple{ck_s,\tuple{k_s,v_t}} = \emptyset$}\r
528         \State $KV_s \gets KV_s \cup \{\tuple{ck_p, \tuple{k_s,v_s}}\}$\r
529         \State $ck_p \gets ck_p + 1$\r
530 \Else\r
531         \State $KV_s \gets (KV_s - \{\tuple{ck_s, \tuple{k_s,v_t}}\}) \cup \r
532         \{\tuple{ck_s, \tuple{k_s,v_s}}\}$\r
533 \EndIf\r
534 \State \Return{$KV_s$}\r
535 \EndFunction\r
536 \end{algorithmic}\r
537 \r
538 \begin{algorithmic}[1]\r
539 \Function{CheckResize}{$MS_s,th_s,max'_t,m'_s$}\r
540 \State $s_{last_{min}} \gets MinLastSeqN(MS_s)$\r
541 \State $s_{last_{max}} \gets MaxLastSeqN(MS_s)$\r
542 \State $n_{live} \gets s_{last_{max}} - s_{last_{min}}$\Comment{Number of live slots}\r
543 \State $n_{dead} \gets max'_t - n_{live}$\r
544 \If{$n_{dead} \leq th_s$}\r
545         \State $max'_s \gets max'_t + m'_s$\r
546 \Else\r
547         \State $max'_s \gets \emptyset$\r
548 \EndIf\r
549 \State \Return{$max'_s$}\r
550 \EndFunction\r
551 \end{algorithmic}\r
552 \r
553 \begin{algorithmic}[1]\r
554 \Function{CheckNeedSS}{$MS_s,max'_t$}\Comment{Check if $ss$ is needed}\r
555 \State $s_{last_{min}} \gets MinLastSeqN(MS_s)$\r
556 \State $s_{last_{max}} \gets MaxLastSeqN(MS_s)$\r
557 \State $n_{live} \gets s_{last_{max}} - s_{last_{min}}$\Comment{Number of live slots}\r
558 \State $n_{dead} \gets max'_t - n_{live}$\r
559 \State \Return {$n_{dead} = 0$}\r
560 \EndFunction\r
561 \end{algorithmic}\r
562 \r
563 \begin{algorithmic}[1]\r
564 \Function{HandleCollision}{$\tuple{stat_s,SL_s}$}\r
565 \State $stat_s \gets GetStatus(\tuple{stat_s,SL_s})$\r
566 \State $SL_s \gets GetSL(\tuple{stat_s,SL_s})$\r
567 \If{$\neg stat_s$}\Comment{Handle collision}\r
568         \State $\tuple{s_{col},sv_{col}} \gets GetColSeqN(SL_s,s_s)$\r
569         \State $s_{col} \gets SeqN(\tuple{s_{col},sv_{col}})$\r
570         \State $sv_{col} \gets SlotVal(\tuple{s_{col},sv_{col}})$\r
571         \State $Dat_{col} \gets Decrypt(SK,sv_{col})$\r
572         \State $id_{col} \gets GetMacId(Dat_{col})$\r
573         \State $\tuple{s_{col},id_{col}} \gets CreateCR(s_{col},id_{col})$\r
574         \State $cr_s \gets \tuple{s_{col},id_{col}}$\r
575 \Else\r
576         \State $cr_s \gets \emptyset$\r
577 \EndIf\r
578 \State $\Call{ProcessSL}{SL_s}$\r
579 \State \Return{$cr_s$}\r
580 \EndFunction\r
581 \end{algorithmic}\r
582 \r
583 \begin{algorithmic}[1]\r
584 \Function{ReinsertLastSlot}{$need_s,sl_{s_{last}},max'_s$}\r
585 \If{$need_s$}\r
586         \State $s_s \gets GetLastS(sl_{s_{last}})$\r
587         \State $sv_s \gets GetSV(sl_{s_{last}})$\r
588         \State $\tuple{stat_s,SL_s} \gets \Call{PutSlot}{s_s,sv_s,max'_s}$\r
589         \State $cr_s \gets \Call{HandleCollision}{\tuple{stat_s,SL_s}}$\r
590 \EndIf\r
591 \State \Return{$cr_s$}\r
592 \EndFunction\r
593 \end{algorithmic}\r
594 \note{Shouldn't this function do something pretty sophisticated about seeing what data we actually need to keep from the last slot and not just insert the entire thing?}\r
595 \r
596 \note{Probably best to just not call this function is $need_s$ is false and not pass in such parameters.  It makes it harder to read.}\r
597 \r
598 \r
599 \begin{algorithmic}[1]\r
600 \Function{GetDEPairs}{$KV_s,max'_s,need_s,sl_s$}\r
601 \State $DE_{ret} \gets \emptyset$\r
602 \State $cp_s \gets cp$\r
603 \If{$cr_p \neq \emptyset$}\Comment{Check and insert a $cr$}\r
604         \State $DE_{ret} \gets DE_{ret} \cup cr_p$\r
605         \State $cp_s \gets cp_s - 1$\r
606 \EndIf\r
607 \If{$cr_{p_{last}} \neq \emptyset$}\Comment{Check and insert a $cr$}\r
608         \State $DE_{ret} \gets DE_{ret} \cup cr_{p_{last}}$\r
609         \State $cp_s \gets cp_s - 1$\r
610 \EndIf\r
611 \If{$max'_s \neq \emptyset$}\Comment{Check and insert a $qs$}\r
612         \State $qs_s \gets max'_s$\r
613         \State $DE_{ret} \gets DE_{ret} \cup qs_s$\r
614         \State $cp_s \gets cp_s - 1$\r
615 \EndIf\r
616 \If{$need_s$}\Comment{Check and insert a $ss$}\r
617         \State $id_s \gets GetID(sl_s)$\r
618         \State $s_{s_{last}} \gets GetLastS(sl_s)$\r
619         \State $ss_s \gets CreateSS(id_s,s_{s_{last}})$\r
620         \State $DE_{ret} \gets DE_{ret} \cup ss_s$\r
621         \State $cp_s \gets cp_s - 1$\r
622 \EndIf\r
623 \If{$|KV_s| \leq cp$}\Comment{$KV$ set can extend multiple slots}\r
624         \State $DE_{ret} \gets DE_{ret} \cup\r
625         \{\tuple{k_s,v_s} \mid \tuple{ck_s,\tuple{k_s,v_s}} \in KV_s\}$\r
626 \Else\r
627         \State $DE_{ret} \gets DE_{ret} \cup\r
628         \{\tuple{k_s,v_s} \mid \tuple{ck_s,\tuple{k_s,v_s}} \in KV_s,\r
629                 ck_g \leq ck_s < ck_g + cp_s\}$\r
630         \If{$|DE_{ret}| = cp$}\r
631                 \State $ck_g \gets ck_g + cp_s$\Comment{Middle of KV set}\r
632         \Else\r
633                 \State $ck_g \gets 0$\Comment{End of KV set}\r
634         \EndIf\r
635 \EndIf\r
636 \State \Return{$DE_{ret}$}\r
637 \EndFunction\r
638 \end{algorithmic}\r
639 \r
640 \begin{algorithmic}[1]\r
641 \Procedure{PutDataEntries}{$th_p,m'_p$}\r
642 \State $s_p \gets MaxLastSeqN(MS)$\r
643 \State $max'_p \gets \Call{CheckResize}{MS,th_p,max'_g,m'_p}$\r
644 \State $need_p \gets \Call{CheckNeedSS}{MS,max'_g}$\r
645 \State $DE_p \gets \Call{GetDEPairs}{KV,max'_p,need_p,sl_{last}}$\r
646 \State $hmac_{c_p} \gets Hmac(DE_p,SK)$\r
647 \State $Dat_p \gets CreateDat(s_p,id_{self},hmac_{p_p},DE_p,hmac_{c_p})$\r
648 \State $hmac_{p_p} \gets hmac_{c_p}$\r
649 \State $sv_p \gets Encrypt(Dat_p,SK)$\r
650 \State $\tuple{stat_p,SL_p} \gets \Call{PutSlot}{s_p,sv_p,max'_p}$\r
651 \State $cr_p \gets \Call{HandleCollision}{\tuple{stat_p,SL_p}}$\r
652 \State $cr_{p_{last}} \gets \Call{ReinsertLastSlot}{need_p,sl_{last},max'_p}$\r
653 \EndProcedure\r
654 \end{algorithmic}\r
655 \r
656 \note{Lots of problems with PutDataEntries: (1) What happens if lose network connectivity after adding the key value pair, but before reinserting the last slot?  You probably need to create space first and then insert your data entry...  (2) What if reinsertlastslot kicks something else important out?  What if the server rejects our update because it is out of date?  At the very least, any putdataentries function w/o a loop is wrong!}\r
657 \r
658 \note{General comments...  Work on structuring things to improve\r
659   readability...  This include names of functions/variables, how\r
660   things are partitioned into functions, adding useful comments,...}\r
661 \r
662 \r
663 \subsection{Definitions for Formal Guarantees}\r
664 \r
665 \begin{enumerate}\r
666 \item Equality: Two messages $t$ and $u$ are equal if their sequence numbers, senders, and contents are exactly the same.\r
667 \item Message: A message $t$, is the tuple $t = (i(t), s(t), contents(t))$ containing the sequence number, machine ID of the sender, and contents of $t$ respectively.\r
668 \item Parent: A parent of a message $t$ is the message $A(t)$, unique by the correctness of HMACs, such that $HMAC_C(t) = HMAC_P(A(t))$.\r
669 \item Partial message sequence: A partial message sequence is a sequence of messages, no two with the same sequence number, that can be divided into disjoint chains, where a chain of messages with length $n \ge 1$ is a message sequence $(t_i, t_{i+1}, ..., t_{i+n-1})$ such that for every index $i < k \le i+n-1$, $t_k$ has sequence number $k$ and is the parent of $t_{k-1}$.\r
670 \item Total message sequence: A total message sequence $T$ with length $n$ is a chain of messages that starts at $i = 1$.\r
671 \item Path: The path of a message $t$ is the total message sequence whose last message is $t$.\r
672 \item Consistency: A partial message sequence $P$ is consistent with a total message sequence $T$ of length $n$ if for every message $p \in P$ with $i(p) < n$, $T_{i(p)} = p$. This implies that $\{p \in P | i(p) \le n\}$ is a subsequence of T.\r
673 \item Transitive closure set at index $n$: A set $\mathscr{S}$ of clients comprising a connected component of an undirected graph, where two clients are connected by an edge if they both received the same message $t$ with index $i(t) > n$.\r
674 \r
675 \end{enumerate}\r
676 \r
677 \subsection{Formal Guarantee}\r
678 \r
679 \begin{prop} Every client $J$ who sends a message $t$ has $A(t)$ as its latest stored message, and $i(t) = i(A(t)) + 1$. \end{prop}\r
680 \begin{proof} True by definition, because $J$ sets $HMAC_P(t) = HMAC_C(A(t))$ and $i(t) = i(A(t)) + 1$ when a message is sent. \end{proof}\r
681 \r
682 \begin{prop} If a rejected message entry is added to the RML at index $i$, the message will remain in the RML until every client has seen it. \end{prop}\r
683 \begin{proof} Every RML entry $e$ remains in the queue until it reaches the tail, and is refreshed by the next sender $J$ at that time if $min(MS) > i(e)$; that is, until every client has sent a message with sequence number greater than $i(e)$. Because every client who sends a message with index $i$ has the state of the queue at $i - 1$, this client will have seen the message at $i(e)$. \end{proof}\r
684 \r
685 \begin{lem} \r
686 \r
687 \end{lem}\r
688 \r
689 \begin{figure}[h]\r
690   \centering\r
691       \xymatrix{\r
692 \dots \ar[r] & q \ar[dr]_{J} \ar[r]^{K} & S_1 \ar[r] & S_2 \ar[rr] & & \dots \ar[r] & S_n = u \\\r
693 & & R_1 \ar[r] & R_2 \ar[r] & \dots \ar[r] & R_m = t}\r
694 \caption{By Lemma 2, receiving $t$ before $u$ is impossible.}\r
695 \end{figure}\r
696 \r
697 \begin{lem} If two packets $t$ and $u$, with $i(t) \le i(u)$, are received without errors by a client $C$, then $t$ is in the path of $u$. \end{lem}\r
698 \begin{proof}\r
699 Assume that $t$ is not in the path of $u$. Take $u$ to be the packet of smallest index for which this occurs, and $t$ be the packet with largest index for this $u$. We will prove that an error occurs upon receipt of $u$.\r
700 \r
701 Let $R_1$ be the earliest member of the path of $t$ that is not in the path of $u$, and $q$ be its parent. $q$, the last common ancestor of $t$ and $u$, must exist, since all clients and the server were initialized with the same state. Let $S_1$ be the successor of $q$ that is in the path of $u$; we know $S_1 \neq R_1$. Let $R = (R_1, R_2, \dots, R_m = t)$ be the distinct portion of the path of $t$, and similarly let $S$ be the distinct portion of the path of $S_n = u$.\r
702 \r
703 Let $J = s(R_1)$, and $K = s(S_1)$. Because no client can send two messages with the same index, and $i(R_1) = i(S_1) = i(q) + 1$, we know that $J \neq K$.\r
704 \r
705 There are two cases:\r
706 \r
707 \begin{itemize}\r
708 \item Case 1: $J$ did not send a message in $S$. Then $v_J(t) > v_J(q) = v_J(u)$.\r
709 \begin{itemize}\r
710 \item Case 1.1: $C$ will throw an error, because the latest index of $J$ changes in the opposite direction of the sequence number: $v_J(u) < v_J(t)$ but $i(u) > i(t)$.\r
711 \r
712 \r
713 \r
714 \end{itemize}\r
715 \r
716 \r
717 \r
718 \item Case 2: $J$ sent at least one message in $S$. Call the first one $p$. We know that $i(p) > i(S_1)$, since $J \neq K$. $R_1$ must be sent either before or after $p$.\r
719 \begin{itemize}\r
720 \item Case 2.1: Client $J$ sends $p$, and then $R_1$. When $p$ was sent, whether it was accepted or rejected, $i(J, p) \geq i(p)$. Since $i(p) > i(S_1)$, $i(J, p) > q$. So $i(q) < i(J, p)$, which would cause $J$ to fail to send $R_1$, a contradiction.\r
721 \begin{itemize}\r
722 \item Case 2.2.1: \r
723 \r
724 \r
725 \r
726 \end{itemize}\r
727 \item Case 2.2: Client $J$ sends $R_1$, and then $p$. Let $X = (R_1, \dots )$ be the list of messages $J$ sends starting before $R_1$ and ending before $p$.\r
728 \begin{itemize}\r
729 \item Case 2.2.1: Some message in $X$ was accepted. In this case, before sending $p$, $J$'s value for its own latest index would be strictly greater than $v_J(q)$. ($J$ could not have sent a message with index less than $i(q)$ after receiving $q$). When preparing to send $p$, $J$ would have received its own latest index as at most $v_J(q)$. $J$ throws an error before sending $p$, because its own latest index decreases.\r
730 \item Case 2.2.2: All messages in $X$ were rejected. Client $J$ will always add the latest rejected message to the rejected-message list in the next update; so for every $i$, $1 \leq i < |X|$, the $i$th element of $X$ will be recorded in the RML of all further elements of $X$; and every element of $X$ will be recorded in $RML(p)$. Since every rejected message in $RML(p)$ will be in $RML(C, u)$, and $u$ is the first message that $C$ sees which does not have $t$ in its path, $R_1$ will be recorded in $RML(C, p)$. When $C$ receives $u$, $C$ will throw an error from the match $(J, iq+1)$ in $RML(C, p)$.\r
731 \end{itemize}\r
732 \end{itemize}\r
733 \end{itemize}\r
734 \end{proof}\r
735 \r
736 \begin{theorem}\r
737 Suppose that there is a transitive closure set $\mathscr{S}$ of clients, at index $n$. Then there is some total message sequence $T$ of length $n$ such that every client $C$ in $\mathscr{S}$ sees a partial sequence $P_C$ consistent with $T$. \end{theorem}\r
738 \r
739 \begin{proof}\r
740 The definition of consistency of $P_C$ with $T$ is that every message $p \in P_C$ with index $i(p) \le n$ is equal to the message in that slot in $T$. Let $C_1$ be some client in the transitive closure set, with partial message sequence $P_{C_1}$, and let $u$ be some message with $i(u) > i$ that $C_1$ shares with another client. Then let $T$ be the portion of the path of $u$ ending at index $i$ and $t$ be the message at that index. Clearly, by Lemma 1, $P_{C_1}$ is consistent with $T$, and furthermore. We will show that, for every other client $D$ with partial sequence $P_D$, $P_D$ has some message whose path includes $t$. Because $D$ is in the transitive closure, there is a sequence of edges from $C_1$ to $D$. Call this $\mathscr{C} = (C_1, C_2, ..., D)$. I will prove by induction that $D$ has a message whose path includes $t$.\r
741 \r
742 For the base case, $P_{C_1}$ includes $u$, whose path includes $t$. For the inductive step, suppose $P_{C_k}$ has an message $w$ with a path that includes $t$, and shares message $x$ with $P_{C_{k+1}}$ such that $i(x) > i$. If $i(w) = i(x)$, then $w = x$. If $i(w) < i(x)$, then, by Lemma 1, $w$ is in the path of $x$. If $i(w) > i(x)$, $x$ is in the path of $w$; note again that its index is greater than $i$. In any case, $t$ is in the path of $u_k+1$.\r
743 \r
744 Let $w$ the message of $D$ whose path includes $t$. By Lemma 1, every message in $P_D$ with index smaller than $i(w)$ is in the path of $w$. Since $t$ is in the path of $w$, every message in $P_D$ with smaller index than $i(t)$ is in $T$. Therefore, $P_D$ is consistent with $T$.\r
745 \end{proof}\r
746 \r
747 \subsection{Future Work}\r
748 \paragraph{Support Messages}\r
749   A message is dead once receiving machine sends an entry with a newer\r
750   sequence identifier\r
751 \r
752 \paragraph{Persistent data structures}\r
753         Root object w/ fields\r
754         Other objects can be reachable from root\r
755         Each object has its own entries\r
756         Dead objects correspond to dead \r
757 \r
758 \paragraph{Multiple App Sharing}\r
759 \r
760 Idea is to separate subspace of entries...  Shared with other cloud...\r
761 \end{document}\r