ShellSession.cpp 4.97 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
xiongziliang committed
3 4 5
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
xiongziliang committed
6 7 8
 * Use of this source code is governed by MIT license that can be found in the
 * LICENSE file in the root of the source tree. All contributing project authors
 * may be found in the AUTHORS file in the root of the source tree.
xzl committed
9 10 11
 */

#include "ShellSession.h"
12
#include "Util/CMD.h"
xzl committed
13
#include "Util/onceToken.h"
14
#include "Util/NoticeCenter.h"
15 16
#include "Common/config.h"
#include "ShellCMD.h"
xiongziliang committed
17
using namespace toolkit;
xzl committed
18

xiongziliang committed
19
namespace mediakit {
xzl committed
20

21 22 23
static onceToken s_token([]() {
    REGIST_CMD(media);
}, nullptr);
24

xiongziliang committed
25
ShellSession::ShellSession(const Socket::Ptr &_sock) : TcpSession(_sock) {
xiongziliang committed
26
    DebugP(this);
27
    pleaseInputUser();
xzl committed
28 29 30
}

ShellSession::~ShellSession() {
xiongziliang committed
31
    DebugP(this);
xzl committed
32 33
}

34
void ShellSession::onRecv(const Buffer::Ptr&buf) {
35
    //DebugL << hexdump(buf->data(), buf->size());
36
    GET_CONFIG(uint32_t,maxReqSize,Shell::kMaxReqSize);
37
    if (_strRecvBuf.size() + buf->size() >= maxReqSize) {
38 39 40 41 42 43
        shutdown(SockException(Err_other,"recv buffer overflow"));
        return;
    }
    _beatTicker.resetTime();
    _strRecvBuf.append(buf->data(), buf->size());
    if (_strRecvBuf.find("\xff\xf4\xff\0xfd\x06") != std::string::npos) {
xiongziliang committed
44
        SockSender::send("\033[0m\r\n	Bye bye!\r\n");
45 46 47 48 49 50 51 52 53 54 55 56 57
        shutdown(SockException(Err_other,"received Ctrl+C"));
        return;
    }
    size_t index;
    string line;
    while ((index = _strRecvBuf.find("\r\n")) != std::string::npos) {
        line = _strRecvBuf.substr(0, index);
        _strRecvBuf.erase(0, index + 2);
        if (!onCommandLine(line)) {
            shutdown(SockException(Err_other,"exit cmd"));
            return;
        }
    }
xzl committed
58 59
}

xiongziliang committed
60 61 62 63
void ShellSession::onError(const SockException &err){
    WarnP(this) << err.what();
}

xzl committed
64
void ShellSession::onManager() {
65 66
    if (_beatTicker.elapsedTime() > 1000 * 60 * 5) {
        //5 miniutes for alive
xiongziliang committed
67
        shutdown(SockException(Err_timeout,"session timeout"));
68 69
        return;
    }
xzl committed
70 71
}

72
inline bool ShellSession::onCommandLine(const string& line) {
73
    auto loginInterceptor = _loginInterceptor;
74
    if (loginInterceptor) {
75 76 77
        bool ret = loginInterceptor(line);
        return ret;
    }
78 79 80
    try {
        std::shared_ptr<stringstream> ss(new stringstream);
        CMDRegister::Instance()(line,ss);
xiongziliang committed
81
        SockSender::send(ss->str());
82 83 84
    }catch(ExitException &ex){
        return false;
    }catch(std::exception &ex){
xiongziliang committed
85 86
        SockSender::send(ex.what());
        SockSender::send("\r\n");
87 88
    }
    printShellPrefix();
89
    return true;
xzl committed
90 91
}

92
inline void ShellSession::pleaseInputUser() {
xiongziliang committed
93 94
    SockSender::send("\033[0m");
    SockSender::send(StrPrinter << SERVER_NAME << " login: " << endl);
95 96
    _loginInterceptor = [this](const string &user_name) {
        _strUserName=user_name;
97
        pleaseInputPasswd();
98 99
        return true;
    };
xzl committed
100
}
101
inline void ShellSession::pleaseInputPasswd() {
xiongziliang committed
102
    SockSender::send("Password: \033[8m");
103
    _loginInterceptor = [this](const string &passwd) {
104 105 106
        auto onAuth = [this](const string &errMessage){
            if(!errMessage.empty()){
                //鉴权失败
xiongziliang committed
107 108 109 110 111 112 113
                SockSender::send(StrPrinter
                                 << "\033[0mAuth failed("
                                 << errMessage
                                 << "), please try again.\r\n"
                                 << _strUserName << "@" << SERVER_NAME
                                 << "'s password: \033[8m"
                                 << endl);
114 115
                return;
            }
xiongziliang committed
116 117 118 119
            SockSender::send("\033[0m");
            SockSender::send("-----------------------------------------\r\n");
            SockSender::send(StrPrinter<<"欢迎来到"<<SERVER_NAME<<", 你可输入\"help\"查看帮助.\r\n"<<endl);
            SockSender::send("-----------------------------------------\r\n");
120
            printShellPrefix();
121
            _loginInterceptor=nullptr;
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
        };

        weak_ptr<ShellSession> weakSelf = dynamic_pointer_cast<ShellSession>(shared_from_this());
        Broadcast::AuthInvoker invoker = [weakSelf,onAuth](const string &errMessage){
            auto strongSelf =  weakSelf.lock();
            if(!strongSelf){
                return;
            }
            strongSelf->async([errMessage,weakSelf,onAuth](){
                auto strongSelf =  weakSelf.lock();
                if(!strongSelf){
                    return;
                }
                onAuth(errMessage);
            });
        };

139
        auto flag = NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastShellLogin,_strUserName,passwd,invoker,static_cast<SockInfo &>(*this));
140 141 142 143
        if(!flag){
            //如果无人监听shell登录事件,那么默认shell无法登录
            onAuth("please listen kBroadcastShellLogin event");
        }
144 145
        return true;
    };
xzl committed
146 147
}

148
inline void ShellSession::printShellPrefix() {
xiongziliang committed
149
    SockSender::send(StrPrinter << _strUserName << "@" << SERVER_NAME << "# " << endl);
xzl committed
150 151
}

xiongziliang committed
152
}/* namespace mediakit */