mirror of
https://github.com/SpaceVim/SpaceVim.git
synced 2025-01-26 15:40:06 +08:00
522 lines
14 KiB
Lua
522 lines
14 KiB
Lua
--=============================================================================
|
|
-- M.lua
|
|
-- Copyright (c) 2016-2022 Wang Shidong & Contributors
|
|
-- Author: Wang Shidong < wsdjeg@outlook.com >
|
|
-- URL: https://spacevim.org
|
|
-- License: GPLv3
|
|
--=============================================================================
|
|
|
|
local M = {}
|
|
|
|
local runners = {}
|
|
|
|
local logger = require('spacevim.logger').derive('runner')
|
|
local job = require('spacevim.api').import('job')
|
|
local file = require('spacevim.api').import('file')
|
|
local str = require('spacevim.api').import('data.string')
|
|
|
|
local code_runner_bufnr = 0
|
|
|
|
local winid = -1
|
|
|
|
local target = ''
|
|
|
|
local runner_lines = 0
|
|
|
|
local runner_jobid = 0
|
|
|
|
local runner_status = {
|
|
is_running = false,
|
|
has_errors = false,
|
|
exit_code = 0,
|
|
exit_single = 0,
|
|
}
|
|
|
|
local selected_file = ''
|
|
--- @type any[]
|
|
local start_time
|
|
--- @type any[]
|
|
local end_time
|
|
|
|
local task_status = {}
|
|
|
|
local task_stdout = {}
|
|
|
|
local task_stderr = {}
|
|
|
|
local task_problem_matcher = {}
|
|
|
|
local selected_language = ''
|
|
|
|
local function stop_runner()
|
|
if runner_status.is_running then
|
|
logger.debug('stop runner:' .. runner_jobid)
|
|
job.stop(runner_jobid)
|
|
end
|
|
end
|
|
|
|
local function close_win()
|
|
stop_runner()
|
|
if code_runner_bufnr ~= 0 and vim.api.nvim_buf_is_valid(code_runner_bufnr) then
|
|
vim.cmd('bd ' .. code_runner_bufnr)
|
|
end
|
|
end
|
|
|
|
local function insert()
|
|
vim.fn.inputsave()
|
|
local input = vim.fn.input('input >')
|
|
if vim.fn.empty(input) == 0 and runner_status.is_running then
|
|
job.send(runner_jobid, input)
|
|
end
|
|
vim.cmd('normal! :')
|
|
vim.fn.inputrestore()
|
|
end
|
|
|
|
local function open_win()
|
|
if
|
|
code_runner_bufnr ~= 0
|
|
and vim.api.nvim_buf_is_valid(code_runner_bufnr)
|
|
and vim.fn.index(vim.fn.tabpagebuflist(), code_runner_bufnr) ~= -1
|
|
then
|
|
return
|
|
end
|
|
logger.debug('open code runner windows')
|
|
local previous_wind = vim.api.nvim_get_current_win()
|
|
vim.cmd('botright split __runner__')
|
|
code_runner_bufnr = vim.fn.bufnr('%')
|
|
local lines = vim.o.lines * 30 / 100
|
|
vim.cmd('resize ' .. lines)
|
|
vim.cmd([[
|
|
setlocal buftype=nofile bufhidden=wipe nobuflisted nolist noswapfile nowrap cursorline nospell nonu norelativenumber winfixheight nomodifiable
|
|
set filetype=SpaceVimRunner
|
|
]])
|
|
vim.api.nvim_buf_set_keymap(code_runner_bufnr, 'n', 'q', '', {
|
|
callback = close_win,
|
|
})
|
|
vim.api.nvim_buf_set_keymap(code_runner_bufnr, 'n', 'i', '', {
|
|
callback = insert,
|
|
})
|
|
vim.api.nvim_buf_set_keymap(code_runner_bufnr, 'n', '<C-c>', '', {
|
|
callback = stop_runner,
|
|
})
|
|
local id = vim.api.nvim_create_augroup('spacevim_runner', {
|
|
clear = true,
|
|
})
|
|
vim.api.nvim_create_autocmd({ 'BufWipeout' }, {
|
|
group = id,
|
|
buffer = code_runner_bufnr,
|
|
callback = stop_runner,
|
|
})
|
|
winid = vim.api.nvim_get_current_win()
|
|
if vim.g.spacevim_code_runner_focus == 0 then
|
|
vim.api.nvim_set_current_win(previous_wind)
|
|
end
|
|
end
|
|
|
|
local function extend(t1, t2)
|
|
for k, v in pairs(t2) do
|
|
t1[k] = v
|
|
end
|
|
end
|
|
|
|
local function update_statusline()
|
|
vim.cmd('redrawstatus!')
|
|
end
|
|
|
|
local function on_stdout(id, data, event)
|
|
if id ~= runner_jobid then
|
|
return
|
|
end
|
|
if vim.api.nvim_buf_is_valid(code_runner_bufnr) then
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(code_runner_bufnr, runner_lines, runner_lines + 1, false, data)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
runner_lines = runner_lines + #data
|
|
if winid >= 0 then
|
|
vim.api.nvim_win_set_cursor(winid, { vim.api.nvim_buf_line_count(code_runner_bufnr), 1 })
|
|
end
|
|
update_statusline()
|
|
end
|
|
end
|
|
local function on_stderr(id, data, event)
|
|
if id ~= runner_jobid then
|
|
return
|
|
end
|
|
runner_status.has_errors = true
|
|
if vim.api.nvim_buf_is_valid(code_runner_bufnr) then
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(code_runner_bufnr, runner_lines, runner_lines + 1, false, data)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
runner_lines = runner_lines + #data
|
|
if winid >= 0 then
|
|
vim.api.nvim_win_set_cursor(winid, { vim.api.nvim_buf_line_count(code_runner_bufnr), 1 })
|
|
end
|
|
update_statusline()
|
|
end
|
|
end
|
|
local function on_exit(id, code, single)
|
|
if id ~= runner_jobid then
|
|
return
|
|
end
|
|
end_time = vim.fn.reltime(start_time)
|
|
runner_status.is_running = false
|
|
runner_status.exit_single = single
|
|
runner_status.exit_code = code
|
|
local done = {
|
|
'',
|
|
'[Done] exited with code=' .. code .. ', single=' .. single .. ' in ' .. str.trim(
|
|
vim.fn.reltimestr(end_time)
|
|
) .. ' seconds',
|
|
}
|
|
if vim.api.nvim_buf_is_valid(code_runner_bufnr) then
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(code_runner_bufnr, runner_lines, runner_lines + 1, false, done)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
if winid >= 0 then
|
|
vim.api.nvim_win_set_cursor(winid, { vim.api.nvim_buf_line_count(code_runner_bufnr), 1 })
|
|
end
|
|
update_statusline()
|
|
end
|
|
end
|
|
|
|
local function merge_list(...)
|
|
local t = {}
|
|
|
|
for _, tb in ipairs({ ... }) do
|
|
for _, v in ipairs(tb) do
|
|
table.insert(t, v)
|
|
end
|
|
end
|
|
return t
|
|
end
|
|
|
|
local function on_compile_exit(id, code, single)
|
|
if id ~= runner_jobid then
|
|
return
|
|
end
|
|
if code == 0 and single == 0 then
|
|
runner_jobid = job.start(target, {
|
|
on_stdout = on_stdout,
|
|
on_stderr = on_stderr,
|
|
on_exit = on_exit,
|
|
})
|
|
if runner_jobid > 0 then
|
|
runner_status = {
|
|
is_running = true,
|
|
has_errors = false,
|
|
exit_code = 0,
|
|
exit_single = 0,
|
|
}
|
|
end
|
|
else
|
|
end_time = vim.fn.reltime(start_time)
|
|
runner_status.is_running = false
|
|
runner_status.exit_code = code
|
|
runner_status.exit_single = single
|
|
local done = {
|
|
'',
|
|
'[Done] exited with code=' .. code .. ', single=' .. single .. ' in ' .. str.trim(
|
|
vim.fn.reltimestr(end_time)
|
|
) .. ' seconds',
|
|
}
|
|
if vim.api.nvim_buf_is_valid(code_runner_bufnr) then
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(code_runner_bufnr, runner_lines, runner_lines + 1, false, done)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
if winid >= 0 then
|
|
vim.api.nvim_win_set_cursor(winid, { vim.api.nvim_buf_line_count(code_runner_bufnr), 1 })
|
|
end
|
|
update_statusline()
|
|
end
|
|
end
|
|
end
|
|
|
|
local function async_run(runner, ...)
|
|
if type(runner) == 'string' then
|
|
local cmd = runner
|
|
pcall(function()
|
|
local f
|
|
if selected_file ~= '' then
|
|
f = selected_file
|
|
else
|
|
f = vim.fn.bufname('%')
|
|
end
|
|
cmd = vim.fn.printf(runner, f)
|
|
end)
|
|
logger.info(' cmd:' .. cmd)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(
|
|
code_runner_bufnr,
|
|
runner_lines,
|
|
-1,
|
|
false,
|
|
{ '[Running] ' .. cmd, '', vim.fn['repeat']('-', 20) }
|
|
)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
runner_lines = runner_lines + 3
|
|
start_time = vim.fn.reltime()
|
|
local opts = select(1, ...) or {}
|
|
extend(opts, {
|
|
on_stdout = on_stdout,
|
|
on_stderr = on_stderr,
|
|
on_exit = on_exit,
|
|
})
|
|
runner_jobid = job.start(cmd, opts)
|
|
elseif type(runner) == 'table' and #runner == 2 then
|
|
target = file.unify_path(vim.fn.tempname(), ':p')
|
|
local dir = vim.fn.fnamemodify(target, ':h')
|
|
if vim.fn.isdirectory(dir) == 0 then
|
|
vim.fn.mkdir(dir, 'p')
|
|
end
|
|
local compile_cmd
|
|
local usestdin
|
|
local compile_cmd_info
|
|
if type(runner[1]) == 'table' then
|
|
local exe
|
|
if type(runner[1].exe) == 'function' then
|
|
exe = runner[1].exe()
|
|
elseif type(runner[1].exe) == 'string' then
|
|
exe = { runner[1].exe }
|
|
end
|
|
usestdin = runner[1].usestdin or false
|
|
compile_cmd = merge_list(exe, { runner[1].targetopt or '' }, { target }, runner[1].opt)
|
|
if not usestdin then
|
|
local f
|
|
if selected_file == '' then
|
|
f = vim.fn.bufname('%')
|
|
else
|
|
f = selected_file
|
|
end
|
|
compile_cmd = merge_list(compile_cmd, { f })
|
|
end
|
|
elseif type(runner[1]) == 'string' then
|
|
end
|
|
|
|
if type(compile_cmd) == 'table' then
|
|
if usestdin then
|
|
compile_cmd_info = vim.inspect(merge_list(compile_cmd, { 'STDIN' }))
|
|
else
|
|
compile_cmd_info = vim.inspect(compile_cmd)
|
|
end
|
|
else
|
|
if usestdin then
|
|
compile_cmd_info = compile_cmd .. ' STDIN'
|
|
else
|
|
compile_cmd_info = compile_cmd
|
|
end
|
|
end
|
|
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(
|
|
code_runner_bufnr,
|
|
runner_lines,
|
|
-1,
|
|
false,
|
|
{ '[Compile] ' .. compile_cmd_info, '[Running] ' .. target, '', vim.fn['repeat']('-', 20) }
|
|
)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
|
|
runner_lines = runner_lines + 4
|
|
start_time = vim.fn.reltime()
|
|
if
|
|
type(compile_cmd) == 'string'
|
|
or type(compile_cmd) == 'table' and vim.fn.executable(compile_cmd[1] or '') == 1
|
|
then
|
|
runner_jobid = job.start(compile_cmd, {
|
|
on_stdout = on_stdout,
|
|
on_stderr = on_stderr,
|
|
on_exit = on_compile_exit,
|
|
})
|
|
if usestdin and runner_jobid > 0 then
|
|
local range = runner[1].range or { 1, '$' }
|
|
job.send(runner_jobid, vim.fn.getline(unpack(range)))
|
|
job.chanclose(runner_jobid, 'stdin')
|
|
job.stop(runner_jobid)
|
|
end
|
|
else
|
|
local exe = compile_cmd[1] or ''
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(
|
|
code_runner_bufnr,
|
|
runner_lines,
|
|
-1,
|
|
false,
|
|
{ exe .. ' is not executable, make sure ' .. exe .. ' is in your PATH' }
|
|
)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
end
|
|
elseif type(runner) == 'table' then
|
|
local cmd = {}
|
|
if type(runner.exe) == 'function' then
|
|
cmd = merge_list(cmd, runner.exe())
|
|
elseif type(runner.exe) == 'string' then
|
|
cmd = { runner.exe }
|
|
end
|
|
local usestdin = runner.usestdin or false
|
|
cmd = merge_list(cmd, runner.opt)
|
|
if not usestdin then
|
|
if selected_file == '' then
|
|
cmd = merge_list(cmd, vim.fn.bufname('%'))
|
|
else
|
|
cmd = merge_list(cmd, selected_file)
|
|
end
|
|
end
|
|
logger.info(' cmd:' .. vim.inspect(cmd))
|
|
local running_command = table.concat(cmd, ' ')
|
|
if usestdin then
|
|
running_command = running_command .. ' STDIN'
|
|
end
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(
|
|
code_runner_bufnr,
|
|
runner_lines,
|
|
-1,
|
|
false,
|
|
{ '[Running] ' .. running_command, '', vim.fn['repeat']('-', 20) }
|
|
)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
runner_lines = runner_lines + 3
|
|
start_time = vim.fn.reltime()
|
|
if vim.fn.empty(cmd) == 0 and vim.fn.executable(cmd[1]) == 1 then
|
|
runner_jobid = job.start(cmd, {
|
|
on_stdout = on_stdout,
|
|
on_stderr = on_stderr,
|
|
on_exit = on_exit,
|
|
})
|
|
if usestdin and runner_jobid > 0 then
|
|
local range = runner.range or { 1, '$' }
|
|
-- if selected file is not empty
|
|
-- read the context from selected file.
|
|
local text
|
|
if selected_file == '' then
|
|
text = vim.fn.getline(unpack(range))
|
|
else
|
|
text = vim.fn.readfile(selected_file, '')
|
|
end
|
|
job.send(runner_jobid, text)
|
|
job.chanclose(runner_jobid, 'stdin')
|
|
job.stop(runner_jobid)
|
|
end
|
|
else
|
|
local exe = cmd[1] or ''
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', true)
|
|
vim.api.nvim_buf_set_lines(
|
|
code_runner_bufnr,
|
|
runner_lines,
|
|
-1,
|
|
false,
|
|
{ exe .. ' is not executable, make sure ' .. exe .. ' is in your PATH' }
|
|
)
|
|
vim.api.nvim_buf_set_option(code_runner_bufnr, 'modifiable', false)
|
|
end
|
|
end
|
|
|
|
if runner_jobid > 0 then
|
|
runner_status = {
|
|
is_running = true,
|
|
has_errors = false,
|
|
exit_code = 0,
|
|
exit_single = 0,
|
|
}
|
|
end
|
|
end
|
|
|
|
function M.open(...)
|
|
stop_runner()
|
|
runner_jobid = 0
|
|
runner_lines = 0
|
|
runner_status = {
|
|
is_running = false,
|
|
has_errors = false,
|
|
exit_code = 0,
|
|
exit_single = 0,
|
|
}
|
|
local language = vim.o.filetype
|
|
local runner = select(1, ...) or runners[language] or ''
|
|
local opts = select(2, ...) or {}
|
|
logger.debug('runner is:\n' .. vim.inspect(runner))
|
|
logger.debug('opt is:\n' .. vim.inspect(opts))
|
|
if vim.fn.empty(runner) == 0 then
|
|
open_win()
|
|
async_run(runner, opts)
|
|
update_statusline()
|
|
else
|
|
end
|
|
end
|
|
|
|
function M.reg_runner(ft, runner)
|
|
runners[ft] = runner
|
|
end
|
|
|
|
function M.status()
|
|
local running_nr = 0
|
|
local running_done = 0
|
|
for _, v in ipairs(task_status) do
|
|
if v.is_running then
|
|
running_nr = running_nr + 1
|
|
else
|
|
running_done = running_done + 1
|
|
end
|
|
end
|
|
|
|
if runner_status.is_running then
|
|
running_nr = running_nr + 1
|
|
end
|
|
return string.format(' %s running, %s done', running_nr, running_done)
|
|
end
|
|
|
|
function M.close()
|
|
close_win()
|
|
end
|
|
|
|
function M.select_file()
|
|
runner_lines = 0
|
|
runner_status = {
|
|
is_running = false,
|
|
has_errors = false,
|
|
exit_code = 0,
|
|
exit_single = 0
|
|
}
|
|
|
|
if vim.loop.os_uname().sysname == 'Windows_NT' then
|
|
-- what the fuck, why need trim?
|
|
-- because powershell comamnd output has `\n` at the end, and filetype detection failed.
|
|
selected_file = str.trim(vim.fn.system({'powershell', "Add-Type -AssemblyName System.windows.forms|Out-Null;$f=New-Object System.Windows.Forms.OpenFileDialog;$f.Filter='Model Files All files (*.*)|*.*';$f.showHelp=$true;$f.ShowDialog()|Out-Null;$f.FileName"}))
|
|
end
|
|
|
|
if selected_file == '' then
|
|
logger.debug('file to get selected filename!')
|
|
return
|
|
else
|
|
logger.debug('selected file is:' .. selected_file)
|
|
local ft = vim.filetype.match({filename = selected_file})
|
|
if not ft then
|
|
logger.debug('failed to detect filetype of selected file:' .. selected_file)
|
|
return
|
|
end
|
|
local runner = runners[ft]
|
|
logger.info(vim.inspect(runner))
|
|
if runner then
|
|
open_win()
|
|
async_run(runner)
|
|
update_statusline()
|
|
end
|
|
end
|
|
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
function M.select_language()
|
|
|
|
end
|
|
|
|
|
|
function M.get(ft)
|
|
return runners[ft] or ''
|
|
end
|
|
|
|
return M
|