On Fri, 22 Aug 2014, Gregory Farnum wrote: > On Fri, Aug 22, 2014 at 2:30 PM, Sage Weil <sweil@xxxxxxxxxx> wrote: > > On Fri, 22 Aug 2014, Gregory Farnum wrote: > >> Whereas if the notify timeout is the same time length as a watch > >> timeout, we can affirmatively know on a notify reply (with or without > >> error return codes) that every client has either: > >> 1) seen the notify, or > >> 2) seen the watch connection's timeout period elapse on their side. > >> So no matter what happens in the network, after a notify cycle has > >> elapsed, every client has either seen the new data or knows that they > >> have failed and needs to re-read everything. > > > > Okay, this makes some sense. I think we still have several problems, > > though, if we want this sort of guarantee. > > > > 1) Notify delivery is distinct from notify ack, even more so with the > > changes I made. Before we acked when we returned from the callback, which > > could take who knows how long. Now, the client explicitly acks and need > > not block in the callback doing whatever work they need to do. > > I think I'm missing something here; can you elaborate? It's just a long way of saying just because the notify arrived intact and the client is still connected to the OSDs and the watch is still alive doesn't mean that the notify is acked. The app might want to perform some action on notify before acking, and who knows what or how long it will take. > > 2) The watch timeout generally means we give the client *at least* this > > much time to reconnect, but frequently more. > > > > I think what we probably need to do is mark the Session on the OSD if a > > notify times out so that the guarantee is actually that either > > > > 1) The client acked the notify, or > > 2) The client's watch disconnected (and they will be able to tell that > > they may have missed notifies), or > > 3) The client's Session was marked (and they will be notified that they > > missed notifies) > > > > 2 and 3 will boil down to the same thing as far as the librados API goes. > > We were thinking a combination of a callback (where there is no timeliness > > guarantee for message delivery) and a synchronous call like watch_check() > > where you, say, pass a timestamp and it tells you whether, as of that > > timestamp, you may have missed any events. Implementing that reliably is > > going to need to involve some sort of ping with the OSD to ensure we've > > seen any events, and/or know that we are still connected as of some time. > > > > Anyway, given those 3 options, I don't think we need notify timeout == > > watch timeout. We could do a notify timeout of 1s and any slowish client > > will get their session marked and eventually either find out they missed > > something or find out they've been disconnected. > > > > It seems like anything stronger than 'eventually' has to be handled a bit > > above this interface. As in, the clients agree that they won't take any > > action unless they know they haven't missed events as of 5 seconds ago. > > (This will allow the watch_check(now - 5s) to not block in the general > > case, as 5s is a wide enough window for the pings.) If a peer gets a > > notify timeout, they wait 5 more seconds to ensure that time elapses. > > Can you give some examples of situations in which an eventual delivery > is a useful building block for something other than "I know this was > delivered"? I'm having trouble coming up with any; in particular both > of our existing use cases (RBD header sync, RGW cache invalidations) > want guaranteed delivery. Otherwise we're stuck delaying every > metadata change on RGW buckets for the timeout period to ensure we're > following ACL policies! And users who are quiescing IO on RBD in order > to take snapshots could get them dirtied if they resume writing on a > node before it's actually processed the header updates. Eventual delivery is necessary but not sufficient. That's what watch_check() is for: it tells us whether, as of some timestamp, we have seen the notifies, or that we've possibly missed some. A simple callback is never sufficient for any notification guarantee because schedulers and everything else can arbitrarily delay it. For the RGW cache use-case: Each client registers a watch. Timeout is, say, 30s. On each cache use, client calls watch_check(now - 5s). If that is true, we consider our cache valid. Specifically, success means that sometime after (now - 5s) we sent a message to the OSD and got an ack that confirms the watch was intact and we had missed no notifies as of that time. If it is false, we consider the cache state unknown and (presumably) fall back to rereading the bucket metadata. On notify callback, we invalidate an entry, and then ack. On modify, we make the update, and then: - send notify, timeout = 5s - on success, we are done (let's say this usually takes 30ms) - on timeout (i.e., 5s), we now know that the other client will eventually discover they missed the timeout. We wait a bit longer (I think < 5s? I need to draw a picture) to ensure their watch_check() alls will fail, and then succeed (the original modify). I think this mainly boils down to making sure that the watch heartbeats are done properly (they need to check the session state to ensure there weren't notifies we missed, and also verify we are still talking to the correct OSD for the given object). sage -- To unsubscribe from this list: send the line "unsubscribe ceph-devel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html