/*
 * Copyright (c) Contributors, http://opensimulator.org/
 * See CONTRIBUTORS.TXT for a full list of copyright holders.
 *
 * 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.
 *     * Neither the name of the OpenSimulator Project nor the
 *       names of its contributors may be used to endorse or promote products
 *       derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE DEVELOPERS ``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 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.Xml;
using System.Collections.Generic;
using System.Reflection;
using System.Threading;
using System.Timers;
using Timer = System.Timers.Timer;
using OpenMetaverse;
using OpenMetaverse.StructuredData;
using log4net;
using Nini.Config;
using OpenSim.Framework;
using OpenSim.Framework.Client;
using OpenSim.Framework.Monitoring;
using OpenSim.Region.Framework.Interfaces;
using OpenSim.Region.Framework.Scenes.Types;
using OpenSim.Services.Interfaces;

namespace OpenSim.Region.Framework.Scenes
{
    public class GodController
    {
        public enum ImplicitGodLevels : int
        {
            EstateManager = 210,    // estate manager implicit god level
            RegionOwner = 220       // region owner implicit god level should be >= than estate
        }

        ScenePresence m_scenePresence;
        Scene m_scene;
        protected bool m_allowGridGods;
        protected bool m_forceGridGodsOnly;
        protected bool m_regionOwnerIsGod;
        protected bool m_regionManagerIsGod;
        protected bool m_forceGodModeAlwaysOn;
        protected bool m_allowGodActionsWithoutGodMode;

        protected int m_userLevel = 0;
        // the god level from local or grid user rights
        protected int m_rightsGodLevel = 0;
        // the level seen by viewers
        protected int m_viewergodlevel = 0;
        // new level that can be fixed or equal to godlevel, acording to options
        protected int m_godlevel = 0;
        protected int m_lastLevelToViewer = 0;

        public GodController(Scene scene, ScenePresence sp, int userlevel)
        {
            m_scene = scene;
            m_scenePresence = sp;
            m_userLevel = userlevel;

            IConfigSource config = scene.Config;

            string[] sections = new string[] { "Startup", "Permissions" };

            // God level is based on UserLevel. Gods will have that
            // level grid-wide. Others may become god locally but grid
            // gods are god everywhere.
            m_allowGridGods =
                    Util.GetConfigVarFromSections<bool>(config,
                    "allow_grid_gods", sections, false);

            // If grid gods are active, dont allow any other gods
            m_forceGridGodsOnly =
                    Util.GetConfigVarFromSections<bool>(config,
                    "force_grid_gods_only", sections, false);

            if(!m_forceGridGodsOnly)
            {
                // The owner of a region is a god in his region only.
                m_regionOwnerIsGod =
                    Util.GetConfigVarFromSections<bool>(config,
                    "region_owner_is_god", sections, true);

                // Region managers are gods in the regions they manage.
                m_regionManagerIsGod =
                    Util.GetConfigVarFromSections<bool>(config,
                    "region_manager_is_god", sections, false);

            }
            else
                m_allowGridGods = true; // reduce potencial user mistakes
                 
            // God mode should be turned on in the viewer whenever
            // the user has god rights somewhere. They may choose
            // to turn it off again, though.
            m_forceGodModeAlwaysOn =
                    Util.GetConfigVarFromSections<bool>(config,
                    "automatic_gods", sections, false);

            // The user can execute any and all god functions, as
            // permitted by the viewer UI, without actually "godding
            // up". This is the default state in 0.8.2.
            m_allowGodActionsWithoutGodMode =
                    Util.GetConfigVarFromSections<bool>(config,
                    "implicit_gods", sections, false);

            m_rightsGodLevel = CalcRightsGodLevel();

            if(m_allowGodActionsWithoutGodMode)
            {
                m_godlevel = m_rightsGodLevel;
                m_forceGodModeAlwaysOn = false;
            }

            else if(m_forceGodModeAlwaysOn)
            {
                m_viewergodlevel = m_rightsGodLevel;
                m_godlevel = m_rightsGodLevel;
            }

            m_scenePresence.IsGod = (m_godlevel >= 200);
            m_scenePresence.IsViewerUIGod = (m_viewergodlevel >= 200);
        }

        // calculates god level at sp creation from local and grid user god rights
        // for now this is assumed static until user leaves region.
        // later estate and gride level updates may update this
        protected int CalcRightsGodLevel()
        {
            int level = 0;
            if (m_allowGridGods && m_userLevel >= 200)
                level = m_userLevel;

            if(m_forceGridGodsOnly || level >= (int)ImplicitGodLevels.RegionOwner)
                return level;

            if (m_regionOwnerIsGod && m_scene.RegionInfo.EstateSettings.IsEstateOwner(m_scenePresence.UUID))
                level = (int)ImplicitGodLevels.RegionOwner;

            if(level >= (int)ImplicitGodLevels.EstateManager)
                return level;

            if (m_regionManagerIsGod && m_scene.Permissions.IsEstateManager(m_scenePresence.UUID))
                level = (int)ImplicitGodLevels.EstateManager;

            return level;
        }

        protected bool CanBeGod()
        {
            return m_rightsGodLevel >= 200;
        }

        protected void UpdateGodLevels(bool viewerState)
        {
            if(!CanBeGod())
            {
                m_viewergodlevel = 0;
                m_godlevel = 0;
                m_scenePresence.IsGod = false;
                m_scenePresence.IsViewerUIGod = false;
                return;
            }

            // legacy some are controled by viewer, others are static
            if(m_allowGodActionsWithoutGodMode)
            {
                if(viewerState)
                    m_viewergodlevel = m_rightsGodLevel;
                else
                    m_viewergodlevel = 0;

                m_godlevel = m_rightsGodLevel;
            }
            else
            {
                // new all change with viewer
                if(viewerState)
                {
                    m_viewergodlevel = m_rightsGodLevel;
                    m_godlevel = m_rightsGodLevel;
                }
                else
                {
                    m_viewergodlevel = 0;
                    m_godlevel = 0;
                }
            }
            m_scenePresence.IsGod = (m_godlevel >= 200);
            m_scenePresence.IsViewerUIGod = (m_viewergodlevel >= 200);
        }

        public void SyncViewerState()
        {
            if(m_lastLevelToViewer == m_viewergodlevel)
                return;

            m_lastLevelToViewer = m_viewergodlevel;

            if(m_scenePresence.IsChildAgent)
                return;            

            m_scenePresence.ControllingClient.SendAdminResponse(UUID.Zero, (uint)m_viewergodlevel);
        }

        public void RequestGodMode(bool god)
        {
            UpdateGodLevels(god);

            if(m_lastLevelToViewer != m_viewergodlevel)
            {
                m_scenePresence.ControllingClient.SendAdminResponse(UUID.Zero, (uint)m_viewergodlevel);
                m_lastLevelToViewer = m_viewergodlevel;
            }
        }

       public OSD State()
        {
            OSDMap godMap = new OSDMap(2);
            bool m_viewerUiIsGod = m_viewergodlevel >= 200;
            godMap.Add("ViewerUiIsGod", OSD.FromBoolean(m_viewerUiIsGod));

            return godMap;
        }

        public void SetState(OSD state)
        {
            bool newstate = false;
            if(m_forceGodModeAlwaysOn)
                newstate = m_viewergodlevel >= 200;
            if(state != null)
            {
                OSDMap s = (OSDMap)state;

                if (s.ContainsKey("ViewerUiIsGod"))
                    newstate = s["ViewerUiIsGod"].AsBoolean();
                m_lastLevelToViewer = m_viewergodlevel; // we are not changing viewer level by default
            }       
            UpdateGodLevels(newstate);
        }

        public void HasMovedAway()
        {
            m_lastLevelToViewer = 0;
            if(m_forceGodModeAlwaysOn)
            {
                m_viewergodlevel = m_rightsGodLevel;
                m_godlevel = m_rightsGodLevel;
            }
        }

        public int UserLevel
        {
            get { return m_userLevel; }
            set { m_userLevel = value; }
        }

        public int ViwerUIGodLevel
        {
            get { return m_viewergodlevel; }
        }

        public int GodLevel
        {
            get { return m_godlevel; }
        }
    }
}