mirror of
https://github.com/KevinMidboe/linguist.git
synced 2025-10-29 17:50:22 +00:00
117 lines
3.3 KiB
Ruby
117 lines
3.3 KiB
Ruby
require 'linguist/tokenizer'
|
|
|
|
module Linguist
|
|
# Language bayesian classifier.
|
|
class Classifier
|
|
# Public: Initialize a Classifier.
|
|
def initialize(attrs = {})
|
|
@tokens_total = attrs['tokens_total'] || 0
|
|
@languages_total = attrs['languages_total'] || 0
|
|
@tokens = attrs['tokens'] || {}
|
|
@language_tokens = attrs['language_tokens'] || {}
|
|
@languages = attrs['languages'] || {}
|
|
end
|
|
|
|
# Public: Train classifier that data is a certain language.
|
|
#
|
|
# language - String language of data
|
|
# data - String contents of file
|
|
#
|
|
# Examples
|
|
#
|
|
# train('Ruby', "def hello; end")
|
|
#
|
|
# Returns nothing.
|
|
def train(language, data)
|
|
tokens = Tokenizer.tokenize(data)
|
|
|
|
tokens.each do |token|
|
|
@tokens[language] ||= {}
|
|
@tokens[language][token] ||= 0
|
|
@tokens[language][token] += 1
|
|
@language_tokens[language] ||= 0
|
|
@language_tokens[language] += 1
|
|
@tokens_total += 1
|
|
end
|
|
@languages[language] ||= 0
|
|
@languages[language] += 1
|
|
@languages_total += 1
|
|
|
|
nil
|
|
end
|
|
|
|
# Public: Guess language of data.
|
|
#
|
|
# data - Array of tokens or String data to analyze.
|
|
# languages - Array of language name Strings to restrict to.
|
|
#
|
|
# Examples
|
|
#
|
|
# classify("def hello; end")
|
|
# # => [ 'Ruby', 0.90], ['Python', 0.2], ... ]
|
|
#
|
|
# Returns sorted Array of result pairs. Each pair contains the
|
|
# String language name and a Float score.
|
|
def classify(tokens, languages = @languages.keys)
|
|
return [] if tokens.nil?
|
|
tokens = Tokenizer.tokenize(tokens) if tokens.is_a?(String)
|
|
|
|
scores = {}
|
|
languages.each do |language|
|
|
scores[language] = tokens_probability(tokens, language) +
|
|
language_probability(language)
|
|
end
|
|
|
|
scores.sort { |a, b| b[1] <=> a[1] }.map { |score| [score[0], score[1]] }
|
|
end
|
|
|
|
# Internal: Probably of set of tokens in a language occuring - P(D | C)
|
|
#
|
|
# tokens - Array of String tokens.
|
|
# language - Language to check.
|
|
#
|
|
# Returns Float between 0.0 and 1.0.
|
|
def tokens_probability(tokens, language)
|
|
tokens.inject(0.0) do |sum, token|
|
|
sum += Math.log(token_probability(token, language))
|
|
end
|
|
end
|
|
|
|
# Internal: Probably of token in language occuring - P(F | C)
|
|
#
|
|
# token - String token.
|
|
# language - Language to check.
|
|
#
|
|
# Returns Float between 0.0 and 1.0.
|
|
def token_probability(token, language)
|
|
if @tokens[language][token].to_f == 0.0
|
|
1 / @tokens_total.to_f
|
|
else
|
|
@tokens[language][token].to_f / @language_tokens[language].to_f
|
|
end
|
|
end
|
|
|
|
# Internal: Probably of a language occuring - P(C)
|
|
#
|
|
# language - Language to check.
|
|
#
|
|
# Returns Float between 0.0 and 1.0.
|
|
def language_probability(language)
|
|
Math.log(@languages[language].to_f / @languages_total.to_f)
|
|
end
|
|
|
|
# Public: Returns serializable hash representation.
|
|
#
|
|
# Returns Hash.
|
|
def to_hash
|
|
{
|
|
'tokens_total' => @tokens_total,
|
|
'languages_total' => @languages_total,
|
|
'tokens' => @tokens,
|
|
'language_tokens' => @language_tokens,
|
|
'languages' => @languages
|
|
}
|
|
end
|
|
end
|
|
end
|