mirror of
https://github.com/neovim/neovim.git
synced 2026-05-08 09:20:00 -04:00
188 lines
5.8 KiB
Lua
188 lines
5.8 KiB
Lua
local M = {}
|
|
|
|
local http_methods = {
|
|
GET = true,
|
|
POST = true,
|
|
PUT = true,
|
|
PATCH = true,
|
|
HEAD = true,
|
|
DELETE = true,
|
|
}
|
|
|
|
---@alias vim.net.request.ResponseFunc fun(err: string?, response: vim.net.request.Response?)
|
|
---@alias vim.net.HttpMethod string "GET" | "POST" | "PUT" | "PATCH" | "HEAD" | "DELETE"
|
|
|
|
---@class vim.net.request.Opts
|
|
---@inlinedoc
|
|
---
|
|
---Enables verbose output.
|
|
---@field verbose? boolean
|
|
---
|
|
---Number of retries on transient failures (default: 3).
|
|
---@field retry? integer
|
|
---
|
|
---Request body for POST/PUT/PATCH requests.
|
|
---@field body? string
|
|
---
|
|
---File path to save the response body to.
|
|
---@field outpath? string
|
|
---
|
|
---Buffer to save the response body to.
|
|
---@field outbuf? integer
|
|
---
|
|
---Custom headers to send with the request. Supports basic key/value headers and empty headers as
|
|
---supported by curl. Does not support "@filename" style, internal header deletion ("Header:").
|
|
---@field headers? table<string, string>
|
|
|
|
---@class vim.net.request.Response
|
|
---
|
|
---The HTTP body of the request
|
|
---@field body string
|
|
|
|
--- Makes an HTTP request to the given URL, asynchronously passing the result to the specified
|
|
--- `on_response`, `outpath` or `outbuf`.
|
|
---
|
|
--- Examples:
|
|
--- ```lua
|
|
--- -- Write response body to file.
|
|
--- vim.net.request('https://neovim.io/charter/', {
|
|
--- outpath = 'vision.html',
|
|
--- })
|
|
---
|
|
--- -- Process the response.
|
|
--- vim.net.request(
|
|
--- 'https://api.github.com/repos/neovim/neovim',
|
|
--- {},
|
|
--- function (err, res)
|
|
--- if err then return end
|
|
--- local stars = vim.json.decode(res.body).stargazers_count
|
|
--- vim.print(('Neovim currently has %d stars'):format(stars))
|
|
--- end
|
|
--- )
|
|
---
|
|
--- -- Write to both file and current buffer, but cancel it.
|
|
--- local job = vim.net.request('https://neovim.io/charter/', {
|
|
--- outpath = 'vision.html',
|
|
--- outbuf = 0,
|
|
--- })
|
|
--- job:close()
|
|
---
|
|
--- -- Add custom headers in the request.
|
|
--- vim.net.request('https://neovim.io/charter/', {
|
|
--- headers = { Authorization = 'Bearer XYZ' },
|
|
--- })
|
|
---
|
|
--- -- POST request with body.
|
|
--- vim.net.request('POST', 'https://example.com/api', {
|
|
--- body = '{"key": "value"}',
|
|
--- headers = {['Content-Type'] = 'application/json' }
|
|
--- })
|
|
--- ```
|
|
---
|
|
--- @param method vim.net.HttpMethod (default: GET) The HTTP method (GET, POST, PUT, PATCH, HEAD, DELETE).
|
|
--- @param url string The URL for the request.
|
|
--- @param opts? vim.net.request.Opts
|
|
--- @param on_response? vim.net.request.ResponseFunc Callback invoked on request completion.
|
|
--- @overload fun(url: string, opts: vim.net.request.Opts, response: vim.net.request.ResponseFunc)
|
|
--- @overload fun(method: vim.net.HttpMethod, url: string, opts: vim.net.request.Opts, response: vim.net.request.ResponseFunc)
|
|
--- @return { close: fun() } # Object with `close()` method which cancels the request.
|
|
function M.request(method, url, opts, on_response)
|
|
if type(url) ~= 'string' then
|
|
---@diagnostic disable-next-line: cast-local-type
|
|
on_response = opts
|
|
---@diagnostic disable-next-line: no-unknown
|
|
opts = url
|
|
url = method
|
|
method = 'GET'
|
|
end
|
|
opts = opts or {}
|
|
|
|
vim.validate('method', method, function(m)
|
|
return http_methods[m] == true
|
|
end, 'method should be one of GET, POST, PUT, PATCH, HEAD, DELETE')
|
|
vim.validate('url', url, 'string')
|
|
vim.validate('opts', opts, 'table', true)
|
|
vim.validate('opts.headers', opts.headers, 'table', true)
|
|
vim.validate('opts.body', opts.body, function(b)
|
|
return (b == nil and true) or (type(b) == 'string' and not b:match('^@'))
|
|
end, true, 'body should be string and not start with @')
|
|
vim.validate('on_response', on_response, 'function', true)
|
|
|
|
local retry = opts.retry or 3
|
|
|
|
-- Build curl command
|
|
local args = { 'curl' }
|
|
if opts.verbose then
|
|
table.insert(args, '--verbose')
|
|
else
|
|
vim.list_extend(args, { '--silent', '--show-error', '--fail' })
|
|
end
|
|
vim.list_extend(args, { '--location', '--retry', tostring(retry) })
|
|
|
|
if opts.outpath then
|
|
vim.list_extend(args, { '--output', opts.outpath })
|
|
end
|
|
|
|
-- curl -X HEAD does not work and advises to use --head instead
|
|
vim.list_extend(args, method == 'HEAD' and { '--head' } or { '--request', method })
|
|
|
|
if vim.list_contains({ 'POST', 'PUT', 'PATCH' }, method) and opts.body then
|
|
vim.list_extend(args, { '--data-binary', '@-' })
|
|
end
|
|
|
|
if opts.headers then
|
|
for key, value in pairs(opts.headers) do
|
|
if type(key) ~= 'string' or type(value) ~= 'string' then
|
|
error('headers keys and values must be strings')
|
|
end
|
|
|
|
if key:match(':$') or key:match(';$') or key:match('^@') then
|
|
error('header keys must not start with @ or end with : and ;')
|
|
end
|
|
|
|
if value == '' then
|
|
vim.list_extend(args, { '--header', key .. ';' })
|
|
else
|
|
vim.list_extend(args, { '--header', key .. ': ' .. value })
|
|
end
|
|
end
|
|
end
|
|
|
|
table.insert(args, url)
|
|
|
|
local system_opts = opts.body and { stdin = opts.body } or {}
|
|
local job = vim.system(args, system_opts, function(res)
|
|
---@type string?, vim.net.request.Response?
|
|
local err, response = nil, nil
|
|
if res.signal ~= 0 then
|
|
err = ('Request killed with signal %d'):format(res.signal)
|
|
elseif res.code ~= 0 then
|
|
err = res.stderr ~= '' and res.stderr or ('Request failed with exit code %d'):format(res.code)
|
|
else
|
|
if on_response then
|
|
response = { body = res.stdout or '' }
|
|
end
|
|
end
|
|
|
|
-- nvim_buf_is_loaded and nvim_buf_set_lines are not allowed in fast context
|
|
vim.schedule(function()
|
|
if res.code == 0 and opts.outbuf and vim.api.nvim_buf_is_loaded(opts.outbuf) then
|
|
local lines = vim.split(res.stdout, '\n', { plain = true })
|
|
vim.api.nvim_buf_set_lines(opts.outbuf, 0, -1, true, lines)
|
|
end
|
|
end)
|
|
|
|
if on_response then
|
|
on_response(err, response)
|
|
end
|
|
end)
|
|
|
|
return {
|
|
close = function()
|
|
job:kill('sigint')
|
|
end,
|
|
}
|
|
end
|
|
|
|
return M
|