-
Notifications
You must be signed in to change notification settings - Fork 20
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
35d0a3c
commit 5ba2854
Showing
6 changed files
with
257 additions
and
9 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,134 @@ | ||
local async = require("neotest.async") | ||
|
||
local convert = require("neotest-golang.convert") | ||
local json = require("neotest-golang.json") | ||
|
||
local M = {} | ||
|
||
---@param spec neotest.RunSpec | ||
---@param result neotest.StrategyResult | ||
---@param tree neotest.Tree | ||
function M.results(spec, result, tree) | ||
---@type table | ||
local raw_output = async.fn.readfile(result.output) | ||
|
||
---@type List<table> | ||
local jsonlines = json.process_json(raw_output) | ||
|
||
---@type List | ||
local full_test_output = {} | ||
|
||
--- neotest results | ||
---@type table<string, neotest.Result> | ||
local neotest_results = {} | ||
|
||
--- internal results struct | ||
---@type table<string, table> | ||
local internal_results = {} | ||
|
||
-- record test names | ||
for _, line in ipairs(jsonlines) do | ||
if line.Action == "run" and line.Test ~= nil then | ||
internal_results[line.Test] = { | ||
status = "skipped", | ||
output = {}, | ||
errors = {}, | ||
node_data = nil, | ||
} | ||
end | ||
end | ||
|
||
-- record test status | ||
for _, line in ipairs(jsonlines) do | ||
if line.Action == "pass" and line.Test ~= nil then | ||
internal_results[line.Test].status = "passed" | ||
elseif line.Action == "fail" and line.Test ~= nil then | ||
internal_results[line.Test].status = "failed" | ||
end | ||
end | ||
|
||
-- record error output | ||
for _, line in ipairs(jsonlines) do | ||
if line.Action == "output" and line.Output ~= nil and line.Test ~= nil then | ||
-- append line.Output to output field | ||
internal_results[line.Test].output = | ||
vim.list_extend(internal_results[line.Test].output, { line.Output }) | ||
-- search for error message and line number | ||
local matched_line_number = string.match(line.Output, ":(%d+):") | ||
if matched_line_number ~= nil then | ||
local line_number = tonumber(matched_line_number) | ||
local message = string.match(line.Output, ":%d+: (.*)") | ||
if line_number ~= nil and message ~= nil then | ||
table.insert(internal_results[line.Test].errors, { | ||
line = line_number - 1, -- neovim lines are 0-indexed | ||
message = message, | ||
}) | ||
end | ||
end | ||
end | ||
end | ||
|
||
-- associate internal results with neotest node data | ||
for test_name, test_properties in pairs(internal_results) do | ||
local test_name_pattern = convert.to_neotest_test_name_pattern(test_name) | ||
for _, node in tree:iter_nodes() do | ||
local node_data = node:data() | ||
|
||
-- WARNING: workarounds | ||
local tweaked_node_data_id = node_data.id:gsub('"', "") -- workaround, since we cannot know where double quotes might appear | ||
local tweaked_node_data_id = tweaked_node_data_id:gsub("_", " ") -- NOTE: look into making this more clear... | ||
|
||
if | ||
string.find(node_data.path, spec.context.id, 1, true) | ||
and string.find(tweaked_node_data_id, test_name_pattern, 1, false) | ||
then | ||
internal_results[test_name].node_data = node_data | ||
end | ||
end | ||
end | ||
|
||
-- populate neotest results | ||
for test_name, test_properties in pairs(internal_results) do | ||
if test_properties.node_data ~= nil then | ||
local test_output_path = vim.fs.normalize(async.fn.tempname()) | ||
async.fn.writefile(test_properties.output, test_output_path) | ||
neotest_results[test_properties.node_data.id] = { | ||
status = test_properties.status, | ||
output = test_output_path, -- NOTE: could be slow when running many tests? | ||
errors = test_properties.errors, | ||
} | ||
end | ||
end | ||
|
||
---@type neotest.ResultStatus | ||
local test_command_status = "skipped" | ||
if result.code == 0 then | ||
test_command_status = "passed" | ||
else | ||
test_command_status = "failed" | ||
end | ||
|
||
-- write full test command output | ||
local parsed_output_path = vim.fs.normalize(async.fn.tempname()) | ||
for _, line in ipairs(jsonlines) do | ||
if line.Action == "output" then | ||
table.insert(full_test_output, line.Output) | ||
end | ||
end | ||
async.fn.writefile(full_test_output, parsed_output_path) | ||
|
||
-- register properties on the directory node that was run | ||
neotest_results[spec.context.id] = { | ||
status = test_command_status, | ||
output = parsed_output_path, | ||
} | ||
|
||
-- FIXME: once output is parsed, erase file contents, so to avoid JSON in | ||
-- output panel. This is a workaround for now, only because of | ||
-- https://github.com/nvim-neotest/neotest/issues/391 | ||
vim.fn.writefile({ "" }, result.output) | ||
|
||
return neotest_results | ||
end | ||
|
||
return M |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,88 @@ | ||
local _ = require("neotest") -- fix LSP errors | ||
|
||
local options = require("neotest-golang.options") | ||
|
||
local M = {} | ||
|
||
--- Build runspec for a directory. | ||
---@param pos neotest.Position | ||
---@return neotest.RunSpec | ||
function M.build(pos) | ||
-- Strategy: | ||
-- 1. Find the go.mod file from pos.path. | ||
-- 2. Run `go test` from the directory containing the go.mod file. | ||
-- 3. Use the relative path from the go.mod file to pos.path as the test pattern. | ||
|
||
local go_mod_filepath = M.find_file_upwards("go.mod", pos.path) | ||
local go_mod_folderpath = vim.fn.fnamemodify(go_mod_filepath, ":h") | ||
local cwd = go_mod_folderpath | ||
|
||
-- calculate the relative path to pos.path from cwd | ||
local relative_path = M.remove_base_path(cwd, pos.path) | ||
local test_pattern = "./" .. relative_path .. "/..." | ||
|
||
return M.build_dir_test_runspec(pos, cwd, test_pattern) | ||
end | ||
|
||
function M.find_file_upwards(filename, start_path) | ||
local scan = require("plenary.scandir") | ||
local cwd = vim.fn.getcwd() -- get the current working directory | ||
local found_filepath = nil | ||
while start_path ~= cwd do | ||
local files = scan.scan_dir( | ||
start_path, | ||
{ search_pattern = filename, hidden = true, depth = 1 } | ||
) | ||
if #files > 0 then | ||
found_filepath = files[1] | ||
break | ||
end | ||
start_path = vim.fn.fnamemodify(start_path, ":h") -- go up one directory | ||
end | ||
return found_filepath | ||
end | ||
|
||
function M.remove_base_path(base_path, target_path) | ||
if string.find(target_path, base_path, 1, true) == 1 then | ||
return string.sub(target_path, string.len(base_path) + 2) | ||
end | ||
|
||
return target_path | ||
end | ||
|
||
--- Build runspec for a directory of tests | ||
---@param pos neotest.Position | ||
---@param cwd string | ||
---@param test_pattern string | ||
---@return neotest.RunSpec | ||
function M.build_dir_test_runspec(pos, cwd, test_pattern) | ||
local gotest = { | ||
"go", | ||
"test", | ||
"-json", | ||
} | ||
|
||
---@type table | ||
local go_test_args = { | ||
test_pattern, | ||
} | ||
|
||
local combined_args = | ||
vim.list_extend(vim.deepcopy(options._go_test_args), go_test_args) | ||
local gotest_command = vim.list_extend(vim.deepcopy(gotest), combined_args) | ||
|
||
---@type neotest.RunSpec | ||
local run_spec = { | ||
command = gotest_command, | ||
cwd = cwd, | ||
context = { | ||
id = pos.id, | ||
test_filepath = pos.path, | ||
test_type = "dir", | ||
}, | ||
} | ||
|
||
return run_spec | ||
end | ||
|
||
return M |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters