build_test.go 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. package build
  2. import (
  3. "errors"
  4. "os"
  5. "path/filepath"
  6. "testing"
  7. "github.com/ollama/ollama/x/encoding/gguf"
  8. "github.com/ollama/ollama/x/model"
  9. )
  10. const qualifiedRef = "x/y/z:latest+Q4_0"
  11. func TestServerBuildErrors(t *testing.T) {
  12. dir := t.TempDir()
  13. s, err := Open(dir)
  14. if err != nil {
  15. t.Fatal(err)
  16. }
  17. t.Run("unqualified ref", func(t *testing.T) {
  18. err := s.Build("x", model.File{})
  19. if !errors.Is(err, ErrIncompleteRef) {
  20. t.Fatalf("Build() err = %v; want unqualified ref", err)
  21. }
  22. })
  23. t.Run("FROM pragma missing", func(t *testing.T) {
  24. err := s.Build(qualifiedRef, model.File{})
  25. var e *model.FileError
  26. if !errors.As(err, &e) {
  27. t.Fatalf("unexpected error: %v", err)
  28. }
  29. if e.Pragma != "FROM" {
  30. t.Errorf("e.Pragma = %s; want FROM", e.Pragma)
  31. }
  32. if e.Message != "missing" {
  33. t.Errorf("e.Message = %s; want missing", e.Message)
  34. }
  35. })
  36. t.Run("FROM file not found", func(t *testing.T) {
  37. err := s.Build(qualifiedRef, model.File{From: "bar"})
  38. if !errors.Is(err, os.ErrNotExist) {
  39. t.Fatalf("Build() err = %v; want file not found", err)
  40. }
  41. })
  42. t.Run("FROM gguf", func(t *testing.T) {
  43. w := newWorkDir(t)
  44. // Write a gguf file without general.file_type metadata.
  45. w.write("gguf", ""+
  46. "GGUF"+ // magic
  47. "\x03\x00\x00\x00"+ // version
  48. "\x00\x00\x00\x00\x00\x00\x00\x00"+ // numMetaValues
  49. "\x00\x00\x00\x00\x00\x00\x00\x00"+ // numTensors
  50. "",
  51. )
  52. err := s.Build(qualifiedRef, model.File{From: w.fileName("gguf")})
  53. if !errors.Is(err, ErrMissingFileType) {
  54. t.Fatalf("Build() err = %#v; want missing file type", err)
  55. }
  56. })
  57. t.Run("FROM obscure dir", func(t *testing.T) {
  58. w := newWorkDir(t)
  59. w.mkdirAll("unknown")
  60. if err := s.Build(qualifiedRef, model.File{From: w.fileName("unknown")}); err != ErrUnsupportedModelFormat {
  61. t.Fatalf("Build() err = %#v; want unsupported model type", err)
  62. }
  63. })
  64. t.Run("FROM unsupported model type", func(t *testing.T) {
  65. w := newWorkDir(t)
  66. from := w.write("unknown", "unknown content")
  67. err := s.Build(qualifiedRef, model.File{From: from})
  68. if !errors.Is(err, ErrUnsupportedModelFormat) {
  69. t.Fatalf("Build() err = %#v; want unsupported model type", err)
  70. }
  71. })
  72. }
  73. func TestBuildBasicGGUF(t *testing.T) {
  74. w := newWorkDir(t)
  75. w.write("gguf", ""+
  76. "GGUF"+ // magic
  77. "\x03\x00\x00\x00"+ // version
  78. "\x00\x00\x00\x00\x00\x00\x00\x00"+ // numTensors
  79. "\x01\x00\x00\x00\x00\x00\x00\x00"+ // numMetaValues
  80. // general.file_type key
  81. "\x11\x00\x00\x00\x00\x00\x00\x00"+ // key length
  82. "general.file_type"+ // key
  83. "\x04\x00\x00\x00"+ // type (uint32)
  84. "\x02\x00\x00\x00\x00\x00\x00\x00"+ // uint32 value
  85. "",
  86. )
  87. dir := t.TempDir()
  88. s, err := Open(dir)
  89. if err != nil {
  90. t.Fatal(err)
  91. }
  92. if err := s.Build(qualifiedRef, model.File{From: w.fileName("gguf")}); err != nil {
  93. t.Fatal(err)
  94. }
  95. filepath.Walk(dir, func(p string, info os.FileInfo, err error) error {
  96. t.Logf("file: %s", p)
  97. return nil
  98. })
  99. _, err = s.WeightsFile("unknown/y/z:latest+Q4_0")
  100. if !errors.Is(err, ErrNotFound) {
  101. t.Fatalf("WeightsFile() err = %v; want not found", err)
  102. }
  103. path, err := s.WeightsFile("x/y/z:latest+Q4_0")
  104. if err != nil {
  105. t.Fatal(err)
  106. }
  107. info, err := gguf.Stat(path)
  108. if err != nil {
  109. t.Fatal(err)
  110. }
  111. if info.FileType != gguf.TypeQ4_0 {
  112. t.Errorf("info.FileType = %d; want 1", info.FileType)
  113. }
  114. }
  115. type work struct {
  116. t testing.TB
  117. dir string
  118. }
  119. func newWorkDir(t *testing.T) work {
  120. return work{t: t, dir: t.TempDir()}
  121. }
  122. func (w work) write(name, content string) (path string) {
  123. w.t.Helper()
  124. path = w.fileName(name)
  125. if err := os.WriteFile(path, []byte(content), 0644); err != nil {
  126. w.t.Fatal(err)
  127. }
  128. return path
  129. }
  130. func (w work) fileName(name string) string {
  131. w.t.Helper()
  132. return filepath.Join(w.dir, name)
  133. }
  134. func (w work) mkdirAll(path string) {
  135. w.t.Helper()
  136. if err := os.MkdirAll(filepath.Join(w.dir, path), 0755); err != nil {
  137. w.t.Fatal(err)
  138. }
  139. }