shithub: furgit

ref: 673902b14458f32fbf47efa3757279872bdfcb7e
dir: /objectstore/loose/write_test.go/

View raw version
package loose_test

import (
	"bytes"
	"io"
	"testing"

	"codeberg.org/lindenii/furgit/internal/testgit"
	"codeberg.org/lindenii/furgit/objectheader"
	"codeberg.org/lindenii/furgit/objectid"
	"codeberg.org/lindenii/furgit/objecttype"
)

func TestLooseStoreWriteWriterContentAgainstGit(t *testing.T) {
	t.Parallel()
	testgit.ForEachAlgorithm(t, func(t *testing.T, algo objectid.Algorithm) { //nolint:thelper
		testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
		store := openLooseStore(t, testRepo.Dir(), algo)

		content := []byte("written-by-content-writer\n")
		expectedHex := testRepo.RunInput(t, content, "hash-object", "-t", "blob", "--stdin")
		expectedID, err := objectid.ParseHex(algo, expectedHex)
		if err != nil {
			t.Fatalf("ParseHex(expected): %v", err)
		}

		writer, finalize, err := store.WriteWriterContent(objecttype.TypeBlob, int64(len(content)))
		if err != nil {
			t.Fatalf("WriteWriterContent: %v", err)
		}
		if _, err := io.Copy(writer, bytes.NewReader(content)); err != nil {
			t.Fatalf("WriteWriterContent write: %v", err)
		}
		if err := writer.Close(); err != nil {
			t.Fatalf("WriteWriterContent close: %v", err)
		}
		writtenID, err := finalize()
		if err != nil {
			t.Fatalf("WriteWriterContent finalize: %v", err)
		}
		if writtenID != expectedID {
			t.Fatalf("WriteWriterContent id = %s, want %s", writtenID, expectedID)
		}

		gotBody := testRepo.CatFile(t, "blob", writtenID)
		if !bytes.Equal(gotBody, content) {
			t.Fatalf("git cat-file body mismatch")
		}

		// Writing the same object again should succeed and return the same ID.
		writer, finalize, err = store.WriteWriterContent(objecttype.TypeBlob, int64(len(content)))
		if err != nil {
			t.Fatalf("WriteWriterContent second: %v", err)
		}
		if _, err := io.Copy(writer, bytes.NewReader(content)); err != nil {
			t.Fatalf("WriteWriterContent second write: %v", err)
		}
		if err := writer.Close(); err != nil {
			t.Fatalf("WriteWriterContent second close: %v", err)
		}
		writtenID2, err := finalize()
		if err != nil {
			t.Fatalf("WriteWriterContent second finalize: %v", err)
		}
		if writtenID2 != expectedID {
			t.Fatalf("WriteWriterContent second id = %s, want %s", writtenID2, expectedID)
		}
	})
}

func TestLooseStoreWriteWriterFullAgainstGit(t *testing.T) {
	t.Parallel()
	testgit.ForEachAlgorithm(t, func(t *testing.T, algo objectid.Algorithm) { //nolint:thelper
		testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
		store := openLooseStore(t, testRepo.Dir(), algo)

		body := []byte("full-writer-body\n")
		header, ok := objectheader.Encode(objecttype.TypeBlob, int64(len(body)))
		if !ok {
			t.Fatalf("objectheader.Encode failed")
		}
		raw := make([]byte, len(header)+len(body))
		copy(raw, header)
		copy(raw[len(header):], body)

		wantID := algo.Sum(raw)
		writer, finalize, err := store.WriteWriterFull()
		if err != nil {
			t.Fatalf("WriteWriterFull: %v", err)
		}
		if _, err := io.Copy(writer, bytes.NewReader(raw)); err != nil {
			t.Fatalf("WriteWriterFull write: %v", err)
		}
		if err := writer.Close(); err != nil {
			t.Fatalf("WriteWriterFull close: %v", err)
		}
		gotID, err := finalize()
		if err != nil {
			t.Fatalf("WriteWriterFull finalize: %v", err)
		}
		if gotID != wantID {
			t.Fatalf("WriteWriterFull id = %s, want %s", gotID, wantID)
		}

		gotBody := testRepo.CatFile(t, "blob", gotID)
		if !bytes.Equal(gotBody, body) {
			t.Fatalf("git cat-file body mismatch")
		}
	})
}

func TestLooseStoreWriterValidationErrors(t *testing.T) {
	t.Parallel()
	testgit.ForEachAlgorithm(t, func(t *testing.T, algo objectid.Algorithm) { //nolint:thelper
		t.Run("content overflow", func(t *testing.T) {
			t.Parallel()
			testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
			store := openLooseStore(t, testRepo.Dir(), algo)

			writer, finalize, err := store.WriteWriterContent(objecttype.TypeBlob, 1)
			if err != nil {
				t.Fatalf("WriteWriterContent: %v", err)
			}
			if _, err := writer.Write([]byte("hello")); err == nil {
				t.Fatalf("expected overflow error")
			}
			_ = writer.Close()
			if _, err := finalize(); err == nil {
				t.Fatalf("expected finalize error after overflow")
			}
		})

		t.Run("content short", func(t *testing.T) {
			t.Parallel()
			testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
			store := openLooseStore(t, testRepo.Dir(), algo)

			writer, finalize, err := store.WriteWriterContent(objecttype.TypeBlob, 5)
			if err != nil {
				t.Fatalf("WriteWriterContent: %v", err)
			}
			if _, err := writer.Write([]byte("x")); err != nil {
				t.Fatalf("write short: %v", err)
			}
			if err := writer.Close(); err != nil {
				t.Fatalf("close short: %v", err)
			}
			if _, err := finalize(); err == nil {
				t.Fatalf("expected finalize error for short content")
			}
		})

		t.Run("full malformed header", func(t *testing.T) {
			t.Parallel()
			testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
			store := openLooseStore(t, testRepo.Dir(), algo)

			writer, finalize, err := store.WriteWriterFull()
			if err != nil {
				t.Fatalf("WriteWriterFull: %v", err)
			}
			if _, err := writer.Write([]byte("not-a-header")); err != nil {
				t.Fatalf("write malformed header bytes unexpectedly failed: %v", err)
			}
			if err := writer.Close(); err != nil {
				t.Fatalf("close malformed header: %v", err)
			}
			if _, err := finalize(); err == nil {
				t.Fatalf("expected finalize error for malformed header")
			}
		})

		t.Run("full size mismatch", func(t *testing.T) {
			t.Parallel()
			testRepo := testgit.NewRepo(t, testgit.RepoOptions{ObjectFormat: algo, Bare: true})
			store := openLooseStore(t, testRepo.Dir(), algo)

			writer, finalize, err := store.WriteWriterFull()
			if err != nil {
				t.Fatalf("WriteWriterFull: %v", err)
			}
			raw := []byte("blob 1\x00hello")
			if _, err := io.Copy(writer, bytes.NewReader(raw)); err == nil {
				t.Fatalf("expected overflow error")
			}
			_ = writer.Close()
			if _, err := finalize(); err == nil {
				t.Fatalf("expected finalize error after mismatch")
			}
		})
	})
}