models.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. package server
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "os"
  9. "path"
  10. "strconv"
  11. )
  12. const directoryURL = "https://ollama.ai/api/models"
  13. type Model struct {
  14. Name string `json:"name"`
  15. DisplayName string `json:"display_name"`
  16. Parameters string `json:"parameters"`
  17. URL string `json:"url"`
  18. ShortDescription string `json:"short_description"`
  19. Description string `json:"description"`
  20. PublishedBy string `json:"published_by"`
  21. OriginalAuthor string `json:"original_author"`
  22. OriginalURL string `json:"original_url"`
  23. License string `json:"license"`
  24. }
  25. func (m *Model) FullName() string {
  26. home, err := os.UserHomeDir()
  27. if err != nil {
  28. panic(err)
  29. }
  30. return path.Join(home, ".ollama", "models", m.Name+".bin")
  31. }
  32. func (m *Model) TempFile() string {
  33. fullName := m.FullName()
  34. return path.Join(
  35. path.Dir(fullName),
  36. fmt.Sprintf(".%s.part", path.Base(fullName)),
  37. )
  38. }
  39. func getRemote(model string) (*Model, error) {
  40. // resolve the model download from our directory
  41. resp, err := http.Get(directoryURL)
  42. if err != nil {
  43. return nil, fmt.Errorf("failed to get directory: %w", err)
  44. }
  45. defer resp.Body.Close()
  46. body, err := io.ReadAll(resp.Body)
  47. if err != nil {
  48. return nil, fmt.Errorf("failed to read directory: %w", err)
  49. }
  50. var models []Model
  51. err = json.Unmarshal(body, &models)
  52. if err != nil {
  53. return nil, fmt.Errorf("failed to parse directory: %w", err)
  54. }
  55. for _, m := range models {
  56. if m.Name == model {
  57. return &m, nil
  58. }
  59. }
  60. return nil, fmt.Errorf("model not found in directory: %s", model)
  61. }
  62. func saveModel(model *Model, fn func(total, completed int64)) error {
  63. // this models cache directory is created by the server on startup
  64. client := &http.Client{}
  65. req, err := http.NewRequest("GET", model.URL, nil)
  66. if err != nil {
  67. return fmt.Errorf("failed to download model: %w", err)
  68. }
  69. // check if completed file exists
  70. fi, err := os.Stat(model.FullName())
  71. switch {
  72. case errors.Is(err, os.ErrNotExist):
  73. // noop, file doesn't exist so create it
  74. case err != nil:
  75. return fmt.Errorf("stat: %w", err)
  76. default:
  77. fn(fi.Size(), fi.Size())
  78. return nil
  79. }
  80. var size int64
  81. // completed file doesn't exist, check partial file
  82. fi, err = os.Stat(model.TempFile())
  83. switch {
  84. case errors.Is(err, os.ErrNotExist):
  85. // noop, file doesn't exist so create it
  86. case err != nil:
  87. return fmt.Errorf("stat: %w", err)
  88. default:
  89. size = fi.Size()
  90. }
  91. req.Header.Add("Range", fmt.Sprintf("bytes=%d-", size))
  92. resp, err := client.Do(req)
  93. if err != nil {
  94. return fmt.Errorf("failed to download model: %w", err)
  95. }
  96. defer resp.Body.Close()
  97. if resp.StatusCode >= 400 {
  98. return fmt.Errorf("failed to download model: %s", resp.Status)
  99. }
  100. out, err := os.OpenFile(model.TempFile(), os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0o644)
  101. if err != nil {
  102. panic(err)
  103. }
  104. defer out.Close()
  105. remaining, _ := strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
  106. completed := size
  107. total := remaining + completed
  108. for {
  109. fn(total, completed)
  110. if completed >= total {
  111. return os.Rename(model.TempFile(), model.FullName())
  112. }
  113. n , err := io.CopyN(out, resp.Body, 8192)
  114. if err != nil && !errors.Is(err, io.EOF) {
  115. return err
  116. }
  117. completed += n
  118. }
  119. }