diff options
Diffstat (limited to 'test_spec.lua')
-rwxr-xr-x | test_spec.lua | 687 |
1 files changed, 687 insertions, 0 deletions
diff --git a/test_spec.lua b/test_spec.lua new file mode 100755 index 0000000..2ae8015 --- /dev/null +++ b/test_spec.lua @@ -0,0 +1,687 @@ +#!/usr/bin/env lua + +local VISUALDELAY = os.getenv("VISUALDELAY") + +local visual = VISUALDELAY or false +local visual_delay = VISUALDELAY and (tonumber(VISUALDELAY)) or 0.1 +local short_delay = 0.3 +local long_delay = 1 + +local unistd = require("posix.unistd") +local time = require("posix.time") +local curses = require("posix.curses") +local rote = require("rote") + +local rt = rote.RoteTerm(24, 80) + +--[[ +local function os_execread(cmd) + local fd = io.popen(cmd, "r") + local out = fd:read("*a") + fd:close() + return (out:gsub("\n$", "")) +end +]] +--local branch = os_execread("git branch | grep '*'"):sub(3) +--print("Running in branch "..branch) + +os.execute("make coverage") +os.execute("rm -f *.gcda */*.gcda") +os.execute("rm -f coverage.info test.htoprc") +os.execute("rm -rf lcov") +os.execute("killall htop") +os.execute("ps aux | grep '[s]leep 12345' | awk '{print $2}' | xargs kill 2> /dev/null") + +os.execute("cp ./default.htoprc ./test.htoprc") +rt:forkPty("LC_ALL=C HTOPRC=./test.htoprc ./htop 2> htop-valgrind.txt") + +local stdscr, term_win +-- Curses initialization needed even when not in visual mode +-- because luaposix only initializes KEY_* constants after initscr(). +stdscr = curses.initscr() +if visual then + curses.echo(false) + curses.start_color() + curses.raw(true) + curses.halfdelay(1) + stdscr:keypad(true) + term_win = curses.newwin(24, 80, 0, 0) + local function makePair(foreground, background) + return background * 8 + 7 - foreground + end + -- initialize the color pairs the way rt:draw() expects it + for foreground = 0, 7 do + for background = 0, 7 do + if foreground ~= 7 or background ~= 0 then + local pair = makePair(foreground, background) + curses.init_pair(pair, foreground, background) + end + end + end +else + curses.endwin() +end + +local function show(key) + rt:update() + if visual then + rt:draw(term_win, 0, 0) + if key then + term_win:mvaddstr(0, 0, tostring(key)) + end + term_win:refresh() + + delay(visual_delay) + end +end + +local function send(key, times, quick) + if times == 0 then return end + for _ = 1, times or 1 do + delay(0.003) -- 30ms delay to avoid clobbering Esc sequences + if type(key) == "string" then + for c in key:gmatch('.') do + rt:keyPress(string.byte(c)) + end + else + rt:keyPress(key) + end + if not quick then + show(key) + end + end + if quick then + show(key) + end +end + +local function string_at(x, y, len) + rt:update() + local out = {} + for i = 1, len do + out[#out+1] = rt:cellChar(y-1, x+i-2) + end + return table.concat(out) +end + +local function is_string_at(x, y, str) + return string_at(x, y, #str) == str +end + +local function check_string_at(x, y, str) + return { str, string_at(x, y, #str) } +end + +local ESC = "\27\27" + +function delay(t) + time.nanosleep({ tv_sec = math.floor(t), tv_nsec = (t - math.floor(t)) * 1000000000 }) +end + +delay(2) -- give some time for htop to initialize. +rt:update() + +local y_panelhdr = (function() + for y = 1, 24 do + if is_string_at(3, y, "PID") then + return y + end + end +end)() or 1 + +assert.not_equal(y_panelhdr, 1) + +local x_metercol2 = 41 + +show() + +os.execute("sleep 12345 &") + +local function terminated() + return not os.execute("ps aux | grep -q '\\./[h]top'") +end + +local function running_it(desc, fn) + it(desc, function() + assert(not terminated()) + show() + fn() + assert(not terminated()) + end) +end + +local function check(t) + return t[1], t[2] +end + +local attrs = { + black_on_cyan = 6, + red_on_cyan = 22, + white_on_black = 176, + yellow_on_black = 112, +} + +local function find_selected_y(from) + rt:update() + for y = from or (y_panelhdr + 1), rt:rows() - 1 do + local attr = rt:cellAttr(y-1, 1) + if attr == attrs.black_on_cyan then + return y + end + end + return y_panelhdr + 1 +end + +local function find_command_x() + for x = 1, 80 do + if is_string_at(x, y_panelhdr, "Command") then + return x + end + end + return 64 +end + +local function set_display_option(n) + send("S") + send(curses.KEY_DOWN) + send(curses.KEY_RIGHT) + send(curses.KEY_DOWN, n, "quick") + send("\n") + send(curses.KEY_F10) +end + +describe("htop test suite", function() + + running_it("performs incremental filter", function() + send("\\") + send("x\127bux\127sted") -- test backspace + send("\n") + delay(short_delay) + rt:update() + local pid = (" "..tostring(unistd.getpid())):sub(-5) + local ourpid = check_string_at(1, y_panelhdr + 1, pid) + send("\\") + send(ESC) + send(curses.KEY_F5) + send(curses.KEY_HOME) + delay(short_delay) + rt:update() + local initpid = check_string_at(1, y_panelhdr + 1, " 1") + delay(short_delay) + rt:update() + send(curses.KEY_F5) + assert.equal(check(ourpid)) + assert.equal(check(initpid)) + end) + + running_it("performs incremental search", function() + send(curses.KEY_HOME) + send("/") + send("busted") + local attr = rt:cellAttr(rt:rows() - 1, 30) + delay(short_delay) + local line = find_selected_y() + local pid = (" "..tostring(unistd.getpid())):sub(-5) + assert.equal(attr, attrs.black_on_cyan) + local ourpid = check_string_at(1, line, pid) + send("\n") + send(curses.KEY_HOME) + assert.equal(check(ourpid)) + end) + + running_it("performs pid search", function() + send(curses.KEY_F5) + send(curses.KEY_END) + send("1") + delay(short_delay) + local line = find_selected_y() + local initpid = check_string_at(1, line, " 1") + send(curses.KEY_F5) + assert.equal(check(initpid)) + end) + + + running_it("horizontal scroll", function() + local h_scroll = 20 + send(curses.KEY_F5) + delay(short_delay) + local str1 = string_at(1+h_scroll, y_panelhdr+1, 5) + send(curses.KEY_RIGHT) + delay(short_delay) + local str2 = string_at(1, y_panelhdr+1, 5) + send(curses.KEY_LEFT) + delay(short_delay) + local str3 = string_at(1+h_scroll, y_panelhdr+1, 5) + send(curses.KEY_LEFT) + delay(short_delay) + local str4 = string_at(1+h_scroll, y_panelhdr+1, 5) + send(curses.KEY_F5) + assert.equal(str1, str2) + assert.equal(str2, str3) + assert.equal(str3, str4) + end) + + running_it("kills a process", function() + send(curses.KEY_HOME) + send("\\") + send("sleep 12345") + local attr = rt:cellAttr(rt:rows() - 1, 30) + assert.equal(attr, attrs.black_on_cyan) + send("\n") + delay(short_delay) + rt:update() + local col = find_command_x() + local procname = check_string_at(col, y_panelhdr + 1, "sleep 12345") + send("k") + send("\n") + send("\\") + send(ESC) + delay(short_delay) + assert.equal(check(procname)) + assert.not_equal((os.execute("ps aux | grep -q '[s]leep 12345'")), true) + end) + + running_it("runs strace", function() + send(curses.KEY_HOME) + send("/") + send("busted") + send("\n") + send("s") + delay(long_delay) + send(ESC) + end) + + running_it("runs lsof", function() + send(curses.KEY_HOME) + send("/") + send("busted") + send("\n") + send("l") + delay(long_delay) + send(ESC) + end) + + running_it("performs filtering in lsof", function() + send(curses.KEY_HOME) + send("/") + send("htop") + send("\n") + send("l") + send(curses.KEY_F4) + send("pipe") + delay(long_delay) + local pipefd = check_string_at(1, 3, " 3") + send(ESC) + assert.equal(check(pipefd)) + end) + + running_it("performs search in lsof", function() + send(curses.KEY_HOME) + send("/") + send("htop") + send("\n") + send("l") + send(curses.KEY_F3) + send("pipe") + delay(long_delay) + local line = find_selected_y(3) + local pipefd = check_string_at(1, line, " 3") + send(ESC) + assert.equal(check(pipefd)) + end) + + + running_it("cycles through meter modes in the default meters", function() + send("S") + for _ = 1, 2 do + send(curses.KEY_RIGHT) + for _ = 1, 3 do + send("\n", 4) + send(curses.KEY_DOWN) + end + end + send(ESC) + end) + + running_it("show process of a user", function() + send(curses.KEY_F5) + send("u") + send(curses.KEY_DOWN) + delay(short_delay) + rt:update() + local chosen = string_at(1, y_panelhdr + 2, 9) + send("\n") + send(curses.KEY_HOME) + delay(short_delay) + rt:update() + local shown = string_at(7, y_panelhdr + 1, 9) + send("u") + send("\n") + send(curses.KEY_HOME) + delay(short_delay) + rt:update() + local inituser = string_at(7, y_panelhdr + 1, 9) + send(curses.KEY_F5) + assert.equal(shown, chosen) + assert.equal(inituser, "root ") + end) + + running_it("performs failing search", function() + send(curses.KEY_HOME) + send("/") + send("xxxxxxxxxx") + delay(short_delay) + rt:update() + local attr = rt:cellAttr(rt:rows() - 1, 30) + assert.equal(attr, attrs.red_on_cyan) + send("\n") + end) + + running_it("cycles through search", function() + send(curses.KEY_HOME) + send("/") + send("sh") + local lastpid + local pidpairs = {} + for _ = 1, 3 do + send(curses.KEY_F3) + local line = find_selected_y() + local pid = string_at(1, line, 5) + if lastpid then + pidpairs[#pidpairs + 1] = { lastpid, pid } + lastpid = pid + end + end + send(curses.KEY_HOME) + for _, pair in pairs(pidpairs) do + assert.not_equal(pair[1], pair[2]) + end + end) + + running_it("visits each setup screen", function() + send("S") + send(curses.KEY_DOWN, 3) + send(curses.KEY_F10) + end) + + running_it("adds and removes PPID column", function() + send("S") + send(curses.KEY_DOWN, 3) + send(curses.KEY_RIGHT, 2) + send(curses.KEY_DOWN, 2) + send("\n") + send(curses.KEY_F10) + delay(short_delay) + local ppid = check_string_at(2, y_panelhdr, "PPID") + send("S") + send(curses.KEY_DOWN, 3) + send(curses.KEY_RIGHT, 1) + send(curses.KEY_DC) + send(curses.KEY_F10) + delay(short_delay) + local not_ppid = check_string_at(2, y_panelhdr, "PPID") + assert.equal(check(ppid)) + assert.not_equal(check(not_ppid)) + end) + + running_it("changes CPU affinity for a process", function() + send("a") + send(" \n") + send(ESC) + end) + + running_it("renices for a process", function() + send("/") + send("busted") + send("\n") + local line = find_selected_y() + local before = check_string_at(22, line, " 0") + send(curses.KEY_F8) + delay(short_delay) + local after = check_string_at(22, line, " 1") + assert.equal(check(before)) + assert.equal(check(after)) + end) + + running_it("tries to lower nice for a process", function() + send("/") + send("busted") + send("\n") + local line = find_selected_y() + local before = string_at(22, line, 2) + send(curses.KEY_F7) + delay(short_delay) + local after = string_at(22, line, 2) + assert.equal(before, after) -- no permissions + end) + + running_it("invert sort order", function() + local cpu_col = 45 + send("P") + send("I") + send(curses.KEY_HOME) + delay(short_delay) + local zerocpu = check_string_at(cpu_col, y_panelhdr + 1, " 0.0") + send("I") + delay(short_delay) + local nonzerocpu = check_string_at(cpu_col, y_panelhdr + 1, " 0.0") + assert.equal(check(zerocpu)) + assert.not_equal(check(nonzerocpu)) + end) + + running_it("changes IO priority for a process", function() + send("/") + send("htop") + send("\n") + send("i") + send(curses.KEY_END) + send("\n") + send(ESC) + end) + + running_it("shows help", function() + send(curses.KEY_F1) + send("\n") + set_display_option(9) + send(curses.KEY_F1) + send("\n") + set_display_option(9) + end) + + running_it("moves meters around", function() + send("S") + send(curses.KEY_RIGHT) + send(curses.KEY_UP) + send("\n") + send(curses.KEY_DOWN) + send(curses.KEY_UP) + send(curses.KEY_RIGHT) + send(curses.KEY_RIGHT) + send(curses.KEY_LEFT) + send(curses.KEY_LEFT) + send("\n") + send(curses.KEY_F10) + end) + + local meters = { + { name = "clock", down = 0, string = "Time" }, + { name = "load", down = 2, string = "Load" }, + { name = "battery", down = 7, string = "Battery" }, + { name = "hostname", down = 8, string = "Hostname" }, + { name = "memory", down = 3, string = "Mem" }, + { name = "CPU average", down = 16, string = "Avg" }, + } + + running_it("checks various CPU meters", function() + send("S") + send(curses.KEY_RIGHT, 3) + send(curses.KEY_DOWN, 9, "quick") + for _ = 9, 14 do + send("\n") + send("\n") + send(curses.KEY_DC) + send(curses.KEY_RIGHT) + send(curses.KEY_DOWN) + end + end) + + for _, item in ipairs(meters) do + running_it("adds and removes a "..item.name.." widget", function() + send("S") + send(curses.KEY_RIGHT, 3) + send(curses.KEY_DOWN, item.down) + send("\n") + send(curses.KEY_UP, 4) + send("\n") + send(curses.KEY_F4, 4) -- cycle through meter modes + delay(short_delay) + rt:update() + local with = check_string_at(x_metercol2, 2, item.string) + send(curses.KEY_DC) + delay(short_delay) + local without = check_string_at(x_metercol2, 2, item.string) + send(curses.KEY_F10) + assert.equal(check(with)) + assert.not_equal(check(without)) + end) + end + + running_it("goes through themes", function() + send(curses.KEY_F2) + send(curses.KEY_DOWN, 2) + send(curses.KEY_RIGHT) + for _ = 1, 6 do + send("\n") + send(curses.KEY_DOWN) + end + send(curses.KEY_UP, 6) + send("\n") + send(curses.KEY_F10) + end) + + local display_options = { + { name = "tree view", down = 0 }, + { name = "shadow other user's process", down = 1 }, + { name = "hide kernel threads", down = 2 }, + { name = "hide userland threads", down = 3 }, + { name = "display threads in different color", down = 4 }, + { name = "show custom thread names", down = 5 }, + { name = "highlight basename", down = 6 }, + { name = "highlight large numbers", down = 7 }, + { name = "leave margin around header", down = 8 }, + { name = "use detailed CPU time", down = 9 }, + { name = "count from zero", down = 10 }, + { name = "update process names", down = 11 }, + { name = "guest time in CPU%", down = 12 }, + } + + for _, item in ipairs(display_options) do + running_it("checks display option to "..item.name, function() + for _ = 1, 2 do + set_display_option(item.down) + delay(short_delay) + end + end) + end + + running_it("shows detailed CPU with guest time", function() + for _ = 1, 2 do + send("S") + send(curses.KEY_DOWN) + send(curses.KEY_RIGHT) + send(curses.KEY_DOWN, 9) + send("\n") + send(curses.KEY_DOWN, 3) + send("\n") + send(curses.KEY_LEFT) + send(curses.KEY_UP) + send(curses.KEY_RIGHT) + send(curses.KEY_F4, 4) -- cycle through CPU meter modes + send(curses.KEY_F10) + delay(short_delay) + end + end) + + running_it("expands and collapses tree", function() + send(curses.KEY_F5) -- tree view + send(curses.KEY_HOME) + send(curses.KEY_DOWN) -- second process in the tree + send("-") + send("+") + send(curses.KEY_F5) + end) + + running_it("sets sort key", function() + send(".") + send("\n") + end) + + running_it("tags all children", function() + send(curses.KEY_F5) -- tree view + send(curses.KEY_HOME) -- ensure we're at init + send("c") + local taggedattrs = {} + rt:update() + for y = y_panelhdr + 2, 23 do + table.insert(taggedattrs, rt:cellAttr(y-1, 4)) + end + delay(short_delay) + send("U") + local untaggedattrs = {} + rt:update() + for y = y_panelhdr + 2, 23 do + table.insert(untaggedattrs, rt:cellAttr(y-1, 4)) + end + send(curses.KEY_F5) + + for _, taggedattr in ipairs(taggedattrs) do + assert.equal(attrs.yellow_on_black, taggedattr) + end + for _, untaggedattr in ipairs(untaggedattrs) do + assert.equal(attrs.white_on_black, untaggedattr) + end + end) + + for i = 1, 62 do + running_it("show column "..i, function() + send("S") + send(curses.KEY_END) + send(curses.KEY_RIGHT, 1) + if i > 1 then + send(curses.KEY_DC) + end + send(curses.KEY_RIGHT, 1) + local down = i + while down > 13 do + send(curses.KEY_NPAGE) + down = down - 13 + end + send(curses.KEY_DOWN, down, "quick") + send("\n") + send(curses.KEY_F10) + if i == 62 then + send("S") + send(curses.KEY_END) + send(curses.KEY_RIGHT, 1) + if i > 1 then + send(curses.KEY_DC) + end + send(curses.KEY_F10) + end + end) + end + + it("finally quits", function() + assert(not terminated()) + send("q") + while not terminated() do + unistd.sleep(1) + send("q") + end + assert(terminated()) + if visual then + curses.endwin() + end + os.execute("make lcov && xdg-open lcov/index.html") + end) +end) |