Skip to content

Commit 43b0d15

Browse files
authored
Print evaluation output to buffer (#79)
* Simplify work around cleanup It's simper if the nrepl keeps track of the connection internal id, which is something it might want to do anyways. * Fix duplicated call to accessor fn * Rename local table and local vars For better expressing their (new) intent. * Allow supplying connection id or ip-port tuple * Allow for reverse lookup on connections * Allow reverse lookup for sessions * Print input and output text to output window Only if enabled, will pass through if buffer doesn't exist.
1 parent f260aec commit 43b0d15

File tree

8 files changed

+180
-46
lines changed

8 files changed

+180
-46
lines changed

lua/acid/connections.lua

+16-24
Original file line numberDiff line numberDiff line change
@@ -24,33 +24,15 @@ connections.add = function(addr)
2424
return ulid
2525
end
2626

27-
connections.remove = function(pwd, addr)
28-
-- If removed address is current to a pwd
29-
local key = connections.current[pwd]
30-
local conn = connections.store[key]
31-
32-
-- Double-check if address is correct
33-
if key ~= nil and conn ~= nil and conn[2] == addr[2] and conn[1] == addr[1] then
34-
-- Then remove it from current pwd
35-
connections.current[pwd] = nil
36-
-- Remove it's definition
37-
connections.store[key] = nil
38-
39-
-- And remove all other addresses that point to it
40-
for ix, v in pairs(connections.current) do
41-
if v == key then
42-
connections.current[ix] = nil
43-
end
44-
end
45-
else
46-
-- Else, remove it from the connections if no address points to it
47-
for i, v in pairs(connections.store) do
48-
if v[2] == addr[2] and v[1] == addr[1] then
49-
connections.store[i] = nil
50-
end
27+
connections.remove = function(key)
28+
-- Remove all addresses that point to id
29+
for ix, v in pairs(connections.current) do
30+
if v == key then
31+
connections.current[ix] = nil
5132
end
5233
end
5334

35+
connections.store[key] = nil
5436
end
5537

5638
--- Elects selected connection as primary (thus default) for a certain address
@@ -103,6 +85,16 @@ connections.search = function(pwd)
10385
return nil
10486
end
10587

88+
connections.reverse_lookup = function(conn)
89+
for k, v in pairs(connections.store) do
90+
if v[2] == conn[2] and v[1] == conn[1] then
91+
return k
92+
end
93+
end
94+
95+
return nil
96+
end
97+
10698
connections.attempt_get = function(pwd)
10799
local conn = connections.get(pwd)
108100
if conn == nil then

lua/acid/core.lua

+19-3
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@
33
--- low-level connection handler.
44
-- @module acid.core
55
-- @todo merge with acid and acid.connections
6+
local output = require("acid.output")
67
local connections = require("acid.connections")
78
local utils = require("acid.utils")
89
local log = require("acid.log")
@@ -12,24 +13,39 @@ local core = {
1213
}
1314

1415
--- Forward messages to the nrepl and registers the handler.
15-
-- @tparam[opt] {string,string} conn Ip and Port tuple. Will try to get one if nil.
16+
-- @tparam[opt] {string,string} connection Ip and Port tuple. Will try to get one if nil.
1617
-- @tparam table obj Payload to be sent to the nrepl.
1718
-- @tparam function handler Handler function to deal with the response.
18-
core.send = function(conn, obj, handler)
19+
core.send = function(connection, obj, handler)
20+
local conn, conn_id
1921
if handler == nil then
2022
vim.api.nvim_err_writeln("Please provide a handler for that operation.")
2123
return
2224
end
2325

2426
local pwd = vim.api.nvim_call_function("getcwd", {})
2527

26-
conn = conn or connections.attempt_get(pwd)
28+
local ctp = type(connection)
29+
if ctp == "string" then
30+
conn_id = connection
31+
conn = connections.store[connection]
32+
elseif ctp == "table" then
33+
conn = connection
34+
conn_id = connections.reverse_lookup(connection)
35+
elseif connection == nil then
36+
conn_id = connections.peek()
37+
conn = connections.store[conn_id]
38+
end
2739

2840
if conn == nil then
2941
log.msg("No active connection to a nrepl session. Aborting")
3042
return
3143
end
3244

45+
if obj.code ~= nil then
46+
output.draw(conn_id, {(obj.ns or "") .. "=> " .. obj.code})
47+
end
48+
3349
if obj.id == nil then
3450
obj.id = utils.ulid()
3551
end

lua/acid/features.lua

+1
Original file line numberDiff line numberDiff line change
@@ -98,6 +98,7 @@ features.eval_expr = function(mode, replace, ns)
9898
end}))
9999
else
100100
midlws = middlewares
101+
.output{}
101102
.print{}
102103
.clipboard{}
103104
.virtualtext(coord)

lua/acid/middlewares/output.lua

+43
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
-- luacheck: globals vim
2+
local do_print = {}
3+
local output = require("acid.output")
4+
local sessions = require("acid.sessions")
5+
local utils = require("acid.utils")
6+
local log = require("acid.log")
7+
8+
output.middleware = function(config)
9+
return function(middleware)
10+
return function(data)
11+
local session = data.session
12+
local conn_id = sessions.reverse_lookup(session)
13+
14+
if data.out ~= nil then
15+
local out = {}
16+
data.out:gsub("[^\n]+", function(dt) table.insert(out, dt) end)
17+
output.draw(conn_id, out)
18+
end
19+
if data.value ~= nil then
20+
output.draw(conn_id, {"=> " .. data.value})
21+
end
22+
if data.ex ~= nil then
23+
output.draw(conn_id, {"!! " .. data.ex})
24+
end
25+
if data.err ~= nil then
26+
local out = {}
27+
data.err:gsub("[^\n]+", function(dt) table.insert(out, "!! " .. dt) end)
28+
output.draw(conn_id, out)
29+
end
30+
31+
if config.accessor ~= nil then
32+
local msg = config.accessor(data)
33+
if msg ~= nil and msg ~= "" then
34+
output.draw(conn_id, {"<> " .. msg})
35+
end
36+
end
37+
38+
return middleware(data)
39+
end
40+
end
41+
end
42+
43+
return output

lua/acid/middlewares/print.lua

+1-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@ do_print.middleware = function(config)
2222
if config.accessor ~= nil then
2323
local msg = config.accessor(data)
2424
if msg ~= nil and msg ~= "" then
25-
log.msg(config.accessor(data))
25+
log.msg(msg)
2626
end
2727
end
2828

lua/acid/nrepl.lua

+22-18
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,9 @@ local nvim = vim.api
66
local log = require("acid.log")
77
local utils = require("acid.utils")
88
local connections = require("acid.connections")
9+
local output = require("acid.output")
910

10-
local pending = {}
11+
local job_mapping = {}
1112
local nrepl = {}
1213

1314
local deps = {
@@ -104,6 +105,7 @@ nrepl.default_middlewares = {'nrepl/nrepl', 'cider/cider-nrepl', 'refactor-nrepl
104105
-- @tparam[opt] string obj.connect -c parameter for the nrepl process
105106
-- @tparam[opt] string obj.bind -b parameter for the nrepl process
106107
-- @tparam[opt] boolean obj.skip_autocmd don't fire an autocmd after starting this repl
108+
-- @tparam[opt] boolean obj.disable_output_capture disables output capturing.
107109
-- @treturn boolean Whether it was possible to spawn a nrepl process
108110
nrepl.start = function(obj)
109111
local pwd = obj.pwd or vim.api.nvim_call_function("getcwd", {})
@@ -148,25 +150,23 @@ nrepl.start = function(obj)
148150
addr = conn
149151
}
150152

151-
local ix = connections.add(conn)
153+
local conn_id = connections.add(conn)
152154

153-
pending[ret] = {pwd = pwd, ix = ix}
155+
nrepl.cache[pwd].id = conn_id
156+
157+
job_mapping[ret] = {pwd = pwd, conn = conn_id, init = false}
158+
if obj.disable_output_capture == false then
159+
output.buffer(conn_id)
160+
end
154161
return true
155162
end
156163

157164
--- Stops a nrepl process managed by acid
158165
-- @tparam table obj Configuration for the nrepl process to be stopped
159166
-- @tparam string obj.pwd Path where the nrepl process was started
160167
nrepl.stop = function(obj)
161-
local pwd = obj.pwd
162-
163-
if not utils.ends_with(pwd, "/") then
164-
pwd = pwd .. "/"
165-
end
166-
167-
nvim.nvim_call_function("jobstop", {nrepl.cache[pwd].job})
168-
connections.unselect(pwd)
169-
connections.remove(pwd, nrepl.cache[pwd].addr)
168+
nvim.nvim_call_function("jobstop", {nrepl.cache[obj.pwd].job})
169+
connections.remove(nrepl.cache[obj.pwd].id)
170170
nrepl.cache[obj.pwd] = nil
171171
end
172172

@@ -185,21 +185,25 @@ nrepl.handle = {
185185
_store = {},
186186
stdout = function(dt, ch)
187187
nrepl.handle._store[ch] = nrepl.handle._store[ch] or {}
188-
if pending[ch] ~= nil then
188+
local job = job_mapping[ch]
189+
190+
if not job.init then
189191
for _, ln in ipairs(dt) do
190192
if string.sub(ln, 1, 20) == "nREPL server started" then
191-
local opts = pending[ch]
192193
local port = ln:match("%d+")
193-
connections.store[opts.ix][2] = port
194-
connections.select(opts.pwd, opts.ix)
195-
pending[ch] = nil
196-
if not nrepl.cache[opts.pwd].skip_autocmd then
194+
connections.store[job.conn][2] = port
195+
connections.select(job.pwd, job.conn)
196+
job_mapping[ch].init = true
197+
if not nrepl.cache[job.pwd].skip_autocmd then
197198
log.msg("Connected on port", tostring(port))
198199
vim.api.nvim_command("doautocmd User AcidConnected")
199200
end
201+
break
200202
end
201203
end
202204
end
205+
206+
output.draw(job.conn, dt)
203207
table.insert(nrepl.handle._store[ch], dt)
204208
end,
205209
stderr = function(dt, ch)

lua/acid/output.lua

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
-- luacheck: globals vim
2+
3+
local output = {}
4+
5+
local location = function(opts)
6+
local width = vim.api.nvim_get_option("columns")
7+
local height = vim.api.nvim_get_option("lines")
8+
return {
9+
relative = "editor",
10+
width = opts.width or 60,
11+
height = height,
12+
row = 0,
13+
focusable = false,
14+
col = width - (opts.width or 60)
15+
}
16+
end
17+
18+
output.conn_to_buf = {}
19+
20+
output.buffer = function(conn)
21+
local buf = output.conn_to_buf[conn]
22+
if buf == nil then
23+
buf = vim.api.nvim_create_buf(false, true)
24+
vim.api.nvim_buf_set_option(buf, "bufhidden", "wipe")
25+
output.conn_to_buf[conn] = buf
26+
end
27+
28+
return buf
29+
end
30+
31+
output.window = function(conn, opts)
32+
local buf = output.buffer(conn)
33+
local winnr = vim.fn.bufwinnr(buf)
34+
local winid
35+
36+
if winnr == -1 then
37+
winid = vim.api.nvim_open_win(buf, true, location(opts or {}))
38+
vim.api.nvim_win_set_option(winid, "breakindent", true)
39+
vim.api.nvim_win_set_option(winid, "number", false)
40+
vim.api.nvim_win_set_option(winid, "relativenumber", false)
41+
vim.api.nvim_win_set_option(winid, "fillchars", "eob: ")
42+
else
43+
winid = vim.fn.win_getid()
44+
end
45+
vim.api.nvim_set_current_win(winid)
46+
end
47+
48+
output.close_window = function(conn, opts)
49+
local buf = output.buffer(conn)
50+
local winnr = vim.fn.bufwinnr(buf)
51+
52+
if winnr ~= -1 then
53+
local winid = vim.fn.win_getid()
54+
vim.api.nvim_win_close(winid, true)
55+
end
56+
end
57+
58+
59+
output.draw = function(conn, lines)
60+
local buf = output.conn_to_buf[conn]
61+
if buf == nil then
62+
return
63+
end
64+
65+
vim.api.nvim_buf_set_lines(buf, -1, -1, false, lines)
66+
end
67+
68+
return output

lua/acid/sessions.lua

+10
Original file line numberDiff line numberDiff line change
@@ -47,5 +47,15 @@ sessions.new_session = function(connection_ix)
4747
core.send(conn, clone.payload(), handler)
4848
end
4949

50+
sessions.reverse_lookup = function(session_id)
51+
for k, v in pairs(sessions.store) do
52+
for _, v2 in ipairs(v.list) do
53+
if v2 == session_id then
54+
return k
55+
end
56+
end
57+
end
58+
end
59+
5060

5161
return sessions

0 commit comments

Comments
 (0)