models.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. package server
  2. import (
  3. "encoding/json"
  4. "errors"
  5. "fmt"
  6. "io"
  7. "net/http"
  8. "os"
  9. "path/filepath"
  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 filepath.Join(home, ".ollama", "models", m.Name+".bin")
  31. }
  32. func (m *Model) TempFile() string {
  33. fullName := m.FullName()
  34. return filepath.Join(
  35. filepath.Dir(fullName),
  36. fmt.Sprintf(".%s.part", filepath.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. var size int64
  70. // completed file doesn't exist, check partial file
  71. fi, err := os.Stat(model.TempFile())
  72. switch {
  73. case errors.Is(err, os.ErrNotExist):
  74. // noop, file doesn't exist so create it
  75. case err != nil:
  76. return fmt.Errorf("stat: %w", err)
  77. default:
  78. size = fi.Size()
  79. }
  80. req.Header.Add("Range", fmt.Sprintf("bytes=%d-", size))
  81. resp, err := client.Do(req)
  82. if err != nil {
  83. return fmt.Errorf("failed to download model: %w", err)
  84. }
  85. defer resp.Body.Close()
  86. if resp.StatusCode >= 400 {
  87. return fmt.Errorf("failed to download model: %s", resp.Status)
  88. }
  89. out, err := os.OpenFile(model.TempFile(), os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0o644)
  90. if err != nil {
  91. panic(err)
  92. }
  93. defer out.Close()
  94. remaining, _ := strconv.ParseInt(resp.Header.Get("Content-Length"), 10, 64)
  95. completed := size
  96. total := remaining + completed
  97. for {
  98. fn(total, completed)
  99. if completed >= total {
  100. return os.Rename(model.TempFile(), model.FullName())
  101. }
  102. n, err := io.CopyN(out, resp.Body, 8192)
  103. if err != nil && !errors.Is(err, io.EOF) {
  104. return err
  105. }
  106. completed += n
  107. }
  108. }