#!/usr/bin/ruby
# author: nishiki
# mail: nishiki@yaegashi.fr
# info: a simple script who m your passwords

require 'rubygems'
require 'highline/import'
require 'pathname'
require 'readline'
require 'i18n'
require 'colorize'

require "#{APP_ROOT}/lib/MPW"

class Cli

	# Constructor
	# @args: lang -> the operating system language
	#        config_file -> a specify config file
	def initialize(config)
		@config = config
	end

	# Sync the data with the server
	# @rtnr: true if the synchro is finish
	def sync
		if not defined?(@sync)
			case @config.sync_type
			when 'mpw'
				require "#{APP_ROOT}/lib/Sync/MPWSync"
				@sync = MPW::Sync::MPWSync.new
			when 'sftp', 'scp', 'ssh'
				require "#{APP_ROOT}/lib/Sync/SSH"
				@sync = MPW::Sync::SSH.new
			when 'ftp'
				require "#{APP_ROOT}/lib/Sync/FTP"
				@sync = MPW::Sync::FTP.new
			else
				return false
			end
		end
		
		if  not @config.sync_host.nil? and not @config.sync_port.nil?
			if not @sync.connect(@config.sync_host, @config.sync_user, @config.sync_pwd, @config.sync_path, @config.sync_port)
				puts "#{I18n.t('display.error')} #1: #{@sync.error_msg}".red
			end
		end

		if @sync.enable
			if not @mpw.sync(@sync.get(@passwd), @config.last_update)
				puts "#{I18n.t('display.error')} #2: #{@mpw.error_msg}".red  if not @mpw.error_msg.nil?
				puts "#{I18n.t('display.error')} #3: #{@sync.error_msg}".red if not @sync.error_msg.nil?
			elsif not @sync.update(File.open(@config.file_gpg).read)
				puts "#{I18n.t('display.error')} #4: #{@sync.error_msg}".red
			elsif not @config.set_last_update
				puts "#{I18n.t('display.error')} #5: #{@config.error_msg}".red
			elsif not @mpw.encrypt
				puts "#{I18n.t('display.error')} #6: #{@mpw.error_msg}".red
			else
				return true
			end
		end
	rescue Exception => e
		puts "#{I18n.t('display.error')} #7: #{e}".red
		puts @sync.error_msg   if not @sync.error_msg.nil?
		puts @config.error_msg if not @config.error_msg.nil?
		puts @mpw.error_msg    if not @mpw.error_msg.nil?
	else
		return false
	end

	# Create a new config file
	# @args: lang -> the software language
	def setup(lang)
		puts I18n.t('form.setup.title')
		puts '--------------------'
		language    = ask(I18n.t('form.setup.lang', lang: lang)).to_s
		key         = ask(I18n.t('form.setup.gpg_key')).to_s
		share_keys  = ask(I18n.t('form.setup.share_gpg_keys')).to_s
		file_gpg    = ask(I18n.t('form.setup.gpg_file', home: @conf.dir_config)).to_s
		sync_type   = ask(I18n.t('form.setup.sync_type')).to_s

		if ['ssh', 'ftp', 'mpw'].include?(sync_type)
			sync_host   = ask(I18n.t('form.setup.sync_host')).to_s
			sync_port   = ask(I18n.t('form.setup.sync_port')).to_s
			sync_user   = ask(I18n.t('form.setup.sync_user')).to_s
			sync_pwd    = ask(I18n.t('form.setup.sync_pwd')).to_s
			sync_path   = ask(I18n.t('form.setup.sync_path')).to_s
		end
		
		if language.nil? or language.empty?
			language = lang
		end
		I18n.locale = language.to_sym

		sync_type = sync_type.nil? or sync_type.empty? ? nil : sync_type
		sync_host = sync_host.nil? or sync_host.empty? ? nil : sync_host
		sync_port = sync_port.nil? or sync_port.empty? ? nil : sync_port.to_i
		sync_user = sync_user.nil? or sync_user.empty? ? nil : sync_user
		sync_pwd  = sync_pwd.nil?  or sync_pwd.empty?  ? nil : sync_pwd
		sync_path = sync_path.nil? or sync_path.empty? ? nil : sync_path

		if @config.setup(key, share_keys, language, file_gpg, sync_type, sync_host, sync_port, sync_user, sync_pwd, sync_path)
			puts "#{I18n.t('form.setup.valid')}".green
		else
			puts "#{I18n.t('display.error')} #8: #{@config.error_msg}".red
			exit 2
		end

		if not @config.checkconfig
			puts "#{I18n.t('display.error')} #9: #{@config.error_msg}".red
			exit 2
		end
	end

	# Setup a new GPG key
	def setup_gpg_key
		puts I18n.t('form.setup_gpg_key.title')
		puts '--------------------'
		ask      = ask(I18n.t('form.setup_gpg_key.ask')).to_s
		
		if not ['Y', 'y', 'O', 'o'].include?(ask)
			puts I18n.t('form.setup_gpg_key.no_create')
			exit 2
		end

		name     = ask(I18n.t('form.setup_gpg_key.name')).to_s
		password = ask(I18n.t('form.setup_gpg_key.password')) {|q| q.echo = false}
		confirm  = ask(I18n.t('form.setup_gpg_key.confirm_password')) {|q| q.echo = false}

		if password != confirm 
			puts I18n.t('form.setup_gpg_key.error_password')
			exit 2
		end

		length   = ask(I18n.t('form.setup_gpg_key.length')).to_s
		expire   = ask(I18n.t('form.setup_gpg_key.expire')).to_s
		password = password.to_s

		length = length.nil? or length.empty? ? 2048 : length.to_i
		expire = expire.nil? or expire.empty? ? 0    : expire.to_i

		puts I18n.t('form.setup_gpg_key.wait')
		
		if @config.setup_gpg_key(password, name, length, expire)
			puts "#{I18n.t('form.setup_gpg_key.valid')}".green
		else
			puts "#{I18n.t('display.error')} #10: #{@config.error_msg}".red
			exit 2
		end
	end

	# Request the GPG password and decrypt the file
	def decrypt
		if not defined?(@mpw)
			@mpw = MPW::MPW.new(@config.file_gpg, @config.key, @config.share_keys)
		end

		@passwd = ask(I18n.t('display.gpg_password')) {|q| q.echo = false}
		if not @mpw.decrypt(@passwd)
			puts "#{I18n.t('display.error')} #11: #{@mpw.error_msg}".red
			exit 2
		end
	end

	# Display the query's result
	# @args: search -> the string to search
	#        protocol -> search from a particular protocol
	def display(search, protocol=nil, group=nil, format=nil)
		result = @mpw.search(search, group, protocol)

		if not result.empty?
			result.each do |r|
				if format.nil? || !format
					displayFormat(r)
				else
					displayFormatAlt(r)
				end
			end
		else
			puts I18n.t('display.nothing')
		end
	end

	# Display an item in the default format
	# @args: item -> an array with the item information
	def displayFormat(item)
		puts '--------------------'
		puts "Id: #{item['id']}"
		puts "#{I18n.t('display.name')}: #{item['name']}"
		puts "#{I18n.t('display.group')}: #{item['group']}"
		puts "#{I18n.t('display.server')}: #{item['host']}"
		puts "#{I18n.t('display.protocol')}: #{item['protocol']}"
		puts "#{I18n.t('display.login')}: #{item['login']}"
		puts "#{I18n.t('display.password')}: #{item['password']}"
		puts "#{I18n.t('display.port')}: #{item['port']}"
		puts "#{I18n.t('display.comment')}: #{item['comment']}"
	end

	# Display an item in the alternative format
	# @args: item -> an array with the item information
	def displayFormatAlt(item)
		port = item[:port].nil? ? '' : ":#{item[:port]}"

		if item[:password].nil? or item[:password].empty?
			if item[:login].include('@')
				puts "# #{item[:id]} #{item[:protocol]}://#{item[:login]}@#{item[:host]}#{port}"
			else
				puts "# #{item[:id]} #{item[:protocol]}://{#{item[:login]}}@#{item[:host]}#{port}"
			end
		else
			puts "# #{item[:id]} #{item[:protocol]}://{#{item[:login]}:#{item[:password]}}@#{item[:host]}#{port}"
		end
	end

	# Form to add a new item
	def add
		row = []
		puts I18n.t('form.add.title')
		puts '--------------------'
		name     = ask(I18n.t('form.add.name')).to_s
		group    = ask(I18n.t('form.add.group')).to_s
		server   = ask(I18n.t('form.add.server')).to_s
		protocol = ask(I18n.t('form.add.protocol')).to_s
		login    = ask(I18n.t('form.add.login')).to_s
		passwd   = ask(I18n.t('form.add.password')).to_s
		port     = ask(I18n.t('form.add.port')).to_s
		comment  = ask(I18n.t('form.add.comment')).to_s

		if @mpw.update(name, group, server, protocol, login, passwd, port, comment)
			if @mpw.encrypt
				sync
				puts "#{I18n.t('form.add.valid')}".green
			else
				puts "#{I18n.t('display.error')} #12: #{@mpw.error_msg}".red
			end
		else
			puts "#{I18n.t('display.error')} #13: #{@mpw.error_msg}".red
		end
	end

	# Update an item
	# @args: id -> the item's id
	def update(id)
		row = @mpw.search_by_id(id)

		if not row.empty?
			puts I18n.t('form.update.title')
			puts '--------------------'
			name     = ask(I18n.t('form.update.name'    , name:     row[:name])).to_s
			group    = ask(I18n.t('form.update.group'   , group:    row[:group])).to_s
			server   = ask(I18n.t('form.update.server'  , server:   row[:host])).to_s
			protocol = ask(I18n.t('form.update.protocol', protocol: row[:protocol])).to_s
			login    = ask(I18n.t('form.update.login'   , login:    row[:login])).to_s
			passwd   = ask(I18n.t('form.update.password')).to_s
			port     = ask(I18n.t('form.update.port'    , port:     row[:port])).to_s
			comment  = ask(I18n.t('form.update.comment' , comment:  row[:comment])).to_s
				
			if @mpw.update(name, group, server, protocol, login, passwd, port, comment, id)
				if @mpw.encrypt
					sync
					puts "#{I18n.t('form.update.valid')}".green
				else
					puts "#{I18n.t('display.error')} #14: #{@mpw.error_msg}".red
				end
			else
				puts "#{I18n.t('display.error')} #15: #{@mpw.error_msg}".red
			end
		else
			puts I18n.t('display.nothing')
		end
	end

	# Remove an item
	# @args: id -> the item's id
	#        force -> no resquest a validation
	def remove(id, force=false)
		if not force
			result = @mpw.search_by_id(id)

			if result.length > 0
				displayFormat(result)

				confirm = ask("#{I18n.t('form.delete.ask', id: id)} (y/N) ").to_s
				if confirm =~ /^(y|yes|YES|Yes|Y)$/
					force = true
				end
			else
				puts I18n.t('display.nothing')
			end
		end

		if force
			if @mpw.remove(id)
				if @mpw.encrypt
					sync
					puts "#{I18n.t('form.delete.valid', id: id)}".green
				else
					puts "#{I18n.t('display.error')} #16: #{@mpw.error_msg}".red
				end
			else
				puts I18n.t('form.delete.not_valid')
			end
		end
	end

	# Export the items in a CSV file
	# @args: file -> the destination file
	def export(file, type=:yaml)
		if @mpw.export(file, type)
			puts "#{I18n.t('export.valid', file)}".green
		else
			puts "#{I18n.t('display.error')} #17: #{@mpw.error_msg}".red
		end

	end

	# Import items from a CSV file
	# @args: file -> the import file
	#        force -> no resquest a validation
	def import(file, type=:yaml, force=false)

		if not force
			result = @mpw.import_preview(file, type)
			if result.is_a?(Array) and not result.empty?
				result.each do |r|
					displayFormat(r)
				end

				confirm = ask("#{I18n.t('form.import.ask', file: file)} (y/N) ").to_s
				if confirm =~ /^(y|yes|YES|Yes|Y)$/
					force = true
				end
			else
				puts I18n.t('form.import.not_valid')
			end
		end

		if force
			if @mpw.import(file, type) and @mpw.encrypt
				sync
				puts "#{I18n.t('form.import.valid')}".green
			else
				puts "#{I18n.t('display.error')} #18: #{@mpw.error_msg}".red
			end
		end
	end

end