198 lines
6.0 KiB
Go
198 lines
6.0 KiB
Go
package management
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"mime/multipart"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"net/url"
|
|
"os"
|
|
"path/filepath"
|
|
"testing"
|
|
|
|
"github.com/gin-gonic/gin"
|
|
"github.com/router-for-me/CLIProxyAPI/v6/internal/config"
|
|
coreauth "github.com/router-for-me/CLIProxyAPI/v6/sdk/cliproxy/auth"
|
|
)
|
|
|
|
func TestUploadAuthFile_BatchMultipart(t *testing.T) {
|
|
t.Setenv("MANAGEMENT_PASSWORD", "")
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
authDir := t.TempDir()
|
|
manager := coreauth.NewManager(nil, nil, nil)
|
|
h := NewHandlerWithoutConfigFilePath(&config.Config{AuthDir: authDir}, manager)
|
|
|
|
files := []struct {
|
|
name string
|
|
content string
|
|
}{
|
|
{name: "alpha.json", content: `{"type":"codex","email":"alpha@example.com"}`},
|
|
{name: "beta.json", content: `{"type":"claude","email":"beta@example.com"}`},
|
|
}
|
|
|
|
var body bytes.Buffer
|
|
writer := multipart.NewWriter(&body)
|
|
for _, file := range files {
|
|
part, err := writer.CreateFormFile("file", file.name)
|
|
if err != nil {
|
|
t.Fatalf("failed to create multipart file: %v", err)
|
|
}
|
|
if _, err = part.Write([]byte(file.content)); err != nil {
|
|
t.Fatalf("failed to write multipart content: %v", err)
|
|
}
|
|
}
|
|
if err := writer.Close(); err != nil {
|
|
t.Fatalf("failed to close multipart writer: %v", err)
|
|
}
|
|
|
|
rec := httptest.NewRecorder()
|
|
ctx, _ := gin.CreateTestContext(rec)
|
|
req := httptest.NewRequest(http.MethodPost, "/v0/management/auth-files", &body)
|
|
req.Header.Set("Content-Type", writer.FormDataContentType())
|
|
ctx.Request = req
|
|
|
|
h.UploadAuthFile(ctx)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
t.Fatalf("expected upload status %d, got %d with body %s", http.StatusOK, rec.Code, rec.Body.String())
|
|
}
|
|
|
|
var payload map[string]any
|
|
if err := json.Unmarshal(rec.Body.Bytes(), &payload); err != nil {
|
|
t.Fatalf("failed to decode response: %v", err)
|
|
}
|
|
if got, ok := payload["uploaded"].(float64); !ok || int(got) != len(files) {
|
|
t.Fatalf("expected uploaded=%d, got %#v", len(files), payload["uploaded"])
|
|
}
|
|
|
|
for _, file := range files {
|
|
fullPath := filepath.Join(authDir, file.name)
|
|
data, err := os.ReadFile(fullPath)
|
|
if err != nil {
|
|
t.Fatalf("expected uploaded file %s to exist: %v", file.name, err)
|
|
}
|
|
if string(data) != file.content {
|
|
t.Fatalf("expected file %s content %q, got %q", file.name, file.content, string(data))
|
|
}
|
|
}
|
|
|
|
auths := manager.List()
|
|
if len(auths) != len(files) {
|
|
t.Fatalf("expected %d auth entries, got %d", len(files), len(auths))
|
|
}
|
|
}
|
|
|
|
func TestUploadAuthFile_BatchMultipart_InvalidJSONDoesNotOverwriteExistingFile(t *testing.T) {
|
|
t.Setenv("MANAGEMENT_PASSWORD", "")
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
authDir := t.TempDir()
|
|
manager := coreauth.NewManager(nil, nil, nil)
|
|
h := NewHandlerWithoutConfigFilePath(&config.Config{AuthDir: authDir}, manager)
|
|
|
|
existingName := "alpha.json"
|
|
existingContent := `{"type":"codex","email":"alpha@example.com"}`
|
|
if err := os.WriteFile(filepath.Join(authDir, existingName), []byte(existingContent), 0o600); err != nil {
|
|
t.Fatalf("failed to seed existing auth file: %v", err)
|
|
}
|
|
|
|
files := []struct {
|
|
name string
|
|
content string
|
|
}{
|
|
{name: existingName, content: `{"type":"codex"`},
|
|
{name: "beta.json", content: `{"type":"claude","email":"beta@example.com"}`},
|
|
}
|
|
|
|
var body bytes.Buffer
|
|
writer := multipart.NewWriter(&body)
|
|
for _, file := range files {
|
|
part, err := writer.CreateFormFile("file", file.name)
|
|
if err != nil {
|
|
t.Fatalf("failed to create multipart file: %v", err)
|
|
}
|
|
if _, err = part.Write([]byte(file.content)); err != nil {
|
|
t.Fatalf("failed to write multipart content: %v", err)
|
|
}
|
|
}
|
|
if err := writer.Close(); err != nil {
|
|
t.Fatalf("failed to close multipart writer: %v", err)
|
|
}
|
|
|
|
rec := httptest.NewRecorder()
|
|
ctx, _ := gin.CreateTestContext(rec)
|
|
req := httptest.NewRequest(http.MethodPost, "/v0/management/auth-files", &body)
|
|
req.Header.Set("Content-Type", writer.FormDataContentType())
|
|
ctx.Request = req
|
|
|
|
h.UploadAuthFile(ctx)
|
|
|
|
if rec.Code != http.StatusMultiStatus {
|
|
t.Fatalf("expected upload status %d, got %d with body %s", http.StatusMultiStatus, rec.Code, rec.Body.String())
|
|
}
|
|
|
|
data, err := os.ReadFile(filepath.Join(authDir, existingName))
|
|
if err != nil {
|
|
t.Fatalf("expected existing auth file to remain readable: %v", err)
|
|
}
|
|
if string(data) != existingContent {
|
|
t.Fatalf("expected existing auth file to remain %q, got %q", existingContent, string(data))
|
|
}
|
|
|
|
betaData, err := os.ReadFile(filepath.Join(authDir, "beta.json"))
|
|
if err != nil {
|
|
t.Fatalf("expected valid auth file to be created: %v", err)
|
|
}
|
|
if string(betaData) != files[1].content {
|
|
t.Fatalf("expected beta auth file content %q, got %q", files[1].content, string(betaData))
|
|
}
|
|
}
|
|
|
|
func TestDeleteAuthFile_BatchQuery(t *testing.T) {
|
|
t.Setenv("MANAGEMENT_PASSWORD", "")
|
|
gin.SetMode(gin.TestMode)
|
|
|
|
authDir := t.TempDir()
|
|
files := []string{"alpha.json", "beta.json"}
|
|
for _, name := range files {
|
|
if err := os.WriteFile(filepath.Join(authDir, name), []byte(`{"type":"codex"}`), 0o600); err != nil {
|
|
t.Fatalf("failed to write auth file %s: %v", name, err)
|
|
}
|
|
}
|
|
|
|
manager := coreauth.NewManager(nil, nil, nil)
|
|
h := NewHandlerWithoutConfigFilePath(&config.Config{AuthDir: authDir}, manager)
|
|
h.tokenStore = &memoryAuthStore{}
|
|
|
|
rec := httptest.NewRecorder()
|
|
ctx, _ := gin.CreateTestContext(rec)
|
|
req := httptest.NewRequest(
|
|
http.MethodDelete,
|
|
"/v0/management/auth-files?name="+url.QueryEscape(files[0])+"&name="+url.QueryEscape(files[1]),
|
|
nil,
|
|
)
|
|
ctx.Request = req
|
|
|
|
h.DeleteAuthFile(ctx)
|
|
|
|
if rec.Code != http.StatusOK {
|
|
t.Fatalf("expected delete status %d, got %d with body %s", http.StatusOK, rec.Code, rec.Body.String())
|
|
}
|
|
|
|
var payload map[string]any
|
|
if err := json.Unmarshal(rec.Body.Bytes(), &payload); err != nil {
|
|
t.Fatalf("failed to decode response: %v", err)
|
|
}
|
|
if got, ok := payload["deleted"].(float64); !ok || int(got) != len(files) {
|
|
t.Fatalf("expected deleted=%d, got %#v", len(files), payload["deleted"])
|
|
}
|
|
|
|
for _, name := range files {
|
|
if _, err := os.Stat(filepath.Join(authDir, name)); !os.IsNotExist(err) {
|
|
t.Fatalf("expected auth file %s to be removed, stat err: %v", name, err)
|
|
}
|
|
}
|
|
}
|