1
0
Fork 0
mirror of https://github.com/nishiki/manage-password.git synced 2024-11-23 22:03:05 +00:00
mpw/lib/Server.rb

325 lines
8.4 KiB
Ruby
Raw Normal View History

2014-01-10 16:55:05 +00:00
#!/usr/bin/ruby
require 'socket'
require 'json'
require 'highline/import'
require 'digest'
2014-01-25 10:35:12 +00:00
require 'logger'
2014-01-10 16:55:05 +00:00
class Server
2014-01-25 10:35:12 +00:00
2014-01-10 16:55:05 +00:00
attr_accessor :error_msg
# Constructor
2014-02-02 09:16:10 +00:00
def initialize
2014-01-10 16:55:05 +00:00
YAML::ENGINE.yamler='syck'
end
# Start the server
2014-02-02 09:16:10 +00:00
def start
2014-01-31 23:30:52 +00:00
server = TCPServer.open(@host, @port)
@log.info("The server is started on #{@host}:#{@port}")
2014-01-25 10:35:12 +00:00
2014-01-10 16:55:05 +00:00
loop do
Thread.start(server.accept) do |client|
2014-01-25 10:35:12 +00:00
@log.info("#{client.peeraddr[3]} is connected")
2014-01-14 22:00:52 +00:00
while true do
2014-01-26 14:09:48 +00:00
msg = get_client_msg(client)
2014-01-14 22:00:52 +00:00
if !msg
next
end
if msg['gpg_key'].nil? || msg['gpg_key'].empty? || msg['password'].nil? || msg['password'].empty?
2014-01-26 16:26:02 +00:00
@log.warning("#{client.peeraddr[3]} is disconnected because no password or no gpg_key")
2014-01-26 14:09:48 +00:00
close_connection(client)
2014-01-14 22:00:52 +00:00
next
end
case msg['action']
when 'get'
2014-01-25 10:35:12 +00:00
@log.debug("#{client.peeraddr[3]} GET gpg_key=#{msg['gpg_key']} suffix=#{msg['suffix']}")
2014-01-26 14:09:48 +00:00
client.puts get_file(msg)
2014-01-14 22:00:52 +00:00
when 'update'
2014-01-25 10:35:12 +00:00
@log.debug("#{client.peeraddr[3]} UPDATE gpg_key=#{msg['gpg_key']} suffix=#{msg['suffix']}")
2014-01-26 14:09:48 +00:00
client.puts update_file(msg)
2014-01-14 22:00:52 +00:00
when 'delete'
2014-01-25 10:35:12 +00:00
@log.debug("#{client.peeraddr[3]} DELETE gpg_key=#{msg['gpg_key']} suffix=#{msg['suffix']}")
2014-01-26 14:09:48 +00:00
client.puts delete_file(msg)
2014-01-14 22:00:52 +00:00
else
2014-01-26 16:26:02 +00:00
@log.warning("#{client.peeraddr[3]} is disconnected because unkwnow command")
2014-01-16 20:24:08 +00:00
send_msg = {:action => 'unknown',
:gpg_key => msg['gpg_key'],
:error => 'server.error.client.unknown'}
client.puts send_msg
2014-01-26 14:09:48 +00:00
close_connection(client)
2014-01-14 22:00:52 +00:00
end
2014-01-10 16:55:05 +00:00
end
end
end
2014-01-31 23:30:52 +00:00
rescue Exception => e
2014-02-05 21:47:20 +00:00
puts "Impossible to start the server: #{e}"
2014-01-31 23:30:52 +00:00
@log.error("Impossible to start the server: #{e}")
exit 2
2014-01-10 16:55:05 +00:00
end
# Get a gpg file
# @args: msg -> message puts by the client
# @rtrn: json message
2014-01-26 14:09:48 +00:00
def get_file(msg)
2014-01-10 16:55:05 +00:00
gpg_key = msg['gpg_key'].sub('@', '_')
if msg['suffix'].nil? || msg['suffix'].empty?
file_gpg = "#{@data_dir}/#{gpg_key}.yml"
else
file_gpg = "#{@data_dir}/#{gpg_key}-#{msg['suffix']}.yml"
end
if File.exist?(file_gpg)
2014-01-12 10:07:54 +00:00
gpg_data = YAML::load_file(file_gpg)
salt = gpg_data['gpg']['salt']
hash = gpg_data['gpg']['hash']
data = gpg_data['gpg']['data']
2014-01-10 16:55:05 +00:00
2014-01-26 14:09:48 +00:00
if is_authorized?(msg['password'], salt, hash)
2014-01-14 22:00:52 +00:00
send_msg = {:action => 'get',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:data => data,
:error => nil}
2014-01-10 16:55:05 +00:00
else
send_msg = {:action => 'get',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.no_authorized'}
2014-01-10 16:55:05 +00:00
end
else
2014-01-16 20:24:08 +00:00
send_msg = {:action => 'get',
:gpg_key => msg['gpg_key'],
:data => '',
:error => nil}
2014-01-10 16:55:05 +00:00
end
return send_msg.to_json
end
# Update a file
# @args: msg -> message puts by the client
# @rtrn: json message
2014-01-26 14:09:48 +00:00
def update_file(msg)
2014-01-10 16:55:05 +00:00
gpg_key = msg['gpg_key'].sub('@', '_')
data = msg['data']
if data.nil? || data.empty?
send_msg = {:action => 'update',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.no_data'}
2014-01-10 16:55:05 +00:00
return send_msg.to_json
end
if msg['suffix'].nil? || msg['suffix'].empty?
file_gpg = "#{@data_dir}/#{gpg_key}.yml"
else
file_gpg = "#{@data_dir}/#{gpg_key}-#{msg['suffix']}.yml"
end
if File.exist?(file_gpg)
gpg_data = YAML::load_file(file_gpg)
salt = gpg_data['gpg']['salt']
hash = gpg_data['gpg']['hash']
else
2014-04-27 13:13:43 +00:00
salt = generate_salt
2014-01-10 16:55:05 +00:00
hash = Digest::SHA256.hexdigest(salt + msg['password'])
end
2014-01-26 14:09:48 +00:00
if is_authorized?(msg['password'], salt, hash)
2014-01-10 16:55:05 +00:00
begin
2014-01-12 10:07:54 +00:00
config = {'gpg' => {'salt' => salt,
'hash' => hash,
'data' => data}}
2014-01-10 16:55:05 +00:00
2014-01-14 22:00:52 +00:00
File.open(file_gpg, 'w+') do |file|
2014-01-10 16:55:05 +00:00
file << config.to_yaml
end
2014-01-16 20:24:08 +00:00
send_msg = {:action => 'update',
:gpg_key => msg['gpg_key'],
:error => nil}
2014-01-10 16:55:05 +00:00
rescue Exception => e
send_msg = {:action => 'update',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.unknown'}
2014-01-10 16:55:05 +00:00
end
else
send_msg = {:action => 'update',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.no_authorized'}
2014-01-10 16:55:05 +00:00
end
return send_msg.to_json
end
# Remove a gpg file
# @args: msg -> message puts by the client
# @rtrn: json message
2014-01-26 14:09:48 +00:00
def delete_file(msg)
2014-01-10 17:12:24 +00:00
gpg_key = msg['gpg_key'].sub('@', '_')
if msg['suffix'].nil? || msg['suffix'].empty?
file_gpg = "#{@data_dir}/#{gpg_key}.yml"
else
file_gpg = "#{@data_dir}/#{gpg_key}-#{msg['suffix']}.yml"
end
if !File.exist?(file_gpg)
send_msg = {:action => 'delete',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => nil}
2014-01-10 17:12:24 +00:00
return send_msg.to_json
end
gpg_data = YAML::load_file(file_gpg)
salt = gpg_data['gpg']['salt']
hash = gpg_data['gpg']['hash']
2014-01-26 14:09:48 +00:00
if is_authorized?(msg['password'], salt, hash)
2014-01-10 17:12:24 +00:00
begin
File.unlink(file_gpg)
send_msg = {:action => 'delete',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => nil}
2014-01-10 17:12:24 +00:00
rescue Exception => e
send_msg = {:action => 'delete',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.unknown'}
2014-01-10 17:12:24 +00:00
end
else
send_msg = {:action => 'delete',
:gpg_key => msg['gpg_key'],
2014-01-16 20:24:08 +00:00
:error => 'server.error.client.no_authorized'}
2014-01-10 17:12:24 +00:00
end
return send_msg.to_json
2014-01-10 16:55:05 +00:00
end
# Check is the hash equal the password with the salt
# @args: password -> the user password
# salt -> the salt
# hash -> the hash of the password with the salt
# @rtrn: true is is good, else false
2014-01-26 14:09:48 +00:00
def is_authorized?(password, salt, hash)
2014-01-10 16:55:05 +00:00
if hash == Digest::SHA256.hexdigest(salt + password)
return true
else
return false
end
end
# Get message to client
# @args: client -> client connection
# @rtrn: array of the json string, or false if isn't json message
2014-01-26 14:09:48 +00:00
def get_client_msg(client)
2014-01-31 23:30:52 +00:00
msg = client.gets
return JSON.parse(msg)
rescue
closeConnection(client)
return false
2014-01-10 16:55:05 +00:00
end
# Close the client connection
# @args: client -> client connection
2014-01-26 14:09:48 +00:00
def close_connection(client)
2014-01-10 16:55:05 +00:00
client.puts "Closing the connection. Bye!"
client.close
end
# Check the config file
# @args: file_config -> the configuration file
# @rtrn: true if the config file is correct
def checkconfig(file_config)
2014-01-31 23:30:52 +00:00
config = YAML::load_file(file_config)
@host = config['config']['host']
@port = config['config']['port'].to_i
@data_dir = config['config']['data_dir']
@log_file = config['config']['log_file']
@timeout = config['config']['timeout'].to_i
if @host.empty? || @port <= 0 || @data_dir.empty?
puts I18n.t('checkconfig.fail')
puts I18n.t('checkconfig.empty')
return false
end
2014-01-10 21:49:22 +00:00
2014-01-31 23:30:52 +00:00
if !Dir.exist?(@data_dir)
puts I18n.t('checkconfig.fail')
puts I18n.t('checkconfig.datadir')
return false
end
2014-01-10 16:55:05 +00:00
2014-01-31 23:30:52 +00:00
if @log_file.nil? || @log_file.empty?
puts I18n.t('checkconfig.fail')
puts I18n.t('checkconfig.log_file_empty')
return false
else
begin
@log = Logger.new(@log_file)
rescue
2014-01-25 17:40:47 +00:00
puts I18n.t('checkconfig.fail')
2014-01-31 23:30:52 +00:00
puts I18n.t('checkconfig.log_file_create')
2014-01-25 10:35:12 +00:00
return false
end
2014-01-10 16:55:05 +00:00
end
return true
2014-01-31 23:30:52 +00:00
rescue Exception => e
puts "#{I18n.t('checkconfig.fail')}\n#{e}"
return false
2014-01-10 16:55:05 +00:00
end
# Create a new config file
# @args: file_config -> the configuration file
# @rtrn: true if le config file is create
def setup(file_config)
2014-01-25 17:40:47 +00:00
puts I18n.t('form.setup.title')
2014-01-10 16:55:05 +00:00
puts '--------------------'
2014-01-25 17:40:47 +00:00
host = ask(I18n.t('form.setup.host')).to_s
port = ask(I18n.t('form.setup.port')).to_s
data_dir = ask(I18n.t('form.setup.data_dir')).to_s
log_file = ask(I18n.t('form.setup.log_file')).to_s
timeout = ask(I18n.t('form.setup.timeout')).to_s
2014-01-10 16:55:05 +00:00
config = {'config' => {'host' => host,
'port' => port,
'data_dir' => data_dir,
2014-01-25 10:35:12 +00:00
'log_file' => log_file,
2014-01-10 16:55:05 +00:00
'timeout' => timeout}}
2014-01-31 23:30:52 +00:00
File.open(file_config, 'w') do |file|
file << config.to_yaml
2014-01-10 16:55:05 +00:00
end
2014-01-31 23:30:52 +00:00
2014-01-10 16:55:05 +00:00
return true
2014-01-31 23:30:52 +00:00
rescue Exception => e
puts "#{I18n.t('form.setup.not_valid')}\n#{e}"
return false
2014-01-10 16:55:05 +00:00
end
2014-01-27 17:27:37 +00:00
# Generate a random salt
# @args: length -> the length salt
# @rtrn: a random string
2014-04-27 13:13:43 +00:00
def generate_salt(length=4)
2014-01-27 17:27:37 +00:00
if length.to_i <= 0 || length.to_i > 16
length = 4
else
length = length.to_i
end
return ([*('A'..'Z'),*('a'..'z'),*('0'..'9')]).sample(length).join
end
2014-01-10 16:55:05 +00:00
end