| 
									
										
										
										
											2024-08-12 10:30:59 +01:00
										 |  |  | # typed: true # rubocop:todo Sorbet/StrictSigil | 
					
						
							| 
									
										
										
										
											2019-04-19 15:38:03 +09:00
										 |  |  | # frozen_string_literal: true | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-03 13:09:07 +01:00
										 |  |  | require "fcntl" | 
					
						
							| 
									
										
										
										
											2013-07-27 23:53:53 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-30 17:51:02 +01:00
										 |  |  | # A lock file to prevent multiple Homebrew processes from modifying the same path. | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  | class LockFile | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |   class OpenFileChangedOnDisk < RuntimeError; end | 
					
						
							|  |  |  |   private_constant :OpenFileChangedOnDisk | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 16:46:39 +02:00
										 |  |  |   attr_reader :path | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-07-30 17:51:02 +01:00
										 |  |  |   sig { params(type: Symbol, locked_path: Pathname).void } | 
					
						
							|  |  |  |   def initialize(type, locked_path) | 
					
						
							|  |  |  |     @locked_path = locked_path | 
					
						
							|  |  |  |     lock_name = locked_path.basename.to_s | 
					
						
							|  |  |  |     @path = HOMEBREW_LOCKS/"#{lock_name}.#{type}.lock" | 
					
						
							| 
									
										
										
										
											2013-04-05 22:06:52 -05:00
										 |  |  |     @lockfile = nil | 
					
						
							| 
									
										
										
										
											2013-02-09 18:19:50 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |   sig { void } | 
					
						
							| 
									
										
										
										
											2013-02-09 18:19:50 -06:00
										 |  |  |   def lock | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |     ignore_interrupts do | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |       next if @lockfile.present? | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       path.dirname.mkpath | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       begin | 
					
						
							|  |  |  |         lockfile = begin | 
					
						
							|  |  |  |           path.open(File::RDWR | File::CREAT) | 
					
						
							|  |  |  |         rescue Errno::EMFILE | 
					
						
							|  |  |  |           odie "The maximum number of open files on this system has been reached. " \ | 
					
						
							|  |  |  |                "Use `ulimit -n` to increase this limit." | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |         lockfile.fcntl(Fcntl::F_SETFD, Fcntl::FD_CLOEXEC) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if lockfile.flock(File::LOCK_EX | File::LOCK_NB) | 
					
						
							|  |  |  |           # This prevents a race condition in case the file we locked doesn't exist on disk anymore, e.g.: | 
					
						
							|  |  |  |           # | 
					
						
							|  |  |  |           # 1. Process A creates and opens the file. | 
					
						
							|  |  |  |           # 2. Process A locks the file. | 
					
						
							|  |  |  |           # 3. Process B opens the file. | 
					
						
							|  |  |  |           # 4. Process A unlinks the file. | 
					
						
							|  |  |  |           # 5. Process A unlocks the file. | 
					
						
							|  |  |  |           # 6. Process B locks the file. | 
					
						
							|  |  |  |           # 7. Process C creates and opens the file. | 
					
						
							|  |  |  |           # 8. Process C locks the file. | 
					
						
							|  |  |  |           # 9. Process B and C hold locks to files with different inode numbers. 💥 | 
					
						
							|  |  |  |           if !path.exist? || lockfile.stat.ino != path.stat.ino | 
					
						
							|  |  |  |             lockfile.close | 
					
						
							|  |  |  |             raise OpenFileChangedOnDisk | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           @lockfile = lockfile | 
					
						
							|  |  |  |           next | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       rescue OpenFileChangedOnDisk | 
					
						
							|  |  |  |         retry | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |       lockfile.close | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |       raise OperationInProgressError, @locked_path | 
					
						
							|  |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-02-09 18:19:50 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |   sig { params(unlink: T::Boolean).void } | 
					
						
							|  |  |  |   def unlock(unlink: false) | 
					
						
							|  |  |  |     ignore_interrupts do | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |       next if @lockfile.nil? | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |       @path.unlink if unlink | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |       @lockfile.flock(File::LOCK_UN) | 
					
						
							|  |  |  |       @lockfile.close | 
					
						
							| 
									
										
										
										
											2024-11-24 01:29:42 +01:00
										 |  |  |       @lockfile = nil | 
					
						
							| 
									
										
										
										
											2024-11-23 23:45:17 +01:00
										 |  |  |     end | 
					
						
							| 
									
										
										
										
											2013-02-09 18:19:50 -06:00
										 |  |  |   end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   def with_lock | 
					
						
							|  |  |  |     lock | 
					
						
							|  |  |  |     yield | 
					
						
							|  |  |  |   ensure | 
					
						
							|  |  |  |     unlock | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-09 03:16:22 +02:00
										 |  |  | # A lock file for a formula. | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  | class FormulaLock < LockFile | 
					
						
							| 
									
										
										
										
											2024-07-30 17:51:02 +01:00
										 |  |  |   sig { params(rack_name: String).void } | 
					
						
							|  |  |  |   def initialize(rack_name) | 
					
						
							|  |  |  |     super(:formula, HOMEBREW_CELLAR/rack_name) | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-09 03:16:22 +02:00
										 |  |  | # A lock file for a cask. | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  | class CaskLock < LockFile | 
					
						
							| 
									
										
										
										
											2024-07-30 17:51:02 +01:00
										 |  |  |   sig { params(cask_token: String).void } | 
					
						
							|  |  |  |   def initialize(cask_token) | 
					
						
							|  |  |  |     super(:cask, HOMEBREW_PREFIX/"Caskroom/#{cask_token}") | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # A lock file for a download. | 
					
						
							|  |  |  | class DownloadLock < LockFile | 
					
						
							|  |  |  |   sig { params(download_path: Pathname).void } | 
					
						
							|  |  |  |   def initialize(download_path) | 
					
						
							|  |  |  |     super(:download, download_path) | 
					
						
							| 
									
										
										
										
											2017-05-22 03:23:50 +02:00
										 |  |  |   end | 
					
						
							|  |  |  | end |