1
0
Fork 0
mirror of https://github.com/Luzifer/sii.git synced 2025-01-04 13:46:04 +00:00
sii/parser.go

143 lines
2.9 KiB
Go
Raw Normal View History

package sii
import (
"bufio"
"bytes"
"io"
"reflect"
"regexp"
"strings"
"github.com/pkg/errors"
)
var blockStartRegex = regexp.MustCompile(`^([^\s:]+)\s?:\s?([^\s]+)(?:\s?\{)?$`)
func ParseSIIPlainFile(r io.Reader) (*Unit, error) {
var (
blockContent []byte
blockName string
blockClass string
inBlock = false
inComment = false
inUnit = false
scanner = bufio.NewScanner(r)
unit = &Unit{}
)
for scanner.Scan() {
var line = strings.TrimSpace(scanner.Text())
switch {
case line == "{":
if !inUnit {
inUnit = true
continue
}
if !inBlock {
if blockClass == "" || blockName == "" {
return nil, errors.New("Unpexpected block open without unit class / name")
}
inBlock = true
continue
}
return nil, errors.New("Unexpected opening braces")
case line == "}":
if inBlock {
if err := processBlock(unit, blockClass, blockName, bytes.Trim(blockContent, "\n")); err != nil {
return nil, errors.Wrap(err, "Unable to process block")
}
inBlock = false
blockClass = ""
blockContent = nil
blockName = ""
continue
}
if inUnit {
inUnit = false
continue
}
return nil, errors.New("Unexpected closing braces")
case blockStartRegex.MatchString(line) && !inBlock:
if !inUnit {
return nil, errors.New("Unexpected block start outside unit")
}
groups := blockStartRegex.FindStringSubmatch(line)
blockClass = groups[1]
blockName = groups[2]
if strings.HasSuffix(line, `{`) {
inBlock = true
}
case (strings.HasPrefix(line, `\*`) && strings.HasSuffix(line, `*\`)) || strings.HasPrefix(line, `#`) || strings.HasPrefix(line, `//`):
// one-line-comment, just drop
case strings.HasPrefix(line, `/*`):
inComment = true
case strings.HasSuffix(line, `*/`):
inComment = false
case strings.HasPrefix(line, `@include`):
return nil, errors.New("File uses includes (unsupported feature)")
default:
if inComment {
// Inside multi-line-comment, just drop
continue
}
if !inBlock {
// Outside block, drop line
continue
}
// Append line to block content
blockContent = bytes.Join([][]byte{
blockContent,
scanner.Bytes(),
}, []byte{'\n'})
}
}
if scanner.Err() != nil {
return nil, errors.Wrap(scanner.Err(), "Unable to scan file")
}
return unit, nil
}
func processBlock(unit *Unit, blockClass, blockName string, blockContent []byte) error {
block := getBlockInstance(blockClass)
block.Init(blockClass, blockName)
var err error
if reflect.TypeOf(block).Implements(reflect.TypeOf((*Unmarshaler)(nil)).Elem()) {
err = block.(Unmarshaler).UnmarshalSII(blockContent)
} else {
err = genericUnmarshal(blockContent, block, unit)
}
if err != nil {
return errors.Wrap(err, "Unable to unmarshal block content")
}
unit.Entries = append(unit.Entries, block)
return nil
}