| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | #------------------------------------------------------------------------ | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Copyright (C) 2000 Autonomous Zone Industries | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # License:      This is free software.  You may use this software for any | 
					
						
							|  |  |  | #               purpose including modification/redistribution, so long as | 
					
						
							|  |  |  | #               this header remains intact and that you do not claim any | 
					
						
							|  |  |  | #               rights of ownership or authorship of this software.  This | 
					
						
							|  |  |  | #               software has been tested, but no warranty is expressed or | 
					
						
							|  |  |  | #               implied. | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Author: Gregory P. Smith <greg@electricrain.com> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Note: I don't know how useful this is in reality since when a | 
					
						
							| 
									
										
										
										
											2002-11-23 11:26:07 +00:00
										 |  |  | #       DBLockDeadlockError happens the current transaction is supposed to be | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | #       aborted.  If it doesn't then when the operation is attempted again | 
					
						
							|  |  |  | #       the deadlock is still happening... | 
					
						
							|  |  |  | #       --Robin | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #------------------------------------------------------------------------ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # import the time.sleep function in a namespace safe way to allow | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  | # "from bsddb.db import *" | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  | from time import sleep as _sleep | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  | from bsddb import _db | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  | # always sleep at least N seconds between retrys | 
					
						
							|  |  |  | _deadlock_MinSleepTime = 1.0/64 | 
					
						
							|  |  |  | # never sleep more than N seconds between retrys | 
					
						
							|  |  |  | _deadlock_MaxSleepTime = 3.14159 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Assign a file object to this for a "sleeping" message to be written to it | 
					
						
							|  |  |  | # each retry | 
					
						
							|  |  |  | _deadlock_VerboseFile = None | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def DeadlockWrap(function, *_args, **_kwargs): | 
					
						
							|  |  |  |     """DeadlockWrap(function, *_args, **_kwargs) - automatically retries
 | 
					
						
							|  |  |  |     function in case of a database deadlock. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-23 11:26:07 +00:00
										 |  |  |     This is a function intended to be used to wrap database calls such | 
					
						
							|  |  |  |     that they perform retrys with exponentially backing off sleeps in | 
					
						
							|  |  |  |     between when a DBLockDeadlockError exception is raised. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     A 'max_retries' parameter may optionally be passed to prevent it | 
					
						
							|  |  |  |     from retrying forever (in which case the exception will be reraised). | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         d = DB(...) | 
					
						
							|  |  |  |         d.open(...) | 
					
						
							|  |  |  |         DeadlockWrap(d.put, "foo", data="bar")  # set key "foo" to "bar" | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     sleeptime = _deadlock_MinSleepTime | 
					
						
							| 
									
										
										
										
											2002-11-23 11:26:07 +00:00
										 |  |  |     max_retries = _kwargs.get('max_retries', -1) | 
					
						
							|  |  |  |     if _kwargs.has_key('max_retries'): | 
					
						
							|  |  |  |         del _kwargs['max_retries'] | 
					
						
							|  |  |  |     while 1: | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  |             return function(*_args, **_kwargs) | 
					
						
							|  |  |  |         except _db.DBLockDeadlockError: | 
					
						
							| 
									
										
										
										
											2002-11-23 11:26:07 +00:00
										 |  |  |             if _deadlock_VerboseFile: | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  |                 _deadlock_VerboseFile.write( | 
					
						
							|  |  |  |                     'dbutils.DeadlockWrap: sleeping %1.3f\n' % sleeptime) | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  |             _sleep(sleeptime) | 
					
						
							|  |  |  |             # exponential backoff in the sleep time | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  |             sleeptime *= 2 | 
					
						
							|  |  |  |             if sleeptime > _deadlock_MaxSleepTime: | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  |                 sleeptime = _deadlock_MaxSleepTime | 
					
						
							| 
									
										
										
										
											2002-12-30 20:53:52 +00:00
										 |  |  |             max_retries -= 1 | 
					
						
							| 
									
										
										
										
											2002-11-23 11:26:07 +00:00
										 |  |  |             if max_retries == -1: | 
					
						
							|  |  |  |                 raise | 
					
						
							| 
									
										
										
										
											2002-11-19 08:09:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #------------------------------------------------------------------------ |