mirror of
https://github.com/alecthomas/chroma.git
synced 2025-01-12 01:22:30 +02:00
307 lines
7.3 KiB
Go
307 lines
7.3 KiB
Go
package chroma
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
"strings"
|
|
)
|
|
|
|
var (
|
|
defaultOptions = &TokeniseOptions{
|
|
State: "root",
|
|
}
|
|
)
|
|
|
|
// Config for a lexer.
|
|
type Config struct {
|
|
// Name of the lexer.
|
|
Name string
|
|
|
|
// Shortcuts for the lexer
|
|
Aliases []string
|
|
|
|
// File name globs
|
|
Filenames []string
|
|
|
|
// Secondary file name globs
|
|
AliasFilenames []string
|
|
|
|
// MIME types
|
|
MimeTypes []string
|
|
|
|
// Regex matching is case-insensitive.
|
|
CaseInsensitive bool
|
|
|
|
// Regex matches all characters.
|
|
DotAll bool
|
|
|
|
// Regex does not match across lines ($ matches EOL).
|
|
//
|
|
// Defaults to multiline.
|
|
NotMultiline bool
|
|
|
|
// Don't strip leading and trailing newlines from the input.
|
|
// DontStripNL bool
|
|
|
|
// Strip all leading and trailing whitespace from the input
|
|
// StripAll bool
|
|
|
|
// Make sure that the input does not end with a newline. This
|
|
// is required for some lexers that consume input linewise.
|
|
// DontEnsureNL bool
|
|
|
|
// If given and greater than 0, expand tabs in the input.
|
|
// TabSize int
|
|
}
|
|
|
|
// Token output to formatter.
|
|
type Token struct {
|
|
Type TokenType
|
|
Value string
|
|
}
|
|
|
|
func (t *Token) String() string { return t.Value }
|
|
func (t *Token) GoString() string { return fmt.Sprintf("Token{%s, %q}", t.Type, t.Value) }
|
|
|
|
type TokeniseOptions struct {
|
|
// State to start tokenisation in. Defaults to "root".
|
|
State string
|
|
}
|
|
|
|
type Lexer interface {
|
|
Config() *Config
|
|
Tokenise(options *TokeniseOptions, text string, out func(*Token)) error
|
|
}
|
|
|
|
type Lexers []Lexer
|
|
|
|
// Pick attempts to pick the best Lexer for a piece of source code. May return nil.
|
|
func (l Lexers) Pick(text string) Lexer {
|
|
if len(l) == 0 {
|
|
return nil
|
|
}
|
|
var picked Lexer
|
|
highest := float32(-1)
|
|
for _, lexer := range l {
|
|
if analyser, ok := lexer.(Analyser); ok {
|
|
score := analyser.AnalyseText(text)
|
|
if score > highest {
|
|
highest = score
|
|
picked = lexer
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
return picked
|
|
}
|
|
|
|
// Analyser determines if this lexer is appropriate for the given text.
|
|
type Analyser interface {
|
|
AnalyseText(text string) float32
|
|
}
|
|
|
|
type Rule struct {
|
|
Pattern string
|
|
Type Emitter
|
|
Mutator Mutator
|
|
}
|
|
|
|
// An Emitter takes group matches and returns tokens.
|
|
type Emitter interface {
|
|
// Emit tokens for the given regex groups.
|
|
Emit(groups []string, lexer Lexer, out func(*Token))
|
|
}
|
|
|
|
// EmitterFunc is a function that is an Emitter.
|
|
type EmitterFunc func(groups []string, lexer Lexer, out func(*Token))
|
|
|
|
// Emit tokens for groups.
|
|
func (e EmitterFunc) Emit(groups []string, lexer Lexer, out func(*Token)) { e(groups, lexer, out) }
|
|
|
|
// ByGroups emits a token for each matching group in the rule's regex.
|
|
func ByGroups(emitters ...Emitter) Emitter {
|
|
return EmitterFunc(func(groups []string, lexer Lexer, out func(*Token)) {
|
|
for i, group := range groups[1:] {
|
|
emitters[i].Emit([]string{group}, lexer, out)
|
|
}
|
|
return
|
|
})
|
|
}
|
|
|
|
// Using returns an Emitter that uses a given Lexer for parsing and emitting.
|
|
func Using(lexer Lexer, options *TokeniseOptions) Emitter {
|
|
return EmitterFunc(func(groups []string, _ Lexer, out func(*Token)) {
|
|
if err := lexer.Tokenise(options, groups[0], out); err != nil {
|
|
panic(err)
|
|
}
|
|
})
|
|
}
|
|
|
|
// UsingSelf is like Using, but uses the current Lexer.
|
|
func UsingSelf(state string) Emitter {
|
|
return EmitterFunc(func(groups []string, lexer Lexer, out func(*Token)) {
|
|
if err := lexer.Tokenise(&TokeniseOptions{State: state}, groups[0], out); err != nil {
|
|
panic(err)
|
|
}
|
|
})
|
|
}
|
|
|
|
// Words creates a regex that matches any of the given literal words.
|
|
func Words(words ...string) string {
|
|
for i, word := range words {
|
|
words[i] = regexp.QuoteMeta(word)
|
|
}
|
|
return `\b(?:` + strings.Join(words, `|`) + `)\b`
|
|
}
|
|
|
|
// Rules maps from state to a sequence of Rules.
|
|
type Rules map[string][]Rule
|
|
|
|
// MustNewLexer creates a new Lexer or panics.
|
|
func MustNewLexer(config *Config, rules Rules) *RegexLexer {
|
|
lexer, err := NewLexer(config, rules)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return lexer
|
|
}
|
|
|
|
// NewLexer creates a new regex-based Lexer.
|
|
//
|
|
// "rules" is a state machine transitition map. Each key is a state. Values are sets of rules
|
|
// that match input, optionally modify lexer state, and output tokens.
|
|
func NewLexer(config *Config, rules Rules) (*RegexLexer, error) {
|
|
if config == nil {
|
|
config = &Config{}
|
|
}
|
|
if _, ok := rules["root"]; !ok {
|
|
return nil, fmt.Errorf("no \"root\" state")
|
|
}
|
|
compiledRules := map[string][]CompiledRule{}
|
|
for state, rules := range rules {
|
|
for _, rule := range rules {
|
|
crule := CompiledRule{Rule: rule}
|
|
flags := ""
|
|
if !config.NotMultiline {
|
|
flags += "m"
|
|
}
|
|
if config.CaseInsensitive {
|
|
flags += "i"
|
|
}
|
|
if config.DotAll {
|
|
flags += "s"
|
|
}
|
|
re, err := regexp.Compile("^(?" + flags + ")(?:" + rule.Pattern + ")")
|
|
if err != nil {
|
|
return nil, fmt.Errorf("invalid regex %q for state %q: %s", rule.Pattern, state, err)
|
|
}
|
|
crule.Regexp = re
|
|
compiledRules[state] = append(compiledRules[state], crule)
|
|
}
|
|
}
|
|
return &RegexLexer{
|
|
config: config,
|
|
rules: compiledRules,
|
|
}, nil
|
|
}
|
|
|
|
// A CompiledRule is a Rule with a pre-compiled regex.
|
|
type CompiledRule struct {
|
|
Rule
|
|
Regexp *regexp.Regexp
|
|
}
|
|
|
|
type CompiledRules map[string][]CompiledRule
|
|
|
|
type LexerState struct {
|
|
Text string
|
|
Pos int
|
|
Rules map[string][]CompiledRule
|
|
Stack []string
|
|
State string
|
|
Rule int
|
|
// Group matches.
|
|
Groups []string
|
|
}
|
|
|
|
type RegexLexer struct {
|
|
config *Config
|
|
rules map[string][]CompiledRule
|
|
analyser func(text string) float32
|
|
}
|
|
|
|
// SetAnalyser sets the analyser function used to perform content inspection.
|
|
func (r *RegexLexer) SetAnalyser(analyser func(text string) float32) *RegexLexer {
|
|
r.analyser = analyser
|
|
return r
|
|
}
|
|
|
|
func (r *RegexLexer) AnalyseText(text string) float32 {
|
|
if r.analyser != nil {
|
|
return r.analyser(text)
|
|
}
|
|
return 0.0
|
|
}
|
|
|
|
func (r *RegexLexer) Config() *Config {
|
|
return r.config
|
|
}
|
|
|
|
func (r *RegexLexer) Tokenise(options *TokeniseOptions, text string, out func(*Token)) error {
|
|
if options == nil {
|
|
options = defaultOptions
|
|
}
|
|
state := &LexerState{
|
|
Text: text,
|
|
Stack: []string{options.State},
|
|
Rules: r.rules,
|
|
}
|
|
for state.Pos < len(text) && len(state.Stack) > 0 {
|
|
state.State = state.Stack[len(state.Stack)-1]
|
|
ruleIndex, rule, index := matchRules(state.Text[state.Pos:], state.Rules[state.State])
|
|
// fmt.Println(text[state.Pos:state.Pos+1], rule, state.Text[state.Pos:state.Pos+1])
|
|
// No match.
|
|
if index == nil {
|
|
out(&Token{Error, state.Text[state.Pos : state.Pos+1]})
|
|
state.Pos++
|
|
continue
|
|
}
|
|
state.Rule = ruleIndex
|
|
|
|
state.Groups = make([]string, len(index)/2)
|
|
for i := 0; i < len(index); i += 2 {
|
|
start := state.Pos + index[i]
|
|
end := state.Pos + index[i+1]
|
|
if start == -1 || end == -1 {
|
|
continue
|
|
}
|
|
state.Groups[i/2] = text[start:end]
|
|
}
|
|
state.Pos += index[1]
|
|
if rule.Mutator != nil {
|
|
if err := rule.Mutator.Mutate(state); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
if rule.Type != nil {
|
|
rule.Type.Emit(state.Groups, r, out)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Tokenise text using lexer, returning tokens as a slice.
|
|
func Tokenise(lexer Lexer, options *TokeniseOptions, text string) ([]*Token, error) {
|
|
out := []*Token{}
|
|
return out, lexer.Tokenise(options, text, func(token *Token) { out = append(out, token) })
|
|
}
|
|
|
|
func matchRules(text string, rules []CompiledRule) (int, CompiledRule, []int) {
|
|
for i, rule := range rules {
|
|
if index := rule.Regexp.FindStringSubmatchIndex(text); index != nil {
|
|
return i, rule, index
|
|
}
|
|
}
|
|
return 0, CompiledRule{}, nil
|
|
}
|