/* Copyright (c) Citrix Systems Inc. 
 * All rights reserved. 
 * 
 * Redistribution and use in source and binary forms, 
 * with or without modification, are permitted provided 
 * that the following conditions are met: 
 * 
 * *   Redistributions of source code must retain the above 
 *     copyright notice, this list of conditions and the 
 *     following disclaimer. 
 * *   Redistributions in binary form must reproduce the above 
 *     copyright notice, this list of conditions and the 
 *     following disclaimer in the documentation and/or other 
 *     materials provided with the distribution. 
 * 
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 
 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, 
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF 
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR 
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 
 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 
 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING 
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 
 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 
 * SUCH DAMAGE.
 */

using System;
using System.Collections.Generic;
using XenAdmin.Core;
using XenAdmin.Network;
using System.Threading.Tasks;
using XenAPI;

namespace XenServerHealthCheck
{
    public class ServerInfo
    {
        public string HostName { get; set; }
        public string UserName { get; set; }
        public string Password { get; set; }
        public System.Threading.Tasks.Task task { get; set; }
    }

    public class ServerListHelper
    {
        private ServerListHelper() { }
        public static readonly ServerListHelper instance = new ServerListHelper();
        private static readonly log4net.ILog log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
        private List<ServerInfo> serverList;

        private static readonly Object serverListLock = new Object();
        public List<ServerInfo> GetServerList()
        {
            List<ServerInfo> currentList;
            lock (serverListLock)
            {
                currentList = new List<ServerInfo>(serverList);
            }
            return currentList;
        }

        public void Init()
        {
            lock (serverListLock)
            {
                serverList = new List<ServerInfo>();
                string[] encServerList = Properties.Settings.Default.ServerList ?? new string[0];
                foreach (string encServer in encServerList)
                {
                    string decryptCredential = EncryptionUtils.Unprotect(encServer);
                    string[] decryptCredentialComps = decryptCredential.Split(SEPARATOR);
                    ServerInfo connection = new ServerInfo();
                    connection.HostName = decryptCredentialComps[0];
                    connection.UserName = decryptCredentialComps[1];
                    connection.Password = decryptCredentialComps[2];
                    connection.task = null;
                    serverList.Add(connection);
                }
            }
        }

        private const char SEPARATOR = '\x202f'; // narrow non-breaking space.
        private string ProtectCredential(ServerInfo connection)
        {
            string Host = connection.HostName ?? string.Empty;
            string username = connection.UserName ?? string.Empty;
            string passwordSecret = connection.Password ?? string.Empty;

            return EncryptionUtils.Protect(String.Join(SEPARATOR.ToString(), new[] { Host, username, passwordSecret }));
        }

        public void removeServerCredential(string HostName)
        {
            lock (serverListLock)
            {
                foreach (ServerInfo con in serverList)
                {
                    if (HostName == con.HostName)
                    {
                        serverList.Remove(con);
                        updateServerList();
                        return;
                    }
                }
            }
        }

        public bool UpdateServerCredential(ServerInfo server, string masterName)
        {
            lock (serverListLock)
            {
                bool needRefresh = true;
                foreach (ServerInfo con in serverList)
                {
                    if (masterName == con.HostName)
                    {
                        needRefresh = false;
                        break;
                    }
                }

                serverList.Remove(server);
                if (needRefresh)
                {
                    server.HostName = masterName;
                    serverList.Add(server);
                }
                
                updateServerList();
                return needRefresh;
            }
        }

        private void updateServerList()
        {
            List<string> encList = new List<string>();
            foreach (ServerInfo connection in serverList)
            {
                encList.Add(ProtectCredential(connection));
            }
            Properties.Settings.Default.ServerList = encList.ToArray();
            Properties.Settings.Default.Save();
        }

        public bool UpdateServerInfo(ServerInfo server)
        {
            lock (serverListLock)
            {
                foreach (ServerInfo con in serverList)
                {
                    if (server.HostName == con.HostName)
                    {
                        serverList.Remove(con);
                        serverList.Add(server);
                        updateServerList();
                        return true;
                    }
                }
                return false;
            }
        }

        public void UpdateServerCredential(string credential)
        {
            log.Info("Receive credential update message");

            string decryptCredential = EncryptionUtils.UnprotectForLocalMachine(credential);
            string[] decryptCredentialComps = decryptCredential.Split(SEPARATOR);

            lock (serverListLock)
            {
                if (decryptCredentialComps.Length == 3)
                {//Add credential
                    System.Threading.Tasks.Task originalTask = null;
                    foreach (ServerInfo connection in serverList)
                    {
                        if (connection.HostName == decryptCredentialComps[0])
                        {
                            originalTask = connection.task;
                            serverList.Remove(connection);
                            log.Info("Refresh credential");
                            break;
                        }
                    }
                    ServerInfo newConnection = new ServerInfo();
                    newConnection.HostName = decryptCredentialComps[0];
                    newConnection.UserName = decryptCredentialComps[1];
                    newConnection.Password = decryptCredentialComps[2];
                    newConnection.task = originalTask;
                    serverList.Add(newConnection);
                    log.InfoFormat("Add credential, current credential size is {0}", serverList.Count);
                }
                else if (decryptCredentialComps.Length == 1)
                {//remove credential
                    foreach (ServerInfo connection in serverList)
                    {
                        if (connection.HostName == decryptCredentialComps[0])
                        {
                            serverList.Remove(connection);
                            log.InfoFormat("Remove credential, current credential size is {0}", serverList.Count);
                            break;
                        }
                    }
                }

                updateServerList();
            }
        }

        public void UpdateProxy(string proxy)
        {
            log.Info("Receive proxy update message");

            try
            {
                string[] proxySettings = proxy.Split(SEPARATOR);
                HTTPHelper.ProxyStyle proxyStyle = (HTTPHelper.ProxyStyle)Int32.Parse(proxySettings[1]);

                switch (proxyStyle)
                {
                    case HTTPHelper.ProxyStyle.SpecifiedProxy:
                        Properties.Settings.Default.ProxySetting = (Int32)proxyStyle;
                        Properties.Settings.Default.ProxyAddress = proxySettings[2];
                        Properties.Settings.Default.ProxyPort = Int32.Parse(proxySettings[3]);
                        Properties.Settings.Default.ConnectionTimeout = Int32.Parse(proxySettings[4]);
                        Properties.Settings.Default.BypassProxyForLocal = bool.Parse(proxySettings[5]);
                        return;

                    case HTTPHelper.ProxyStyle.SystemProxy:
                        Properties.Settings.Default.ProxySetting = (Int32)proxyStyle;
                        return;

                    default:
                        Properties.Settings.Default.ProxySetting = (Int32)proxyStyle;
                        return;
                }

            }
            catch (Exception e)
            {
                log.Error("Error parsing 'ProxySetting' from XenCenter", e);
            }
        }
    }
}