|
|
@@ -0,0 +1,417 @@
|
|
|
+import { describe, it, expect, vi, beforeEach, afterEach } from "vitest"
|
|
|
+import { generateImageWithImagesApi, generateImageWithProvider } from "../image-generation"
|
|
|
+
|
|
|
+// Mock the i18n module
|
|
|
+vi.mock("../../../i18n", () => ({
|
|
|
+ t: (key: string, options?: any) => {
|
|
|
+ // Return a sensible mock for i18n
|
|
|
+ if (key === "tools:generateImage.failedWithMessage" && options?.message) {
|
|
|
+ return options.message
|
|
|
+ }
|
|
|
+ return key
|
|
|
+ },
|
|
|
+}))
|
|
|
+
|
|
|
+// Mock fetch globally
|
|
|
+global.fetch = vi.fn()
|
|
|
+global.FormData = vi.fn(() => ({
|
|
|
+ append: vi.fn(),
|
|
|
+})) as any
|
|
|
+global.Blob = vi.fn() as any
|
|
|
+global.atob = vi.fn((str: string) => {
|
|
|
+ return Buffer.from(str, "base64").toString("binary")
|
|
|
+})
|
|
|
+
|
|
|
+describe("generateImageWithImagesApi", () => {
|
|
|
+ beforeEach(() => {
|
|
|
+ vi.clearAllMocks()
|
|
|
+ })
|
|
|
+
|
|
|
+ afterEach(() => {
|
|
|
+ vi.clearAllMocks()
|
|
|
+ })
|
|
|
+
|
|
|
+ describe("image generation (text-to-image)", () => {
|
|
|
+ it("should successfully generate an image", async () => {
|
|
|
+ const mockBase64 = Buffer.from("fake image data").toString("base64")
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ b64_json: mockBase64 }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ outputFormat: "png",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+ expect(result.imageData).toContain("data:image/png;base64,")
|
|
|
+ expect(result.imageFormat).toBe("png")
|
|
|
+
|
|
|
+ // Verify fetch was called with correct parameters
|
|
|
+ expect(global.fetch).toHaveBeenCalledWith(
|
|
|
+ "https://api.example.com/v1/images/generations",
|
|
|
+ expect.objectContaining({
|
|
|
+ method: "POST",
|
|
|
+ headers: expect.objectContaining({
|
|
|
+ Authorization: "Bearer test-token",
|
|
|
+ "Content-Type": "application/json",
|
|
|
+ }),
|
|
|
+ }),
|
|
|
+ )
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle API errors gracefully", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: false,
|
|
|
+ status: 400,
|
|
|
+ statusText: "Bad Request",
|
|
|
+ text: vi.fn().mockResolvedValue("{}"),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle missing image data in response", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{}], // Missing b64_json and url
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle URL response instead of b64_json", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ url: "" }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+ expect(result.imageData).toBe("")
|
|
|
+ expect(result.imageFormat).toBe("png")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle external URL response", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ url: "https://example.com/generated-image.png" }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ outputFormat: "png",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+ expect(result.imageData).toBe("https://example.com/generated-image.png")
|
|
|
+ expect(result.imageFormat).toBe("png")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle empty data array in response", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle API error response", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ error: {
|
|
|
+ message: "Rate limit exceeded",
|
|
|
+ type: "rate_limit_error",
|
|
|
+ },
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should include optional parameters when provided", async () => {
|
|
|
+ const mockBase64 = Buffer.from("fake image data").toString("base64")
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ b64_json: mockBase64 }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ size: "1024x1024",
|
|
|
+ quality: "hd",
|
|
|
+ outputFormat: "png",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+
|
|
|
+ // Verify fetch was called with optional parameters
|
|
|
+ const callArgs = vi.mocked(global.fetch).mock.calls[0]
|
|
|
+ const body = JSON.parse(callArgs[1]?.body as string)
|
|
|
+ expect(body.size).toBe("1024x1024")
|
|
|
+ expect(body.quality).toBe("hd")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle network errors", async () => {
|
|
|
+ vi.mocked(global.fetch).mockRejectedValue(new Error("Network error"))
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toContain("Network error")
|
|
|
+ })
|
|
|
+ })
|
|
|
+
|
|
|
+ describe("image editing", () => {
|
|
|
+ it("should use /images/generations endpoint with inputImage in request body", async () => {
|
|
|
+ const mockBase64 = Buffer.from("fake image data").toString("base64")
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ b64_json: mockBase64 }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const inputImageDataUrl = `data:image/png;base64,${mockBase64}`
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "Make it blue",
|
|
|
+ inputImage: inputImageDataUrl,
|
|
|
+ outputFormat: "png",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+
|
|
|
+ // Verify /images/generations endpoint was used (not /images/edits)
|
|
|
+ const callUrl = vi.mocked(global.fetch).mock.calls[0][0]
|
|
|
+ expect(callUrl).toContain("/images/generations")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle edit operation errors", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: false,
|
|
|
+ status: 400,
|
|
|
+ statusText: "Bad Request",
|
|
|
+ text: vi.fn().mockResolvedValue("{}"),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const inputImageDataUrl =
|
|
|
+ ""
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "Make it blue",
|
|
|
+ inputImage: inputImageDataUrl,
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+ })
|
|
|
+
|
|
|
+ describe("output format handling", () => {
|
|
|
+ it("should use png format by default", async () => {
|
|
|
+ const mockBase64 = Buffer.from("fake image data").toString("base64")
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ b64_json: mockBase64 }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.imageFormat).toBe("png")
|
|
|
+ expect(result.imageData).toContain("data:image/png;base64,")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should use specified output format", async () => {
|
|
|
+ const mockBase64 = Buffer.from("fake image data").toString("base64")
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ data: [{ b64_json: mockBase64 }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithImagesApi({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-image-1",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ outputFormat: "jpeg",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.imageFormat).toBe("jpeg")
|
|
|
+ expect(result.imageData).toContain("data:image/jpeg;base64,")
|
|
|
+ })
|
|
|
+ })
|
|
|
+})
|
|
|
+
|
|
|
+describe("generateImageWithProvider (chat completions)", () => {
|
|
|
+ beforeEach(() => {
|
|
|
+ vi.clearAllMocks()
|
|
|
+ })
|
|
|
+
|
|
|
+ afterEach(() => {
|
|
|
+ vi.clearAllMocks()
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should use /chat/completions endpoint", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ choices: [
|
|
|
+ {
|
|
|
+ message: {
|
|
|
+ images: [
|
|
|
+ {
|
|
|
+ image_url: {
|
|
|
+ url: "",
|
|
|
+ },
|
|
|
+ },
|
|
|
+ ],
|
|
|
+ },
|
|
|
+ },
|
|
|
+ ],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithProvider({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-4-vision",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(true)
|
|
|
+
|
|
|
+ // Verify /chat/completions endpoint was used
|
|
|
+ const callUrl = vi.mocked(global.fetch).mock.calls[0][0]
|
|
|
+ expect(callUrl).toContain("/chat/completions")
|
|
|
+ })
|
|
|
+
|
|
|
+ it("should handle missing images in response", async () => {
|
|
|
+ const mockResponse = {
|
|
|
+ ok: true,
|
|
|
+ json: vi.fn().mockResolvedValue({
|
|
|
+ choices: [{ message: { content: "No images" } }],
|
|
|
+ }),
|
|
|
+ }
|
|
|
+
|
|
|
+ vi.mocked(global.fetch).mockResolvedValue(mockResponse as any)
|
|
|
+
|
|
|
+ const result = await generateImageWithProvider({
|
|
|
+ baseURL: "https://api.example.com/v1",
|
|
|
+ authToken: "test-token",
|
|
|
+ model: "gpt-4-vision",
|
|
|
+ prompt: "A cute cat",
|
|
|
+ })
|
|
|
+
|
|
|
+ expect(result.success).toBe(false)
|
|
|
+ expect(result.error).toBeDefined()
|
|
|
+ })
|
|
|
+})
|