mirror of
				https://github.com/prometheus/prometheus.git
				synced 2025-11-04 02:11:01 +01:00 
			
		
		
		
	- Move CONTRIBUTORS.md to the more common AUTHORS. - Added the required NOTICE file. - Changed "Prometheus Team" to "The Prometheus Authors". - Reverted the erroneous changes to the Apache License.
		
			
				
	
	
		
			165 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
// Copyright 2013 The Prometheus Authors
 | 
						|
// Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
// you may not use this file except in compliance with the License.
 | 
						|
// You may obtain a copy of the License at
 | 
						|
//
 | 
						|
// http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
//
 | 
						|
// Unless required by applicable law or agreed to in writing, software
 | 
						|
// distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
// See the License for the specific language governing permissions and
 | 
						|
// limitations under the License.
 | 
						|
 | 
						|
package rules
 | 
						|
 | 
						|
import (
 | 
						|
	"bufio"
 | 
						|
	"errors"
 | 
						|
	"fmt"
 | 
						|
	"io"
 | 
						|
	"os"
 | 
						|
	"strings"
 | 
						|
 | 
						|
	"github.com/golang/glog"
 | 
						|
 | 
						|
	"github.com/prometheus/prometheus/rules/ast"
 | 
						|
)
 | 
						|
 | 
						|
// RulesLexer is the lexer for rule expressions.
 | 
						|
type RulesLexer struct {
 | 
						|
	// Errors encountered during parsing.
 | 
						|
	errors []string
 | 
						|
	// Dummy token to simulate multiple start symbols (see below).
 | 
						|
	startToken int
 | 
						|
	// Parsed full rules.
 | 
						|
	parsedRules []Rule
 | 
						|
	// Parsed single expression.
 | 
						|
	parsedExpr ast.Node
 | 
						|
 | 
						|
	// Current character.
 | 
						|
	current byte
 | 
						|
	// Current token buffer.
 | 
						|
	buf []byte
 | 
						|
	// Input text.
 | 
						|
	src *bufio.Reader
 | 
						|
	// Whether we have a current char.
 | 
						|
	empty bool
 | 
						|
 | 
						|
	// Current input line.
 | 
						|
	line int
 | 
						|
	// Current character position within the current input line.
 | 
						|
	pos int
 | 
						|
}
 | 
						|
 | 
						|
func (lexer *RulesLexer) Error(errorStr string) {
 | 
						|
	err := fmt.Sprintf("Error parsing rules at line %v, char %v: %v", lexer.line, lexer.pos, errorStr)
 | 
						|
	lexer.errors = append(lexer.errors, err)
 | 
						|
}
 | 
						|
 | 
						|
func (lexer *RulesLexer) getChar() byte {
 | 
						|
	if lexer.current != 0 {
 | 
						|
		lexer.buf = append(lexer.buf, lexer.current)
 | 
						|
	}
 | 
						|
	lexer.current = 0
 | 
						|
	if b, err := lexer.src.ReadByte(); err == nil {
 | 
						|
		if b == '\n' {
 | 
						|
			lexer.line++
 | 
						|
			lexer.pos = 0
 | 
						|
		} else {
 | 
						|
			lexer.pos++
 | 
						|
		}
 | 
						|
		lexer.current = b
 | 
						|
	} else if err != io.EOF {
 | 
						|
		glog.Fatal(err)
 | 
						|
	}
 | 
						|
	return lexer.current
 | 
						|
}
 | 
						|
 | 
						|
func (lexer *RulesLexer) token() string {
 | 
						|
	return string(lexer.buf)
 | 
						|
}
 | 
						|
 | 
						|
func newRulesLexer(src io.Reader, singleExpr bool) *RulesLexer {
 | 
						|
	lexer := &RulesLexer{
 | 
						|
		startToken: START_RULES,
 | 
						|
		src:        bufio.NewReader(src),
 | 
						|
		pos:        1,
 | 
						|
		line:       1,
 | 
						|
	}
 | 
						|
 | 
						|
	if singleExpr {
 | 
						|
		lexer.startToken = START_EXPRESSION
 | 
						|
	}
 | 
						|
	lexer.getChar()
 | 
						|
	return lexer
 | 
						|
}
 | 
						|
 | 
						|
func lexAndParse(rulesReader io.Reader, singleExpr bool) (*RulesLexer, error) {
 | 
						|
	lexer := newRulesLexer(rulesReader, singleExpr)
 | 
						|
	ret := yyParse(lexer)
 | 
						|
	if ret != 0 && len(lexer.errors) == 0 {
 | 
						|
		lexer.Error("unknown parser error")
 | 
						|
	}
 | 
						|
 | 
						|
	if len(lexer.errors) > 0 {
 | 
						|
		err := errors.New(strings.Join(lexer.errors, "\n"))
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return lexer, nil
 | 
						|
}
 | 
						|
 | 
						|
// LoadRulesFromReader parses rules from the provided reader and returns them.
 | 
						|
func LoadRulesFromReader(rulesReader io.Reader) ([]Rule, error) {
 | 
						|
	lexer, err := lexAndParse(rulesReader, false)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return lexer.parsedRules, err
 | 
						|
}
 | 
						|
 | 
						|
// LoadRulesFromString parses rules from the provided string returns them.
 | 
						|
func LoadRulesFromString(rulesString string) ([]Rule, error) {
 | 
						|
	rulesReader := strings.NewReader(rulesString)
 | 
						|
	return LoadRulesFromReader(rulesReader)
 | 
						|
}
 | 
						|
 | 
						|
// LoadRulesFromFile parses rules from the file of the provided name and returns
 | 
						|
// them.
 | 
						|
func LoadRulesFromFile(fileName string) ([]Rule, error) {
 | 
						|
	rulesReader, err := os.Open(fileName)
 | 
						|
	if err != nil {
 | 
						|
		return []Rule{}, err
 | 
						|
	}
 | 
						|
	defer rulesReader.Close()
 | 
						|
	return LoadRulesFromReader(rulesReader)
 | 
						|
}
 | 
						|
 | 
						|
// LoadExprFromReader parses a single expression from the provided reader and
 | 
						|
// returns it as an AST node.
 | 
						|
func LoadExprFromReader(exprReader io.Reader) (ast.Node, error) {
 | 
						|
	lexer, err := lexAndParse(exprReader, true)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	return lexer.parsedExpr, err
 | 
						|
}
 | 
						|
 | 
						|
// LoadExprFromString parses a single expression from the provided string and
 | 
						|
// returns it as an AST node.
 | 
						|
func LoadExprFromString(exprString string) (ast.Node, error) {
 | 
						|
	exprReader := strings.NewReader(exprString)
 | 
						|
	return LoadExprFromReader(exprReader)
 | 
						|
}
 | 
						|
 | 
						|
// LoadExprFromFile parses a single expression from the file of the provided
 | 
						|
// name and returns it as an AST node.
 | 
						|
func LoadExprFromFile(fileName string) (ast.Node, error) {
 | 
						|
	exprReader, err := os.Open(fileName)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	defer exprReader.Close()
 | 
						|
	return LoadExprFromReader(exprReader)
 | 
						|
}
 |