2012-05-02 15:21:36 -07:00
|
|
|
# Copyright (c) 2010, 2011, 2012 Nicira, Inc.
|
2010-08-25 10:26:40 -07:00
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at:
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
|
|
|
import os
|
|
|
|
|
2012-02-29 17:20:14 -08:00
|
|
|
import ovs.util
|
2017-02-24 10:03:26 +08:00
|
|
|
import ovs.vlog
|
2011-09-24 17:53:30 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
# Values returned by Reconnect.run()
|
|
|
|
CONNECT = 'connect'
|
|
|
|
DISCONNECT = 'disconnect'
|
|
|
|
PROBE = 'probe'
|
|
|
|
|
2012-02-29 17:20:14 -08:00
|
|
|
EOF = ovs.util.EOF
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog = ovs.vlog.Vlog("reconnect")
|
2010-08-25 10:26:40 -07:00
|
|
|
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
class Reconnect(object):
|
|
|
|
"""A finite-state machine for connecting and reconnecting to a network
|
|
|
|
resource with exponential backoff. It also provides optional support for
|
|
|
|
detecting a connection on which the peer is no longer responding.
|
|
|
|
|
|
|
|
The library does not implement anything networking related, only an FSM for
|
|
|
|
networking code to use.
|
|
|
|
|
|
|
|
Many Reconnect methods take a "now" argument. This makes testing easier
|
|
|
|
since there is no hidden state. When not testing, just pass the return
|
|
|
|
value of ovs.time.msec(). (Perhaps this design should be revisited
|
|
|
|
later.)"""
|
|
|
|
|
|
|
|
class Void(object):
|
|
|
|
name = "VOID"
|
|
|
|
is_connected = False
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
return None
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
|
|
|
return None
|
|
|
|
|
|
|
|
class Listening(object):
|
|
|
|
name = "LISTENING"
|
|
|
|
is_connected = False
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
return None
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
|
|
|
return None
|
|
|
|
|
|
|
|
class Backoff(object):
|
|
|
|
name = "BACKOFF"
|
|
|
|
is_connected = False
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
return fsm.state_entered + fsm.backoff
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
|
|
|
return CONNECT
|
|
|
|
|
|
|
|
class ConnectInProgress(object):
|
2011-03-09 10:42:24 -08:00
|
|
|
name = "CONNECTING"
|
2010-08-25 10:26:40 -07:00
|
|
|
is_connected = False
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
return fsm.state_entered + max(1000, fsm.backoff)
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
|
|
|
return DISCONNECT
|
|
|
|
|
|
|
|
class Active(object):
|
|
|
|
name = "ACTIVE"
|
|
|
|
is_connected = True
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
if fsm.probe_interval:
|
2012-08-08 13:32:57 -07:00
|
|
|
base = max(fsm.last_activity, fsm.state_entered)
|
2020-12-21 16:21:00 -08:00
|
|
|
expiration = base + fsm.probe_interval
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
if (now < expiration or
|
|
|
|
fsm.last_receive_attempt is None or
|
2020-12-21 16:21:00 -08:00
|
|
|
fsm.last_receive_attempt >= expiration):
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
# We still have time before the expiration or the time has
|
|
|
|
# already passed and there was no activity. In the first
|
|
|
|
# case we need to wait for the expiration, in the second -
|
|
|
|
# we're already past the deadline. */
|
2020-12-21 16:21:00 -08:00
|
|
|
return expiration
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
else:
|
|
|
|
# Time has already passed, but we didn't attempt to receive
|
|
|
|
# anything. We need to wake up and try to receive even if
|
|
|
|
# nothing is pending, so we can update the expiration time
|
|
|
|
# or transition to a different state.
|
|
|
|
return now + 1
|
2010-08-25 10:26:40 -07:00
|
|
|
return None
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.dbg("%s: idle %d ms, sending inactivity probe"
|
|
|
|
% (fsm.name,
|
2012-08-08 13:32:57 -07:00
|
|
|
now - max(fsm.last_activity, fsm.state_entered)))
|
2010-08-25 10:26:40 -07:00
|
|
|
fsm._transition(now, Reconnect.Idle)
|
|
|
|
return PROBE
|
|
|
|
|
|
|
|
class Idle(object):
|
|
|
|
name = "IDLE"
|
|
|
|
is_connected = True
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2012-04-11 20:42:01 -07:00
|
|
|
if fsm.probe_interval:
|
2020-12-21 16:21:00 -08:00
|
|
|
expiration = fsm.state_entered + fsm.probe_interval
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
if (now < expiration or
|
|
|
|
fsm.last_receive_attempt is None or
|
2020-12-21 16:21:00 -08:00
|
|
|
fsm.last_receive_attempt >= expiration):
|
|
|
|
return expiration
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
else:
|
|
|
|
return now + 1
|
2012-04-11 20:42:01 -07:00
|
|
|
return None
|
2010-08-25 10:26:40 -07:00
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.err("%s: no response to inactivity probe after %.3g "
|
|
|
|
"seconds, disconnecting"
|
|
|
|
% (fsm.name, (now - fsm.state_entered) / 1000.0))
|
2010-08-25 10:26:40 -07:00
|
|
|
return DISCONNECT
|
|
|
|
|
2011-08-23 10:45:43 -07:00
|
|
|
class Reconnect(object):
|
2010-08-25 10:26:40 -07:00
|
|
|
name = "RECONNECT"
|
|
|
|
is_connected = False
|
|
|
|
|
|
|
|
@staticmethod
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
def deadline(fsm, now):
|
2010-08-25 10:26:40 -07:00
|
|
|
return fsm.state_entered
|
|
|
|
|
|
|
|
@staticmethod
|
|
|
|
def run(fsm, now):
|
|
|
|
return DISCONNECT
|
|
|
|
|
|
|
|
def __init__(self, now):
|
|
|
|
"""Creates and returns a new reconnect FSM with default settings. The
|
|
|
|
FSM is initially disabled. The caller will likely want to call
|
|
|
|
self.enable() and self.set_name() on the returned object."""
|
|
|
|
|
|
|
|
self.name = "void"
|
|
|
|
self.min_backoff = 1000
|
|
|
|
self.max_backoff = 8000
|
|
|
|
self.probe_interval = 5000
|
|
|
|
self.passive = False
|
2011-09-24 17:53:30 -07:00
|
|
|
self.info_level = vlog.info
|
2010-08-25 10:26:40 -07:00
|
|
|
|
|
|
|
self.state = Reconnect.Void
|
|
|
|
self.state_entered = now
|
|
|
|
self.backoff = 0
|
2012-08-08 13:32:57 -07:00
|
|
|
self.last_activity = now
|
2011-03-14 13:10:02 -07:00
|
|
|
self.last_connected = None
|
|
|
|
self.last_disconnected = None
|
2020-12-21 16:21:00 -08:00
|
|
|
self.last_receive_attempt = now
|
2010-08-25 10:26:40 -07:00
|
|
|
self.max_tries = None
|
2018-01-22 11:04:58 -08:00
|
|
|
self.backoff_free_tries = 0
|
2010-08-25 10:26:40 -07:00
|
|
|
|
|
|
|
self.creation_time = now
|
|
|
|
self.n_attempted_connections = 0
|
|
|
|
self.n_successful_connections = 0
|
|
|
|
self.total_connected_duration = 0
|
|
|
|
self.seqno = 0
|
|
|
|
|
|
|
|
def set_quiet(self, quiet):
|
|
|
|
"""If 'quiet' is true, this object will log informational messages at
|
|
|
|
debug level, by default keeping them out of log files. This is
|
|
|
|
appropriate if the connection is one that is expected to be
|
|
|
|
short-lived, so that the log messages are merely distracting.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
If 'quiet' is false, this object logs informational messages at info
|
|
|
|
level. This is the default.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
This setting has no effect on the log level of debugging, warning, or
|
|
|
|
error messages."""
|
|
|
|
if quiet:
|
2011-09-24 17:53:30 -07:00
|
|
|
self.info_level = vlog.dbg
|
2010-08-25 10:26:40 -07:00
|
|
|
else:
|
2011-09-24 17:53:30 -07:00
|
|
|
self.info_level = vlog.info
|
2010-08-25 10:26:40 -07:00
|
|
|
|
|
|
|
def get_name(self):
|
|
|
|
return self.name
|
|
|
|
|
|
|
|
def set_name(self, name):
|
|
|
|
"""Sets this object's name to 'name'. If 'name' is None, then "void"
|
|
|
|
is used instead.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
The name is used in log messages."""
|
|
|
|
if name is None:
|
|
|
|
self.name = "void"
|
|
|
|
else:
|
|
|
|
self.name = name
|
|
|
|
|
|
|
|
def get_min_backoff(self):
|
|
|
|
"""Return the minimum number of milliseconds to back off between
|
|
|
|
consecutive connection attempts. The default is 1000 ms."""
|
|
|
|
return self.min_backoff
|
|
|
|
|
|
|
|
def get_max_backoff(self):
|
|
|
|
"""Return the maximum number of milliseconds to back off between
|
|
|
|
consecutive connection attempts. The default is 8000 ms."""
|
|
|
|
return self.max_backoff
|
|
|
|
|
|
|
|
def get_probe_interval(self):
|
|
|
|
"""Returns the "probe interval" in milliseconds. If this is zero, it
|
|
|
|
disables the connection keepalive feature. If it is nonzero, then if
|
|
|
|
the interval passes while the FSM is connected and without
|
2012-08-08 13:32:57 -07:00
|
|
|
self.activity() being called, self.run() returns ovs.reconnect.PROBE.
|
|
|
|
If the interval passes again without self.activity() being called,
|
2010-08-25 10:26:40 -07:00
|
|
|
self.run() returns ovs.reconnect.DISCONNECT."""
|
|
|
|
return self.probe_interval
|
|
|
|
|
|
|
|
def set_max_tries(self, max_tries):
|
|
|
|
"""Limits the maximum number of times that this object will ask the
|
|
|
|
client to try to reconnect to 'max_tries'. None (the default) means an
|
|
|
|
unlimited number of tries.
|
|
|
|
|
|
|
|
After the number of tries has expired, the FSM will disable itself
|
|
|
|
instead of backing off and retrying."""
|
|
|
|
self.max_tries = max_tries
|
|
|
|
|
|
|
|
def get_max_tries(self):
|
|
|
|
"""Returns the current remaining number of connection attempts,
|
|
|
|
None if the number is unlimited."""
|
|
|
|
return self.max_tries
|
|
|
|
|
|
|
|
def set_backoff(self, min_backoff, max_backoff):
|
|
|
|
"""Configures the backoff parameters for this FSM. 'min_backoff' is
|
|
|
|
the minimum number of milliseconds, and 'max_backoff' is the maximum,
|
|
|
|
between connection attempts.
|
|
|
|
|
|
|
|
'min_backoff' must be at least 1000, and 'max_backoff' must be greater
|
|
|
|
than or equal to 'min_backoff'."""
|
|
|
|
self.min_backoff = max(min_backoff, 1000)
|
|
|
|
if self.max_backoff:
|
|
|
|
self.max_backoff = max(max_backoff, 1000)
|
|
|
|
else:
|
|
|
|
self.max_backoff = 8000
|
|
|
|
if self.min_backoff > self.max_backoff:
|
|
|
|
self.max_backoff = self.min_backoff
|
|
|
|
|
|
|
|
if (self.state == Reconnect.Backoff and
|
|
|
|
self.backoff > self.max_backoff):
|
2019-02-12 18:34:02 +03:00
|
|
|
self.backoff = self.max_backoff
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2018-01-22 11:04:58 -08:00
|
|
|
def set_backoff_free_tries(self, backoff_free_tries):
|
|
|
|
"""Sets the number of connection attempts that will be made without
|
|
|
|
backoff to 'backoff_free_tries'. Values 0 and 1 both
|
|
|
|
represent a single attempt."""
|
|
|
|
self.backoff_free_tries = backoff_free_tries
|
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
def set_probe_interval(self, probe_interval):
|
|
|
|
"""Sets the "probe interval" to 'probe_interval', in milliseconds. If
|
|
|
|
this is zero, it disables the connection keepalive feature. If it is
|
|
|
|
nonzero, then if the interval passes while this FSM is connected and
|
2012-08-08 13:32:57 -07:00
|
|
|
without self.activity() being called, self.run() returns
|
2010-08-25 10:26:40 -07:00
|
|
|
ovs.reconnect.PROBE. If the interval passes again without
|
2012-08-08 13:32:57 -07:00
|
|
|
self.activity() being called, self.run() returns
|
2010-08-25 10:26:40 -07:00
|
|
|
ovs.reconnect.DISCONNECT.
|
|
|
|
|
|
|
|
If 'probe_interval' is nonzero, then it will be forced to a value of at
|
|
|
|
least 1000 ms."""
|
|
|
|
if probe_interval:
|
|
|
|
self.probe_interval = max(1000, probe_interval)
|
|
|
|
else:
|
|
|
|
self.probe_interval = 0
|
|
|
|
|
|
|
|
def is_passive(self):
|
|
|
|
"""Returns true if 'fsm' is in passive mode, false if 'fsm' is in
|
|
|
|
active mode (the default)."""
|
|
|
|
return self.passive
|
|
|
|
|
|
|
|
def set_passive(self, passive, now):
|
|
|
|
"""Configures this FSM for active or passive mode. In active mode (the
|
|
|
|
default), the FSM is attempting to connect to a remote host. In
|
2011-09-23 23:43:12 -07:00
|
|
|
passive mode, the FSM is listening for connections from a remote
|
|
|
|
host."""
|
2010-08-25 10:26:40 -07:00
|
|
|
if self.passive != passive:
|
|
|
|
self.passive = passive
|
|
|
|
|
|
|
|
if ((passive and self.state in (Reconnect.ConnectInProgress,
|
|
|
|
Reconnect.Reconnect)) or
|
|
|
|
(not passive and self.state == Reconnect.Listening
|
|
|
|
and self.__may_retry())):
|
|
|
|
self._transition(now, Reconnect.Backoff)
|
|
|
|
self.backoff = 0
|
|
|
|
|
|
|
|
def is_enabled(self):
|
|
|
|
"""Returns true if this FSM has been enabled with self.enable().
|
|
|
|
Calling another function that indicates a change in connection state,
|
|
|
|
such as self.disconnected() or self.force_reconnect(), will also enable
|
|
|
|
a reconnect FSM."""
|
|
|
|
return self.state != Reconnect.Void
|
|
|
|
|
|
|
|
def enable(self, now):
|
|
|
|
"""If this FSM is disabled (the default for newly created FSMs),
|
|
|
|
enables it, so that the next call to reconnect_run() for 'fsm' will
|
|
|
|
return ovs.reconnect.CONNECT.
|
|
|
|
|
|
|
|
If this FSM is not disabled, this function has no effect."""
|
|
|
|
if self.state == Reconnect.Void and self.__may_retry():
|
|
|
|
self._transition(now, Reconnect.Backoff)
|
|
|
|
self.backoff = 0
|
|
|
|
|
|
|
|
def disable(self, now):
|
|
|
|
"""Disables this FSM. Until 'fsm' is enabled again, self.run() will
|
|
|
|
always return 0."""
|
|
|
|
if self.state != Reconnect.Void:
|
|
|
|
self._transition(now, Reconnect.Void)
|
|
|
|
|
|
|
|
def force_reconnect(self, now):
|
|
|
|
"""If this FSM is enabled and currently connected (or attempting to
|
|
|
|
connect), forces self.run() to return ovs.reconnect.DISCONNECT the next
|
|
|
|
time it is called, which should cause the client to drop the connection
|
|
|
|
(or attempt), back off, and then reconnect."""
|
|
|
|
if self.state in (Reconnect.ConnectInProgress,
|
|
|
|
Reconnect.Active,
|
|
|
|
Reconnect.Idle):
|
|
|
|
self._transition(now, Reconnect.Reconnect)
|
|
|
|
|
|
|
|
def disconnected(self, now, error):
|
|
|
|
"""Tell this FSM that the connection dropped or that a connection
|
|
|
|
attempt failed. 'error' specifies the reason: a positive value
|
|
|
|
represents an errno value, EOF indicates that the connection was closed
|
|
|
|
by the peer (e.g. read() returned 0), and 0 indicates no specific
|
|
|
|
error.
|
|
|
|
|
|
|
|
The FSM will back off, then reconnect."""
|
|
|
|
if self.state not in (Reconnect.Backoff, Reconnect.Void):
|
|
|
|
# Report what happened
|
|
|
|
if self.state in (Reconnect.Active, Reconnect.Idle):
|
|
|
|
if error > 0:
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.warn("%s: connection dropped (%s)"
|
|
|
|
% (self.name, os.strerror(error)))
|
2010-08-25 10:26:40 -07:00
|
|
|
elif error == EOF:
|
|
|
|
self.info_level("%s: connection closed by peer"
|
|
|
|
% self.name)
|
|
|
|
else:
|
|
|
|
self.info_level("%s: connection dropped" % self.name)
|
|
|
|
elif self.state == Reconnect.Listening:
|
|
|
|
if error > 0:
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.warn("%s: error listening for connections (%s)"
|
|
|
|
% (self.name, os.strerror(error)))
|
2010-08-25 10:26:40 -07:00
|
|
|
else:
|
|
|
|
self.info_level("%s: error listening for connections"
|
|
|
|
% self.name)
|
2019-01-25 19:10:01 +00:00
|
|
|
elif self.state == Reconnect.Reconnect:
|
|
|
|
self.info_level("%s: connection closed by client"
|
|
|
|
% self.name)
|
2018-01-22 11:04:58 -08:00
|
|
|
elif self.backoff < self.max_backoff:
|
2010-08-25 10:26:40 -07:00
|
|
|
if self.passive:
|
2011-08-23 09:50:46 -07:00
|
|
|
type_ = "listen"
|
2010-08-25 10:26:40 -07:00
|
|
|
else:
|
2011-08-23 09:50:46 -07:00
|
|
|
type_ = "connection"
|
2010-08-25 10:26:40 -07:00
|
|
|
if error > 0:
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.warn("%s: %s attempt failed (%s)"
|
|
|
|
% (self.name, type_, os.strerror(error)))
|
2010-08-25 10:26:40 -07:00
|
|
|
else:
|
|
|
|
self.info_level("%s: %s attempt timed out"
|
2011-08-23 09:50:46 -07:00
|
|
|
% (self.name, type_))
|
2010-08-25 10:26:40 -07:00
|
|
|
|
2011-03-09 18:36:26 -08:00
|
|
|
if (self.state in (Reconnect.Active, Reconnect.Idle)):
|
|
|
|
self.last_disconnected = now
|
|
|
|
|
2018-01-22 11:04:58 -08:00
|
|
|
if not self.__may_retry():
|
|
|
|
self._transition(now, Reconnect.Void)
|
|
|
|
return
|
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
# Back off
|
2018-01-22 11:04:58 -08:00
|
|
|
if self.backoff_free_tries > 1:
|
|
|
|
self.backoff_free_tries -= 1
|
|
|
|
self.backoff = 0
|
|
|
|
elif (self.state in (Reconnect.Active, Reconnect.Idle) and
|
2012-08-08 13:32:57 -07:00
|
|
|
(self.last_activity - self.last_connected >= self.backoff or
|
2010-08-25 10:26:40 -07:00
|
|
|
self.passive)):
|
|
|
|
if self.passive:
|
|
|
|
self.backoff = 0
|
|
|
|
else:
|
|
|
|
self.backoff = self.min_backoff
|
|
|
|
else:
|
|
|
|
if self.backoff < self.min_backoff:
|
|
|
|
self.backoff = self.min_backoff
|
2018-01-22 11:04:58 -08:00
|
|
|
elif self.backoff < self.max_backoff / 2:
|
2010-08-25 10:26:40 -07:00
|
|
|
self.backoff *= 2
|
2018-01-22 11:04:58 -08:00
|
|
|
if self.passive:
|
|
|
|
action = "trying to listen again"
|
|
|
|
else:
|
|
|
|
action = "reconnect"
|
|
|
|
self.info_level("%s: waiting %.3g seconds before %s"
|
|
|
|
% (self.name, self.backoff / 1000.0,
|
|
|
|
action))
|
2010-08-25 10:26:40 -07:00
|
|
|
else:
|
2018-01-22 11:04:58 -08:00
|
|
|
if self.backoff < self.max_backoff:
|
|
|
|
if self.passive:
|
|
|
|
action = "try to listen"
|
|
|
|
else:
|
|
|
|
action = "reconnect"
|
|
|
|
self.info_level("%s: continuing to %s in the "
|
|
|
|
"background but suppressing further "
|
|
|
|
"logging" % (self.name, action))
|
|
|
|
self.backoff = self.max_backoff
|
|
|
|
self._transition(now, Reconnect.Backoff)
|
2010-08-25 10:26:40 -07:00
|
|
|
|
|
|
|
def connecting(self, now):
|
|
|
|
"""Tell this FSM that a connection or listening attempt is in progress.
|
|
|
|
|
|
|
|
The FSM will start a timer, after which the connection or listening
|
|
|
|
attempt will be aborted (by returning ovs.reconnect.DISCONNECT from
|
|
|
|
self.run())."""
|
|
|
|
if self.state != Reconnect.ConnectInProgress:
|
|
|
|
if self.passive:
|
|
|
|
self.info_level("%s: listening..." % self.name)
|
2018-01-22 11:04:58 -08:00
|
|
|
elif self.backoff < self.max_backoff:
|
2010-08-25 10:26:40 -07:00
|
|
|
self.info_level("%s: connecting..." % self.name)
|
|
|
|
self._transition(now, Reconnect.ConnectInProgress)
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
def listening(self, now):
|
|
|
|
"""Tell this FSM that the client is listening for connection attempts.
|
|
|
|
This state last indefinitely until the client reports some change.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
The natural progression from this state is for the client to report
|
|
|
|
that a connection has been accepted or is in progress of being
|
|
|
|
accepted, by calling self.connecting() or self.connected().
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
The client may also report that listening failed (e.g. accept()
|
|
|
|
returned an unexpected error such as ENOMEM) by calling
|
|
|
|
self.listen_error(), in which case the FSM will back off and eventually
|
|
|
|
return ovs.reconnect.CONNECT from self.run() to tell the client to try
|
|
|
|
listening again."""
|
|
|
|
if self.state != Reconnect.Listening:
|
|
|
|
self.info_level("%s: listening..." % self.name)
|
|
|
|
self._transition(now, Reconnect.Listening)
|
|
|
|
|
|
|
|
def listen_error(self, now, error):
|
|
|
|
"""Tell this FSM that the client's attempt to accept a connection
|
|
|
|
failed (e.g. accept() returned an unexpected error such as ENOMEM).
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
If the FSM is currently listening (self.listening() was called), it
|
|
|
|
will back off and eventually return ovs.reconnect.CONNECT from
|
|
|
|
self.run() to tell the client to try listening again. If there is an
|
|
|
|
active connection, this will be delayed until that connection drops."""
|
|
|
|
if self.state == Reconnect.Listening:
|
|
|
|
self.disconnected(now, error)
|
|
|
|
|
|
|
|
def connected(self, now):
|
|
|
|
"""Tell this FSM that the connection was successful.
|
|
|
|
|
|
|
|
The FSM will start the probe interval timer, which is reset by
|
2012-08-08 13:32:57 -07:00
|
|
|
self.activity(). If the timer expires, a probe will be sent (by
|
2010-08-25 10:26:40 -07:00
|
|
|
returning ovs.reconnect.PROBE from self.run(). If the timer expires
|
|
|
|
again without being reset, the connection will be aborted (by returning
|
|
|
|
ovs.reconnect.DISCONNECT from self.run()."""
|
|
|
|
if not self.state.is_connected:
|
|
|
|
self.connecting(now)
|
|
|
|
|
|
|
|
self.info_level("%s: connected" % self.name)
|
|
|
|
self._transition(now, Reconnect.Active)
|
|
|
|
self.last_connected = now
|
|
|
|
|
|
|
|
def connect_failed(self, now, error):
|
|
|
|
"""Tell this FSM that the connection attempt failed.
|
|
|
|
|
|
|
|
The FSM will back off and attempt to reconnect."""
|
|
|
|
self.connecting(now)
|
|
|
|
self.disconnected(now, error)
|
|
|
|
|
2012-08-08 13:32:57 -07:00
|
|
|
def activity(self, now):
|
|
|
|
"""Tell this FSM that some activity occurred on the connection. This
|
|
|
|
resets the probe interval timer, so that the connection is known not to
|
|
|
|
be idle."""
|
2010-08-25 10:26:40 -07:00
|
|
|
if self.state != Reconnect.Active:
|
|
|
|
self._transition(now, Reconnect.Active)
|
2012-08-08 13:32:57 -07:00
|
|
|
self.last_activity = now
|
2010-08-25 10:26:40 -07:00
|
|
|
|
2020-12-21 16:21:00 -08:00
|
|
|
def receive_attempted(self, now):
|
|
|
|
"""Tell 'fsm' that some attempt to receive data on the connection was
|
|
|
|
made at 'now'. The FSM only allows probe interval timer to expire when
|
|
|
|
some attempt to receive data on the connection was received after the
|
|
|
|
time when it should have expired. This helps in the case where there's
|
|
|
|
a long delay in the poll loop and then reconnect_run() executes before
|
|
|
|
the code to try to receive anything from the remote runs. (To disable
|
|
|
|
this feature, pass None for 'now'.)"""
|
|
|
|
self.last_receive_attempt = now
|
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
def _transition(self, now, state):
|
|
|
|
if self.state == Reconnect.ConnectInProgress:
|
|
|
|
self.n_attempted_connections += 1
|
|
|
|
if state == Reconnect.Active:
|
|
|
|
self.n_successful_connections += 1
|
|
|
|
|
|
|
|
connected_before = self.state.is_connected
|
|
|
|
connected_now = state.is_connected
|
|
|
|
if connected_before != connected_now:
|
|
|
|
if connected_before:
|
|
|
|
self.total_connected_duration += now - self.last_connected
|
|
|
|
self.seqno += 1
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2011-09-24 17:53:30 -07:00
|
|
|
vlog.dbg("%s: entering %s" % (self.name, state.name))
|
2010-08-25 10:26:40 -07:00
|
|
|
self.state = state
|
|
|
|
self.state_entered = now
|
|
|
|
|
|
|
|
def run(self, now):
|
|
|
|
"""Assesses whether any action should be taken on this FSM. The return
|
|
|
|
value is one of:
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
- None: The client need not take any action.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
- Active client, ovs.reconnect.CONNECT: The client should start a
|
|
|
|
connection attempt and indicate this by calling
|
|
|
|
self.connecting(). If the connection attempt has definitely
|
|
|
|
succeeded, it should call self.connected(). If the connection
|
|
|
|
attempt has definitely failed, it should call
|
|
|
|
self.connect_failed().
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
The FSM is smart enough to back off correctly after successful
|
|
|
|
connections that quickly abort, so it is OK to call
|
|
|
|
self.connected() after a low-level successful connection
|
|
|
|
(e.g. connect()) even if the connection might soon abort due to a
|
|
|
|
failure at a high-level (e.g. SSL negotiation failure).
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
- Passive client, ovs.reconnect.CONNECT: The client should try to
|
|
|
|
listen for a connection, if it is not already listening. It
|
|
|
|
should call self.listening() if successful, otherwise
|
|
|
|
self.connecting() or reconnected_connect_failed() if the attempt
|
|
|
|
is in progress or definitely failed, respectively.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
A listening passive client should constantly attempt to accept a
|
|
|
|
new connection and report an accepted connection with
|
|
|
|
self.connected().
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
- ovs.reconnect.DISCONNECT: The client should abort the current
|
|
|
|
connection or connection attempt or listen attempt and call
|
|
|
|
self.disconnected() or self.connect_failed() to indicate it.
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
- ovs.reconnect.PROBE: The client should send some kind of request
|
|
|
|
to the peer that will elicit a response, to ensure that the
|
|
|
|
connection is indeed in working order. (This will only be
|
|
|
|
returned if the "probe interval" is nonzero--see
|
|
|
|
self.set_probe_interval())."""
|
2012-04-11 20:42:01 -07:00
|
|
|
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
deadline = self.state.deadline(self, now)
|
2012-04-11 20:42:01 -07:00
|
|
|
if deadline is not None and now >= deadline:
|
2010-08-25 10:26:40 -07:00
|
|
|
return self.state.run(self, now)
|
|
|
|
else:
|
|
|
|
return None
|
2011-09-23 23:43:12 -07:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
def wait(self, poller, now):
|
|
|
|
"""Causes the next call to poller.block() to wake up when self.run()
|
|
|
|
should be called."""
|
|
|
|
timeout = self.timeout(now)
|
2015-12-17 09:56:44 -05:00
|
|
|
if timeout is not None and timeout >= 0:
|
2010-08-25 10:26:40 -07:00
|
|
|
poller.timer_wait(timeout)
|
|
|
|
|
|
|
|
def timeout(self, now):
|
|
|
|
"""Returns the number of milliseconds after which self.run() should be
|
2011-08-23 11:33:08 -07:00
|
|
|
called if nothing else notable happens in the meantime, or None if this
|
|
|
|
is currently unnecessary."""
|
reconnect: Fix broken inactivity probe if there is no other reason to wake up.
The purpose of reconnect_deadline__() function is twofold:
1. Its result is used to tell if the state has to be changed right now
in reconnect_run().
2. Its result also used to determine when the process need to wake up
and call reconnect_run() for a next time, i.e. when the state may
need to be changed next time.
Since introduction of the 'receive-attempted' feature, the function
returns LLONG_MAX if the deadline is in the future. That works for
the first case, but doesn't for the second one, because we don't
really know when we need to call reconnect_run().
This is the problem for applications where jsonrpc connection is the
only source of wake ups, e.g. ovn-northd. When the network goes down
silently, e.g. server looses IP address due to DHCP failure, ovn-northd
will sleep in the poll loop indefinitely after being told that it
doesn't need to call reconnect_run() (deadline == LLONG_MAX).
Fixing that by actually returning the expected time if it is in the
future, so we will know when to wake up. In order to keep the
'receive-attempted' feature, returning 'now + 1' in case where the
time has already passed, but receive wasn't attempted. That will
trigger a fast wake up, so the application will be able to attempt the
receive even if there was no real events. In a correctly written
application we should not fall into this case more than once in a row.
'+ 1' ensures that we will not transition into a different state
prematurely, i.e. before the receive is actually attempted.
Fixes: 4241d652e465 ("jsonrpc: Avoid disconnecting prematurely due to long poll intervals.")
Acked-by: Dumitru Ceara <dceara@redhat.com>
Signed-off-by: Ilya Maximets <i.maximets@ovn.org>
2022-02-21 13:59:51 +01:00
|
|
|
deadline = self.state.deadline(self, now)
|
2010-08-25 10:26:40 -07:00
|
|
|
if deadline is not None:
|
|
|
|
remaining = deadline - now
|
|
|
|
return max(0, remaining)
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
|
|
|
|
def is_connected(self):
|
|
|
|
"""Returns True if this FSM is currently believed to be connected, that
|
|
|
|
is, if self.connected() was called more recently than any call to
|
|
|
|
self.connect_failed() or self.disconnected() or self.disable(), and
|
|
|
|
False otherwise."""
|
|
|
|
return self.state.is_connected
|
|
|
|
|
2011-03-14 13:10:02 -07:00
|
|
|
def get_last_connect_elapsed(self, now):
|
|
|
|
"""Returns the number of milliseconds since 'fsm' was last connected
|
|
|
|
to its peer. Returns None if never connected."""
|
|
|
|
if self.last_connected:
|
2010-08-25 10:26:40 -07:00
|
|
|
return now - self.last_connected
|
|
|
|
else:
|
2011-03-14 13:10:02 -07:00
|
|
|
return None
|
2010-08-25 10:26:40 -07:00
|
|
|
|
2011-03-14 13:10:02 -07:00
|
|
|
def get_last_disconnect_elapsed(self, now):
|
|
|
|
"""Returns the number of milliseconds since 'fsm' was last disconnected
|
|
|
|
from its peer. Returns None if never disconnected."""
|
|
|
|
if self.last_disconnected:
|
2011-03-09 18:36:26 -08:00
|
|
|
return now - self.last_disconnected
|
|
|
|
else:
|
2011-03-14 13:10:02 -07:00
|
|
|
return None
|
2011-03-09 18:36:26 -08:00
|
|
|
|
2010-08-25 10:26:40 -07:00
|
|
|
def get_stats(self, now):
|
|
|
|
class Stats(object):
|
|
|
|
pass
|
|
|
|
stats = Stats()
|
|
|
|
stats.creation_time = self.creation_time
|
|
|
|
stats.last_connected = self.last_connected
|
2011-03-09 18:36:26 -08:00
|
|
|
stats.last_disconnected = self.last_disconnected
|
2012-08-08 13:32:57 -07:00
|
|
|
stats.last_activity = self.last_activity
|
2010-08-25 10:26:40 -07:00
|
|
|
stats.backoff = self.backoff
|
|
|
|
stats.seqno = self.seqno
|
|
|
|
stats.is_connected = self.is_connected()
|
2011-03-14 13:10:02 -07:00
|
|
|
stats.msec_since_connect = self.get_last_connect_elapsed(now)
|
|
|
|
stats.msec_since_disconnect = self.get_last_disconnect_elapsed(now)
|
2011-03-15 14:42:49 -07:00
|
|
|
stats.total_connected_duration = self.total_connected_duration
|
|
|
|
if self.is_connected():
|
2011-09-23 23:43:12 -07:00
|
|
|
stats.total_connected_duration += (
|
|
|
|
self.get_last_connect_elapsed(now))
|
2010-08-25 10:26:40 -07:00
|
|
|
stats.n_attempted_connections = self.n_attempted_connections
|
|
|
|
stats.n_successful_connections = self.n_successful_connections
|
|
|
|
stats.state = self.state.name
|
|
|
|
stats.state_elapsed = now - self.state_entered
|
|
|
|
return stats
|
|
|
|
|
|
|
|
def __may_retry(self):
|
|
|
|
if self.max_tries is None:
|
|
|
|
return True
|
|
|
|
elif self.max_tries > 0:
|
|
|
|
self.max_tries -= 1
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|