routes.go 5.01 KB
Newer Older
Jeffrey Morgan's avatar
Jeffrey Morgan committed
1
2
3
package server

import (
Michael Yang's avatar
Michael Yang committed
4
	"encoding/json"
Jeffrey Morgan's avatar
Jeffrey Morgan committed
5
6
7
8
	"io"
	"log"
	"net"
	"net/http"
9
	"os"
Michael Yang's avatar
Michael Yang committed
10
	"path/filepath"
Michael Yang's avatar
Michael Yang committed
11
12
	"strings"
	"text/template"
13
	"time"
Jeffrey Morgan's avatar
Jeffrey Morgan committed
14

Michael Yang's avatar
Michael Yang committed
15
	"dario.cat/mergo"
Jeffrey Morgan's avatar
Jeffrey Morgan committed
16
17
	"github.com/gin-gonic/gin"

Jeffrey Morgan's avatar
Jeffrey Morgan committed
18
	"github.com/jmorganca/ollama/api"
Michael Yang's avatar
Michael Yang committed
19
	"github.com/jmorganca/ollama/llama"
Jeffrey Morgan's avatar
Jeffrey Morgan committed
20
21
)

22
23
24
25
26
27
func cacheDir() string {
	home, err := os.UserHomeDir()
	if err != nil {
		panic(err)
	}

Michael Yang's avatar
Michael Yang committed
28
	return filepath.Join(home, ".ollama")
29
30
}

Bruce MacDonald's avatar
Bruce MacDonald committed
31
func generate(c *gin.Context) {
32
33
	start := time.Now()

Michael Yang's avatar
Michael Yang committed
34
	var req api.GenerateRequest
Bruce MacDonald's avatar
Bruce MacDonald committed
35
	if err := c.ShouldBindJSON(&req); err != nil {
36
		c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
Bruce MacDonald's avatar
Bruce MacDonald committed
37
38
		return
	}
39

40
41
42
43
	model, err := GetModel(req.Model)
	if err != nil {
		c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
		return
Bruce MacDonald's avatar
Bruce MacDonald committed
44
	}
Michael Yang's avatar
Michael Yang committed
45

Michael Yang's avatar
Michael Yang committed
46
47
48
49
50
51
52
53
54
55
56
	opts := api.DefaultOptions()
	if err := mergo.Merge(&opts, model.Options, mergo.WithOverride); err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
	}

	if err := mergo.Merge(&opts, req.Options, mergo.WithOverride); err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
	}

57
58
59
60
	templ, err := template.New("").Parse(model.Prompt)
	if err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
Michael Yang's avatar
Michael Yang committed
61
62
	}

63
64
65
66
	var sb strings.Builder
	if err = templ.Execute(&sb, req); err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
Michael Yang's avatar
Michael Yang committed
67
	}
68
	req.Prompt = sb.String()
Michael Yang's avatar
Michael Yang committed
69

Michael Yang's avatar
Michael Yang committed
70
	llm, err := llama.New(model.ModelPath, opts)
Michael Yang's avatar
Michael Yang committed
71
72
73
74
75
	if err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
	}
	defer llm.Close()
Jeffrey Morgan's avatar
Jeffrey Morgan committed
76

Michael Yang's avatar
Michael Yang committed
77
78
79
80
81
82
83
84
85
86
87
88
89
	ch := make(chan any)
	go func() {
		defer close(ch)
		llm.Predict(req.Context, req.Prompt, func(r api.GenerateResponse) {
			r.Model = req.Model
			r.CreatedAt = time.Now().UTC()
			if r.Done {
				r.TotalDuration = time.Since(start)
			}

			ch <- r
		})
	}()
Michael Yang's avatar
Michael Yang committed
90

Michael Yang's avatar
Michael Yang committed
91
	streamResponse(c, ch)
Michael Yang's avatar
Michael Yang committed
92
}
Michael Yang's avatar
Michael Yang committed
93

Michael Yang's avatar
Michael Yang committed
94
95
96
97
98
99
100
func pull(c *gin.Context) {
	var req api.PullRequest
	if err := c.ShouldBindJSON(&req); err != nil {
		c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
		return
	}

101
102
103
	ch := make(chan any)
	go func() {
		defer close(ch)
104
105
		fn := func(r api.ProgressResponse) {
			ch <- r
106
		}
107

108
109
110
111
112
113
114
115
116
117
118
119
120
		if err := PullModel(req.Name, req.Username, req.Password, fn); err != nil {
			c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
			return
		}
	}()

	streamResponse(c, ch)
}

func push(c *gin.Context) {
	var req api.PushRequest
	if err := c.ShouldBindJSON(&req); err != nil {
		c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
Michael Yang's avatar
Michael Yang committed
121
122
		return
	}
Michael Yang's avatar
Michael Yang committed
123

124
125
126
	ch := make(chan any)
	go func() {
		defer close(ch)
127
128
		fn := func(r api.ProgressResponse) {
			ch <- r
129
		}
130

131
132
133
134
135
136
137
138
139
140
141
142
143
		if err := PushModel(req.Name, req.Username, req.Password, fn); err != nil {
			c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
			return
		}
	}()

	streamResponse(c, ch)
}

func create(c *gin.Context) {
	var req api.CreateRequest
	if err := c.ShouldBindJSON(&req); err != nil {
		c.JSON(http.StatusBadRequest, gin.H{"message": err.Error()})
Michael Yang's avatar
Michael Yang committed
144
		return
145
146
	}

Michael Yang's avatar
Michael Yang committed
147
	ch := make(chan any)
Michael Yang's avatar
Michael Yang committed
148
149
	go func() {
		defer close(ch)
150
151
152
		fn := func(status string) {
			ch <- api.CreateProgress{
				Status: status,
Michael Yang's avatar
Michael Yang committed
153
			}
154
155
		}

156
		if err := CreateModel(req.Name, req.Path, fn); err != nil {
157
158
159
			c.JSON(http.StatusBadRequest, gin.H{"message": err.Error()})
			return
		}
Michael Yang's avatar
Michael Yang committed
160
	}()
Michael Yang's avatar
Michael Yang committed
161

Michael Yang's avatar
Michael Yang committed
162
	streamResponse(c, ch)
Bruce MacDonald's avatar
Bruce MacDonald committed
163
164
}

Patrick Devine's avatar
Patrick Devine committed
165
166
167
168
169
170
171
172
173
174
175
176
177
178
func list(c *gin.Context) {
	var models []api.ListResponseModel
	fp, err := GetManifestPath()
	if err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
	}
	err = filepath.Walk(fp, func(path string, info os.FileInfo, err error) error {
		if err != nil {
			return err
		}
		if !info.IsDir() {
			fi, err := os.Stat(path)
			if err != nil {
179
180
				log.Printf("skipping file: %s", fp)
				return nil
Patrick Devine's avatar
Patrick Devine committed
181
182
183
184
185
186
187
188
189
190
			}
			path := path[len(fp)+1:]
			slashIndex := strings.LastIndex(path, "/")
			if slashIndex == -1 {
				return nil
			}
			tag := path[:slashIndex] + ":" + path[slashIndex+1:]
			mp := ParseModelPath(tag)
			manifest, err := GetManifest(mp)
			if err != nil {
191
192
				log.Printf("skipping file: %s", fp)
				return nil
Patrick Devine's avatar
Patrick Devine committed
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
			}
			model := api.ListResponseModel{
				Name:       mp.GetShortTagname(),
				Size:       manifest.GetTotalSize(),
				ModifiedAt: fi.ModTime(),
			}
			models = append(models, model)
		}
		return nil
	})
	if err != nil {
		c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})
		return
	}

	c.JSON(http.StatusOK, api.ListResponse{models})
}

Bruce MacDonald's avatar
Bruce MacDonald committed
211
212
213
func Serve(ln net.Listener) error {
	r := gin.Default()

214
215
216
217
	r.GET("/", func(c *gin.Context) {
		c.String(http.StatusOK, "Ollama is running")
	})

Michael Yang's avatar
Michael Yang committed
218
	r.POST("/api/pull", pull)
Bruce MacDonald's avatar
Bruce MacDonald committed
219
	r.POST("/api/generate", generate)
220
221
	r.POST("/api/create", create)
	r.POST("/api/push", push)
Patrick Devine's avatar
Patrick Devine committed
222
	r.GET("/api/tags", list)
Jeffrey Morgan's avatar
Jeffrey Morgan committed
223
224
225
226
227
228
229
230

	log.Printf("Listening on %s", ln.Addr())
	s := &http.Server{
		Handler: r,
	}

	return s.Serve(ln)
}
Michael Yang's avatar
Michael Yang committed
231

Michael Yang's avatar
Michael Yang committed
232
func streamResponse(c *gin.Context, ch chan any) {
Michael Yang's avatar
Michael Yang committed
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
	c.Stream(func(w io.Writer) bool {
		val, ok := <-ch
		if !ok {
			return false
		}

		bts, err := json.Marshal(val)
		if err != nil {
			return false
		}

		bts = append(bts, '\n')
		if _, err := w.Write(bts); err != nil {
			return false
		}

		return true
	})
}