revert cli to use /api/generate (#1383)

This commit is contained in:
Patrick Devine 2023-12-04 16:35:29 -08:00 committed by GitHub
parent 7a0899d62d
commit bf704423c5
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -159,54 +159,7 @@ func RunHandler(cmd *cobra.Command, args []string) error {
return err return err
} }
interactive := true return RunGenerate(cmd, args)
opts := runOptions{
Model: name,
WordWrap: os.Getenv("TERM") == "xterm-256color",
Options: map[string]interface{}{},
}
format, err := cmd.Flags().GetString("format")
if err != nil {
return err
}
opts.Format = format
prompts := args[1:]
// prepend stdin to the prompt if provided
if !term.IsTerminal(int(os.Stdin.Fd())) {
in, err := io.ReadAll(os.Stdin)
if err != nil {
return err
}
prompts = append([]string{string(in)}, prompts...)
opts.WordWrap = false
interactive = false
}
msg := api.Message{
Role: "user",
Content: strings.Join(prompts, " "),
}
opts.Messages = append(opts.Messages, msg)
if len(prompts) > 0 {
interactive = false
}
nowrap, err := cmd.Flags().GetBool("nowordwrap")
if err != nil {
return err
}
opts.WordWrap = !nowrap
if !interactive {
_, err := chat(cmd, opts)
return err
}
return chatInteractive(cmd, opts)
} }
func PushHandler(cmd *cobra.Command, args []string) error { func PushHandler(cmd *cobra.Command, args []string) error {
@ -458,26 +411,83 @@ func PullHandler(cmd *cobra.Command, args []string) error {
return nil return nil
} }
type runOptions struct { func RunGenerate(cmd *cobra.Command, args []string) error {
interactive := true
opts := generateOptions{
Model: args[0],
WordWrap: os.Getenv("TERM") == "xterm-256color",
Options: map[string]interface{}{},
}
format, err := cmd.Flags().GetString("format")
if err != nil {
return err
}
opts.Format = format
prompts := args[1:]
// prepend stdin to the prompt if provided
if !term.IsTerminal(int(os.Stdin.Fd())) {
in, err := io.ReadAll(os.Stdin)
if err != nil {
return err
}
prompts = append([]string{string(in)}, prompts...)
opts.WordWrap = false
interactive = false
}
opts.Prompt = strings.Join(prompts, " ")
if len(prompts) > 0 {
interactive = false
}
nowrap, err := cmd.Flags().GetBool("nowordwrap")
if err != nil {
return err
}
opts.WordWrap = !nowrap
if !interactive {
return generate(cmd, opts)
}
return generateInteractive(cmd, opts)
}
type generateContextKey string
type generateOptions struct {
Model string Model string
Messages []api.Message Prompt string
WordWrap bool WordWrap bool
Format string Format string
System string
Template string Template string
Options map[string]interface{} Options map[string]interface{}
} }
func chat(cmd *cobra.Command, opts runOptions) (*api.Message, error) { func generate(cmd *cobra.Command, opts generateOptions) error {
client, err := api.ClientFromEnvironment() client, err := api.ClientFromEnvironment()
if err != nil { if err != nil {
return nil, err return err
} }
p := progress.NewProgress(os.Stderr) p := progress.NewProgress(os.Stderr)
defer p.StopAndClear() defer p.StopAndClear()
spinner := progress.NewSpinner("") spinner := progress.NewSpinner("")
p.Add("", spinner) p.Add("", spinner)
var latest api.GenerateResponse
generateContext, ok := cmd.Context().Value(generateContextKey("context")).([]int)
if !ok {
generateContext = []int{}
}
termWidth, _, err := term.GetSize(int(os.Stdout.Fd())) termWidth, _, err := term.GetSize(int(os.Stdout.Fd()))
if err != nil { if err != nil {
opts.WordWrap = false opts.WordWrap = false
@ -496,24 +506,24 @@ func chat(cmd *cobra.Command, opts runOptions) (*api.Message, error) {
var currentLineLength int var currentLineLength int
var wordBuffer string var wordBuffer string
var latest api.ChatResponse
var fullResponse strings.Builder
var role string
fn := func(response api.ChatResponse) error { request := api.GenerateRequest{
p.StopAndClear() Model: opts.Model,
latest = response Prompt: opts.Prompt,
if response.Message == nil { Context: generateContext,
// warm-up response or done Format: opts.Format,
return nil System: opts.System,
Template: opts.Template,
Options: opts.Options,
} }
role = response.Message.Role fn := func(response api.GenerateResponse) error {
content := response.Message.Content p.StopAndClear()
fullResponse.WriteString(content)
latest = response
termWidth, _, _ = term.GetSize(int(os.Stdout.Fd())) termWidth, _, _ = term.GetSize(int(os.Stdout.Fd()))
if opts.WordWrap && termWidth >= 10 { if opts.WordWrap && termWidth >= 10 {
for _, ch := range content { for _, ch := range response.Response {
if currentLineLength+1 > termWidth-5 { if currentLineLength+1 > termWidth-5 {
if len(wordBuffer) > termWidth-10 { if len(wordBuffer) > termWidth-10 {
fmt.Printf("%s%c", wordBuffer, ch) fmt.Printf("%s%c", wordBuffer, ch)
@ -541,7 +551,7 @@ func chat(cmd *cobra.Command, opts runOptions) (*api.Message, error) {
} }
} }
} else { } else {
fmt.Printf("%s%s", wordBuffer, content) fmt.Printf("%s%s", wordBuffer, response.Response)
if len(wordBuffer) > 0 { if len(wordBuffer) > 0 {
wordBuffer = "" wordBuffer = ""
} }
@ -550,35 +560,35 @@ func chat(cmd *cobra.Command, opts runOptions) (*api.Message, error) {
return nil return nil
} }
req := &api.ChatRequest{ if err := client.Generate(cancelCtx, &request, fn); err != nil {
Model: opts.Model,
Messages: opts.Messages,
Format: opts.Format,
Template: opts.Template,
Options: opts.Options,
}
if err := client.Chat(cancelCtx, req, fn); err != nil {
if errors.Is(err, context.Canceled) { if errors.Is(err, context.Canceled) {
return nil, nil return nil
} }
return nil, err return err
}
if opts.Prompt != "" {
fmt.Println()
fmt.Println()
} }
if len(opts.Messages) > 0 { if !latest.Done {
fmt.Println() return nil
fmt.Println()
} }
verbose, err := cmd.Flags().GetBool("verbose") verbose, err := cmd.Flags().GetBool("verbose")
if err != nil { if err != nil {
return nil, err return err
} }
if verbose { if verbose {
latest.Summary() latest.Summary()
} }
return &api.Message{Role: role, Content: fullResponse.String()}, nil ctx := cmd.Context()
ctx = context.WithValue(ctx, generateContextKey("context"), latest.Context)
cmd.SetContext(ctx)
return nil
} }
type MultilineState int type MultilineState int
@ -590,10 +600,13 @@ const (
MultilineTemplate MultilineTemplate
) )
func chatInteractive(cmd *cobra.Command, opts runOptions) error { func generateInteractive(cmd *cobra.Command, opts generateOptions) error {
// load the model // load the model
loadOpts := runOptions{Model: opts.Model} loadOpts := generateOptions{
if _, err := chat(cmd, loadOpts); err != nil { Model: opts.Model,
Prompt: "",
}
if err := generate(cmd, loadOpts); err != nil {
return err return err
} }
@ -664,9 +677,7 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
defer fmt.Printf(readline.EndBracketedPaste) defer fmt.Printf(readline.EndBracketedPaste)
var multiline MultilineState var multiline MultilineState
var content string var prompt string
var systemContent string
opts.Messages = make([]api.Message, 0)
for { for {
line, err := scanner.Readline() line, err := scanner.Readline()
@ -680,7 +691,7 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
} }
scanner.Prompt.UseAlt = false scanner.Prompt.UseAlt = false
content = "" prompt = ""
continue continue
case err != nil: case err != nil:
@ -688,37 +699,37 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
} }
switch { switch {
case strings.HasPrefix(content, `"""`): case strings.HasPrefix(prompt, `"""`):
// if the prompt so far starts with """ then we're in multiline mode // 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 """ // and we need to keep reading until we find a line that ends with """
cut, found := strings.CutSuffix(line, `"""`) cut, found := strings.CutSuffix(line, `"""`)
content += cut + "\n" prompt += cut + "\n"
if !found { if !found {
continue continue
} }
content = strings.TrimPrefix(content, `"""`) prompt = strings.TrimPrefix(prompt, `"""`)
scanner.Prompt.UseAlt = false scanner.Prompt.UseAlt = false
switch multiline { switch multiline {
case MultilineSystem: case MultilineSystem:
systemContent = content opts.System = prompt
content = "" prompt = ""
fmt.Println("Set system template.\n") fmt.Println("Set system template.\n")
case MultilineTemplate: case MultilineTemplate:
opts.Template = content opts.Template = prompt
content = "" prompt = ""
fmt.Println("Set model template.\n") fmt.Println("Set model template.\n")
} }
multiline = MultilineNone multiline = MultilineNone
case strings.HasPrefix(line, `"""`) && len(content) == 0: case strings.HasPrefix(line, `"""`) && len(prompt) == 0:
scanner.Prompt.UseAlt = true scanner.Prompt.UseAlt = true
multiline = MultilinePrompt multiline = MultilinePrompt
content += line + "\n" prompt += line + "\n"
continue continue
case scanner.Pasting: case scanner.Pasting:
content += line + "\n" prompt += line + "\n"
continue continue
case strings.HasPrefix(line, "/list"): case strings.HasPrefix(line, "/list"):
args := strings.Fields(line) args := strings.Fields(line)
@ -780,17 +791,17 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
line = strings.TrimPrefix(line, `"""`) line = strings.TrimPrefix(line, `"""`)
if strings.HasPrefix(args[2], `"""`) { if strings.HasPrefix(args[2], `"""`) {
cut, found := strings.CutSuffix(line, `"""`) cut, found := strings.CutSuffix(line, `"""`)
content += cut + "\n" prompt += cut + "\n"
if found { if found {
systemContent = content opts.System = prompt
if args[1] == "system" { if args[1] == "system" {
fmt.Println("Set system template.\n") fmt.Println("Set system template.\n")
} else { } else {
fmt.Println("Set prompt template.\n") fmt.Println("Set prompt template.\n")
} }
content = "" prompt = ""
} else { } else {
content = `"""` + content prompt = `"""` + prompt
if args[1] == "system" { if args[1] == "system" {
multiline = MultilineSystem multiline = MultilineSystem
} else { } else {
@ -799,7 +810,7 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
scanner.Prompt.UseAlt = true scanner.Prompt.UseAlt = true
} }
} else { } else {
systemContent = line opts.System = line
fmt.Println("Set system template.\n") fmt.Println("Set system template.\n")
} }
default: default:
@ -847,8 +858,8 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
} }
case "system": case "system":
switch { switch {
case systemContent != "": case opts.System != "":
fmt.Println(systemContent + "\n") fmt.Println(opts.System + "\n")
case resp.System != "": case resp.System != "":
fmt.Println(resp.System + "\n") fmt.Println(resp.System + "\n")
default: default:
@ -888,23 +899,16 @@ func chatInteractive(cmd *cobra.Command, opts runOptions) error {
fmt.Printf("Unknown command '%s'. Type /? for help\n", args[0]) fmt.Printf("Unknown command '%s'. Type /? for help\n", args[0])
continue continue
default: default:
content += line prompt += line
} }
if len(content) > 0 && multiline == MultilineNone { if len(prompt) > 0 && multiline == MultilineNone {
if systemContent != "" { opts.Prompt = prompt
opts.Messages = append(opts.Messages, api.Message{Role: "system", Content: systemContent}) if err := generate(cmd, opts); err != nil {
}
opts.Messages = append(opts.Messages, api.Message{Role: "user", Content: content})
assistant, err := chat(cmd, opts)
if err != nil {
return err return err
} }
if assistant != nil {
opts.Messages = append(opts.Messages, *assistant)
}
content = "" prompt = ""
} }
} }
} }