From 9fb5e8399c61217a4deef60a773de19ebd87caca Mon Sep 17 00:00:00 2001 From: Jeffrey Morgan Date: Sat, 25 Nov 2023 23:30:34 -0500 Subject: [PATCH] Fix issues with inputting and formatting multi line strings in `ollama run` Co-authored-by: Wen Sun --- cmd/cmd.go | 47 ++++++++++++++++++++++++++------------------ readline/readline.go | 19 ++++++++---------- readline/types.go | 8 -------- 3 files changed, 36 insertions(+), 38 deletions(-) diff --git a/cmd/cmd.go b/cmd/cmd.go index 56d3471c..970a398b 100644 --- a/cmd/cmd.go +++ b/cmd/cmd.go @@ -602,14 +602,12 @@ func generateInteractive(cmd *cobra.Command, model string, wordWrap bool, format fmt.Fprintln(os.Stderr, "") } - prompt := readline.Prompt{ + scanner, err := readline.New(readline.Prompt{ Prompt: ">>> ", AltPrompt: "... ", Placeholder: "Send a message (/? for help)", AltPlaceholder: `Use """ to end multi-line input`, - } - - scanner, err := readline.New(prompt) + }) if err != nil { return err } @@ -617,7 +615,7 @@ func generateInteractive(cmd *cobra.Command, model string, wordWrap bool, format fmt.Print(readline.StartBracketedPaste) defer fmt.Printf(readline.EndBracketedPaste) - var multiLineBuffer string + var prompt string for { line, err := scanner.Readline() @@ -630,27 +628,33 @@ func generateInteractive(cmd *cobra.Command, model string, wordWrap bool, format fmt.Println("\nUse Ctrl-D or /bye to exit.") } + scanner.Prompt.UseAlt = false + prompt = "" + continue case err != nil: return err } - line = strings.TrimSpace(line) - switch { - case scanner.Prompt.UseAlt: - if strings.HasSuffix(line, `"""`) { - scanner.Prompt.UseAlt = false - multiLineBuffer += strings.TrimSuffix(line, `"""`) - line = multiLineBuffer - multiLineBuffer = "" - } else { - multiLineBuffer += line + " " + case strings.HasPrefix(prompt, `"""`): + // if the prompt so far starts with """ then we're in multiline mode + // and we need to keep reading until we find a line that ends with """ + cut, found := strings.CutSuffix(line, `"""`) + prompt += cut + "\n" + + if !found { continue } - case strings.HasPrefix(line, `"""`): + + prompt = strings.TrimPrefix(prompt, `"""`) + scanner.Prompt.UseAlt = false + case strings.HasPrefix(line, `"""`) && len(prompt) == 0: scanner.Prompt.UseAlt = true - multiLineBuffer = strings.TrimPrefix(line, `"""`) + " " + prompt += line + "\n" + continue + case scanner.Pasting: + prompt += line + "\n" continue case strings.HasPrefix(line, "/list"): args := strings.Fields(line) @@ -757,12 +761,17 @@ func generateInteractive(cmd *cobra.Command, model string, wordWrap bool, format case strings.HasPrefix(line, "/"): args := strings.Fields(line) fmt.Printf("Unknown command '%s'. Type /? for help\n", args[0]) + continue + default: + prompt += line } - if len(line) > 0 && line[0] != '/' { - if err := generate(cmd, model, line, wordWrap, format); err != nil { + if len(prompt) > 0 && prompt[0] != '/' { + if err := generate(cmd, model, prompt, wordWrap, format); err != nil { return err } + + prompt = "" } } } diff --git a/readline/readline.go b/readline/readline.go index 82b5351b..88813fc6 100644 --- a/readline/readline.go +++ b/readline/readline.go @@ -24,6 +24,7 @@ type Instance struct { Prompt *Prompt Terminal *Terminal History *History + Pasting bool } func New(prompt Prompt) (*Instance, error) { @@ -46,7 +47,7 @@ func New(prompt Prompt) (*Instance, error) { func (i *Instance) Readline() (string, error) { prompt := i.Prompt.Prompt - if i.Prompt.UseAlt { + if i.Prompt.UseAlt || i.Pasting { prompt = i.Prompt.AltPrompt } fmt.Print(prompt) @@ -63,12 +64,13 @@ func (i *Instance) Readline() (string, error) { var esc bool var escex bool var metaDel bool - var pasteMode PasteMode var currentLineBuf []rune for { - if buf.IsEmpty() { + // don't show placeholder when pasting unless we're in multiline mode + showPlaceholder := !i.Pasting || i.Prompt.UseAlt + if buf.IsEmpty() && showPlaceholder { ph := i.Prompt.Placeholder if i.Prompt.UseAlt { ph = i.Prompt.AltPlaceholder @@ -119,9 +121,9 @@ func (i *Instance) Readline() (string, error) { code += string(r) } if code == CharBracketedPasteStart { - pasteMode = PasteModeStart + i.Pasting = true } else if code == CharBracketedPasteEnd { - pasteMode = PasteModeEnd + i.Pasting = false } case KeyDel: if buf.Size() > 0 { @@ -196,12 +198,7 @@ func (i *Instance) Readline() (string, error) { } buf.MoveToEnd() fmt.Println() - switch pasteMode { - case PasteModeStart: - output = `"""` + output - case PasteModeEnd: - output = output + `"""` - } + return output, nil default: if metaDel { diff --git a/readline/types.go b/readline/types.go index 03fa526d..3b88588f 100644 --- a/readline/types.go +++ b/readline/types.go @@ -76,11 +76,3 @@ const ( CharBracketedPasteStart = "00~" CharBracketedPasteEnd = "01~" ) - -type PasteMode int - -const ( - PastModeOff = iota - PasteModeStart - PasteModeEnd -)