mirror of
				https://github.com/KevinMidboe/linguist.git
				synced 2025-10-29 17:50:22 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			186 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			186 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
module Linguist
 | 
						|
  # A collection of simple heuristics that can be used to better analyze languages.
 | 
						|
  class Heuristics
 | 
						|
    # Public: Use heuristics to detect language of the blob.
 | 
						|
    #
 | 
						|
    # blob               - An object that quacks like a blob.
 | 
						|
    # possible_languages - Array of Language objects
 | 
						|
    #
 | 
						|
    # Examples
 | 
						|
    #
 | 
						|
    #   Heuristics.call(FileBlob.new("path/to/file"), [
 | 
						|
    #     Language["Ruby"], Language["Python"]
 | 
						|
    #   ])
 | 
						|
    #
 | 
						|
    # Returns an Array of languages, or empty if none matched or were inconclusive.
 | 
						|
    def self.call(blob, languages)
 | 
						|
      data = blob.data
 | 
						|
 | 
						|
      @heuristics.each do |heuristic|
 | 
						|
        return Array(heuristic.call(data)) if heuristic.matches?(languages)
 | 
						|
      end
 | 
						|
 | 
						|
      [] # No heuristics matched
 | 
						|
    end
 | 
						|
 | 
						|
    # Internal: Define a new heuristic.
 | 
						|
    #
 | 
						|
    # languages - String names of languages to disambiguate.
 | 
						|
    # heuristic - Block which takes data as an argument and returns a Language or nil.
 | 
						|
    #
 | 
						|
    # Examples
 | 
						|
    #
 | 
						|
    #     disambiguate "Perl", "Prolog" do |data|
 | 
						|
    #       if data.include?("use strict")
 | 
						|
    #         Language["Perl"]
 | 
						|
    #       elsif data.include?(":-")
 | 
						|
    #         Language["Prolog"]
 | 
						|
    #       end
 | 
						|
    #     end
 | 
						|
    #
 | 
						|
    def self.disambiguate(*languages, &heuristic)
 | 
						|
      @heuristics << new(languages, &heuristic)
 | 
						|
    end
 | 
						|
 | 
						|
    # Internal: Array of defined heuristics
 | 
						|
    @heuristics = []
 | 
						|
 | 
						|
    # Internal
 | 
						|
    def initialize(languages, &heuristic)
 | 
						|
      @languages = languages
 | 
						|
      @heuristic = heuristic
 | 
						|
    end
 | 
						|
 | 
						|
    # Internal: Check if this heuristic matches the candidate languages.
 | 
						|
    def matches?(candidates)
 | 
						|
      candidates.any? && candidates.all? { |l| @languages.include?(l.name) }
 | 
						|
    end
 | 
						|
 | 
						|
    # Internal: Perform the heuristic
 | 
						|
    def call(data)
 | 
						|
      @heuristic.call(data)
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "BitBake", "BlitzBasic" do |data|
 | 
						|
      if /^\s*; /.match(data) || data.include?("End Function")
 | 
						|
        Language["BlitzBasic"]
 | 
						|
      elsif /^\s*(# |include|require)\b/.match(data)
 | 
						|
        Language["BitBake"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Objective-C", "C++", "C" do |data|
 | 
						|
      if (/@(interface|class|protocol|property|end|synchronised|selector|implementation)\b/.match(data))
 | 
						|
        Language["Objective-C"]
 | 
						|
      elsif (/^\s*#\s*include <(cstdint|string|vector|map|list|array|bitset|queue|stack|forward_list|unordered_map|unordered_set|(i|o|io)stream)>/.match(data) ||
 | 
						|
        /^\s*template\s*</.match(data) || /^[^@]class\s+\w+/.match(data) || /^[^@](private|public|protected):$/.match(data) || /std::.+$/.match(data))
 | 
						|
        Language["C++"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Perl", "Perl6", "Prolog" do |data|
 | 
						|
      if data.include?("use v6")
 | 
						|
        Language["Perl6"]
 | 
						|
      elsif data.include?("use strict")
 | 
						|
        Language["Perl"]
 | 
						|
      elsif data.include?(":-")
 | 
						|
        Language["Prolog"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "ECL", "Prolog" do |data|
 | 
						|
      if data.include?(":-")
 | 
						|
        Language["Prolog"]
 | 
						|
      elsif data.include?(":=")
 | 
						|
        Language["ECL"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "IDL", "Prolog" do |data|
 | 
						|
      if data.include?(":-")
 | 
						|
        Language["Prolog"]
 | 
						|
      else
 | 
						|
        Language["IDL"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Common Lisp", "OpenCL", "Cool" do |data|
 | 
						|
      if data.include?("(defun ")
 | 
						|
        Language["Common Lisp"]
 | 
						|
      elsif /^class/x.match(data)
 | 
						|
        Language["Cool"]
 | 
						|
      elsif /\/\* |\/\/ |^\}/.match(data)
 | 
						|
        Language["OpenCL"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Hack", "PHP" do |data|
 | 
						|
      if data.include?("<?hh")
 | 
						|
        Language["Hack"]
 | 
						|
      elsif /<?[^h]/.match(data)
 | 
						|
        Language["PHP"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Scala", "SuperCollider" do |data|
 | 
						|
      if /\^(this|super)\./.match(data) || /^\s*(\+|\*)\s*\w+\s*{/.match(data) || /^\s*~\w+\s*=\./.match(data)
 | 
						|
        Language["SuperCollider"]
 | 
						|
      elsif /^\s*import (scala|java)\./.match(data) || /^\s*val\s+\w+\s*=/.match(data) || /^\s*class\b/.match(data)
 | 
						|
        Language["Scala"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "AsciiDoc", "AGS Script" do |data|
 | 
						|
      Language["AsciiDoc"] if /^=+(\s|\n)/.match(data)
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "FORTRAN", "Forth" do |data|
 | 
						|
      if /^: /.match(data)
 | 
						|
        Language["Forth"]
 | 
						|
      elsif /^([c*][^a-z]|      subroutine\s)/i.match(data)
 | 
						|
        Language["FORTRAN"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "F#", "Forth", "GLSL" do |data|
 | 
						|
      if /^(: |new-device)/.match(data)
 | 
						|
        Language["Forth"]
 | 
						|
      elsif /^\s*(#light|import|let|module|namespace|open|type)/.match(data)
 | 
						|
        Language["F#"]
 | 
						|
      elsif /^\s*(#include|#pragma|precision|uniform|varying|void)/.match(data)
 | 
						|
        Language["GLSL"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Gosu", "JavaScript" do |data|
 | 
						|
      Language["Gosu"] if /^uses java\./.match(data)
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "LoomScript", "LiveScript" do |data|
 | 
						|
      if /^\s*package\s*[\w\.\/\*\s]*\s*{/.match(data)
 | 
						|
        Language["LoomScript"]
 | 
						|
      else
 | 
						|
        Language["LiveScript"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "TypeScript", "XML" do |data|
 | 
						|
      if data.include?("<TS ")
 | 
						|
        Language["XML"]
 | 
						|
      else
 | 
						|
        Language["TypeScript"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
 | 
						|
    disambiguate "Frege", "Forth", "text" do |data|
 | 
						|
      if /^(: |also |new-device|previous )/.match(data)
 | 
						|
        Language["Forth"]
 | 
						|
      elsif /\s*(import|module|package|data|type) /.match(data)
 | 
						|
        Language["Frege"]
 | 
						|
      else
 | 
						|
        Language["text"]
 | 
						|
      end
 | 
						|
    end
 | 
						|
  end
 | 
						|
end
 |