/**
 * @file llfloaterregiondebugconsole.h
 * @author Brad Kittenbrink <brad@lindenlab.com>
 * @brief Quick and dirty console for region debug settings
 *
 * $LicenseInfo:firstyear=2010&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2010, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

#include "llviewerprecompiledheaders.h"

#include "llfloaterregiondebugconsole.h"

#include "llagent.h"
#include "llhttpnode.h"
#include "lllineeditor.h"
#include "lltexteditor.h"
#include "llviewerregion.h"
#include "llcorehttputil.h"

// Two versions of the sim console API are supported.
//
// SimConsole capability (deprecated):
// This is the initial implementation that is supported by some versions of the
// simulator. It is simple and straight forward, just POST a command and the
// body of the response has the result. This API is deprecated because it
// doesn't allow the sim to use any asynchronous API.
//
// SimConsoleAsync capability:
// This capability replaces the original SimConsole capability. It is similar
// in that the command is POSTed to the SimConsoleAsync cap, but the response
// comes in through the event poll, which gives the simulator more flexibility
// and allows it to perform complex operations without blocking any frames.
//
// We will assume the SimConsoleAsync capability is available, and fall back to
// the SimConsole cap if it is not. The simulator will only support one or the
// other.

namespace
{
    // Signal used to notify the floater of responses from the asynchronous
    // API.
    console_reply_signal_t sConsoleReplySignal;

    const std::string PROMPT("\n\n> ");
    const std::string UNABLE_TO_SEND_COMMAND(
        "ERROR: The last command was not received by the server.");
    const std::string CONSOLE_UNAVAILABLE(
        "ERROR: No console available for this region/simulator.");
    const std::string CONSOLE_NOT_SUPPORTED(
        "This region does not support the simulator console.");

    // This handles responses for console commands sent via the asynchronous
    // API.
    class ConsoleResponseNode : public LLHTTPNode
    {
    public:
        /* virtual */
        void post(
            LLHTTPNode::ResponsePtr reponse,
            const LLSD& context,
            const LLSD& input) const
        {
            LL_INFOS() << "Received response from the debug console: "
                << input << LL_ENDL;
            sConsoleReplySignal(input["body"].asString());
        }
    };
}

boost::signals2::connection LLFloaterRegionDebugConsole::setConsoleReplyCallback(const console_reply_signal_t::slot_type& cb)
{
    return sConsoleReplySignal.connect(cb);
}

LLFloaterRegionDebugConsole::LLFloaterRegionDebugConsole(LLSD const & key)
: LLFloater(key), mOutput(NULL)
{
    mReplySignalConnection = sConsoleReplySignal.connect(
        boost::bind(
            &LLFloaterRegionDebugConsole::onReplyReceived,
            this,
            _1));
}

LLFloaterRegionDebugConsole::~LLFloaterRegionDebugConsole()
{
    mReplySignalConnection.disconnect();
}

bool LLFloaterRegionDebugConsole::postBuild()
{
    LLLineEditor* input = getChild<LLLineEditor>("region_debug_console_input");
    input->setEnableLineHistory(true);
    input->setCommitCallback(boost::bind(&LLFloaterRegionDebugConsole::onInput, this, _1, _2));
    input->setFocus(true);
    input->setCommitOnFocusLost(false);

    mOutput = getChild<LLTextEditor>("region_debug_console_output");

    std::string url = gAgent.getRegionCapability("SimConsoleAsync");
    if (url.empty())
    {
        // Fall back to see if the old API is supported.
        url = gAgent.getRegionCapability("SimConsole");
        if (url.empty())
        {
            mOutput->appendText(
                CONSOLE_NOT_SUPPORTED + PROMPT,
                false);
            return true;
        }
    }

    mOutput->appendText("> ", false);
    return true;
}

void LLFloaterRegionDebugConsole::onInput(LLUICtrl* ctrl, const LLSD& param)
{
    LLLineEditor* input = static_cast<LLLineEditor*>(ctrl);
    std::string text = input->getText() + "\n";

    std::string url = gAgent.getRegionCapability("SimConsoleAsync");
    if (url.empty())
    {
        // Fall back to the old API
        url = gAgent.getRegionCapability("SimConsole");
        if (url.empty())
        {
            text += CONSOLE_UNAVAILABLE + PROMPT;
        }
        else
        {
            LLSD postData = LLSD(input->getText());
            LLCoreHttpUtil::HttpCoroutineAdapter::callbackHttpPost(url, postData,
                boost::bind(&LLFloaterRegionDebugConsole::onConsoleSuccess, this, _1),
                boost::bind(&LLFloaterRegionDebugConsole::onConsoleError, this, _1));
        }
    }
    else
    {
        LLSD postData = LLSD(input->getText());
        LLCoreHttpUtil::HttpCoroutineAdapter::callbackHttpPost(url, postData,
            NULL,
            boost::bind(&LLFloaterRegionDebugConsole::onAsyncConsoleError, this, _1));

    }

    mOutput->appendText(text, false);
    input->clear();
}

void LLFloaterRegionDebugConsole::onAsyncConsoleError(LLSD result)
{
    LL_WARNS("Console") << UNABLE_TO_SEND_COMMAND << LL_ENDL;
    sConsoleReplySignal(UNABLE_TO_SEND_COMMAND);
}

void LLFloaterRegionDebugConsole::onConsoleError(LLSD result)
{
    LL_WARNS("Console") << UNABLE_TO_SEND_COMMAND << LL_ENDL;
    if (mOutput)
    {
        mOutput->appendText(
            UNABLE_TO_SEND_COMMAND + PROMPT,
            false);
    }

}

void LLFloaterRegionDebugConsole::onConsoleSuccess(LLSD result)
{
    if (mOutput)
    {
        LLSD response = result;
        if (response.isMap() && response.has(LLCoreHttpUtil::HttpCoroutineAdapter::HTTP_RESULTS_CONTENT))
        {
            response = response[LLCoreHttpUtil::HttpCoroutineAdapter::HTTP_RESULTS_CONTENT];
        }
        mOutput->appendText(
            response.asString() + PROMPT, false);
    }
}

void LLFloaterRegionDebugConsole::onReplyReceived(const std::string& output)
{
    mOutput->appendText(output + PROMPT, false);
}

LLHTTPRegistration<ConsoleResponseNode>
    gHTTPRegistrationMessageDebugConsoleResponse(
        "/message/SimConsoleResponse");