2024-06-26 21:38:21 -07:00
|
|
|
package server
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/zip"
|
|
|
|
"bytes"
|
2024-06-30 11:10:40 -07:00
|
|
|
"errors"
|
2024-06-26 21:38:21 -07:00
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"slices"
|
2024-06-30 11:10:40 -07:00
|
|
|
"strings"
|
2024-06-26 21:38:21 -07:00
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/ollama/ollama/api"
|
|
|
|
)
|
|
|
|
|
|
|
|
func createZipFile(t *testing.T, name string) *os.File {
|
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
f, err := os.CreateTemp(t.TempDir(), "")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
zf := zip.NewWriter(f)
|
|
|
|
defer zf.Close()
|
|
|
|
|
|
|
|
zh, err := zf.CreateHeader(&zip.FileHeader{Name: name})
|
|
|
|
if err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, err := io.Copy(zh, bytes.NewReader([]byte(""))); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return f
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestExtractFromZipFile(t *testing.T) {
|
|
|
|
cases := []struct {
|
|
|
|
name string
|
|
|
|
expect []string
|
2024-06-30 11:10:40 -07:00
|
|
|
err error
|
2024-06-26 21:38:21 -07:00
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "good",
|
|
|
|
expect: []string{"good"},
|
|
|
|
},
|
|
|
|
{
|
2024-06-30 11:10:40 -07:00
|
|
|
name: strings.Join([]string{"path", "..", "to", "good"}, string(os.PathSeparator)),
|
|
|
|
expect: []string{filepath.Join("to", "good")},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: strings.Join([]string{"path", "..", "to", "..", "good"}, string(os.PathSeparator)),
|
|
|
|
expect: []string{"good"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: strings.Join([]string{"path", "to", "..", "..", "good"}, string(os.PathSeparator)),
|
|
|
|
expect: []string{"good"},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: strings.Join([]string{"..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "..", "bad"}, string(os.PathSeparator)),
|
|
|
|
err: zip.ErrInsecurePath,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: strings.Join([]string{"path", "..", "..", "to", "bad"}, string(os.PathSeparator)),
|
|
|
|
err: zip.ErrInsecurePath,
|
2024-06-26 21:38:21 -07:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tt := range cases {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
f := createZipFile(t, tt.name)
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
tempDir := t.TempDir()
|
2024-06-30 11:10:40 -07:00
|
|
|
if err := extractFromZipFile(tempDir, f, func(api.ProgressResponse) {}); !errors.Is(err, tt.err) {
|
2024-06-26 21:38:21 -07:00
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var matches []string
|
|
|
|
if err := filepath.Walk(tempDir, func(p string, fi os.FileInfo, err error) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if !fi.IsDir() {
|
|
|
|
matches = append(matches, p)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}); err != nil {
|
|
|
|
t.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
var actual []string
|
|
|
|
for _, match := range matches {
|
|
|
|
rel, err := filepath.Rel(tempDir, match)
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
actual = append(actual, rel)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !slices.Equal(actual, tt.expect) {
|
|
|
|
t.Fatalf("expected %d files, got %d", len(tt.expect), len(matches))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|