mirror of
https://github.com/kovidgoyal/kitty
synced 2026-06-08 22:28:24 +02:00
Start work on implementing edit-in-kitty in kitty-tool
This commit is contained in:
253
tools/cmd/edit_in_kitty/main.go
Normal file
253
tools/cmd/edit_in_kitty/main.go
Normal file
@@ -0,0 +1,253 @@
|
||||
// License: GPLv3 Copyright: 2022, Kovid Goyal, <kovid at kovidgoyal.net>
|
||||
|
||||
package edit_in_kitty
|
||||
|
||||
import (
|
||||
"encoding/base64"
|
||||
"fmt"
|
||||
"io"
|
||||
"io/fs"
|
||||
"os"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"golang.org/x/sys/unix"
|
||||
|
||||
"kitty/tools/cli"
|
||||
"kitty/tools/tui"
|
||||
"kitty/tools/tui/loop"
|
||||
"kitty/tools/utils"
|
||||
"kitty/tools/utils/humanize"
|
||||
)
|
||||
|
||||
var _ = fmt.Print
|
||||
|
||||
func encode(x string) string {
|
||||
return base64.StdEncoding.EncodeToString(utils.UnsafeStringToBytes(x))
|
||||
}
|
||||
|
||||
type OnDataCallback = func(data_type string, data []byte) error
|
||||
|
||||
func edit_loop(data_to_send string, kill_if_signaled bool, on_data OnDataCallback) (err error) {
|
||||
lp, err := loop.New(loop.NoAlternateScreen, loop.NoRestoreColors, loop.NoMouseTracking)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
current_text := strings.Builder{}
|
||||
data := strings.Builder{}
|
||||
data.Grow(4096)
|
||||
started := false
|
||||
canceled := false
|
||||
update_type := ""
|
||||
|
||||
handle_line := func(line string) error {
|
||||
if canceled {
|
||||
return nil
|
||||
}
|
||||
if started {
|
||||
if update_type == "" {
|
||||
update_type = line
|
||||
} else {
|
||||
if line == "KITTY_DATA_END" {
|
||||
lp.QueueWriteString(update_type + "\r\n")
|
||||
if update_type == "DONE" {
|
||||
lp.Quit(0)
|
||||
return nil
|
||||
}
|
||||
b, err := base64.StdEncoding.DecodeString(data.String())
|
||||
data.Reset()
|
||||
data.Grow(4096)
|
||||
started = false
|
||||
if err == nil {
|
||||
err = on_data(update_type, b)
|
||||
}
|
||||
update_type = ""
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
} else {
|
||||
data.WriteString(line)
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if line == "KITTY_DATA_START" {
|
||||
started = true
|
||||
update_type = ""
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
check_for_line := func() error {
|
||||
if canceled {
|
||||
return nil
|
||||
}
|
||||
s := current_text.String()
|
||||
for {
|
||||
idx := strings.Index(s, "\n")
|
||||
if idx < 0 {
|
||||
break
|
||||
}
|
||||
err = handle_line(s[:idx])
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
s = s[idx+1:]
|
||||
}
|
||||
current_text.Reset()
|
||||
current_text.Grow(4096)
|
||||
if s != "" {
|
||||
current_text.WriteString(s)
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
lp.OnInitialize = func() (string, error) {
|
||||
pos, chunk_num := 0, 0
|
||||
for {
|
||||
limit := utils.Min(pos+2048, len(data_to_send))
|
||||
if limit <= pos {
|
||||
break
|
||||
}
|
||||
lp.QueueWriteString("\x1bP@kitty-edit|" + strconv.Itoa(chunk_num) + ":")
|
||||
lp.QueueWriteString(data_to_send[pos:limit])
|
||||
lp.QueueWriteString("\x1b\\")
|
||||
chunk_num++
|
||||
pos = limit
|
||||
}
|
||||
lp.QueueWriteString("\x1bP@kitty-edit|\x1b\\")
|
||||
return "", nil
|
||||
}
|
||||
|
||||
lp.OnText = func(text string, from_key_event bool, in_bracketed_paste bool) error {
|
||||
if !from_key_event {
|
||||
current_text.WriteString(text)
|
||||
err = check_for_line()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
const abort_msg = "\x1bP@kitty-edit|0:abort_signaled=interrupt\x1b\\\x1bP@kitty-edit|\x1b\\"
|
||||
|
||||
lp.OnKeyEvent = func(event *loop.KeyEvent) error {
|
||||
if event.MatchesPressOrRepeat("ctrl+c") || event.MatchesPressOrRepeat("esc") {
|
||||
event.Handled = true
|
||||
canceled = true
|
||||
lp.QueueWriteString(abort_msg)
|
||||
if !started {
|
||||
return tui.Canceled
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
err = lp.Run()
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
if canceled {
|
||||
return tui.Canceled
|
||||
}
|
||||
|
||||
ds := lp.DeathSignalName()
|
||||
if ds != "" {
|
||||
fmt.Print(abort_msg)
|
||||
if kill_if_signaled {
|
||||
lp.KillIfSignalled()
|
||||
return
|
||||
}
|
||||
return &tui.KilledBySignal{Msg: fmt.Sprint("Killed by signal: ", ds), SignalName: ds}
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func edit_in_kitty(path string) (err error) {
|
||||
read_file, err := os.Open(path)
|
||||
if err != nil {
|
||||
return fmt.Errorf("Failed to open %s for reading with error: %w", path, err)
|
||||
}
|
||||
defer read_file.Close()
|
||||
var s unix.Stat_t
|
||||
err = unix.Fstat(int(read_file.Fd()), &s)
|
||||
if err != nil {
|
||||
return fmt.Errorf("Failed to stat %s with error: %w", path, err)
|
||||
}
|
||||
if s.Size > 8*1024*1024 {
|
||||
return fmt.Errorf("File size %s is too large for performant editing", humanize.Bytes(uint64(s.Size)))
|
||||
}
|
||||
|
||||
file_data, err := io.ReadAll(read_file)
|
||||
if err != nil {
|
||||
return fmt.Errorf("Failed to read from %s with error: %w", path, err)
|
||||
}
|
||||
read_file.Close()
|
||||
data := strings.Builder{}
|
||||
data.Grow(len(file_data) * 4)
|
||||
|
||||
add := func(key, val string) {
|
||||
if data.Len() > 0 {
|
||||
data.WriteString(",")
|
||||
}
|
||||
data.WriteString(key)
|
||||
data.WriteString("=")
|
||||
data.WriteString(val)
|
||||
}
|
||||
add_encoded := func(key, val string) { add(key, encode(val)) }
|
||||
|
||||
if unix.Access(path, unix.R_OK|unix.W_OK) != nil {
|
||||
return fmt.Errorf("%s is not readable and writeable", path)
|
||||
}
|
||||
cwd, err := os.Getwd()
|
||||
if err != nil {
|
||||
return fmt.Errorf("Failed to get the current working directory with error: %w", err)
|
||||
}
|
||||
add_encoded("cwd", cwd)
|
||||
for _, arg := range os.Args[2:] {
|
||||
add_encoded("a", arg)
|
||||
}
|
||||
add("file_inode", fmt.Sprintf("%d:%d:%d", s.Dev, s.Ino, s.Mtim.Nano()))
|
||||
add_encoded("file_data", utils.UnsafeBytesToString(file_data))
|
||||
fmt.Println("Waiting for editing to be completed, press Esc to abort...")
|
||||
write_data := func(data_type string, rdata []byte) (err error) {
|
||||
err = utils.AtomicWriteFile(path, rdata, fs.FileMode(s.Mode).Perm())
|
||||
if err != nil {
|
||||
err = fmt.Errorf("Failed to write data to %s with error: %w", path, err)
|
||||
}
|
||||
return
|
||||
}
|
||||
err = edit_loop(data.String(), true, write_data)
|
||||
if err != nil {
|
||||
if err == tui.Canceled {
|
||||
return err
|
||||
}
|
||||
return fmt.Errorf("Failed to receive edited file back from terminal with error: %w", err)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
func EntryPoint(parent *cli.Command) *cli.Command {
|
||||
sc := parent.AddSubCommand(&cli.Command{
|
||||
Name: "edit-in-kitty",
|
||||
Usage: "edit-in-kitty [options] file-to-edit",
|
||||
ShortDescription: "Edit a file in a kitty overlay window",
|
||||
HelpText: "Edit the specified file in a kitty overlay window. Works over SSH as well.\n\n" +
|
||||
"For usage instructions see: https://sw.kovidgoyal.net/kitty/shell-integration/#edit-file",
|
||||
Run: func(cmd *cli.Command, args []string) (ret int, err error) {
|
||||
if len(args) == 0 {
|
||||
fmt.Fprintln(os.Stderr, "Usage:", cmd.Usage)
|
||||
return 1, fmt.Errorf("No file to edit specified.")
|
||||
}
|
||||
if len(args) != 1 {
|
||||
fmt.Fprintln(os.Stderr, "Usage:", cmd.Usage)
|
||||
return 1, fmt.Errorf("Only one file to edit must be specified")
|
||||
}
|
||||
err = edit_in_kitty(args[0])
|
||||
return 0, err
|
||||
},
|
||||
})
|
||||
AddCloneSafeOpts(sc)
|
||||
return sc
|
||||
}
|
||||
@@ -6,6 +6,7 @@ import (
|
||||
"fmt"
|
||||
"kitty/tools/cli"
|
||||
"kitty/tools/cmd/at"
|
||||
"kitty/tools/cmd/edit_in_kitty"
|
||||
"kitty/tools/cmd/update_self"
|
||||
)
|
||||
|
||||
@@ -18,4 +19,6 @@ func KittyToolEntryPoints(root *cli.Command) {
|
||||
at.EntryPoint(root)
|
||||
// update-self
|
||||
update_self.EntryPoint(root)
|
||||
// edit-in-kitty
|
||||
edit_in_kitty.EntryPoint(root)
|
||||
}
|
||||
|
||||
@@ -78,7 +78,7 @@ func update_self(version string) (err error) {
|
||||
return unix.Exec(exe, []string{"kitty-tool", "--version"}, os.Environ())
|
||||
}
|
||||
|
||||
func EntryPoint(root *cli.Command) {
|
||||
func EntryPoint(root *cli.Command) *cli.Command {
|
||||
sc := root.AddSubCommand(&cli.Command{
|
||||
Name: "update-self",
|
||||
Usage: "update-self [options ...]",
|
||||
@@ -101,4 +101,5 @@ func EntryPoint(root *cli.Command) {
|
||||
Default: "latest",
|
||||
Help: "The version to fetch. The special words :code:`latest` and :code:`nightly` fetch the latest stable and nightly release respectively. Other values can be, for example: 0.27.1.",
|
||||
})
|
||||
return sc
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user