| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | """Core implementation of import.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This module is NOT meant to be directly imported! It has been designed such | 
					
						
							|  |  |  | that it can be bootstrapped into Python as the implementation of import. As | 
					
						
							|  |  |  | such it requires the injection of specific modules and attributes in order to | 
					
						
							|  |  |  | work. One should use importlib as the public-facing version of this module. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2012-04-25 10:54:48 +02:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2019-02-13 15:21:17 -08:00
										 |  |  | # IMPORTANT: Whenever making changes to this module, be sure to run a top-level | 
					
						
							|  |  |  | # `make regen-importlib` followed by `make` in order to get the frozen version | 
					
						
							|  |  |  | # of the module updated. Not doing so will result in the Makefile to fail for | 
					
						
							| 
									
										
										
										
											2012-04-25 10:54:48 +02:00
										 |  |  | # all others who don't have a ./python around to freeze the module | 
					
						
							|  |  |  | # in the early stages of compilation. | 
					
						
							|  |  |  | # | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-20 21:19:53 -04:00
										 |  |  | # See importlib._setup() for what is injected into the global namespace. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | # When editing this code be aware that code executed at import time CANNOT | 
					
						
							|  |  |  | # reference any injected objects! This includes not only global code but also | 
					
						
							|  |  |  | # anything specified at the class level. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  | def _object_name(obj): | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         return obj.__qualname__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         return type(obj).__qualname__ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 22:28:55 +00:00
										 |  |  | # Bootstrap-related code ###################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-19 13:43:43 +01:00
										 |  |  | # Modules injected manually by _setup() | 
					
						
							|  |  |  | _thread = None | 
					
						
							|  |  |  | _warnings = None | 
					
						
							|  |  |  | _weakref = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # Import done by _install_external_importers() | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  | _bootstrap_external = None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-19 13:43:43 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 22:07:17 +00:00
										 |  |  | def _wrap(new, old): | 
					
						
							| 
									
										
										
										
											2012-07-07 13:16:44 +02:00
										 |  |  |     """Simple substitute for functools.update_wrapper.""" | 
					
						
							| 
									
										
										
										
											2011-12-14 22:53:13 -06:00
										 |  |  |     for replace in ['__module__', '__name__', '__qualname__', '__doc__']: | 
					
						
							| 
									
										
										
										
											2012-02-08 18:44:14 -05:00
										 |  |  |         if hasattr(old, replace): | 
					
						
							|  |  |  |             setattr(new, replace, getattr(old, replace)) | 
					
						
							| 
									
										
										
										
											2009-02-07 02:13:28 +00:00
										 |  |  |     new.__dict__.update(old.__dict__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | def _new_module(name): | 
					
						
							|  |  |  |     return type(sys)(name) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | # Module-level locking ######################################################## | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # A dict mapping module names to weakrefs of _ModuleLock instances | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  | # Dictionary protected by the global import lock | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | _module_locks = {} | 
					
						
							|  |  |  | # A dict mapping thread ids to _ModuleLock instances | 
					
						
							|  |  |  | _blocking_on = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _DeadlockError(RuntimeError): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _ModuleLock: | 
					
						
							|  |  |  |     """A recursive lock implementation which is able to detect deadlocks
 | 
					
						
							|  |  |  |     (e.g. thread 1 trying to take locks A then B, and thread 2 trying to | 
					
						
							|  |  |  |     take locks B then A). | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name): | 
					
						
							|  |  |  |         self.lock = _thread.allocate_lock() | 
					
						
							|  |  |  |         self.wakeup = _thread.allocate_lock() | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.owner = None | 
					
						
							|  |  |  |         self.count = 0 | 
					
						
							|  |  |  |         self.waiters = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def has_deadlock(self): | 
					
						
							|  |  |  |         # Deadlock avoidance for concurrent circular imports. | 
					
						
							|  |  |  |         me = _thread.get_ident() | 
					
						
							|  |  |  |         tid = self.owner | 
					
						
							| 
									
										
										
										
											2020-03-03 02:37:25 +01:00
										 |  |  |         seen = set() | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |         while True: | 
					
						
							|  |  |  |             lock = _blocking_on.get(tid) | 
					
						
							|  |  |  |             if lock is None: | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |             tid = lock.owner | 
					
						
							|  |  |  |             if tid == me: | 
					
						
							|  |  |  |                 return True | 
					
						
							| 
									
										
										
										
											2020-03-03 02:37:25 +01:00
										 |  |  |             if tid in seen: | 
					
						
							|  |  |  |                 # bpo 38091: the chain of tid's we encounter here | 
					
						
							|  |  |  |                 # eventually leads to a fixpoint or a cycle, but | 
					
						
							|  |  |  |                 # does not reach 'me'.  This means we would not | 
					
						
							|  |  |  |                 # actually deadlock.  This can happen if other | 
					
						
							|  |  |  |                 # threads are at the beginning of acquire() below. | 
					
						
							|  |  |  |                 return False | 
					
						
							|  |  |  |             seen.add(tid) | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def acquire(self): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Acquire the module lock.  If a potential deadlock is detected, | 
					
						
							|  |  |  |         a _DeadlockError is raised. | 
					
						
							|  |  |  |         Otherwise, the lock is always acquired and True is returned. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         tid = _thread.get_ident() | 
					
						
							|  |  |  |         _blocking_on[tid] = self | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             while True: | 
					
						
							|  |  |  |                 with self.lock: | 
					
						
							|  |  |  |                     if self.count == 0 or self.owner == tid: | 
					
						
							|  |  |  |                         self.owner = tid | 
					
						
							|  |  |  |                         self.count += 1 | 
					
						
							|  |  |  |                         return True | 
					
						
							|  |  |  |                     if self.has_deadlock(): | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |                         raise _DeadlockError('deadlock detected by %r' % self) | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |                     if self.wakeup.acquire(False): | 
					
						
							|  |  |  |                         self.waiters += 1 | 
					
						
							|  |  |  |                 # Wait for a release() call | 
					
						
							|  |  |  |                 self.wakeup.acquire() | 
					
						
							|  |  |  |                 self.wakeup.release() | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             del _blocking_on[tid] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def release(self): | 
					
						
							|  |  |  |         tid = _thread.get_ident() | 
					
						
							|  |  |  |         with self.lock: | 
					
						
							|  |  |  |             if self.owner != tid: | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |                 raise RuntimeError('cannot release un-acquired lock') | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |             assert self.count > 0 | 
					
						
							|  |  |  |             self.count -= 1 | 
					
						
							|  |  |  |             if self.count == 0: | 
					
						
							|  |  |  |                 self.owner = None | 
					
						
							|  |  |  |                 if self.waiters: | 
					
						
							|  |  |  |                     self.waiters -= 1 | 
					
						
							|  |  |  |                     self.wakeup.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |         return '_ModuleLock({!r}) at {}'.format(self.name, id(self)) | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class _DummyModuleLock: | 
					
						
							|  |  |  |     """A simple _ModuleLock equivalent for Python builds without
 | 
					
						
							|  |  |  |     multi-threading support."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.count = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def acquire(self): | 
					
						
							|  |  |  |         self.count += 1 | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def release(self): | 
					
						
							|  |  |  |         if self.count == 0: | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |             raise RuntimeError('cannot release un-acquired lock') | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |         self.count -= 1 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |         return '_DummyModuleLock({!r}) at {}'.format(self.name, id(self)) | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | class _ModuleLockManager: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name): | 
					
						
							|  |  |  |         self._name = name | 
					
						
							|  |  |  |         self._lock = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |         self._lock = _get_module_lock(self._name) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         self._lock.acquire() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, *args, **kwargs): | 
					
						
							|  |  |  |         self._lock.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | # The following two functions are for consumption by Python/import.c. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _get_module_lock(name): | 
					
						
							|  |  |  |     """Get or create the module lock for a given module name.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |     Acquire/release internally the global import lock to protect | 
					
						
							|  |  |  |     _module_locks."""
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _imp.acquire_lock() | 
					
						
							| 
									
										
										
										
											2012-08-28 00:24:52 +02:00
										 |  |  |     try: | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             lock = _module_locks[name]() | 
					
						
							|  |  |  |         except KeyError: | 
					
						
							|  |  |  |             lock = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if lock is None: | 
					
						
							|  |  |  |             if _thread is None: | 
					
						
							|  |  |  |                 lock = _DummyModuleLock(name) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 lock = _ModuleLock(name) | 
					
						
							| 
									
										
										
										
											2017-08-09 14:29:12 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  |             def cb(ref, name=name): | 
					
						
							|  |  |  |                 _imp.acquire_lock() | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     # bpo-31070: Check if another thread created a new lock | 
					
						
							|  |  |  |                     # after the previous lock was destroyed | 
					
						
							|  |  |  |                     # but before the weakref callback was called. | 
					
						
							|  |  |  |                     if _module_locks.get(name) is ref: | 
					
						
							|  |  |  |                         del _module_locks[name] | 
					
						
							|  |  |  |                 finally: | 
					
						
							|  |  |  |                     _imp.release_lock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |             _module_locks[name] = _weakref.ref(lock, cb) | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         _imp.release_lock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |     return lock | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | def _lock_unlock_module(name): | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |     """Acquires then releases the module lock for a given module name.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |     This is used to ensure a module is completely initialized, in the | 
					
						
							|  |  |  |     event it is being imported by another thread. | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |     lock = _get_module_lock(name) | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         lock.acquire() | 
					
						
							|  |  |  |     except _DeadlockError: | 
					
						
							|  |  |  |         # Concurrent circular import, we'll accept a partially initialized | 
					
						
							|  |  |  |         # module object. | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         lock.release() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-31 21:14:18 +10:00
										 |  |  | # Frame stripping magic ############################################### | 
					
						
							|  |  |  | def _call_with_frames_removed(f, *args, **kwds): | 
					
						
							|  |  |  |     """remove_importlib_frames in import.c will always remove sequences
 | 
					
						
							|  |  |  |     of importlib frames that end with a call to this function | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Use it instead of a normal call in places where including the importlib | 
					
						
							|  |  |  |     frames introduces unwanted noise into the traceback (e.g. when executing | 
					
						
							|  |  |  |     module code) | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     return f(*args, **kwds) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-04-01 13:10:51 -04:00
										 |  |  | def _verbose_message(message, *args, verbosity=1): | 
					
						
							| 
									
										
										
										
											2012-04-14 14:10:13 -04:00
										 |  |  |     """Print the message to stderr if -v/PYTHONVERBOSE is turned on.""" | 
					
						
							| 
									
										
										
										
											2013-04-01 13:10:51 -04:00
										 |  |  |     if sys.flags.verbose >= verbosity: | 
					
						
							| 
									
										
										
										
											2012-04-15 12:21:32 -07:00
										 |  |  |         if not message.startswith(('#', 'import ')): | 
					
						
							| 
									
										
										
										
											2012-04-14 14:10:13 -04:00
										 |  |  |             message = '# ' + message | 
					
						
							|  |  |  |         print(message.format(*args), file=sys.stderr) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-20 01:48:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  | def _requires_builtin(fxn): | 
					
						
							|  |  |  |     """Decorator to verify the named module is built-in.""" | 
					
						
							| 
									
										
										
										
											2012-01-16 11:46:22 -05:00
										 |  |  |     def _requires_builtin_wrapper(self, fullname): | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  |         if fullname not in sys.builtin_module_names: | 
					
						
							| 
									
										
										
										
											2013-11-22 14:52:36 -05:00
										 |  |  |             raise ImportError('{!r} is not a built-in module'.format(fullname), | 
					
						
							| 
									
										
										
										
											2012-04-12 21:09:01 -04:00
										 |  |  |                               name=fullname) | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  |         return fxn(self, fullname) | 
					
						
							| 
									
										
										
										
											2012-01-16 11:46:22 -05:00
										 |  |  |     _wrap(_requires_builtin_wrapper, fxn) | 
					
						
							|  |  |  |     return _requires_builtin_wrapper | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  | def _requires_frozen(fxn): | 
					
						
							|  |  |  |     """Decorator to verify the named module is frozen.""" | 
					
						
							| 
									
										
										
										
											2012-01-16 11:46:22 -05:00
										 |  |  |     def _requires_frozen_wrapper(self, fullname): | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         if not _imp.is_frozen(fullname): | 
					
						
							| 
									
										
										
										
											2013-11-22 14:52:36 -05:00
										 |  |  |             raise ImportError('{!r} is not a frozen module'.format(fullname), | 
					
						
							| 
									
										
										
										
											2012-04-12 21:09:01 -04:00
										 |  |  |                               name=fullname) | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  |         return fxn(self, fullname) | 
					
						
							| 
									
										
										
										
											2012-01-16 11:46:22 -05:00
										 |  |  |     _wrap(_requires_frozen_wrapper, fxn) | 
					
						
							|  |  |  |     return _requires_frozen_wrapper | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 16:28:00 -04:00
										 |  |  | # Typically used by loader classes as a method replacement. | 
					
						
							|  |  |  | def _load_module_shim(self, fullname): | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |     """Load the specified module into sys.modules and return it.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-24 08:26:56 -07:00
										 |  |  |     This method is deprecated.  Use loader.exec_module() instead. | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  |     msg = ("the load_module() method is deprecated and slated for removal in " | 
					
						
							|  |  |  |           "Python 3.12; use exec_module() instead") | 
					
						
							|  |  |  |     _warnings.warn(msg, DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2014-05-30 16:28:00 -04:00
										 |  |  |     spec = spec_from_loader(fullname, self) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     if fullname in sys.modules: | 
					
						
							|  |  |  |         module = sys.modules[fullname] | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |         _exec(spec, module) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         return sys.modules[fullname] | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |         return _load(spec) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Module specifications ####################################################### | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _module_repr(module): | 
					
						
							| 
									
										
										
										
											2021-03-24 08:26:56 -07:00
										 |  |  |     """The implementation of ModuleType.__repr__().""" | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     loader = getattr(module, '__loader__', None) | 
					
						
							| 
									
										
										
										
											2021-03-24 08:26:56 -07:00
										 |  |  |     if spec := getattr(module, "__spec__", None): | 
					
						
							|  |  |  |         return _module_repr_from_spec(spec) | 
					
						
							|  |  |  |     elif hasattr(loader, 'module_repr'): | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         try: | 
					
						
							|  |  |  |             return loader.module_repr(module) | 
					
						
							|  |  |  |         except Exception: | 
					
						
							|  |  |  |             pass | 
					
						
							| 
									
										
										
										
											2021-03-24 08:26:56 -07:00
										 |  |  |     # Fall through to a catch-all which always succeeds. | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     try: | 
					
						
							|  |  |  |         name = module.__name__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         name = '?' | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         filename = module.__file__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         if loader is None: | 
					
						
							|  |  |  |             return '<module {!r}>'.format(name) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return '<module {!r} ({!r})>'.format(name, loader) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         return '<module {!r} from {!r}>'.format(name, filename) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class ModuleSpec: | 
					
						
							|  |  |  |     """The specification for a module, used for loading.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     A module's spec is the source for information about the module.  For | 
					
						
							|  |  |  |     data associated with the module, including source, use the spec's | 
					
						
							|  |  |  |     loader. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `name` is the absolute name of the module.  `loader` is the loader | 
					
						
							|  |  |  |     to use when loading the module.  `parent` is the name of the | 
					
						
							|  |  |  |     package the module is in.  The parent is derived from the name. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `is_package` determines if the module is considered a package or | 
					
						
							|  |  |  |     not.  On modules this is reflected by the `__path__` attribute. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `origin` is the specific location used by the loader from which to | 
					
						
							|  |  |  |     load the module, if that information is available.  When filename is | 
					
						
							|  |  |  |     set, origin will match. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `has_location` indicates that a spec's "origin" reflects a location. | 
					
						
							|  |  |  |     When this is True, `__file__` attribute of the module is set. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `cached` is the location of the cached bytecode file, if any.  It | 
					
						
							|  |  |  |     corresponds to the `__cached__` attribute. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     `submodule_search_locations` is the sequence of path entries to | 
					
						
							|  |  |  |     search when importing submodules.  If set, is_package should be | 
					
						
							|  |  |  |     True--and False otherwise. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Packages are simply modules that (may) have submodules.  If a spec | 
					
						
							|  |  |  |     has a non-None value in `submodule_search_locations`, the import | 
					
						
							|  |  |  |     system will consider modules loaded from the spec as packages. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Only finders (see importlib.abc.MetaPathFinder and | 
					
						
							|  |  |  |     importlib.abc.PathEntryFinder) should modify ModuleSpec instances. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self, name, loader, *, origin=None, loader_state=None, | 
					
						
							|  |  |  |                  is_package=None): | 
					
						
							|  |  |  |         self.name = name | 
					
						
							|  |  |  |         self.loader = loader | 
					
						
							|  |  |  |         self.origin = origin | 
					
						
							|  |  |  |         self.loader_state = loader_state | 
					
						
							|  |  |  |         self.submodule_search_locations = [] if is_package else None | 
					
						
							| 
									
										
										
										
											2021-07-24 23:44:46 +01:00
										 |  |  |         self._uninitialized_submodules = [] | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # file-location attributes | 
					
						
							|  |  |  |         self._set_fileattr = False | 
					
						
							|  |  |  |         self._cached = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __repr__(self): | 
					
						
							|  |  |  |         args = ['name={!r}'.format(self.name), | 
					
						
							|  |  |  |                 'loader={!r}'.format(self.loader)] | 
					
						
							|  |  |  |         if self.origin is not None: | 
					
						
							|  |  |  |             args.append('origin={!r}'.format(self.origin)) | 
					
						
							|  |  |  |         if self.submodule_search_locations is not None: | 
					
						
							|  |  |  |             args.append('submodule_search_locations={}' | 
					
						
							|  |  |  |                         .format(self.submodule_search_locations)) | 
					
						
							|  |  |  |         return '{}({})'.format(self.__class__.__name__, ', '.join(args)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __eq__(self, other): | 
					
						
							|  |  |  |         smsl = self.submodule_search_locations | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             return (self.name == other.name and | 
					
						
							|  |  |  |                     self.loader == other.loader and | 
					
						
							|  |  |  |                     self.origin == other.origin and | 
					
						
							|  |  |  |                     smsl == other.submodule_search_locations and | 
					
						
							|  |  |  |                     self.cached == other.cached and | 
					
						
							|  |  |  |                     self.has_location == other.has_location) | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2019-08-08 08:42:54 +03:00
										 |  |  |             return NotImplemented | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def cached(self): | 
					
						
							|  |  |  |         if self._cached is None: | 
					
						
							|  |  |  |             if self.origin is not None and self._set_fileattr: | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  |                 if _bootstrap_external is None: | 
					
						
							|  |  |  |                     raise NotImplementedError | 
					
						
							| 
									
										
										
										
											2015-05-02 19:15:18 -06:00
										 |  |  |                 self._cached = _bootstrap_external._get_cached(self.origin) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         return self._cached | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @cached.setter | 
					
						
							|  |  |  |     def cached(self, cached): | 
					
						
							|  |  |  |         self._cached = cached | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def parent(self): | 
					
						
							|  |  |  |         """The name of the module's parent.""" | 
					
						
							|  |  |  |         if self.submodule_search_locations is None: | 
					
						
							|  |  |  |             return self.name.rpartition('.')[0] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return self.name | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @property | 
					
						
							|  |  |  |     def has_location(self): | 
					
						
							|  |  |  |         return self._set_fileattr | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-12-10 22:16:41 -07:00
										 |  |  |     @has_location.setter | 
					
						
							|  |  |  |     def has_location(self, value): | 
					
						
							|  |  |  |         self._set_fileattr = bool(value) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | def spec_from_loader(name, loader, *, origin=None, is_package=None): | 
					
						
							|  |  |  |     """Return a module spec based on various loader methods.""" | 
					
						
							|  |  |  |     if hasattr(loader, 'get_filename'): | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  |         if _bootstrap_external is None: | 
					
						
							|  |  |  |             raise NotImplementedError | 
					
						
							|  |  |  |         spec_from_file_location = _bootstrap_external.spec_from_file_location | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         if is_package is None: | 
					
						
							|  |  |  |             return spec_from_file_location(name, loader=loader) | 
					
						
							|  |  |  |         search = [] if is_package else None | 
					
						
							|  |  |  |         return spec_from_file_location(name, loader=loader, | 
					
						
							|  |  |  |                                        submodule_search_locations=search) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if is_package is None: | 
					
						
							|  |  |  |         if hasattr(loader, 'is_package'): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 is_package = loader.is_package(name) | 
					
						
							|  |  |  |             except ImportError: | 
					
						
							|  |  |  |                 is_package = None  # aka, undefined | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # the default | 
					
						
							|  |  |  |             is_package = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ModuleSpec(name, loader, origin=origin, is_package=is_package) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _spec_from_module(module, loader=None, origin=None): | 
					
						
							|  |  |  |     # This function is meant for use in _setup(). | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         spec = module.__spec__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         if spec is not None: | 
					
						
							|  |  |  |             return spec | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     name = module.__name__ | 
					
						
							|  |  |  |     if loader is None: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             loader = module.__loader__ | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             # loader will stay None. | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         location = module.__file__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         location = None | 
					
						
							|  |  |  |     if origin is None: | 
					
						
							|  |  |  |         if location is None: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 origin = loader._ORIGIN | 
					
						
							|  |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 origin = None | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             origin = location | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         cached = module.__cached__ | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         cached = None | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         submodule_search_locations = list(module.__path__) | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         submodule_search_locations = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     spec = ModuleSpec(name, loader, origin=origin) | 
					
						
							|  |  |  |     spec._set_fileattr = False if location is None else True | 
					
						
							|  |  |  |     spec.cached = cached | 
					
						
							|  |  |  |     spec.submodule_search_locations = submodule_search_locations | 
					
						
							|  |  |  |     return spec | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | def _init_module_attrs(spec, module, *, override=False): | 
					
						
							|  |  |  |     # The passed-in module may be not support attribute assignment, | 
					
						
							|  |  |  |     # in which case we simply don't set the attributes. | 
					
						
							|  |  |  |     # __name__ | 
					
						
							|  |  |  |     if (override or getattr(module, '__name__', None) is None): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             module.__name__ = spec.name | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     # __loader__ | 
					
						
							|  |  |  |     if override or getattr(module, '__loader__', None) is None: | 
					
						
							|  |  |  |         loader = spec.loader | 
					
						
							|  |  |  |         if loader is None: | 
					
						
							|  |  |  |             # A backward compatibility hack. | 
					
						
							|  |  |  |             if spec.submodule_search_locations is not None: | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  |                 if _bootstrap_external is None: | 
					
						
							|  |  |  |                     raise NotImplementedError | 
					
						
							|  |  |  |                 _NamespaceLoader = _bootstrap_external._NamespaceLoader | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |                 loader = _NamespaceLoader.__new__(_NamespaceLoader) | 
					
						
							|  |  |  |                 loader._path = spec.submodule_search_locations | 
					
						
							| 
									
										
										
										
											2018-02-02 15:15:58 -05:00
										 |  |  |                 spec.loader = loader | 
					
						
							|  |  |  |                 # While the docs say that module.__file__ is not set for | 
					
						
							|  |  |  |                 # built-in modules, and the code below will avoid setting it if | 
					
						
							|  |  |  |                 # spec.has_location is false, this is incorrect for namespace | 
					
						
							|  |  |  |                 # packages.  Namespace packages have no location, but their | 
					
						
							|  |  |  |                 # __spec__.origin is None, and thus their module.__file__ | 
					
						
							|  |  |  |                 # should also be None for consistency.  While a bit of a hack, | 
					
						
							|  |  |  |                 # this is the best place to ensure this consistency. | 
					
						
							|  |  |  |                 # | 
					
						
							|  |  |  |                 # See # https://docs.python.org/3/library/importlib.html#importlib.abc.Loader.load_module | 
					
						
							|  |  |  |                 # and bpo-32305 | 
					
						
							|  |  |  |                 module.__file__ = None | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |         try: | 
					
						
							|  |  |  |             module.__loader__ = loader | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     # __package__ | 
					
						
							|  |  |  |     if override or getattr(module, '__package__', None) is None: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             module.__package__ = spec.parent | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     # __spec__ | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         module.__spec__ = spec | 
					
						
							|  |  |  |     except AttributeError: | 
					
						
							|  |  |  |         pass | 
					
						
							|  |  |  |     # __path__ | 
					
						
							|  |  |  |     if override or getattr(module, '__path__', None) is None: | 
					
						
							|  |  |  |         if spec.submodule_search_locations is not None: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |                 module.__path__ = spec.submodule_search_locations | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 pass | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     # __file__/__cached__ | 
					
						
							|  |  |  |     if spec.has_location: | 
					
						
							|  |  |  |         if override or getattr(module, '__file__', None) is None: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             try: | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |                 module.__file__ = spec.origin | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |         if override or getattr(module, '__cached__', None) is None: | 
					
						
							|  |  |  |             if spec.cached is not None: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                 try: | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |                     module.__cached__ = spec.cached | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                 except AttributeError: | 
					
						
							|  |  |  |                     pass | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     return module | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | def module_from_spec(spec): | 
					
						
							|  |  |  |     """Create a module based on the provided spec.""" | 
					
						
							|  |  |  |     # Typically loaders will not implement create_module(). | 
					
						
							|  |  |  |     module = None | 
					
						
							|  |  |  |     if hasattr(spec.loader, 'create_module'): | 
					
						
							|  |  |  |         # If create_module() returns `None` then it means default | 
					
						
							|  |  |  |         # module creation should be used. | 
					
						
							|  |  |  |         module = spec.loader.create_module(spec) | 
					
						
							| 
									
										
										
										
											2015-01-09 11:39:21 -05:00
										 |  |  |     elif hasattr(spec.loader, 'exec_module'): | 
					
						
							| 
									
										
										
										
											2016-09-08 11:12:31 -07:00
										 |  |  |         raise ImportError('loaders that define exec_module() ' | 
					
						
							|  |  |  |                           'must also define create_module()') | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     if module is None: | 
					
						
							|  |  |  |         module = _new_module(spec.name) | 
					
						
							|  |  |  |     _init_module_attrs(spec, module) | 
					
						
							|  |  |  |     return module | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | def _module_repr_from_spec(spec): | 
					
						
							|  |  |  |     """Return the repr to use for the module.""" | 
					
						
							|  |  |  |     # We mostly replicate _module_repr() using the spec attributes. | 
					
						
							|  |  |  |     name = '?' if spec.name is None else spec.name | 
					
						
							|  |  |  |     if spec.origin is None: | 
					
						
							|  |  |  |         if spec.loader is None: | 
					
						
							|  |  |  |             return '<module {!r}>'.format(name) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         else: | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |             return '<module {!r} ({!r})>'.format(name, spec.loader) | 
					
						
							|  |  |  |     else: | 
					
						
							|  |  |  |         if spec.has_location: | 
					
						
							|  |  |  |             return '<module {!r} from {!r}>'.format(name, spec.origin) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return '<module {!r} ({})>'.format(spec.name, spec.origin) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | # Used by importlib.reload() and _load_module_shim(). | 
					
						
							|  |  |  | def _exec(spec, module): | 
					
						
							| 
									
										
										
										
											2016-07-08 11:09:35 -07:00
										 |  |  |     """Execute the spec's specified module in an existing module's namespace.""" | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     name = spec.name | 
					
						
							|  |  |  |     with _ModuleLockManager(name): | 
					
						
							|  |  |  |         if sys.modules.get(name) is not module: | 
					
						
							|  |  |  |             msg = 'module {!r} not in sys.modules'.format(name) | 
					
						
							|  |  |  |             raise ImportError(msg, name=name) | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |         try: | 
					
						
							|  |  |  |             if spec.loader is None: | 
					
						
							|  |  |  |                 if spec.submodule_search_locations is None: | 
					
						
							|  |  |  |                     raise ImportError('missing loader', name=spec.name) | 
					
						
							|  |  |  |                 # Namespace package. | 
					
						
							|  |  |  |                 _init_module_attrs(spec, module, override=True) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 _init_module_attrs(spec, module, override=True) | 
					
						
							|  |  |  |                 if not hasattr(spec.loader, 'exec_module'): | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  |                     msg = (f"{_object_name(spec.loader)}.exec_module() not found; " | 
					
						
							|  |  |  |                            "falling back to load_module()") | 
					
						
							|  |  |  |                     _warnings.warn(msg, ImportWarning) | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |                     spec.loader.load_module(name) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     spec.loader.exec_module(module) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             # Update the order of insertion into sys.modules for module | 
					
						
							|  |  |  |             # clean-up at shutdown. | 
					
						
							|  |  |  |             module = sys.modules.pop(spec.name) | 
					
						
							|  |  |  |             sys.modules[spec.name] = module | 
					
						
							|  |  |  |     return module | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def _load_backward_compatible(spec): | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  |     # It is assumed that all callers have been warned about using load_module() | 
					
						
							|  |  |  |     # appropriately before calling this function. | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |     try: | 
					
						
							|  |  |  |         spec.loader.load_module(spec.name) | 
					
						
							|  |  |  |     except: | 
					
						
							|  |  |  |         if spec.name in sys.modules: | 
					
						
							|  |  |  |             module = sys.modules.pop(spec.name) | 
					
						
							|  |  |  |             sys.modules[spec.name] = module | 
					
						
							|  |  |  |         raise | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     # The module must be in sys.modules at this point! | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |     # Move it to the end of sys.modules. | 
					
						
							|  |  |  |     module = sys.modules.pop(spec.name) | 
					
						
							|  |  |  |     sys.modules[spec.name] = module | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     if getattr(module, '__loader__', None) is None: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             module.__loader__ = spec.loader | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     if getattr(module, '__package__', None) is None: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             # Since module.__path__ may not line up with | 
					
						
							|  |  |  |             # spec.submodule_search_paths, we can't necessarily rely | 
					
						
							|  |  |  |             # on spec.parent here. | 
					
						
							|  |  |  |             module.__package__ = module.__name__ | 
					
						
							|  |  |  |             if not hasattr(module, '__path__'): | 
					
						
							|  |  |  |                 module.__package__ = spec.name.rpartition('.')[0] | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     if getattr(module, '__spec__', None) is None: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             module.__spec__ = spec | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             pass | 
					
						
							|  |  |  |     return module | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | def _load_unlocked(spec): | 
					
						
							|  |  |  |     # A helper for direct use by the import system. | 
					
						
							|  |  |  |     if spec.loader is not None: | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |         # Not a namespace package. | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |         if not hasattr(spec.loader, 'exec_module'): | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  |             msg = (f"{_object_name(spec.loader)}.exec_module() not found; " | 
					
						
							|  |  |  |                     "falling back to load_module()") | 
					
						
							|  |  |  |             _warnings.warn(msg, ImportWarning) | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |             return _load_backward_compatible(spec) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     module = module_from_spec(spec) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-10-29 19:30:16 +02:00
										 |  |  |     # This must be done before putting the module in sys.modules | 
					
						
							|  |  |  |     # (otherwise an optimization shortcut in import.c becomes | 
					
						
							|  |  |  |     # wrong). | 
					
						
							|  |  |  |     spec._initializing = True | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         sys.modules[spec.name] = module | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             if spec.loader is None: | 
					
						
							|  |  |  |                 if spec.submodule_search_locations is None: | 
					
						
							|  |  |  |                     raise ImportError('missing loader', name=spec.name) | 
					
						
							|  |  |  |                 # A namespace package so do nothing. | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 spec.loader.exec_module(module) | 
					
						
							|  |  |  |         except: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 del sys.modules[spec.name] | 
					
						
							|  |  |  |             except KeyError: | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  |             raise | 
					
						
							|  |  |  |         # Move the module to the end of sys.modules. | 
					
						
							|  |  |  |         # We don't ensure that the import-related module attributes get | 
					
						
							|  |  |  |         # set in the sys.modules replacement case.  Such modules are on | 
					
						
							|  |  |  |         # their own. | 
					
						
							|  |  |  |         module = sys.modules.pop(spec.name) | 
					
						
							|  |  |  |         sys.modules[spec.name] = module | 
					
						
							|  |  |  |         _verbose_message('import {!r} # {!r}', spec.name, spec.loader) | 
					
						
							|  |  |  |     finally: | 
					
						
							|  |  |  |         spec._initializing = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return module | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  | # A method used during testing of _load_unlocked() and by | 
					
						
							|  |  |  | # _load_module_shim(). | 
					
						
							|  |  |  | def _load(spec): | 
					
						
							|  |  |  |     """Return a new module object, loaded by the spec's loader.
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     The module is not added to its parent. | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     If a module is already in sys.modules, that existing module gets | 
					
						
							|  |  |  |     clobbered. | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     """
 | 
					
						
							|  |  |  |     with _ModuleLockManager(spec.name): | 
					
						
							|  |  |  |         return _load_unlocked(spec) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 22:37:06 +00:00
										 |  |  | # Loaders ##################################################################### | 
					
						
							| 
									
										
										
										
											2009-03-12 22:28:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  | class BuiltinImporter: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 00:53:05 +00:00
										 |  |  |     """Meta path import for built-in modules.
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 00:53:05 +00:00
										 |  |  |     All methods are either class or static methods to avoid the need to | 
					
						
							|  |  |  |     instantiate the class. | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-09-12 01:00:02 +09:00
										 |  |  |     _ORIGIN = "built-in" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def module_repr(module): | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         """Return repr for the module.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The method is deprecated.  The import machinery does the job itself. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-03-26 11:55:07 -07:00
										 |  |  |         _warnings.warn("BuiltinImporter.module_repr() is deprecated and " | 
					
						
							|  |  |  |                        "slated for removal in Python 3.12", DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2019-09-12 01:00:02 +09:00
										 |  |  |         return f'<module {module.__name__!r} ({BuiltinImporter._ORIGIN})>' | 
					
						
							| 
									
										
										
										
											2012-05-24 20:21:04 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def find_spec(cls, fullname, path=None, target=None): | 
					
						
							|  |  |  |         if path is not None: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  |         if _imp.is_builtin(fullname): | 
					
						
							| 
									
										
										
										
											2019-09-12 01:00:02 +09:00
										 |  |  |             return spec_from_loader(fullname, cls, origin=cls._ORIGIN) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         else: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def find_module(cls, fullname, path=None): | 
					
						
							| 
									
										
										
										
											2009-03-15 00:53:05 +00:00
										 |  |  |         """Find the built-in module.
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         If 'path' is ever specified then the search is considered a failure. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         This method is deprecated.  Use find_spec() instead. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-06 08:56:57 -07:00
										 |  |  |         _warnings.warn("BuiltinImporter.find_module() is deprecated and " | 
					
						
							|  |  |  |                        "slated for removal in Python 3.12; use find_spec() instead", | 
					
						
							|  |  |  |                        DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         spec = cls.find_spec(fullname, path) | 
					
						
							|  |  |  |         return spec.loader if spec is not None else None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 14:44:02 +01:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def create_module(spec): | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |         """Create a built-in module""" | 
					
						
							|  |  |  |         if spec.name not in sys.builtin_module_names: | 
					
						
							|  |  |  |             raise ImportError('{!r} is not a built-in module'.format(spec.name), | 
					
						
							|  |  |  |                               name=spec.name) | 
					
						
							|  |  |  |         return _call_with_frames_removed(_imp.create_builtin, spec) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 14:44:02 +01:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def exec_module(module): | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |         """Exec a built-in module""" | 
					
						
							| 
									
										
										
										
											2015-08-24 19:53:56 -07:00
										 |  |  |         _call_with_frames_removed(_imp.exec_builtin, module) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_builtin | 
					
						
							|  |  |  |     def get_code(cls, fullname): | 
					
						
							|  |  |  |         """Return None as built-in modules do not have code objects.""" | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_builtin | 
					
						
							|  |  |  |     def get_source(cls, fullname): | 
					
						
							|  |  |  |         """Return None as built-in modules do not have source code.""" | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_builtin | 
					
						
							|  |  |  |     def is_package(cls, fullname): | 
					
						
							| 
									
										
										
										
											2012-07-07 13:16:44 +02:00
										 |  |  |         """Return False as built-in modules are never packages.""" | 
					
						
							| 
									
										
										
										
											2009-03-15 01:41:33 +00:00
										 |  |  |         return False | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-23 22:24:10 +10:00
										 |  |  |     load_module = classmethod(_load_module_shim) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  | class FrozenImporter: | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 00:53:05 +00:00
										 |  |  |     """Meta path import for frozen modules.
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 00:53:05 +00:00
										 |  |  |     All methods are either class or static methods to avoid the need to | 
					
						
							|  |  |  |     instantiate the class. | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-04 16:56:26 -08:00
										 |  |  |     _ORIGIN = "frozen" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def module_repr(m): | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         """Return repr for the module.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         The method is deprecated.  The import machinery does the job itself. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-03-26 11:55:07 -07:00
										 |  |  |         _warnings.warn("FrozenImporter.module_repr() is deprecated and " | 
					
						
							|  |  |  |                        "slated for removal in Python 3.12", DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2019-02-04 16:56:26 -08:00
										 |  |  |         return '<module {!r} ({})>'.format(m.__name__, FrozenImporter._ORIGIN) | 
					
						
							| 
									
										
										
										
											2012-05-24 20:21:04 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def _setup_module(cls, module): | 
					
						
							|  |  |  |         assert not hasattr(module, '__file__'), module.__file__ | 
					
						
							|  |  |  |         ispkg = hasattr(module, '__path__') | 
					
						
							|  |  |  |         assert not ispkg or not module.__path__, module.__path__ | 
					
						
							|  |  |  |         spec = module.__spec__ | 
					
						
							|  |  |  |         assert not ispkg or not spec.submodule_search_locations | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if spec.loader_state is None: | 
					
						
							|  |  |  |             spec.loader_state = type(sys.implementation)( | 
					
						
							|  |  |  |                 data=None, | 
					
						
							|  |  |  |                 origname=None, | 
					
						
							|  |  |  |             ) | 
					
						
							|  |  |  |         elif not hasattr(spec.loader_state, 'data'): | 
					
						
							|  |  |  |             spec.loader_state.data = None | 
					
						
							|  |  |  |         if not getattr(spec.loader_state, 'origname', None): | 
					
						
							|  |  |  |             origname = vars(module).pop('__origname__', None) | 
					
						
							|  |  |  |             assert origname, 'see PyImport_ImportFrozenModuleObject()' | 
					
						
							|  |  |  |             spec.loader_state.origname = origname | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def find_spec(cls, fullname, path=None, target=None): | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         info = _call_with_frames_removed(_imp.find_frozen, fullname) | 
					
						
							|  |  |  |         if info is None: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             return None | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |         data, ispkg, origname = info | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         spec = spec_from_loader(fullname, cls, | 
					
						
							|  |  |  |                                 origin=cls._ORIGIN, | 
					
						
							|  |  |  |                                 is_package=ispkg) | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |         spec.loader_state = type(sys.implementation)( | 
					
						
							|  |  |  |             data=data, | 
					
						
							|  |  |  |             origname=origname, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         return spec | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def find_module(cls, fullname, path=None): | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         """Find a frozen module.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This method is deprecated.  Use find_spec() instead. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2021-04-06 08:56:57 -07:00
										 |  |  |         _warnings.warn("FrozenImporter.find_module() is deprecated and " | 
					
						
							|  |  |  |                        "slated for removal in Python 3.12; use find_spec() instead", | 
					
						
							|  |  |  |                        DeprecationWarning) | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         return cls if _imp.is_frozen(fullname) else None | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-20 14:44:02 +01:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def create_module(spec): | 
					
						
							| 
									
										
										
										
											2015-01-09 11:39:21 -05:00
										 |  |  |         """Use default semantics for module creation.""" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     @staticmethod | 
					
						
							|  |  |  |     def exec_module(module): | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         spec = module.__spec__ | 
					
						
							|  |  |  |         name = spec.name | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |             data = spec.loader_state.data | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         except AttributeError: | 
					
						
							|  |  |  |             if not _imp.is_frozen(name): | 
					
						
							|  |  |  |                 raise ImportError('{!r} is not a frozen module'.format(name), | 
					
						
							|  |  |  |                                   name=name) | 
					
						
							|  |  |  |             data = None | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # We clear the extra data we got from the finder, to save memory. | 
					
						
							|  |  |  |             # Note that if this method is called again (e.g. by | 
					
						
							|  |  |  |             # importlib.reload()) then _imp.get_frozen_object() will notice | 
					
						
							|  |  |  |             # no data was provided and will look it up. | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |             spec.loader_state.data = None | 
					
						
							| 
									
										
										
										
											2021-10-05 10:01:27 -06:00
										 |  |  |         code = _call_with_frames_removed(_imp.get_frozen_object, name, data) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         exec(code, module.__dict__) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-22 22:43:07 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     def load_module(cls, fullname): | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |         """Load a frozen module.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         This method is deprecated.  Use exec_module() instead. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2020-12-04 15:39:21 -08:00
										 |  |  |         # Warning about deprecation implemented in _load_module_shim(). | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |         return _load_module_shim(cls, fullname) | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_frozen | 
					
						
							|  |  |  |     def get_code(cls, fullname): | 
					
						
							|  |  |  |         """Return the code object for the frozen module.""" | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         return _imp.get_frozen_object(fullname) | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_frozen | 
					
						
							|  |  |  |     def get_source(cls, fullname): | 
					
						
							|  |  |  |         """Return None as frozen modules do not have source code.""" | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     @classmethod | 
					
						
							|  |  |  |     @_requires_frozen | 
					
						
							|  |  |  |     def is_package(cls, fullname): | 
					
						
							| 
									
										
										
										
											2012-08-10 16:21:35 -07:00
										 |  |  |         """Return True if the frozen module is a package.""" | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         return _imp.is_frozen_package(fullname) | 
					
						
							| 
									
										
										
										
											2009-03-15 02:20:16 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 22:37:06 +00:00
										 |  |  | # Import itself ############################################################### | 
					
						
							| 
									
										
										
										
											2009-02-02 19:19:36 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-12 22:37:06 +00:00
										 |  |  | class _ImportLockContext: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """Context manager for the import lock.""" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         """Acquire the import lock.""" | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         _imp.acquire_lock() | 
					
						
							| 
									
										
										
										
											2009-03-12 22:37:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, exc_type, exc_value, exc_traceback): | 
					
						
							|  |  |  |         """Release the import lock regardless of any raised exceptions.""" | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |         _imp.release_lock() | 
					
						
							| 
									
										
										
										
											2009-03-12 22:37:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | def _resolve_name(name, package, level): | 
					
						
							|  |  |  |     """Resolve a relative module name to an absolute one.""" | 
					
						
							| 
									
										
										
										
											2012-02-24 21:48:17 -08:00
										 |  |  |     bits = package.rsplit('.', level - 1) | 
					
						
							| 
									
										
										
										
											2012-02-24 11:20:54 -05:00
										 |  |  |     if len(bits) < level: | 
					
						
							| 
									
										
										
										
											2019-08-03 12:46:02 +07:00
										 |  |  |         raise ImportError('attempted relative import beyond top-level package') | 
					
						
							| 
									
										
										
										
											2012-02-24 11:20:54 -05:00
										 |  |  |     base = bits[0] | 
					
						
							| 
									
										
										
										
											2012-07-07 13:16:44 +02:00
										 |  |  |     return '{}.{}'.format(base, name) if name else base | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  | def _find_spec_legacy(finder, name, path): | 
					
						
							| 
									
										
										
										
											2021-03-30 08:43:03 -07:00
										 |  |  |     msg = (f"{_object_name(finder)}.find_spec() not found; " | 
					
						
							|  |  |  |                            "falling back to find_module()") | 
					
						
							|  |  |  |     _warnings.warn(msg, ImportWarning) | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |     loader = finder.find_module(name, path) | 
					
						
							|  |  |  |     if loader is None: | 
					
						
							|  |  |  |         return None | 
					
						
							|  |  |  |     return spec_from_loader(name, loader) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | def _find_spec(name, path, target=None): | 
					
						
							| 
									
										
										
										
											2016-07-08 11:09:35 -07:00
										 |  |  |     """Find a module's spec.""" | 
					
						
							| 
									
										
										
										
											2016-03-25 00:40:59 +01:00
										 |  |  |     meta_path = sys.meta_path | 
					
						
							|  |  |  |     if meta_path is None: | 
					
						
							|  |  |  |         # PyImport_Cleanup() is running or has been called. | 
					
						
							|  |  |  |         raise ImportError("sys.meta_path is None, Python is likely " | 
					
						
							|  |  |  |                           "shutting down") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if not meta_path: | 
					
						
							| 
									
										
										
										
											2012-04-27 14:01:58 -04:00
										 |  |  |         _warnings.warn('sys.meta_path is empty', ImportWarning) | 
					
						
							| 
									
										
										
										
											2016-03-25 00:40:59 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     # We check sys.modules here for the reload case.  While a passed-in | 
					
						
							|  |  |  |     # target will usually indicate a reload there is no guarantee, whereas | 
					
						
							|  |  |  |     # sys.modules provides one. | 
					
						
							| 
									
										
										
										
											2013-10-31 22:22:15 -06:00
										 |  |  |     is_reload = name in sys.modules | 
					
						
							| 
									
										
										
										
											2016-03-25 00:40:59 +01:00
										 |  |  |     for finder in meta_path: | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  |         with _ImportLockContext(): | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 find_spec = finder.find_spec | 
					
						
							|  |  |  |             except AttributeError: | 
					
						
							| 
									
										
										
										
											2014-01-06 20:49:04 -07:00
										 |  |  |                 spec = _find_spec_legacy(finder, name, path) | 
					
						
							|  |  |  |                 if spec is None: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                     continue | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |             else: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                 spec = find_spec(name, path, target) | 
					
						
							|  |  |  |         if spec is not None: | 
					
						
							|  |  |  |             # The parent import may have already imported this module. | 
					
						
							|  |  |  |             if not is_reload and name in sys.modules: | 
					
						
							|  |  |  |                 module = sys.modules[name] | 
					
						
							| 
									
										
										
										
											2013-10-31 22:22:15 -06:00
										 |  |  |                 try: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                     __spec__ = module.__spec__ | 
					
						
							| 
									
										
										
										
											2013-10-31 22:22:15 -06:00
										 |  |  |                 except AttributeError: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |                     # We use the found spec since that is the one that | 
					
						
							|  |  |  |                     # we would have used if the parent module hadn't | 
					
						
							|  |  |  |                     # beaten us to the punch. | 
					
						
							|  |  |  |                     return spec | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     if __spec__ is None: | 
					
						
							|  |  |  |                         return spec | 
					
						
							|  |  |  |                     else: | 
					
						
							|  |  |  |                         return __spec__ | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return spec | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     else: | 
					
						
							|  |  |  |         return None | 
					
						
							| 
									
										
										
										
											2009-02-15 05:48:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-11-18 03:03:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | def _sanity_check(name, package, level): | 
					
						
							|  |  |  |     """Verify arguments are "sane".""" | 
					
						
							| 
									
										
										
										
											2012-02-23 18:18:48 -05:00
										 |  |  |     if not isinstance(name, str): | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |         raise TypeError('module name must be str, not {}'.format(type(name))) | 
					
						
							| 
									
										
										
										
											2012-02-22 18:33:05 -05:00
										 |  |  |     if level < 0: | 
					
						
							|  |  |  |         raise ValueError('level must be >= 0') | 
					
						
							| 
									
										
										
										
											2016-02-20 12:52:06 -08:00
										 |  |  |     if level > 0: | 
					
						
							| 
									
										
										
										
											2012-02-23 18:18:48 -05:00
										 |  |  |         if not isinstance(package, str): | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |             raise TypeError('__package__ not set to a string') | 
					
						
							| 
									
										
										
										
											2016-02-20 12:59:36 -08:00
										 |  |  |         elif not package: | 
					
						
							|  |  |  |             raise ImportError('attempted relative import with no known parent ' | 
					
						
							|  |  |  |                               'package') | 
					
						
							| 
									
										
										
										
											2009-02-07 01:15:27 +00:00
										 |  |  |     if not name and level == 0: | 
					
						
							| 
									
										
										
										
											2013-07-12 11:04:23 -04:00
										 |  |  |         raise ValueError('Empty module name') | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-07-06 14:48:18 -04:00
										 |  |  | _ERR_MSG_PREFIX = 'No module named ' | 
					
						
							|  |  |  | _ERR_MSG = _ERR_MSG_PREFIX + '{!r}' | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | def _find_and_load_unlocked(name, import_): | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     path = None | 
					
						
							|  |  |  |     parent = name.rpartition('.')[0] | 
					
						
							| 
									
										
										
										
											2021-07-24 23:44:46 +01:00
										 |  |  |     parent_spec = None | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     if parent: | 
					
						
							|  |  |  |         if parent not in sys.modules: | 
					
						
							| 
									
										
										
										
											2012-07-31 21:14:18 +10:00
										 |  |  |             _call_with_frames_removed(import_, parent) | 
					
						
							| 
									
										
										
										
											2012-04-02 20:33:56 -04:00
										 |  |  |         # Crazy side-effects! | 
					
						
							|  |  |  |         if name in sys.modules: | 
					
						
							|  |  |  |             return sys.modules[name] | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |         parent_module = sys.modules[parent] | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             path = parent_module.__path__ | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							| 
									
										
										
										
											2013-11-22 14:52:36 -05:00
										 |  |  |             msg = (_ERR_MSG + '; {!r} is not a package').format(name, parent) | 
					
						
							| 
									
										
										
										
											2016-09-07 16:56:15 -07:00
										 |  |  |             raise ModuleNotFoundError(msg, name=name) from None | 
					
						
							| 
									
										
										
										
											2021-07-24 23:44:46 +01:00
										 |  |  |         parent_spec = parent_module.__spec__ | 
					
						
							|  |  |  |         child = name.rpartition('.')[2] | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     spec = _find_spec(name, path) | 
					
						
							|  |  |  |     if spec is None: | 
					
						
							| 
									
										
										
										
											2016-09-07 16:56:15 -07:00
										 |  |  |         raise ModuleNotFoundError(_ERR_MSG.format(name), name=name) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2021-07-24 23:44:46 +01:00
										 |  |  |         if parent_spec: | 
					
						
							|  |  |  |             # Temporarily add child we are currently importing to parent's | 
					
						
							|  |  |  |             # _uninitialized_submodules for circular import tracking. | 
					
						
							|  |  |  |             parent_spec._uninitialized_submodules.append(child) | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             module = _load_unlocked(spec) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             if parent_spec: | 
					
						
							|  |  |  |                 parent_spec._uninitialized_submodules.pop() | 
					
						
							| 
									
										
										
										
											2012-02-22 18:33:05 -05:00
										 |  |  |     if parent: | 
					
						
							|  |  |  |         # Set the module as an attribute on its parent. | 
					
						
							|  |  |  |         parent_module = sys.modules[parent] | 
					
						
							| 
									
										
										
										
											2020-01-22 16:42:38 -08:00
										 |  |  |         try: | 
					
						
							|  |  |  |             setattr(parent_module, child, module) | 
					
						
							|  |  |  |         except AttributeError: | 
					
						
							|  |  |  |             msg = f"Cannot set an attribute on {parent!r} for child module {child!r}" | 
					
						
							|  |  |  |             _warnings.warn(msg, ImportWarning) | 
					
						
							| 
									
										
										
										
											2012-02-22 18:33:05 -05:00
										 |  |  |     return module | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-21 13:00:46 +02:00
										 |  |  | _NEEDS_LOADING = object() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | def _find_and_load(name, import_): | 
					
						
							| 
									
										
										
										
											2017-07-06 08:09:03 +03:00
										 |  |  |     """Find and load the module.""" | 
					
						
							| 
									
										
										
										
											2021-08-12 11:23:29 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # Optimization: we avoid unneeded module locking if the module | 
					
						
							|  |  |  |     # already exists in sys.modules and is fully initialized. | 
					
						
							|  |  |  |     module = sys.modules.get(name, _NEEDS_LOADING) | 
					
						
							|  |  |  |     if (module is _NEEDS_LOADING or | 
					
						
							|  |  |  |         getattr(getattr(module, "__spec__", None), "_initializing", False)): | 
					
						
							|  |  |  |         with _ModuleLockManager(name): | 
					
						
							|  |  |  |             module = sys.modules.get(name, _NEEDS_LOADING) | 
					
						
							|  |  |  |             if module is _NEEDS_LOADING: | 
					
						
							|  |  |  |                 return _find_and_load_unlocked(name, import_) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Optimization: only call _bootstrap._lock_unlock_module() if | 
					
						
							|  |  |  |         # module.__spec__._initializing is True. | 
					
						
							|  |  |  |         # NOTE: because of this, initializing must be set *before* | 
					
						
							|  |  |  |         # putting the new module in sys.modules. | 
					
						
							|  |  |  |         _lock_unlock_module(name) | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 08:09:03 +03:00
										 |  |  |     if module is None: | 
					
						
							|  |  |  |         message = ('import of {} halted; ' | 
					
						
							|  |  |  |                    'None in sys.modules'.format(name)) | 
					
						
							|  |  |  |         raise ModuleNotFoundError(message, name=name) | 
					
						
							| 
									
										
										
										
											2017-07-10 22:52:32 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 08:09:03 +03:00
										 |  |  |     return module | 
					
						
							| 
									
										
										
										
											2012-05-17 18:55:59 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | def _gcd_import(name, package=None, level=0): | 
					
						
							|  |  |  |     """Import and return the module based on its name, the package the call is
 | 
					
						
							|  |  |  |     being made from, and the level adjustment. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     This function represents the greatest common denominator of functionality | 
					
						
							|  |  |  |     between import_module and __import__. This includes setting __package__ if | 
					
						
							|  |  |  |     the loader did not. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     _sanity_check(name, package, level) | 
					
						
							| 
									
										
										
										
											2009-02-06 02:47:33 +00:00
										 |  |  |     if level > 0: | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |         name = _resolve_name(name, package, level) | 
					
						
							| 
									
										
										
										
											2017-07-06 08:09:03 +03:00
										 |  |  |     return _find_and_load(name, _gcd_import) | 
					
						
							| 
									
										
										
										
											2009-02-06 02:47:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-07 16:56:15 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-26 10:41:59 +03:00
										 |  |  | def _handle_fromlist(module, fromlist, import_, *, recursive=False): | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     """Figure out what __import__ should return.
 | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     The import_ parameter is a callable which takes the name of module to | 
					
						
							|  |  |  |     import. It is required to decouple the function from assuming importlib's | 
					
						
							|  |  |  |     import implementation is desired. | 
					
						
							| 
									
										
										
										
											2009-01-18 00:24:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2009-02-07 01:15:27 +00:00
										 |  |  |     # The hell that is fromlist ... | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |     # If a package was imported, try to import stuff from fromlist. | 
					
						
							| 
									
										
										
										
											2018-03-11 10:52:37 +02:00
										 |  |  |     for x in fromlist: | 
					
						
							|  |  |  |         if not isinstance(x, str): | 
					
						
							|  |  |  |             if recursive: | 
					
						
							|  |  |  |                 where = module.__name__ + '.__all__' | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 where = "``from list''" | 
					
						
							|  |  |  |             raise TypeError(f"Item in {where} must be str, " | 
					
						
							|  |  |  |                             f"not {type(x).__name__}") | 
					
						
							|  |  |  |         elif x == '*': | 
					
						
							|  |  |  |             if not recursive and hasattr(module, '__all__'): | 
					
						
							|  |  |  |                 _handle_fromlist(module, module.__all__, import_, | 
					
						
							|  |  |  |                                  recursive=True) | 
					
						
							|  |  |  |         elif not hasattr(module, x): | 
					
						
							|  |  |  |             from_name = '{}.{}'.format(module.__name__, x) | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 _call_with_frames_removed(import_, from_name) | 
					
						
							|  |  |  |             except ModuleNotFoundError as exc: | 
					
						
							|  |  |  |                 # Backwards-compatibility dictates we ignore failed | 
					
						
							|  |  |  |                 # imports triggered by fromlist for modules that don't | 
					
						
							|  |  |  |                 # exist. | 
					
						
							|  |  |  |                 if (exc.name == from_name and | 
					
						
							|  |  |  |                     sys.modules.get(from_name, _NEEDS_LOADING) is not None): | 
					
						
							|  |  |  |                     continue | 
					
						
							|  |  |  |                 raise | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |     return module | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | def _calc___package__(globals): | 
					
						
							|  |  |  |     """Calculate what __package__ should be.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     __package__ is not guaranteed to be defined or could be set to None | 
					
						
							|  |  |  |     to represent that its proper value is unknown. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2016-01-22 15:25:50 -08:00
										 |  |  |     package = globals.get('__package__') | 
					
						
							| 
									
										
										
										
											2016-01-15 13:33:03 -08:00
										 |  |  |     spec = globals.get('__spec__') | 
					
						
							| 
									
										
										
										
											2016-01-22 15:25:50 -08:00
										 |  |  |     if package is not None: | 
					
						
							|  |  |  |         if spec is not None and package != spec.parent: | 
					
						
							|  |  |  |             _warnings.warn("__package__ != __spec__.parent " | 
					
						
							|  |  |  |                            f"({package!r} != {spec.parent!r})", | 
					
						
							|  |  |  |                            ImportWarning, stacklevel=3) | 
					
						
							|  |  |  |         return package | 
					
						
							|  |  |  |     elif spec is not None: | 
					
						
							| 
									
										
										
										
											2016-01-15 13:33:03 -08:00
										 |  |  |         return spec.parent | 
					
						
							| 
									
										
										
										
											2016-01-22 15:25:50 -08:00
										 |  |  |     else: | 
					
						
							| 
									
										
										
										
											2016-01-15 13:33:03 -08:00
										 |  |  |         _warnings.warn("can't resolve package from __spec__ or __package__, " | 
					
						
							|  |  |  |                        "falling back on __name__ and __path__", | 
					
						
							|  |  |  |                        ImportWarning, stacklevel=3) | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |         package = globals['__name__'] | 
					
						
							|  |  |  |         if '__path__' not in globals: | 
					
						
							|  |  |  |             package = package.rpartition('.')[0] | 
					
						
							|  |  |  |     return package | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-06 16:34:44 -04:00
										 |  |  | def __import__(name, globals=None, locals=None, fromlist=(), level=0): | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     """Import a module.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-26 05:35:26 +00:00
										 |  |  |     The 'globals' argument is used to infer where the import is occurring from | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |     to handle relative imports. The 'locals' argument is ignored. The | 
					
						
							|  |  |  |     'fromlist' argument specifies what should exist as attributes on the module | 
					
						
							|  |  |  |     being imported (e.g. ``from module import <fromlist>``).  The 'level' | 
					
						
							|  |  |  |     argument represents the package location to import from in a relative | 
					
						
							|  |  |  |     import (e.g. ``from ..pkg import mod`` would have a 'level' of 2). | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							|  |  |  |     if level == 0: | 
					
						
							|  |  |  |         module = _gcd_import(name) | 
					
						
							|  |  |  |     else: | 
					
						
							| 
									
										
										
										
											2012-08-06 16:34:44 -04:00
										 |  |  |         globals_ = globals if globals is not None else {} | 
					
						
							|  |  |  |         package = _calc___package__(globals_) | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  |         module = _gcd_import(name, package, level) | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |     if not fromlist: | 
					
						
							|  |  |  |         # Return up to the first dot in 'name'. This is complicated by the fact | 
					
						
							|  |  |  |         # that 'name' may be relative. | 
					
						
							|  |  |  |         if level == 0: | 
					
						
							| 
									
										
										
										
											2012-05-07 21:41:59 +02:00
										 |  |  |             return _gcd_import(name.partition('.')[0]) | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |         elif not name: | 
					
						
							|  |  |  |             return module | 
					
						
							|  |  |  |         else: | 
					
						
							| 
									
										
										
										
											2012-09-28 16:41:39 -04:00
										 |  |  |             # Figure out where to slice the module's name up to the first dot | 
					
						
							|  |  |  |             # in 'name'. | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |             cut_off = len(name) - len(name.partition('.')[0]) | 
					
						
							| 
									
										
										
										
											2012-09-28 16:41:39 -04:00
										 |  |  |             # Slice end needs to be positive to alleviate need to special-case | 
					
						
							|  |  |  |             # when ``'.' not in name``. | 
					
						
							| 
									
										
										
										
											2012-04-14 21:50:00 -04:00
										 |  |  |             return sys.modules[module.__name__[:len(module.__name__)-cut_off]] | 
					
						
							| 
									
										
										
										
											2018-03-11 10:52:37 +02:00
										 |  |  |     elif hasattr(module, '__path__'): | 
					
						
							| 
									
										
										
										
											2012-02-23 20:47:57 -05:00
										 |  |  |         return _handle_fromlist(module, fromlist, _gcd_import) | 
					
						
							| 
									
										
										
										
											2018-03-11 10:52:37 +02:00
										 |  |  |     else: | 
					
						
							|  |  |  |         return module | 
					
						
							| 
									
										
										
										
											2012-02-16 13:43:41 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | def _builtin_from_name(name): | 
					
						
							|  |  |  |     spec = BuiltinImporter.find_spec(name) | 
					
						
							| 
									
										
										
										
											2013-11-22 11:58:17 -05:00
										 |  |  |     if spec is None: | 
					
						
							|  |  |  |         raise ImportError('no built-in module named ' + name) | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |     return _load_unlocked(spec) | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-20 18:04:03 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  | def _setup(sys_module, _imp_module): | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  |     """Setup importlib by importing needed built-in modules and injecting them
 | 
					
						
							|  |  |  |     into the global namespace. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |     As sys is needed for sys.modules access and _imp is needed to load built-in | 
					
						
							| 
									
										
										
										
											2012-02-14 18:38:11 -05:00
										 |  |  |     modules, those two modules must be explicitly passed in. | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |     """
 | 
					
						
							| 
									
										
										
										
											2015-04-13 14:21:02 -04:00
										 |  |  |     global _imp, sys | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |     _imp = _imp_module | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  |     sys = sys_module | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     # Set up the spec for existing builtin/frozen modules. | 
					
						
							| 
									
										
										
										
											2013-02-01 14:04:12 -05:00
										 |  |  |     module_type = type(sys) | 
					
						
							| 
									
										
										
										
											2013-02-01 15:31:49 -05:00
										 |  |  |     for name, module in sys.modules.items(): | 
					
						
							| 
									
										
										
										
											2013-02-01 14:04:12 -05:00
										 |  |  |         if isinstance(module, module_type): | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             if name in sys.builtin_module_names: | 
					
						
							|  |  |  |                 loader = BuiltinImporter | 
					
						
							|  |  |  |             elif _imp.is_frozen(name): | 
					
						
							|  |  |  |                 loader = FrozenImporter | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             spec = _spec_from_module(module, loader) | 
					
						
							| 
									
										
										
										
											2014-05-30 14:55:29 -04:00
										 |  |  |             _init_module_attrs(spec, module) | 
					
						
							| 
									
										
										
										
											2021-10-05 11:26:37 -06:00
										 |  |  |             if loader is FrozenImporter: | 
					
						
							|  |  |  |                 loader._setup_module(module) | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |     # Directly load built-in modules needed during bootstrap. | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  |     self_module = sys.modules[__name__] | 
					
						
							| 
									
										
										
										
											2017-09-18 23:50:44 +02:00
										 |  |  |     for builtin_name in ('_thread', '_warnings', '_weakref'): | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  |         if builtin_name not in sys.modules: | 
					
						
							| 
									
										
										
										
											2013-11-22 09:05:39 -07:00
										 |  |  |             builtin_module = _builtin_from_name(builtin_name) | 
					
						
							| 
									
										
										
										
											2012-02-08 18:50:22 -05:00
										 |  |  |         else: | 
					
						
							|  |  |  |             builtin_module = sys.modules[builtin_name] | 
					
						
							|  |  |  |         setattr(self_module, builtin_name, builtin_module) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-02-14 18:38:11 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  | def _install(sys_module, _imp_module): | 
					
						
							| 
									
										
										
										
											2017-05-22 21:36:03 -07:00
										 |  |  |     """Install importers for builtin and frozen modules""" | 
					
						
							| 
									
										
										
										
											2012-04-15 16:08:47 -04:00
										 |  |  |     _setup(sys_module, _imp_module) | 
					
						
							| 
									
										
										
										
											2015-05-02 19:15:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-28 21:33:05 +02:00
										 |  |  |     sys.meta_path.append(BuiltinImporter) | 
					
						
							|  |  |  |     sys.meta_path.append(FrozenImporter) | 
					
						
							| 
									
										
										
										
											2015-05-02 19:15:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-22 21:36:03 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | def _install_external_importers(): | 
					
						
							|  |  |  |     """Install importers that require external filesystem access""" | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  |     global _bootstrap_external | 
					
						
							| 
									
										
										
										
											2015-05-02 19:15:18 -06:00
										 |  |  |     import _frozen_importlib_external | 
					
						
							| 
									
										
										
										
											2015-05-15 21:54:59 -06:00
										 |  |  |     _bootstrap_external = _frozen_importlib_external | 
					
						
							| 
									
										
										
										
											2015-05-02 19:15:18 -06:00
										 |  |  |     _frozen_importlib_external._install(sys.modules[__name__]) |