brew/Library/Homebrew/lock_file.rb
2024-11-24 01:29:42 +01:00

109 lines
2.8 KiB
Ruby

# typed: true # rubocop:todo Sorbet/StrictSigil
# frozen_string_literal: true
require "fcntl"
# A lock file to prevent multiple Homebrew processes from modifying the same path.
class LockFile
class OpenFileChangedOnDisk < RuntimeError; end
private_constant :OpenFileChangedOnDisk
attr_reader :path
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"
@lockfile = nil
end
sig { void }
def lock
ignore_interrupts do
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
lockfile.close
raise OperationInProgressError, @locked_path
end
end
sig { params(unlink: T::Boolean).void }
def unlock(unlink: false)
ignore_interrupts do
next if @lockfile.nil?
@path.unlink if unlink
@lockfile.flock(File::LOCK_UN)
@lockfile.close
@lockfile = nil
end
end
def with_lock
lock
yield
ensure
unlock
end
end
# A lock file for a formula.
class FormulaLock < LockFile
sig { params(rack_name: String).void }
def initialize(rack_name)
super(:formula, HOMEBREW_CELLAR/rack_name)
end
end
# A lock file for a cask.
class CaskLock < LockFile
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)
end
end