mirror of
https://github.com/ikechan8370/chatgpt-plugin.git
synced 2025-12-16 21:37:11 +00:00
123 lines
3.4 KiB
JavaScript
123 lines
3.4 KiB
JavaScript
import { v4 as uuidv4 } from 'uuid'
|
||
import {Config, defaultChatGPTAPI, officialChatGPTAPI} from '../utils/config.js'
|
||
import _ from 'lodash'
|
||
import fetch from 'node-fetch'
|
||
let proxy
|
||
if (Config.proxy) {
|
||
try {
|
||
proxy = (await import('https-proxy-agent')).default
|
||
} catch (e) {
|
||
console.warn('未安装https-proxy-agent,请在插件目录下执行pnpm add https-proxy-agent')
|
||
}
|
||
}
|
||
// API3
|
||
export class OfficialChatGPTClient {
|
||
constructor (opts = {}) {
|
||
const {
|
||
accessToken,
|
||
apiReverseUrl,
|
||
timeoutMs
|
||
} = opts
|
||
this._accessToken = accessToken
|
||
this._apiReverseUrl = apiReverseUrl
|
||
this._timeoutMs = timeoutMs
|
||
this._fetch = (url, options = {}) => {
|
||
const defaultOptions = Config.proxy
|
||
? {
|
||
agent: proxy(Config.proxy)
|
||
}
|
||
: {}
|
||
const mergedOptions = {
|
||
...defaultOptions,
|
||
...options
|
||
}
|
||
|
||
return fetch(url, mergedOptions)
|
||
}
|
||
}
|
||
|
||
async sendMessage (prompt, opts = {}) {
|
||
let {
|
||
timeoutMs = this._timeoutMs,
|
||
conversationId,
|
||
parentMessageId = uuidv4(),
|
||
messageId = uuidv4(),
|
||
action = 'next'
|
||
} = opts
|
||
let abortController = null
|
||
if (timeoutMs) {
|
||
abortController = new AbortController()
|
||
}
|
||
let url = this._apiReverseUrl || officialChatGPTAPI
|
||
if (this._apiReverseUrl && Config.proxy && !Config.apiForceUseReverse) {
|
||
// 如果配了proxy,而且有反代,但是没开启强制反代
|
||
url = officialChatGPTAPI
|
||
}
|
||
const body = {
|
||
action,
|
||
messages: [
|
||
{
|
||
id: messageId,
|
||
role: 'user',
|
||
content: {
|
||
content_type: 'text',
|
||
parts: [prompt]
|
||
}
|
||
}
|
||
],
|
||
model: Config.useGPT4 ? 'gpt-4' : 'text-davinci-002-render-sha',
|
||
parent_message_id: parentMessageId
|
||
}
|
||
if (conversationId) {
|
||
body.conversation_id = conversationId
|
||
}
|
||
let option = {
|
||
method: 'POST',
|
||
body: JSON.stringify(body),
|
||
signal: abortController?.signal,
|
||
headers: {
|
||
accept: 'text/event-stream',
|
||
'x-openai-assistant-app-id': '',
|
||
authorization: `Bearer ${this._accessToken}`,
|
||
'content-type': 'application/json',
|
||
referer: 'https://chat.openai.com/chat',
|
||
library: 'chatgpt-plugin'
|
||
},
|
||
referrer: 'https://chat.openai.com/chat'
|
||
}
|
||
const res = await this._fetch(url, option)
|
||
if (res.status === 403) {
|
||
return await this.sendMessage(prompt, opts)
|
||
}
|
||
if (res.status !== 200) {
|
||
let body = await res.json()
|
||
throw new Error(JSON.stringify(body, null, 2))
|
||
}
|
||
const decoder = new TextDecoder('utf-8')
|
||
const bodyBytes = await res.arrayBuffer()
|
||
const bodyText = decoder.decode(bodyBytes)
|
||
const events = bodyText.split('\n\n').filter(item => !_.isEmpty(item))
|
||
let fullResponse
|
||
for (let i = 0; i < events.length; i++) {
|
||
let event = events[i]
|
||
event = _.trimStart(event, 'data: ')
|
||
try {
|
||
fullResponse = JSON.parse(event)
|
||
} catch (err) {
|
||
console.log(event)
|
||
}
|
||
}
|
||
if (Config.debug) {
|
||
logger.mark(fullResponse)
|
||
}
|
||
if (!fullResponse?.message) {
|
||
throw new Error(bodyText || 'unkown error, please check log')
|
||
}
|
||
return {
|
||
text: fullResponse.message.content.parts[0],
|
||
conversationId: fullResponse.conversation_id,
|
||
id: fullResponse.message.id,
|
||
parentMessageId
|
||
}
|
||
}
|
||
}
|