2023-07-03 15:22:44 -04:00
package server
import (
2023-07-25 17:08:51 -04:00
"context"
2023-07-06 10:40:11 -07:00
"encoding/json"
2023-10-06 16:06:20 -04:00
"errors"
2023-07-21 23:02:12 -07:00
"fmt"
2023-07-03 15:22:44 -04:00
"io"
2023-10-06 16:06:20 -04:00
"io/fs"
2024-01-18 10:52:01 -08:00
"log/slog"
2024-03-13 13:29:40 -07:00
"math"
2023-07-03 15:22:44 -04:00
"net"
"net/http"
2024-03-08 22:23:47 -08:00
"net/netip"
2023-07-07 15:27:43 -04:00
"os"
2023-08-30 16:35:03 -04:00
"os/signal"
2023-07-14 17:27:14 -07:00
"path/filepath"
2023-07-31 21:35:18 -04:00
"reflect"
2023-09-12 11:04:35 -04:00
"runtime"
2024-03-13 13:29:40 -07:00
"strconv"
2023-07-06 10:40:11 -07:00
"strings"
2023-07-18 11:59:42 -07:00
"sync"
2023-08-30 16:35:03 -04:00
"syscall"
2023-07-12 18:18:06 -07:00
"time"
2023-07-03 15:22:44 -04:00
2023-07-21 18:01:24 -07:00
"github.com/gin-contrib/cors"
2023-07-03 15:22:44 -04:00
"github.com/gin-gonic/gin"
2024-02-12 11:16:20 -08:00
"golang.org/x/exp/slices"
2023-07-03 15:22:44 -04:00
2023-07-03 16:32:48 -04:00
"github.com/jmorganca/ollama/api"
2023-11-29 11:00:37 -08:00
"github.com/jmorganca/ollama/gpu"
2023-07-21 13:33:56 -07:00
"github.com/jmorganca/ollama/llm"
2024-02-07 17:24:29 -05:00
"github.com/jmorganca/ollama/openai"
2023-11-14 12:30:34 -08:00
"github.com/jmorganca/ollama/parser"
2023-10-13 16:08:35 -07:00
"github.com/jmorganca/ollama/version"
2023-07-03 15:22:44 -04:00
)
2023-08-22 09:48:35 -07:00
var mode string = gin . DebugMode
2023-12-14 16:47:40 -08:00
type Server struct {
2024-03-08 22:23:47 -08:00
addr net . Addr
2023-12-14 16:47:40 -08:00
}
2023-08-22 09:48:35 -07:00
func init ( ) {
switch mode {
case gin . DebugMode :
case gin . ReleaseMode :
case gin . TestMode :
default :
mode = gin . DebugMode
}
gin . SetMode ( mode )
}
2023-07-31 21:35:18 -04:00
var loaded struct {
2023-07-19 15:00:28 -07:00
mu sync . Mutex
2023-10-19 10:39:58 -04:00
runner llm . LLM
2023-07-19 15:00:28 -07:00
expireAt time . Time
expireTimer * time . Timer
2023-07-31 21:35:18 -04:00
2023-10-19 10:39:58 -04:00
* Model
* api . Options
2023-07-18 11:59:42 -07:00
}
2023-08-15 10:35:39 -03:00
var defaultSessionDuration = 5 * time . Minute
2023-08-08 15:13:22 -04:00
// load a model into memory if it is not already loaded, it is up to the caller to lock loaded.mu before calling this function
2024-01-03 12:01:42 -05:00
func load ( c * gin . Context , model * Model , opts api . Options , sessionDuration time . Duration ) error {
2023-10-19 10:39:58 -04:00
needLoad := loaded . runner == nil || // is there a model loaded?
loaded . ModelPath != model . ModelPath || // has the base model changed?
! reflect . DeepEqual ( loaded . AdapterPaths , model . AdapterPaths ) || // have the adapters changed?
! reflect . DeepEqual ( loaded . Options . Runner , opts . Runner ) // have the runner options changed?
if needLoad {
if loaded . runner != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( "changing loaded model" )
2023-10-19 10:39:58 -04:00
loaded . runner . Close ( )
loaded . runner = nil
loaded . Model = nil
loaded . Options = nil
2023-07-18 11:59:42 -07:00
}
2023-07-17 12:08:10 -07:00
2024-02-15 17:15:09 -08:00
llmRunner , err := llm . New ( model . ModelPath , model . AdapterPaths , model . ProjectorPaths , opts )
2023-07-18 11:59:42 -07:00
if err != nil {
2023-10-19 14:50:45 -04:00
// some older models are not compatible with newer versions of llama.cpp
// show a generalized compatibility error until there is a better way to
// check for model compatibility
2023-11-24 13:58:09 -05:00
if errors . Is ( llm . ErrUnsupportedFormat , err ) || strings . Contains ( err . Error ( ) , "failed to load model" ) {
2023-10-19 14:50:45 -04:00
err = fmt . Errorf ( "%v: this model may be incompatible with your version of Ollama. If you previously pulled this model, try updating it by running `ollama pull %s`" , err , model . ShortName )
}
2024-01-03 12:01:42 -05:00
return err
2023-07-18 11:59:42 -07:00
}
2023-10-19 10:39:58 -04:00
loaded . Model = model
loaded . runner = llmRunner
loaded . Options = & opts
2023-07-19 15:00:28 -07:00
}
2023-09-21 20:38:49 +01:00
2023-07-31 21:35:18 -04:00
loaded . expireAt = time . Now ( ) . Add ( sessionDuration )
2023-08-08 15:13:22 -04:00
2023-07-31 21:35:18 -04:00
if loaded . expireTimer == nil {
loaded . expireTimer = time . AfterFunc ( sessionDuration , func ( ) {
loaded . mu . Lock ( )
defer loaded . mu . Unlock ( )
2023-07-19 15:00:28 -07:00
2023-07-31 21:35:18 -04:00
if time . Now ( ) . Before ( loaded . expireAt ) {
2023-07-19 15:00:28 -07:00
return
}
2023-10-19 10:39:58 -04:00
if loaded . runner != nil {
loaded . runner . Close ( )
2023-07-19 15:00:28 -07:00
}
2023-10-19 10:39:58 -04:00
loaded . runner = nil
loaded . Model = nil
loaded . Options = nil
2023-07-19 15:00:28 -07:00
} )
2023-07-06 10:40:11 -07:00
}
2023-09-21 20:38:49 +01:00
2023-07-31 21:35:18 -04:00
loaded . expireTimer . Reset ( sessionDuration )
2024-01-03 12:01:42 -05:00
return nil
}
func modelOptions ( model * Model , requestOpts map [ string ] interface { } ) ( api . Options , error ) {
opts := api . DefaultOptions ( )
if err := opts . FromMap ( model . Options ) ; err != nil {
return api . Options { } , err
}
if err := opts . FromMap ( requestOpts ) ; err != nil {
return api . Options { } , err
}
return opts , nil
2023-08-08 15:13:22 -04:00
}
2024-02-12 11:16:20 -08:00
func isSupportedImageType ( image [ ] byte ) bool {
contentType := http . DetectContentType ( image )
allowedTypes := [ ] string { "image/jpeg" , "image/jpg" , "image/png" }
return slices . Contains ( allowedTypes , contentType )
}
2023-08-08 15:13:22 -04:00
func GenerateHandler ( c * gin . Context ) {
loaded . mu . Lock ( )
defer loaded . mu . Unlock ( )
checkpointStart := time . Now ( )
var req api . GenerateRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
2023-12-11 13:56:22 -08:00
2023-10-18 16:08:42 -07:00
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-08-08 15:13:22 -04:00
return
}
2023-11-08 14:05:02 -08:00
// validate the request
switch {
case req . Model == "" :
2023-10-18 15:56:34 -07:00
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
return
2023-11-09 16:44:02 -08:00
case len ( req . Format ) > 0 && req . Format != "json" :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "format must be json" } )
return
2023-11-08 14:05:02 -08:00
case req . Raw && ( req . Template != "" || req . System != "" || len ( req . Context ) > 0 ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "raw mode does not support template, system, or context" } )
return
2023-10-18 15:56:34 -07:00
}
2024-02-12 11:16:20 -08:00
for _ , img := range req . Images {
if ! isSupportedImageType ( img ) {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "unsupported image format" } )
return
}
}
2024-01-03 12:01:42 -05:00
model , err := GetModel ( req . Model )
2023-08-08 15:13:22 -04:00
if err != nil {
2023-10-06 16:06:20 -04:00
var pErr * fs . PathError
2024-01-03 12:01:42 -05:00
if errors . As ( err , & pErr ) {
2023-10-06 16:06:20 -04:00
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found, try pulling it first" , req . Model ) } )
2024-01-03 12:01:42 -05:00
return
}
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-02-20 21:37:29 -05:00
if model . IsEmbedding ( ) {
2024-02-20 21:53:45 -05:00
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "embedding models do not support generate" } )
2024-02-20 21:37:29 -05:00
return
}
2024-01-03 12:01:42 -05:00
opts , err := modelOptions ( model , req . Options )
if err != nil {
if errors . Is ( err , api . ErrInvalidOpts ) {
2023-12-05 14:57:33 -05:00
c . JSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2024-01-03 12:01:42 -05:00
return
2023-11-08 14:05:02 -08:00
}
2024-01-03 12:01:42 -05:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-01-26 14:28:02 -08:00
var sessionDuration time . Duration
if req . KeepAlive == nil {
2024-03-13 13:29:40 -07:00
sessionDuration = getDefaultSessionDuration ( )
2024-01-26 14:28:02 -08:00
} else {
sessionDuration = req . KeepAlive . Duration
}
2024-01-03 12:01:42 -05:00
if err := load ( c , model , opts , sessionDuration ) ; err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
2023-08-08 15:13:22 -04:00
return
}
2023-12-05 14:57:33 -05:00
// an empty request loads the model
2024-02-12 15:06:57 -08:00
// note: for a short while template was used in lieu
// of `raw` mode so we need to check for it too
2023-12-05 14:57:33 -05:00
if req . Prompt == "" && req . Template == "" && req . System == "" {
2023-12-01 11:37:17 -08:00
c . JSON ( http . StatusOK , api . GenerateResponse {
2023-12-09 02:05:43 -08:00
CreatedAt : time . Now ( ) . UTC ( ) ,
Model : req . Model ,
2023-12-15 14:25:12 -08:00
Done : true ,
} )
2023-12-04 18:01:06 -05:00
return
}
checkpointLoaded := time . Now ( )
2023-12-05 14:57:33 -05:00
var prompt string
switch {
case req . Raw :
prompt = req . Prompt
case req . Prompt != "" :
2024-02-12 15:06:57 -08:00
if req . Template == "" {
req . Template = model . Template
2023-12-05 14:57:33 -05:00
}
2024-02-12 15:06:57 -08:00
if req . System == "" {
req . System = model . System
}
slog . Debug ( "generate handler" , "prompt" , req . Prompt )
slog . Debug ( "generate handler" , "template" , req . Template )
slog . Debug ( "generate handler" , "system" , req . System )
var sb strings . Builder
2024-02-29 11:30:14 -08:00
for i := range req . Images {
fmt . Fprintf ( & sb , "[img-%d] " , i )
}
sb . WriteString ( req . Prompt )
p , err := Prompt ( req . Template , req . System , sb . String ( ) , "" , true )
if err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
sb . Reset ( )
2023-12-05 14:57:33 -05:00
if req . Context != nil {
2024-02-12 15:06:57 -08:00
prev , err := loaded . runner . Decode ( c . Request . Context ( ) , req . Context )
2023-12-05 14:57:33 -05:00
if err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-02-12 15:06:57 -08:00
sb . WriteString ( prev )
2024-01-28 14:15:56 -08:00
}
2024-02-12 15:06:57 -08:00
sb . WriteString ( p )
prompt = sb . String ( )
2023-08-08 15:13:22 -04:00
}
2024-01-31 16:47:26 -08:00
slog . Debug ( "generate handler" , "prompt" , prompt )
2024-01-28 15:22:35 -08:00
2023-12-04 18:01:06 -05:00
ch := make ( chan any )
2023-12-05 14:57:33 -05:00
var generated strings . Builder
2023-12-04 18:01:06 -05:00
go func ( ) {
defer close ( ch )
2023-12-05 14:57:33 -05:00
fn := func ( r llm . PredictResult ) {
// Update model expiration
2023-12-04 18:01:06 -05:00
loaded . expireAt = time . Now ( ) . Add ( sessionDuration )
loaded . expireTimer . Reset ( sessionDuration )
2023-12-05 14:57:33 -05:00
// Build up the full response
if _ , err := generated . WriteString ( r . Content ) ; err != nil {
ch <- gin . H { "error" : err . Error ( ) }
return
2023-12-04 18:01:06 -05:00
}
2023-12-05 14:57:33 -05:00
resp := api . GenerateResponse {
2023-12-10 11:42:15 -05:00
Model : req . Model ,
2023-12-14 12:15:50 -05:00
CreatedAt : time . Now ( ) . UTC ( ) ,
2023-12-09 02:05:43 -08:00
Done : r . Done ,
Response : r . Content ,
2023-12-05 14:57:33 -05:00
Metrics : api . Metrics {
PromptEvalCount : r . PromptEvalCount ,
PromptEvalDuration : r . PromptEvalDuration ,
EvalCount : r . EvalCount ,
EvalDuration : r . EvalDuration ,
} ,
2023-12-04 18:01:06 -05:00
}
2023-12-14 12:15:50 -05:00
if r . Done {
resp . TotalDuration = time . Since ( checkpointStart )
resp . LoadDuration = checkpointLoaded . Sub ( checkpointStart )
if ! req . Raw {
2024-02-12 15:06:57 -08:00
p , err := Prompt ( req . Template , req . System , req . Prompt , generated . String ( ) , false )
2023-12-22 17:07:05 -05:00
if err != nil {
2024-02-12 15:06:57 -08:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
2023-12-22 17:07:05 -05:00
return
}
2024-02-12 15:06:57 -08:00
// TODO (jmorganca): encode() should not strip special tokens
tokens , err := loaded . runner . Encode ( c . Request . Context ( ) , p )
2023-12-14 12:15:50 -05:00
if err != nil {
ch <- gin . H { "error" : err . Error ( ) }
return
}
2024-02-12 15:06:57 -08:00
resp . Context = append ( req . Context , tokens ... )
2023-12-05 14:57:33 -05:00
}
}
ch <- resp
2023-12-04 18:01:06 -05:00
}
2024-01-31 18:56:12 -08:00
var images [ ] llm . ImageData
2024-01-31 17:39:38 -08:00
for i := range req . Images {
2024-01-31 18:56:12 -08:00
images = append ( images , llm . ImageData {
ID : i ,
Data : req . Images [ i ] ,
} )
2024-01-31 17:39:38 -08:00
}
2023-12-05 14:57:33 -05:00
// Start prediction
predictReq := llm . PredictOpts {
2024-01-03 12:01:42 -05:00
Prompt : prompt ,
Format : req . Format ,
2024-01-31 17:39:38 -08:00
Images : images ,
2024-01-03 12:01:42 -05:00
Options : opts ,
2023-12-05 14:57:33 -05:00
}
if err := loaded . runner . Predict ( c . Request . Context ( ) , predictReq , fn ) ; err != nil {
2023-12-04 18:01:06 -05:00
ch <- gin . H { "error" : err . Error ( ) }
}
} ( )
if req . Stream != nil && ! * req . Stream {
2023-12-10 10:53:38 -05:00
// Accumulate responses into the final response
var final api . GenerateResponse
2023-12-05 14:57:33 -05:00
var sb strings . Builder
2023-12-04 18:01:06 -05:00
for resp := range ch {
2023-12-10 10:53:38 -05:00
switch r := resp . ( type ) {
case api . GenerateResponse :
sb . WriteString ( r . Response )
final = r
case gin . H :
if errorMsg , ok := r [ "error" ] . ( string ) ; ok {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : errorMsg } )
return
} else {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected error format in response" } )
return
}
default :
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected error" } )
2023-12-04 18:01:06 -05:00
return
}
}
2023-12-10 10:53:38 -05:00
final . Response = sb . String ( )
c . JSON ( http . StatusOK , final )
2023-12-04 18:01:06 -05:00
return
}
streamResponse ( c , ch )
}
2024-03-13 13:29:40 -07:00
func getDefaultSessionDuration ( ) time . Duration {
if t , exists := os . LookupEnv ( "OLLAMA_KEEP_ALIVE" ) ; exists {
v , err := strconv . Atoi ( t )
if err != nil {
d , err := time . ParseDuration ( t )
if err != nil {
return defaultSessionDuration
}
if d < 0 {
return time . Duration ( math . MaxInt64 )
}
return d
}
d := time . Duration ( v ) * time . Second
if d < 0 {
return time . Duration ( math . MaxInt64 )
}
return d
}
return defaultSessionDuration
}
2024-02-29 17:40:56 -08:00
func EmbeddingsHandler ( c * gin . Context ) {
2023-12-04 18:01:06 -05:00
loaded . mu . Lock ( )
defer loaded . mu . Unlock ( )
var req api . EmbeddingRequest
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
if req . Model == "" {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
return
}
2024-01-03 12:01:42 -05:00
model , err := GetModel ( req . Model )
2023-12-04 18:01:06 -05:00
if err != nil {
2023-12-05 14:57:33 -05:00
var pErr * fs . PathError
2024-01-03 12:01:42 -05:00
if errors . As ( err , & pErr ) {
2023-12-05 14:57:33 -05:00
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found, try pulling it first" , req . Model ) } )
2024-01-03 12:01:42 -05:00
return
}
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
opts , err := modelOptions ( model , req . Options )
if err != nil {
if errors . Is ( err , api . ErrInvalidOpts ) {
2023-12-05 14:57:33 -05:00
c . JSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2024-01-03 12:01:42 -05:00
return
2023-12-05 14:57:33 -05:00
}
2024-01-03 12:01:42 -05:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-01-26 14:28:02 -08:00
var sessionDuration time . Duration
if req . KeepAlive == nil {
2024-03-13 13:29:40 -07:00
sessionDuration = getDefaultSessionDuration ( )
2024-01-26 14:28:02 -08:00
} else {
sessionDuration = req . KeepAlive . Duration
}
2024-01-03 12:01:42 -05:00
if err := load ( c , model , opts , sessionDuration ) ; err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
2023-12-04 18:01:06 -05:00
return
}
2024-02-29 17:40:56 -08:00
// an empty request loads the model
if req . Prompt == "" {
c . JSON ( http . StatusOK , api . EmbeddingResponse { Embedding : [ ] float64 { } } )
2023-08-08 15:13:22 -04:00
return
}
2023-10-19 10:39:58 -04:00
embedding , err := loaded . runner . Embedding ( c . Request . Context ( ) , req . Prompt )
2023-08-08 15:13:22 -04:00
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "embedding generation failed: %v" , err ) )
2023-08-08 15:13:22 -04:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "failed to generate embedding" } )
return
}
resp := api . EmbeddingResponse {
Embedding : embedding ,
}
c . JSON ( http . StatusOK , resp )
}
2023-07-20 16:09:23 -07:00
func PullModelHandler ( c * gin . Context ) {
2023-07-11 11:54:22 -07:00
var req api . PullRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-07-11 11:54:22 -07:00
return
}
2024-01-11 14:07:54 -08:00
var model string
if req . Model != "" {
model = req . Model
} else if req . Name != "" {
model = req . Name
} else {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
2023-10-18 15:56:34 -07:00
return
}
2023-07-16 17:02:22 -07:00
ch := make ( chan any )
go func ( ) {
defer close ( ch )
2023-07-18 18:51:30 -07:00
fn := func ( r api . ProgressResponse ) {
ch <- r
2023-07-16 17:02:22 -07:00
}
2023-07-18 18:51:30 -07:00
2024-02-14 11:29:49 -08:00
regOpts := & registryOptions {
2023-07-21 15:42:19 -07:00
Insecure : req . Insecure ,
}
2023-07-25 17:08:51 -04:00
ctx , cancel := context . WithCancel ( c . Request . Context ( ) )
defer cancel ( )
2024-01-11 14:07:54 -08:00
if err := PullModel ( ctx , model , regOpts , fn ) ; err != nil {
2023-07-20 12:12:08 -07:00
ch <- gin . H { "error" : err . Error ( ) }
2023-07-16 17:02:22 -07:00
}
} ( )
2023-10-11 12:54:27 -04:00
if req . Stream != nil && ! * req . Stream {
waitForStream ( c , ch )
return
}
2023-07-16 17:02:22 -07:00
streamResponse ( c , ch )
}
2023-07-20 16:09:23 -07:00
func PushModelHandler ( c * gin . Context ) {
2023-07-16 17:02:22 -07:00
var req api . PushRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-07-11 11:54:22 -07:00
return
}
2023-07-06 10:40:11 -07:00
2024-01-11 14:07:54 -08:00
var model string
if req . Model != "" {
model = req . Model
} else if req . Name != "" {
model = req . Name
} else {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
2023-10-18 15:56:34 -07:00
return
}
2023-07-16 17:02:22 -07:00
ch := make ( chan any )
go func ( ) {
defer close ( ch )
2023-07-18 18:51:30 -07:00
fn := func ( r api . ProgressResponse ) {
ch <- r
2023-07-16 17:02:22 -07:00
}
2023-07-18 18:51:30 -07:00
2024-02-14 11:29:49 -08:00
regOpts := & registryOptions {
2023-07-21 15:42:19 -07:00
Insecure : req . Insecure ,
}
2023-10-09 10:24:27 -07:00
ctx , cancel := context . WithCancel ( c . Request . Context ( ) )
defer cancel ( )
2024-01-11 14:07:54 -08:00
if err := PushModel ( ctx , model , regOpts , fn ) ; err != nil {
2023-07-20 12:12:08 -07:00
ch <- gin . H { "error" : err . Error ( ) }
2023-07-16 17:02:22 -07:00
}
} ( )
2023-10-11 12:54:27 -04:00
if req . Stream != nil && ! * req . Stream {
waitForStream ( c , ch )
return
}
2023-07-16 17:02:22 -07:00
streamResponse ( c , ch )
}
2023-07-20 16:09:23 -07:00
func CreateModelHandler ( c * gin . Context ) {
2023-07-16 17:02:22 -07:00
var req api . CreateRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-07-12 19:07:15 -07:00
return
2023-07-16 17:02:22 -07:00
}
2024-01-11 14:07:54 -08:00
var model string
if req . Model != "" {
model = req . Model
} else if req . Name != "" {
model = req . Name
} else {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
2023-10-18 15:56:34 -07:00
return
}
2024-01-11 14:07:54 -08:00
if err := ParseModelPath ( model ) . Validate ( ) ; err != nil {
2023-11-29 15:54:29 -05:00
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-11-29 13:33:45 -05:00
return
}
2023-11-14 13:45:07 -08:00
if req . Path == "" && req . Modelfile == "" {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "path or modelfile are required" } )
2023-11-14 12:30:34 -08:00
return
}
2023-11-14 13:45:07 -08:00
var modelfile io . Reader = strings . NewReader ( req . Modelfile )
if req . Path != "" && req . Modelfile == "" {
2023-11-21 15:43:17 -05:00
mf , err := os . Open ( req . Path )
2023-11-14 13:45:07 -08:00
if err != nil {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : fmt . Sprintf ( "error reading modelfile: %s" , err ) } )
return
}
2023-11-21 15:43:17 -05:00
defer mf . Close ( )
2023-11-14 13:45:07 -08:00
2023-11-21 15:43:17 -05:00
modelfile = mf
2023-11-14 13:45:07 -08:00
}
2023-11-14 12:30:34 -08:00
commands , err := parser . Parse ( modelfile )
if err != nil {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
2023-07-11 11:54:22 -07:00
ch := make ( chan any )
2023-07-14 14:15:53 -07:00
go func ( ) {
defer close ( ch )
2023-07-25 14:25:13 -04:00
fn := func ( resp api . ProgressResponse ) {
ch <- resp
2023-07-16 17:02:22 -07:00
}
2023-07-25 17:08:51 -04:00
ctx , cancel := context . WithCancel ( c . Request . Context ( ) )
defer cancel ( )
2024-01-11 14:07:54 -08:00
if err := CreateModel ( ctx , model , filepath . Dir ( req . Path ) , commands , fn ) ; err != nil {
2023-07-20 12:12:08 -07:00
ch <- gin . H { "error" : err . Error ( ) }
2023-07-16 17:02:22 -07:00
}
2023-07-14 14:15:53 -07:00
} ( )
2023-07-07 15:29:17 -07:00
2023-10-11 12:54:27 -04:00
if req . Stream != nil && ! * req . Stream {
waitForStream ( c , ch )
return
}
2023-07-14 14:15:53 -07:00
streamResponse ( c , ch )
2023-07-05 15:37:33 -04:00
}
2023-07-20 16:09:23 -07:00
func DeleteModelHandler ( c * gin . Context ) {
var req api . DeleteRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-07-20 16:09:23 -07:00
return
}
2024-01-11 14:07:54 -08:00
var model string
if req . Model != "" {
model = req . Model
} else if req . Name != "" {
model = req . Name
} else {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
2023-10-18 15:56:34 -07:00
return
}
2024-01-11 14:07:54 -08:00
if err := DeleteModel ( model ) ; err != nil {
2023-07-21 23:02:12 -07:00
if os . IsNotExist ( err ) {
2024-01-11 14:07:54 -08:00
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found" , model ) } )
2023-07-21 23:02:12 -07:00
} else {
2023-07-20 16:09:23 -07:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
}
2023-07-21 23:02:12 -07:00
return
}
2023-09-26 17:28:14 -07:00
manifestsPath , err := GetManifestPath ( )
if err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
if err := PruneDirectory ( manifestsPath ) ; err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2023-09-11 11:46:35 -07:00
c . JSON ( http . StatusOK , nil )
2023-07-20 16:09:23 -07:00
}
2023-09-06 11:04:17 -07:00
func ShowModelHandler ( c * gin . Context ) {
var req api . ShowRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-09-06 11:04:17 -07:00
return
}
2024-01-11 14:07:54 -08:00
if req . Model != "" {
2024-01-18 15:36:50 -08:00
// noop
2024-01-11 14:07:54 -08:00
} else if req . Name != "" {
2024-01-18 15:36:50 -08:00
req . Model = req . Name
2024-01-11 14:07:54 -08:00
} else {
2024-01-04 17:23:11 -08:00
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
2023-10-18 15:56:34 -07:00
return
}
2024-01-04 17:23:11 -08:00
resp , err := GetModelInfo ( req )
2023-09-06 11:04:17 -07:00
if err != nil {
if os . IsNotExist ( err ) {
2024-01-18 15:36:50 -08:00
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found" , req . Model ) } )
2023-09-06 11:04:17 -07:00
} else {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
}
return
}
c . JSON ( http . StatusOK , resp )
}
2024-01-04 17:23:11 -08:00
func GetModelInfo ( req api . ShowRequest ) ( * api . ShowResponse , error ) {
model , err := GetModel ( req . Model )
2023-09-06 11:04:17 -07:00
if err != nil {
return nil , err
}
2023-12-11 13:56:22 -08:00
modelDetails := api . ModelDetails {
2024-01-25 12:12:36 -08:00
ParentModel : model . ParentModel ,
2023-12-11 13:56:22 -08:00
Format : model . Config . ModelFormat ,
Family : model . Config . ModelFamily ,
Families : model . Config . ModelFamilies ,
ParameterSize : model . Config . ModelType ,
QuantizationLevel : model . Config . FileType ,
}
2024-01-04 17:23:11 -08:00
if req . System != "" {
model . System = req . System
}
if req . Template != "" {
model . Template = req . Template
}
2024-01-25 12:12:36 -08:00
msgs := make ( [ ] api . Message , 0 )
for _ , msg := range model . Messages {
msgs = append ( msgs , api . Message { Role : msg . Role , Content : msg . Content } )
}
2023-09-06 11:04:17 -07:00
resp := & api . ShowResponse {
License : strings . Join ( model . License , "\n" ) ,
System : model . System ,
Template : model . Template ,
2023-12-11 13:56:22 -08:00
Details : modelDetails ,
2024-01-25 12:12:36 -08:00
Messages : msgs ,
2023-09-06 11:04:17 -07:00
}
var params [ ] string
cs := 30
for k , v := range model . Options {
switch val := v . ( type ) {
case [ ] interface { } :
for _ , nv := range val {
2024-01-16 10:34:44 -08:00
params = append ( params , fmt . Sprintf ( "%-*s %#v" , cs , k , nv ) )
2023-09-06 11:04:17 -07:00
}
2024-01-16 10:34:44 -08:00
default :
params = append ( params , fmt . Sprintf ( "%-*s %#v" , cs , k , v ) )
2023-09-06 11:04:17 -07:00
}
}
resp . Parameters = strings . Join ( params , "\n" )
2024-01-04 17:23:11 -08:00
for k , v := range req . Options {
if _ , ok := req . Options [ k ] ; ok {
model . Options [ k ] = v
}
}
mf , err := ShowModelfile ( model )
if err != nil {
return nil , err
}
resp . Modelfile = mf
2023-09-06 11:04:17 -07:00
return resp , nil
}
2023-07-20 16:09:23 -07:00
func ListModelsHandler ( c * gin . Context ) {
2023-10-17 19:02:43 +02:00
models := make ( [ ] api . ModelResponse , 0 )
2023-12-15 15:50:51 -08:00
manifestsPath , err := GetManifestPath ( )
2023-07-18 09:09:45 -07:00
if err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2023-08-30 14:14:12 -04:00
2023-12-11 13:56:22 -08:00
modelResponse := func ( modelName string ) ( api . ModelResponse , error ) {
model , err := GetModel ( modelName )
if err != nil {
return api . ModelResponse { } , err
}
modelDetails := api . ModelDetails {
Format : model . Config . ModelFormat ,
Family : model . Config . ModelFamily ,
Families : model . Config . ModelFamilies ,
ParameterSize : model . Config . ModelType ,
QuantizationLevel : model . Config . FileType ,
}
return api . ModelResponse {
2024-01-18 14:32:55 -08:00
Model : model . ShortName ,
2023-12-11 13:56:22 -08:00
Name : model . ShortName ,
Size : model . Size ,
Digest : model . Digest ,
Details : modelDetails ,
} , nil
}
2023-08-30 14:14:12 -04:00
walkFunc := func ( path string , info os . FileInfo , _ error ) error {
2023-07-18 09:09:45 -07:00
if ! info . IsDir ( ) {
2023-12-15 15:50:51 -08:00
path , tag := filepath . Split ( path )
model := strings . Trim ( strings . TrimPrefix ( path , manifestsPath ) , string ( os . PathSeparator ) )
modelPath := strings . Join ( [ ] string { model , tag } , ":" )
canonicalModelPath := strings . ReplaceAll ( modelPath , string ( os . PathSeparator ) , "/" )
2023-08-21 21:56:56 -07:00
2023-12-15 15:50:51 -08:00
resp , err := modelResponse ( canonicalModelPath )
2023-07-18 09:09:45 -07:00
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "skipping file: %s" , canonicalModelPath ) )
2023-12-15 14:07:34 -08:00
// nolint: nilerr
2023-07-18 12:39:08 -07:00
return nil
2023-07-18 09:09:45 -07:00
}
2023-08-30 14:14:12 -04:00
2023-12-11 13:56:22 -08:00
resp . ModifiedAt = info . ModTime ( )
models = append ( models , resp )
2023-07-18 09:09:45 -07:00
}
2023-08-30 14:14:12 -04:00
2023-07-18 09:09:45 -07:00
return nil
2023-08-30 14:14:12 -04:00
}
2023-12-15 15:50:51 -08:00
if err := filepath . Walk ( manifestsPath , walkFunc ) ; err != nil {
2023-07-18 09:09:45 -07:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2023-07-19 15:00:28 -07:00
c . JSON ( http . StatusOK , api . ListResponse { Models : models } )
2023-07-18 09:09:45 -07:00
}
2023-07-24 11:27:28 -04:00
func CopyModelHandler ( c * gin . Context ) {
var req api . CopyRequest
2023-10-18 16:08:42 -07:00
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2023-07-24 11:27:28 -04:00
return
}
2023-10-18 15:56:34 -07:00
if req . Source == "" || req . Destination == "" {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "source add destination are required" } )
return
}
2023-11-29 15:54:29 -05:00
if err := ParseModelPath ( req . Destination ) . Validate ( ) ; err != nil {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
2023-07-24 11:27:28 -04:00
if err := CopyModel ( req . Source , req . Destination ) ; err != nil {
if os . IsNotExist ( err ) {
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found" , req . Source ) } )
} else {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
}
return
}
}
2023-11-15 10:59:38 -08:00
func HeadBlobHandler ( c * gin . Context ) {
2023-11-14 14:07:40 -08:00
path , err := GetBlobsPath ( c . Param ( "digest" ) )
if err != nil {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
if _ , err := os . Stat ( path ) ; err != nil {
c . AbortWithStatusJSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "blob %q not found" , c . Param ( "digest" ) ) } )
return
}
2023-11-15 13:55:37 -08:00
c . Status ( http . StatusOK )
2023-11-14 14:07:40 -08:00
}
func CreateBlobHandler ( c * gin . Context ) {
2023-11-24 12:01:23 -08:00
layer , err := NewLayer ( c . Request . Body , "" )
2023-11-17 15:21:57 -08:00
if err != nil {
c . AbortWithStatusJSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2023-11-24 12:01:23 -08:00
if layer . Digest != c . Param ( "digest" ) {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : fmt . Sprintf ( "digest mismatch, expected %q, got %q" , c . Param ( "digest" ) , layer . Digest ) } )
2023-11-14 14:07:40 -08:00
return
}
2023-11-24 12:01:23 -08:00
if _ , err := layer . Commit ( ) ; err != nil {
2023-11-14 14:07:40 -08:00
c . AbortWithStatusJSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2023-11-15 13:55:37 -08:00
c . Status ( http . StatusCreated )
2023-11-14 14:07:40 -08:00
}
2023-09-21 09:42:16 -07:00
var defaultAllowOrigins = [ ] string {
"localhost" ,
"127.0.0.1" ,
"0.0.0.0" ,
}
2024-03-09 00:22:08 -08:00
func isLocalIP ( ip netip . Addr ) bool {
if interfaces , err := net . Interfaces ( ) ; err == nil {
for _ , iface := range interfaces {
addrs , err := iface . Addrs ( )
if err != nil {
continue
}
for _ , a := range addrs {
if parsed , _ , err := net . ParseCIDR ( a . String ( ) ) ; err == nil {
if parsed . String ( ) == ip . String ( ) {
return true
}
}
}
}
}
return false
}
2024-03-08 22:23:47 -08:00
func allowedHost ( host string ) bool {
2024-03-09 00:22:08 -08:00
if host == "" || host == "localhost" {
2024-03-08 22:23:47 -08:00
return true
}
if hostname , err := os . Hostname ( ) ; err == nil && host == hostname {
return true
}
var tlds = [ ] string {
2024-03-08 23:23:59 -08:00
"localhost" ,
"local" ,
"internal" ,
2023-12-14 16:47:40 -08:00
}
2023-10-30 11:10:18 -04:00
2024-03-08 23:29:53 -08:00
// check if the host is a local TLD
2024-03-08 22:23:47 -08:00
for _ , tld := range tlds {
if strings . HasSuffix ( host , "." + tld ) {
return true
}
}
2024-03-08 23:29:53 -08:00
return false
2024-03-08 23:23:59 -08:00
}
2024-03-08 22:23:47 -08:00
2024-03-08 23:23:59 -08:00
func allowedHostsMiddleware ( addr net . Addr ) gin . HandlerFunc {
return func ( c * gin . Context ) {
if addr == nil {
2024-03-08 22:23:47 -08:00
c . Next ( )
return
}
2024-03-09 00:22:08 -08:00
if addr , err := netip . ParseAddrPort ( addr . String ( ) ) ; err == nil && ! addr . Addr ( ) . IsLoopback ( ) {
2024-03-08 22:23:47 -08:00
c . Next ( )
return
}
host , _ , err := net . SplitHostPort ( c . Request . Host )
if err != nil {
host = c . Request . Host
}
2024-03-08 23:23:59 -08:00
if addr , err := netip . ParseAddr ( host ) ; err == nil {
2024-03-09 00:22:08 -08:00
if addr . IsLoopback ( ) || addr . IsPrivate ( ) || addr . IsUnspecified ( ) || isLocalIP ( addr ) {
2024-03-08 23:23:59 -08:00
c . Next ( )
return
}
}
2024-03-08 22:23:47 -08:00
if allowedHost ( host ) {
c . Next ( )
return
}
c . AbortWithStatus ( http . StatusForbidden )
}
2023-12-14 16:47:40 -08:00
}
2023-10-30 11:10:18 -04:00
2023-12-14 16:47:40 -08:00
func ( s * Server ) GenerateRoutes ( ) http . Handler {
var origins [ ] string
if o := os . Getenv ( "OLLAMA_ORIGINS" ) ; o != "" {
origins = strings . Split ( o , "," )
2023-10-30 11:10:18 -04:00
}
2023-07-21 18:01:24 -07:00
config := cors . DefaultConfig ( )
config . AllowWildcard = true
2024-01-05 08:55:47 +07:00
config . AllowBrowserExtensions = true
2023-09-21 09:42:16 -07:00
2023-12-14 16:47:40 -08:00
config . AllowOrigins = origins
2023-09-21 09:42:16 -07:00
for _ , allowOrigin := range defaultAllowOrigins {
config . AllowOrigins = append ( config . AllowOrigins ,
fmt . Sprintf ( "http://%s" , allowOrigin ) ,
fmt . Sprintf ( "https://%s" , allowOrigin ) ,
fmt . Sprintf ( "http://%s:*" , allowOrigin ) ,
fmt . Sprintf ( "https://%s:*" , allowOrigin ) ,
)
}
2023-07-21 18:01:24 -07:00
2023-07-05 15:37:33 -04:00
r := gin . Default ( )
2023-09-21 20:38:49 +01:00
r . Use (
cors . New ( config ) ,
2024-03-08 22:23:47 -08:00
allowedHostsMiddleware ( s . addr ) ,
2023-09-21 20:38:49 +01:00
)
2023-07-05 15:37:33 -04:00
2023-07-20 16:09:23 -07:00
r . POST ( "/api/pull" , PullModelHandler )
r . POST ( "/api/generate" , GenerateHandler )
2023-12-05 14:57:33 -05:00
r . POST ( "/api/chat" , ChatHandler )
2024-02-29 17:40:56 -08:00
r . POST ( "/api/embeddings" , EmbeddingsHandler )
2023-07-20 16:09:23 -07:00
r . POST ( "/api/create" , CreateModelHandler )
r . POST ( "/api/push" , PushModelHandler )
2023-07-24 11:27:28 -04:00
r . POST ( "/api/copy" , CopyModelHandler )
2023-07-20 16:09:23 -07:00
r . DELETE ( "/api/delete" , DeleteModelHandler )
2023-09-06 11:04:17 -07:00
r . POST ( "/api/show" , ShowModelHandler )
2023-11-14 14:07:40 -08:00
r . POST ( "/api/blobs/:digest" , CreateBlobHandler )
2023-11-15 15:22:12 -08:00
r . HEAD ( "/api/blobs/:digest" , HeadBlobHandler )
2023-07-03 15:22:44 -04:00
2024-02-07 17:24:29 -05:00
// Compatibility endpoints
r . POST ( "/v1/chat/completions" , openai . Middleware ( ) , ChatHandler )
2023-09-21 16:38:03 -07:00
for _ , method := range [ ] string { http . MethodGet , http . MethodHead } {
r . Handle ( method , "/" , func ( c * gin . Context ) {
c . String ( http . StatusOK , "Ollama is running" )
} )
r . Handle ( method , "/api/tags" , ListModelsHandler )
2023-10-12 15:45:07 -07:00
r . Handle ( method , "/api/version" , func ( c * gin . Context ) {
c . JSON ( http . StatusOK , gin . H { "version" : version . Version } )
} )
2023-09-21 16:38:03 -07:00
}
2023-12-14 16:47:40 -08:00
return r
}
func Serve ( ln net . Listener ) error {
2024-01-31 14:59:32 -08:00
level := slog . LevelInfo
2024-01-18 10:52:01 -08:00
if debug := os . Getenv ( "OLLAMA_DEBUG" ) ; debug != "" {
2024-01-31 14:59:32 -08:00
level = slog . LevelDebug
2024-01-18 10:52:01 -08:00
}
2024-01-31 14:59:32 -08:00
handler := slog . NewTextHandler ( os . Stderr , & slog . HandlerOptions {
Level : level ,
AddSource : true ,
ReplaceAttr : func ( _ [ ] string , attr slog . Attr ) slog . Attr {
if attr . Key == slog . SourceKey {
source := attr . Value . Any ( ) . ( * slog . Source )
source . File = filepath . Base ( source . File )
}
return attr
} ,
} )
slog . SetDefault ( slog . New ( handler ) )
2024-03-14 20:18:06 -07:00
blobsDir , err := GetBlobsPath ( "" )
if err != nil {
return err
}
if err := fixBlobs ( blobsDir ) ; err != nil {
return err
}
2023-12-14 16:47:40 -08:00
if noprune := os . Getenv ( "OLLAMA_NOPRUNE" ) ; noprune == "" {
// clean up unused layers and manifests
if err := PruneLayers ( ) ; err != nil {
return err
}
manifestsPath , err := GetManifestPath ( )
if err != nil {
return err
}
if err := PruneDirectory ( manifestsPath ) ; err != nil {
return err
}
}
2024-03-08 22:23:47 -08:00
s := & Server { addr : ln . Addr ( ) }
2023-12-14 16:47:40 -08:00
r := s . GenerateRoutes ( )
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "Listening on %s (version %s)" , ln . Addr ( ) , version . Version ) )
2023-12-14 16:47:40 -08:00
srvr := & http . Server {
2023-07-03 15:22:44 -04:00
Handler : r ,
}
2023-08-30 16:35:03 -04:00
// listen for a ctrl+c and stop any loaded llm
signals := make ( chan os . Signal , 1 )
2023-09-21 20:38:49 +01:00
signal . Notify ( signals , syscall . SIGINT , syscall . SIGTERM )
2023-08-30 16:35:03 -04:00
go func ( ) {
<- signals
2023-10-19 10:39:58 -04:00
if loaded . runner != nil {
loaded . runner . Close ( )
2023-09-22 19:41:52 +01:00
}
2024-03-08 09:45:55 -08:00
gpu . Cleanup ( )
2023-08-30 16:35:03 -04:00
os . Exit ( 0 )
} ( )
2024-02-15 17:15:09 -08:00
if err := llm . Init ( ) ; err != nil {
2023-11-29 11:00:37 -08:00
return fmt . Errorf ( "unable to initialize llm library %w" , err )
}
if runtime . GOOS == "linux" { // TODO - windows too
2023-09-12 11:04:35 -04:00
// check compatibility to log warnings
2023-11-29 11:00:37 -08:00
if _ , err := gpu . CheckVRAM ( ) ; err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( err . Error ( ) )
2023-09-12 11:04:35 -04:00
}
}
2023-12-14 16:47:40 -08:00
return srvr . Serve ( ln )
2023-07-03 15:22:44 -04:00
}
2023-07-06 10:40:11 -07:00
2023-10-11 12:54:27 -04:00
func waitForStream ( c * gin . Context , ch chan interface { } ) {
c . Header ( "Content-Type" , "application/json" )
for resp := range ch {
switch r := resp . ( type ) {
case api . ProgressResponse :
if r . Status == "success" {
c . JSON ( http . StatusOK , r )
return
}
case gin . H :
if errorMsg , ok := r [ "error" ] . ( string ) ; ok {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : errorMsg } )
return
} else {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected error format in progress response" } )
return
}
default :
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected progress response" } )
return
}
}
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected end of progress response" } )
}
2023-07-14 14:15:53 -07:00
func streamResponse ( c * gin . Context , ch chan any ) {
2023-08-08 21:38:10 -07:00
c . Header ( "Content-Type" , "application/x-ndjson" )
2023-07-11 11:54:22 -07:00
c . Stream ( func ( w io . Writer ) bool {
val , ok := <- ch
if ! ok {
return false
}
bts , err := json . Marshal ( val )
if err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "streamResponse: json.Marshal failed with %s" , err ) )
2023-07-11 11:54:22 -07:00
return false
}
2023-09-30 00:45:52 -04:00
// Delineate chunks with new-line delimiter
2023-07-11 11:54:22 -07:00
bts = append ( bts , '\n' )
if _ , err := w . Write ( bts ) ; err != nil {
2024-01-18 10:52:01 -08:00
slog . Info ( fmt . Sprintf ( "streamResponse: w.Write failed with %s" , err ) )
2023-07-11 11:54:22 -07:00
return false
}
return true
} )
}
2023-12-05 14:57:33 -05:00
2024-02-12 15:06:57 -08:00
// ChatPrompt builds up a prompt from a series of messages for the currently `loaded` model
2024-02-16 14:42:43 -05:00
func chatPrompt ( ctx context . Context , template string , messages [ ] api . Message , numCtx int ) ( string , error ) {
2024-02-12 15:06:57 -08:00
encode := func ( s string ) ( [ ] int , error ) {
return loaded . runner . Encode ( ctx , s )
}
2024-02-16 14:42:43 -05:00
prompt , err := ChatPrompt ( template , messages , numCtx , encode )
2024-02-12 15:06:57 -08:00
if err != nil {
return "" , err
}
return prompt , nil
}
2023-12-05 14:57:33 -05:00
func ChatHandler ( c * gin . Context ) {
loaded . mu . Lock ( )
defer loaded . mu . Unlock ( )
checkpointStart := time . Now ( )
var req api . ChatRequest
err := c . ShouldBindJSON ( & req )
switch {
case errors . Is ( err , io . EOF ) :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "missing request body" } )
return
case err != nil :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
// validate the request
switch {
case req . Model == "" :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "model is required" } )
return
case len ( req . Format ) > 0 && req . Format != "json" :
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "format must be json" } )
return
}
2024-01-03 12:01:42 -05:00
model , err := GetModel ( req . Model )
2023-12-05 14:57:33 -05:00
if err != nil {
var pErr * fs . PathError
2024-01-03 12:01:42 -05:00
if errors . As ( err , & pErr ) {
2023-12-05 14:57:33 -05:00
c . JSON ( http . StatusNotFound , gin . H { "error" : fmt . Sprintf ( "model '%s' not found, try pulling it first" , req . Model ) } )
2024-01-03 12:01:42 -05:00
return
}
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-02-20 21:37:29 -05:00
if model . IsEmbedding ( ) {
2024-02-20 21:53:45 -05:00
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "embedding models do not support chat" } )
2024-02-20 21:37:29 -05:00
return
}
2024-01-03 12:01:42 -05:00
opts , err := modelOptions ( model , req . Options )
if err != nil {
if errors . Is ( err , api . ErrInvalidOpts ) {
2023-12-05 14:57:33 -05:00
c . JSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
2024-01-03 12:01:42 -05:00
return
2023-12-05 14:57:33 -05:00
}
2024-01-03 12:01:42 -05:00
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
return
}
2024-01-26 14:28:02 -08:00
var sessionDuration time . Duration
if req . KeepAlive == nil {
2024-03-13 13:29:40 -07:00
sessionDuration = getDefaultSessionDuration ( )
2024-01-26 14:28:02 -08:00
} else {
sessionDuration = req . KeepAlive . Duration
}
2024-01-03 12:01:42 -05:00
if err := load ( c , model , opts , sessionDuration ) ; err != nil {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : err . Error ( ) } )
2023-12-05 14:57:33 -05:00
return
}
checkpointLoaded := time . Now ( )
2024-02-16 14:42:43 -05:00
// if the first message is not a system message, then add the model's default system message
if len ( req . Messages ) > 0 && req . Messages [ 0 ] . Role != "system" {
req . Messages = append ( [ ] api . Message {
{
Role : "system" ,
Content : model . System ,
} ,
} , req . Messages ... )
}
prompt , err := chatPrompt ( c . Request . Context ( ) , model . Template , req . Messages , opts . NumCtx )
2023-12-05 14:57:33 -05:00
if err != nil {
c . JSON ( http . StatusBadRequest , gin . H { "error" : err . Error ( ) } )
return
}
2024-01-31 17:39:38 -08:00
2024-02-07 19:30:33 -05:00
// an empty request loads the model
2024-02-12 15:06:57 -08:00
if len ( req . Messages ) == 0 || prompt == "" {
2024-02-07 19:30:33 -05:00
resp := api . ChatResponse {
CreatedAt : time . Now ( ) . UTC ( ) ,
Model : req . Model ,
Done : true ,
Message : api . Message { Role : "assistant" } ,
}
c . JSON ( http . StatusOK , resp )
return
}
2024-02-12 15:06:57 -08:00
// only send images that are in the prompt
var i int
var images [ ] llm . ImageData
for _ , m := range req . Messages {
for _ , img := range m . Images {
if ! isSupportedImageType ( img ) {
c . AbortWithStatusJSON ( http . StatusBadRequest , gin . H { "error" : "unsupported image format" } )
return
}
if strings . Contains ( prompt , fmt . Sprintf ( "[img-%d]" , i ) ) {
images = append ( images , llm . ImageData { Data : img , ID : i } )
}
i += 1
}
}
slog . Debug ( "chat handler" , "prompt" , prompt , "images" , len ( images ) )
2024-01-28 15:22:35 -08:00
2023-12-05 14:57:33 -05:00
ch := make ( chan any )
go func ( ) {
defer close ( ch )
fn := func ( r llm . PredictResult ) {
// Update model expiration
loaded . expireAt = time . Now ( ) . Add ( sessionDuration )
loaded . expireTimer . Reset ( sessionDuration )
resp := api . ChatResponse {
2023-12-10 11:42:15 -05:00
Model : req . Model ,
2023-12-14 12:15:50 -05:00
CreatedAt : time . Now ( ) . UTC ( ) ,
2023-12-18 14:23:38 -05:00
Message : api . Message { Role : "assistant" , Content : r . Content } ,
2023-12-05 14:57:33 -05:00
Done : r . Done ,
Metrics : api . Metrics {
PromptEvalCount : r . PromptEvalCount ,
PromptEvalDuration : r . PromptEvalDuration ,
EvalCount : r . EvalCount ,
EvalDuration : r . EvalDuration ,
} ,
}
2023-12-14 12:15:50 -05:00
if r . Done {
resp . TotalDuration = time . Since ( checkpointStart )
resp . LoadDuration = checkpointLoaded . Sub ( checkpointStart )
2023-12-05 14:57:33 -05:00
}
ch <- resp
}
// Start prediction
predictReq := llm . PredictOpts {
2024-01-03 12:01:42 -05:00
Prompt : prompt ,
Format : req . Format ,
2024-01-31 19:18:25 -08:00
Images : images ,
2024-01-03 12:01:42 -05:00
Options : opts ,
2023-12-05 14:57:33 -05:00
}
if err := loaded . runner . Predict ( c . Request . Context ( ) , predictReq , fn ) ; err != nil {
ch <- gin . H { "error" : err . Error ( ) }
}
} ( )
if req . Stream != nil && ! * req . Stream {
2023-12-10 10:53:38 -05:00
// Accumulate responses into the final response
var final api . ChatResponse
2023-12-05 14:57:33 -05:00
var sb strings . Builder
for resp := range ch {
2023-12-10 10:53:38 -05:00
switch r := resp . ( type ) {
case api . ChatResponse :
2023-12-18 14:23:38 -05:00
sb . WriteString ( r . Message . Content )
2023-12-10 10:53:38 -05:00
final = r
case gin . H :
if errorMsg , ok := r [ "error" ] . ( string ) ; ok {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : errorMsg } )
return
} else {
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected error format in response" } )
return
}
default :
c . JSON ( http . StatusInternalServerError , gin . H { "error" : "unexpected error" } )
return
2023-12-05 14:57:33 -05:00
}
}
2023-12-10 10:53:38 -05:00
2023-12-18 14:23:38 -05:00
final . Message = api . Message { Role : "assistant" , Content : sb . String ( ) }
2023-12-10 10:53:38 -05:00
c . JSON ( http . StatusOK , final )
2023-12-05 14:57:33 -05:00
return
}
streamResponse ( c , ch )
}