mirror of
https://github.com/gocsaf/csaf.git
synced 2025-12-22 05:40:11 +01:00
* Factor JSON evaluation and construction base URLs out of of checker. * Move json path matching to util. * Add csaf_aggregator (as additional command) * Improve itest workflow to checkout the branch where it is running on. resolve #105 resolve #72 Co-authored-by: tschmidtb51 <65305130+tschmidtb51@users.noreply.github.com> Co-authored-by: Bernhard Reiter <bernhard@intevation.de> Co-authored-by: Fadi Abbud <fadi.abbud@intevation.de>
148 lines
3.5 KiB
Go
148 lines
3.5 KiB
Go
// This file is Free Software under the MIT License
|
|
// without warranty, see README.md and LICENSES/MIT.txt for details.
|
|
//
|
|
// SPDX-License-Identifier: MIT
|
|
//
|
|
// SPDX-FileCopyrightText: 2021 German Federal Office for Information Security (BSI) <https://www.bsi.bund.de>
|
|
// Software-Engineering: 2021 Intevation GmbH <https://intevation.de>
|
|
|
|
package util
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/PaesslerAG/gval"
|
|
"github.com/PaesslerAG/jsonpath"
|
|
)
|
|
|
|
// ReMarshalJSON transforms data from src to dst via JSON marshalling.
|
|
func ReMarshalJSON(dst, src interface{}) error {
|
|
intermediate, err := json.Marshal(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return json.Unmarshal(intermediate, dst)
|
|
}
|
|
|
|
// PathEval is a helper to evaluate JSON paths on documents.
|
|
type PathEval struct {
|
|
builder gval.Language
|
|
exprs map[string]gval.Evaluable
|
|
}
|
|
|
|
// NewPathEval creates a new PathEval.
|
|
func NewPathEval() *PathEval {
|
|
return &PathEval{
|
|
builder: gval.Full(jsonpath.Language()),
|
|
exprs: map[string]gval.Evaluable{},
|
|
}
|
|
}
|
|
|
|
// Eval evalutes expression expr on document doc.
|
|
// Returns the result of the expression.
|
|
func (pe *PathEval) Eval(expr string, doc interface{}) (interface{}, error) {
|
|
if doc == nil {
|
|
return nil, errors.New("no document to extract data from")
|
|
}
|
|
eval := pe.exprs[expr]
|
|
if eval == nil {
|
|
var err error
|
|
if eval, err = pe.builder.NewEvaluable(expr); err != nil {
|
|
return nil, err
|
|
}
|
|
pe.exprs[expr] = eval
|
|
}
|
|
return eval(context.Background(), doc)
|
|
}
|
|
|
|
// PathEvalMatcher is a pair of an expression and an action
|
|
// when doing extractions via PathEval.Match.
|
|
type PathEvalMatcher struct {
|
|
// Expr is the expression to evaluate
|
|
Expr string
|
|
// Action is executed with the result of the match.
|
|
Action func(interface{}) error
|
|
// Optional expresses if the expression is optional.
|
|
Optional bool
|
|
}
|
|
|
|
// ReMarshalMatcher is an action to re-marshal the result to another type.
|
|
func ReMarshalMatcher(dst interface{}) func(interface{}) error {
|
|
return func(src interface{}) error {
|
|
return ReMarshalJSON(dst, src)
|
|
}
|
|
}
|
|
|
|
// BoolMatcher stores the matched result in a bool.
|
|
func BoolMatcher(dst *bool) func(interface{}) error {
|
|
return func(x interface{}) error {
|
|
b, ok := x.(bool)
|
|
if !ok {
|
|
return errors.New("not a bool")
|
|
}
|
|
*dst = b
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// StringMatcher stores the matched result in a string.
|
|
func StringMatcher(dst *string) func(interface{}) error {
|
|
return func(x interface{}) error {
|
|
s, ok := x.(string)
|
|
if !ok {
|
|
return errors.New("not a string")
|
|
}
|
|
*dst = s
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// TimeMatcher stores a time with a given format.
|
|
func TimeMatcher(dst *time.Time, format string) func(interface{}) error {
|
|
return func(x interface{}) error {
|
|
s, ok := x.(string)
|
|
if !ok {
|
|
return errors.New("not a string")
|
|
}
|
|
t, err := time.Parse(format, s)
|
|
if err != nil {
|
|
return nil
|
|
}
|
|
*dst = t
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// Extract extracts a value from a given document with a given expression/action.
|
|
func (pe *PathEval) Extract(
|
|
expr string,
|
|
action func(interface{}) error,
|
|
optional bool,
|
|
doc interface{},
|
|
) error {
|
|
optErr := func(err error) error {
|
|
if err == nil || optional {
|
|
return nil
|
|
}
|
|
return fmt.Errorf("extract failed '%s': %v", expr, err)
|
|
}
|
|
x, err := pe.Eval(expr, doc)
|
|
if err != nil {
|
|
return optErr(err)
|
|
}
|
|
return optErr(action(x))
|
|
}
|
|
|
|
// Match matches a list of PathEvalMatcher pairs against a document.
|
|
func (pe *PathEval) Match(matcher []PathEvalMatcher, doc interface{}) error {
|
|
for _, m := range matcher {
|
|
if err := pe.Extract(m.Expr, m.Action, m.Optional, doc); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|