2019-01-31 01:14:18 -06:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import (
|
2019-01-31 18:36:28 -06:00
|
|
|
"bytes"
|
2019-01-31 01:14:18 -06:00
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/actions/workflow-parser/model"
|
|
|
|
)
|
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
type Error struct {
|
2019-01-31 01:14:18 -06:00
|
|
|
message string
|
2019-02-07 00:36:08 -06:00
|
|
|
Errors []*ParseError
|
2019-01-31 01:14:18 -06:00
|
|
|
Actions []*model.Action
|
|
|
|
Workflows []*model.Workflow
|
|
|
|
}
|
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
func (e *Error) Error() string {
|
2019-01-31 18:36:28 -06:00
|
|
|
buffer := bytes.NewBuffer(nil)
|
2019-02-07 00:36:08 -06:00
|
|
|
buffer.WriteString(e.message)
|
|
|
|
for _, pe := range e.Errors {
|
2019-01-31 18:36:28 -06:00
|
|
|
buffer.WriteString("\n ")
|
2019-02-07 00:36:08 -06:00
|
|
|
buffer.WriteString(pe.Error())
|
2019-01-31 18:36:28 -06:00
|
|
|
}
|
|
|
|
return buffer.String()
|
2019-01-31 01:14:18 -06:00
|
|
|
}
|
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
// FirstError searches a Configuration for the first error at or above a
|
|
|
|
// given severity level. Checking the return value against nil is a good
|
|
|
|
// way to see if the file has any errors at or above the given severity.
|
|
|
|
// A caller intending to execute the file might check for
|
|
|
|
// `errors.FirstError(parser.WARNING)`, while a caller intending to
|
|
|
|
// display the file might check for `errors.FirstError(parser.FATAL)`.
|
|
|
|
func (e *Error) FirstError(severity Severity) error {
|
|
|
|
for _, pe := range e.Errors {
|
|
|
|
if pe.Severity >= severity {
|
|
|
|
return pe
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseError represents an error identified by the parser, either syntactic
|
2019-01-31 01:14:18 -06:00
|
|
|
// (HCL) or semantic (.workflow) in nature. There are fields for location
|
|
|
|
// (File, Line, Column), severity, and base error string. The `Error()`
|
|
|
|
// function on this type concatenates whatever bits of the location are
|
|
|
|
// available with the message. The severity is only used for filtering.
|
2019-02-07 00:36:08 -06:00
|
|
|
type ParseError struct {
|
2019-01-31 01:14:18 -06:00
|
|
|
message string
|
|
|
|
Pos ErrorPos
|
|
|
|
Severity Severity
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErrorPos represents the location of an error in a user's workflow
|
|
|
|
// file(s).
|
|
|
|
type ErrorPos struct {
|
|
|
|
File string
|
|
|
|
Line int
|
|
|
|
Column int
|
|
|
|
}
|
|
|
|
|
|
|
|
// newFatal creates a new error at the FATAL level, indicating that the
|
|
|
|
// file is so broken it should not be displayed.
|
2019-02-07 00:36:08 -06:00
|
|
|
func newFatal(pos ErrorPos, format string, a ...interface{}) *ParseError {
|
|
|
|
return &ParseError{
|
2019-01-31 01:14:18 -06:00
|
|
|
message: fmt.Sprintf(format, a...),
|
|
|
|
Pos: pos,
|
|
|
|
Severity: FATAL,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// newError creates a new error at the ERROR level, indicating that the
|
|
|
|
// file can be displayed but cannot be run.
|
2019-02-07 00:36:08 -06:00
|
|
|
func newError(pos ErrorPos, format string, a ...interface{}) *ParseError {
|
|
|
|
return &ParseError{
|
2019-01-31 01:14:18 -06:00
|
|
|
message: fmt.Sprintf(format, a...),
|
|
|
|
Pos: pos,
|
|
|
|
Severity: ERROR,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// newWarning creates a new error at the WARNING level, indicating that
|
|
|
|
// the file might be runnable but might not execute as intended.
|
2019-02-07 00:36:08 -06:00
|
|
|
func newWarning(pos ErrorPos, format string, a ...interface{}) *ParseError {
|
|
|
|
return &ParseError{
|
2019-01-31 01:14:18 -06:00
|
|
|
message: fmt.Sprintf(format, a...),
|
|
|
|
Pos: pos,
|
|
|
|
Severity: WARNING,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
func (e *ParseError) Error() string {
|
2019-01-31 01:14:18 -06:00
|
|
|
var sb strings.Builder
|
|
|
|
if e.Pos.Line != 0 {
|
|
|
|
sb.WriteString("Line ") // nolint: errcheck
|
|
|
|
sb.WriteString(strconv.Itoa(e.Pos.Line)) // nolint: errcheck
|
|
|
|
sb.WriteString(": ") // nolint: errcheck
|
|
|
|
}
|
|
|
|
if sb.Len() > 0 {
|
|
|
|
sb.WriteString(e.message) // nolint: errcheck
|
|
|
|
return sb.String()
|
|
|
|
}
|
|
|
|
return e.message
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
_ = iota
|
|
|
|
|
|
|
|
// WARNING indicates a mistake that might affect correctness
|
|
|
|
WARNING
|
|
|
|
|
|
|
|
// ERROR indicates a mistake that prevents execution of any workflows in the file
|
|
|
|
ERROR
|
|
|
|
|
|
|
|
// FATAL indicates a mistake that prevents even drawing the file
|
|
|
|
FATAL
|
|
|
|
)
|
|
|
|
|
|
|
|
// Severity represents the level of an error encountered while parsing a
|
|
|
|
// workflow file. See the comments for WARNING, ERROR, and FATAL, above.
|
|
|
|
type Severity int
|
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
type errorList []*ParseError
|
2019-01-31 01:14:18 -06:00
|
|
|
|
2019-02-07 00:36:08 -06:00
|
|
|
func (a errorList) Len() int { return len(a) }
|
|
|
|
func (a errorList) Swap(i, j int) { a[i], a[j] = a[j], a[i] }
|
|
|
|
func (a errorList) Less(i, j int) bool { return a[i].Pos.Line < a[j].Pos.Line }
|
2019-01-31 01:14:18 -06:00
|
|
|
|
|
|
|
// sortErrors sorts the errors reported by the parser. Do this after
|
|
|
|
// parsing is complete. The sort is stable, so order is preserved within
|
|
|
|
// a single line: left to right, syntax errors before validation errors.
|
2019-02-07 00:36:08 -06:00
|
|
|
func (errors errorList) sort() {
|
2019-01-31 01:14:18 -06:00
|
|
|
sort.Stable(errors)
|
|
|
|
}
|