rspamd/lualib/rspamadm/keypair.lua
2018-06-09 16:00:28 +01:00

485 rivejä
12 KiB
Lua

--[[
Copyright (c) 2018, Vsevolod Stakhov <vsevolod@highsecure.ru>
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
]]--
local argparse = require "argparse"
local rspamd_keypair = require "rspamd_cryptobox_keypair"
local rspamd_pubkey = require "rspamd_cryptobox_pubkey"
local rspamd_signature = require "rspamd_cryptobox_signature"
local rspamd_crypto = require "rspamd_cryptobox"
local rspamd_util = require "rspamd_util"
local ucl = require "ucl"
local logger = require "rspamd_logger"
-- Define command line options
local parser = argparse()
:name "rspamadm keypair"
:description "Manages keypairs for Rspamd"
:help_description_margin(30)
:command_target("command")
:require_command(false)
-- Generate subcommand
local generate = parser:command "generate gen g"
:description "Creates a new keypair"
generate:flag "-s --sign"
:description "Generates a sign keypair instead of the encryption one"
generate:flag "-n --nist"
:description "Uses nist encryption algorithm"
generate:option "-o --output"
:description "Write keypair to file"
:argname "<file>"
generate:mutex(
generate:flag "-j --json"
:description "Output JSON instead of UCL",
generate:flag "-u --ucl"
:description "Output UCL"
:default(true)
)
-- Sign subcommand
local sign = parser:command "sign sig s"
:description "Signs a file using keypair"
sign:option "-k --keypair"
:description "Keypair to use"
:argname "<file>"
sign:option "-s --suffix"
:description "Suffix for signature"
:argname "<suffix>"
:default("sig")
sign:argument "file"
:description "File to sign"
:argname "<file>"
:args "*"
-- Verify subcommand
local verify = parser:command "verify ver v"
:description "Verifies a file using keypair or a public key"
verify:mutex(
verify:option "-p --pubkey"
:description "Load pubkey from the specified file"
:argname "<file>",
verify:option "-P --pubstring"
:description "Load pubkey from the base32 encoded string"
:argname "<base32>",
verify:option "-k --keypair"
:description "Get pubkey from the keypair file"
:argname "<file>"
)
verify:argument "file"
:description "File to verify"
:argname "<file>"
:args "*"
verify:flag "-n --nist"
:description "Uses nistp curves (P256)"
verify:option "-s --suffix"
:description "Suffix for signature"
:argname "<suffix>"
:default("sig")
-- Encrypt subcommand
local encrypt = parser:command "encrypt crypt enc e"
:description "Encrypts a file using keypair (or a pubkey)"
encrypt:mutex(
encrypt:option "-p --pubkey"
:description "Load pubkey from the specified file"
:argname "<file>",
encrypt:option "-P --pubstring"
:description "Load pubkey from the base32 encoded string"
:argname "<base32>",
encrypt:option "-k --keypair"
:description "Get pubkey from the keypair file"
:argname "<file>"
)
encrypt:option "-s --suffix"
:description "Suffix for encrypted file"
:argname "<suffix>"
:default("enc")
encrypt:argument "file"
:description "File to encrypt"
:argname "<file>"
:args "*"
encrypt:flag "-r --rm"
:description "Remove unencrypted file"
encrypt:flag "-f --force"
:description "Remove destination file if it exists"
-- Decrypt subcommand
local decrypt = parser:command "decrypt dec d"
:description "Decrypts a file using keypair"
decrypt:option "-k --keypair"
:description "Get pubkey from the keypair file"
:argname "<file>"
decrypt:flag "-S --keep-suffix"
:description "Preserve suffix for decrypted file (overwrite encrypted)"
decrypt:argument "file"
:description "File to encrypt"
:argname "<file>"
:args "*"
decrypt:flag "-f --force"
:description "Remove destination file if it exists (implied with -S)"
decrypt:flag "-r --rm"
:description "Remove encrypted file"
-- Default command is generate, so duplicate options to be compatible
parser:flag "-s --sign"
:description "Generates a sign keypair instead of the encryption one"
parser:flag "-n --nist"
:description "Uses nistp curves (P256)"
parser:mutex(
parser:flag "-j --json"
:description "Output JSON instead of UCL",
parser:flag "-u --ucl"
:description "Output UCL"
:default(true)
)
parser:option "-o --output"
:description "Write keypair to file"
:argname "<file>"
local function fatal(...)
logger.errx(...)
os.exit(1)
end
local function ask_yes_no(greet, default)
local def_str
if default then
greet = greet .. "[Y/n]: "
def_str = "yes"
else
greet = greet .. "[y/N]: "
def_str = "no"
end
local reply = rspamd_util.readline(greet)
if not reply then os.exit(0) end
if #reply == 0 then reply = def_str end
reply = reply:lower()
if reply == 'y' or reply == 'yes' then return true end
return false
end
local function generate_handler(opts)
local mode = 'encryption'
if opts.sign then
mode = 'sign'
end
local alg = 'curve25519'
if opts.nist then
alg = 'nist'
end
-- TODO: probably, do it in a more safe way
local kp = rspamd_keypair.create(mode, alg):totable()
local format = 'ucl'
if opts.json then
format = 'json'
end
if opts.output then
local out = io.open(opts.output, 'w')
if not out then
fatal('cannot open output to write: ' .. opts.output)
end
out:write(ucl.to_format(kp, format))
out:close()
else
io.write(ucl.to_format(kp, format))
end
end
local function sign_handler(opts)
if opts.file then
if type(opts.file) == 'string' then
opts.file = {opts.file}
end
else
parser:error('no files to sign')
end
if not opts.keypair then
parser:error("no keypair specified")
end
local ucl_parser = ucl.parser()
local res,err = ucl_parser:parse_file(opts.keypair)
if not res then
fatal(string.format('cannot load %s: %s', opts.keypair, err))
end
local kp = rspamd_keypair.load(ucl_parser:get_object())
if not kp then
fatal("cannot load keypair: " .. opts.keypair)
end
for _,fname in ipairs(opts.file) do
local sig = rspamd_crypto.sign_file(kp, fname)
if not sig then
fatal(string.format("cannot sign %s\n", fname))
end
local out = string.format('%s.%s', fname, opts.suffix or 'sig')
local of = io.open(out, 'w')
if not of then
fatal('cannot open output to write: ' .. out)
end
of:write(sig:bin())
of:close()
io.write(string.format('signed %s -> %s (%s)\n', fname, out, sig:hex()))
end
end
local function verify_handler(opts)
if opts.file then
if type(opts.file) == 'string' then
opts.file = {opts.file}
end
else
parser:error('no files to verify')
end
local pk
local alg = 'curve25519'
if opts.keypair then
local ucl_parser = ucl.parser()
local res,err = ucl_parser:parse_file(opts.keypair)
if not res then
fatal(string.format('cannot load %s: %s', opts.keypair, err))
end
local kp = rspamd_keypair.load(ucl_parser:get_object())
if not kp then
fatal("cannot load keypair: " .. opts.keypair)
end
pk = kp:pk()
alg = kp:alg()
elseif opts.pubkey then
if opts.nist then alg = 'nist' end
pk = rspamd_pubkey.load(opts.pubkey, 'sign', alg)
elseif opts.pubstr then
if opts.nist then alg = 'nist' end
pk = rspamd_pubkey.create(opts.pubstr, 'sign', alg)
end
if not pk then
fatal("cannot create pubkey")
end
local valid = true
for _,fname in ipairs(opts.file) do
local sig_fname = string.format('%s.%s', fname, opts.suffix or 'sig')
local sig = rspamd_signature.load(sig_fname, alg)
if not sig then
fatal(string.format("cannot load signature for %s -> %s",
fname, sig_fname))
end
if rspamd_crypto.verify_file(pk, sig, fname, alg) then
io.write(string.format('verified %s -> %s (%s)\n', fname, sig_fname, sig:hex()))
else
valid = false
io.write(string.format('FAILED to verify %s -> %s (%s)\n', fname,
sig_fname, sig:hex()))
end
end
if not valid then
os.exit(1)
end
end
local function encrypt_handler(opts)
if opts.file then
if type(opts.file) == 'string' then
opts.file = {opts.file}
end
else
parser:error('no files to sign')
end
local pk
local alg = 'curve25519'
if opts.keypair then
local ucl_parser = ucl.parser()
local res,err = ucl_parser:parse_file(opts.keypair)
if not res then
fatal(string.format('cannot load %s: %s', opts.keypair, err))
end
local kp = rspamd_keypair.load(ucl_parser:get_object())
if not kp then
fatal("cannot load keypair: " .. opts.keypair)
end
pk = kp:pk()
alg = kp:alg()
elseif opts.pubkey then
if opts.nist then alg = 'nist' end
pk = rspamd_pubkey.load(opts.pubkey, 'sign', alg)
elseif opts.pubstr then
if opts.nist then alg = 'nist' end
pk = rspamd_pubkey.create(opts.pubstr, 'sign', alg)
end
if not pk then
fatal("cannot load keypair: " .. opts.keypair)
end
for _,fname in ipairs(opts.file) do
local enc = rspamd_crypto.encrypt_file(pk, fname, alg)
if not enc then
fatal(string.format("cannot encrypt %s\n", fname))
end
local out
if opts.suffix and #opts.suffix > 0 then
out = string.format('%s.%s', fname, opts.suffix)
else
out = string.format('%s', fname)
end
if rspamd_util.file_exists(out) then
if opts.force or ask_yes_no(string.format('File %s already exists, overwrite?',
out), true) then
os.remove(out)
else
os.exit(1)
end
end
enc:save_in_file(out)
if opts.rm then
os.remove(fname)
io.write(string.format('encrypted %s (deleted) -> %s\n', fname, out))
else
io.write(string.format('encrypted %s -> %s\n', fname, out))
end
end
end
local function decrypt_handler(opts)
if opts.file then
if type(opts.file) == 'string' then
opts.file = {opts.file}
end
else
parser:error('no files to decrypt')
end
if not opts.keypair then
parser:error("no keypair specified")
end
local ucl_parser = ucl.parser()
local res,err = ucl_parser:parse_file(opts.keypair)
if not res then
fatal(string.format('cannot load %s: %s', opts.keypair, err))
end
local kp = rspamd_keypair.load(ucl_parser:get_object())
if not kp then
fatal("cannot load keypair: " .. opts.keypair)
end
for _,fname in ipairs(opts.file) do
local decrypted = rspamd_crypto.decrypt_file(kp, fname)
if not decrypted then
fatal(string.format("cannot decrypt %s\n", fname))
end
local out
if not opts['keep-suffix'] then
-- Strip the last suffix
out = fname:match("^(.+)%..+$")
else
out = fname
end
local removed = false
if rspamd_util.file_exists(out) then
if (opts.force or opts['keep-suffix'])
or ask_yes_no(string.format('File %s already exists, overwrite?', out), true) then
os.remove(out)
removed = true
else
os.exit(1)
end
end
if opts.rm then
os.remove(fname)
removed = true
end
if removed then
io.write(string.format('decrypted %s (removed) -> %s\n', fname, out))
else
io.write(string.format('decrypted %s -> %s\n', fname, out))
end
end
end
local function handler(args)
local opts = parser:parse(args)
local command = opts.command or "generate"
if command == 'generate' then
generate_handler(opts)
elseif command == 'sign' then
sign_handler(opts)
elseif command == 'verify' then
verify_handler(opts)
elseif command == 'encrypt' then
encrypt_handler(opts)
elseif command == 'decrypt' then
decrypt_handler(opts)
else
parser:error('command %s is not implemented', command)
end
end
return {
name = 'keypair',
aliases = {'kp', 'key'},
handler = handler,
description = parser._description
}