From e0abcf14010f1c300e7321ae320b890f2273ad79 Mon Sep 17 00:00:00 2001 From: Chris Lu Date: Tue, 10 Feb 2026 17:38:25 -0800 Subject: [PATCH] s3api: expand namespace validation REST tests --- .../s3api_tables_rest_validation_test.go | 103 ++++++++++++------ 1 file changed, 71 insertions(+), 32 deletions(-) diff --git a/weed/s3api/s3api_tables_rest_validation_test.go b/weed/s3api/s3api_tables_rest_validation_test.go index 3445e42df..38305ea6d 100644 --- a/weed/s3api/s3api_tables_rest_validation_test.go +++ b/weed/s3api/s3api_tables_rest_validation_test.go @@ -16,50 +16,89 @@ import ( const testTableBucketARN = "arn:aws:s3tables:us-east-1:123456789012:bucket/test-bucket" func TestBuildListTablesRequestRejectsInvalidNamespaceQuery(t *testing.T) { - req := httptest.NewRequest(http.MethodGet, "/tables?namespace=InvalidNamespace", nil) - req = mux.SetURLVars(req, map[string]string{"tableBucketARN": testTableBucketARN}) - - _, err := buildListTablesRequest(req) - if err == nil { - t.Fatalf("expected invalid namespace query to return an error") + testCases := []struct { + name string + namespace string + }{ + {name: "uppercase", namespace: "InvalidNamespace"}, + {name: "hyphen", namespace: "invalid-ns"}, + {name: "slash", namespace: "a/b"}, } - if !strings.Contains(err.Error(), "invalid namespace") { - t.Fatalf("expected invalid namespace error, got %q", err.Error()) + + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + req := httptest.NewRequest(http.MethodGet, "/tables?namespace="+url.QueryEscape(tc.namespace), nil) + req = mux.SetURLVars(req, map[string]string{"tableBucketARN": testTableBucketARN}) + + _, err := buildListTablesRequest(req) + if err == nil { + t.Fatalf("expected invalid namespace query to return an error") + } + if !strings.Contains(err.Error(), "invalid namespace") { + t.Fatalf("expected invalid namespace error, got %q", err.Error()) + } + }) } } func TestBuildGetTableRequestRejectsInvalidNamespaceQuery(t *testing.T) { - req := httptest.NewRequest(http.MethodGet, "/get-table?tableBucketARN="+url.QueryEscape(testTableBucketARN)+"&namespace=InvalidNamespace&name=table1", nil) - - _, err := buildGetTableRequest(req) - if err == nil { - t.Fatalf("expected invalid namespace query to return an error") + testCases := []struct { + name string + namespace string + }{ + {name: "uppercase", namespace: "InvalidNamespace"}, + {name: "hyphen", namespace: "invalid-ns"}, + {name: "slash", namespace: "a/b"}, } - if !strings.Contains(err.Error(), "invalid namespace") { - t.Fatalf("expected invalid namespace error, got %q", err.Error()) + + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + req := httptest.NewRequest(http.MethodGet, "/get-table?tableBucketARN="+url.QueryEscape(testTableBucketARN)+"&namespace="+url.QueryEscape(tc.namespace)+"&name=table1", nil) + + _, err := buildGetTableRequest(req) + if err == nil { + t.Fatalf("expected invalid namespace query to return an error") + } + if !strings.Contains(err.Error(), "invalid namespace") { + t.Fatalf("expected invalid namespace error, got %q", err.Error()) + } + }) } } func TestHandleRestOperationReturnsBadRequestForInvalidNamespaceQuery(t *testing.T) { - req := httptest.NewRequest(http.MethodGet, "/tables?namespace=InvalidNamespace", nil) - req = mux.SetURLVars(req, map[string]string{"tableBucketARN": testTableBucketARN}) - rr := httptest.NewRecorder() + testCases := []struct { + name string + namespace string + }{ + {name: "uppercase", namespace: "InvalidNamespace"}, + {name: "hyphen", namespace: "invalid-ns"}, + {name: "slash", namespace: "a/b"}, + } - st := &S3TablesApiServer{} - st.handleRestOperation("ListTables", buildListTablesRequest).ServeHTTP(rr, req) + for _, tc := range testCases { + t.Run(tc.name, func(t *testing.T) { + req := httptest.NewRequest(http.MethodGet, "/tables?namespace="+url.QueryEscape(tc.namespace), nil) + req = mux.SetURLVars(req, map[string]string{"tableBucketARN": testTableBucketARN}) + rr := httptest.NewRecorder() - if rr.Code != http.StatusBadRequest { - t.Fatalf("expected status %d, got %d", http.StatusBadRequest, rr.Code) - } + st := &S3TablesApiServer{} + st.handleRestOperation("ListTables", buildListTablesRequest).ServeHTTP(rr, req) - var body map[string]string - if err := json.Unmarshal(rr.Body.Bytes(), &body); err != nil { - t.Fatalf("failed to decode error response: %v", err) - } - if got, want := body["__type"], s3tables.ErrCodeInvalidRequest; got != want { - t.Fatalf("expected __type=%q, got %q", want, got) - } - if !strings.Contains(body["message"], "invalid namespace") { - t.Fatalf("expected invalid namespace error message, got %q", body["message"]) + if rr.Code != http.StatusBadRequest { + t.Fatalf("expected status %d, got %d", http.StatusBadRequest, rr.Code) + } + + var body map[string]string + if err := json.Unmarshal(rr.Body.Bytes(), &body); err != nil { + t.Fatalf("failed to decode error response: %v", err) + } + if got, want := body["__type"], s3tables.ErrCodeInvalidRequest; got != want { + t.Fatalf("expected __type=%q, got %q", want, got) + } + if !strings.Contains(body["message"], "invalid namespace") { + t.Fatalf("expected invalid namespace error message, got %q", body["message"]) + } + }) } }