mirror of
https://github.com/LLM-Red-Team/kimi-free-api.git
synced 2024-11-01 18:29:19 +08:00
优化功能
This commit is contained in:
parent
78b267c3e8
commit
9a50c30ef7
@ -14,6 +14,7 @@
|
|||||||
],
|
],
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "tsup src/index.ts --format cjs,esm --sourcemap --dts --publicDir public --watch --onSuccess \"node dist/index.js\"",
|
"dev": "tsup src/index.ts --format cjs,esm --sourcemap --dts --publicDir public --watch --onSuccess \"node dist/index.js\"",
|
||||||
|
"start": "node dist/index.js",
|
||||||
"build": "tsup src/index.ts --format cjs,esm --sourcemap --dts --clean --publicDir public"
|
"build": "tsup src/index.ts --format cjs,esm --sourcemap --dts --clean --publicDir public"
|
||||||
},
|
},
|
||||||
"author": "Vinlic",
|
"author": "Vinlic",
|
||||||
|
@ -2,4 +2,5 @@ export default {
|
|||||||
API_TEST: [-9999, 'API异常错误'],
|
API_TEST: [-9999, 'API异常错误'],
|
||||||
API_REQUEST_PARAMS_INVALID: [-2000, '请求参数非法'],
|
API_REQUEST_PARAMS_INVALID: [-2000, '请求参数非法'],
|
||||||
API_REQUEST_FAILED: [-2001, '请求失败'],
|
API_REQUEST_FAILED: [-2001, '请求失败'],
|
||||||
|
API_TOKEN_EXPIRES: [-2002, 'Token已失效']
|
||||||
}
|
}
|
@ -8,69 +8,70 @@ import { createParser } from 'eventsource-parser'
|
|||||||
import logger from '@/lib/logger.ts';
|
import logger from '@/lib/logger.ts';
|
||||||
import util from '@/lib/util.ts';
|
import util from '@/lib/util.ts';
|
||||||
|
|
||||||
const TOKEN_EXPIRES = 120;
|
const ACCESS_TOKEN_EXPIRES = 300;
|
||||||
let currentAccessToken: string | null = null;
|
const accessTokenMap = new Map();
|
||||||
let currentRefreshToken: string | null = null;
|
|
||||||
let latestRefreshTime = 0;
|
|
||||||
|
|
||||||
function setRefreshToken(refreshToken: string) {
|
async function requestToken(refreshToken: string) {
|
||||||
currentRefreshToken = refreshToken || 'eyJhbGciOiJIUzUxMiIsInR5cCI6IkpXVCJ9.eyJpc3MiOiJ1c2VyLWNlbnRlciIsImV4cCI6MTcxNzY2NzkzMSwiaWF0IjoxNzA5ODkxOTMxLCJqdGkiOiJjbmxlMm1wcmRpamFpbGxzcHJuMCIsInR5cCI6InJlZnJlc2giLCJzdWIiOiJjbmVyMGgybG5sOTU3N3MzMmluZyIsInNwYWNlX2lkIjoiY25lcXA1ODNyMDdkajd1a3JqcjAiLCJhYnN0cmFjdF91c2VyX2lkIjoiY25lcXA1ODNyMDdkajd1a3JqcWcifQ.XMDecAmBq817_n3xtRqIwIlS9QQLIClS1PaVh4EY8bqhiHr8SxFxbiTEyuRuPPTnCB90eUJNc_LchLMjUo8cKA';
|
|
||||||
}
|
|
||||||
|
|
||||||
async function refreshToken() {
|
|
||||||
const refreshToken = currentRefreshToken;
|
|
||||||
const result = await axios.get('https://kimi.moonshot.cn/api/auth/token/refresh', {
|
const result = await axios.get('https://kimi.moonshot.cn/api/auth/token/refresh', {
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${refreshToken}`
|
Authorization: `Bearer ${refreshToken}`
|
||||||
}
|
},
|
||||||
|
validateStatus: () => true
|
||||||
});
|
});
|
||||||
const {
|
const {
|
||||||
access_token,
|
access_token,
|
||||||
refresh_token
|
refresh_token
|
||||||
} = checkResult(result);
|
} = checkResult(result, refreshToken);
|
||||||
currentAccessToken = access_token;
|
return {
|
||||||
currentRefreshToken = refresh_token;
|
accessToken: access_token,
|
||||||
logger.info(`Current access_token: ${currentAccessToken}`);
|
refreshToken: refresh_token,
|
||||||
logger.info(`Current refresh_token: ${currentRefreshToken}`);
|
refreshTime: util.unixTimestamp() + ACCESS_TOKEN_EXPIRES
|
||||||
logger.success('Token refresh completed');
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function requestToken() {
|
async function acquireToken(refreshToken: string): Promise<string> {
|
||||||
if (util.unixTimestamp() - latestRefreshTime > TOKEN_EXPIRES)
|
let result = accessTokenMap.get(refreshToken);
|
||||||
await refreshToken();
|
if (!result) {
|
||||||
return currentAccessToken;
|
result = await requestToken(refreshToken);
|
||||||
|
accessTokenMap.set(refreshToken, result);
|
||||||
|
}
|
||||||
|
if (util.unixTimestamp() > result.refreshTime)
|
||||||
|
result = await requestToken(refreshToken);
|
||||||
|
return result.accessToken;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function createConversation(name: string) {
|
async function createConversation(name: string, refreshToken: string) {
|
||||||
const token = await requestToken();
|
const token = await acquireToken(refreshToken);
|
||||||
const result = await axios.post('https://kimi.moonshot.cn/api/chat', {
|
const result = await axios.post('https://kimi.moonshot.cn/api/chat', {
|
||||||
name,
|
name,
|
||||||
is_example: false
|
is_example: false
|
||||||
}, {
|
}, {
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${token}`
|
Authorization: `Bearer ${token}`
|
||||||
}
|
},
|
||||||
|
validateStatus: () => true
|
||||||
});
|
});
|
||||||
const {
|
const {
|
||||||
id: convId
|
id: convId
|
||||||
} = checkResult(result);
|
} = checkResult(result, refreshToken);
|
||||||
return convId;
|
return convId;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function removeConversation(convId: string) {
|
async function removeConversation(convId: string, refreshToken: string) {
|
||||||
const token = await requestToken();
|
const token = await acquireToken(refreshToken);
|
||||||
const result = await axios.delete(`https://kimi.moonshot.cn/api/chat/${convId}`, {
|
const result = await axios.delete(`https://kimi.moonshot.cn/api/chat/${convId}`, {
|
||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${token}`
|
Authorization: `Bearer ${token}`
|
||||||
}
|
},
|
||||||
|
validateStatus: () => true
|
||||||
});
|
});
|
||||||
checkResult(result);
|
checkResult(result, refreshToken);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function createCompletionStream(messages: any[], useSearch = true) {
|
async function createCompletion(messages: any[], refreshToken: string, useSearch = true) {
|
||||||
console.log(messages);
|
logger.info(messages);
|
||||||
const convId = await createConversation(`cmpl-${util.uuid(false)}`);
|
const convId = await createConversation(`cmpl-${util.uuid(false)}`, refreshToken);
|
||||||
const token = await requestToken();
|
const token = await acquireToken(refreshToken);
|
||||||
const result = await axios.post(`https://kimi.moonshot.cn/api/chat/${convId}/completion/stream`, {
|
const result = await axios.post(`https://kimi.moonshot.cn/api/chat/${convId}/completion/stream`, {
|
||||||
messages,
|
messages,
|
||||||
use_search: useSearch
|
use_search: useSearch
|
||||||
@ -78,25 +79,87 @@ async function createCompletionStream(messages: any[], useSearch = true) {
|
|||||||
headers: {
|
headers: {
|
||||||
Authorization: `Bearer ${token}`
|
Authorization: `Bearer ${token}`
|
||||||
},
|
},
|
||||||
|
validateStatus: () => true,
|
||||||
responseType: 'stream'
|
responseType: 'stream'
|
||||||
});
|
});
|
||||||
return createTransStream(convId, result.data);
|
const answer = await receiveStream(convId, result.data);
|
||||||
|
removeConversation(convId, refreshToken)
|
||||||
|
.catch(err => console.error(err));
|
||||||
|
return answer;
|
||||||
}
|
}
|
||||||
|
|
||||||
function checkResult(result: AxiosResponse) {
|
async function createCompletionStream(messages: any[], refreshToken: string, useSearch = true) {
|
||||||
if(!result.data)
|
logger.info(messages);
|
||||||
|
const convId = await createConversation(`cmpl-${util.uuid(false)}`, refreshToken);
|
||||||
|
const token = await acquireToken(refreshToken);
|
||||||
|
const result = await axios.post(`https://kimi.moonshot.cn/api/chat/${convId}/completion/stream`, {
|
||||||
|
messages,
|
||||||
|
use_search: useSearch
|
||||||
|
}, {
|
||||||
|
headers: {
|
||||||
|
Authorization: `Bearer ${token}`
|
||||||
|
},
|
||||||
|
validateStatus: () => true,
|
||||||
|
responseType: 'stream'
|
||||||
|
});
|
||||||
|
return createTransStream(convId, result.data, () => {
|
||||||
|
removeConversation(convId, refreshToken)
|
||||||
|
.catch(err => console.error(err));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function checkResult(result: AxiosResponse, refreshToken: string) {
|
||||||
|
if(result.status == 401) {
|
||||||
|
accessTokenMap.delete(refreshToken);
|
||||||
|
throw new APIException(EX.API_REQUEST_FAILED);
|
||||||
|
}
|
||||||
|
if (!result.data)
|
||||||
return null;
|
return null;
|
||||||
const { error_type, message } = result.data;
|
const { error_type, message } = result.data;
|
||||||
if (!_.isString(error_type))
|
if (!_.isString(error_type))
|
||||||
return result.data;
|
return result.data;
|
||||||
console.log(result.data);
|
if (error_type == 'auth.token.invalid')
|
||||||
throw new APIException(EX.API_REQUEST_FAILED, message);
|
accessTokenMap.delete(refreshToken);
|
||||||
|
throw new APIException(EX.API_REQUEST_FAILED, `[请求kimi失败]: ${message}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
function createTransStream(convId: string, stream: any) {
|
async function receiveStream(convId: string, stream: any) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
const data = {
|
||||||
|
id: convId,
|
||||||
|
model: 'kimi',
|
||||||
|
object: 'chat.completion',
|
||||||
|
choices: [
|
||||||
|
{ index: 0, message: { role: 'assistant', content: '' }, finish_reason: 'stop' }
|
||||||
|
],
|
||||||
|
created: parseInt(performance.now() as any)
|
||||||
|
};
|
||||||
|
const parser = createParser(event => {
|
||||||
|
try {
|
||||||
|
if (event.type !== "event") return;
|
||||||
|
const result = _.attempt(() => JSON.parse(event.data));
|
||||||
|
if (_.isError(result))
|
||||||
|
throw new Error(`stream response invalid: ${event.data}`);
|
||||||
|
if (result.event == 'cmpl') {
|
||||||
|
data.choices[0].message.content += result.text;
|
||||||
|
}
|
||||||
|
else if (result.event == 'all_done')
|
||||||
|
resolve(data);
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
logger.error(err);
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
stream.on("data", buffer => parser.feed(buffer.toString()));
|
||||||
|
stream.once("error", err => reject(err));
|
||||||
|
stream.once("close", () => resolve(data));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function createTransStream(convId: string, stream: any, endCallback?: Function) {
|
||||||
const created = parseInt(performance.now() as any);
|
const created = parseInt(performance.now() as any);
|
||||||
const transStream = new PassThrough();
|
const transStream = new PassThrough();
|
||||||
|
|
||||||
!transStream.closed && transStream.write(`data: ${JSON.stringify({
|
!transStream.closed && transStream.write(`data: ${JSON.stringify({
|
||||||
id: convId,
|
id: convId,
|
||||||
model: 'kimi',
|
model: 'kimi',
|
||||||
@ -135,8 +198,8 @@ function createTransStream(convId: string, stream: any) {
|
|||||||
created
|
created
|
||||||
})}\n\n`;
|
})}\n\n`;
|
||||||
!transStream.closed && transStream.write(data);
|
!transStream.closed && transStream.write(data);
|
||||||
!transStream.closed && transStream.end('[DONE]');
|
!transStream.closed && transStream.end('data: [DONE]\n\n');
|
||||||
removeConversation(convId).catch(err => console.error(err));
|
endCallback && endCallback();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
@ -145,14 +208,13 @@ function createTransStream(convId: string, stream: any) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
stream.on("data", buffer => parser.feed(buffer.toString()));
|
stream.on("data", buffer => parser.feed(buffer.toString()));
|
||||||
stream.once("error", () => !transStream.closed && transStream.end('[DONE]'));
|
stream.once("error", () => !transStream.closed && transStream.end('data: [DONE]\n\n'));
|
||||||
stream.once("close", () => !transStream.closed && transStream.end('[DONE]'));
|
stream.once("close", () => !transStream.closed && transStream.end('data: [DONE]\n\n'));
|
||||||
return transStream;
|
return transStream;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
setRefreshToken,
|
|
||||||
refreshToken,
|
|
||||||
createConversation,
|
createConversation,
|
||||||
|
createCompletion,
|
||||||
createCompletionStream
|
createCompletionStream
|
||||||
};
|
};
|
||||||
|
@ -3,9 +3,10 @@ import _ from 'lodash';
|
|||||||
import Request from '@/lib/request/Request.ts';
|
import Request from '@/lib/request/Request.ts';
|
||||||
import Response from '@/lib/response/Response.ts';
|
import Response from '@/lib/response/Response.ts';
|
||||||
import chat from '@/api/controllers/chat.ts';
|
import chat from '@/api/controllers/chat.ts';
|
||||||
|
import logger from '@/lib/logger.ts';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
|
|
||||||
prefix: '/v1/chat',
|
prefix: '/v1/chat',
|
||||||
|
|
||||||
post: {
|
post: {
|
||||||
@ -13,11 +14,19 @@ export default {
|
|||||||
'/completions': async (request: Request) => {
|
'/completions': async (request: Request) => {
|
||||||
request
|
request
|
||||||
.validate('body.messages', _.isArray)
|
.validate('body.messages', _.isArray)
|
||||||
chat.setRefreshToken(request.body.refresh_token);
|
.validate('headers.authorization', _.isString)
|
||||||
const stream = await chat.createCompletionStream(request.body.messages, request.body.use_search);
|
const token = request.headers.authorization;
|
||||||
return new Response(stream, {
|
const refreshToken = token.replace('Bearer ', '');
|
||||||
type: "text/event-stream"
|
logger.info(`Refresh token: ${refreshToken}`);
|
||||||
});
|
const messages = request.body.messages;
|
||||||
|
if (request.body.stream) {
|
||||||
|
const stream = await chat.createCompletionStream(request.body.messages, refreshToken, request.body.use_search);
|
||||||
|
return new Response(stream, {
|
||||||
|
type: "text/event-stream"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return await chat.createCompletion(messages, refreshToken, request.body.use_search);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user