 07cc4d056e
			
		
	
	07cc4d056e
	
	
	
		
			
			* Adds a GreenLockFile that always works in greenthreads * Adds test to verify that regular Lockfile is broken * Adds test to verify that GreenLockfile works * Adds note about limitation of external locks * Adds test showing limitation of nested locks * Fixes bug 956313 Change-Id: I11cd1206611aa4862dadd2fcc077c4c2e0f798f6
		
			
				
	
	
		
			185 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			185 lines
		
	
	
		
			6.2 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| # vim: tabstop=4 shiftwidth=4 softtabstop=4
 | |
| 
 | |
| #    Copyright 2010 OpenStack LLC
 | |
| #
 | |
| #    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 commands
 | |
| import errno
 | |
| import glob
 | |
| import os
 | |
| import select
 | |
| 
 | |
| from eventlet import greenpool
 | |
| from eventlet import greenthread
 | |
| import lockfile
 | |
| 
 | |
| from nova import exception
 | |
| from nova import test
 | |
| from nova import utils
 | |
| 
 | |
| 
 | |
| class ExceptionTestCase(test.TestCase):
 | |
|     @staticmethod
 | |
|     def _raise_exc(exc):
 | |
|         raise exc()
 | |
| 
 | |
|     def test_exceptions_raise(self):
 | |
|         for name in dir(exception):
 | |
|             exc = getattr(exception, name)
 | |
|             if isinstance(exc, type):
 | |
|                 self.assertRaises(exc, self._raise_exc, exc)
 | |
| 
 | |
| 
 | |
| class ProjectTestCase(test.TestCase):
 | |
|     def test_authors_up_to_date(self):
 | |
|         topdir = os.path.normpath(os.path.dirname(__file__) + '/../../')
 | |
|         missing = set()
 | |
|         contributors = set()
 | |
|         mailmap = utils.parse_mailmap(os.path.join(topdir, '.mailmap'))
 | |
|         authors_file = open(os.path.join(topdir,
 | |
|                                          'Authors'), 'r').read().lower()
 | |
| 
 | |
|         if os.path.exists(os.path.join(topdir, '.git')):
 | |
|             for email in commands.getoutput('git log --format=%ae').split():
 | |
|                 if not email:
 | |
|                     continue
 | |
|                 if "jenkins" in email and "openstack.org" in email:
 | |
|                     continue
 | |
|                 email = '<' + email.lower() + '>'
 | |
|                 contributors.add(utils.str_dict_replace(email, mailmap))
 | |
|         else:
 | |
|             return
 | |
| 
 | |
|         for contributor in contributors:
 | |
|             if contributor == 'nova-core':
 | |
|                 continue
 | |
|             if not contributor in authors_file:
 | |
|                 missing.add(contributor)
 | |
| 
 | |
|         self.assertTrue(len(missing) == 0,
 | |
|                         '%r not listed in Authors' % missing)
 | |
| 
 | |
|     def test_all_migrations_have_downgrade(self):
 | |
|         topdir = os.path.normpath(os.path.dirname(__file__) + '/../../')
 | |
|         py_glob = os.path.join(topdir, "nova", "db", "sqlalchemy",
 | |
|                                "migrate_repo", "versions", "*.py")
 | |
|         missing_downgrade = []
 | |
|         for path in glob.iglob(py_glob):
 | |
|             has_upgrade = False
 | |
|             has_downgrade = False
 | |
|             with open(path, "r") as f:
 | |
|                 for line in f:
 | |
|                     if 'def upgrade(' in line:
 | |
|                         has_upgrade = True
 | |
|                     if 'def downgrade(' in line:
 | |
|                         has_downgrade = True
 | |
| 
 | |
|                 if has_upgrade and not has_downgrade:
 | |
|                     fname = os.path.basename(path)
 | |
|                     missing_downgrade.append(fname)
 | |
| 
 | |
|         helpful_msg = (_("The following migrations are missing a downgrade:"
 | |
|                          "\n\t%s") % '\n\t'.join(sorted(missing_downgrade)))
 | |
|         self.assert_(not missing_downgrade, helpful_msg)
 | |
| 
 | |
| 
 | |
| class LockTestCase(test.TestCase):
 | |
|     def test_synchronized_wrapped_function_metadata(self):
 | |
|         @utils.synchronized('whatever')
 | |
|         def foo():
 | |
|             """Bar"""
 | |
|             pass
 | |
|         self.assertEquals(foo.__doc__, 'Bar', "Wrapped function's docstring "
 | |
|                                               "got lost")
 | |
|         self.assertEquals(foo.__name__, 'foo', "Wrapped function's name "
 | |
|                                                "got mangled")
 | |
| 
 | |
|     def test_synchronized_internally(self):
 | |
|         """We can lock across multiple green threads"""
 | |
|         saved_sem_num = len(utils._semaphores)
 | |
|         seen_threads = list()
 | |
| 
 | |
|         @utils.synchronized('testlock2', external=False)
 | |
|         def f(id):
 | |
|             for x in range(10):
 | |
|                 seen_threads.append(id)
 | |
|                 greenthread.sleep(0)
 | |
| 
 | |
|         threads = []
 | |
|         pool = greenpool.GreenPool(10)
 | |
|         for i in range(10):
 | |
|             threads.append(pool.spawn(f, i))
 | |
| 
 | |
|         for thread in threads:
 | |
|             thread.wait()
 | |
| 
 | |
|         self.assertEquals(len(seen_threads), 100)
 | |
|         # Looking at the seen threads, split it into chunks of 10, and verify
 | |
|         # that the last 9 match the first in each chunk.
 | |
|         for i in range(10):
 | |
|             for j in range(9):
 | |
|                 self.assertEquals(seen_threads[i * 10],
 | |
|                                   seen_threads[i * 10 + 1 + j])
 | |
| 
 | |
|         self.assertEqual(saved_sem_num, len(utils._semaphores),
 | |
|                          "Semaphore leak detected")
 | |
| 
 | |
|     def test_nested_external_fails(self):
 | |
|         """We can not nest external syncs"""
 | |
| 
 | |
|         @utils.synchronized('testlock1', external=True)
 | |
|         def outer_lock():
 | |
| 
 | |
|             @utils.synchronized('testlock2', external=True)
 | |
|             def inner_lock():
 | |
|                 pass
 | |
|             inner_lock()
 | |
|         try:
 | |
|             self.assertRaises(lockfile.NotMyLock, outer_lock)
 | |
|         finally:
 | |
|             utils.cleanup_file_locks()
 | |
| 
 | |
|     def test_synchronized_externally(self):
 | |
|         """We can lock across multiple processes"""
 | |
|         rpipe1, wpipe1 = os.pipe()
 | |
|         rpipe2, wpipe2 = os.pipe()
 | |
| 
 | |
|         @utils.synchronized('testlock1', external=True)
 | |
|         def f(rpipe, wpipe):
 | |
|             try:
 | |
|                 os.write(wpipe, "foo")
 | |
|             except OSError, e:
 | |
|                 self.assertEquals(e.errno, errno.EPIPE)
 | |
|                 return
 | |
| 
 | |
|             rfds, _wfds, _efds = select.select([rpipe], [], [], 1)
 | |
|             self.assertEquals(len(rfds), 0, "The other process, which was"
 | |
|                                             " supposed to be locked, "
 | |
|                                             "wrote on its end of the "
 | |
|                                             "pipe")
 | |
|             os.close(rpipe)
 | |
| 
 | |
|         pid = os.fork()
 | |
|         if pid > 0:
 | |
|             os.close(wpipe1)
 | |
|             os.close(rpipe2)
 | |
| 
 | |
|             f(rpipe1, wpipe2)
 | |
|         else:
 | |
|             os.close(rpipe1)
 | |
|             os.close(wpipe2)
 | |
| 
 | |
|             f(rpipe2, wpipe1)
 | |
|             os._exit(0)
 |