Compare commits

...

26 Commits

Author SHA1 Message Date
Arfon Smith
8c8434ed64 Merge pull request #1335 from github/1318-local
1318 local
2014-07-01 11:48:24 -05:00
Arfon Smith
9281bd043a Version 2014-07-01 11:19:05 -05:00
Arfon Smith
6771f7c272 Merge branch 'master' into 1318-local 2014-07-01 11:12:44 -05:00
Vicent Marti
df09a746a0 b3 2014-06-27 16:57:58 +02:00
Vicent Marti
d9be472ccb Skip submodules when diffing 2014-06-27 16:41:23 +02:00
Vicent Marti
32828a9af5 b2 2014-06-27 13:51:56 +02:00
Vicent Marti
d206131df0 Hardcode OIDs for test 2014-06-27 13:51:37 +02:00
Vicent Marti
65eaf98d0b docs 2014-06-26 21:26:26 +02:00
Vicent Marti
12429b90fe Bring back missing test 2014-06-26 21:24:30 +02:00
Vicent Marti
621042e639 Remove whitespace 2014-06-26 18:42:43 +02:00
Arfon Smith
526244be11 Samples 2014-06-26 17:38:39 +01:00
Arfon Smith
bc53d0b55e Merge pull request #1311 from maximusvladimir/master
Added 3 character glsl extensions.
2014-06-26 17:37:42 +01:00
Vicent Marti
907d3c5a36 b1 2014-06-26 18:17:51 +02:00
Max K.
898f1e215e Added sample files for glsl. 2014-06-26 09:25:40 -05:00
Vicent Marti
324ac83489 Use the new Rugged release 2014-06-26 14:12:00 +02:00
Vicent Marti
00a873dcc7 Bump 3.0.0b0 2014-06-26 13:03:41 +02:00
Vicent Marti
bc34345a56 Fix the linguist binary 2014-06-26 13:03:30 +02:00
Vicent Marti
659d27cae5 DOCS 2014-06-26 12:54:08 +02:00
Vicent Marti
29072d6eae Fix travis build 2014-06-26 12:27:02 +02:00
Vicent Marti
1fd59361b5 Proper incremental diffing 2014-06-25 20:26:44 +02:00
Vicent Marti
5896bb8fa3 Missing file. Duh. 2014-06-24 17:52:43 +02:00
Vicent Marti
ea1fc90cf5 Handle nil blob names 2014-06-24 17:43:01 +02:00
Vicent Marti
463f48f04f Mode must always be a String 2014-06-24 17:41:16 +02:00
Vicent Marti
cd58a30c7c Only cache strings, thanks 2014-06-24 17:41:16 +02:00
Vicent Marti
c4260ae681 Use Rugged when computing Repository stats 2014-06-24 17:41:16 +02:00
Max K.
b83a364b0e Added 3 character glsl extensions. 2014-06-23 15:46:28 -05:00
14 changed files with 229 additions and 105 deletions

View File

@@ -1,4 +1,6 @@
before_install: before_install:
- git fetch origin master:master
- git fetch origin v2.0.0:v2.0.0
- sudo apt-get install libicu-dev -y - sudo apt-get install libicu-dev -y
- gem update --system 2.1.11 - gem update --system 2.1.11
rvm: rvm:

View File

@@ -5,6 +5,7 @@
require 'linguist/file_blob' require 'linguist/file_blob'
require 'linguist/repository' require 'linguist/repository'
require 'rugged'
path = ARGV[0] || Dir.pwd path = ARGV[0] || Dir.pwd
@@ -18,7 +19,8 @@ ARGV.shift
breakdown = true if ARGV[0] == "--breakdown" breakdown = true if ARGV[0] == "--breakdown"
if File.directory?(path) if File.directory?(path)
repo = Linguist::Repository.from_directory(path) rugged = Rugged::Repository.new(path)
repo = Linguist::Repository.new(rugged, rugged.head.target_id)
repo.languages.sort_by { |_, size| size }.reverse.each do |language, size| repo.languages.sort_by { |_, size| size }.reverse.each do |language, size|
percentage = ((size / repo.size.to_f) * 100) percentage = ((size / repo.size.to_f) * 100)
percentage = sprintf '%.2f' % percentage percentage = sprintf '%.2f' % percentage

View File

@@ -17,6 +17,7 @@ Gem::Specification.new do |s|
s.add_dependency 'escape_utils', '~> 1.0.1' s.add_dependency 'escape_utils', '~> 1.0.1'
s.add_dependency 'mime-types', '~> 1.19' s.add_dependency 'mime-types', '~> 1.19'
s.add_dependency 'pygments.rb', '~> 0.6.0' s.add_dependency 'pygments.rb', '~> 0.6.0'
s.add_dependency 'rugged', '~> 0.21.0'
s.add_development_dependency 'json' s.add_development_dependency 'json'
s.add_development_dependency 'mocha' s.add_development_dependency 'mocha'

View File

@@ -313,15 +313,7 @@ module Linguist
# #
# Returns a Language or nil if none is detected # Returns a Language or nil if none is detected
def language def language
return @language if defined? @language @language ||= Language.detect(self)
if defined?(@data) && @data.is_a?(String)
data = @data
else
data = lambda { (binary_mime_type? || binary?) ? "" : self.data }
end
@language = Language.detect(name.to_s, data, mode)
end end
# Internal: Get the lexer of the blob. # Internal: Get the lexer of the blob.

View File

@@ -92,18 +92,17 @@ module Linguist
# Public: Detects the Language of the blob. # Public: Detects the Language of the blob.
# #
# name - String filename # blob - an object that implements the Linguist `Blob` interface;
# data - String blob data. A block also maybe passed in for lazy # see Linguist::LazyBlob and Linguist::FileBlob for examples
# loading. This behavior is deprecated and you should always
# pass in a String.
# mode - Optional String mode (defaults to nil)
# #
# Returns Language or nil. # Returns Language or nil.
def self.detect(name, data, mode = nil) def self.detect(blob)
name = blob.name.to_s
# A bit of an elegant hack. If the file is executable but extensionless, # A bit of an elegant hack. If the file is executable but extensionless,
# append a "magic" extension so it can be classified with other # append a "magic" extension so it can be classified with other
# languages that have shebang scripts. # languages that have shebang scripts.
if File.extname(name).empty? && mode && (mode.to_i(8) & 05) == 05 if File.extname(name).empty? && blob.mode && (blob.mode.to_i(8) & 05) == 05
name += ".script!" name += ".script!"
end end
@@ -114,7 +113,7 @@ module Linguist
# extension at all, in the case of extensionless scripts), we need to continue # extension at all, in the case of extensionless scripts), we need to continue
# our detection work # our detection work
if possible_languages.length > 1 if possible_languages.length > 1
data = data.call() if data.respond_to?(:call) data = blob.data
possible_language_names = possible_languages.map(&:name) possible_language_names = possible_languages.map(&:name)
# Don't bother with emptiness # Don't bother with emptiness

View File

@@ -735,12 +735,14 @@ GLSL:
- .glsl - .glsl
- .fp - .fp
- .frag - .frag
- .frg
- .fshader - .fshader
- .geom - .geom
- .glslv - .glslv
- .gshader - .gshader
- .shader - .shader
- .vert - .vert
- .vrx
- .vshader - .vshader
Genshi: Genshi:

37
lib/linguist/lazy_blob.rb Normal file
View File

@@ -0,0 +1,37 @@
require 'linguist/blob_helper'
require 'rugged'
module Linguist
class LazyBlob
include BlobHelper
MAX_SIZE = 128 * 1024
attr_reader :repository
attr_reader :oid
attr_reader :name
attr_reader :mode
def initialize(repo, oid, name, mode = nil)
@repository = repo
@oid = oid
@name = name
@mode = mode
end
def data
load_blob!
@data
end
def size
load_blob!
@size
end
protected
def load_blob!
@data, @size = Rugged::Blob.to_buffer(repository, oid, MAX_SIZE) if @data.nil?
end
end
end

View File

@@ -1,4 +1,5 @@
require 'linguist/file_blob' require 'linguist/lazy_blob'
require 'rugged'
module Linguist module Linguist
# A Repository is an abstraction of a Grit::Repo or a basic file # A Repository is an abstraction of a Grit::Repo or a basic file
@@ -7,100 +8,146 @@ module Linguist
# Its primary purpose is for gathering language statistics across # Its primary purpose is for gathering language statistics across
# the entire project. # the entire project.
class Repository class Repository
# Public: Initialize a new Repository from a File directory attr_reader :repository
#
# base_path - A path String # Public: Create a new Repository based on the stats of
# # an existing one
# Returns a Repository def self.incremental(repo, commit_oid, old_commit_oid, old_stats)
def self.from_directory(base_path) repo = self.new(repo, commit_oid)
new Dir["#{base_path}/**/*"]. repo.load_existing_stats(old_commit_oid, old_stats)
select { |f| File.file?(f) }. repo
map { |path| FileBlob.new(path, base_path) }
end end
# Public: Initialize a new Repository # Public: Initialize a new Repository to be analyzed for language
# data
# #
# enum - Enumerator that responds to `each` and # repo - a Rugged::Repository object
# yields Blob objects # commit_oid - the sha1 of the commit that will be analyzed;
# this is usually the master branch
# #
# Returns a Repository # Returns a Repository
def initialize(enum) def initialize(repo, commit_oid)
@enum = enum @repository = repo
@computed_stats = false @commit_oid = commit_oid
@language = @size = nil
@sizes = Hash.new { 0 } raise TypeError, 'commit_oid must be a commit SHA1' unless commit_oid.is_a?(String)
@file_breakdown = Hash.new { |h,k| h[k] = Array.new } end
# Public: Load the results of a previous analysis on this repository
# to speed up the new scan.
#
# The new analysis will be performed incrementally as to only take
# into account the file changes since the last time the repository
# was scanned
#
# old_commit_oid - the sha1 of the commit that was previously analyzed
# old_stats - the result of the previous analysis, obtained by calling
# Repository#cache on the old repository
#
# Returns nothing
def load_existing_stats(old_commit_oid, old_stats)
@old_commit_oid = old_commit_oid
@old_stats = old_stats
nil
end end
# Public: Returns a breakdown of language stats. # Public: Returns a breakdown of language stats.
# #
# Examples # Examples
# #
# # => { Language['Ruby'] => 46319, # # => { 'Ruby' => 46319,
# Language['JavaScript'] => 258 } # 'JavaScript' => 258 }
# #
# Returns a Hash of Language keys and Integer size values. # Returns a Hash of language names and Integer size values.
def languages def languages
compute_stats @sizes ||= begin
@sizes sizes = Hash.new { 0 }
cache.each do |_, (language, size)|
sizes[language] += size
end
sizes
end
end end
# Public: Get primary Language of repository. # Public: Get primary Language of repository.
# #
# Returns a Language # Returns a language name
def language def language
compute_stats @language ||= begin
@language primary = languages.max_by { |(_, size)| size }
primary && primary[0]
end
end end
# Public: Get the total size of the repository. # Public: Get the total size of the repository.
# #
# Returns a byte size Integer # Returns a byte size Integer
def size def size
compute_stats @size ||= languages.inject(0) { |s,(_,v)| s + v }
@size
end end
# Public: Return the language breakdown of this repository by file # Public: Return the language breakdown of this repository by file
#
# Returns a map of language names => [filenames...]
def breakdown_by_file def breakdown_by_file
compute_stats @file_breakdown ||= begin
@file_breakdown breakdown = Hash.new { |h,k| h[k] = Array.new }
cache.each do |filename, (language, _)|
breakdown[language] << filename
end
breakdown
end
end end
# Internal: Compute language breakdown for each blob in the Repository. # Public: Return the cached results of the analysis
# #
# Returns nothing # This is a per-file breakdown that can be passed to other instances
def compute_stats # of Linguist::Repository to perform incremental scans
return if @computed_stats #
# Returns a map of filename => [language, size]
def cache
@cache ||= begin
if @old_commit_oid == @commit_oid
@old_stats
else
compute_stats(@old_commit_oid, @commit_oid, @old_stats)
end
end
end
@enum.each do |blob| protected
# Skip files that are likely binary def compute_stats(old_commit_oid, commit_oid, cache = nil)
next if blob.likely_binary? file_map = cache ? cache.dup : {}
old_tree = old_commit_oid && Rugged::Commit.lookup(repository, old_commit_oid).tree
new_tree = Rugged::Commit.lookup(repository, commit_oid).tree
diff = Rugged::Tree.diff(repository, old_tree, new_tree)
diff.each_delta do |delta|
old = delta.old_file[:path]
new = delta.new_file[:path]
file_map.delete(old)
next if delta.binary
if [:added, :modified].include? delta.status
# Skip submodules
mode = delta.new_file[:mode]
next if (mode & 040000) != 0
blob = Linguist::LazyBlob.new(repository, delta.new_file[:oid], new, mode.to_s(8))
# Skip vendored or generated blobs # Skip vendored or generated blobs
next if blob.vendored? || blob.generated? || blob.language.nil? next if blob.vendored? || blob.generated? || blob.language.nil?
# Only include programming languages and acceptable markup languages # Only include programming languages and acceptable markup languages
if blob.language.type == :programming || Language.detectable_markup.include?(blob.language.name) if blob.language.type == :programming || Language.detectable_markup.include?(blob.language.name)
file_map[new] = [blob.language.group.name, blob.size]
# Build up the per-file breakdown stats end
@file_breakdown[blob.language.group.name] << blob.name end
end
@sizes[blob.language.group] += blob.size
end file_map
end
# Compute total size
@size = @sizes.inject(0) { |s,(_,v)| s + v }
# Get primary language
if primary = @sizes.max_by { |(_, size)| size }
@language = primary[0]
end
@computed_stats = true
nil
end end
end end
end end

View File

@@ -170,7 +170,9 @@
"GLSL": [ "GLSL": [
".fp", ".fp",
".frag", ".frag",
".glsl" ".frg",
".glsl",
".vrx"
], ],
"Game Maker Language": [ "Game Maker Language": [
".gml" ".gml"
@@ -781,8 +783,8 @@
"exception.zep.php" "exception.zep.php"
] ]
}, },
"tokens_total": 643487, "tokens_total": 643530,
"languages_total": 843, "languages_total": 845,
"tokens": { "tokens": {
"ABAP": { "ABAP": {
"*/**": 1, "*/**": 1,
@@ -23553,14 +23555,14 @@
"////": 4, "////": 4,
"High": 1, "High": 1,
"quality": 2, "quality": 2,
"(": 435, "(": 437,
"Some": 1, "Some": 1,
"browsers": 1, "browsers": 1,
"may": 1, "may": 1,
"freeze": 1, "freeze": 1,
"or": 1, "or": 1,
"crash": 1, "crash": 1,
")": 435, ")": 437,
"//#define": 10, "//#define": 10,
"HIGHQUALITY": 2, "HIGHQUALITY": 2,
"Medium": 1, "Medium": 1,
@@ -23602,7 +23604,7 @@
"eps": 5, "eps": 5,
"e": 4, "e": 4,
"-": 108, "-": 108,
";": 383, ";": 391,
"PI": 3, "PI": 3,
"vec3": 165, "vec3": 165,
"sunDir": 5, "sunDir": 5,
@@ -23621,13 +23623,13 @@
"tonemapping": 1, "tonemapping": 1,
"mod289": 4, "mod289": 4,
"x": 11, "x": 11,
"{": 82, "{": 84,
"return": 47, "return": 47,
"floor": 8, "floor": 8,
"*": 115, "*": 116,
"/": 24, "/": 24,
"}": 82, "}": 84,
"vec4": 73, "vec4": 77,
"permute": 4, "permute": 4,
"x*34.0": 1, "x*34.0": 1,
"+": 108, "+": 108,
@@ -23743,7 +23745,7 @@
"p*0.5": 1, "p*0.5": 1,
"Other": 1, "Other": 1,
"bump": 2, "bump": 2,
"pos": 42, "pos": 43,
"rayDir": 43, "rayDir": 43,
"s": 23, "s": 23,
"Fade": 1, "Fade": 1,
@@ -23931,8 +23933,8 @@
"iResolution.x/iResolution.y*0.5": 1, "iResolution.x/iResolution.y*0.5": 1,
"rd.x": 1, "rd.x": 1,
"rd.y": 1, "rd.y": 1,
"void": 31, "void": 33,
"main": 5, "main": 7,
"gl_FragCoord.xy": 7, "gl_FragCoord.xy": 7,
"*0.25": 4, "*0.25": 4,
"*0.5": 1, "*0.5": 1,
@@ -23944,7 +23946,16 @@
"col*exposure": 1, "col*exposure": 1,
"x*": 2, "x*": 2,
".5": 1, ".5": 1,
"gl_FragColor": 3, "gl_FragColor": 4,
"varying": 6,
"v_color": 4,
"uniform": 8,
"mat4": 1,
"u_MVPMatrix": 2,
"attribute": 2,
"a_position": 1,
"a_color": 2,
"gl_Position": 1,
"#version": 2, "#version": 2,
"core": 1, "core": 1,
"cbar": 2, "cbar": 2,
@@ -23980,11 +23991,9 @@
"AMBIENT": 2, "AMBIENT": 2,
"MAX_DIST": 3, "MAX_DIST": 3,
"MAX_DIST_SQUARED": 3, "MAX_DIST_SQUARED": 3,
"uniform": 7,
"lightColor": 3, "lightColor": 3,
"[": 29, "[": 29,
"]": 29, "]": 29,
"varying": 4,
"fragmentNormal": 2, "fragmentNormal": 2,
"cameraVector": 2, "cameraVector": 2,
"lightVector": 4, "lightVector": 4,
@@ -69657,7 +69666,7 @@
"GAMS": 363, "GAMS": 363,
"GAP": 9944, "GAP": 9944,
"GAS": 133, "GAS": 133,
"GLSL": 4033, "GLSL": 4076,
"Game Maker Language": 13310, "Game Maker Language": 13310,
"Gnuplot": 1023, "Gnuplot": 1023,
"Gosu": 410, "Gosu": 410,
@@ -69854,7 +69863,7 @@
"GAMS": 1, "GAMS": 1,
"GAP": 7, "GAP": 7,
"GAS": 1, "GAS": 1,
"GLSL": 5, "GLSL": 7,
"Game Maker Language": 13, "Game Maker Language": 13,
"Gnuplot": 6, "Gnuplot": 6,
"Gosu": 4, "Gosu": 4,
@@ -70004,5 +70013,5 @@
"fish": 3, "fish": 3,
"wisp": 1 "wisp": 1
}, },
"md5": "59afe9ab875947c5df3590aef023152d" "md5": "bb637c5d1f457edff3f8c2676d10807a"
} }

View File

@@ -1,3 +1,3 @@
module Linguist module Linguist
VERSION = "2.12.0" VERSION = "3.0.0"
end end

View File

@@ -0,0 +1,6 @@
varying vec4 v_color;
void main()
{
gl_FragColor = v_color;
}

12
samples/GLSL/myvertex.vrx Normal file
View File

@@ -0,0 +1,12 @@
uniform mat4 u_MVPMatrix;
attribute vec4 a_position;
attribute vec4 a_color;
varying vec4 v_color;
void main()
{
v_color = a_color;
gl_Position = u_MVPMatrix * pos;
}

View File

@@ -1,6 +1,7 @@
require 'linguist/heuristics' require 'linguist/heuristics'
require 'linguist/language' require 'linguist/language'
require 'linguist/samples' require 'linguist/samples'
require 'linguist/file_blob'
require 'test/unit' require 'test/unit'
@@ -35,7 +36,8 @@ class TestHeuristcs < Test::Unit::TestCase
end end
def test_detect_still_works_if_nothing_matches def test_detect_still_works_if_nothing_matches
match = Language.detect("Hello.m", fixture("Objective-C/hello.m")) blob = Linguist::FileBlob.new(File.join(samples_path, "Objective-C/hello.m"))
match = Language.detect(blob)
assert_equal Language["Objective-C"], match assert_equal Language["Objective-C"], match
end end

View File

@@ -3,22 +3,24 @@ require 'linguist/repository'
require 'test/unit' require 'test/unit'
class TestRepository < Test::Unit::TestCase class TestRepository < Test::Unit::TestCase
include Linguist def rugged_repository
@rugged ||= Rugged::Repository.new(File.expand_path("../../.git", __FILE__))
def repo(base_path)
Repository.from_directory(base_path)
end end
def linguist_repo def master_oid
repo(File.expand_path("../..", __FILE__)) 'd40b4a33deba710e2f494db357c654fbe5d4b419'
end
def linguist_repo(oid = master_oid)
Linguist::Repository.new(rugged_repository, oid)
end end
def test_linguist_language def test_linguist_language
# assert_equal Language['Ruby'], linguist_repo.language assert_equal 'Ruby', linguist_repo.language
end end
def test_linguist_languages def test_linguist_languages
# assert linguist_repo.languages[Language['Ruby']] > 10_000 assert linguist_repo.languages['Ruby'] > 10_000
end end
def test_linguist_size def test_linguist_size
@@ -31,7 +33,18 @@ class TestRepository < Test::Unit::TestCase
assert linguist_repo.breakdown_by_file["Ruby"].include?("lib/linguist/language.rb") assert linguist_repo.breakdown_by_file["Ruby"].include?("lib/linguist/language.rb")
end end
def test_binary_override def test_incremental_stats
assert_equal repo(File.expand_path("../../samples/Nimrod", __FILE__)).language, Language["Nimrod"] old_commit = '3d7364877d6794f6cc2a86b493e893968a597332'
old_repo = linguist_repo(old_commit)
assert old_repo.languages['Ruby'] > 10_000
assert old_repo.size > 30_000
new_repo = Linguist::Repository.incremental(rugged_repository, master_oid, old_commit, old_repo.cache)
assert new_repo.languages['Ruby'] > old_repo.languages['Ruby']
assert new_repo.size > old_repo.size
assert_equal linguist_repo.cache, new_repo.cache
end end
end end