From 7d907f35aac1df274c2e195fedd09237700ad2c4 Mon Sep 17 00:00:00 2001
From: Brian Warner <warner@allmydata.com>
Date: Tue, 10 Feb 2009 23:39:38 -0700
Subject: [PATCH] storage: make add-lease work, change default ownernum=1 since
 0 is reserved to mean 'no lease here'

---
 src/allmydata/storage.py           |  9 ++++---
 src/allmydata/test/test_storage.py | 42 +++++++++++++++++++++++-------
 2 files changed, 39 insertions(+), 12 deletions(-)

diff --git a/src/allmydata/storage.py b/src/allmydata/storage.py
index d5859c8f..e2fbb4f8 100644
--- a/src/allmydata/storage.py
+++ b/src/allmydata/storage.py
@@ -615,6 +615,7 @@ class MutableShareFile:
         return leases
 
     def add_lease(self, lease_info):
+        precondition(lease_info.owner_num != 0) # 0 means "no lease here"
         f = open(self.home, 'rb+')
         num_lease_slots = self._get_num_lease_slots(f)
         empty_slot = self._get_first_empty_lease_slot(f)
@@ -649,6 +650,7 @@ class MutableShareFile:
         raise IndexError(msg)
 
     def add_or_renew_lease(self, lease_info):
+        precondition(lease_info.owner_num != 0) # 0 means "no lease here"
         try:
             self.renew_lease(lease_info.renew_secret,
                              lease_info.expiration_time)
@@ -829,11 +831,12 @@ class StorageServer(service.MultiService, Referenceable):
                           "write": [],
                           "close": [],
                           "read": [],
-                          "renew": [],
-                          "cancel": [],
                           "get": [],
                           "writev": [], # mutable
                           "readv": [],
+                          "add-lease": [], # both
+                          "renew": [],
+                          "cancel": [],
                           }
 
     def count(self, name, delta=1):
@@ -1064,7 +1067,7 @@ class StorageServer(service.MultiService, Referenceable):
             yield sf
 
     def remote_add_lease(self, storage_index, renew_secret, cancel_secret,
-                         owner_num=0):
+                         owner_num=1):
         start = time.time()
         self.count("add-lease")
         new_expire_time = time.time() + 31*24*60*60
diff --git a/src/allmydata/test/test_storage.py b/src/allmydata/test/test_storage.py
index a3b18af5..96d1c410 100644
--- a/src/allmydata/test/test_storage.py
+++ b/src/allmydata/test/test_storage.py
@@ -494,6 +494,17 @@ class Server(unittest.TestCase):
         self.failUnlessEqual(len(leases), 2)
         self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs1, rs2]))
 
+        # and a third lease, using add-lease
+        rs2a,cs2a = (hashutil.tagged_hash("blah", "%d" % self._lease_secret.next()),
+                     hashutil.tagged_hash("blah", "%d" % self._lease_secret.next()))
+        ss.remote_add_lease("si1", rs2a, cs2a)
+        leases = list(ss.get_leases("si1"))
+        self.failUnlessEqual(len(leases), 3)
+        self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs1, rs2, rs2a]))
+
+        # add-lease on a missing storage index is an error
+        self.failUnlessRaises(IndexError, ss.remote_add_lease, "si18", "", "")
+
         # check that si0 is readable
         readers = ss.remote_get_buckets("si0")
         self.failUnlessEqual(len(readers), 5)
@@ -519,7 +530,7 @@ class Server(unittest.TestCase):
         self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si0", rs0)
 
 
-        # cancel the first lease on si1, leaving the second in place
+        # cancel the first lease on si1, leaving the second and third in place
         ss.remote_cancel_lease("si1", cs1)
         readers = ss.remote_get_buckets("si1")
         self.failUnlessEqual(len(readers), 5)
@@ -527,16 +538,18 @@ class Server(unittest.TestCase):
         self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs1)
 
         leases = list(ss.get_leases("si1"))
-        self.failUnlessEqual(len(leases), 1)
-        self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs2]))
+        self.failUnlessEqual(len(leases), 2)
+        self.failUnlessEqual(set([l.renew_secret for l in leases]), set([rs2, rs2a]))
 
         ss.remote_renew_lease("si1", rs2)
-        # cancelling the second should make it go away
+        # cancelling the second and third should make it go away
         ss.remote_cancel_lease("si1", cs2)
+        ss.remote_cancel_lease("si1", cs2a)
         readers = ss.remote_get_buckets("si1")
         self.failUnlessEqual(len(readers), 0)
         self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs1)
         self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs2)
+        self.failUnlessRaises(IndexError, ss.remote_renew_lease, "si1", rs2a)
 
         leases = list(ss.get_leases("si1"))
         self.failUnlessEqual(len(leases), 0)
@@ -1046,28 +1059,39 @@ class MutableServer(unittest.TestCase):
         f.write("you ought to be ignoring me\n")
         f.close()
 
+        s0 = MutableShareFile(os.path.join(bucket_dir, "0"))
+        self.failUnlessEqual(len(s0.debug_get_leases()), 1)
+
+        # add-lease on a missing storage index is an error
+        self.failUnlessRaises(IndexError, ss.remote_add_lease, "si18", "", "")
+
         # re-allocate the slots and use the same secrets, that should update
         # the lease
         write("si1", secrets(0), {0: ([], [(0,data)], None)}, [])
+        self.failUnlessEqual(len(s0.debug_get_leases()), 1)
 
         # renew it directly
         ss.remote_renew_lease("si1", secrets(0)[1])
+        self.failUnlessEqual(len(s0.debug_get_leases()), 1)
 
         # now allocate them with a bunch of different secrets, to trigger the
-        # extended lease code
+        # extended lease code. Use add_lease for one of them.
         write("si1", secrets(1), {0: ([], [(0,data)], None)}, [])
-        write("si1", secrets(2), {0: ([], [(0,data)], None)}, [])
+        self.failUnlessEqual(len(s0.debug_get_leases()), 2)
+        secrets2 = secrets(2)
+        ss.remote_add_lease("si1", secrets2[1], secrets2[2])
+        self.failUnlessEqual(len(s0.debug_get_leases()), 3)
         write("si1", secrets(3), {0: ([], [(0,data)], None)}, [])
         write("si1", secrets(4), {0: ([], [(0,data)], None)}, [])
         write("si1", secrets(5), {0: ([], [(0,data)], None)}, [])
 
+        self.failUnlessEqual(len(s0.debug_get_leases()), 6)
+
         # cancel one of them
         ss.remote_cancel_lease("si1", secrets(5)[2])
+        self.failUnlessEqual(len(s0.debug_get_leases()), 5)
 
-        s0 = MutableShareFile(os.path.join(bucket_dir, "0"))
         all_leases = s0.debug_get_leases()
-        self.failUnlessEqual(len(all_leases), 5)
-
         # and write enough data to expand the container, forcing the server
         # to move the leases
         write("si1", secrets(0),
-- 
2.45.2