1
0
mirror of https://github.com/alecthomas/chroma.git synced 2025-03-19 21:10:15 +02:00
chroma/lexers/api.go

117 lines
2.5 KiB
Go
Raw Normal View History

2017-06-02 00:17:21 +10:00
package lexers
import (
"path/filepath"
2017-09-19 10:47:22 +10:00
"sort"
2017-06-02 00:17:21 +10:00
"github.com/danwakefield/fnmatch"
"github.com/alecthomas/chroma"
)
// Registry of Lexers.
var Registry = struct {
Lexers chroma.Lexers
byName map[string]chroma.Lexer
byAlias map[string]chroma.Lexer
}{
byName: map[string]chroma.Lexer{},
byAlias: map[string]chroma.Lexer{},
2017-06-02 00:17:21 +10:00
}
// Names of all lexers, optionally including aliases.
func Names(withAliases bool) []string {
2017-06-02 00:17:21 +10:00
out := []string{}
for _, lexer := range Registry.Lexers {
2017-06-02 00:17:21 +10:00
config := lexer.Config()
out = append(out, config.Name)
if withAliases {
out = append(out, config.Aliases...)
}
}
2017-09-19 10:47:22 +10:00
sort.Strings(out)
2017-06-02 00:17:21 +10:00
return out
}
// Get a Lexer by name.
func Get(name string) chroma.Lexer {
if lexer := Registry.byName[name]; lexer != nil {
return lexer
}
return Registry.byAlias[name]
2017-06-02 00:17:21 +10:00
}
2017-09-19 11:52:23 +10:00
// MatchMimeType attempts to find a lexer for the given MIME type.
func MatchMimeType(mimeType string) chroma.Lexer {
for _, l := range Registry.Lexers {
for _, lmt := range l.Config().MimeTypes {
if mimeType == lmt {
return l
}
}
}
return nil
}
// Match returns the first lexer matching filename.
2017-06-07 10:27:10 +10:00
func Match(filename string) chroma.Lexer {
filename = filepath.Base(filename)
// First, try primary filename matches.
for _, lexer := range Registry.Lexers {
2017-06-02 00:17:21 +10:00
config := lexer.Config()
for _, glob := range config.Filenames {
if fnmatch.Match(glob, filename, 0) {
2017-06-07 10:27:10 +10:00
return lexer
2017-06-02 00:17:21 +10:00
}
}
}
// Next, try filename aliases.
for _, lexer := range Registry.Lexers {
config := lexer.Config()
for _, glob := range config.AliasFilenames {
if fnmatch.Match(glob, filename, 0) {
return lexer
}
}
}
2017-06-07 10:27:10 +10:00
return nil
2017-06-02 00:17:21 +10:00
}
2017-06-07 10:27:10 +10:00
// Analyse text content and return the "best" lexer..
func Analyse(text string) chroma.Lexer {
var picked chroma.Lexer
highest := float32(0.0)
for _, lexer := range Registry.Lexers {
if analyser, ok := lexer.(chroma.Analyser); ok {
weight := analyser.AnalyseText(text)
if weight > highest {
picked = lexer
highest = weight
}
}
2017-06-02 00:17:21 +10:00
}
2017-06-07 10:27:10 +10:00
return picked
}
// Register a Lexer with the global registry.
func Register(lexer chroma.Lexer) chroma.Lexer {
2017-06-02 00:17:21 +10:00
config := lexer.Config()
Registry.byName[config.Name] = lexer
for _, alias := range config.Aliases {
Registry.byAlias[alias] = lexer
2017-06-02 00:17:21 +10:00
}
Registry.Lexers = append(Registry.Lexers, lexer)
return lexer
}
// Fallback lexer if no other is found.
var Fallback chroma.Lexer = chroma.MustNewLexer(&chroma.Config{
Name: "fallback",
Filenames: []string{"*"},
}, chroma.Rules{
"root": []chroma.Rule{
{`.+`, chroma.Text, nil},
{`\n`, chroma.Text, nil},
},
})