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.
123 lines
4.8 KiB
123 lines
4.8 KiB
package volume_server_http_test
|
|
|
|
import (
|
|
"net/http"
|
|
"strconv"
|
|
"testing"
|
|
|
|
"github.com/seaweedfs/seaweedfs/test/volume_server/framework"
|
|
"github.com/seaweedfs/seaweedfs/test/volume_server/matrix"
|
|
)
|
|
|
|
func TestUploadReadRangeHeadDeleteRoundTrip(t *testing.T) {
|
|
if testing.Short() {
|
|
t.Skip("skipping integration test in short mode")
|
|
}
|
|
|
|
cluster := framework.StartSingleVolumeCluster(t, matrix.P1())
|
|
conn, grpcClient := framework.DialVolumeServer(t, cluster.VolumeGRPCAddress())
|
|
defer conn.Close()
|
|
|
|
const volumeID = uint32(7)
|
|
framework.AllocateVolume(t, grpcClient, volumeID, "")
|
|
|
|
fid := framework.NewFileID(volumeID, 123456, 0xA1B2C3D4)
|
|
data := []byte("hello-volume-server-integration")
|
|
client := framework.NewHTTPClient()
|
|
|
|
uploadResp := framework.UploadBytes(t, client, cluster.VolumeAdminURL(), fid, data)
|
|
_ = framework.ReadAllAndClose(t, uploadResp)
|
|
if uploadResp.StatusCode != http.StatusCreated {
|
|
t.Fatalf("upload status: expected 201, got %d", uploadResp.StatusCode)
|
|
}
|
|
|
|
getResp := framework.ReadBytes(t, client, cluster.VolumeAdminURL(), fid)
|
|
getBody := framework.ReadAllAndClose(t, getResp)
|
|
if getResp.StatusCode != http.StatusOK {
|
|
t.Fatalf("get status: expected 200, got %d", getResp.StatusCode)
|
|
}
|
|
if string(getBody) != string(data) {
|
|
t.Fatalf("get body mismatch: got %q want %q", string(getBody), string(data))
|
|
}
|
|
etag := getResp.Header.Get("ETag")
|
|
if etag == "" {
|
|
t.Fatalf("expected ETag header from GET response")
|
|
}
|
|
|
|
notModifiedReq := mustNewRequest(t, http.MethodGet, cluster.VolumeAdminURL()+"/"+fid)
|
|
notModifiedReq.Header.Set("If-None-Match", etag)
|
|
notModifiedResp := framework.DoRequest(t, client, notModifiedReq)
|
|
_ = framework.ReadAllAndClose(t, notModifiedResp)
|
|
if notModifiedResp.StatusCode != http.StatusNotModified {
|
|
t.Fatalf("if-none-match expected 304, got %d", notModifiedResp.StatusCode)
|
|
}
|
|
|
|
rangeReq := mustNewRequest(t, http.MethodGet, cluster.VolumeAdminURL()+"/"+fid)
|
|
rangeReq.Header.Set("Range", "bytes=0-4")
|
|
rangeResp := framework.DoRequest(t, client, rangeReq)
|
|
rangeBody := framework.ReadAllAndClose(t, rangeResp)
|
|
if rangeResp.StatusCode != http.StatusPartialContent {
|
|
t.Fatalf("range status: expected 206, got %d", rangeResp.StatusCode)
|
|
}
|
|
if got, want := string(rangeBody), "hello"; got != want {
|
|
t.Fatalf("range body mismatch: got %q want %q", got, want)
|
|
}
|
|
|
|
invalidRangeReq := mustNewRequest(t, http.MethodGet, cluster.VolumeAdminURL()+"/"+fid)
|
|
invalidRangeReq.Header.Set("Range", "bytes=9999-10000")
|
|
invalidRangeResp := framework.DoRequest(t, client, invalidRangeReq)
|
|
_ = framework.ReadAllAndClose(t, invalidRangeResp)
|
|
if invalidRangeResp.StatusCode != http.StatusRequestedRangeNotSatisfiable {
|
|
t.Fatalf("invalid range expected 416, got %d", invalidRangeResp.StatusCode)
|
|
}
|
|
|
|
headResp := framework.DoRequest(t, client, mustNewRequest(t, http.MethodHead, cluster.VolumeAdminURL()+"/"+fid))
|
|
headBody := framework.ReadAllAndClose(t, headResp)
|
|
if headResp.StatusCode != http.StatusOK {
|
|
t.Fatalf("head status: expected 200, got %d", headResp.StatusCode)
|
|
}
|
|
if got := headResp.Header.Get("Content-Length"); got != strconv.Itoa(len(data)) {
|
|
t.Fatalf("head content-length mismatch: got %q want %d", got, len(data))
|
|
}
|
|
if len(headBody) != 0 {
|
|
t.Fatalf("head body should be empty, got %d bytes", len(headBody))
|
|
}
|
|
|
|
headNotModifiedReq := mustNewRequest(t, http.MethodHead, cluster.VolumeAdminURL()+"/"+fid)
|
|
headNotModifiedReq.Header.Set("If-None-Match", etag)
|
|
headNotModifiedResp := framework.DoRequest(t, client, headNotModifiedReq)
|
|
headNotModifiedBody := framework.ReadAllAndClose(t, headNotModifiedResp)
|
|
if headNotModifiedResp.StatusCode != http.StatusNotModified {
|
|
t.Fatalf("head if-none-match expected 304, got %d", headNotModifiedResp.StatusCode)
|
|
}
|
|
if len(headNotModifiedBody) != 0 {
|
|
t.Fatalf("head if-none-match body should be empty, got %d bytes", len(headNotModifiedBody))
|
|
}
|
|
|
|
deleteResp := framework.DoRequest(t, client, mustNewRequest(t, http.MethodDelete, cluster.VolumeAdminURL()+"/"+fid))
|
|
_ = framework.ReadAllAndClose(t, deleteResp)
|
|
if deleteResp.StatusCode != http.StatusAccepted {
|
|
t.Fatalf("delete status: expected 202, got %d", deleteResp.StatusCode)
|
|
}
|
|
|
|
notFoundResp := framework.ReadBytes(t, client, cluster.VolumeAdminURL(), fid)
|
|
_ = framework.ReadAllAndClose(t, notFoundResp)
|
|
if notFoundResp.StatusCode != http.StatusNotFound {
|
|
t.Fatalf("read after delete: expected 404, got %d", notFoundResp.StatusCode)
|
|
}
|
|
}
|
|
|
|
func TestInvalidReadPathReturnsBadRequest(t *testing.T) {
|
|
if testing.Short() {
|
|
t.Skip("skipping integration test in short mode")
|
|
}
|
|
|
|
cluster := framework.StartSingleVolumeCluster(t, matrix.P1())
|
|
client := framework.NewHTTPClient()
|
|
|
|
resp := framework.DoRequest(t, client, mustNewRequest(t, http.MethodGet, cluster.VolumeAdminURL()+"/invalid,needle"))
|
|
_ = framework.ReadAllAndClose(t, resp)
|
|
if resp.StatusCode != http.StatusBadRequest {
|
|
t.Fatalf("invalid read expected 400, got %d", resp.StatusCode)
|
|
}
|
|
}
|