| 
									
										
										
										
											2018-07-14 22:10:56 +03:00
										 |  |  | # Copyright (c) 2014-present PlatformIO <contact@platformio.org> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  | # you may not use this file except in compliance with the License. | 
					
						
							|  |  |  | # You may obtain a copy of the License at | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | #    http://www.apache.org/licenses/LICENSE-2.0 | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  | # distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  | # See the License for the specific language governing permissions and | 
					
						
							|  |  |  | # limitations under the License. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from os import remove | 
					
						
							|  |  |  | from os.path import abspath, exists, getmtime | 
					
						
							|  |  |  | from time import sleep, time | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | from platformio import exception | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LOCKFILE_TIMEOUT = 3600  # in seconds, 1 hour | 
					
						
							|  |  |  | LOCKFILE_DELAY = 0.2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | LOCKFILE_INTERFACE_FCNTL = 1 | 
					
						
							|  |  |  | LOCKFILE_INTERFACE_MSVCRT = 2 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | try: | 
					
						
							|  |  |  |     import fcntl | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 22:10:56 +03:00
										 |  |  |     LOCKFILE_CURRENT_INTERFACE = LOCKFILE_INTERFACE_FCNTL | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         import msvcrt | 
					
						
							| 
									
										
										
										
											2019-09-23 23:13:48 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-14 22:10:56 +03:00
										 |  |  |         LOCKFILE_CURRENT_INTERFACE = LOCKFILE_INTERFACE_MSVCRT | 
					
						
							|  |  |  |     except ImportError: | 
					
						
							|  |  |  |         LOCKFILE_CURRENT_INTERFACE = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class LockFileExists(Exception): | 
					
						
							|  |  |  |     pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class LockFile(object): | 
					
						
							|  |  |  |     def __init__(self, path, timeout=LOCKFILE_TIMEOUT, delay=LOCKFILE_DELAY): | 
					
						
							|  |  |  |         self.timeout = timeout | 
					
						
							|  |  |  |         self.delay = delay | 
					
						
							|  |  |  |         self._lock_path = abspath(path) + ".lock" | 
					
						
							|  |  |  |         self._fp = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _lock(self): | 
					
						
							|  |  |  |         if not LOCKFILE_CURRENT_INTERFACE and exists(self._lock_path): | 
					
						
							|  |  |  |             # remove stale lock | 
					
						
							|  |  |  |             if time() - getmtime(self._lock_path) > 10: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     remove(self._lock_path) | 
					
						
							|  |  |  |                 except:  # pylint: disable=bare-except | 
					
						
							|  |  |  |                     pass | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 raise LockFileExists | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._fp = open(self._lock_path, "w") | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             if LOCKFILE_CURRENT_INTERFACE == LOCKFILE_INTERFACE_FCNTL: | 
					
						
							|  |  |  |                 fcntl.flock(self._fp.fileno(), fcntl.LOCK_EX | fcntl.LOCK_NB) | 
					
						
							|  |  |  |             elif LOCKFILE_CURRENT_INTERFACE == LOCKFILE_INTERFACE_MSVCRT: | 
					
						
							|  |  |  |                 msvcrt.locking(self._fp.fileno(), msvcrt.LK_NBLCK, 1) | 
					
						
							|  |  |  |         except IOError: | 
					
						
							|  |  |  |             self._fp = None | 
					
						
							|  |  |  |             raise LockFileExists | 
					
						
							|  |  |  |         return True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _unlock(self): | 
					
						
							|  |  |  |         if not self._fp: | 
					
						
							|  |  |  |             return | 
					
						
							|  |  |  |         if LOCKFILE_CURRENT_INTERFACE == LOCKFILE_INTERFACE_FCNTL: | 
					
						
							|  |  |  |             fcntl.flock(self._fp.fileno(), fcntl.LOCK_UN) | 
					
						
							|  |  |  |         elif LOCKFILE_CURRENT_INTERFACE == LOCKFILE_INTERFACE_MSVCRT: | 
					
						
							|  |  |  |             msvcrt.locking(self._fp.fileno(), msvcrt.LK_UNLCK, 1) | 
					
						
							|  |  |  |         self._fp.close() | 
					
						
							|  |  |  |         self._fp = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def acquire(self): | 
					
						
							|  |  |  |         elapsed = 0 | 
					
						
							|  |  |  |         while elapsed < self.timeout: | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 return self._lock() | 
					
						
							|  |  |  |             except LockFileExists: | 
					
						
							|  |  |  |                 sleep(self.delay) | 
					
						
							|  |  |  |                 elapsed += self.delay | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         raise exception.LockFileTimeoutError() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def release(self): | 
					
						
							|  |  |  |         self._unlock() | 
					
						
							|  |  |  |         if exists(self._lock_path): | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 remove(self._lock_path) | 
					
						
							|  |  |  |             except:  # pylint: disable=bare-except | 
					
						
							|  |  |  |                 pass | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __enter__(self): | 
					
						
							|  |  |  |         self.acquire() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __exit__(self, type_, value, traceback): | 
					
						
							|  |  |  |         self.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __del__(self): | 
					
						
							|  |  |  |         self.release() |