Sha256: 28d708e97dee77caf95a2f1d8dce9fcd144f01f35909e3af14bfc4cb501d7886

Contents?: true

Size: 1.78 KB

Versions: 14

Compression:

Stored size: 1.78 KB

Contents

module Linguist
  class Shebang
    # Public: Use shebang to detect language of the blob.
    #
    # blob               - An object that quacks like a blob.
    # candidates         - A list of candidate languages.
    #
    # Examples
    #
    #   Shebang.call(FileBlob.new("path/to/file"))
    #
    # Returns an array of languages from the candidate list for which the
    # blob's shebang is valid. Returns an empty list if there is no shebang.
    # If the candidate list is empty, any language is a valid candidate.
    def self.call(blob, candidates)
      return [] if blob.symlink?

      languages = Language.find_by_interpreter interpreter(blob.data)
      candidates.any? ? candidates & languages : languages
    end

    # Public: Get the interpreter from the shebang
    #
    # Returns a String or nil
    def self.interpreter(data)
      shebang = data.lines.first

      # First line must start with #!
      return unless shebang && shebang.start_with?("#!")

      s = StringScanner.new(shebang)

      # There was nothing after the #!
      return unless path = s.scan(/^#!\s*\S+/)

      # Keep going
      script = path.split('/').last

      # if /usr/bin/env type shebang then walk the string
      if script == 'env'
        s.scan(/\s+/)
        s.scan(/.*=[^\s]+\s+/) # skip over variable arguments e.g. foo=bar
        script = s.scan(/\S+/)
      end

      # Interpreter was /usr/bin/env with no arguments
      return unless script

      # "python2.6" -> "python2"
      script.sub!(/(\.\d+)$/, '')

      # #! perl -> perl
      script.sub!(/^#!\s*/, '')

      # Check for multiline shebang hacks that call `exec`
      if script == 'sh' &&
        data.lines.first(5).any? { |l| l.match(/exec (\w+).+\$0.+\$@/) }
        script = $1
      end

      File.basename(script)
    end
  end
end

Version data entries

14 entries across 14 versions & 1 rubygems

Version Path
github-linguist-7.3.1 lib/linguist/shebang.rb
github-linguist-7.3.0 lib/linguist/shebang.rb
github-linguist-7.2.0 lib/linguist/shebang.rb
github-linguist-7.1.3 lib/linguist/shebang.rb
github-linguist-7.1.2 lib/linguist/shebang.rb
github-linguist-7.1.1 lib/linguist/shebang.rb
github-linguist-7.1.0 lib/linguist/shebang.rb
github-linguist-7.0.0 lib/linguist/shebang.rb
github-linguist-6.4.1 lib/linguist/shebang.rb
github-linguist-6.4.0 lib/linguist/shebang.rb
github-linguist-6.3.1 lib/linguist/shebang.rb
github-linguist-6.3.0 lib/linguist/shebang.rb
github-linguist-6.2.0 lib/linguist/shebang.rb
github-linguist-6.1.0 lib/linguist/shebang.rb