HttpSession.cpp 22.2 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
 */
xiongziliang committed
10

11 12 13 14
#if !defined(_WIN32)
#include <dirent.h>
#endif //!defined(_WIN32)

xzl committed
15 16 17
#include <stdio.h>
#include <sys/stat.h>
#include <algorithm>
18
#include <iomanip>
xzl committed
19

xiongziliang committed
20
#include "Common/config.h"
xzl committed
21 22
#include "strCoding.h"
#include "HttpSession.h"
xiongziliang committed
23 24
#include "Util/base64.h"
#include "Util/SHA1.h"
xiongziliang committed
25
using namespace toolkit;
26

xiongziliang committed
27
namespace mediakit {
xzl committed
28

xiongziliang committed
29
HttpSession::HttpSession(const Socket::Ptr &pSock) : TcpSession(pSock) {
xiongziliang committed
30
    TraceP(this);
31 32
    GET_CONFIG(uint32_t,keep_alive_sec,Http::kKeepAliveSecond);
    pSock->setSendTimeOutSecond(keep_alive_sec);
33 34
    //起始接收buffer缓存设置为4K,节省内存
    pSock->setReadBuffer(std::make_shared<BufferRaw>(4 * 1024));
xzl committed
35 36 37
}

HttpSession::~HttpSession() {
xiongziliang committed
38
    TraceP(this);
xzl committed
39 40
}

41 42 43 44 45
void HttpSession::Handle_Req_HEAD(int64_t &content_len){
    //暂时全部返回200 OK,因为HTTP GET存在按需生成流的操作,所以不能按照HTTP GET的流程返回
    //如果直接返回404,那么又会导致按需生成流的逻辑失效,所以HTTP HEAD在静态文件或者已存在资源时才有效
    //对于按需生成流的直播场景并不适用
    sendResponse("200 OK", true);
46 47
}

48
int64_t HttpSession::onRecvHeader(const char *header,uint64_t len) {
49 50 51 52 53
    typedef void (HttpSession::*HttpCMDHandle)(int64_t &);
    static unordered_map<string, HttpCMDHandle> s_func_map;
    static onceToken token([]() {
        s_func_map.emplace("GET",&HttpSession::Handle_Req_GET);
        s_func_map.emplace("POST",&HttpSession::Handle_Req_POST);
54
        s_func_map.emplace("HEAD",&HttpSession::Handle_Req_HEAD);
55
    }, nullptr);
xiongziliang committed
56

57 58 59 60 61
    _parser.Parse(header);
    urlDecode(_parser);
    string cmd = _parser.Method();
    auto it = s_func_map.find(cmd);
    if (it == s_func_map.end()) {
xiongziliang committed
62
        WarnP(this) << "不支持该命令:" << cmd;
63
        sendResponse("405 Not Allowed", true);
xiongziliang committed
64
        return 0;
65
    }
66

xiongziliang committed
67 68 69 70
    //跨域
    _origin = _parser["Origin"];

    //默认后面数据不是content而是header
71 72
    int64_t content_len = 0;
    auto &fun = it->second;
xiongziliang committed
73 74 75 76 77 78
    try {
        (this->*fun)(content_len);
    }catch (exception &ex){
        shutdown(SockException(Err_shutdown,ex.what()));
    }

79 80 81 82
    //清空解析器节省内存
    _parser.Clear();
    //返回content长度
    return content_len;
xzl committed
83
}
84

85
void HttpSession::onRecvContent(const char *data,uint64_t len) {
86 87 88 89 90
    if(_contentCallBack){
        if(!_contentCallBack(data,len)){
            _contentCallBack = nullptr;
        }
    }
91 92 93
}

void HttpSession::onRecv(const Buffer::Ptr &pBuf) {
94
    _ticker.resetTime();
95
    input(pBuf->data(),pBuf->size());
96 97
}

xzl committed
98
void HttpSession::onError(const SockException& err) {
xiongziliang committed
99
    if(_is_flv_stream){
100
        uint64_t duration = _ticker.createdTime()/1000;
xiongziliang committed
101
        //flv播放器
102
        WarnP(this) << "FLV播放器("
xiongziliang committed
103 104 105
                    << _mediaInfo._vhost << "/"
                    << _mediaInfo._app << "/"
                    << _mediaInfo._streamid
106 107
                    << ")断开:" << err.what()
                    << ",耗时(s):" << duration;
xiongziliang committed
108 109 110

        GET_CONFIG(uint32_t,iFlowThreshold,General::kFlowThreshold);
        if(_ui64TotalBytes > iFlowThreshold * 1024){
111
            NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastFlowReport, _mediaInfo, _ui64TotalBytes, duration , true, static_cast<SockInfo &>(*this));
xiongziliang committed
112 113 114 115 116
        }
        return;
    }

    //http客户端
xiongziliang committed
117 118 119 120 121
    if(_ticker.createdTime() < 10 * 1000){
        TraceP(this) << err.what();
    }else{
        WarnP(this) << err.what();
    }
xzl committed
122 123 124
}

void HttpSession::onManager() {
125
    GET_CONFIG(uint32_t,keepAliveSec,Http::kKeepAliveSecond);
126

127
    if(_ticker.elapsedTime() > keepAliveSec * 1000){
128 129 130
        //1分钟超时
        shutdown(SockException(Err_timeout,"session timeouted"));
    }
xzl committed
131
}
xiongziliang committed
132

133
bool HttpSession::checkWebSocket(){
134
    auto Sec_WebSocket_Key = _parser["Sec-WebSocket-Key"];
135
    if (Sec_WebSocket_Key.empty()) {
136 137
        return false;
    }
138 139
    auto Sec_WebSocket_Accept = encodeBase64(
            SHA1::encode_bin(Sec_WebSocket_Key + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"));
140 141 142 143 144

    KeyValue headerOut;
    headerOut["Upgrade"] = "websocket";
    headerOut["Connection"] = "Upgrade";
    headerOut["Sec-WebSocket-Accept"] = Sec_WebSocket_Accept;
145
    if (!_parser["Sec-WebSocket-Protocol"].empty()) {
146 147
        headerOut["Sec-WebSocket-Protocol"] = _parser["Sec-WebSocket-Protocol"];
    }
148

149
    auto res_cb = [this, headerOut]() {
150
        _flv_over_websocket = true;
151
        sendResponse("101 Switching Protocols", false, nullptr, headerOut, nullptr, true);
152 153 154
    };

    //判断是否为websocket-flv
155
    if (checkLiveFlvStream(res_cb)) {
156 157 158 159 160
        //这里是websocket-flv直播请求
        return true;
    }

    //如果checkLiveFlvStream返回false,则代表不是websocket-flv,而是普通的websocket连接
161 162
    if (!onWebSocketConnect(_parser)) {
        sendResponse("501 Not Implemented", true, nullptr, headerOut);
163 164
        return true;
    }
165
    sendResponse("101 Switching Protocols", false, nullptr, headerOut, nullptr, true);
166
    return true;
xiongziliang committed
167
}
168

169 170
//http-flv 链接格式:http://vhost-url:port/app/streamid.flv?key1=value1&key2=value2
//如果url(除去?以及后面的参数)后缀是.flv,那么表明该url是一个http-flv直播。
171
bool HttpSession::checkLiveFlvStream(const function<void()> &cb){
172 173 174 175 176 177 178 179 180 181 182
    auto pos = strrchr(_parser.Url().data(),'.');
    if(!pos){
        //未找到".flv"后缀
        return false;
    }
    if(strcasecmp(pos,".flv") != 0){
        //未找到".flv"后缀
        return false;
    }

    //这是个.flv的流
183
    _mediaInfo.parse(string(RTMP_SCHEMA) + "://" + _parser["Host"] + _parser.FullUrl());
184 185
    if(_mediaInfo._app.empty() || _mediaInfo._streamid.size() < 5){
        //url不合法
186
        return false;
187
    }
188
    _mediaInfo._streamid.erase(_mediaInfo._streamid.size() - 4);//去除.flv后缀
189
    bool bClose = !strcasecmp(_parser["Connection"].data(),"close");
190 191

    weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
192 193 194

    //鉴权结果回调
    auto onRes = [cb, weakSelf, bClose](const string &err){
195
        auto strongSelf = weakSelf.lock();
196
        if (!strongSelf) {
197 198 199
            //本对象已经销毁
            return;
        }
200 201 202 203

        if(!err.empty()){
            //播放鉴权失败
            strongSelf->sendResponse("401 Unauthorized", bClose, nullptr, KeyValue(), std::make_shared<HttpStringBody>(err));
204 205
            return;
        }
206 207 208 209 210 211 212 213 214 215 216 217 218

        //异步查找rtmp流
        MediaSource::findAsync(strongSelf->_mediaInfo, strongSelf, [weakSelf, bClose, cb](const MediaSource::Ptr &src) {
            auto strongSelf = weakSelf.lock();
            if (!strongSelf) {
                //本对象已经销毁
                return;
            }
            auto rtmp_src = dynamic_pointer_cast<RtmpMediaSource>(src);
            if (!rtmp_src) {
                //未找到该流
                strongSelf->sendNotFound(bClose);
                return;
219 220
            }

221
            if (!cb) {
222
                //找到rtmp源,发送http头,负载后续发送
223 224 225
                strongSelf->sendResponse("200 OK", false, "video/x-flv", KeyValue(), nullptr, true);
            } else {
                //自定义发送http头
226
                cb();
227
            }
228

229
            //http-flv直播牺牲延时提升发送性能
230 231 232 233 234
            strongSelf->setSocketFlags();
            strongSelf->start(strongSelf->getPoller(), rtmp_src);
            strongSelf->_is_flv_stream = true;
        });
    };
235

236 237 238 239
    Broadcast::AuthInvoker invoker = [weakSelf, onRes](const string &err) {
        auto strongSelf = weakSelf.lock();
        if (!strongSelf) {
            return;
240
        }
241 242 243 244 245 246 247 248 249 250
        strongSelf->async([onRes, err]() {
            onRes(err);
        });
    };

    auto flag = NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastMediaPlayed, _mediaInfo, invoker, static_cast<SockInfo &>(*this));
    if (!flag) {
        //该事件无人监听,默认不鉴权
        onRes("");
    }
251
    return true;
xiongziliang committed
252
}
253

254
void HttpSession::Handle_Req_GET(int64_t &content_len) {
255 256 257 258
    Handle_Req_GET_l(content_len, true);
}

void HttpSession::Handle_Req_GET_l(int64_t &content_len, bool sendBody) {
259 260 261 262
    //先看看是否为WebSocket请求
    if(checkWebSocket()){
        content_len = -1;
        _contentCallBack = [this](const char *data,uint64_t len){
263
            WebSocketSplitter::decode((uint8_t *)data,len);
264 265 266 267 268
            //_contentCallBack是可持续的,后面还要处理后续数据
            return true;
        };
        return;
    }
xiongziliang committed
269

270
    if(emitHttpEvent(false)){
271
        //拦截http api事件
272 273
        return;
    }
xiongziliang committed
274

275 276 277 278 279
    if(checkLiveFlvStream()){
        //拦截http-flv播放器
        return;
    }

280
    bool bClose = !strcasecmp(_parser["Connection"].data(),"close");
281

282 283 284 285 286
    weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
    HttpFileManager::onAccessPath(*this, _parser, [weakSelf, bClose](const string &status_code, const string &content_type,
                                                                     const StrCaseMap &responseHeader, const HttpBody::Ptr &body) {
        auto strongSelf = weakSelf.lock();
        if (!strongSelf) {
287
            return;
288
        }
289 290 291 292
        strongSelf->async([weakSelf, bClose, status_code, content_type, responseHeader, body]() {
            auto strongSelf = weakSelf.lock();
            if (!strongSelf) {
                return;
293
            }
xiongziliang committed
294
            strongSelf->sendResponse(status_code.data(), bClose, content_type.data(), responseHeader, body);
295
        });
296
    });
xzl committed
297 298
}

299 300 301 302 303
static string dateStr() {
    char buf[64];
    time_t tt = time(NULL);
    strftime(buf, sizeof buf, "%a, %b %d %Y %H:%M:%S GMT", gmtime(&tt));
    return buf;
xzl committed
304
}
305

306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378
class AsyncSenderData {
public:
    friend class AsyncSender;
    typedef std::shared_ptr<AsyncSenderData> Ptr;
    AsyncSenderData(const TcpSession::Ptr &session, const HttpBody::Ptr &body, bool close_when_complete) {
        _session = dynamic_pointer_cast<HttpSession>(session);
        _body = body;
        _close_when_complete = close_when_complete;
    }
    ~AsyncSenderData() = default;
private:
    std::weak_ptr<HttpSession> _session;
    HttpBody::Ptr _body;
    bool _close_when_complete;
    bool _read_complete = false;
};

class AsyncSender {
public:
    typedef std::shared_ptr<AsyncSender> Ptr;
    static bool onSocketFlushed(const AsyncSenderData::Ptr &data) {
        if (data->_read_complete) {
            if (data->_close_when_complete) {
                //发送完毕需要关闭socket
                shutdown(data->_session.lock());
            }
            return false;
        }

        GET_CONFIG(uint32_t, sendBufSize, Http::kSendBufSize);
        data->_body->readDataAsync(sendBufSize, [data](const Buffer::Ptr &sendBuf) {
            auto session = data->_session.lock();
            if (!session) {
                //本对象已经销毁
                return;
            }
            session->async([data, sendBuf]() {
                auto session = data->_session.lock();
                if (!session) {
                    //本对象已经销毁
                    return;
                }
                onRequestData(data, session, sendBuf);
            }, false);
        });
        return true;
    }

private:
    static void onRequestData(const AsyncSenderData::Ptr &data, const std::shared_ptr<HttpSession> &session, const Buffer::Ptr &sendBuf) {
        session->_ticker.resetTime();
        if (sendBuf && session->send(sendBuf) != -1) {
            //文件还未读完,还需要继续发送
            if (!session->isSocketBusy()) {
                //socket还可写,继续请求数据
                onSocketFlushed(data);
            }
            return;
        }
        //文件写完了
        data->_read_complete = true;
        if (!session->isSocketBusy() && data->_close_when_complete) {
            shutdown(session);
        }
    }

    static void shutdown(const std::shared_ptr<HttpSession> &session) {
        if(session){
            session->shutdown(SockException(Err_shutdown, StrPrinter << "close connection after send http body completed."));
        }
    }
};

379 380 381 382 383 384 385 386 387
static const string kDate = "Date";
static const string kServer = "Server";
static const string kConnection = "Connection";
static const string kKeepAlive = "Keep-Alive";
static const string kContentType = "Content-Type";
static const string kContentLength = "Content-Length";
static const string kAccessControlAllowOrigin = "Access-Control-Allow-Origin";
static const string kAccessControlAllowCredentials = "Access-Control-Allow-Credentials";

xiongziliang committed
388 389 390 391 392
void HttpSession::sendResponse(const char *pcStatus,
                               bool bClose,
                               const char *pcContentType,
                               const HttpSession::KeyValue &header,
                               const HttpBody::Ptr &body,
393
                               bool no_content_length ){
xiongziliang committed
394 395 396 397 398 399 400 401 402 403
    GET_CONFIG(string,charSet,Http::kCharSet);
    GET_CONFIG(uint32_t,keepAliveSec,Http::kKeepAliveSecond);

    //body默认为空
    int64_t size = 0;
    if (body && body->remainSize()) {
        //有body,获取body大小
        size = body->remainSize();
    }

404
    if(no_content_length){
405
        //http-flv直播是Keep-Alive类型
xiongziliang committed
406
        bClose = false;
407 408 409
    }else if(size >= INT64_MAX){
        //不固定长度的body,那么发送完body后应该关闭socket,以便浏览器做下载完毕的判断
        bClose = true;
xiongziliang committed
410 411 412
    }

    HttpSession::KeyValue &headerOut = const_cast<HttpSession::KeyValue &>(header);
413
    headerOut.emplace(kDate, dateStr());
xiongziliang committed
414
    headerOut.emplace(kServer, SERVER_NAME);
415
    headerOut.emplace(kConnection, bClose ? "close" : "keep-alive");
xiongziliang committed
416
    if(!bClose){
417 418 419 420
        string keepAliveString = "timeout=";
        keepAliveString += to_string(keepAliveSec);
        keepAliveString += ", max=100";
        headerOut.emplace(kKeepAlive,std::move(keepAliveString));
xiongziliang committed
421 422 423 424
    }

    if(!_origin.empty()){
        //设置跨域
425 426
        headerOut.emplace(kAccessControlAllowOrigin,_origin);
        headerOut.emplace(kAccessControlAllowCredentials, "true");
xiongziliang committed
427 428
    }

429
    if(!no_content_length && size >= 0 && size < INT64_MAX){
430
        //文件长度为固定值,且不是http-flv强制设置Content-Length
431
        headerOut[kContentLength] = to_string(size);
xiongziliang committed
432 433 434 435 436 437 438 439 440
    }

    if(size && !pcContentType){
        //有body时,设置缺省类型
        pcContentType = "text/plain";
    }

    if(size && pcContentType){
        //有body时,设置文件类型
441 442 443 444
        string strContentType = pcContentType;
        strContentType += "; charset=";
        strContentType += charSet;
        headerOut.emplace(kContentType,std::move(strContentType));
xiongziliang committed
445 446
    }

447
    //发送http头
448 449 450 451 452
    string str;
    str.reserve(256);
    str += "HTTP/1.1 " ;
    str += pcStatus ;
    str += "\r\n";
453
    for (auto &pr : header) {
454 455 456 457
        str += pr.first ;
        str += ": ";
        str += pr.second;
        str += "\r\n";
458
    }
459
    str += "\r\n";
xiongziliang committed
460
    SockSender::send(std::move(str));
461 462
    _ticker.resetTime();

xiongziliang committed
463
    if(!size){
464 465
        //没有body
        if(bClose){
466
            shutdown(SockException(Err_shutdown,StrPrinter << "close connection after send http header completed with status code:" << pcStatus));
467 468 469 470
        }
        return;
    }

471
    GET_CONFIG(uint32_t, sendBufSize, Http::kSendBufSize);
472 473 474 475
    if(body->remainSize() > sendBufSize){
        //文件下载提升发送性能
        setSocketFlags();
    }
476 477 478

    //发送http body
    AsyncSenderData::Ptr data = std::make_shared<AsyncSenderData>(shared_from_this(),body,bClose);
479
    getSock()->setOnFlush([data](){
480 481 482
        return AsyncSender::onSocketFlushed(data);
    });
    AsyncSender::onSocketFlushed(data);
483 484
}

xiongziliang committed
485
string HttpSession::urlDecode(const string &str){
486
    auto ret = strCoding::UrlDecode(str);
487
#ifdef _WIN32
488
    GET_CONFIG(string,charSet,Http::kCharSet);
489 490 491 492
    bool isGb2312 = !strcasecmp(charSet.data(), "gb2312");
    if (isGb2312) {
        ret = strCoding::UTF8ToGB2312(ret);
    }
493
#endif // _WIN32
xiongziliang committed
494 495
    return ret;
}
496

497
void HttpSession::urlDecode(Parser &parser){
498 499 500 501
    parser.setUrl(urlDecode(parser.Url()));
    for(auto &pr : _parser.getUrlArgs()){
        const_cast<string &>(pr.second) = urlDecode(pr.second);
    }
xiongziliang committed
502
}
xzl committed
503

504
bool HttpSession::emitHttpEvent(bool doInvoke){
505
    bool bClose = !strcasecmp(_parser["Connection"].data(),"close");
506 507 508 509 510 511 512 513 514 515
    /////////////////////异步回复Invoker///////////////////////////////
    weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
    HttpResponseInvoker invoker = [weakSelf,bClose](const string &codeOut, const KeyValue &headerOut, const HttpBody::Ptr &body){
        auto strongSelf = weakSelf.lock();
        if(!strongSelf) {
            return;
        }
        strongSelf->async([weakSelf,bClose,codeOut,headerOut,body]() {
            auto strongSelf = weakSelf.lock();
            if(!strongSelf) {
516
                //本对象已经销毁
517 518
                return;
            }
xiongziliang committed
519
            strongSelf->sendResponse(codeOut.data(), bClose, nullptr, headerOut, body);
520 521 522 523
        });
    };
    ///////////////////广播HTTP事件///////////////////////////
    bool consumed = false;//该事件是否被消费
524
    NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastHttpRequest,_parser,invoker,consumed,static_cast<SockInfo &>(*this));
525 526 527 528 529
    if(!consumed && doInvoke){
        //该事件无人消费,所以返回404
        invoker("404 Not Found",KeyValue(), HttpBody::Ptr());
    }
    return consumed;
xiongziliang committed
530
}
531 532

void HttpSession::Handle_Req_POST(int64_t &content_len) {
533
    GET_CONFIG(uint64_t,maxReqSize,Http::kMaxReqSize);
534

535
    int64_t totalContentLen = _parser["Content-Length"].empty() ? -1 : atoll(_parser["Content-Length"].data());
536

537 538 539 540 541 542
    if(totalContentLen == 0){
        //content为空
        //emitHttpEvent内部会选择是否关闭连接
        emitHttpEvent(true);
        return;
    }
543

544 545
    //根据Content-Length设置接收缓存大小
    if(totalContentLen > 0){
546
        getSock()->setReadBuffer(std::make_shared<BufferRaw>(MIN(totalContentLen + 1,256 * 1024)));
547
    }else{
548
        //不定长度的Content-Length
549
        getSock()->setReadBuffer(std::make_shared<BufferRaw>(256 * 1024));
550
    }
551 552

    if(totalContentLen > 0 && totalContentLen < maxReqSize ){
553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581
        //返回固定长度的content
        content_len = totalContentLen;
        auto parserCopy = _parser;
        _contentCallBack = [this,parserCopy](const char *data,uint64_t len){
            //恢复http头
            _parser = parserCopy;
            //设置content
            _parser.setContent(string(data,len));
            //触发http事件,emitHttpEvent内部会选择是否关闭连接
            emitHttpEvent(true);
            //清空数据,节省内存
            _parser.Clear();
            //content已经接收完毕
            return false;
        };
    }else{
        //返回不固定长度的content
        content_len = -1;
        auto parserCopy = _parser;
        std::shared_ptr<uint64_t> recvedContentLen = std::make_shared<uint64_t>(0);
        bool bClose = !strcasecmp(_parser["Connection"].data(),"close");

        _contentCallBack = [this,parserCopy,totalContentLen,recvedContentLen,bClose](const char *data,uint64_t len){
            *(recvedContentLen) += len;

            onRecvUnlimitedContent(parserCopy,data,len,totalContentLen,*(recvedContentLen));

            if(*(recvedContentLen) < totalContentLen){
                //数据还没接收完毕
582
                //_contentCallBack是可持续的,后面还要处理后续content数据
583
                return true;
584
            }
585

586
            //数据接收完毕
587
            if(!bClose){
588 589 590
                //keep-alive类型连接
                //content接收完毕,后续都是http header
                setContentLen(0);
591
                //content已经接收完毕
592 593 594 595
                return false;
            }

            //连接类型是close类型,收完content就关闭连接
xiongziliang committed
596
            shutdown(SockException(Err_shutdown,"recv http content completed"));
597
            //content已经接收完毕
598
            return false ;
599 600 601
        };
    }
    //有后续content数据要处理,暂时不关闭连接
602
}
603 604

void HttpSession::sendNotFound(bool bClose) {
605
    GET_CONFIG(string,notFound,Http::kNotFound);
xiongziliang committed
606
    sendResponse("404 Not Found", bClose,"text/html",KeyValue(),std::make_shared<HttpStringBody>(notFound));
xzl committed
607 608
}

609
void HttpSession::setSocketFlags(){
610 611
    GET_CONFIG(int, mergeWriteMS, General::kMergeWriteMS);
    if(mergeWriteMS > 0) {
612
        //推流模式下,关闭TCP_NODELAY会增加推流端的延时,但是服务器性能将提高
613
        SockUtil::setNoDelay(getSock()->rawFD(), false);
614
        //播放模式下,开启MSG_MORE会增加延时,但是能提高发送性能
xiongziliang committed
615
        setSendFlags(SOCKET_DEFAULE_FLAGS | FLAG_MORE);
616 617
    }
}
xiongziliang committed
618

xiongziliang committed
619 620 621 622 623 624
void HttpSession::onWrite(const Buffer::Ptr &buffer, bool flush) {
    if(flush){
        //需要flush那么一次刷新缓存
        HttpSession::setSendFlushFlag(true);
    }

625
    _ticker.resetTime();
626 627 628
    if(!_flv_over_websocket){
        _ui64TotalBytes += buffer->size();
        send(buffer);
xiongziliang committed
629 630 631 632 633 634 635
    }else{
        WebSocketHeader header;
        header._fin = true;
        header._reserved = 0;
        header._opcode = WebSocketHeader::BINARY;
        header._mask_flag = false;
        WebSocketSplitter::encode(header,buffer);
636 637
    }

xiongziliang committed
638 639 640 641
    if(flush){
        //本次刷新缓存后,下次不用刷新缓存
        HttpSession::setSendFlushFlag(false);
    }
642 643
}

644 645 646
void HttpSession::onWebSocketEncodeData(const Buffer::Ptr &buffer){
    _ui64TotalBytes += buffer->size();
    send(buffer);
xiongziliang committed
647 648
}

649 650 651 652 653 654 655 656 657 658 659 660 661 662 663
void HttpSession::onWebSocketDecodeComplete(const WebSocketHeader &header_in){
    WebSocketHeader& header = const_cast<WebSocketHeader&>(header_in);
    header._mask_flag = false;

    switch (header._opcode) {
        case WebSocketHeader::CLOSE: {
            encode(header, nullptr);
            shutdown(SockException(Err_shutdown, "recv close request from client"));
            break;
        }

        default : break;
    }
}

xiongziliang committed
664
void HttpSession::onDetach() {
665
    shutdown(SockException(Err_shutdown,"rtmp ring buffer detached"));
xiongziliang committed
666 667 668
}

std::shared_ptr<FlvMuxer> HttpSession::getSharedPtr(){
669
    return dynamic_pointer_cast<FlvMuxer>(shared_from_this());
670 671
}

xiongziliang committed
672
} /* namespace mediakit */