package operation import ( "bytes" "encoding/json" "fmt" "io" "log" "os/exec" "github.com/aziis98/cabret" ) func init() { registerType("program", &Program{}) } // Program is a [cabret.ItemOperation] that passes the incoming item as input to the given command. // Options are the following: // // command: // io: # optional, by default is "raw" // // The io format can be one of the following // // - "raw" will just pass the item data to the program // - "json" will pass the whole item as JSON to the given program, this should be useful for // making external plugins compatible with cabret. type Program struct { IOFormat string ShellCommand string } type Format interface { Input(item cabret.Content) (stdin io.Reader, err error) Output(item cabret.Content, stdout io.Reader) (*cabret.Content, error) } var ioProgramFormats = map[string]Format{} func init() { ioProgramFormats["json"] = JsonFormat{} ioProgramFormats["raw"] = RawFormat{} } type JsonFormat struct{} func (JsonFormat) Input(item cabret.Content) (stdin io.Reader, err error) { buf := &bytes.Buffer{} if err := json.NewEncoder(buf).Encode(map[string]any{ "type": item.Type, "metadata": item.Metadata, "data": item.Data, }); err != nil { return nil, err } return buf, nil } func (JsonFormat) Output(item cabret.Content, stdout io.Reader) (*cabret.Content, error) { var result struct { Type string `json:"type"` Metadata map[string]any `json:"metadata"` Data string `json:"data"` } if err := json.NewDecoder(stdout).Decode(&result); err != nil { return nil, err } return &cabret.Content{ Type: result.Type, Metadata: result.Metadata, Data: []byte(result.Data), }, nil } type RawFormat struct{} func (RawFormat) Input(item cabret.Content) (stdin io.Reader, err error) { return bytes.NewReader(item.Data), nil } func (RawFormat) Output(item cabret.Content, stdout io.Reader) (*cabret.Content, error) { data, err := io.ReadAll(stdout) if err != nil { return nil, err } item.Data = data return &item, nil } func (op *Program) Configure(options map[string]any) error { var err error op.IOFormat, err = getKey(options, "io", "raw") if err != nil { return err } op.ShellCommand, err = getKey[string](options, "command") if err != nil { return err } return nil } var _ cabret.ItemOperation = &Program{} func (op *Program) ProcessItem(ctx *cabret.Context, item cabret.Content) (*cabret.Content, error) { ioFmt, ok := ioProgramFormats[op.IOFormat] if !ok { return nil, fmt.Errorf(`unknown io format "%s"`, op.IOFormat) } r, err := ioFmt.Input(item) if err != nil { return nil, err } log.Printf(`[operation.Program] running external program "%s" with "%s" input`, op.ShellCommand, op.IOFormat) cmd := exec.Command("sh", "-c", op.ShellCommand) cmd.Stdin = r var buf bytes.Buffer cmd.Stdout = &buf if err := cmd.Run(); err != nil { return nil, err } return ioFmt.Output(item, &buf) }