2023-07-16 17:02:22 -07:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
2023-07-25 22:52:23 +05:30
|
|
|
"bytes"
|
2023-07-17 14:21:27 -07:00
|
|
|
"errors"
|
2023-07-16 17:02:22 -07:00
|
|
|
"io"
|
2023-07-25 11:50:23 -07:00
|
|
|
"log"
|
2023-07-16 17:02:22 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
type Command struct {
|
|
|
|
Name string
|
2023-07-17 14:21:27 -07:00
|
|
|
Args string
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Command) Reset() {
|
|
|
|
c.Name = ""
|
|
|
|
c.Args = ""
|
2023-07-16 17:02:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
func Parse(reader io.Reader) ([]Command, error) {
|
|
|
|
var commands []Command
|
2023-07-17 14:21:27 -07:00
|
|
|
var command, modelCommand Command
|
2023-07-16 17:02:22 -07:00
|
|
|
|
|
|
|
scanner := bufio.NewScanner(reader)
|
2023-07-17 14:21:27 -07:00
|
|
|
scanner.Split(scanModelfile)
|
2023-07-16 17:02:22 -07:00
|
|
|
for scanner.Scan() {
|
2023-07-17 14:21:27 -07:00
|
|
|
line := scanner.Bytes()
|
|
|
|
|
|
|
|
fields := bytes.SplitN(line, []byte(" "), 2)
|
2023-07-25 11:50:23 -07:00
|
|
|
if len(fields) == 0 || len(fields[0]) == 0 {
|
2023-07-16 17:02:22 -07:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2023-07-25 22:52:23 +05:30
|
|
|
switch string(bytes.ToUpper(fields[0])) {
|
2023-07-16 17:02:22 -07:00
|
|
|
case "FROM":
|
|
|
|
command.Name = "model"
|
2023-07-17 14:21:27 -07:00
|
|
|
command.Args = string(fields[1])
|
|
|
|
// copy command for validation
|
|
|
|
modelCommand = command
|
2023-07-20 01:13:30 -07:00
|
|
|
case "LICENSE", "TEMPLATE", "SYSTEM", "PROMPT":
|
2023-07-25 22:52:23 +05:30
|
|
|
command.Name = string(bytes.ToLower(fields[0]))
|
2023-07-17 14:21:27 -07:00
|
|
|
command.Args = string(fields[1])
|
2023-07-16 17:02:22 -07:00
|
|
|
case "PARAMETER":
|
2023-07-17 14:21:27 -07:00
|
|
|
fields = bytes.SplitN(fields[1], []byte(" "), 2)
|
|
|
|
command.Name = string(fields[0])
|
|
|
|
command.Args = string(fields[1])
|
2023-07-16 17:02:22 -07:00
|
|
|
default:
|
2023-07-25 22:52:23 +05:30
|
|
|
// log a warning for unknown commands
|
2023-07-25 11:50:23 -07:00
|
|
|
log.Printf("WARNING: Unknown command: %s", fields[0])
|
2023-07-25 22:52:23 +05:30
|
|
|
continue
|
2023-07-16 17:02:22 -07:00
|
|
|
}
|
2023-07-17 14:21:27 -07:00
|
|
|
|
|
|
|
commands = append(commands, command)
|
|
|
|
command.Reset()
|
2023-07-16 17:02:22 -07:00
|
|
|
}
|
|
|
|
|
2023-07-17 14:21:27 -07:00
|
|
|
if modelCommand.Args == "" {
|
2023-07-24 18:11:13 +05:30
|
|
|
return nil, errors.New("no FROM line for the model was specified")
|
2023-07-16 17:02:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return commands, scanner.Err()
|
|
|
|
}
|
2023-07-17 14:21:27 -07:00
|
|
|
|
|
|
|
func scanModelfile(data []byte, atEOF bool) (advance int, token []byte, err error) {
|
2023-07-25 22:52:23 +05:30
|
|
|
const multilineString = `"""`
|
|
|
|
|
2023-07-17 14:21:27 -07:00
|
|
|
newline := bytes.IndexByte(data, '\n')
|
|
|
|
|
2023-07-24 18:11:13 +05:30
|
|
|
if start := bytes.Index(data, []byte(multilineString)); start >= 0 && start < newline {
|
|
|
|
end := bytes.Index(data[start+len(multilineString):], []byte(multilineString))
|
2023-07-17 14:21:27 -07:00
|
|
|
if end < 0 {
|
2023-07-20 00:37:52 -07:00
|
|
|
if atEOF {
|
2023-07-24 18:11:13 +05:30
|
|
|
return 0, nil, errors.New("unterminated multiline string: " + multilineString)
|
2023-07-20 00:37:52 -07:00
|
|
|
} else {
|
|
|
|
return 0, nil, nil
|
|
|
|
}
|
2023-07-17 14:21:27 -07:00
|
|
|
}
|
|
|
|
|
2023-07-24 18:11:13 +05:30
|
|
|
n := start + len(multilineString) + end + len(multilineString)
|
2023-07-25 11:50:23 -07:00
|
|
|
|
|
|
|
newData := data[:start]
|
|
|
|
newData = append(newData, data[start+len(multilineString):n-len(multilineString)]...)
|
|
|
|
return n, newData, nil
|
2023-07-17 14:21:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
return bufio.ScanLines(data, atEOF)
|
|
|
|
}
|