diff --git a/FUNDING.yml b/FUNDING.yml new file mode 100644 index 0000000..392c477 --- /dev/null +++ b/FUNDING.yml @@ -0,0 +1,14 @@ +# These are supported funding model platforms + +github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] +patreon: # Replace with a single Patreon username +open_collective: # Replace with a single Open Collective username +ko_fi: # Replace with a single Ko-fi username +tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel +community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry +liberapay: # Replace with a single Liberapay username +issuehunt: # Replace with a single IssueHunt username +lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry +polar: # Replace with a single Polar username +buy_me_a_coffee: gcui # Replace with a single Buy Me a Coffee username +custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] diff --git a/README.md b/README.md index c985958..fdbe56e 100644 --- a/README.md +++ b/README.md @@ -13,7 +13,7 @@ | Deploy with Vercel
 @@ -33,6 +33,7 @@ We have deployed an example bound to a free Suno account, so it has daily usage - Perfectly implements the creation API from app.suno.ai - Automatically keep the account active. +- Compatible with the format of OpenAI’s `/v1/chat/completions` API. - Supports Custom Mode - One-click deployment to Vercel - In addition to the standard API, it also adapts to the API Schema of Agent platforms like GPTs and Coze, so you can use it as a tool/plugin/Action for LLMs and integrate it into any AI Agent. @@ -116,6 +117,7 @@ Suno API currently mainly implements the following APIs: ```bash - `/api/generate`: Generate music +- `/v1/chat/completions`: Generate music - Call the generate API in a format that works with OpenAI’s API. - `/api/custom_generate`: Generate music (Custom Mode, support setting lyrics, music style, title, etc.) - `/api/generate_lyrics`: Generate lyrics based on prompt - `/api/get`: Get music information based on the id. Use “,” to separate multiple ids. diff --git a/README_CN.md b/README_CN.md index e96bc36..1bd5ec9 100644 --- a/README_CN.md +++ b/README_CN.md @@ -14,8 +14,9 @@ +  ## 简介 @@ -32,6 +33,7 @@ Suno.ai v3 是一个令人惊叹的 AI 音乐服务,虽然官方还没有开 - 完美的实现了 app.suno.ai 中的大部分 API - 自动保持账号活跃 +- 兼容OpenAI的 `/v1/chat/completions` API 格式 - 支持 Custom Mode - 一键部署到 vercel - 除了标准 API,还适配了 GPTs、coze 等 Agent 平台的 API Schema,所以你可以把它当做一个 LLM 的工具/插件/Action,集成到任意 AI Agent 中。 @@ -113,6 +115,7 @@ Suno API 目前主要实现了以下 API: ```bash - `/api/generate`: 创建音乐 +- `/v1/chat/completions`: 创建音乐 - 用OpenAI API 兼容的格式调用 generate API - `/api/custom_generate`: 创建音乐(自定义模式,支持设置歌词、音乐风格、设置标题等) - `/api/generate_lyrics`: 根据Prompt创建歌词 - `/api/get`: 根据id获取音乐信息。获取多个请用","分隔,不传ids则返回所有音乐 diff --git a/package-lock.json b/package-lock.json index 505fa0c..8ccfffc 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,12 +1,12 @@ { "name": "suno-api", - "version": "1.0.0", + "version": "1.1.0", "lockfileVersion": 3, "requires": true, "packages": { "": { "name": "suno-api", - "version": "1.0.0", + "version": "1.1.0", "license": "LGPL-3.0-or-later", "dependencies": { "@vercel/analytics": "^1.2.2", diff --git a/src/app/api/custom_generate/route.ts b/src/app/api/custom_generate/route.ts index dcd2104..12e9964 100644 --- a/src/app/api/custom_generate/route.ts +++ b/src/app/api/custom_generate/route.ts @@ -1,6 +1,9 @@ import { NextResponse, NextRequest } from "next/server"; import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + export const dynamic = "force-dynamic"; + export async function POST(req: NextRequest) { if (req.method === 'POST') { try { @@ -9,7 +12,10 @@ export async function POST(req: NextRequest) { if (!prompt || !tags || !title) { return new NextResponse(JSON.stringify({ error: 'Prompt, tags, and title are required' }), { status: 400, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } const audioInfo = await (await sunoApi).custom_generate( @@ -19,25 +25,44 @@ export async function POST(req: NextRequest) { ); return new NextResponse(JSON.stringify(audioInfo), { status: 200, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } catch (error: any) { console.error('Error generating custom audio:', error.response.data); if (error.response.status === 402) { return new NextResponse(JSON.stringify({ error: error.response.data.detail }), { status: 402, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } return new NextResponse(JSON.stringify({ error: 'Internal server error' }), { status: 500, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } } else { return new NextResponse('Method Not Allowed', { - headers: { Allow: 'POST' }, + headers: { + Allow: 'POST', + ...corsHeaders + }, status: 405 }); } +} + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); } \ No newline at end of file diff --git a/src/app/api/generate/route.ts b/src/app/api/generate/route.ts index ea8d06a..3469f41 100644 --- a/src/app/api/generate/route.ts +++ b/src/app/api/generate/route.ts @@ -1,6 +1,9 @@ import { NextResponse, NextRequest } from "next/server"; import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + export const dynamic = "force-dynamic"; + export async function POST(req: NextRequest) { if (req.method === 'POST') { try { @@ -10,7 +13,10 @@ export async function POST(req: NextRequest) { if (!prompt) { return new NextResponse(JSON.stringify({ error: 'Prompt is required' }), { status: 400, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } @@ -18,25 +24,45 @@ export async function POST(req: NextRequest) { return new NextResponse(JSON.stringify(audioInfo), { status: 200, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } catch (error: any) { console.error('Error generating custom audio:', JSON.stringify(error.response.data)); if (error.response.status === 402) { return new NextResponse(JSON.stringify({ error: error.response.data.detail }), { status: 402, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } return new NextResponse(JSON.stringify({ error: 'Internal server error: ' + JSON.stringify(error.response.data.detail) }), { status: 500, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } } else { return new NextResponse('Method Not Allowed', { - headers: { Allow: 'POST' }, + headers: { + Allow: 'POST', + ...corsHeaders + }, status: 405 }); } +} + + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); } \ No newline at end of file diff --git a/src/app/api/generate_lyrics/route.ts b/src/app/api/generate_lyrics/route.ts index 732ae6b..91360ad 100644 --- a/src/app/api/generate_lyrics/route.ts +++ b/src/app/api/generate_lyrics/route.ts @@ -1,6 +1,9 @@ import { NextResponse, NextRequest } from "next/server"; import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + export const dynamic = "force-dynamic"; + export async function POST(req: NextRequest) { if (req.method === 'POST') { try { @@ -10,7 +13,10 @@ export async function POST(req: NextRequest) { if (!prompt) { return new NextResponse(JSON.stringify({ error: 'Prompt is required' }), { status: 400, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } @@ -18,25 +24,44 @@ export async function POST(req: NextRequest) { return new NextResponse(JSON.stringify(lyrics), { status: 200, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } catch (error: any) { console.error('Error generating lyrics:', JSON.stringify(error.response.data)); if (error.response.status === 402) { return new NextResponse(JSON.stringify({ error: error.response.data.detail }), { status: 402, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } return new NextResponse(JSON.stringify({ error: 'Internal server error: ' + JSON.stringify(error.response.data.detail) }), { status: 500, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } } else { return new NextResponse('Method Not Allowed', { - headers: { Allow: 'POST' }, + headers: { + Allow: 'POST', + ...corsHeaders + }, status: 405 }); } +} + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); } \ No newline at end of file diff --git a/src/app/api/get/route.ts b/src/app/api/get/route.ts index 991e0be..f3f34f2 100644 --- a/src/app/api/get/route.ts +++ b/src/app/api/get/route.ts @@ -1,5 +1,7 @@ import { NextResponse, NextRequest } from "next/server"; import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + export const dynamic = "force-dynamic"; export async function GET(req: NextRequest) { @@ -17,20 +19,36 @@ export async function GET(req: NextRequest) { return new NextResponse(JSON.stringify(audioInfo), { status: 200, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } catch (error) { console.error('Error fetching audio:', error); return new NextResponse(JSON.stringify({ error: 'Internal server error' }), { status: 500, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } } else { return new NextResponse('Method Not Allowed', { - headers: { Allow: 'GET' }, + headers: { + Allow: 'GET', + ...corsHeaders + }, status: 405 }); } +} + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); } \ No newline at end of file diff --git a/src/app/api/get_limit/route.ts b/src/app/api/get_limit/route.ts index 9568a44..514c920 100644 --- a/src/app/api/get_limit/route.ts +++ b/src/app/api/get_limit/route.ts @@ -1,6 +1,9 @@ import { NextResponse, NextRequest } from "next/server"; import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + export const dynamic = "force-dynamic"; + export async function GET(req: NextRequest) { if (req.method === 'GET') { try { @@ -10,20 +13,36 @@ export async function GET(req: NextRequest) { return new NextResponse(JSON.stringify(limit), { status: 200, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } catch (error) { console.error('Error fetching limit:', error); return new NextResponse(JSON.stringify({ error: 'Internal server error. ' + error }), { status: 500, - headers: { 'Content-Type': 'application/json' } + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } }); } } else { return new NextResponse('Method Not Allowed', { - headers: { Allow: 'GET' }, + headers: { + Allow: 'GET', + ...corsHeaders + }, status: 405 }); } +} + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); } \ No newline at end of file diff --git a/src/app/docs/page.tsx b/src/app/docs/page.tsx index a0af96c..ea931c2 100644 --- a/src/app/docs/page.tsx +++ b/src/app/docs/page.tsx @@ -20,6 +20,8 @@ export default function Docs() { \`\`\`bash - \`/api/generate\`: Generate music +- \`/v1/chat/completions\`: Generate music - Call the generate API in a format + that works with OpenAI’s API. - \`/api/custom_generate\`: Generate music (Custom Mode, support setting lyrics, music style, title, etc.) - \`/api/generate_lyrics\`: Generate lyrics based on prompt diff --git a/src/app/docs/swagger-suno-api.json b/src/app/docs/swagger-suno-api.json index 9019d63..f2fe496 100644 --- a/src/app/docs/swagger-suno-api.json +++ b/src/app/docs/swagger-suno-api.json @@ -11,32 +11,22 @@ } ], "paths": { - "/api/custom_generate": { + "/api/generate": { "post": { - "summary": "Generate Audio - Custom Mode", - "description": "The custom mode enables users to provide additional details about the music, such as music genre, lyrics, and more.\n\n 2 audio files will be generated for each request, consuming a total of 10 credits. \n\n `wait_audio` can be set to API mode:\n\n\u2022 By default, it is set to false, which indicates the background mode. It will only return audio task information, and you will need to call the get API to retrieve detailed audio information.\n\n\u2022 If set to true, it simulates synchronous mode. The API will wait for a maximum of 100s until the audio is generated, and will directly return the audio link and other information. Recommend using in GPTs and other agents.", + "summary": "Generate audio based on Prompt.", + "description": "It will automatically fill in the lyrics.\n\n2 audio files will be generated for each request, consuming a total of 10 credits.\n\n`wait_audio` can be set to API mode:\n\n\u2022 By default, it is set to `false`, which indicates the background mode. It will only return audio task information, and you will need to call the get API to retrieve detailed audio information.\n\n\u2022 If set to `true`, it simulates synchronous mode. The API will wait for a maximum of 100s until the audio is generated, and will directly return the audio link and other information. Recommend using in GPTs and other agents.", "tags": ["default"], "requestBody": { "content": { "application/json": { "schema": { "type": "object", - "required": ["prompt", "tags", "title"], + "required": ["prompt", "make_instrumental", "wait_audio"], "properties": { "prompt": { "type": "string", - "description": "Detailed prompt, including information such as music lyrics.", - "example": "[Verse 1]\nCruel flames of war engulf this land\nBattlefields filled with death and dread\nInnocent souls in darkness, they rest\nMy heart trembles in this silent test\n\n[Verse 2]\nPeople weep for loved ones lost\nBattered bodies bear the cost\nSeeking peace and hope once known\nOur grief transforms to hearts of stone\n\n[Chorus]\nSilent battlegrounds, no birds' song\nShadows of war, where we don't belong\nMay flowers of peace bloom in this place\nLet's guard this precious dream with grace\n\n[Bridge]\nThrough the ashes, we will rise\nHand in hand, towards peaceful skies\nNo more sorrow, no more pain\nTogether, we'll break these chains\n\n[Chorus]\nSilent battlegrounds, no birds' song\nShadows of war, where we don't belong\nMay flowers of peace bloom in this place\nLet's guard this precious dream with grace\n\n[Outro]\nIn unity, our strength will grow\nA brighter future, we'll soon know\nFrom the ruins, hope will spring\nA new dawn, we'll together bring" - }, - "tags": { - "type": "string", - "description": "Music genre", - "example": "pop metal male melancholic" - }, - "title": { - "type": "string", - "description": "Music title", - "example": "Silent Battlefield" + "description": "Prompt", + "example": "A popular heavy metal song about war, sung by a deep-voiced male singer, slowly and melodiously. The lyrics depict the sorrow of people after the war." }, "make_instrumental": { "type": "boolean", @@ -79,10 +69,10 @@ } } }, - "/api/generate": { + "/v1/chat/completions": { "post": { - "summary": "Generate audio based on Prompt.", - "description": "It will automatically fill in the lyrics.\n\n2 audio files will be generated for each request, consuming a total of 10 credits.\n\n`wait_audio` can be set to API mode:\n\n\u2022 By default, it is set to `false`, which indicates the background mode. It will only return audio task information, and you will need to call the get API to retrieve detailed audio information.\n\n\u2022 If set to `true`, it simulates synchronous mode. The API will wait for a maximum of 100s until the audio is generated, and will directly return the audio link and other information. Recommend using in GPTs and other agents.", + "summary": "Generate audio based on Prompt - OpenAI API format compatibility.", + "description": "Convert the `/api/generate` API to be compatible with the OpenAI `/v1/chat/completions` API format. \n\nGenerally used in OpenAI compatible clients.", "tags": ["default"], "requestBody": { "content": { @@ -95,6 +85,58 @@ "type": "string", "description": "Prompt", "example": "A popular heavy metal song about war, sung by a deep-voiced male singer, slowly and melodiously. The lyrics depict the sorrow of people after the war." + } + } + } + } + } + }, + "responses": { + "200": { + "description": "success", + "content": { + "application/json": { + "schema": { + "type": "object", + "properties": { + "data": { + "type": "string", + "description": "Text description for music, with details like title, album cover, lyrics, and more." + } + } + } + } + } + } + } + } + }, + "/api/custom_generate": { + "post": { + "summary": "Generate Audio - Custom Mode", + "description": "The custom mode enables users to provide additional details about the music, such as music genre, lyrics, and more.\n\n 2 audio files will be generated for each request, consuming a total of 10 credits. \n\n `wait_audio` can be set to API mode:\n\n\u2022 By default, it is set to false, which indicates the background mode. It will only return audio task information, and you will need to call the get API to retrieve detailed audio information.\n\n\u2022 If set to true, it simulates synchronous mode. The API will wait for a maximum of 100s until the audio is generated, and will directly return the audio link and other information. Recommend using in GPTs and other agents.", + "tags": ["default"], + "requestBody": { + "content": { + "application/json": { + "schema": { + "type": "object", + "required": ["prompt", "tags", "title"], + "properties": { + "prompt": { + "type": "string", + "description": "Detailed prompt, including information such as music lyrics.", + "example": "[Verse 1]\nCruel flames of war engulf this land\nBattlefields filled with death and dread\nInnocent souls in darkness, they rest\nMy heart trembles in this silent test\n\n[Verse 2]\nPeople weep for loved ones lost\nBattered bodies bear the cost\nSeeking peace and hope once known\nOur grief transforms to hearts of stone\n\n[Chorus]\nSilent battlegrounds, no birds' song\nShadows of war, where we don't belong\nMay flowers of peace bloom in this place\nLet's guard this precious dream with grace\n\n[Bridge]\nThrough the ashes, we will rise\nHand in hand, towards peaceful skies\nNo more sorrow, no more pain\nTogether, we'll break these chains\n\n[Chorus]\nSilent battlegrounds, no birds' song\nShadows of war, where we don't belong\nMay flowers of peace bloom in this place\nLet's guard this precious dream with grace\n\n[Outro]\nIn unity, our strength will grow\nA brighter future, we'll soon know\nFrom the ruins, hope will spring\nA new dawn, we'll together bring" + }, + "tags": { + "type": "string", + "description": "Music genre", + "example": "pop metal male melancholic" + }, + "title": { + "type": "string", + "description": "Music title", + "example": "Silent Battlefield" }, "make_instrumental": { "type": "boolean", diff --git a/src/app/page.tsx b/src/app/page.tsx index cb38d45..3be0ccc 100644 --- a/src/app/page.tsx +++ b/src/app/page.tsx @@ -18,6 +18,7 @@ We update quickly, please star us on Github: [github.com/gcui-art/suno-api](htt ## 🌟 Features - Perfectly implements the creation API from \`app.suno.ai\` +- Compatible with the format of OpenAI’s \`/v1/chat/completions\` API. - Automatically keep the account active. - Supports \`Custom Mode\` - One-click deployment to Vercel @@ -96,6 +97,8 @@ Suno API currently mainly implements the following APIs: \`\`\`bash - \`/api/generate\`: Generate music +- \`/v1/chat/completions\`: Generate music - Call the generate API in a format + that works with OpenAI’s API. - \`/api/custom_generate\`: Generate music (Custom Mode, support setting lyrics, music style, title, etc.) - \`/api/generate_lyrics\`: Generate lyrics based on prompt diff --git a/src/app/v1/chat/completions/route.ts b/src/app/v1/chat/completions/route.ts new file mode 100644 index 0000000..e2f44fe --- /dev/null +++ b/src/app/v1/chat/completions/route.ts @@ -0,0 +1,61 @@ +import { NextResponse, NextRequest } from "next/server"; +import { sunoApi } from "@/lib/SunoApi"; +import { corsHeaders } from "@/lib/utils"; + +export const dynamic = "force-dynamic"; + +/** + * desc + * + */ +export async function POST(req: NextRequest) { + try { + + const body = await req.json(); + + let userMessage = null; + const { messages } = body; + for (let message of messages) { + if (message.role == 'user') { + userMessage = message; + } + } + + if (!userMessage) { + return new NextResponse(JSON.stringify({ error: 'Prompt message is required' }), { + status: 400, + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } + }); + } + + + const audioInfo = await (await sunoApi).generate(userMessage.content, true, true); + + const audio = audioInfo[0] + const data = `## Song Title: ${audio.title}\n\n### Lyrics:\n${audio.lyric}\n### Listen to the song: ${audio.audio_url}` + + return new NextResponse(data, { + status: 200, + headers: corsHeaders + }); + } catch (error: any) { + console.error('Error generating audio:', JSON.stringify(error.response.data)); + return new NextResponse(JSON.stringify({ error: 'Internal server error: ' + JSON.stringify(error.response.data.detail) }), { + status: 500, + headers: { + 'Content-Type': 'application/json', + ...corsHeaders + } + }); + } +} + +export async function OPTIONS(request: Request) { + return new Response(null, { + status: 200, + headers: corsHeaders + }); +} \ No newline at end of file diff --git a/src/lib/SunoApi.ts b/src/lib/SunoApi.ts index b77b98c..cde4851 100644 --- a/src/lib/SunoApi.ts +++ b/src/lib/SunoApi.ts @@ -27,7 +27,7 @@ export interface AudioInfo { class SunoApi { private static BASE_URL: string = 'https://studio-api.suno.ai'; - private static CLERK_BASE_URL: string = 'https://clerk.suno.ai'; + private static CLERK_BASE_URL: string = 'https://clerk.suno.com'; private readonly client: AxiosInstance; private sid?: string; @@ -327,4 +327,4 @@ const newSunoApi = async (cookie: string) => { return await sunoApi.init(); } -export const sunoApi = newSunoApi(process.env.SUNO_COOKIE || ''); \ No newline at end of file +export const sunoApi = newSunoApi(process.env.SUNO_COOKIE || ''); diff --git a/src/lib/utils.ts b/src/lib/utils.ts index ddabf9d..4822741 100644 --- a/src/lib/utils.ts +++ b/src/lib/utils.ts @@ -18,4 +18,11 @@ export const sleep = (x: number, y?: number): Promise