2019-09-10 21:39:53 +00:00
|
|
|
// Package hclparse has the main API entry point for parsing both HCL native
|
|
|
|
// syntax and HCL JSON.
|
|
|
|
//
|
|
|
|
// The main HCL package also includes SimpleParse and SimpleParseFile which
|
|
|
|
// can be a simpler interface for the common case where an application just
|
|
|
|
// needs to parse a single file. The gohcl package simplifies that further
|
|
|
|
// in its SimpleDecode function, which combines hcl.SimpleParse with decoding
|
|
|
|
// into Go struct values
|
|
|
|
//
|
|
|
|
// Package hclparse, then, is useful for applications that require more fine
|
|
|
|
// control over parsing or which need to load many separate files and keep
|
|
|
|
// track of them for possible error reporting or other analysis.
|
2017-09-11 23:00:31 +00:00
|
|
|
package hclparse
|
2017-05-18 15:41:48 +00:00
|
|
|
|
|
|
|
import (
|
2017-05-30 15:03:38 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
|
2019-09-09 23:08:19 +00:00
|
|
|
"github.com/hashicorp/hcl/v2"
|
|
|
|
"github.com/hashicorp/hcl/v2/hclsyntax"
|
|
|
|
"github.com/hashicorp/hcl/v2/json"
|
2017-05-18 15:41:48 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NOTE: This is the public interface for parsing. The actual parsers are
|
|
|
|
// in other packages alongside this one, with this package just wrapping them
|
|
|
|
// to provide a unified interface for the caller across all supported formats.
|
|
|
|
|
|
|
|
// Parser is the main interface for parsing configuration files. As well as
|
|
|
|
// parsing files, a parser also retains a registry of all of the files it
|
|
|
|
// has parsed so that multiple attempts to parse the same file will return
|
|
|
|
// the same object and so the collected files can be used when printing
|
|
|
|
// diagnostics.
|
|
|
|
//
|
|
|
|
// Any diagnostics for parsing a file are only returned once on the first
|
|
|
|
// call to parse that file. Callers are expected to collect up diagnostics
|
|
|
|
// and present them together, so returning diagnostics for the same file
|
|
|
|
// multiple times would create a confusing result.
|
|
|
|
type Parser struct {
|
2017-09-11 23:40:37 +00:00
|
|
|
files map[string]*hcl.File
|
2017-05-18 15:41:48 +00:00
|
|
|
}
|
|
|
|
|
2017-06-04 15:40:49 +00:00
|
|
|
// NewParser creates a new parser, ready to parse configuration files.
|
|
|
|
func NewParser() *Parser {
|
|
|
|
return &Parser{
|
2017-09-11 23:40:37 +00:00
|
|
|
files: map[string]*hcl.File{},
|
2017-06-04 15:40:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-12 00:22:10 +00:00
|
|
|
// ParseHCL parses the given buffer (which is assumed to have been loaded from
|
2017-05-30 15:03:38 +00:00
|
|
|
// the given filename) as a native-syntax configuration file and returns the
|
2017-09-11 23:40:37 +00:00
|
|
|
// hcl.File object representing it.
|
2017-09-12 00:22:10 +00:00
|
|
|
func (p *Parser) ParseHCL(src []byte, filename string) (*hcl.File, hcl.Diagnostics) {
|
2017-05-30 15:03:38 +00:00
|
|
|
if existing := p.files[filename]; existing != nil {
|
|
|
|
return existing, nil
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
file, diags := hclsyntax.ParseConfig(src, filename, hcl.Pos{Byte: 0, Line: 1, Column: 1})
|
2017-06-04 16:02:41 +00:00
|
|
|
p.files[filename] = file
|
|
|
|
return file, diags
|
2017-05-30 15:03:38 +00:00
|
|
|
}
|
|
|
|
|
2017-09-12 00:22:10 +00:00
|
|
|
// ParseHCLFile reads the given filename and parses it as a native-syntax HCL
|
2017-05-30 15:03:38 +00:00
|
|
|
// configuration file. An error diagnostic is returned if the given file
|
|
|
|
// cannot be read.
|
2017-09-12 00:22:10 +00:00
|
|
|
func (p *Parser) ParseHCLFile(filename string) (*hcl.File, hcl.Diagnostics) {
|
2017-05-30 15:03:38 +00:00
|
|
|
if existing := p.files[filename]; existing != nil {
|
|
|
|
return existing, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
src, err := ioutil.ReadFile(filename)
|
|
|
|
if err != nil {
|
2017-09-11 23:40:37 +00:00
|
|
|
return nil, hcl.Diagnostics{
|
2017-05-30 15:03:38 +00:00
|
|
|
{
|
2017-09-11 23:40:37 +00:00
|
|
|
Severity: hcl.DiagError,
|
2017-05-30 15:03:38 +00:00
|
|
|
Summary: "Failed to read file",
|
|
|
|
Detail: fmt.Sprintf("The configuration file %q could not be read.", filename),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-12 00:22:10 +00:00
|
|
|
return p.ParseHCL(src, filename)
|
2017-05-30 15:03:38 +00:00
|
|
|
}
|
|
|
|
|
2017-05-18 15:41:48 +00:00
|
|
|
// ParseJSON parses the given JSON buffer (which is assumed to have been loaded
|
2017-09-11 23:40:37 +00:00
|
|
|
// from the given filename) and returns the hcl.File object representing it.
|
|
|
|
func (p *Parser) ParseJSON(src []byte, filename string) (*hcl.File, hcl.Diagnostics) {
|
2017-05-18 15:41:48 +00:00
|
|
|
if existing := p.files[filename]; existing != nil {
|
|
|
|
return existing, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
file, diags := json.Parse(src, filename)
|
|
|
|
p.files[filename] = file
|
|
|
|
return file, diags
|
|
|
|
}
|
|
|
|
|
|
|
|
// ParseJSONFile reads the given filename and parses it as JSON, similarly to
|
|
|
|
// ParseJSON. An error diagnostic is returned if the given file cannot be read.
|
2017-09-11 23:40:37 +00:00
|
|
|
func (p *Parser) ParseJSONFile(filename string) (*hcl.File, hcl.Diagnostics) {
|
2017-05-18 15:41:48 +00:00
|
|
|
if existing := p.files[filename]; existing != nil {
|
|
|
|
return existing, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
file, diags := json.ParseFile(filename)
|
2017-05-22 05:34:23 +00:00
|
|
|
p.files[filename] = file
|
2017-05-21 23:13:04 +00:00
|
|
|
return file, diags
|
2017-05-21 20:04:33 +00:00
|
|
|
}
|
|
|
|
|
2017-05-18 15:41:48 +00:00
|
|
|
// AddFile allows a caller to record in a parser a file that was parsed some
|
|
|
|
// other way, thus allowing it to be included in the registry of sources.
|
2017-09-11 23:40:37 +00:00
|
|
|
func (p *Parser) AddFile(filename string, file *hcl.File) {
|
2017-05-18 15:41:48 +00:00
|
|
|
p.files[filename] = file
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sources returns a map from filenames to the raw source code that was
|
|
|
|
// read from them. This is intended to be used, for example, to print
|
|
|
|
// diagnostics with contextual information.
|
|
|
|
//
|
|
|
|
// The arrays underlying the returned slices should not be modified.
|
|
|
|
func (p *Parser) Sources() map[string][]byte {
|
|
|
|
ret := make(map[string][]byte)
|
|
|
|
for fn, f := range p.files {
|
|
|
|
ret[fn] = f.Bytes
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
2017-05-20 01:56:39 +00:00
|
|
|
|
|
|
|
// Files returns a map from filenames to the File objects produced from them.
|
|
|
|
// This is intended to be used, for example, to print diagnostics with
|
|
|
|
// contextual information.
|
|
|
|
//
|
|
|
|
// The returned map and all of the objects it refers to directly or indirectly
|
|
|
|
// must not be modified.
|
2017-09-11 23:40:37 +00:00
|
|
|
func (p *Parser) Files() map[string]*hcl.File {
|
2017-05-20 01:56:39 +00:00
|
|
|
return p.files
|
|
|
|
}
|