clipman/main.go
2020-05-03 11:00:35 +02:00

184 lines
5.5 KiB
Go

// GPL v3.0
// 2019- (C) yory8 <yory8@users.noreply.github.com>
package main
import (
"bufio"
"encoding/json"
"fmt"
"io/ioutil"
"os"
"os/exec"
"strings"
"gopkg.in/alecthomas/kingpin.v2"
)
const version = "1.5.0"
var (
app = kingpin.New("clipman", "A clipboard manager for Wayland")
histpath = app.Flag("histpath", "Path of history file").Default("~/.local/share/clipman.json").String()
alert = app.Flag("notify", "Send desktop notifications on errors").Bool()
storer = app.Command("store", "Record clipboard events (run as argument to `wl-paste --watch`)")
maxDemon = storer.Flag("max-items", "history size").Default("15").Int()
noPersist = storer.Flag("no-persist", "Don't persist a copy buffer after a program exits").Short('P').Default("false").Bool()
picker = app.Command("pick", "Pick an item from clipboard history")
maxPicker = picker.Flag("max-items", "scrollview length").Default("15").Int()
pickTool = picker.Flag("tool", "Which selector to use: wofi/bemenu/CUSTOM/dmenu/rofi/STDOUT").Short('t').Required().String()
pickToolArgs = picker.Flag("tool-args", "Extra arguments to pass to the --tool").Short('T').Default("").String()
pickEsc = picker.Flag("print0", "Separate items using NULL; recommended if your tool supports --read0 or similar").Default("false").Bool()
clearer = app.Command("clear", "Remove item/s from history")
maxClearer = clearer.Flag("max-items", "scrollview length").Default("15").Int()
clearTool = clearer.Flag("tool", "Which selector to use: wofi/bemenu/CUSTOM/dmenu/rofi/STDOUT").Short('t').String()
clearToolArgs = clearer.Flag("tool-args", "Extra arguments to pass to the --tool").Short('T').Default("").String()
clearAll = clearer.Flag("all", "Remove all items").Short('a').Default("false").Bool()
clearEsc = clearer.Flag("print0", "Separate items using NULL; recommended if your tool supports --read0 or similar").Default("false").Bool()
_ = app.Command("restore", "Serve the last recorded item from history")
)
func main() {
app.Version(version)
app.HelpFlag.Short('h')
app.VersionFlag.Short('v')
action := kingpin.MustParse(app.Parse(os.Args[1:]))
histfile, history, err := getHistory(*histpath)
if err != nil {
smartLog(err.Error(), "critical", *alert)
}
switch action {
case "store":
// read copy from stdin
var stdin []string
scanner := bufio.NewScanner(os.Stdin)
for scanner.Scan() {
stdin = append(stdin, scanner.Text())
}
if err := scanner.Err(); err != nil {
smartLog("Couldn't get input from stdin.", "critical", *alert)
}
text := strings.Join(stdin, "\n")
persist := !*noPersist
if err := store(text, history, histfile, *maxDemon, persist); err != nil {
smartLog(err.Error(), "critical", *alert)
}
case "pick":
selection, err := selector(history, *maxPicker, *pickTool, "pick", *pickToolArgs, *pickEsc)
if err != nil {
smartLog(err.Error(), "normal", *alert)
}
if selection != "" {
// serve selection to the OS
serveTxt(selection)
}
case "restore":
if len(history) == 0 {
fmt.Println("Nothing to restore")
return
}
serveTxt(history[len(history)-1])
case "clear":
// remove all history
if *clearAll {
if err := wipeAll(histfile); err != nil {
smartLog(err.Error(), "normal", *alert)
}
return
}
if *clearTool == "" {
fmt.Println("clipman: error: required flag --tool or --all not provided, try --help")
os.Exit(1)
}
selection, err := selector(history, *maxClearer, *clearTool, "clear", *clearToolArgs, *clearEsc)
if err != nil {
smartLog(err.Error(), "normal", *alert)
}
if selection == "" {
return
}
if len(history) < 2 {
// there was only one possible item we could select, and we selected it,
// so wipe everything
if err := wipeAll(histfile); err != nil {
smartLog(err.Error(), "normal", *alert)
}
return
}
if selection == history[len(history)-1] {
// wl-copy is still serving the copy, so replace with next latest
// note: we alread exited if less than 2 items
serveTxt(history[len(history)-2])
}
if err := write(filter(history, selection), histfile); err != nil {
smartLog(err.Error(), "critical", *alert)
}
}
}
func wipeAll(histfile string) error {
// clear WM's clipboard
if err := exec.Command("wl-copy", "-c").Run(); err != nil {
return err
}
if err := os.Remove(histfile); err != nil {
return err
}
return nil
}
func getHistory(rawPath string) (string, []string, error) {
// set histfile; expand user home
histfile := rawPath
if strings.HasPrefix(histfile, "~") {
home, err := os.UserHomeDir()
if err != nil {
return "", nil, err
}
histfile = strings.Replace(histfile, "~", home, 1)
}
// read history if it exists
var history []string
b, err := ioutil.ReadFile(histfile)
if err != nil {
if !os.IsNotExist(err) {
return "", nil, fmt.Errorf("failure reading history file: %s", err)
}
} else {
if err := json.Unmarshal(b, &history); err != nil {
return "", nil, fmt.Errorf("failure parsing history: %s", err)
}
}
return histfile, history, nil
}
func serveTxt(s string) {
bin, err := exec.LookPath("wl-copy")
if err != nil {
smartLog(fmt.Sprintf("couldn't find wl-copy: %v\n", err), "low", *alert)
}
// we mandate the mime type because we know we can only serve text; not doing this leads to weird bugs like #35
cmd := exec.Cmd{Path: bin, Args: []string{bin, "-t", "TEXT"}, Stdin: strings.NewReader(s)}
if err := cmd.Run(); err != nil {
smartLog(fmt.Sprintf("error running wl-copy: %s\n", err), "low", *alert)
}
}