#!/usr/bin/env lua5.3 -- needs xsel, clipnotify, pyclip, wclip -- luarocks-5.3 install --local luaposix -- luarocks-5.3 install --local argparse -- luarocks-5.3 install --local lsqlite3 -- front-end example: sqlite3 $(cat /tmp/lclipd/lclipd_db_name) 'select content from lclipd;' | dmenu -l 10 | xsel -ib local string = require("string") -- Adds the lua rocks modules to the require path for this script local function default_luarocks_modules() local luarocks_handle = io.popen("luarocks-5.3 path --bin") local path_b = false local cpath_b = false for line in luarocks_handle:lines() do local path = string.match(line, "LUA_PATH%s*=%s*('.+')") local cpath = string.match(line, "LUA_CPATH%s*=%s*('.+')") if path ~= nil then package.path = package.path .. ";" .. string.sub(path, 2, -2) end if cpath ~= nil then package.cpath = package.cpath .. ";" .. string.sub(cpath, 2, -2) end end if path_b then os.exit(1) end if cpath_b then os.exit(1) end end default_luarocks_modules() -- we want to delete a pidfile if we wrote one, otherwise we won't local wrote_a_pidfile = false local signal = require("posix.signal") local argparse = require("argparse") local sys_stat = require("posix.sys.stat") local unistd = require("posix.unistd") local posix_syslog = require("posix.syslog") local sqlite3 = require("lsqlite3") local posix_wait = require("posix.sys.wait") local sql_create_table = [=[ create table if not exists lclipd ( id integer primary key, content text unique not null, dateAdded integer unique not null ); ]=] -- We are deleting old entries in groups of 20 instead of one by one local sql_old_reap_trigger = [=[ create trigger if not exists hist_old_reap before insert on lclipd begin delete from lclipd where id = ( select id from lclipd order by dateAdded asc limit 20 ) and ( select count(id) from lclipd ) >= %s; end; ]=] local sql_insert = [=[ insert into lclipd(content,dateAdded) values('%s', unixepoch()); ]=] -- using a heredoc string without expansion bypasses the need for escaping local detect_secrets_cmd = [=[ detect-secrets scan %s --string <<- STR | grep -v False %s STR ]=] local tmp_dir = "/tmp/lclipd" local pid_file = "/tmp/lclipd/lclipd.pid" local db_file_name = "/tmp/lclipd/lclipd_db_name" --- We are not longer running. local function remove_pid_file() if wrote_a_pidfile then os.remove(pid_file) end end --- Adds LUA_PATH and LUA_CPATH to the current interpreters path. local function lclip_exit(n) os.exit(n) remove_pid_file() end local parser = argparse() parser:option("-s --hist_size", "number of distinct entries for clipboard history", 200) parser:option("-d --detect_secrets_args", "options that will be passed to detect secrets", "") --- Log the given string to syslog with the given priority. -- @param log_str the string passed to the logging facility -- @param log_priority the priority of the log string -- functions called through pcall will return nil when we -- try to get their name from debug.getinfo local function log_to_syslog(log_str, log_priority) local caller_name = debug.getinfo(2, "n").name posix_syslog.openlog("clipd", posix_syslog.LOG_NDELAY | posix_syslog.LOG_PID, posix_syslog.LOG_LOCAL0) posix_syslog.syslog(log_priority, tostring(caller_name) .. ": " .. log_str) posix_syslog.closelog() end local function check_uid_gid() log_to_syslog(tostring(unistd.getuid()) .. ":" .. tostring(unistd.getgid()), posix_syslog.LOG_INFO) end --- Change the permission to user read/write i.e. chmod 600 -- @param path to the database file whose permissions will be set local function set_db_permissions(db_path) local ret = sys_stat.chmod(db_path, sys_stat.S_IRUSR | sys_stat.S_IWUSR) if ret ~= 0 then log_to_syslog(tostring(ret), posix_syslog.LOG_CRIT) lclip_exit(1) end end --- Creates the necessary dirs local function make_tmp_dirs() local f = sys_stat.stat(tmp_dir) if f == nil then local ret = sys_stat.mkdir(tmp_dir) if ret ~= 0 then log_to_syslog(ret, posix_syslog.LOG_CRIT) os.exit(1) end end f = sys_stat.stat(tmp_dir .. "/secrets") if f == nil then local ret = sys_stat.mkdir(tmp_dir .. "/secrets") if ret ~= 0 then log_to_syslog(ret, posix_syslog.LOG_CRIT) os.exit(1) end end end --- Tries to determine whether another instance is running, if yes, quits -- obvisouly doing it like this is imprecise but the chances of it failing -- are very low unless we have a constant known way of calling the script -- so that we can match for that exactly for the procfs cmdline check. local function check_pid_file() local f = sys_stat.stat(pid_file) if f ~= nil then local pid_file_handle = io.open(pid_file, "r") local pid_file_content = pid_file_handle:read("*a") pid_file_content = pid_file_content:gsub("\n", "") log_to_syslog(pid_file_content, posix_syslog.LOG_INFO) local old_pid_file = sys_stat.stat("/proc/" .. pid_file_content) if old_pid_file ~= nil then local pid_cmdline = io.open("/proc/" .. pid_file_content .. "/cmdline", "r") local pid_cmdline_content = pid_cmdline:read("*a") if string.match(pid_cmdline_content, "lclipd") then -- we assume a lclipd instance is already running at this point log_to_syslog("clipd is already running", posix_syslog.LOG_CRIT) lclip_exit(1) end -- the old pid file is stale, meaning the previous instance -- died without being able to clean up after itself because -- e.g. it received a SIGKILL end end end --- Write a pidfile local function write_pid_file() local f = io.open(pid_file, "w") if f == nil then log_to_syslog("cant open pid file for writing", posix_syslog.LOG_CRIT) lclip_exit(1) end f:write(tostring(unistd.getpid())) wrote_a_pidfile = true end --- Runs secret detection tests -- returns true if the string is not a secret -- @param clipboard_content the content that will be checked against detect-secrets -- @param detect_secrets_arg extra args that will be passed to detect-secrets scan local function detect_secrets(clipboard_content, detect_secrets_args) if clipboard_content == nil or clipboard_content == "" then return false end local pipe_read, pipe_write = unistd.pipe() if pipe_read == nil then log_to_syslog("could not create pipe", posix_syslog.LOG_CRIT) log_to_syslog(pipe_write, posix_syslog.LOG_CRIT) lclip_exit(1) end local pid, errmsg = unistd.fork() if pid == nil then -- error unistd.closr(pipe_read) unistd.closr(pipe_write) log_to_syslog("could not fork", posix_syslog.LOG_CRIT) log_to_syslog(errmsg, posix_syslog.LOG_CRIT) lclip_exit(1) elseif pid == 0 then -- child unistd.close(pipe_read) local cmd = string.format(detect_secrets_cmd, detect_secrets_args, clipboard_content) local _, secrets_baseline_handle = pcall(io.popen, cmd) local secrets_baseline = secrets_baseline_handle:read("*a") if secrets_baseline == "" then unistd.write(pipe_write, "1") else unistd.write(pipe_write, "0") end unistd.close(pipe_write) unistd._exit(0) elseif pid > 0 then -- parent log_to_syslog("spawned " .. tostring(pid), posix_syslog.LOG_INFO) unistd.close(pipe_write) posix_wait.wait(pid) local result = unistd.read(pipe_read, 1) unistd.close(pipe_read) if result == "0" then return false else return true end end end --- Get the clipboard content from X or wayland. local function get_clipboard_content() -- if we use a plain os.execute for clipnotify the parent wont get the -- SIGINT when it is passed.clipnotify will end up getting it. -- if we fork though, the parent receives the SIGINT just fine. local pid, errmsg = unistd.fork() if pid == nil then -- error log_to_syslog("could not fork", posix_syslog.LOG_CRIT) log_to_syslog(errmsg, posix_syslog.LOG_CRIT) lclip_exit(1) elseif pid == 0 then -- child os.execute("clipnotify") unistd._exit(0) elseif pid > 0 then -- parent log_to_syslog("spawned " .. tostring(pid), posix_syslog.LOG_INFO) -- clipnotify exits when there is a new entry on the clipboard -- so we do want a blocking call here posix_wait.wait(pid) -- we dont care whether all the calls to the different clipboard apps -- succeed or not so we just ignore the errors. local _, handle_x = pcall(io.popen, "xsel -ob") if handle_x ~= nil then local last_clip_entry_x = handle_x:read("*a") if last_clip_entry_x ~= "" and last_clip_entry_x ~= nil then return last_clip_entry_x end end local _, handle_w = pcall(io.popen, "wl-paste") if handle_w ~= nil then local last_clip_entry_w = handle_w:read("*a") if last_clip_entry_w ~= "" and last_clip_entry_w ~= nil then return last_clip_entry_w end end return nil end end --- Get the sqlite DB handle. local function get_sqlite_handle() local tmp_db_name = "/tmp/" .. io.popen( "tr -dc A-Za-z0-9