tiki/plugin/parser.go
2026-01-27 18:53:53 -05:00

149 lines
3.9 KiB
Go

package plugin
import (
"fmt"
"github.com/gdamore/tcell/v2"
"gopkg.in/yaml.v3"
"github.com/boolean-maybe/tiki/plugin/filter"
)
// parsePluginConfig parses a pluginFileConfig into a Plugin
func parsePluginConfig(cfg pluginFileConfig, source string) (Plugin, error) {
// Common fields
// Use ColorDefault as sentinel so views can detect "not specified" and use theme-appropriate colors
fg := parseColor(cfg.Foreground, tcell.ColorDefault)
bg := parseColor(cfg.Background, tcell.ColorDefault)
key, r, mod, err := parseKey(cfg.Key)
if err != nil {
return nil, fmt.Errorf("plugin %q (%s): parsing key: %w", cfg.Name, source, err)
}
pluginType := cfg.Type
if pluginType == "" {
pluginType = "tiki"
}
base := BasePlugin{
Name: cfg.Name,
Key: key,
Rune: r,
Modifier: mod,
Foreground: fg,
Background: bg,
FilePath: source,
Type: pluginType,
ConfigIndex: -1, // default, will be set by caller if needed
}
switch pluginType {
case "doki":
// Strict validation for Doki
if cfg.Filter != "" {
return nil, fmt.Errorf("doki plugin cannot have 'filter'")
}
if cfg.Sort != "" {
return nil, fmt.Errorf("doki plugin cannot have 'sort'")
}
if cfg.View != "" {
return nil, fmt.Errorf("doki plugin cannot have 'view'")
}
if len(cfg.Panes) > 0 {
return nil, fmt.Errorf("doki plugin cannot have 'panes'")
}
if cfg.Fetcher != "file" && cfg.Fetcher != "internal" {
return nil, fmt.Errorf("doki plugin fetcher must be 'file' or 'internal', got '%s'", cfg.Fetcher)
}
if cfg.Fetcher == "file" && cfg.URL == "" {
return nil, fmt.Errorf("doki plugin with file fetcher requires 'url'")
}
if cfg.Fetcher == "internal" && cfg.Text == "" {
return nil, fmt.Errorf("doki plugin with internal fetcher requires 'text'")
}
return &DokiPlugin{
BasePlugin: base,
Fetcher: cfg.Fetcher,
Text: cfg.Text,
URL: cfg.URL,
}, nil
case "tiki":
// Strict validation for Tiki
if cfg.Fetcher != "" {
return nil, fmt.Errorf("tiki plugin cannot have 'fetcher'")
}
if cfg.Text != "" {
return nil, fmt.Errorf("tiki plugin cannot have 'text'")
}
if cfg.URL != "" {
return nil, fmt.Errorf("tiki plugin cannot have 'url'")
}
if cfg.Filter != "" {
return nil, fmt.Errorf("tiki plugin cannot have 'filter'")
}
if len(cfg.Panes) == 0 {
return nil, fmt.Errorf("tiki plugin requires 'panes'")
}
if len(cfg.Panes) > 10 {
return nil, fmt.Errorf("tiki plugin has too many panes (%d), max is 10", len(cfg.Panes))
}
panes := make([]TikiPane, 0, len(cfg.Panes))
for i, pane := range cfg.Panes {
if pane.Name == "" {
return nil, fmt.Errorf("pane %d missing name", i)
}
columns := pane.Columns
if columns == 0 {
columns = 1
}
if columns < 0 {
return nil, fmt.Errorf("pane %q has invalid columns %d", pane.Name, columns)
}
filterExpr, err := filter.ParseFilter(pane.Filter)
if err != nil {
return nil, fmt.Errorf("parsing filter for pane %q: %w", pane.Name, err)
}
action, err := ParsePaneAction(pane.Action)
if err != nil {
return nil, fmt.Errorf("parsing action for pane %q: %w", pane.Name, err)
}
panes = append(panes, TikiPane{
Name: pane.Name,
Columns: columns,
Filter: filterExpr,
Action: action,
})
}
// Parse sort rules
sortRules, err := ParseSort(cfg.Sort)
if err != nil {
return nil, fmt.Errorf("parsing sort: %w", err)
}
return &TikiPlugin{
BasePlugin: base,
Panes: panes,
Sort: sortRules,
ViewMode: cfg.View,
}, nil
default:
return nil, fmt.Errorf("unknown plugin type: %s", pluginType)
}
}
// parsePluginYAML parses plugin YAML data into a Plugin
func parsePluginYAML(data []byte, source string) (Plugin, error) {
var cfg pluginFileConfig
if err := yaml.Unmarshal(data, &cfg); err != nil {
return nil, fmt.Errorf("parsing yaml: %w", err)
}
return parsePluginConfig(cfg, source)
}