-
Notifications
You must be signed in to change notification settings - Fork 25
/
openai.lua
165 lines (141 loc) · 4.33 KB
/
openai.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
local util = require('model.util')
local sse = require('model.util.sse')
local M = {}
local default_params = {
model = 'gpt-4o-mini',
stream = true,
}
M.default_prompt = {
provider = M,
builder = function(input)
return {
messages = {
{
role = 'user',
content = input,
},
},
}
end,
}
local function extract_chat_data(item)
local data = util.json.decode(item)
if data ~= nil and data.choices ~= nil then
return {
content = (data.choices[1].delta or {}).content,
finish_reason = data.choices[1].finish_reason,
}
end
end
---@deprecated Completion endpoints are pretty outdated
local function extract_completion_data(item)
local data = util.json.decode(item)
if data ~= nil and data.choices ~= nil then
return {
content = (data.choices[1] or {}).text,
finish_reason = data.choices[1].finish_reason,
}
end
end
---@param handlers StreamHandlers
---@param params? any Additional options for OpenAI endpoint
---@param options? { url?: string, endpoint?: string, authorization?: string } Request endpoint and url. Defaults to 'https://api.openai.com/v1/' and 'chat/completions'. `authorization` overrides the request auth header. If url is provided the environment key will not be sent, you'll need to provide an authorization.
function M.request_completion(handlers, params, options)
options = options or {}
local headers = { ['Content-Type'] = 'application/json' }
if options.authorization then
headers.Authorization = options.authorization
elseif not options.url then -- only check the OpenAI env key if options.url wasn't set
headers.Authorization = 'Bearer ' .. util.env('OPENAI_API_KEY')
end
local endpoint = options.endpoint or 'chat/completions' -- TODO does this make compat harder?
local extract_data = endpoint == 'chat/completions' and extract_chat_data
or extract_completion_data
local completion = ''
return sse.curl_client({
headers = headers,
method = 'POST',
url = util.path.join(options.url or 'https://api.openai.com/v1/', endpoint),
body = vim.tbl_deep_extend('force', default_params, params),
}, {
on_message = function(message, pending)
local data = extract_data(message.data)
if data == nil then
if not message.data == '[DONE]' then
handlers.on_error(
vim.inspect({
data = message.data,
pending = pending,
}),
'Unrecognized SSE message data'
)
end
else
if data.content ~= nil then
completion = completion .. data.content
handlers.on_partial(data.content)
end
if data.finish_reason ~= nil then
handlers.on_finish(completion, data.finish_reason)
end
end
end,
on_other = function(content)
-- Non-SSE message likely means there was an error
handlers.on_error(content, 'OpenAI API error')
end,
on_error = handlers.on_error,
})
end
---@param standard_prompt StandardPrompt
function M.adapt(standard_prompt)
return {
messages = util.table.flatten({
{
role = 'system',
content = standard_prompt.instruction,
},
standard_prompt.fewshot,
standard_prompt.messages,
}),
}
end
--- Sets default openai provider params. Currently enforces `stream = true`.
function M.initialize(opts)
default_params = vim.tbl_deep_extend('force', default_params, opts or {}, {
stream = true, -- force streaming since data parsing will break otherwise
})
end
-- These are convenience exports for building prompt params specific to this provider
M.prompt = {}
function M.prompt.input_as_message(input)
return {
role = 'user',
content = input,
}
end
function M.prompt.add_args_as_last_message(messages, context)
if #context.args > 0 then
table.insert(messages, {
role = 'user',
content = context.args,
})
end
return messages
end
function M.prompt.input_and_args_as_messages(input, context)
return {
messages = M.add_args_as_last_message(M.input_as_message(input), context),
}
end
function M.prompt.with_system_message(text)
return function(input, context)
local body = M.input_and_args_as_messages(input, context)
table.insert(body.messages, 1, {
role = 'system',
content = text,
})
return body
end
end
return M