mirror of
https://github.com/Homebrew/brew.git
synced 2025-07-14 16:09:03 +08:00
125 lines
4.0 KiB
Ruby
125 lines
4.0 KiB
Ruby
# typed: strict
|
|
# frozen_string_literal: true
|
|
|
|
module Language
|
|
# Helper functions for Node formulae.
|
|
#
|
|
# @api public
|
|
module Node
|
|
sig { returns(String) }
|
|
def self.npm_cache_config
|
|
"cache=#{HOMEBREW_CACHE}/npm_cache"
|
|
end
|
|
|
|
sig { returns(String) }
|
|
def self.pack_for_installation
|
|
# Homebrew assumes the buildpath/testpath will always be disposable
|
|
# and from npm 5.0.0 the logic changed so that when a directory is
|
|
# fed to `npm install` only symlinks are created linking back to that
|
|
# directory, consequently breaking that assumption. We require a tarball
|
|
# because npm install creates a "real" installation when fed a tarball.
|
|
package = Pathname("package.json")
|
|
if package.exist?
|
|
begin
|
|
pkg_json = JSON.parse(package.read)
|
|
rescue JSON::ParserError
|
|
opoo "Could not parse package.json!"
|
|
raise
|
|
end
|
|
prepare_removed = pkg_json["scripts"]&.delete("prepare")
|
|
prepack_removed = pkg_json["scripts"]&.delete("prepack")
|
|
postpack_removed = pkg_json["scripts"]&.delete("postpack")
|
|
package.atomic_write(JSON.pretty_generate(pkg_json)) if prepare_removed || prepack_removed || postpack_removed
|
|
end
|
|
output = Utils.popen_read("npm", "pack", "--ignore-scripts")
|
|
raise "npm failed to pack #{Dir.pwd}" if !$CHILD_STATUS.exitstatus.zero? || output.lines.empty?
|
|
|
|
output.lines.last.chomp
|
|
end
|
|
|
|
sig { void }
|
|
def self.setup_npm_environment
|
|
# guard that this is only run once
|
|
return if @env_set
|
|
|
|
@env_set = T.let(true, T.nilable(T::Boolean))
|
|
# explicitly use our npm and node-gyp executables instead of the user
|
|
# managed ones in HOMEBREW_PREFIX/lib/node_modules which might be broken
|
|
begin
|
|
ENV.prepend_path "PATH", Formula["node"].opt_libexec/"bin"
|
|
rescue FormulaUnavailableError
|
|
nil
|
|
end
|
|
end
|
|
|
|
sig { params(libexec: Pathname).returns(T::Array[String]) }
|
|
def self.std_npm_install_args(libexec)
|
|
setup_npm_environment
|
|
|
|
pack = pack_for_installation
|
|
|
|
# npm 7 requires that these dirs exist before install
|
|
(libexec/"lib").mkpath
|
|
|
|
# npm install args for global style module format installed into libexec
|
|
args = %W[
|
|
-ddd
|
|
--global
|
|
--build-from-source
|
|
--#{npm_cache_config}
|
|
--prefix=#{libexec}
|
|
#{Dir.pwd}/#{pack}
|
|
]
|
|
|
|
args << "--unsafe-perm" if Process.uid.zero?
|
|
|
|
args
|
|
end
|
|
|
|
sig { returns(T::Array[String]) }
|
|
def self.local_npm_install_args
|
|
setup_npm_environment
|
|
# npm install args for local style module format
|
|
%W[
|
|
-ddd
|
|
--build-from-source
|
|
--#{npm_cache_config}
|
|
]
|
|
end
|
|
|
|
# Mixin module for {Formula} adding shebang rewrite features.
|
|
module Shebang
|
|
extend T::Helpers
|
|
|
|
requires_ancestor { Formula }
|
|
|
|
module_function
|
|
|
|
# A regex to match potential shebang permutations.
|
|
NODE_SHEBANG_REGEX = %r{^#! ?(?:/usr/bin/(?:env )?)?node( |$)}
|
|
|
|
# The length of the longest shebang matching `SHEBANG_REGEX`.
|
|
NODE_SHEBANG_MAX_LENGTH = T.let("#! /usr/bin/env node ".length, Integer)
|
|
|
|
# @private
|
|
sig { params(node_path: T.any(String, Pathname)).returns(Utils::Shebang::RewriteInfo) }
|
|
def node_shebang_rewrite_info(node_path)
|
|
Utils::Shebang::RewriteInfo.new(
|
|
NODE_SHEBANG_REGEX,
|
|
NODE_SHEBANG_MAX_LENGTH,
|
|
"#{node_path}\\1",
|
|
)
|
|
end
|
|
|
|
sig { params(formula: Formula).returns(Utils::Shebang::RewriteInfo) }
|
|
def detected_node_shebang(formula = T.cast(self, Formula))
|
|
node_deps = formula.deps.select(&:required?).map(&:name).grep(/^node(@.+)?$/)
|
|
raise ShebangDetectionError.new("Node", "formula does not depend on Node") if node_deps.empty?
|
|
raise ShebangDetectionError.new("Node", "formula has multiple Node dependencies") if node_deps.length > 1
|
|
|
|
node_shebang_rewrite_info(Formula[node_deps.first].opt_bin/"node")
|
|
end
|
|
end
|
|
end
|
|
end
|