build core, basic and go variants and add template
This commit is contained in:
57
README.md
57
README.md
@@ -1,3 +1,56 @@
|
|||||||
# code
|
# Containerized `code` IDE
|
||||||
|
`code` is a containerized IDE, that runs in the terminal and is based on
|
||||||
|
[tmux][tmux] and [neovim][neovim]. That means you can run it on a machine
|
||||||
|
without X server or even within an SSH terminal session.
|
||||||
|
|
||||||
Containerized IDE, that runs in the terminal.
|
The [podman][podman] containerization makes the development environment not
|
||||||
|
only more secure but also reproducible. If something in the editor were to
|
||||||
|
break, restarting it would create a fresh container. Since only the current
|
||||||
|
working directory is mounted into the container, the editor plugins and code
|
||||||
|
dependencies can't directly access any other files on your system. In
|
||||||
|
combination with code reviews on your code hosting platform of trust, this
|
||||||
|
might even be a reasonable trade-off between security and productivity.
|
||||||
|
|
||||||
|
[tmux]: https://tmux.github.io/
|
||||||
|
[neovim]: https://neovim.io/
|
||||||
|
[podman]: https://podman.io/
|
||||||
|
|
||||||
|
## Variants
|
||||||
|
The beauty of `code` is, that you can easily specify a separate development
|
||||||
|
environment for each of your coding workflows by writing a Containerfile and
|
||||||
|
building it. When starting `code`, you simply specify the variant you'd like to
|
||||||
|
use on the command line.
|
||||||
|
|
||||||
|
The default and most basic variant is *core*. It contains the shell
|
||||||
|
configuration and a basic neovim setup, that doesn't use any third party
|
||||||
|
plugins and therefore is the most secure choice. The *basic* variant contains
|
||||||
|
[git][git] and a bunch of plugins and configurations for integration with tmux,
|
||||||
|
file tree navigation, completion, quick editing and visuals. It doesn't contain
|
||||||
|
any language runtimes, libraries or LSP servers though. Any language or
|
||||||
|
workflow-specific variants can be built on top of *basic*.
|
||||||
|
|
||||||
|
[git]: https://git-scm.com/
|
||||||
|
|
||||||
|
## Using `code`
|
||||||
|
After cloning the repository and making sure, that podman is installed on your
|
||||||
|
machine, you can build all images by executing `./build.sh` or only selected
|
||||||
|
variants by passing them as command line arguments. The *core* and *basic*
|
||||||
|
variants need to be built before any variants depending on them.
|
||||||
|
|
||||||
|
Now you can link the `code` script to some directory in your `$PATH` (e. g.
|
||||||
|
`ln -s "$(pwd)/code" ~/.local/bin/code`) and edit your first file by executing
|
||||||
|
`code my-file.txt`. Some flags at the beginning of the command are interpreted
|
||||||
|
by `code`, everything else gets passed to neovim. To learn more about `code`s
|
||||||
|
command line flags, execute `code --help`.
|
||||||
|
|
||||||
|
## Building new variants
|
||||||
|
Simply copy the template directory, add you [alpine packages][alpine] and
|
||||||
|
further instructions to the Containerfile, add additional neovim configuration
|
||||||
|
files etc. and build your variant with `./build.sh my-variant`. You can now run
|
||||||
|
it by executing `code --my-variant`.
|
||||||
|
|
||||||
|
Please consider the files in the template directory for further instructions
|
||||||
|
on installing packages, configuring neovim, installing and configuring plugins
|
||||||
|
etc.
|
||||||
|
|
||||||
|
[alpine]: https://pkgs.alpinelinux.org/packages
|
||||||
|
|||||||
25
basic/Containerfile
Normal file
25
basic/Containerfile
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
FROM code-core
|
||||||
|
USER root
|
||||||
|
ENV HOME=/root
|
||||||
|
|
||||||
|
# Install alpine packages
|
||||||
|
RUN apk add --no-cache procps curl curl-doc make make-doc git git-doc
|
||||||
|
|
||||||
|
# Install neovim plugins
|
||||||
|
WORKDIR /usr/local/share/nvim/site/pack/plugins/opt
|
||||||
|
RUN printf '%s\n' \
|
||||||
|
christoomey/vim-tmux-navigator benmills/vimux ap/vim-buftabline \
|
||||||
|
nvim-tree/nvim-tree.lua nvim-tree/nvim-web-devicons ctrlpvim/ctrlp.vim \
|
||||||
|
hrsh7th/nvim-cmp hrsh7th/cmp-buffer hrsh7th/cmp-path hrsh7th/cmp-nvim-lua \
|
||||||
|
hrsh7th/cmp-cmdline hrsh7th/cmp-nvim-lsp neovim/nvim-lspconfig \
|
||||||
|
saadparwaiz1/cmp_luasnip L3MON4D3/LuaSnip jiangmiao/auto-pairs \
|
||||||
|
tpope/vim-surround tpope/vim-repeat kaiuri/nvim-juliana \
|
||||||
|
| xargs -n1 -P0 -I'{}' git clone --depth 1 'https://github.com/{}'
|
||||||
|
RUN printf 'helptags %s\n' */doc | nvim --noplugin -es
|
||||||
|
|
||||||
|
# Add neovim config
|
||||||
|
COPY nvim /etc/xdg/nvim
|
||||||
|
|
||||||
|
# Configure environment
|
||||||
|
USER 1000:1000
|
||||||
|
ENV HOME=/tmp
|
||||||
32
basic/nvim/init/00-basic.lua
Normal file
32
basic/nvim/init/00-basic.lua
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
local pack = require('pack')
|
||||||
|
|
||||||
|
-- tmux integration
|
||||||
|
pack 'vim-tmux-navigator'
|
||||||
|
pack 'vimux'
|
||||||
|
|
||||||
|
-- File tree navigation
|
||||||
|
pack 'nvim-web-devicons'
|
||||||
|
pack 'nvim-tree.lua'
|
||||||
|
pack 'ctrlp.vim'
|
||||||
|
|
||||||
|
-- Language servers and snippets
|
||||||
|
pack 'nvim-lspconfig'
|
||||||
|
pack 'LuaSnip'
|
||||||
|
|
||||||
|
-- Completion
|
||||||
|
pack 'cmp-buffer'
|
||||||
|
pack 'cmp-path'
|
||||||
|
pack 'cmp-cmdline'
|
||||||
|
pack 'cmp-nvim-lua'
|
||||||
|
pack 'cmp-nvim-lsp'
|
||||||
|
pack 'cmp_luasnip'
|
||||||
|
pack 'nvim-cmp'
|
||||||
|
|
||||||
|
-- Editing
|
||||||
|
pack 'auto-pairs'
|
||||||
|
pack 'vim-surround'
|
||||||
|
pack 'vim-repeat'
|
||||||
|
|
||||||
|
-- Visuals
|
||||||
|
pack 'vim-buftabline'
|
||||||
|
pack 'nvim-juliana'
|
||||||
9
basic/nvim/lua/pack.lua
Normal file
9
basic/nvim/lua/pack.lua
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
local conf_path = 'plugconf/%s.lua'
|
||||||
|
|
||||||
|
function pack(name)
|
||||||
|
vim.cmd('packadd! ' .. name)
|
||||||
|
name = name:gsub('^n?vim%-', ''):gsub('%..*$', '')
|
||||||
|
vim.cmd.runtime(conf_path:format(name))
|
||||||
|
end
|
||||||
|
|
||||||
|
return pack
|
||||||
25
basic/nvim/luasnippets/all.lua
Normal file
25
basic/nvim/luasnippets/all.lua
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
return {
|
||||||
|
parse(
|
||||||
|
{ trig = 'lorem', name = 'Lorem ipsum sentence' },
|
||||||
|
table.concat({
|
||||||
|
'Lorem ipsum dolor sit amet, qui minim labore adipisicing minim sint',
|
||||||
|
'cillum sint consectetur cupidatat.',
|
||||||
|
}, " ")
|
||||||
|
),
|
||||||
|
parse(
|
||||||
|
{ trig = 'loremipsum', name = "Lorem ipsum paragraph" },
|
||||||
|
table.concat({
|
||||||
|
'Lorem ipsum dolor sit amet, officia excepteur ex fugiat reprehenderit',
|
||||||
|
'enim labore culpa sint ad nisi Lorem pariatur mollit ex esse',
|
||||||
|
'exercitation amet. Nisi anim cupidatat excepteur officia.',
|
||||||
|
'Reprehenderit nostrud nostrud ipsum Lorem est aliquip amet voluptate',
|
||||||
|
'voluptate dolor minim nulla est proident. Nostrud officia pariatur ut',
|
||||||
|
'officia. Sit irure elit esse ea nulla sunt ex occaecat reprehenderit',
|
||||||
|
'commodo officia dolor Lorem duis laboris cupidatat officia voluptate.',
|
||||||
|
'Culpa proident adipisicing id nulla nisi laboris ex in Lorem sunt duis',
|
||||||
|
'officia eiusmod. Aliqua reprehenderit commodo ex non excepteur duis',
|
||||||
|
'sunt velit enim. Voluptate laboris sint cupidatat ullamco ut ea',
|
||||||
|
'consectetur et est culpa et culpa duis.',
|
||||||
|
}, ' ')
|
||||||
|
),
|
||||||
|
}
|
||||||
13
basic/nvim/plugconf/LuaSnip.lua
Normal file
13
basic/nvim/plugconf/LuaSnip.lua
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
local luasnip = require 'luasnip'
|
||||||
|
local loader = require 'luasnip.loaders.from_lua'
|
||||||
|
|
||||||
|
luasnip.setup {
|
||||||
|
-- Enable live updates
|
||||||
|
update_events = { 'TextChanged', 'TextChangedI' },
|
||||||
|
|
||||||
|
-- Stop jumping when deleted
|
||||||
|
delete_check_events = { 'TextChanged' },
|
||||||
|
}
|
||||||
|
|
||||||
|
-- Automatically load lua snippets
|
||||||
|
loader.lazy_load()
|
||||||
4
basic/nvim/plugconf/auto-pairs.lua
Normal file
4
basic/nvim/plugconf/auto-pairs.lua
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
-- Replace <c-b> mapping
|
||||||
|
vim.keymap.set('i', '<c-b>', '{', { remap = true })
|
||||||
|
vim.keymap.del('i', '<c-b> ')
|
||||||
|
vim.keymap.del('i', '<c-b><cr>')
|
||||||
6
basic/nvim/plugconf/cmp-nvim-lsp.lua
Normal file
6
basic/nvim/plugconf/cmp-nvim-lsp.lua
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
local lspconfig = require 'lspconfig'
|
||||||
|
local cmp_nvim = require 'cmp_nvim_lsp'
|
||||||
|
|
||||||
|
-- Set default capabilities
|
||||||
|
local capabilities = cmp_nvim.default_capabilities()
|
||||||
|
lspconfig.util.default_config.capabilities = capabilities
|
||||||
94
basic/nvim/plugconf/cmp.lua
Normal file
94
basic/nvim/plugconf/cmp.lua
Normal file
@@ -0,0 +1,94 @@
|
|||||||
|
local cmp = require 'cmp'
|
||||||
|
local context = require 'cmp.config.context'
|
||||||
|
local luasnip = require 'luasnip'
|
||||||
|
|
||||||
|
vim.o.completeopt = nil
|
||||||
|
|
||||||
|
-- General setup
|
||||||
|
cmp.setup {
|
||||||
|
snippet = {
|
||||||
|
expand = function(args) luasnip.lsp_expand(args.body) end,
|
||||||
|
},
|
||||||
|
window = {
|
||||||
|
documentation = cmp.config.window.bordered(),
|
||||||
|
},
|
||||||
|
formatting = {
|
||||||
|
fields = { 'menu', 'abbr', 'kind' },
|
||||||
|
format = function(entry, item)
|
||||||
|
item.menu = ({
|
||||||
|
nvim_lsp = 'λ',
|
||||||
|
luasnip = '⋗',
|
||||||
|
buffer = 'Ω',
|
||||||
|
path = '🖫',
|
||||||
|
})[entry.source.name]
|
||||||
|
return item
|
||||||
|
end,
|
||||||
|
},
|
||||||
|
preselect = cmp.PreselectMode.None,
|
||||||
|
mapping = {
|
||||||
|
-- Confirm, jump or select with tab
|
||||||
|
['<tab>'] = cmp.mapping(function(fallback)
|
||||||
|
if cmp.get_selected_entry() then
|
||||||
|
cmp.confirm()
|
||||||
|
elseif luasnip.jumpable(1) then
|
||||||
|
luasnip.jump(1)
|
||||||
|
elseif luasnip.expandable() then
|
||||||
|
luasnip.expand()
|
||||||
|
elseif cmp.visible() then
|
||||||
|
cmp.select_next_item()
|
||||||
|
else
|
||||||
|
fallback()
|
||||||
|
end
|
||||||
|
end, { 'i', 's' }),
|
||||||
|
-- Jump back with shift tab
|
||||||
|
['<s-tab>'] = cmp.mapping(function(fallback)
|
||||||
|
if luasnip.jumpable(-1) then
|
||||||
|
luasnip.jump(-1)
|
||||||
|
else
|
||||||
|
fallback()
|
||||||
|
end
|
||||||
|
end, { 'i', 's' }),
|
||||||
|
-- Navigate options with ctrl n and ctrl p
|
||||||
|
['<c-n>'] = cmp.mapping(function(fallback)
|
||||||
|
if not cmp.visible() then
|
||||||
|
cmp.complete()
|
||||||
|
else
|
||||||
|
cmp.select_next_item()
|
||||||
|
end
|
||||||
|
end),
|
||||||
|
['<c-p>'] = cmp.mapping(function(fallback)
|
||||||
|
if cmp.visible() then cmp.select_prev_item() else fallback() end
|
||||||
|
end),
|
||||||
|
},
|
||||||
|
sources = cmp.config.sources({
|
||||||
|
{ name = 'nvim_lsp' },
|
||||||
|
{ name = 'luasnip' },
|
||||||
|
{ name = 'nvim_lua' },
|
||||||
|
}, {
|
||||||
|
{ name = 'buffer', keyword_length = 5 },
|
||||||
|
{ name = 'path' },
|
||||||
|
}),
|
||||||
|
completion = {
|
||||||
|
keyword_length = 3
|
||||||
|
},
|
||||||
|
experimental = {
|
||||||
|
ghost_text = true,
|
||||||
|
},
|
||||||
|
-- Disable in comments
|
||||||
|
enabled = function()
|
||||||
|
return not context.in_syntax_group('Comment')
|
||||||
|
end,
|
||||||
|
}
|
||||||
|
|
||||||
|
-- Command line setup
|
||||||
|
cmp.setup.cmdline(':', {
|
||||||
|
mapping = cmp.mapping.preset.cmdline(),
|
||||||
|
sources = cmp.config.sources({
|
||||||
|
{ name = 'path' },
|
||||||
|
},{
|
||||||
|
{ name = 'cmdline' },
|
||||||
|
}),
|
||||||
|
completion = {
|
||||||
|
keyword_length = 2
|
||||||
|
},
|
||||||
|
})
|
||||||
5
basic/nvim/plugconf/ctrlp.lua
Normal file
5
basic/nvim/plugconf/ctrlp.lua
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
-- Ctrlp
|
||||||
|
vim.g.ctrlp_open_multiple_files = '1r'
|
||||||
|
|
||||||
|
-- Keyboard shortcuts
|
||||||
|
vim.g.ctrlp_map = '<c-j>'
|
||||||
5
basic/nvim/plugconf/juliana.lua
Normal file
5
basic/nvim/plugconf/juliana.lua
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
local juliana = require 'nvim-juliana'
|
||||||
|
|
||||||
|
-- Set color scheme
|
||||||
|
juliana.setup { colors = { fg4 = '#596875' } }
|
||||||
|
vim.cmd.colorscheme('juliana')
|
||||||
12
basic/nvim/plugconf/lspconfig.lua
Normal file
12
basic/nvim/plugconf/lspconfig.lua
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
-- Set keymaps
|
||||||
|
vim.api.nvim_create_autocmd('LspAttach', {
|
||||||
|
group = vim.api.nvim_create_augroup('UserLspConfig', {}),
|
||||||
|
callback = function(ev)
|
||||||
|
local opts = { buffer = ev.buf }
|
||||||
|
vim.keymap.set('n', 'gd', vim.lsp.buf.definition, opts)
|
||||||
|
vim.keymap.set('n', 'gr', vim.lsp.buf.references, opts)
|
||||||
|
vim.keymap.set('n', 'K', vim.lsp.buf.hover, opts)
|
||||||
|
vim.keymap.set('n', 'cr', vim.lsp.buf.rename, opts)
|
||||||
|
vim.keymap.set('n', 'ca', vim.lsp.buf.code_action, opts)
|
||||||
|
end,
|
||||||
|
})
|
||||||
9
basic/nvim/plugconf/tmux-navigator.lua
Normal file
9
basic/nvim/plugconf/tmux-navigator.lua
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
-- Disable wrapping
|
||||||
|
vim.g.tmux_navigator_no_wrap = 1
|
||||||
|
|
||||||
|
-- Keyboard shortcuts
|
||||||
|
vim.g.tmux_navigator_no_mappings = 1
|
||||||
|
vim.keymap.set('n', '<m-h>', vim.cmd.TmuxNavigateLeft, { silent = true })
|
||||||
|
vim.keymap.set('n', '<m-j>', vim.cmd.TmuxNavigateDown, { silent = true })
|
||||||
|
vim.keymap.set('n', '<m-k>', vim.cmd.TmuxNavigateUp, { silent = true })
|
||||||
|
vim.keymap.set('n', '<m-l>', vim.cmd.TmuxNavigateRight, { silent = true })
|
||||||
106
basic/nvim/plugconf/tree.lua
Normal file
106
basic/nvim/plugconf/tree.lua
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
local tree = require 'nvim-tree'
|
||||||
|
|
||||||
|
-- Disable NetRw
|
||||||
|
vim.g.loaded_netrw = 1
|
||||||
|
vim.g.loaded_netrwPlugin = 1
|
||||||
|
|
||||||
|
-- Keyboard shortcuts
|
||||||
|
vim.keymap.set('n', 'U', vim.cmd.NvimTreeToggle, { silent = true })
|
||||||
|
|
||||||
|
-- Configure Nvim Tree
|
||||||
|
tree.setup {
|
||||||
|
sync_root_with_cwd = true,
|
||||||
|
update_focused_file = { enable = true },
|
||||||
|
git = { ignore = false },
|
||||||
|
renderer = {
|
||||||
|
highlight_git = true,
|
||||||
|
highlight_opened_files = "icon",
|
||||||
|
root_folder_label = ":~:s?$?",
|
||||||
|
},
|
||||||
|
on_attach = function(buf)
|
||||||
|
-- Mapping helpers
|
||||||
|
local api = require 'nvim-tree.api'
|
||||||
|
local function opts(desc) return {
|
||||||
|
desc = 'nvim-tree: ' .. desc, buffer = buf,
|
||||||
|
noremap = true, silent = true, nowait = true,
|
||||||
|
} end
|
||||||
|
|
||||||
|
-- Directories
|
||||||
|
vim.keymap.set('n', 'C', api.tree.change_root_to_node, opts('Change Directory'))
|
||||||
|
vim.keymap.set('n', '-', api.tree.change_root_to_parent, opts('Up'))
|
||||||
|
vim.keymap.set('n', 'P', api.node.navigate.parent, opts('Parent Directory'))
|
||||||
|
|
||||||
|
-- Open
|
||||||
|
vim.keymap.set('n', '<cr>', api.node.open.edit, opts('Open: Toggle'))
|
||||||
|
vim.keymap.set('n', '<space>', api.node.open.edit, opts('Open: Toggle'))
|
||||||
|
vim.keymap.set('n', 'v', api.node.open.vertical, opts('Open: Vertical Split'))
|
||||||
|
vim.keymap.set('n', 's', api.node.open.horizontal, opts('Open: Horizontal Split'))
|
||||||
|
vim.keymap.set('n', '<Tab>', api.node.open.preview, opts('Open: Preview'))
|
||||||
|
vim.keymap.set('n', 'i', api.node.show_info_popup, opts('Info'))
|
||||||
|
|
||||||
|
-- File operations
|
||||||
|
vim.keymap.set('n', 'r', api.fs.rename_sub, opts('Rename'))
|
||||||
|
vim.keymap.set('n', 'R', api.fs.rename_basename, opts('Rename: Basename'))
|
||||||
|
vim.keymap.set('n', 'a', api.fs.create, opts('Create'))
|
||||||
|
vim.keymap.set('n', 'c', api.fs.copy.node, opts('Copy'))
|
||||||
|
vim.keymap.set('n', 'x', api.fs.cut, opts('Cut'))
|
||||||
|
vim.keymap.set('n', 'p', api.fs.paste, opts('Paste'))
|
||||||
|
vim.keymap.set('n', 'd', api.fs.remove, opts('Delete'))
|
||||||
|
vim.keymap.set('n', '.', api.node.run.cmd, opts('Run Command'))
|
||||||
|
|
||||||
|
-- Path copying
|
||||||
|
vim.keymap.set('n', 'y', api.fs.copy.relative_path, opts('Copy Relative Path'))
|
||||||
|
vim.keymap.set('n', 'Y', api.fs.copy.absolute_path, opts('Copy Absolute Path'))
|
||||||
|
|
||||||
|
-- Navigation
|
||||||
|
vim.keymap.set('n', 'q', api.tree.close, opts('Close'))
|
||||||
|
vim.keymap.set('n', '/', api.tree.search_node, opts('Search'))
|
||||||
|
vim.keymap.set('n', '?', api.tree.toggle_help, opts('Help'))
|
||||||
|
|
||||||
|
-- Filtering
|
||||||
|
vim.keymap.set('n', 'f', api.live_filter.start, opts('Filter'))
|
||||||
|
vim.keymap.set('n', 'F', api.live_filter.clear, opts('Clean Filter'))
|
||||||
|
vim.keymap.set('n', 'H', api.tree.toggle_hidden_filter, opts('Toggle Dotfiles'))
|
||||||
|
vim.keymap.set('n', 'T', api.tree.toggle_git_clean_filter, opts('Toggle Git Clean'))
|
||||||
|
vim.keymap.set('n', 'Z', api.tree.collapse_all, opts('Collapse'))
|
||||||
|
vim.keymap.set('n', 'z', api.tree.expand_all, opts('Expand All'))
|
||||||
|
|
||||||
|
-- Bookmarks
|
||||||
|
vim.keymap.set('n', 'm', api.marks.toggle, opts('Toggle Bookmark'))
|
||||||
|
vim.keymap.set('n', 'M', api.marks.bulk.move, opts('Move Bookmarked'))
|
||||||
|
|
||||||
|
-- Custom open and close
|
||||||
|
vim.keymap.set('n', 'l', function()
|
||||||
|
local node = api.tree.get_node_under_cursor()
|
||||||
|
if node.type == 'directory' then
|
||||||
|
if not node.open then api.node.open.edit() end
|
||||||
|
else
|
||||||
|
api.node.open.edit()
|
||||||
|
end
|
||||||
|
end, opts('Open'))
|
||||||
|
|
||||||
|
vim.keymap.set('n', 'h', function()
|
||||||
|
local node = api.tree.get_node_under_cursor()
|
||||||
|
if node.type == 'directory' then
|
||||||
|
if node.open then api.node.open.edit() end
|
||||||
|
else
|
||||||
|
local path = node.absolute_path
|
||||||
|
if vim.fn.buflisted(path) == 1 then
|
||||||
|
if vim.fn.getbufinfo(path)[1].changed == 0 then
|
||||||
|
vim.cmd.bdelete(path)
|
||||||
|
else
|
||||||
|
vim.notify('Buffer has unsaved changes', vim.log.levels.ERROR)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end, opts('Close'))
|
||||||
|
|
||||||
|
vim.keymap.set('n', 'o', function()
|
||||||
|
local node = api.tree.get_node_under_cursor()
|
||||||
|
if node.type == 'file' then
|
||||||
|
api.node.open.edit()
|
||||||
|
api.tree.focus()
|
||||||
|
end
|
||||||
|
end, opts('Open: Keep Focus'))
|
||||||
|
end
|
||||||
|
}
|
||||||
10
basic/nvim/plugconf/vimux.lua
Normal file
10
basic/nvim/plugconf/vimux.lua
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
-- Keyboard shortcuts
|
||||||
|
if vim.env.TMUX ~= nil then
|
||||||
|
vim.keymap.set('n', '<f5>', function()
|
||||||
|
vim.cmd.write()
|
||||||
|
local file = vim.api.nvim_buf_get_name(0)
|
||||||
|
local command = vim.g.vimux_command or 'make'
|
||||||
|
vim.fn.VimuxRunCommand(command:format(file))
|
||||||
|
end)
|
||||||
|
vim.keymap.set('n', '<f6>', vim.fn.VimuxCloseRunner, { silent = true })
|
||||||
|
end
|
||||||
15
build.sh
Executable file
15
build.sh
Executable file
@@ -0,0 +1,15 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
shopt -s extglob
|
||||||
|
|
||||||
|
cd "$(dirname "$0")"
|
||||||
|
|
||||||
|
if (( $# == 0 )); then
|
||||||
|
readonly images=(core basic !(template|core|basic)/)
|
||||||
|
else
|
||||||
|
readonly images=("$@")
|
||||||
|
fi
|
||||||
|
|
||||||
|
for image in "${images[@]}"; do
|
||||||
|
podman build -t "code-${image%/}" -f "${image%/}/Containerfile"
|
||||||
|
done
|
||||||
72
code
Executable file
72
code
Executable file
@@ -0,0 +1,72 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
# Config
|
||||||
|
readonly DEFAULT_VARIANT='core'
|
||||||
|
readonly LOOPBACK_ARGS=(
|
||||||
|
--network slirp4netns:allow_host_loopback=true
|
||||||
|
--add-host host:10.0.2.2
|
||||||
|
)
|
||||||
|
|
||||||
|
# Help text
|
||||||
|
readonly HELP="code [--publish <port-spec>] [--loopback]
|
||||||
|
[--<variant>] [--variants] [--help] [vim-params...]
|
||||||
|
|
||||||
|
--publish Publish port(s) from container
|
||||||
|
--loopback Allow host loopback
|
||||||
|
--<variant> Use editor variant <variant> (default: ${DEFAULT_VARIANT})
|
||||||
|
--variants Print list of available editor variants and exit
|
||||||
|
--help Show this help and exit"
|
||||||
|
|
||||||
|
# Get script directory
|
||||||
|
readonly DIR="$(dirname "$(readlink -f -- "$0")")"
|
||||||
|
|
||||||
|
# Parse arguments
|
||||||
|
ports=()
|
||||||
|
loopback=false
|
||||||
|
variant="$DEFAULT_VARIANT"
|
||||||
|
|
||||||
|
for arg in "$@"; do
|
||||||
|
case "$arg" in
|
||||||
|
--publish ) shift; ports+=("$1");;
|
||||||
|
--loopback ) loopback=true;;
|
||||||
|
--variants )
|
||||||
|
for f in "$DIR"/*; do
|
||||||
|
[[ "$f" == */template || ! -d "$f" ]] || echo "${f##*/}";
|
||||||
|
done
|
||||||
|
exit;;
|
||||||
|
-h | --help ) echo >&2 "$HELP"; exit;;
|
||||||
|
--* )
|
||||||
|
if [[ ! -d "${DIR}/${arg#--}" ]]; then break; fi
|
||||||
|
variant="${arg#--}";;
|
||||||
|
* ) break;;
|
||||||
|
esac
|
||||||
|
shift
|
||||||
|
done
|
||||||
|
|
||||||
|
# Collect additional arguments
|
||||||
|
args=()
|
||||||
|
readonly NAME="$(basename "$(pwd)")"
|
||||||
|
if [[ -f "${DIR}/${variant}/podman.args" ]]; then
|
||||||
|
mapfile -t custom < "${DIR}/${variant}/podman.args"
|
||||||
|
for arg in "${custom[@]}"; do
|
||||||
|
arg="${arg//'{NAME}'/"$NAME"}"
|
||||||
|
arg="${arg//'{DIR}'/"$DIR"}"
|
||||||
|
arg="${arg//'{HOME}'/"$HOME"}"
|
||||||
|
args+=("$arg")
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
for port in "${ports[@]}"; do
|
||||||
|
[[ ! "$port" =~ ^[0-9]+$ ]] || port="${port}:${port}"
|
||||||
|
args+=(--publish "$port")
|
||||||
|
done
|
||||||
|
if $loopback; then args+=("${LOOPBACK_ARGS[@]}"); fi
|
||||||
|
|
||||||
|
# Start editor
|
||||||
|
exec podman run --rm -it \
|
||||||
|
--tz local \
|
||||||
|
--detach-keys ctrl-q,ctrl-q \
|
||||||
|
--volume ".:/work/$NAME" \
|
||||||
|
--workdir "/work/$NAME" \
|
||||||
|
--userns keep-id --user "$(id -u):$(id -g)" \
|
||||||
|
"${args[@]}" "code-${variant}" "$@"
|
||||||
19
core/Containerfile
Normal file
19
core/Containerfile
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
FROM alpine:latest
|
||||||
|
|
||||||
|
# Install alpine packages
|
||||||
|
RUN apk add --no-cache \
|
||||||
|
mandoc mandoc-doc less less-doc bash bash-doc bash-completion \
|
||||||
|
tmux tmux-doc neovim neovim-doc
|
||||||
|
|
||||||
|
# Add shell configs
|
||||||
|
COPY shell /etc
|
||||||
|
|
||||||
|
# Add neovim config
|
||||||
|
COPY nvim /etc/xdg/nvim
|
||||||
|
ENV VIMINIT=":luafile /etc/xdg/nvim/init.lua"
|
||||||
|
|
||||||
|
# Configure environment
|
||||||
|
USER 1000:1000
|
||||||
|
WORKDIR /work
|
||||||
|
ENV HOME=/tmp TERM=xterm-256color LANG=en_US.UTF-8
|
||||||
|
ENTRYPOINT ["/usr/bin/tmux", "new-session", "/usr/bin/nvim"]
|
||||||
69
core/nvim/init.lua
Normal file
69
core/nvim/init.lua
Normal file
@@ -0,0 +1,69 @@
|
|||||||
|
-- Change files in place
|
||||||
|
vim.o.backupcopy = 'yes'
|
||||||
|
|
||||||
|
-- Disable auto comment
|
||||||
|
vim.cmd.autocmd('FileType', '*', 'setlocal', 'formatoptions-=cro')
|
||||||
|
|
||||||
|
-- Disable double spaces when joining
|
||||||
|
vim.o.joinspaces = false
|
||||||
|
|
||||||
|
-- Code folding
|
||||||
|
vim.o.foldmethod = 'indent'
|
||||||
|
vim.o.foldlevel = 99
|
||||||
|
|
||||||
|
-- Indentation
|
||||||
|
vim.o.tabstop = 2
|
||||||
|
vim.o.shiftwidth = 2
|
||||||
|
vim.o.softtabstop = -1
|
||||||
|
vim.o.expandtab = true
|
||||||
|
|
||||||
|
-- Visible line breaks
|
||||||
|
vim.o.list = true
|
||||||
|
vim.o.listchars = 'eol:¬'
|
||||||
|
|
||||||
|
-- Relative line numbers
|
||||||
|
vim.o.number = true
|
||||||
|
vim.o.relativenumber = true
|
||||||
|
|
||||||
|
-- 80 character indication
|
||||||
|
vim.o.colorcolumn = '80'
|
||||||
|
|
||||||
|
-- Status window height
|
||||||
|
vim.o.previewheight = 30
|
||||||
|
|
||||||
|
-- Theme
|
||||||
|
vim.o.termguicolors = true
|
||||||
|
vim.cmd.colorscheme('slate')
|
||||||
|
|
||||||
|
-- Keyboard shortcuts
|
||||||
|
vim.keymap.set('n', '<c-n>', ':bn<cr>', { silent = true })
|
||||||
|
vim.keymap.set('n', '<c-p>', ':bp<cr>', { silent = true })
|
||||||
|
vim.keymap.set('n', '<c-l>', ':noh<cr><c-l>')
|
||||||
|
vim.keymap.set('n', 'gp', "'[V']", { remap = true })
|
||||||
|
vim.keymap.set('i', '<c-b>', '{}<left>', { silent = true } )
|
||||||
|
vim.keymap.set('i', '<c-b> ', '{ }<left><left>', { silent = true } )
|
||||||
|
vim.keymap.set('i', '<c-b><cr>', '{<cr>}O', { silent = true } )
|
||||||
|
vim.keymap.set('i', '<c-space>', '<space>', { remap = true })
|
||||||
|
|
||||||
|
-- Add command to show help in current window
|
||||||
|
vim.api.nvim_create_user_command('Help', function(a)
|
||||||
|
-- Create empty 'help' buffer
|
||||||
|
vim.cmd.enew()
|
||||||
|
vim.o.buftype = 'help'
|
||||||
|
local buf = vim.fn.bufnr('%')
|
||||||
|
|
||||||
|
-- Show help and handle errors
|
||||||
|
ok, msg = pcall(vim.cmd.help, a.fargs[1] or "help.txt")
|
||||||
|
if not ok then vim.notify(msg, vim.log.levels.ERROR); vim.cmd.bp() end
|
||||||
|
|
||||||
|
-- Wipe out empty buffer if not taken over
|
||||||
|
if vim.fn.bufnr('%') ~= buf then vim.cmd.bwipeout(buf) end
|
||||||
|
end, { nargs = '?', bar = true, complete = 'help' })
|
||||||
|
|
||||||
|
-- Add command line abbreviation if at start of line
|
||||||
|
vim.cmd.cabbrev { 'help',
|
||||||
|
'<c-r>=(getcmdtype() == ":" && getcmdpos() == 1) ? "Help" : "help"<cr>'
|
||||||
|
}
|
||||||
|
|
||||||
|
-- Source init directory
|
||||||
|
vim.cmd 'runtime! init/*.lua'
|
||||||
7
core/shell/inputrc
Normal file
7
core/shell/inputrc
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
set editing-mode vi
|
||||||
|
|
||||||
|
$if mode=vi
|
||||||
|
set keymap vi-insert
|
||||||
|
Control-l: clear-screen
|
||||||
|
Control-r: vi-put
|
||||||
|
$endif
|
||||||
24
core/shell/profile
Normal file
24
core/shell/profile
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
# Set editor
|
||||||
|
EDITOR='nvim'
|
||||||
|
alias vim='nvim'
|
||||||
|
alias vi='nvim'
|
||||||
|
|
||||||
|
# Configure history
|
||||||
|
HISTCONTROL=ignoreboth
|
||||||
|
unset HISTFILE
|
||||||
|
|
||||||
|
# Handy aliases
|
||||||
|
alias la='ls -A'
|
||||||
|
alias ll='la -l'
|
||||||
|
alias ..='cd ..'
|
||||||
|
alias e='exit'
|
||||||
|
|
||||||
|
# Git branch helper
|
||||||
|
git_branch_prompt() {
|
||||||
|
local branch="$(git branch --show-current 2> /dev/null)"
|
||||||
|
[[ -z "$branch" ]] || echo "[${branch}]"
|
||||||
|
}
|
||||||
|
|
||||||
|
# Colorful command prompt
|
||||||
|
PS1='\[\e[33m\][\[\e[0m\]\W\[\e[33m\]]\[\e[32m\]$(git_branch_prompt)\[\e[33m\]\$\[\e[37m\] '
|
||||||
|
trap 'printf \\e[0m' DEBUG
|
||||||
83
core/shell/tmux.conf
Normal file
83
core/shell/tmux.conf
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
# Set default shell
|
||||||
|
set -g default-shell /bin/bash
|
||||||
|
|
||||||
|
# Increase history size
|
||||||
|
set -g history-limit 50000
|
||||||
|
|
||||||
|
# Disable escape delay
|
||||||
|
set-option -sg escape-time 0
|
||||||
|
|
||||||
|
# Enable mouse and focus events
|
||||||
|
set -g mouse on
|
||||||
|
set -g focus-events on
|
||||||
|
|
||||||
|
################
|
||||||
|
# Key bindings #
|
||||||
|
################
|
||||||
|
|
||||||
|
# Change prefix
|
||||||
|
set-option -g prefix C-a
|
||||||
|
|
||||||
|
# Remap split commands
|
||||||
|
unbind '"'
|
||||||
|
unbind %
|
||||||
|
bind s split-window -v
|
||||||
|
bind v split-window -h
|
||||||
|
|
||||||
|
# Remap pane selection
|
||||||
|
unbind Up
|
||||||
|
unbind Down
|
||||||
|
unbind Left
|
||||||
|
unbind Right
|
||||||
|
|
||||||
|
is_vim="ps -o state= -o comm= -t '#{pane_tty}' \
|
||||||
|
| grep -iqE '^[^TXZ ]+ +(\\S+\\/)?g?(view|l?n?vim?x?)(diff)?$'"
|
||||||
|
bind-key -n 'M-h' if-shell "$is_vim" { send-keys M-h } { if-shell -F '#{pane_at_left}' {} { select-pane -L } }
|
||||||
|
bind-key -n 'M-j' if-shell "$is_vim" { send-keys M-j } { if-shell -F '#{pane_at_bottom}' {} { select-pane -D } }
|
||||||
|
bind-key -n 'M-k' if-shell "$is_vim" { send-keys M-k } { if-shell -F '#{pane_at_top}' {} { select-pane -U } }
|
||||||
|
bind-key -n 'M-l' if-shell "$is_vim" { send-keys M-l } { if-shell -F '#{pane_at_right}' {} { select-pane -R } }
|
||||||
|
|
||||||
|
# Remap resize
|
||||||
|
unbind C-Up
|
||||||
|
unbind C-Down
|
||||||
|
unbind C-Left
|
||||||
|
unbind C-Right
|
||||||
|
unbind M-Up
|
||||||
|
unbind M-Down
|
||||||
|
unbind M-Left
|
||||||
|
unbind M-Right
|
||||||
|
bind -nr M-K resize-pane -U
|
||||||
|
bind -nr M-J resize-pane -D
|
||||||
|
bind -nr M-H resize-pane -L
|
||||||
|
bind -nr M-L resize-pane -R
|
||||||
|
|
||||||
|
##########
|
||||||
|
# Design #
|
||||||
|
##########
|
||||||
|
|
||||||
|
# Borders
|
||||||
|
set -g pane-border-style 'fg=colour12'
|
||||||
|
set -g pane-active-border-style 'fg=colour12'
|
||||||
|
|
||||||
|
# Statusbar
|
||||||
|
set -g status-position bottom
|
||||||
|
set -g status-justify left
|
||||||
|
set -g status-style 'bg=colour235'
|
||||||
|
set -g status-left ''
|
||||||
|
|
||||||
|
# Command line and messages
|
||||||
|
set -g message-style 'bg=color235'
|
||||||
|
set -g display-time 4000
|
||||||
|
|
||||||
|
# Tabs
|
||||||
|
setw -g window-status-current-style 'bg=colour238 bold'
|
||||||
|
setw -g window-status-current-format ' #[fg=color245]#I#[fg=colour250] #W '
|
||||||
|
setw -g window-status-style 'bg=colour236'
|
||||||
|
setw -g window-status-format ' #[fg=colour240]#I#[fg=colour245] #W '
|
||||||
|
|
||||||
|
# Clock
|
||||||
|
set -g status-right '#[fg=colour0,bg=colour240] %H:%M '
|
||||||
|
set -g status-interval 10
|
||||||
|
|
||||||
|
# Scroll number
|
||||||
|
setw -g mode-style 'fg=colour240'
|
||||||
17
go/Containerfile
Normal file
17
go/Containerfile
Normal file
@@ -0,0 +1,17 @@
|
|||||||
|
FROM code-basic
|
||||||
|
USER root
|
||||||
|
ENV HOME=/root
|
||||||
|
|
||||||
|
# Install alpine packages
|
||||||
|
RUN apk add --no-cache go go-doc
|
||||||
|
|
||||||
|
# Install go packages
|
||||||
|
ENV GOPATH=/usr/local/lib/go PATH=/usr/local/lib/go/bin:$PATH
|
||||||
|
RUN go install golang.org/x/tools/gopls@latest && rm -rf $(go env GOCACHE)
|
||||||
|
|
||||||
|
# Add neovim config
|
||||||
|
COPY nvim /etc/xdg/nvim
|
||||||
|
|
||||||
|
# Configure environment
|
||||||
|
USER 1000:1000
|
||||||
|
ENV HOME=/tmp
|
||||||
1
go/nvim/init/go.lua
Normal file
1
go/nvim/init/go.lua
Normal file
@@ -0,0 +1 @@
|
|||||||
|
require('lspconfig').gopls.setup {}
|
||||||
10
go/nvim/luasnippets/go.lua
Normal file
10
go/nvim/luasnippets/go.lua
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
return {
|
||||||
|
parse(
|
||||||
|
{ trig = 'func', name = 'Function' },
|
||||||
|
'func ${1:name}($2) $3{\n\t$4\n}'
|
||||||
|
),
|
||||||
|
parse(
|
||||||
|
{ trig = 'err', name = 'Handle error' },
|
||||||
|
'if ${2:err} != nil { return ${1}${2} }'
|
||||||
|
),
|
||||||
|
}
|
||||||
4
go/podman.args
Normal file
4
go/podman.args
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
--env
|
||||||
|
GOPATH=/work/{NAME}/.go
|
||||||
|
--env
|
||||||
|
GOCACHE=/work/{NAME}/.go/cache
|
||||||
33
template/Containerfile
Normal file
33
template/Containerfile
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
FROM code-basic
|
||||||
|
USER root
|
||||||
|
ENV HOME=/root
|
||||||
|
|
||||||
|
# Install alpine packages
|
||||||
|
###
|
||||||
|
### Replace ... with the alpine packaes you'd like to install.
|
||||||
|
### You can browser them on https://pkgs.alpinelinux.org/packages.
|
||||||
|
###
|
||||||
|
RUN apk add --no-cache ...
|
||||||
|
|
||||||
|
# Install neovim plugins
|
||||||
|
###
|
||||||
|
### Replace the first ... with <user>/<repo> strings for every neovim plugin
|
||||||
|
### you'd like to install from GitHub. Replace the second ... with a comma
|
||||||
|
### separated list of the <repo>s, that contain a doc directory.
|
||||||
|
###
|
||||||
|
WORKDIR /usr/local/share/nvim/site/pack/plugins/opt
|
||||||
|
RUN printf '%s\n' \
|
||||||
|
... \
|
||||||
|
| xargs -n1 -P0 -I'{}' git clone --depth 1 'https://github.com/{}'
|
||||||
|
RUN printf 'helptags %s\n' {...}/doc | nvim --noplugin -es
|
||||||
|
|
||||||
|
# Add neovim config
|
||||||
|
COPY nvim /etc/xdg/nvim
|
||||||
|
|
||||||
|
###
|
||||||
|
### Execute further instructions and copy furhter configurations here.
|
||||||
|
###
|
||||||
|
|
||||||
|
# Configure environment
|
||||||
|
USER 1000:1000
|
||||||
|
ENV HOME=/tmp
|
||||||
12
template/nvim/init/template.lua
Normal file
12
template/nvim/init/template.lua
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
local pack = require 'pack'
|
||||||
|
|
||||||
|
---
|
||||||
|
--- Add your custom neovim configuration here. This file should be named after
|
||||||
|
--- your editor variant. You can use `pack '<user>/<repo>'` to load a plugin
|
||||||
|
--- and source it's configuration file. For a plugin
|
||||||
|
--- 'my-user/nvim-my-repo.nvim', the configuration file plugconf/my-repo.lua
|
||||||
|
--- would be source, if it exists ("nvim-" and "vim-" prefixes and everyting
|
||||||
|
--- after the first "." get removed from <repo>).
|
||||||
|
---
|
||||||
|
|
||||||
|
pack 'my-user/nvim-my-repo.nvim'
|
||||||
11
template/nvim/luasnippets/all.lua
Normal file
11
template/nvim/luasnippets/all.lua
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
---
|
||||||
|
--- Add your lua snippets here. Rename this file to match the file type your
|
||||||
|
--- snippets are supposed to be availlable for instead of 'all'.
|
||||||
|
---
|
||||||
|
|
||||||
|
return {
|
||||||
|
parse(
|
||||||
|
{ trig = 'my-snippet', name = 'My Snippet' },
|
||||||
|
'The quick brown ${1:fox} jumps over the lazy ${2:dog,cat,elephant}'
|
||||||
|
),
|
||||||
|
}
|
||||||
5
template/plugconf/my-repo.lua
Normal file
5
template/plugconf/my-repo.lua
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
---
|
||||||
|
--- You can import your plugin and set it up or set vim options to configure it
|
||||||
|
--- here. If any of your plugins don't require configuration, you can simply
|
||||||
|
--- omit creating a plugconf/*.lua file for it.
|
||||||
|
---
|
||||||
8
template/podman.args
Normal file
8
template/podman.args
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
###
|
||||||
|
### Add variant specific podman arguments here, one per line, or remove the
|
||||||
|
### file entirely. You can use {DIR} as placeholder for the directory of this
|
||||||
|
### repository, {NAME} for the name of the current working directory, that will
|
||||||
|
### be mounted to /work/{NAME} in the container and {HOME} for the users home
|
||||||
|
### directory on the host system. That way you can change the confinement or
|
||||||
|
### network configuration, mount additional directories into the container etc.
|
||||||
|
###
|
||||||
Reference in New Issue
Block a user