/*
  * Replication slot on-disk data structure.
@@ -225,10 +226,25 @@ ReplicationSlotCreate(const char *name, bool db_specific,
        ReplicationSlot *slot = NULL;
        int                     i;

-       Assert(MyReplicationSlot == NULL);
+       /* Only aka ephemeral slots can survive across commands. */

What does this comment mean?


+       Assert(!MyReplicationSlot ||
+                  MyReplicationSlot->data.persistency == RS_EPHEMERAL);

+       if (MyReplicationSlot)
+       {
+               /* Already acquired? Nothis to do. */

typo.

+               if (namestrcmp(&MyReplicationSlot->data.name, name) == 0)
+                       return;
+
+               ereport(ERROR,
+                               
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
+                                errmsg("cannot create replication slot %s, 
another slot %s is "
+                                               "already active in this 
session",
+                                               name, 
NameStr(MyReplicationSlot->data.name))));
+       }
+

Why do we now create slots that are already created? That seems like an
odd API change.

        /*
         * If some other backend ran this code concurrently with us, we'd likely
         * both allocate the same slot, and that would be bad.  We'd also be at
@@ -331,10 +347,25 @@ ReplicationSlotAcquire(const char *name)
        int                     i;
        int                     active_pid = 0;

-       Assert(MyReplicationSlot == NULL);
+       /* Only aka ephemeral slots can survive across commands. */
+       Assert(!MyReplicationSlot ||
+                  MyReplicationSlot->data.persistency == RS_EPHEMERAL);

        ReplicationSlotValidateName(name, ERROR);

+       if (MyReplicationSlot)
+       {
+               /* Already acquired? Nothis to do. */
+               if (namestrcmp(&MyReplicationSlot->data.name, name) == 0)
+                       return;
+
+               ereport(ERROR,
+                               
(errcode(ERRCODE_OBJECT_NOT_IN_PREREQUISITE_STATE),
+                                errmsg("cannot acquire replication slot %s, 
another slot %s is "
+                                               "already active in this 
session",
+                                               name, 
NameStr(MyReplicationSlot->data.name))));
+       }
+
        /* Search for the named slot and mark it active if we find it. */
        LWLockAcquire(ReplicationSlotControlLock, LW_SHARED);
        for (i = 0; i < max_replication_slots; i++)
@@ -406,12 +437,26 @@ ReplicationSlotRelease(void)
 }

Uh? We shouldn't ever have to acquire ephemeral

 /*
+ * Same as above but only if currently acquired slot is peristent one.
+ */

s/peristent/persistent/

+void
+ReplicationSlotReleasePersistent(void)
+{
+       Assert(MyReplicationSlot);
+
+       if (MyReplicationSlot->data.persistency == RS_PERSISTENT)
+               ReplicationSlotRelease();
+}

Ick.



Hm. I think I have to agree a bit with Peter here.  Overloading
MyReplicationSlot this way seems ugly, and I think there's a bunch of
bugs around it too.


Sounds what we really want is a) two different lifetimes for ephemeral
slots, session and "command" b) have a number of slots that are released
either after a failed transaction / command or at session end.   The
easiest way for that appears to have a list of slots to be checked at
end-of-xact and backend shutdown. 

Regards,

Andres


-- 
Sent via pgsql-hackers mailing list (pgsql-hackers@postgresql.org)
To make changes to your subscription:
http://www.postgresql.org/mailpref/pgsql-hackers

Reply via email to