parser.go 2.46 KB
Newer Older
1
2
3
4
package parser

import (
	"bufio"
5
	"bytes"
6
	"errors"
7
	"fmt"
8
	"io"
Michael Yang's avatar
Michael Yang committed
9
	"log"
10
11
12
13
)

type Command struct {
	Name string
14
15
16
17
18
19
	Args string
}

func (c *Command) Reset() {
	c.Name = ""
	c.Args = ""
20
21
22
23
}

func Parse(reader io.Reader) ([]Command, error) {
	var commands []Command
24
	var command, modelCommand Command
25
26

	scanner := bufio.NewScanner(reader)
27
	scanner.Buffer(make([]byte, 0, bufio.MaxScanTokenSize), bufio.MaxScanTokenSize)
28
	scanner.Split(scanModelfile)
29
	for scanner.Scan() {
30
31
32
		line := scanner.Bytes()

		fields := bytes.SplitN(line, []byte(" "), 2)
Michael Yang's avatar
Michael Yang committed
33
		if len(fields) == 0 || len(fields[0]) == 0 {
34
35
36
			continue
		}

37
		switch string(bytes.ToUpper(fields[0])) {
38
39
		case "FROM":
			command.Name = "model"
40
41
42
			command.Args = string(fields[1])
			// copy command for validation
			modelCommand = command
Jeffrey Morgan's avatar
Jeffrey Morgan committed
43
		case "LICENSE", "TEMPLATE", "SYSTEM", "PROMPT":
44
			command.Name = string(bytes.ToLower(fields[0]))
45
			command.Args = string(fields[1])
46
		case "PARAMETER":
47
48
49
			fields = bytes.SplitN(fields[1], []byte(" "), 2)
			command.Name = string(fields[0])
			command.Args = string(fields[1])
50
		default:
51
			// log a warning for unknown commands
Michael Yang's avatar
Michael Yang committed
52
			log.Printf("WARNING: Unknown command: %s", fields[0])
53
			continue
54
		}
55
56
57

		commands = append(commands, command)
		command.Reset()
58
59
	}

60
	if modelCommand.Args == "" {
61
		return nil, errors.New("no FROM line for the model was specified")
62
63
64
65
	}

	return commands, scanner.Err()
}
66
67

func scanModelfile(data []byte, atEOF bool) (advance int, token []byte, err error) {
68
69
70
71
72
73
74
75
	advance, token, err = scan([]byte(`"""`), []byte(`"""`), data, atEOF)
	if err != nil {
		return 0, nil, err
	}

	if advance > 0 && token != nil {
		return advance, token, nil
	}
76

77
78
79
80
81
82
83
84
85
86
87
88
89
	advance, token, err = scan([]byte(`"`), []byte(`"`), data, atEOF)
	if err != nil {
		return 0, nil, err
	}

	if advance > 0 && token != nil {
		return advance, token, nil
	}

	return bufio.ScanLines(data, atEOF)
}

func scan(openBytes, closeBytes, data []byte, atEOF bool) (advance int, token []byte, err error) {
90
91
	newline := bytes.IndexByte(data, '\n')

92
93
	if start := bytes.Index(data, openBytes); start >= 0 && start < newline {
		end := bytes.Index(data[start+len(openBytes):], closeBytes)
94
		if end < 0 {
95
			if atEOF {
96
				return 0, nil, fmt.Errorf("unterminated %s: expecting %s", openBytes, closeBytes)
97
98
99
			} else {
				return 0, nil, nil
			}
100
101
		}

102
		n := start + len(openBytes) + end + len(closeBytes)
Michael Yang's avatar
Michael Yang committed
103
104

		newData := data[:start]
105
		newData = append(newData, data[start+len(openBytes):n-len(closeBytes)]...)
Michael Yang's avatar
Michael Yang committed
106
		return n, newData, nil
107
108
	}

109
	return 0, nil, nil
110
}