You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
ci_validator/gitlabcivalidator/parser.go

161 lines
4.7 KiB
Go

package gitlabcivalidator
import (
"fmt"
"gitflame.ru/CI_VLDR/errors"
"gopkg.in/yaml.v3"
)
// ParseGitLabCIConfig parses the YAML and returns the GitLabCIConfig structure.
func ParseGitLabCIConfig(data []byte, ve *errors.ValidationError) (*GitLabCIConfig, error) {
var config GitLabCIConfig
var root yaml.Node
// Parse YAML into the root node
err := yaml.Unmarshal(data, &root)
if err != nil {
return nil, fmt.Errorf("YAML syntax error: %w", err)
}
// If root is empty, the YAML file is empty or invalid
if len(root.Content) == 0 {
return nil, fmt.Errorf(".gitlab-ci.yml file is empty or does not contain valid data")
}
// Expect the root node to be a YAML document
if root.Kind != yaml.DocumentNode || len(root.Content) == 0 {
return nil, fmt.Errorf("invalid YAML format: expected a root object")
}
// The main object is the first child
mainNode := root.Content[0]
// Decode the root object into the configuration structure
err = mainNode.Decode(&config)
if err != nil {
return nil, fmt.Errorf("failed to decode GitLab CI configuration: %w", err)
}
// Manually parse the 'jobs' node
for i := 0; i < len(mainNode.Content)-1; i += 2 {
keyNode := mainNode.Content[i]
valNode := mainNode.Content[i+1]
if keyNode.Value == "jobs" && valNode.Kind == yaml.MappingNode {
// 'jobs' found, parsing manually
config.Jobs = make(map[string]*Job)
for j := 0; j < len(valNode.Content)-1; j += 2 {
jobKeyNode := valNode.Content[j] // job name
jobValNode := valNode.Content[j+1] // job definition (mapping or otherwise)
// Create a new job with its name and positional info.
job := &Job{
Name: jobKeyNode.Value,
Line: jobKeyNode.Line,
Column: jobKeyNode.Column,
DependencyLines: make(map[string]struct{ Line, Column int }),
}
// If the job value is a mapping node, decode its content.
if jobValNode.Kind == yaml.MappingNode {
err := jobValNode.Decode(job)
if err != nil {
return nil, fmt.Errorf("failed to decode job '%s': %w", job.Name, err)
}
validateJobMapping(jobValNode, ve)
// Process inner keys (dependencies, needs, only, etc.)
for d := 0; d < len(jobValNode.Content)-1; d += 2 {
depKey := jobValNode.Content[d]
depVal := jobValNode.Content[d+1]
if depKey.Value == "dependencies" && depVal.Kind == yaml.SequenceNode {
for k := 0; k < len(depVal.Content); k++ {
depNode := depVal.Content[k]
job.Dependencies = append(job.Dependencies, depNode.Value)
job.DependencyLines[depNode.Value] = struct {
Line int
Column int
}{depNode.Line, depNode.Column}
}
}
if depKey.Value == "needs" && depVal.Kind == yaml.SequenceNode {
job.NeedLines = make(map[string]struct {
Line int
Column int
})
for k := 0; k < len(depVal.Content); k++ {
needNode := depVal.Content[k]
job.Needs = append(job.Needs, needNode.Value)
job.NeedLines[needNode.Value] = struct {
Line int
Column int
}{needNode.Line, needNode.Column}
}
}
if depKey.Value == "only" {
if depVal.Kind == yaml.ScalarNode {
job.Only = depVal.Value
} else if depVal.Kind == yaml.SequenceNode {
var arr []string
for k := 0; k < len(depVal.Content); k++ {
arr = append(arr, depVal.Content[k].Value)
}
job.Only = arr
}
job.OnlyLine = depKey.Line
job.OnlyColumn = depKey.Column
}
}
}
// If jobValNode is not a mapping, leave the job with default (nil) fields.
// Now add the job to the configuration regardless.
config.Jobs[job.Name] = job
}
}
}
return &config, nil
}
// findUnknownRootKeysWithSpan scans the mainNode for unknown keys and returns their information.
func findUnknownRootKeysWithSpan(mainNode *yaml.Node) []errors.UnknownKeyError {
allowedKeys := map[string]struct{}{
"default": {},
"include": {},
"before_script": {},
"image": {},
"services": {},
"after_script": {},
"variables": {},
"stages": {},
"cache": {},
"workflow": {},
"jobs": {},
"changes": {},
"paths": {},
}
var unknowns []errors.UnknownKeyError
if mainNode.Kind != yaml.MappingNode {
return unknowns
}
for i := 0; i < len(mainNode.Content)-1; i += 2 {
keyNode := mainNode.Content[i]
if _, ok := allowedKeys[keyNode.Value]; !ok {
endLine, endCol := keyNode.Line, keyNode.Column+len(keyNode.Value)
unknowns = append(unknowns, errors.UnknownKeyError{
Key: keyNode.Value,
Line: keyNode.Line,
Column: keyNode.Column,
EndLine: endLine,
EndColumn: endCol,
})
}
}
return unknowns
}