You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
118 lines
4.8 KiB
118 lines
4.8 KiB
package volume_server_http_test
|
|
|
|
import (
|
|
"encoding/json"
|
|
"net/http"
|
|
"testing"
|
|
|
|
"github.com/seaweedfs/seaweedfs/test/volume_server/framework"
|
|
"github.com/seaweedfs/seaweedfs/test/volume_server/matrix"
|
|
)
|
|
|
|
func TestWriteUnchangedAndDeleteEdgeVariants(t *testing.T) {
|
|
if testing.Short() {
|
|
t.Skip("skipping integration test in short mode")
|
|
}
|
|
|
|
clusterHarness := framework.StartSingleVolumeCluster(t, matrix.P1())
|
|
conn, grpcClient := framework.DialVolumeServer(t, clusterHarness.VolumeGRPCAddress())
|
|
defer conn.Close()
|
|
|
|
const volumeID = uint32(87)
|
|
framework.AllocateVolume(t, grpcClient, volumeID, "")
|
|
|
|
const key = uint64(999001)
|
|
const cookie = uint32(0xDEADBEEF)
|
|
fid := framework.NewFileID(volumeID, key, cookie)
|
|
client := framework.NewHTTPClient()
|
|
payload := []byte("unchanged-write-content")
|
|
|
|
firstUpload := newUploadRequest(t, clusterHarness.VolumeAdminURL()+"/"+fid, payload)
|
|
firstUploadResp := framework.DoRequest(t, client, firstUpload)
|
|
_ = framework.ReadAllAndClose(t, firstUploadResp)
|
|
if firstUploadResp.StatusCode != http.StatusCreated {
|
|
t.Fatalf("first upload expected 201, got %d", firstUploadResp.StatusCode)
|
|
}
|
|
|
|
secondUpload := newUploadRequest(t, clusterHarness.VolumeAdminURL()+"/"+fid, payload)
|
|
secondUploadResp := framework.DoRequest(t, client, secondUpload)
|
|
_ = framework.ReadAllAndClose(t, secondUploadResp)
|
|
if secondUploadResp.StatusCode != http.StatusNoContent {
|
|
t.Fatalf("second unchanged upload expected 204, got %d", secondUploadResp.StatusCode)
|
|
}
|
|
if secondUploadResp.Header.Get("ETag") == "" {
|
|
t.Fatalf("second unchanged upload expected ETag header")
|
|
}
|
|
|
|
wrongCookieFid := framework.NewFileID(volumeID, key, cookie+1)
|
|
wrongCookieDelete := framework.DoRequest(t, client, mustNewRequest(t, http.MethodDelete, clusterHarness.VolumeAdminURL()+"/"+wrongCookieFid))
|
|
_ = framework.ReadAllAndClose(t, wrongCookieDelete)
|
|
if wrongCookieDelete.StatusCode != http.StatusBadRequest {
|
|
t.Fatalf("delete with mismatched cookie expected 400, got %d", wrongCookieDelete.StatusCode)
|
|
}
|
|
|
|
missingDelete := framework.DoRequest(t, client, mustNewRequest(t, http.MethodDelete, clusterHarness.VolumeAdminURL()+"/"+framework.NewFileID(volumeID, key+1, cookie)))
|
|
missingDeleteBody := framework.ReadAllAndClose(t, missingDelete)
|
|
if missingDelete.StatusCode != http.StatusNotFound {
|
|
t.Fatalf("delete missing needle expected 404, got %d", missingDelete.StatusCode)
|
|
}
|
|
|
|
var payloadMap map[string]int64
|
|
if err := json.Unmarshal(missingDeleteBody, &payloadMap); err != nil {
|
|
t.Fatalf("decode delete missing response: %v", err)
|
|
}
|
|
if payloadMap["size"] != 0 {
|
|
t.Fatalf("delete missing needle expected size=0, got %d", payloadMap["size"])
|
|
}
|
|
}
|
|
|
|
func TestDeleteTimestampOverrideKeepsReadDeletedLastModifiedParity(t *testing.T) {
|
|
if testing.Short() {
|
|
t.Skip("skipping integration test in short mode")
|
|
}
|
|
|
|
clusterHarness := framework.StartSingleVolumeCluster(t, matrix.P1())
|
|
conn, grpcClient := framework.DialVolumeServer(t, clusterHarness.VolumeGRPCAddress())
|
|
defer conn.Close()
|
|
|
|
const volumeID = uint32(88)
|
|
framework.AllocateVolume(t, grpcClient, volumeID, "")
|
|
|
|
client := framework.NewHTTPClient()
|
|
fid := framework.NewFileID(volumeID, 999002, 0xABCD1234)
|
|
uploadResp := framework.UploadBytes(t, client, clusterHarness.VolumeAdminURL(), fid, []byte("delete-ts-override"))
|
|
_ = framework.ReadAllAndClose(t, uploadResp)
|
|
if uploadResp.StatusCode != http.StatusCreated {
|
|
t.Fatalf("upload expected 201, got %d", uploadResp.StatusCode)
|
|
}
|
|
|
|
beforeDeleteResp := framework.ReadBytes(t, client, clusterHarness.VolumeAdminURL(), fid)
|
|
_ = framework.ReadAllAndClose(t, beforeDeleteResp)
|
|
if beforeDeleteResp.StatusCode != http.StatusOK {
|
|
t.Fatalf("pre-delete read expected 200, got %d", beforeDeleteResp.StatusCode)
|
|
}
|
|
lastModifiedBeforeDelete := beforeDeleteResp.Header.Get("Last-Modified")
|
|
if lastModifiedBeforeDelete == "" {
|
|
t.Fatalf("expected Last-Modified before delete")
|
|
}
|
|
|
|
deleteReq := mustNewRequest(t, http.MethodDelete, clusterHarness.VolumeAdminURL()+"/"+fid+"?ts=1700000000")
|
|
deleteResp := framework.DoRequest(t, client, deleteReq)
|
|
_ = framework.ReadAllAndClose(t, deleteResp)
|
|
if deleteResp.StatusCode != http.StatusAccepted {
|
|
t.Fatalf("delete with ts override expected 202, got %d", deleteResp.StatusCode)
|
|
}
|
|
|
|
readDeletedResp := framework.DoRequest(t, client, mustNewRequest(t, http.MethodGet, clusterHarness.VolumeAdminURL()+"/"+fid+"?readDeleted=true"))
|
|
_ = framework.ReadAllAndClose(t, readDeletedResp)
|
|
if readDeletedResp.StatusCode != http.StatusOK {
|
|
t.Fatalf("readDeleted after ts override expected 200, got %d", readDeletedResp.StatusCode)
|
|
}
|
|
lastModified := readDeletedResp.Header.Get("Last-Modified")
|
|
if lastModified == "" {
|
|
t.Fatalf("expected Last-Modified header on readDeleted response")
|
|
}
|
|
if lastModified != lastModifiedBeforeDelete {
|
|
t.Fatalf("expected readDeleted Last-Modified parity with pre-delete header, got %q want %q", lastModified, lastModifiedBeforeDelete)
|
|
}
|
|
}
|