159
159
160
160
SELECT_TIMEOUT = 60
161
161
ERROR_RECOVERY_DELAY = 5
162
+ PG_ADVISORY_LOCK_ID = 2293787760715711918
162
163
163
164
_logger = logging .getLogger (__name__ )
164
165
165
166
select = selectors .DefaultSelector
166
167
167
168
169
+ class MasterElectionLost (Exception ):
170
+ pass
171
+
172
+
168
173
# Unfortunately, it is not possible to extend the Odoo
169
174
# server command line arguments, so we resort to environment variables
170
175
# to configure the runner (channels mostly).
@@ -268,6 +273,7 @@ def __init__(self, db_name):
268
273
self .conn .set_isolation_level (ISOLATION_LEVEL_AUTOCOMMIT )
269
274
self .has_queue_job = self ._has_queue_job ()
270
275
if self .has_queue_job :
276
+ self ._acquire_master_lock ()
271
277
self ._initialize ()
272
278
except BaseException :
273
279
self .close ()
@@ -284,6 +290,14 @@ def close(self):
284
290
pass
285
291
self .conn = None
286
292
293
+ def _acquire_master_lock (self ):
294
+ """Acquire the master runner lock or raise MasterElectionLost"""
295
+ with closing (self .conn .cursor ()) as cr :
296
+ cr .execute ("SELECT pg_try_advisory_lock(%s)" , (PG_ADVISORY_LOCK_ID ,))
297
+ if not cr .fetchone ()[0 ]:
298
+ msg = f"Could not acquire master runner lock on { self .db_name } "
299
+ raise MasterElectionLost (msg )
300
+
287
301
def _has_queue_job (self ):
288
302
with closing (self .conn .cursor ()) as cr :
289
303
cr .execute (
@@ -413,7 +427,7 @@ def get_db_names(self):
413
427
db_names = config ["db_name" ].split ("," )
414
428
else :
415
429
db_names = odoo .service .db .list_dbs (True )
416
- return db_names
430
+ return sorted ( db_names )
417
431
418
432
def close_databases (self , remove_jobs = True ):
419
433
for db_name , db in self .db_by_name .items ():
@@ -522,7 +536,7 @@ def run(self):
522
536
while not self ._stop :
523
537
# outer loop does exception recovery
524
538
try :
525
- _logger .info ("initializing database connections" )
539
+ _logger .debug ("initializing database connections" )
526
540
# TODO: how to detect new databases or databases
527
541
# on which queue_job is installed after server start?
528
542
self .initialize_databases ()
@@ -537,6 +551,14 @@ def run(self):
537
551
except InterruptedError :
538
552
# Interrupted system call, i.e. KeyboardInterrupt during select
539
553
self .stop ()
554
+ except MasterElectionLost as e :
555
+ _logger .debug (
556
+ "master election lost: %s, sleeping %ds and retrying" ,
557
+ e ,
558
+ ERROR_RECOVERY_DELAY ,
559
+ )
560
+ self .close_databases ()
561
+ time .sleep (ERROR_RECOVERY_DELAY )
540
562
except Exception :
541
563
_logger .exception (
542
564
"exception: sleeping %ds and retrying" , ERROR_RECOVERY_DELAY
0 commit comments