2019-11-13 13:55:26 -07:00
|
|
|
--- TODO: This is implemented only for files now.
|
|
|
|
-- https://tools.ietf.org/html/rfc3986
|
|
|
|
-- https://tools.ietf.org/html/rfc2732
|
|
|
|
-- https://tools.ietf.org/html/rfc2396
|
|
|
|
|
|
|
|
|
|
|
|
local uri_decode
|
|
|
|
do
|
|
|
|
local schar = string.char
|
2020-09-14 06:12:17 -07:00
|
|
|
|
|
|
|
--- Convert hex to char
|
2021-08-22 13:55:28 -07:00
|
|
|
---@private
|
2019-11-13 13:55:26 -07:00
|
|
|
local function hex_to_char(hex)
|
|
|
|
return schar(tonumber(hex, 16))
|
|
|
|
end
|
|
|
|
uri_decode = function(str)
|
|
|
|
return str:gsub("%%([a-fA-F0-9][a-fA-F0-9])", hex_to_char)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
local uri_encode
|
|
|
|
do
|
|
|
|
local PATTERNS = {
|
|
|
|
--- RFC 2396
|
|
|
|
-- https://tools.ietf.org/html/rfc2396#section-2.2
|
|
|
|
rfc2396 = "^A-Za-z0-9%-_.!~*'()";
|
|
|
|
--- RFC 2732
|
|
|
|
-- https://tools.ietf.org/html/rfc2732
|
|
|
|
rfc2732 = "^A-Za-z0-9%-_.!~*'()[]";
|
|
|
|
--- RFC 3986
|
|
|
|
-- https://tools.ietf.org/html/rfc3986#section-2.2
|
|
|
|
rfc3986 = "^A-Za-z0-9%-._~!$&'()*+,;=:@/";
|
|
|
|
}
|
|
|
|
local sbyte, tohex = string.byte
|
|
|
|
if jit then
|
|
|
|
tohex = require'bit'.tohex
|
|
|
|
else
|
|
|
|
tohex = function(b) return string.format("%02x", b) end
|
|
|
|
end
|
2020-09-14 06:12:17 -07:00
|
|
|
|
2021-08-22 13:55:28 -07:00
|
|
|
---@private
|
2019-11-13 13:55:26 -07:00
|
|
|
local function percent_encode_char(char)
|
|
|
|
return "%"..tohex(sbyte(char), 2)
|
|
|
|
end
|
|
|
|
uri_encode = function(text, rfc)
|
|
|
|
if not text then return end
|
|
|
|
local pattern = PATTERNS[rfc] or PATTERNS.rfc3986
|
|
|
|
return text:gsub("(["..pattern.."])", percent_encode_char)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2021-08-22 13:55:28 -07:00
|
|
|
---@private
|
2019-11-13 13:55:26 -07:00
|
|
|
local function is_windows_file_uri(uri)
|
2021-07-07 15:34:22 -07:00
|
|
|
return uri:match('^file:/+[a-zA-Z]:') ~= nil
|
2019-11-13 13:55:26 -07:00
|
|
|
end
|
|
|
|
|
2020-09-14 06:12:17 -07:00
|
|
|
--- Get a URI from a file path.
|
2021-08-22 13:55:28 -07:00
|
|
|
---@param path (string): Path to file
|
|
|
|
---@return URI
|
2019-11-13 13:55:26 -07:00
|
|
|
local function uri_from_fname(path)
|
|
|
|
local volume_path, fname = path:match("^([a-zA-Z]:)(.*)")
|
|
|
|
local is_windows = volume_path ~= nil
|
|
|
|
if is_windows then
|
|
|
|
path = volume_path..uri_encode(fname:gsub("\\", "/"))
|
|
|
|
else
|
|
|
|
path = uri_encode(path)
|
|
|
|
end
|
|
|
|
local uri_parts = {"file://"}
|
|
|
|
if is_windows then
|
|
|
|
table.insert(uri_parts, "/")
|
|
|
|
end
|
|
|
|
table.insert(uri_parts, path)
|
|
|
|
return table.concat(uri_parts)
|
|
|
|
end
|
|
|
|
|
2021-07-02 07:27:04 -07:00
|
|
|
local URI_SCHEME_PATTERN = '^([a-zA-Z]+[a-zA-Z0-9+-.]*):.*'
|
2021-10-15 12:03:41 -07:00
|
|
|
local WINDOWS_URI_SCHEME_PATTERN = '^([a-zA-Z]+[a-zA-Z0-9+-.]*):[a-zA-Z]:.*'
|
2020-05-21 09:17:21 -07:00
|
|
|
|
2020-09-14 06:12:17 -07:00
|
|
|
--- Get a URI from a bufnr
|
2021-08-22 13:55:28 -07:00
|
|
|
---@param bufnr (number): Buffer number
|
|
|
|
---@return URI
|
2019-11-13 13:55:26 -07:00
|
|
|
local function uri_from_bufnr(bufnr)
|
2020-05-04 20:06:40 -07:00
|
|
|
local fname = vim.api.nvim_buf_get_name(bufnr)
|
2021-10-15 12:03:41 -07:00
|
|
|
local volume_path = fname:match("^([a-zA-Z]:).*")
|
|
|
|
local is_windows = volume_path ~= nil
|
|
|
|
local scheme
|
|
|
|
if is_windows then
|
|
|
|
fname = fname:gsub("\\", "/")
|
|
|
|
scheme = fname:match(WINDOWS_URI_SCHEME_PATTERN)
|
|
|
|
else
|
|
|
|
scheme = fname:match(URI_SCHEME_PATTERN)
|
|
|
|
end
|
2020-05-04 20:06:40 -07:00
|
|
|
if scheme then
|
|
|
|
return fname
|
|
|
|
else
|
|
|
|
return uri_from_fname(fname)
|
|
|
|
end
|
2019-11-13 13:55:26 -07:00
|
|
|
end
|
|
|
|
|
2020-09-14 06:12:17 -07:00
|
|
|
--- Get a filename from a URI
|
2021-08-22 13:55:28 -07:00
|
|
|
---@param uri (string): The URI
|
|
|
|
---@return Filename
|
2019-11-13 13:55:26 -07:00
|
|
|
local function uri_to_fname(uri)
|
2020-05-21 09:17:21 -07:00
|
|
|
local scheme = assert(uri:match(URI_SCHEME_PATTERN), 'URI must contain a scheme: ' .. uri)
|
|
|
|
if scheme ~= 'file' then
|
|
|
|
return uri
|
|
|
|
end
|
2020-03-30 05:30:24 -07:00
|
|
|
uri = uri_decode(uri)
|
2019-11-13 13:55:26 -07:00
|
|
|
-- TODO improve this.
|
|
|
|
if is_windows_file_uri(uri) then
|
2021-07-07 15:34:22 -07:00
|
|
|
uri = uri:gsub('^file:/+', '')
|
2019-11-13 13:55:26 -07:00
|
|
|
uri = uri:gsub('/', '\\')
|
|
|
|
else
|
2021-07-07 15:34:22 -07:00
|
|
|
uri = uri:gsub('^file:/+', '/')
|
2019-11-13 13:55:26 -07:00
|
|
|
end
|
2020-03-30 05:30:24 -07:00
|
|
|
return uri
|
2019-11-13 13:55:26 -07:00
|
|
|
end
|
|
|
|
|
2020-09-14 06:12:17 -07:00
|
|
|
--- Return or create a buffer for a uri.
|
2021-08-22 13:55:28 -07:00
|
|
|
---@param uri (string): The URI
|
|
|
|
---@return bufnr.
|
|
|
|
---@note Creates buffer but does not load it
|
2019-11-20 12:39:54 -07:00
|
|
|
local function uri_to_bufnr(uri)
|
2020-05-21 09:17:21 -07:00
|
|
|
local scheme = assert(uri:match(URI_SCHEME_PATTERN), 'URI must contain a scheme: ' .. uri)
|
2020-05-04 20:06:40 -07:00
|
|
|
if scheme == 'file' then
|
|
|
|
return vim.fn.bufadd(uri_to_fname(uri))
|
|
|
|
else
|
|
|
|
return vim.fn.bufadd(uri)
|
|
|
|
end
|
2019-11-20 12:39:54 -07:00
|
|
|
end
|
|
|
|
|
2019-11-13 13:55:26 -07:00
|
|
|
return {
|
|
|
|
uri_from_fname = uri_from_fname,
|
|
|
|
uri_from_bufnr = uri_from_bufnr,
|
|
|
|
uri_to_fname = uri_to_fname,
|
2019-11-20 12:39:54 -07:00
|
|
|
uri_to_bufnr = uri_to_bufnr,
|
2019-11-13 13:55:26 -07:00
|
|
|
}
|
|
|
|
-- vim:sw=2 ts=2 et
|