2021-04-08 18:46:17 -04:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"context"
|
|
|
|
"encoding/json"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
2021-09-02 11:09:48 -04:00
|
|
|
"github.com/rs/zerolog"
|
2021-04-08 18:46:17 -04:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
2021-09-07 16:33:30 -04:00
|
|
|
"os/signal"
|
2021-09-02 11:09:48 -04:00
|
|
|
"path/filepath"
|
2021-04-08 18:46:17 -04:00
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/MontFerret/ferret"
|
|
|
|
"github.com/MontFerret/ferret/pkg/drivers/cdp"
|
|
|
|
"github.com/MontFerret/ferret/pkg/drivers/http"
|
|
|
|
"github.com/MontFerret/ferret/pkg/runtime"
|
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/core"
|
2021-09-02 11:09:48 -04:00
|
|
|
"github.com/MontFerret/ferret/pkg/runtime/logging"
|
2021-04-08 18:46:17 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
type Params []string
|
|
|
|
|
|
|
|
func (p *Params) String() string {
|
|
|
|
return "[" + strings.Join(*p, ",") + "]"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Params) Set(value string) error {
|
|
|
|
*p = append(*p, value)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (p *Params) ToMap() (map[string]interface{}, error) {
|
|
|
|
res := make(map[string]interface{})
|
|
|
|
|
|
|
|
for _, entry := range *p {
|
|
|
|
pair := strings.SplitN(entry, ":", 2)
|
|
|
|
|
|
|
|
if len(pair) < 2 {
|
|
|
|
return nil, core.Error(core.ErrInvalidArgument, entry)
|
|
|
|
}
|
|
|
|
|
|
|
|
var value interface{}
|
|
|
|
key := pair[0]
|
|
|
|
|
|
|
|
err := json.Unmarshal([]byte(pair[1]), &value)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(pair[1])
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
res[key] = value
|
|
|
|
}
|
|
|
|
|
|
|
|
return res, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
conn = flag.String(
|
|
|
|
"cdp",
|
|
|
|
"",
|
|
|
|
"set CDP address",
|
|
|
|
)
|
2021-09-02 11:09:48 -04:00
|
|
|
|
|
|
|
logLevel = flag.String(
|
|
|
|
"log-level",
|
|
|
|
logging.ErrorLevel.String(),
|
|
|
|
"log level",
|
|
|
|
)
|
2021-04-08 18:46:17 -04:00
|
|
|
)
|
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
var logger zerolog.Logger
|
|
|
|
|
2021-04-08 18:46:17 -04:00
|
|
|
func main() {
|
|
|
|
var params Params
|
|
|
|
|
|
|
|
flag.Var(
|
|
|
|
¶ms,
|
|
|
|
"param",
|
|
|
|
`query parameter (--param=foo:\"bar\", --param=id:1)`,
|
|
|
|
)
|
|
|
|
|
|
|
|
flag.Parse()
|
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
console := zerolog.ConsoleWriter{
|
|
|
|
Out: os.Stderr,
|
|
|
|
TimeFormat: "15:04:05.999",
|
|
|
|
}
|
|
|
|
logger = zerolog.New(console).
|
|
|
|
Level(zerolog.Level(logging.MustParseLevel(*logLevel))).
|
|
|
|
With().
|
|
|
|
Timestamp().
|
|
|
|
Logger()
|
2021-04-08 18:46:17 -04:00
|
|
|
|
|
|
|
stat, _ := os.Stdin.Stat()
|
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
var query string
|
|
|
|
var files []string
|
|
|
|
|
2021-04-08 18:46:17 -04:00
|
|
|
if (stat.Mode() & os.ModeCharDevice) == 0 {
|
|
|
|
// check whether the app is getting a query via standard input
|
|
|
|
std := bufio.NewReader(os.Stdin)
|
|
|
|
|
|
|
|
b, err := ioutil.ReadAll(std)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
query = string(b)
|
|
|
|
} else if flag.NArg() > 0 {
|
2021-09-02 11:09:48 -04:00
|
|
|
files = flag.Args()
|
2021-04-08 18:46:17 -04:00
|
|
|
} else {
|
|
|
|
fmt.Println(flag.NArg())
|
2021-09-02 11:09:48 -04:00
|
|
|
fmt.Println("File or input stream are required")
|
2021-04-08 18:46:17 -04:00
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
p, err := params.ToMap()
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
engine := ferret.New()
|
|
|
|
_ = engine.Drivers().Register(http.NewDriver())
|
|
|
|
_ = engine.Drivers().Register(cdp.NewDriver(cdp.WithAddress(*conn)))
|
|
|
|
|
|
|
|
opts := []runtime.Option{
|
|
|
|
runtime.WithParams(p),
|
|
|
|
runtime.WithLog(console),
|
|
|
|
runtime.WithLogLevel(logging.MustParseLevel(*logLevel)),
|
|
|
|
}
|
|
|
|
|
2021-09-07 16:33:30 -04:00
|
|
|
c := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(c, os.Interrupt)
|
|
|
|
signal.Notify(c, os.Kill)
|
|
|
|
|
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
|
|
|
|
go func() {
|
|
|
|
for {
|
|
|
|
<-c
|
|
|
|
cancel()
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
if query != "" {
|
2021-09-07 16:33:30 -04:00
|
|
|
err = execQuery(ctx, engine, opts, query)
|
2021-09-02 11:09:48 -04:00
|
|
|
} else {
|
2021-09-07 16:33:30 -04:00
|
|
|
err = execFiles(ctx, engine, opts, files)
|
2021-09-02 11:09:48 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
2021-04-08 18:46:17 -04:00
|
|
|
fmt.Println(err)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-07 16:33:30 -04:00
|
|
|
func execFiles(ctx context.Context, engine *ferret.Instance, opts []runtime.Option, files []string) error {
|
2021-09-02 11:09:48 -04:00
|
|
|
errList := make([]error, 0, len(files))
|
2021-04-08 18:46:17 -04:00
|
|
|
|
2021-09-02 11:09:48 -04:00
|
|
|
for _, path := range files {
|
|
|
|
log := logger.With().Str("path", path).Logger()
|
|
|
|
log.Debug().Msg("checking path...")
|
|
|
|
|
|
|
|
info, err := os.Stat(path)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Err(err).Msg("failed to get path info")
|
|
|
|
|
|
|
|
errList = append(errList, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if info.IsDir() {
|
|
|
|
log.Debug().Msg("path points to a directory. retrieving list of files...")
|
|
|
|
|
|
|
|
fileInfos, err := ioutil.ReadDir(path)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Err(err).Msg("failed to retrieve list of files")
|
|
|
|
|
|
|
|
errList = append(errList, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Int("size", len(fileInfos)).Msg("retrieved list of files. starting to iterate...")
|
|
|
|
|
|
|
|
dirFiles := make([]string, 0, len(fileInfos))
|
|
|
|
|
|
|
|
for _, info := range fileInfos {
|
|
|
|
if filepath.Ext(info.Name()) == ".fql" {
|
|
|
|
dirFiles = append(dirFiles, filepath.Join(path, info.Name()))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(dirFiles) > 0 {
|
2021-09-07 16:33:30 -04:00
|
|
|
if err := execFiles(ctx, engine, opts, dirFiles); err != nil {
|
2021-09-02 11:09:48 -04:00
|
|
|
log.Debug().Err(err).Msg("failed to execute files")
|
|
|
|
|
|
|
|
errList = append(errList, err)
|
|
|
|
} else {
|
|
|
|
log.Debug().Int("size", len(fileInfos)).Err(err).Msg("successfully executed files")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
log.Debug().Int("size", len(fileInfos)).Err(err).Msg("no FQL files found")
|
|
|
|
}
|
|
|
|
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Msg("path points to a file. starting to read content")
|
|
|
|
|
|
|
|
out, err := ioutil.ReadFile(path)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Err(err).Msg("failed to read content")
|
|
|
|
|
|
|
|
errList = append(errList, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Msg("successfully read file")
|
|
|
|
log.Debug().Msg("executing file...")
|
2021-09-07 16:33:30 -04:00
|
|
|
err = execQuery(ctx, engine, opts, string(out))
|
2021-09-02 11:09:48 -04:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Debug().Err(err).Msg("failed to execute file")
|
|
|
|
|
|
|
|
errList = append(errList, err)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug().Msg("successfully executed file")
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(errList) > 0 {
|
|
|
|
if len(errList) == len(files) {
|
|
|
|
logger.Debug().Errs("errors", errList).Msg("failed to execute file(s)")
|
|
|
|
} else {
|
|
|
|
logger.Debug().Errs("errors", errList).Msg("executed with errors")
|
|
|
|
}
|
|
|
|
|
|
|
|
return core.Errors(errList...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2021-04-08 18:46:17 -04:00
|
|
|
|
2021-09-07 16:33:30 -04:00
|
|
|
func execQuery(ctx context.Context, engine *ferret.Instance, opts []runtime.Option, query string) error {
|
|
|
|
out, err := engine.Exec(ctx, query, opts...)
|
2021-04-08 18:46:17 -04:00
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
fmt.Println(string(out))
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|