mirror of
https://github.com/lewis6991/gitsigns.nvim
synced 2025-04-04 23:39:47 +00:00
Some checks are pending
CI / commit_lint (push) Waiting to run
CI / test (nightly) (push) Waiting to run
CI / test (v0.10.0) (push) Waiting to run
CI / test (v0.9.5) (push) Waiting to run
CI / stylua (push) Waiting to run
CI / luals (push) Waiting to run
CI / doc (push) Waiting to run
release-please / release-please (push) Waiting to run
release-please / luarocks-upload (push) Blocked by required conditions
release-please / update-doc (push) Blocked by required conditions
Fixes #1168
437 lines
10 KiB
Lua
437 lines
10 KiB
Lua
local log = require('gitsigns.debug.log')
|
|
local async = require('gitsigns.async')
|
|
local util = require('gitsigns.util')
|
|
local Repo = require('gitsigns.git.repo')
|
|
|
|
local check_version = require('gitsigns.git.version').check
|
|
|
|
local M = {}
|
|
|
|
M.Repo = Repo
|
|
|
|
--- @param file string
|
|
--- @return boolean
|
|
local function in_git_dir(file)
|
|
for _, p in ipairs(vim.split(file, util.path_sep)) do
|
|
if p == '.git' then
|
|
return true
|
|
end
|
|
end
|
|
return false
|
|
end
|
|
|
|
--- @class Gitsigns.GitObj
|
|
--- @field file string
|
|
--- @field encoding string
|
|
--- @field i_crlf boolean Object has crlf
|
|
--- @field w_crlf boolean Working copy has crlf
|
|
--- @field mode_bits string
|
|
--- @field revision? string Revision the object is tracking against. Nil for index
|
|
--- @field object_name string The fixed object name to use.
|
|
--- @field relpath string
|
|
--- @field orig_relpath? string Use for tracking moved files
|
|
--- @field repo Gitsigns.Repo
|
|
--- @field has_conflicts? boolean
|
|
---
|
|
--- @field lock? true
|
|
local Obj = {}
|
|
|
|
M.Obj = Obj
|
|
|
|
local git_command = require('gitsigns.git.cmd')
|
|
|
|
--- @async
|
|
--- @param file_cmp string
|
|
--- @param file_buf string
|
|
--- @param indent_heuristic? boolean
|
|
--- @param diff_algo string
|
|
--- @return string[] stdout
|
|
--- @return string? stderr
|
|
--- @return integer code
|
|
function M.diff(file_cmp, file_buf, indent_heuristic, diff_algo)
|
|
return git_command({
|
|
'-c',
|
|
'core.safecrlf=false',
|
|
'diff',
|
|
'--color=never',
|
|
'--' .. (indent_heuristic and '' or 'no-') .. 'indent-heuristic',
|
|
'--diff-algorithm=' .. diff_algo,
|
|
'--patch-with-raw',
|
|
'--unified=0',
|
|
file_cmp,
|
|
file_buf,
|
|
}, {
|
|
-- git-diff implies --exit-code
|
|
ignore_error = true,
|
|
})
|
|
end
|
|
|
|
--- @async
|
|
--- @param revision? string
|
|
function Obj:update_revision(revision)
|
|
self.revision = util.norm_base(revision)
|
|
self:update()
|
|
end
|
|
|
|
--- @async
|
|
--- @param update_relpath? boolean
|
|
--- @param silent? boolean
|
|
--- @return boolean
|
|
function Obj:update(update_relpath, silent)
|
|
local old_object_name = self.object_name
|
|
local props = self:file_info(self.file, silent)
|
|
|
|
if update_relpath then
|
|
self.relpath = props.relpath
|
|
end
|
|
self.object_name = props.object_name
|
|
self.mode_bits = props.mode_bits
|
|
self.has_conflicts = props.has_conflicts
|
|
self.i_crlf = props.i_crlf
|
|
self.w_crlf = props.w_crlf
|
|
|
|
return old_object_name ~= self.object_name
|
|
end
|
|
|
|
--- @class (exact) Gitsigns.FileInfo
|
|
--- @field relpath string
|
|
--- @field i_crlf? boolean
|
|
--- @field w_crlf? boolean
|
|
--- @field mode_bits? string
|
|
--- @field object_name? string
|
|
--- @field has_conflicts? true
|
|
|
|
--- @return boolean
|
|
function Obj:from_tree()
|
|
return self.revision ~= nil and not vim.startswith(self.revision, ':')
|
|
end
|
|
|
|
--- @async
|
|
--- @param file? string
|
|
--- @param silent? boolean
|
|
--- @return Gitsigns.FileInfo
|
|
function Obj:file_info(file, silent)
|
|
if self:from_tree() then
|
|
return self:file_info_tree(file, silent)
|
|
else
|
|
return self:file_info_index(file, silent)
|
|
end
|
|
end
|
|
|
|
--- @private
|
|
--- @async
|
|
--- Get information about files in the index and the working tree
|
|
--- @param file? string
|
|
--- @param silent? boolean
|
|
--- @return Gitsigns.FileInfo
|
|
function Obj:file_info_index(file, silent)
|
|
local has_eol = check_version(2, 9)
|
|
|
|
-- --others + --exclude-standard means ignored files won't return info, but
|
|
-- untracked files will. Unlike file_info_tree which won't return untracked
|
|
-- files.
|
|
local cmd = {
|
|
'-c',
|
|
'core.quotepath=off',
|
|
'ls-files',
|
|
'--stage',
|
|
'--others',
|
|
'--exclude-standard',
|
|
}
|
|
|
|
if has_eol then
|
|
cmd[#cmd + 1] = '--eol'
|
|
end
|
|
|
|
cmd[#cmd + 1] = file or self.file
|
|
|
|
local results, stderr = self.repo:command(cmd, { ignore_error = true })
|
|
|
|
if stderr and not silent then
|
|
-- ignore_error for the cases when we run:
|
|
-- git ls-files --others exists/nonexist
|
|
if not stderr:match('^warning: could not open directory .*: No such file or directory') then
|
|
log.eprint(stderr)
|
|
end
|
|
end
|
|
|
|
local relpath_idx = has_eol and 2 or 1
|
|
|
|
local result = {}
|
|
for _, line in ipairs(results) do
|
|
local parts = vim.split(line, '\t')
|
|
if #parts > relpath_idx then -- tracked file
|
|
local attrs = vim.split(parts[1], '%s+')
|
|
local stage = tonumber(attrs[3])
|
|
if stage <= 1 then
|
|
result.mode_bits = attrs[1]
|
|
result.object_name = attrs[2]
|
|
else
|
|
result.has_conflicts = true
|
|
end
|
|
|
|
if has_eol then
|
|
result.relpath = parts[3]
|
|
local eol = vim.split(parts[2], '%s+')
|
|
result.i_crlf = eol[1] == 'i/crlf'
|
|
result.w_crlf = eol[2] == 'w/crlf'
|
|
else
|
|
result.relpath = parts[2]
|
|
end
|
|
else -- untracked file
|
|
result.relpath = parts[relpath_idx]
|
|
end
|
|
end
|
|
|
|
return result
|
|
end
|
|
|
|
--- @private
|
|
--- @async
|
|
--- Get information about files in a certain revision
|
|
--- @param file? string
|
|
--- @param silent? boolean
|
|
--- @return Gitsigns.FileInfo
|
|
function Obj:file_info_tree(file, silent)
|
|
local results, stderr = self.repo:command({
|
|
'-c',
|
|
'core.quotepath=off',
|
|
'ls-tree',
|
|
self.revision,
|
|
file or self.file,
|
|
}, { ignore_error = true })
|
|
|
|
if stderr then
|
|
if not silent then
|
|
log.eprint(stderr)
|
|
end
|
|
return {}
|
|
end
|
|
|
|
local info_line = results[1]
|
|
if not info_line then
|
|
return {}
|
|
end
|
|
|
|
local info, relpath = unpack(vim.split(info_line, '\t'))
|
|
local mode_bits, objtype, object_name = unpack(vim.split(info, '%s+'))
|
|
assert(objtype == 'blob')
|
|
|
|
return {
|
|
mode_bits = mode_bits,
|
|
object_name = object_name,
|
|
relpath = relpath,
|
|
}
|
|
end
|
|
|
|
--- @async
|
|
--- @param revision? string
|
|
--- @return string[] stdout, string? stderr
|
|
function Obj:get_show_text(revision)
|
|
if revision and not self.relpath then
|
|
log.dprint('no relpath')
|
|
return {}
|
|
end
|
|
|
|
local object = revision and (revision .. ':' .. self.relpath) or self.object_name
|
|
|
|
if not object then
|
|
log.dprint('no revision or object_name')
|
|
return { '' }
|
|
end
|
|
|
|
local stdout, stderr = self.repo:get_show_text(object, self.encoding)
|
|
|
|
if not self.i_crlf and self.w_crlf then
|
|
-- Add cr
|
|
-- Do not add cr to the newline at the end of file
|
|
for i = 1, #stdout - 1 do
|
|
stdout[i] = stdout[i] .. '\r'
|
|
end
|
|
end
|
|
|
|
return stdout, stderr
|
|
end
|
|
|
|
local function autocmd_changed(file)
|
|
vim.schedule(function()
|
|
vim.api.nvim_exec_autocmds('User', {
|
|
pattern = 'GitSignsChanged',
|
|
modeline = false,
|
|
data = { file = file },
|
|
})
|
|
end)
|
|
end
|
|
|
|
--- @async
|
|
function Obj:unstage_file()
|
|
self.lock = true
|
|
self.repo:command({ 'reset', self.file })
|
|
self.lock = nil
|
|
autocmd_changed(self.file)
|
|
end
|
|
|
|
--- @async
|
|
--- @param contents? string[]
|
|
--- @param lnum? integer
|
|
--- @param revision? string
|
|
--- @param opts? Gitsigns.BlameOpts
|
|
--- @return table<integer,Gitsigns.BlameInfo?>
|
|
function Obj:run_blame(contents, lnum, revision, opts)
|
|
return require('gitsigns.git.blame').run_blame(self, contents, lnum, revision, opts)
|
|
end
|
|
|
|
--- @async
|
|
--- @private
|
|
function Obj:ensure_file_in_index()
|
|
self.lock = true
|
|
if self.object_name and not self.has_conflicts then
|
|
return
|
|
end
|
|
|
|
if not self.object_name then
|
|
-- If there is no object_name then it is not yet in the index so add it
|
|
self.repo:command({ 'add', '--intent-to-add', self.file })
|
|
else
|
|
-- Update the index with the common ancestor (stage 1) which is what bcache
|
|
-- stores
|
|
local info = string.format('%s,%s,%s', self.mode_bits, self.object_name, self.relpath)
|
|
self.repo:command({ 'update-index', '--add', '--cacheinfo', info })
|
|
end
|
|
|
|
self:update()
|
|
self.lock = nil
|
|
end
|
|
|
|
--- @async
|
|
--- Stage 'lines' as the entire contents of the file
|
|
--- @param lines string[]
|
|
function Obj:stage_lines(lines)
|
|
self.lock = true
|
|
|
|
-- Concatenate the lines into a single string to ensure EOL
|
|
-- is respected
|
|
local text = table.concat(lines, '\n')
|
|
|
|
local new_object = self.repo:command({
|
|
'hash-object',
|
|
'-w',
|
|
'--path',
|
|
self.relpath,
|
|
'--stdin',
|
|
}, { stdin = text })[1]
|
|
|
|
self.repo:command({
|
|
'update-index',
|
|
'--cacheinfo',
|
|
string.format('%s,%s,%s', self.mode_bits, new_object, self.relpath),
|
|
})
|
|
|
|
self.lock = nil
|
|
autocmd_changed(self.file)
|
|
end
|
|
|
|
local sleep = async.awrap(2, function(duration, cb)
|
|
vim.defer_fn(cb, duration)
|
|
end)
|
|
|
|
--- @async
|
|
--- @param hunks Gitsigns.Hunk.Hunk[]
|
|
--- @param invert? boolean
|
|
--- @return string? err
|
|
function Obj:stage_hunks(hunks, invert)
|
|
self.lock = true
|
|
self:ensure_file_in_index()
|
|
|
|
local patch = require('gitsigns.hunks').create_patch(self.relpath, hunks, self.mode_bits, invert)
|
|
|
|
if not self.i_crlf and self.w_crlf then
|
|
-- Remove cr
|
|
for i = 1, #patch do
|
|
patch[i] = patch[i]:gsub('\r$', '')
|
|
end
|
|
end
|
|
|
|
local stat, err = pcall(function()
|
|
self.repo:command({
|
|
'apply',
|
|
'--whitespace=nowarn',
|
|
'--cached',
|
|
'--unidiff-zero',
|
|
'-',
|
|
}, {
|
|
stdin = patch,
|
|
})
|
|
end)
|
|
|
|
if not stat then
|
|
self.lock = nil
|
|
return err
|
|
end
|
|
|
|
-- Staging operations cause IO of the git directory so wait some time
|
|
-- for the changes to settle.
|
|
sleep(100)
|
|
|
|
self.lock = nil
|
|
autocmd_changed(self.file)
|
|
end
|
|
|
|
--- @async
|
|
--- @return string?
|
|
function Obj:has_moved()
|
|
local out = self.repo:command({ 'diff', '--name-status', '-C', '--cached' })
|
|
local orig_relpath = self.orig_relpath or self.relpath
|
|
for _, l in ipairs(out) do
|
|
local parts = vim.split(l, '%s+')
|
|
if #parts == 3 then
|
|
local orig, new = parts[2], parts[3]
|
|
if orig_relpath == orig then
|
|
self.orig_relpath = orig_relpath
|
|
self.relpath = new
|
|
self.file = self.repo.toplevel .. '/' .. new
|
|
return new
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
--- @async
|
|
--- @param file string
|
|
--- @param revision string?
|
|
--- @param encoding string
|
|
--- @param gitdir string?
|
|
--- @param toplevel string?
|
|
--- @return Gitsigns.GitObj?
|
|
function Obj.new(file, revision, encoding, gitdir, toplevel)
|
|
if in_git_dir(file) then
|
|
log.dprint('In git dir')
|
|
return nil
|
|
end
|
|
|
|
if not vim.startswith(file, '/') and toplevel then
|
|
file = toplevel .. util.path_sep .. file
|
|
end
|
|
|
|
local repo = Repo.get(util.dirname(file), gitdir, toplevel)
|
|
if not repo then
|
|
log.dprint('Not in git repo')
|
|
return
|
|
end
|
|
|
|
local self = setmetatable({}, { __index = Obj })
|
|
self.repo = repo
|
|
self.file = file
|
|
self.revision = util.norm_base(revision)
|
|
self.encoding = encoding
|
|
|
|
-- When passing gitdir and toplevel, suppress stderr when resolving the file
|
|
local silent = gitdir ~= nil and toplevel ~= nil
|
|
|
|
self:update(true, silent)
|
|
|
|
return self
|
|
end
|
|
|
|
return M
|