Class: Bundler::Dsl
Relationships & Source Files | |
Namespace Children | |
Exceptions:
| |
Extension / Inclusion / Inheritance Descendants | |
Subclasses:
|
|
Super Chains via Extension / Inclusion / Inheritance | |
Instance Chain:
self,
RubyDsl
|
|
Inherits: | Object |
Defined in: | lib/bundler/dsl.rb |
Constant Summary
-
GITHUB_PULL_REQUEST_URL =
# File 'lib/bundler/dsl.rb', line 21%r{\Ahttps://github\.com/([A-Za-z0-9_\-\.]/[A-Za-z0-9_\-\.])/pull/(\d+)\z}
-
GITLAB_MERGE_REQUEST_URL =
# File 'lib/bundler/dsl.rb', line 22%r{\Ahttps://gitlab\.com/([A-Za-z0-9_\-\./])/-/merge_requests/(\d)\z}
-
VALID_KEYS =
# File 'lib/bundler/dsl.rb', line 18%w[group groups git path glob name branch ref tag require submodules platform platforms type source install_if gemfile force_ruby_platform].freeze
-
VALID_PLATFORMS =
# File 'lib/bundler/dsl.rb', line 16Bundler::Dependency::PLATFORM_MAP.keys.freeze
Class Method Summary
- .evaluate(gemfile, lockfile, unlock)
- .new ⇒ Dsl constructor
Instance Attribute Summary
- #dependencies rw
- #gemfile readonly
- #gemspecs readonly
Instance Method Summary
- #check_primary_source_safety
- #env(name)
- #eval_gemfile(gemfile, contents = nil)
- #gem(name, *args) (also: #_gem)
- #gemfile_root
- #gemspec(opts = nil)
- #git(uri, options = {}, &blk)
- #git_source(name, &block)
- #github(repo, options = {})
- #group(*args, &blk)
- #install_if(*args)
- #method_missing(name, *args)
- #path(path, options = {}, &blk)
-
#platform(*platforms)
Alias for #platforms.
- #platforms(*platforms) (also: #platform)
- #plugin(*args)
- #source(source, *args, &blk)
- #to_definition(lockfile, unlock)
- #add_git_sources private
- #check_path_source_safety private
- #check_rubygems_source_safety private
- #multiple_global_source_warning private
- #normalize_group_options(opts, groups) private
- #normalize_hash(opts) private
- #normalize_options(name, version, opts) private
- #normalize_source(source) private
- #valid_keys private
- #validate_keys(command, opts, valid_keys) private
- #with_gemfile(gemfile) private
- #with_source(source) private
RubyDsl
- Included
#normalize_ruby_file | Support the various file formats found in .ruby-version files. |
#ruby |
Constructor Details
.new ⇒ Dsl
# File 'lib/bundler/dsl.rb', line 27
def initialize @source = nil @sources = SourceList.new @git_sources = {} @dependencies = [] @groups = [] @install_conditionals = [] @optional_groups = [] @platforms = [] @env = nil @ruby_version = nil @gemspecs = [] @gemfile = nil @gemfiles = [] add_git_sources end
Dynamic Method Handling
This class handles dynamic methods through the method_missing method
#method_missing(name, *args)
# File 'lib/bundler/dsl.rb', line 293
def method_missing(name, *args) raise GemfileError, "Undefined local variable or method `#{name}' for Gemfile" end
Class Method Details
.evaluate(gemfile, lockfile, unlock)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 10
def self.evaluate(gemfile, lockfile, unlock) builder = new builder.eval_gemfile(gemfile) builder.to_definition(lockfile, unlock) end
Instance Attribute Details
#dependencies (rw)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 25
attr_accessor :dependencies
#gemfile (readonly)
[ GitHub ]#gemspecs (readonly)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 24
attr_reader :gemspecs, :gemfile
Instance Method Details
#add_git_sources (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 314
def add_git_sources git_source(:github) do |repo_name| if repo_name =~ GITHUB_PULL_REQUEST_URL { "git" => "https://github.com/#{$1}.git", "branch" => nil, "ref" => "refs/pull/#{$2}/head", "tag" => nil, } else repo_name = "#{repo_name}/#{repo_name}" unless repo_name.include?("/") "https://github.com/#{repo_name}.git" end end git_source(:gist) do |repo_name| "https://gist.github.com/#{repo_name}.git" end git_source(:bitbucket) do |repo_name| user_name, repo_name = repo_name.split("/") repo_name ||= user_name "https://#{user_name}@bitbucket.org/#{user_name}/#{repo_name}.git" end git_source(:gitlab) do |repo_name| if repo_name =~ GITLAB_MERGE_REQUEST_URL { "git" => "https://gitlab.com/#{$1}.git", "branch" => nil, "ref" => "refs/merge-requests/#{$2}/head", "tag" => nil, } else repo_name = "#{repo_name}/#{repo_name}" unless repo_name.include?("/") "https://gitlab.com/#{repo_name}.git" end end end
#check_path_source_safety (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 491
def check_path_source_safety return if @sources.global_path_source.nil? msg = "You can no longer specify a path source by itself. Instead, \n" \ "either use the :path option on a gem, or specify the gems that \n" \ "bundler should find in the path source by passing a block to \n" \ "the path method, like: \n\n" \ " path 'dir/containing/rails' do\n" \ " gem 'rails'\n" \ " end\n\n" SharedHelpers.major_deprecation(2, msg.strip) end
#check_primary_source_safety
[ GitHub ]# File 'lib/bundler/dsl.rb', line 297
def check_primary_source_safety check_path_source_safety check_rubygems_source_safety end
#check_rubygems_source_safety (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 505
def check_rubygems_source_safety multiple_global_source_warning if @sources.aggregate_global_source? end
#env(name)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 281
def env(name) old = @env @env = name yield ensure @env = old end
#eval_gemfile(gemfile, contents = nil)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 44
def eval_gemfile(gemfile, contents = nil) with_gemfile(gemfile) do |current_gemfile| contents ||= Bundler.read_file(current_gemfile) instance_eval(contents, current_gemfile, 1) rescue GemfileEvalError => e = "There was an error evaluating `#{File.basename current_gemfile}`: #{e.}" raise DSLError.new(, current_gemfile, e.backtrace, contents) rescue GemfileError, InvalidArgumentError, InvalidOption, DeprecatedError, ScriptError => e = "There was an error parsing `#{File.basename current_gemfile}`: #{e.}" raise DSLError.new(, current_gemfile, e.backtrace, contents) rescue StandardError => e raise unless e.backtrace_locations.first.path == current_gemfile = "There was an error parsing `#{File.basename current_gemfile}`: #{e.}" raise DSLError.new(, current_gemfile, e.backtrace, contents) end end
#gem(name, *args) Also known as: #_gem
[ GitHub ]# File 'lib/bundler/dsl.rb', line 95
def gem(name, *args) = args.last.is_a?(Hash) ? args.pop.dup : {} ["gemfile"] = @gemfile version = args || [">= 0"] (name, version, ) dep = Dependency.new(name, version, ) # if there's already a dependency with this name we try to prefer one if current = @dependencies.find {|d| d.name == dep.name } if current.requirement != dep.requirement current_requirement_open = current.requirements_list.include?(">= 0") gemspec_dep = [dep, current].find(&:gemspec_dev_dep?) if gemspec_dep gemfile_dep = [dep, current].find(&:runtime?) if gemfile_dep && !current_requirement_open Bundler.ui.warn "A gemspec development dependency (#{gemspec_dep.name}, #{gemspec_dep.requirement}) is being overridden by a Gemfile dependency (#{gemfile_dep.name}, #{gemfile_dep.requirement}).\n" \ "This behaviour may change in the future. Please remove either of them, or make sure they both have the same requirement\n" elsif gemfile_dep.nil? require_relative "vendor/pub_grub/lib/pub_grub/version_range" require_relative "vendor/pub_grub/lib/pub_grub/version_constraint" require_relative "vendor/pub_grub/lib/pub_grub/version_union" require_relative "vendor/pub_grub/lib/pub_grub/rubygems" current_gemspec_range = PubGrub::RubyGems.requirement_to_range(current.requirement) next_gemspec_range = PubGrub::RubyGems.requirement_to_range(dep.requirement) if current_gemspec_range.intersects?(next_gemspec_range) dep = Dependency.new(name, current.requirement.as_list + dep.requirement.as_list, ) else raise GemfileError, "Two gemspecs have conflicting requirements on the same gem: #{dep} and #{current}" end end else update_prompt = "" if File.basename(@gemfile) == Injector::INJECTED_GEMS if dep.requirements_list.include?(">= 0") && !current_requirement_open update_prompt = ". Gem already added" else update_prompt = ". If you want to update the gem version, run `bundle update #{current.name}`" update_prompt += ". You may also need to change the version requirement specified in the Gemfile if it's too restrictive." unless current_requirement_open end end raise GemfileError, "You cannot specify the same gem twice with different version requirements.\n" \ "You specified: #{current.name} (#{current.requirement}) and #{dep.name} (#{dep.requirement})" \ "#{update_prompt}" end end unless current.gemspec_dev_dep? && dep.gemspec_dev_dep? # Always prefer the dependency from the Gemfile if current.gemspec_dev_dep? @dependencies.delete(current) elsif dep.gemspec_dev_dep? return elsif current.source != dep.source raise GemfileError, "You cannot specify the same gem twice coming from different sources.\n" \ "You specified that #{dep.name} (#{dep.requirement}) should come from " \ "#{current.source || "an unspecified source"} and #{dep.source}\n" else Bundler.ui.warn "Your Gemfile lists the gem #{current.name} (#{current.requirement}) more than once.\n" \ "You should probably keep only one of them.\n" \ "Remove any duplicate entries and specify the gem only once.\n" \ "While it's not a problem now, it could cause errors if you change the version of one of them later." end end end @dependencies << dep end
#gemfile_root
[ GitHub ]# File 'lib/bundler/dsl.rb', line 629
def gemfile_root @gemfile ||= Bundler.default_gemfile @gemfile.dirname end
#gemspec(opts = nil)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 61
def gemspec(opts = nil) opts ||= {} path = opts[:path] || "." glob = opts[:glob] name = opts[:name] development_group = opts[:development_group] || :development = gemfile_root.join(path) gemspecs = Gem::Util.glob_files_in_dir("{,*}.gemspec", ).map {|g| Bundler.load_gemspec(g) }.compact gemspecs.reject! {|s| s.name != name } if name specs_by_name_and_version = gemspecs.group_by {|s| [s.name, s.version] } case specs_by_name_and_version.size when 1 specs = specs_by_name_and_version.values.first spec = specs.find {|s| s.match_platform(Bundler.local_platform) } || specs.first @gemspecs << spec gem spec.name, name: spec.name, path: path, glob: glob group(development_group) do spec.development_dependencies.each do |dep| gem dep.name, *(dep.requirement.as_list + [type: :development]) end end when 0 raise InvalidOption, "There are no gemspecs at #{}" else raise InvalidOption, "There are multiple gemspecs at #{}. " \ "Please use the :name option to specify which one should be used" end end
#git(uri, options = {}, &blk)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 222
def git(uri, = {}, &blk) unless block_given? msg = "You can no longer specify a git source by itself. Instead, \n" \ "either use the :git option on a gem, or specify the gems that \n" \ "bundler should find in the git source by passing a block to \n" \ "the git method, like: \n\n" \ " git 'git://github.com/rails/rails.git' do\n" \ " gem 'rails'\n" \ " end" raise DeprecatedError, msg end with_source(@sources.add_git_source(normalize_hash( ).merge("uri" => uri)), &blk) end
#git_source(name, &block)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 196
def git_source(name, &block) unless block_given? raise InvalidOption, "You need to pass a block to #git_source" end if valid_keys.include?(name.to_s) raise InvalidOption, "You cannot use #{name} as a git source. It " \ "is a reserved key. Reserved keys are: #{valid_keys.join(", ")}" end @git_sources[name.to_s] = block end
#github(repo, options = {})
# File 'lib/bundler/dsl.rb', line 237
def github(repo, = {}) raise InvalidArgumentError, "GitHub sources require a block" unless block_given? github_uri = @git_sources["github"].call(repo) = normalize_hash( ).merge("uri" => github_uri) git_source = @sources.add_git_source( ) with_source(git_source) { yield } end
#group(*args, &blk)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 250
def group(*args, &blk) = args.last.is_a?(Hash) ? args.pop.dup : {} (, args) @groups.concat args if ["optional"] optional_groups = args - @optional_groups @optional_groups.concat optional_groups end yield ensure args.each { @groups.pop } end
#install_if(*args)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 266
def install_if(*args) @install_conditionals.concat args yield ensure args.each { @install_conditionals.pop } end
#multiple_global_source_warning (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 509
def multiple_global_source_warning if Bundler.feature_flag.bundler_3_mode? msg = "This Gemfile contains multiple global sources. " \ "Each source after the first must include a block to indicate which gems " \ "should come from that source" raise GemfileEvalError, msg else = "Your Gemfile contains multiple global sources. " \ "Using `source` more than once without a block is a security risk, and " \ "may result in installing unexpected gems. To resolve this warning, use " \ "a block to indicate which gems should come from the secondary source." = "Your Gemfile contains multiple global sources. " \ "Using `source` more than once without a block is a security risk, and " \ "may result in installing unexpected gems. To resolve this error, use " \ "a block to indicate which gems should come from the secondary source." Bundler::SharedHelpers.major_deprecation 2, , removed_message: end end
#normalize_group_options(opts, groups) (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 443
def (opts, groups) normalize_hash(opts) groups = groups.map {|group| ":#{group}" }.join(", ") validate_keys("group #{groups}", opts, %w[optional]) opts["optional"] ||= false end
#normalize_hash(opts) (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 365
def normalize_hash(opts) opts.keys.each do |k| opts[k.to_s] = opts.delete(k) unless k.is_a?(String) end opts end
#normalize_options(name, version, opts) (private)
# File 'lib/bundler/dsl.rb', line 376
def (name, version, opts) if name.is_a?(Symbol) raise GemfileError, %(You need to specify gem names as Strings. Use 'gem "#{name}"' instead) end if /\s/.match?(name) raise GemfileError, %('#{name}' is not a valid gem name because it contains whitespace) end raise GemfileError, %(an empty gem name is not valid) if name.empty? normalize_hash(opts) git_names = @git_sources.keys.map(&:to_s) validate_keys("gem '#{name}'", opts, valid_keys + git_names) groups = @groups.dup opts["group"] = opts.delete("groups") || opts["group"] groups.concat Array(opts.delete("group")) groups = [:default] if groups.empty? install_if = @install_conditionals.dup install_if.concat Array(opts.delete("install_if")) install_if = install_if.reduce(true) do |memo, val| memo && (val.respond_to?(:call) ? val.call : val) end platforms = @platforms.dup opts["platforms"] = opts["platform"] || opts["platforms"] platforms.concat Array(opts.delete("platforms")) platforms.map!(&:to_sym) platforms.each do |p| next if VALID_PLATFORMS.include?(p) raise GemfileError, "`#{p}` is not a valid platform. The available options are: #{VALID_PLATFORMS.inspect}" end # Save sources passed in a key if opts.key?("source") source = normalize_source(opts["source"]) opts["source"] = @sources.add_rubygems_source("remotes" => source) end git_name = (git_names & opts.keys).last if @git_sources[git_name] git_opts = @git_sources[git_name].call(opts[git_name]) git_opts = { "git" => git_opts } if git_opts.is_a?(String) opts.merge!(git_opts) do |key, _gemfile_value, _git_source_value| raise GemfileError, %(The :#{key} option can't be used with `#{git_name}: #{opts[git_name].inspect}`) end end %w[git path].each do |type| next unless param = opts[type] if version.first && version.first =~ /^\s*=?\s*(\d[^\s]*)\s*$/ = opts.merge("name" => name, "version" => $1) else = opts.dup end source = send(type, param, ) {} opts["source"] = source end opts["source"] ||= @source opts["env"] ||= @env opts["platforms"] = platforms.dup opts["group"] = groups opts["should_include"] = install_if end
#normalize_source(source) (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 473
def normalize_source(source) case source when :gemcutter, :rubygems, :rubyforge = "The source :#{source} is deprecated because HTTP requests are insecure.\n" \ "Please change your source to 'https://rubygems.org' if possible, or 'http://rubygems.org' if not." = "The source :#{source} is disallowed because HTTP requests are insecure.\n" \ "Please change your source to 'https://rubygems.org' if possible, or 'http://rubygems.org' if not." Bundler::SharedHelpers.major_deprecation 2, , removed_message: "http://rubygems.org" when String source else raise GemfileError, "Unknown source '#{source}'" end end
#path(path, options = {}, &blk)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 209
def path(path, = {}, &blk) = normalize_hash( ).merge( "path" => Pathname.new(path), "root_path" => gemfile_root, "gemspec" => gemspecs.find {|g| g.name == ["name"] } ) ["global"] = true unless block_given? source = @sources.add_path_source( ) with_source(source, &blk) end
#platform(*platforms)
Alias for #platforms.
# File 'lib/bundler/dsl.rb', line 279
alias_method :platform, :platforms
#platforms(*platforms) Also known as: #platform
[ GitHub ]# File 'lib/bundler/dsl.rb', line 273
def platforms(*platforms) @platforms.concat platforms yield ensure platforms.each { @platforms.pop } end
#plugin(*args)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 289
def plugin(*args) # Pass on end
#source(source, *args, &blk)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 172
def source(source, *args, &blk) = args.last.is_a?(Hash) ? args.pop.dup : {} = normalize_hash( ) source = normalize_source(source) if .key?("type") ["type"] = ["type"].to_s unless Plugin.source?( ["type"]) raise InvalidOption, "No plugin sources available for #{ ["type"]}" end unless block_given? raise InvalidOption, "You need to pass a block to #source with :type option" end source_opts = .merge("uri" => source) with_source(@sources.add_plugin_source( ["type"], source_opts), &blk) elsif block_given? with_source(@sources.add_rubygems_source("remotes" => source), &blk) else @sources.add_global_rubygems_remote(source) end end
#to_definition(lockfile, unlock)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 245
def to_definition(lockfile, unlock) check_primary_source_safety Definition.new(lockfile, @dependencies, @sources, unlock, @ruby_version, @optional_groups, @gemfiles) end
#valid_keys (private)
[ GitHub ]# File 'lib/bundler/dsl.rb', line 372
def valid_keys @valid_keys ||= VALID_KEYS end
#validate_keys(command, opts, valid_keys) (private)
# File 'lib/bundler/dsl.rb', line 452
def validate_keys(command, opts, valid_keys) if opts["branch"] && !(opts["git"] || opts["github"] || (opts.keys & @git_sources.keys.map(&:to_s)).any?) raise GemfileError, %(The `branch` option for `#{command}` is not allowed. Only gems with a git source can specify a branch) end invalid_keys = opts.keys - valid_keys return true unless invalid_keys.any? = String.new << "You passed #{invalid_keys.map {|k| ":" + k }.join(", ")} " << if invalid_keys.size > 1 "as options for #{command}, but they are invalid." else "as an option for #{command}, but it is invalid." end << " Valid options are: #{valid_keys.join(", ")}." << " You may be able to resolve this by upgrading Bundler to the newest version." raise InvalidOption, end