HttpSession.cpp 33.6 KB
Newer Older
xiongziliang committed
1
/*
xiongziliang committed
2
 * MIT License
xzl committed
3
 *
xiongziliang committed
4
 * Copyright (c) 2016-2019 xiongziliang <771730766@qq.com>
xiongziliang committed
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
 *
 * This file is part of ZLMediaKit(https://github.com/xiongziliang/ZLMediaKit).
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy
 * of this software and associated documentation files (the "Software"), to deal
 * in the Software without restriction, including without limitation the rights
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 * copies of the Software, and to permit persons to whom the Software is
 * furnished to do so, subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
xzl committed
25
 */
26 27 28 29
#if !defined(_WIN32)
#include <dirent.h>
#endif //!defined(_WIN32)

xzl committed
30 31 32
#include <stdio.h>
#include <sys/stat.h>
#include <algorithm>
33
#include <iomanip>
xzl committed
34

xiongziliang committed
35
#include "Common/config.h"
xzl committed
36 37 38 39 40 41
#include "strCoding.h"
#include "HttpSession.h"
#include "Util/File.h"
#include "Util/util.h"
#include "Util/TimeTicker.h"
#include "Util/onceToken.h"
xiongzilaing committed
42
#include "Util/mini.h"
xzl committed
43
#include "Util/NoticeCenter.h"
xiongziliang committed
44 45
#include "Util/base64.h"
#include "Util/SHA1.h"
xiongziliang committed
46
#include "Rtmp/utils.h"
xiongziliang committed
47
using namespace toolkit;
48

xiongziliang committed
49
namespace mediakit {
xzl committed
50

51
static int kHlsCookieSecond = 10 * 60;
52
static const string kCookieName = "ZL_COOKIE";
53 54
static const string kCookiePathKey = "kCookiePathKey";
static const string kAccessErrKey = "kAccessErrKey";
55

xzl committed
56 57 58 59 60 61 62 63 64 65
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;
}
static const char*
get_mime_type(const char* name) {
	const char* dot;
	dot = strrchr(name, '.');
xiongziliang committed
66
	static HttpSession::KeyValue mapType;
xzl committed
67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
	static onceToken token([&]() {
		mapType.emplace(".html","text/html");
		mapType.emplace(".htm","text/html");
		mapType.emplace(".mp4","video/mp4");
		mapType.emplace(".m3u8","application/vnd.apple.mpegurl");
		mapType.emplace(".jpg","image/jpeg");
		mapType.emplace(".jpeg","image/jpeg");
		mapType.emplace(".gif","image/gif");
		mapType.emplace(".png","image/png");
		mapType.emplace(".ico","image/x-icon");
		mapType.emplace(".css","text/css");
		mapType.emplace(".js","application/javascript");
		mapType.emplace(".au","audio/basic");
		mapType.emplace(".wav","audio/wav");
		mapType.emplace(".avi","video/x-msvideo");
		mapType.emplace(".mov","video/quicktime");
		mapType.emplace(".qt","video/quicktime");
		mapType.emplace(".mpeg","video/mpeg");
		mapType.emplace(".mpe","video/mpeg");
		mapType.emplace(".vrml","model/vrml");
		mapType.emplace(".wrl","model/vrml");
		mapType.emplace(".midi","audio/midi");
		mapType.emplace(".mid","audio/midi");
		mapType.emplace(".mp3","audio/mpeg");
		mapType.emplace(".ogg","application/ogg");
		mapType.emplace(".pac","application/x-ns-proxy-autoconfig");
xiongziliang committed
93
        mapType.emplace(".flv","video/x-flv");
xzl committed
94 95 96 97
	}, nullptr);
	if(!dot){
		return "text/plain";
	}
xiongziliang committed
98
	auto it = mapType.find(dot);
xzl committed
99 100 101 102 103 104 105
	if (it == mapType.end()) {
		return "text/plain";
	}
	return it->second.data();
}


xiongziliang committed
106
HttpSession::HttpSession(const Socket::Ptr &pSock) : TcpSession(pSock) {
xiongziliang committed
107
    TraceP(this);
108 109
    GET_CONFIG(uint32_t,keep_alive_sec,Http::kKeepAliveSecond);
    pSock->setSendTimeOutSecond(keep_alive_sec);
110 111
	//起始接收buffer缓存设置为4K,节省内存
	pSock->setReadBuffer(std::make_shared<BufferRaw>(4 * 1024));
xzl committed
112 113 114
}

HttpSession::~HttpSession() {
xiongziliang committed
115
    TraceP(this);
xzl committed
116 117
}

118
int64_t HttpSession::onRecvHeader(const char *header,uint64_t len) {
xiongziliang committed
119
	typedef void (HttpSession::*HttpCMDHandle)(int64_t &);
xiongziliang committed
120 121 122 123 124 125
	static unordered_map<string, HttpCMDHandle> g_mapCmdIndex;
	static onceToken token([]() {
		g_mapCmdIndex.emplace("GET",&HttpSession::Handle_Req_GET);
		g_mapCmdIndex.emplace("POST",&HttpSession::Handle_Req_POST);
	}, nullptr);

126 127 128
	_parser.Parse(header);
	urlDecode(_parser);
	string cmd = _parser.Method();
xzl committed
129 130 131
	auto it = g_mapCmdIndex.find(cmd);
	if (it == g_mapCmdIndex.end()) {
		sendResponse("403 Forbidden", makeHttpHeader(true), "");
xiongziliang committed
132 133
        shutdown(SockException(Err_shutdown,StrPrinter << "403 Forbidden:" << cmd));
        return 0;
134 135
	}

xiongziliang committed
136 137 138 139
    //跨域
    _origin = _parser["Origin"];

    //默认后面数据不是content而是header
140 141
	int64_t content_len = 0;
	auto &fun = it->second;
xiongziliang committed
142 143 144 145 146 147 148 149 150 151
    try {
        (this->*fun)(content_len);
    }catch (SockException &ex){
        if(ex){
            shutdown(ex);
        }
    }catch (exception &ex){
        shutdown(SockException(Err_shutdown,ex.what()));
    }

152
	//清空解析器节省内存
153
	_parser.Clear();
154 155
	//返回content长度
	return content_len;
xzl committed
156
}
157

158
void HttpSession::onRecvContent(const char *data,uint64_t len) {
159 160 161
	if(_contentCallBack){
		if(!_contentCallBack(data,len)){
			_contentCallBack = nullptr;
162 163 164 165 166
		}
	}
}

void HttpSession::onRecv(const Buffer::Ptr &pBuf) {
167
    _ticker.resetTime();
168
    input(pBuf->data(),pBuf->size());
169 170
}

xzl committed
171
void HttpSession::onError(const SockException& err) {
xiongziliang committed
172 173 174 175 176
    if(_ticker.createdTime() < 10 * 1000){
        TraceP(this) << err.what();
    }else{
        WarnP(this) << err.what();
    }
177

xiongziliang committed
178
    GET_CONFIG(uint32_t,iFlowThreshold,General::kFlowThreshold);
179
    if(_ui64TotalBytes > iFlowThreshold * 1024){
180
        NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastFlowReport,
181 182 183
										   _mediaInfo,
										   _ui64TotalBytes,
										   _ticker.createdTime()/1000,
184
										   true,
185
										   *this);
186
    }
xzl committed
187 188 189
}

void HttpSession::onManager() {
190
    GET_CONFIG(uint32_t,keepAliveSec,Http::kKeepAliveSecond);
191

192
    if(_ticker.elapsedTime() > keepAliveSec * 1000){
xzl committed
193
		//1分钟超时
xiongziliang committed
194
		shutdown(SockException(Err_timeout,"session timeouted"));
xzl committed
195 196
	}
}
xiongziliang committed
197

198
bool HttpSession::checkWebSocket(){
199
	auto Sec_WebSocket_Key = _parser["Sec-WebSocket-Key"];
200
	if(Sec_WebSocket_Key.empty()){
xiongziliang committed
201 202 203 204
		return false;
	}
	auto Sec_WebSocket_Accept = encodeBase64(SHA1::encode_bin(Sec_WebSocket_Key + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"));

205
	KeyValue headerOut = makeHttpHeader();
xiongziliang committed
206 207 208
	headerOut["Upgrade"] = "websocket";
	headerOut["Connection"] = "Upgrade";
	headerOut["Sec-WebSocket-Accept"] = Sec_WebSocket_Accept;
xiongziliang committed
209 210 211
	if(!_parser["Sec-WebSocket-Protocol"].empty()){
		headerOut["Sec-WebSocket-Protocol"] = _parser["Sec-WebSocket-Protocol"];
	}
212 213 214 215 216 217 218 219 220 221 222 223 224

    auto res_cb = [this,headerOut](){
        _flv_over_websocket = true;
        sendResponse("101 Switching Protocols",headerOut,"");
    };

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

    //如果checkLiveFlvStream返回false,则代表不是websocket-flv,而是普通的websocket连接
225 226 227 228 229
    if(!onWebSocketConnect(_parser)){
        sendResponse("501 Not Implemented",headerOut,"");
        shutdown(SockException(Err_shutdown,"WebSocket server not implemented"));
        return true;
    }
230
    sendResponse("101 Switching Protocols",headerOut,"");
xiongziliang committed
231 232
	return true;
}
233 234
//http-flv 链接格式:http://vhost-url:port/app/streamid.flv?key1=value1&key2=value2
//如果url(除去?以及后面的参数)后缀是.flv,那么表明该url是一个http-flv直播。
235
bool HttpSession::checkLiveFlvStream(const function<void()> &cb){
236
	auto pos = strrchr(_parser.Url().data(),'.');
xiongziliang committed
237 238 239 240
	if(!pos){
		//未找到".flv"后缀
		return false;
	}
xiongziliang committed
241
	if(strcasecmp(pos,".flv") != 0){
xiongziliang committed
242 243 244
		//未找到".flv"后缀
		return false;
	}
245

246 247 248 249 250
	//这是个.flv的流
    _mediaInfo.parse(string(RTMP_SCHEMA) + "://" + _parser["Host"] + _parser.FullUrl());
	if(_mediaInfo._app.empty() || _mediaInfo._streamid.size() < 5){
	    //url不合法
        return false;
xiongziliang committed
251
	}
252
    _mediaInfo._streamid.erase(_mediaInfo._streamid.size() - 4);//去除.flv后缀
xiongziliang committed
253

254
    GET_CONFIG(uint32_t,reqCnt,Http::kMaxReqCount);
255
    bool bClose = (strcasecmp(_parser["Connection"].data(),"close") == 0) || ( ++_iReqCnt > reqCnt);
256 257

    weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
258
    MediaSource::findAsync(_mediaInfo,weakSelf.lock(), true,[weakSelf,bClose,this,cb](const MediaSource::Ptr &src){
259 260
        auto strongSelf = weakSelf.lock();
        if(!strongSelf){
261 262 263 264 265 266
            //本对象已经销毁
            return;
        }
        auto rtmp_src = dynamic_pointer_cast<RtmpMediaSource>(src);
        if(!rtmp_src){
            //未找到该流
267
            sendNotFound(bClose);
268
            if(bClose){
xiongziliang committed
269
                shutdown(SockException(Err_shutdown,"flv stream not found"));
270
            }
271 272
            return;
        }
273
        //找到流了
274
        auto onRes = [this,rtmp_src,cb](const string &err){
275 276
            bool authSuccess = err.empty();
            if(!authSuccess){
277
                sendResponse("401 Unauthorized", makeHttpHeader(true,err.size()),err);
xiongziliang committed
278
                shutdown(SockException(Err_shutdown,StrPrinter << "401 Unauthorized:" << err));
279 280 281
                return ;
            }

282
            if(!cb) {
283 284
                //找到rtmp源,发送http头,负载后续发送
                sendResponse("200 OK", makeHttpHeader(false, 0, get_mime_type(".flv")), "");
285 286
            }else{
                cb();
287
            }
288

289 290 291
            //http-flv直播牺牲延时提升发送性能
            setSocketFlags();

292 293 294 295
            try{
                start(getPoller(),rtmp_src);
            }catch (std::exception &ex){
                //该rtmp源不存在
xiongziliang committed
296
                shutdown(SockException(Err_shutdown,"rtmp mediasource released"));
297 298 299 300 301
            }
        };

        weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
        Broadcast::AuthInvoker invoker = [weakSelf,onRes](const string &err){
302 303 304 305
            auto strongSelf = weakSelf.lock();
            if(!strongSelf){
                return;
            }
306 307 308 309 310 311 312 313 314 315 316 317 318 319
            strongSelf->async([weakSelf,onRes,err](){
                auto strongSelf = weakSelf.lock();
                if(!strongSelf){
                    return;
                }
                onRes(err);
            });
        };
        auto flag = NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastMediaPlayed,_mediaInfo,invoker,*this);
        if(!flag){
            //该事件无人监听,默认不鉴权
            onRes("");
        }
    });
320
    return true;
xiongziliang committed
321
}
322

323
bool makeMeun(const string &httpPath,const string &strFullPath, string &strRet) ;
324

325
static string findIndexFile(const string &dir){
326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342
    DIR *pDir;
    dirent *pDirent;
    if ((pDir = opendir(dir.data())) == NULL) {
        return "";
    }
    set<string> setFile;
    while ((pDirent = readdir(pDir)) != NULL) {
        static set<const char *,StrCaseCompare> indexSet = {"index.html","index.htm","index"};
        if(indexSet.find(pDirent->d_name) !=  indexSet.end()){
            closedir(pDir);
            return pDirent->d_name;
        }
    }
    closedir(pDir);
    return "";
}

343
string HttpSession::getClientUid(){
344 345 346 347 348 349 350
    //如果http客户端不支持cookie,那么我们可以通过url参数来追踪用户
    //如果url参数也没有,那么只能通过ip+端口号来追踪用户
    //追踪用户的目的是为了减少http短链接情况的重复鉴权验证,通过缓存记录鉴权结果,提高性能
    string uid = _parser.Params();
    if(uid.empty()){
        uid = StrPrinter << get_peer_ip() << ":" << get_peer_port();
    }
351
    return uid;
352
}
353 354 355


//字符串是否以xx结尾
356
static bool end_of(const string &str, const string &substr){
357 358 359 360 361
    auto pos = str.rfind(substr);
    return pos != string::npos && pos == str.size() - substr.size();
};

//拦截hls的播放请求
362
static bool checkHls(BroadcastHttpAccessArgs){
363 364 365
    if(!end_of(args._streamid,("/hls.m3u8"))) {
        //不是hls
        return false;
366
    }
367 368
    //访问的hls.m3u8结尾,我们转换成kBroadcastMediaPlayed事件
    Broadcast::AuthInvoker mediaAuthInvoker = [invoker,path](const string &err){
369 370
        //cookie有效期为kHlsCookieSecond
        invoker(err,"",kHlsCookieSecond);
371 372 373 374 375 376 377
    };

    auto args_copy = args;
    replace(args_copy._streamid,"/hls.m3u8","");
    return NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastMediaPlayed,args_copy,mediaAuthInvoker,sender);
}

378
void HttpSession::canAccessPath(const string &path_in,bool is_dir,const function<void(const string &errMsg,const HttpServerCookie::Ptr &cookie)> &callback_in){
379 380 381
    auto path = path_in;
    replace(const_cast<string &>(path),"//","/");

382
    auto callback = [callback_in,this](const string &errMsg,const HttpServerCookie::Ptr &cookie){
383
        try {
384
            callback_in(errMsg,cookie);
385 386 387 388 389 390 391 392 393
        }catch (SockException &ex){
            if(ex){
                shutdown(ex);
            }
        }catch (exception &ex){
            shutdown(SockException(Err_shutdown,ex.what()));
        }
    };

394 395 396 397
    //获取用户唯一id
    auto uid = getClientUid();
    //先根据http头中的cookie字段获取cookie
    HttpServerCookie::Ptr cookie = HttpCookieManager::Instance().getCookie(kCookieName, _parser.getValues());
398 399
    //如果不是从http头中找到的cookie,我们让http客户端设置下cookie
    bool cookie_from_header = true;
400 401 402
    if(!cookie){
        //客户端请求中无cookie,再根据该用户的用户id获取cookie
        cookie = HttpCookieManager::Instance().getCookieByUid(kCookieName, uid);
403
        cookie_from_header = false;
404 405
    }

406
    if(cookie){
407 408
        //找到了cookie,对cookie上锁先
        auto lck = cookie->getLock();
409 410 411
        auto accessErr = (*cookie)[kAccessErrKey].get<string>();
        auto cookiePath = (*cookie)[kCookiePathKey].get<string>();
        if(path.find(cookiePath) == 0){
412 413 414
            //上次cookie是限定本目录
            if(accessErr.empty()){
                //上次鉴权成功
415
                callback("", cookie_from_header ? nullptr : cookie);
416 417
                return;
            }
418
            //上次鉴权失败,但是如果url参数发生变更,那么也重新鉴权下
419
            if (_parser.Params().empty() || _parser.Params() == cookie->getUid()) {
420 421
                //url参数未变,或者本来就没有url参数,那么判断本次请求为重复请求,无访问权限
                callback(accessErr, cookie_from_header ? nullptr : cookie);
422 423
                return;
            }
424
        }
425
        //如果url参数变了或者不是限定本目录,那么旧cookie失效,重新鉴权
426
        HttpCookieManager::Instance().delCookie(cookie);
427 428 429 430
    }

    //该用户从来未获取过cookie,这个时候我们广播是否允许该用户访问该http目录
    weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
431 432 433 434
    HttpAccessPathInvoker accessPathInvoker = [weakSelf,callback,uid,path,is_dir] (const string &errMsg,const string &cookie_path_in, int cookieLifeSecond) {
        HttpServerCookie::Ptr cookie ;
        if(cookieLifeSecond) {
            //本次鉴权设置了有效期,我们把鉴权结果缓存在cookie中
xiongziliang committed
435 436 437 438 439 440
            string cookie_path = cookie_path_in;
            if(cookie_path.empty()){
                //如果未设置鉴权目录,那么我们采用当前目录
                cookie_path = is_dir ? path : path.substr(0,path.rfind("/") + 1);
            }

xiongziliang committed
441 442 443
            cookie = HttpCookieManager::Instance().addCookie(kCookieName, uid, cookieLifeSecond);
            //对cookie上锁
            auto lck = cookie->getLock();
444
            //记录用户能访问的路径
445
            (*cookie)[kCookiePathKey].set<string>(cookie_path);
446
            //记录能否访问
447
            (*cookie)[kAccessErrKey].set<string>(errMsg);
448 449
        }

450 451 452 453 454
        auto strongSelf = weakSelf.lock();
        if (!strongSelf) {
            //自己已经销毁
            return;
        }
455
        strongSelf->async([weakSelf,callback,cookie,errMsg]() {
456 457 458 459 460 461
            //切换到自己线程
            auto strongSelf = weakSelf.lock();
            if (!strongSelf) {
                //自己已经销毁
                return;
            }
462
            callback(errMsg, cookie);
463 464 465
        });
    };

466 467 468 469 470
    if(checkHls(_parser,_mediaInfo,path,is_dir,accessPathInvoker,*this)){
        //是hls的播放鉴权,拦截之
        return;
    }

471 472 473
    bool flag = NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastHttpAccess,_parser,_mediaInfo,path,is_dir,accessPathInvoker,*this);
    if(!flag){
        //此事件无人监听,我们默认都有权限访问
474
        callback("", nullptr);
475 476 477
    }

}
478

479
void HttpSession::Handle_Req_GET(int64_t &content_len) {
xiongziliang committed
480 481 482
	//先看看是否为WebSocket请求
	if(checkWebSocket()){
		content_len = -1;
483 484
		_contentCallBack = [this](const char *data,uint64_t len){
            WebSocketSplitter::decode((uint8_t *)data,len);
485
			//_contentCallBack是可持续的,后面还要处理后续数据
xiongziliang committed
486 487
			return true;
		};
xiongziliang committed
488
		return;
xiongziliang committed
489 490
	}

xiongziliang committed
491 492
	//先看看该http事件是否被拦截
	if(emitHttpEvent(false)){
xiongziliang committed
493
		return;
494
	}
xiongziliang committed
495

496
    //再看看是否为http-flv直播请求
497
	if(checkLiveFlvStream()){
3503207480@qq.com committed
498
		//若是,return!
xiongziliang committed
499
		return;
xiongziliang committed
500
	}
501

xiongziliang committed
502
	//事件未被拦截,则认为是http下载请求
503 504
	auto fullUrl = string(HTTP_SCHEMA) + "://" + _parser["Host"] + _parser.FullUrl();
    _mediaInfo.parse(fullUrl);
505

xiongziliang committed
506
	/////////////HTTP连接是否需要被关闭////////////////
507 508 509
    GET_CONFIG(uint32_t,reqCnt,Http::kMaxReqCount);
    GET_CONFIG(bool,enableVhost,General::kEnableVhost);
    GET_CONFIG(string,rootPath,Http::kRootPath);
510
    string strFile = enableVhost ?  rootPath + "/" + _mediaInfo._vhost + _parser.Url() :rootPath + _parser.Url();
xiongziliang committed
511
    bool bClose = (strcasecmp(_parser["Connection"].data(),"close") == 0) || ( ++_iReqCnt > reqCnt);
512 513 514 515 516 517 518 519

    do{
        //访问的是文件夹
        if (strFile.back() == '/' || File::is_dir(strFile.data())) {
            auto indexFile = findIndexFile(strFile);
            if(!indexFile.empty()){
                //发现该文件夹下有index文件
                strFile = strFile + "/" + indexFile;
520
                _parser.setUrl(_parser.Url() + "/" + indexFile);
521 522 523
                break;
            }
            string strMeun;
524
            //生成文件夹菜单索引
525
            if (!makeMeun(_parser.Url(),strFile,strMeun)) {
526 527
                //文件夹不存在
                sendNotFound(bClose);
xiongziliang committed
528
                throw SockException(bClose ? Err_shutdown : Err_success,"close connection after send 404 not found on folder");
529
            }
530 531

            //判断是否有权限访问该目录
xiongziliang committed
532
            canAccessPath(_parser.Url(),true,[this,bClose,strFile,strMeun](const string &errMsg,const HttpServerCookie::Ptr &cookie){
533 534
                if(!errMsg.empty()){
                    const_cast<string &>(strMeun) = errMsg;
535 536 537
                }
                auto headerOut = makeHttpHeader(bClose,strMeun.size());
                if(cookie){
538
                    headerOut["Set-Cookie"] = cookie->getCookie((*cookie)[kCookiePathKey].get<string>());
539
                }
540
                sendResponse(errMsg.empty() ? "200 OK" : "401 Unauthorized" , headerOut, strMeun);
541 542 543
                throw SockException(bClose ? Err_shutdown : Err_success,"close connection after access folder");
            });
            return;
544 545 546
        }
    }while(0);

xiongziliang committed
547
	//访问的是文件
xzl committed
548 549
	struct stat tFileStat;
	if (0 != stat(strFile.data(), &tFileStat)) {
xiongziliang committed
550
		//文件不存在
xzl committed
551
		sendNotFound(bClose);
xiongziliang committed
552
        throw SockException(bClose ? Err_shutdown : Err_success,"close connection after send 404 not found on file");
xzl committed
553
	}
554 555 556 557 558 559 560 561
    //文件智能指针,防止退出时未关闭
    std::shared_ptr<FILE> pFilePtr(fopen(strFile.data(), "rb"), [](FILE *pFile) {
        if(pFile){
            fclose(pFile);
        }
    });

	if (!pFilePtr) {
xiongziliang committed
562
		//打开文件失败
xzl committed
563
		sendNotFound(bClose);
xiongziliang committed
564
        throw SockException(bClose ? Err_shutdown : Err_success,"close connection after send 404 not found on open file failed");
xzl committed
565
	}
566

567 568
	auto parser = _parser;
    //判断是否有权限访问该文件
569 570 571 572
    canAccessPath(_parser.Url(),false,[this,parser,tFileStat,pFilePtr,bClose,strFile](const string &errMsg,const HttpServerCookie::Ptr &cookie){
        if(!errMsg.empty()){
            auto headerOut = makeHttpHeader(bClose,errMsg.size());
            if(cookie){
573
                headerOut["Set-Cookie"] = cookie->getCookie((*cookie)[kCookiePathKey].get<string>());
574 575 576 577 578
            }
            sendResponse("401 Unauthorized" , headerOut, errMsg);
            throw SockException(bClose ? Err_shutdown : Err_success,"close connection after access file failed");
        }

579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595
        //判断是不是分节下载
        auto &strRange = parser["Range"];
        int64_t iRangeStart = 0, iRangeEnd = 0;
        iRangeStart = atoll(FindField(strRange.data(), "bytes=", "-").data());
        iRangeEnd = atoll(FindField(strRange.data(), "-", "\r\n").data());
        if (iRangeEnd == 0) {
            iRangeEnd = tFileStat.st_size - 1;
        }
        const char *pcHttpResult = NULL;
        if (strRange.size() == 0) {
            //全部下载
            pcHttpResult = "200 OK";
        } else {
            //分节下载
            pcHttpResult = "206 Partial Content";
            fseek(pFilePtr.get(), iRangeStart, SEEK_SET);
        }
596
        auto httpHeader =  makeHttpHeader(bClose, iRangeEnd - iRangeStart + 1, get_mime_type(strFile.data()));
597 598 599 600 601
        if (strRange.size() != 0) {
            //分节下载返回Content-Range头
            httpHeader.emplace("Content-Range",StrPrinter<<"bytes " << iRangeStart << "-" << iRangeEnd << "/" << tFileStat.st_size<< endl);
        }

602 603 604
        if(cookie){
            httpHeader["Set-Cookie"] = cookie->getCookie((*cookie)[kCookiePathKey].get<string>());
        }
605
        //先回复HTTP头部分
606 607 608
        sendResponse(pcHttpResult,httpHeader,"");
        
        if (iRangeEnd - iRangeStart < 0) {
609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659
            //文件是空的!
            throw SockException(bClose ? Err_shutdown : Err_success,"close connection after access file");
        }
        //回复Content部分
        std::shared_ptr<int64_t> piLeft(new int64_t(iRangeEnd - iRangeStart + 1));

        GET_CONFIG(uint32_t,sendBufSize,Http::kSendBufSize);

        weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
        auto onFlush = [pFilePtr,bClose,weakSelf,piLeft]() {
            TimeTicker();
            auto strongSelf = weakSelf.lock();
            while(*piLeft && strongSelf){
                //更新超时定时器
                strongSelf->_ticker.resetTime();
                //从循环池获取一个内存片
                auto sendBuf = strongSelf->obtainBuffer();
                sendBuf->setCapacity(sendBufSize);
                //本次需要读取文件字节数
                int64_t iReq = MIN(sendBufSize,*piLeft);
                //读文件
                int iRead;
                do{
                    iRead = fread(sendBuf->data(), 1, iReq, pFilePtr.get());
                }while(-1 == iRead && UV_EINTR == get_uv_error(false));
                //文件剩余字节数
                *piLeft -= iRead;

                if (iRead < iReq || !*piLeft) {
                    //文件读完
                    if(iRead>0) {
                        sendBuf->setSize(iRead);
                        strongSelf->send(sendBuf);
                    }
                    if(bClose) {
                        strongSelf->shutdown(SockException(Err_shutdown,"read file eof"));
                    }
                    return false;
                }
                //文件还未读完
                sendBuf->setSize(iRead);
                int iSent = strongSelf->send(sendBuf);
                if(iSent == -1) {
                    //套机制销毁
                    return false;
                }
                if(strongSelf->isSocketBusy()){
                    //套接字忙,那么停止继续写
                    return true;
                }
                //继续写套接字
xiongziliang committed
660
            }
661 662
            return false;
        };
663 664 665

        //文件下载提升发送性能
        setSocketFlags();
666 667 668 669

        onFlush();
        _sock->setOnFlush(onFlush);
    });
xzl committed
670 671
}

672
bool makeMeun(const string &httpPath,const string &strFullPath, string &strRet) {
xzl committed
673
	string strPathPrefix(strFullPath);
674 675 676 677 678 679 680
	string last_dir_name;
	if(strPathPrefix.back() == '/'){
		strPathPrefix.pop_back();
	}else{
		last_dir_name = split(strPathPrefix,"/").back();
	}

xzl committed
681 682 683
	if (!File::is_dir(strPathPrefix.data())) {
		return false;
	}
684 685
	stringstream ss;
	ss <<   "<html>\r\n"
xzl committed
686 687 688 689 690 691
			"<head>\r\n"
			"<title>文件索引</title>\r\n"
			"</head>\r\n"
			"<body>\r\n"
			"<h1>文件索引:";

692
	ss << httpPath;
693
	ss << "</h1>\r\n";
694
	if (httpPath != "/") {
695 696 697 698 699 700 701 702 703 704 705 706 707 708 709
		ss << "<li><a href=\"";
		ss << "/";
		ss << "\">";
		ss << "根目录";
		ss << "</a></li>\r\n";

		ss << "<li><a href=\"";
		if(!last_dir_name.empty()){
			ss << "./";
		}else{
			ss << "../";
		}
		ss << "\">";
		ss << "上级目录";
		ss << "</a></li>\r\n";
xzl committed
710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726
	}

	DIR *pDir;
	dirent *pDirent;
	if ((pDir = opendir(strPathPrefix.data())) == NULL) {
		return false;
	}
	set<string> setFile;
	while ((pDirent = readdir(pDir)) != NULL) {
		if (File::is_special_dir(pDirent->d_name)) {
			continue;
		}
		if(pDirent->d_name[0] == '.'){
			continue;
		}
		setFile.emplace(pDirent->d_name);
	}
727
	int i = 0;
xzl committed
728 729
	for(auto &strFile :setFile ){
		string strAbsolutePath = strPathPrefix + "/" + strFile;
730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759
		bool isDir = File::is_dir(strAbsolutePath.data());
		ss << "<li><span>" << i++ << "</span>\t";
		ss << "<a href=\"";
		if(!last_dir_name.empty()){
			ss << last_dir_name << "/" << strFile;
		}else{
			ss << strFile;
		}

		if(isDir){
			ss << "/";
		}
		ss << "\">";
		ss << strFile;
		if (isDir) {
			ss << "/</a></li>\r\n";
			continue;
		}
		//是文件
		struct stat fileData;
		if (0 == stat(strAbsolutePath.data(), &fileData)) {
			auto &fileSize = fileData.st_size;
			if (fileSize < 1024) {
				ss << " (" << fileData.st_size << "B)" << endl;
			} else if (fileSize < 1024 * 1024) {
				ss << fixed << setprecision(2) << " (" << fileData.st_size / 1024.0 << "KB)";
			} else if (fileSize < 1024 * 1024 * 1024) {
				ss << fixed << setprecision(2) << " (" << fileData.st_size / 1024 / 1024.0 << "MB)";
			} else {
				ss << fixed << setprecision(2) << " (" << fileData.st_size / 1024 / 1024 / 1024.0 << "GB)";
xzl committed
760 761
			}
		}
762
		ss << "</a></li>\r\n";
xzl committed
763 764
	}
	closedir(pDir);
765 766 767
	ss << "<ul>\r\n";
	ss << "</ul>\r\n</body></html>";
	ss.str().swap(strRet);
xzl committed
768 769
	return true;
}
770 771

void HttpSession::sendResponse(const char* pcStatus, const KeyValue& header, const string& strContent) {
xzl committed
772
	_StrPrinter printer;
xiongziliang committed
773
	printer << "HTTP/1.1 " << pcStatus << "\r\n";
xzl committed
774 775 776 777 778 779
	for (auto &pr : header) {
		printer << pr.first << ": " << pr.second << "\r\n";
	}
	printer << "\r\n" << strContent;
	auto strSend = printer << endl;
	send(strSend);
780
	_ticker.resetTime();
xzl committed
781
}
782 783

HttpSession::KeyValue HttpSession::makeHttpHeader(bool bClose, int64_t iContentSize,const char* pcContentType) {
xzl committed
784
	KeyValue headerOut;
785 786 787
    GET_CONFIG(string,charSet,Http::kCharSet);
    GET_CONFIG(uint32_t,keepAliveSec,Http::kKeepAliveSecond);
    GET_CONFIG(uint32_t,reqCnt,Http::kMaxReqCount);
xzl committed
788

xiongziliang committed
789
	headerOut.emplace("Date", dateStr());
790
	headerOut.emplace("Server", SERVER_NAME);
xiongziliang committed
791
	headerOut.emplace("Connection", bClose ? "close" : "keep-alive");
xiongziliang committed
792 793 794 795
	if(!bClose){
		headerOut.emplace("Keep-Alive",StrPrinter << "timeout=" << keepAliveSec << ", max=" << reqCnt << endl);
	}
	if(pcContentType){
xzl committed
796
		auto strContentType = StrPrinter << pcContentType << "; charset=" << charSet << endl;
xiongziliang committed
797 798 799
		headerOut.emplace("Content-Type",strContentType);
	}
	if(iContentSize > 0){
xzl committed
800 801
		headerOut.emplace("Content-Length", StrPrinter<<iContentSize<<endl);
	}
xiongziliang committed
802 803 804 805 806

    if(!_origin.empty()){
        headerOut.emplace("Access-Control-Allow-Origin",_origin);
        headerOut.emplace("Access-Control-Allow-Credentials", "true");
    }
xzl committed
807 808 809
	return headerOut;
}

xiongziliang committed
810
string HttpSession::urlDecode(const string &str){
xiongziliang committed
811
	auto ret = strCoding::UrlDecode(str);
812
#ifdef _WIN32
813
    GET_CONFIG(string,charSet,Http::kCharSet);
xiongziliang committed
814
	bool isGb2312 = !strcasecmp(charSet.data(), "gb2312");
815
	if (isGb2312) {
xiongziliang committed
816
		ret = strCoding::UTF8ToGB2312(ret);
817 818
	}
#endif // _WIN32
xiongziliang committed
819 820
    return ret;
}
821

822
void HttpSession::urlDecode(Parser &parser){
xiongziliang committed
823
	parser.setUrl(urlDecode(parser.Url()));
824
	for(auto &pr : _parser.getUrlArgs()){
xiongziliang committed
825 826 827
		const_cast<string &>(pr.second) = urlDecode(pr.second);
	}
}
xzl committed
828

829
bool HttpSession::emitHttpEvent(bool doInvoke){
xiongziliang committed
830
	///////////////////是否断开本链接///////////////////////
831
    GET_CONFIG(uint32_t,reqCnt,Http::kMaxReqCount);
832

xiongziliang committed
833
    bool bClose = (strcasecmp(_parser["Connection"].data(),"close") == 0) || ( ++_iReqCnt > reqCnt);
xiongziliang committed
834
	/////////////////////异步回复Invoker///////////////////////////////
835
	weak_ptr<HttpSession> weakSelf = dynamic_pointer_cast<HttpSession>(shared_from_this());
xiongziliang committed
836
	HttpResponseInvoker invoker = [weakSelf,bClose](const string &codeOut, const KeyValue &headerOut, const string &contentOut){
837 838 839 840
		auto strongSelf = weakSelf.lock();
		if(!strongSelf) {
			return;
		}
xiongziliang committed
841
		strongSelf->async([weakSelf,bClose,codeOut,headerOut,contentOut]() {
842 843 844 845
			auto strongSelf = weakSelf.lock();
			if(!strongSelf) {
				return;
			}
xiongziliang committed
846
			strongSelf->responseDelay(bClose,codeOut,headerOut,contentOut);
847
			if(bClose){
xiongziliang committed
848
				strongSelf->shutdown(SockException(Err_shutdown,"Connection: close"));
849 850 851
			}
		});
	};
xiongziliang committed
852 853
	///////////////////广播HTTP事件///////////////////////////
	bool consumed = false;//该事件是否被消费
xiongziliang committed
854
	NoticeCenter::Instance().emitEvent(Broadcast::kBroadcastHttpRequest,_parser,invoker,consumed,*this);
xiongziliang committed
855 856
	if(!consumed && doInvoke){
		//该事件无人消费,所以返回404
857
		invoker("404 Not Found",KeyValue(),"");
858 859
		if(bClose){
			//close类型,回复完毕,关闭连接
xiongziliang committed
860
			shutdown(SockException(Err_shutdown,"404 Not Found"));
861
		}
xzl committed
862
	}
xiongziliang committed
863 864
	return consumed;
}
865 866

void HttpSession::Handle_Req_POST(int64_t &content_len) {
867 868
	GET_CONFIG(uint64_t,maxReqSize,Http::kMaxReqSize);
    GET_CONFIG(int,maxReqCnt,Http::kMaxReqCount);
869

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

872
	if(totalContentLen == 0){
873 874
		//content为空
		//emitHttpEvent内部会选择是否关闭连接
875
		emitHttpEvent(true);
xiongziliang committed
876
		return;
877 878
	}

879 880 881 882 883 884 885 886 887
    //根据Content-Length设置接收缓存大小
    if(totalContentLen > 0){
        _sock->setReadBuffer(std::make_shared<BufferRaw>(MIN(totalContentLen + 1,256 * 1024)));
    }else{
	    //不定长度的Content-Length
        _sock->setReadBuffer(std::make_shared<BufferRaw>(256 * 1024));
	}

    if(totalContentLen > 0 && totalContentLen < maxReqSize ){
888
		//返回固定长度的content
889
		content_len = totalContentLen;
890 891
		auto parserCopy = _parser;
		_contentCallBack = [this,parserCopy](const char *data,uint64_t len){
892
			//恢复http头
893
			_parser = parserCopy;
894
			//设置content
895
			_parser.setContent(string(data,len));
896
			//触发http事件,emitHttpEvent内部会选择是否关闭连接
897 898
			emitHttpEvent(true);
			//清空数据,节省内存
899
			_parser.Clear();
900
			//content已经接收完毕
901 902 903 904 905
			return false;
		};
	}else{
		//返回不固定长度的content
		content_len = -1;
906
		auto parserCopy = _parser;
907
		std::shared_ptr<uint64_t> recvedContentLen = std::make_shared<uint64_t>(0);
xiongziliang committed
908
		bool bClose = (strcasecmp(_parser["Connection"].data(),"close") == 0) || ( ++_iReqCnt > maxReqCnt);
909

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

913
		    onRecvUnlimitedContent(parserCopy,data,len,totalContentLen,*(recvedContentLen));
914 915 916

			if(*(recvedContentLen) < totalContentLen){
			    //数据还没接收完毕
917
                //_contentCallBack是可持续的,后面还要处理后续content数据
918 919 920 921 922 923 924 925
                return true;
			}

			//数据接收完毕
            if(!bClose){
			    //keep-alive类型连接
				//content接收完毕,后续都是http header
				setContentLen(0);
926
                //content已经接收完毕
927 928 929 930
                return false;
            }

            //连接类型是close类型,收完content就关闭连接
xiongziliang committed
931
            shutdown(SockException(Err_shutdown,"recv http content completed"));
932
            //content已经接收完毕
933
            return false ;
934 935
		};
	}
936
	//有后续content数据要处理,暂时不关闭连接
937
}
xiongziliang committed
938 939 940
void HttpSession::responseDelay(bool bClose,
                                const string &codeOut,
                                const KeyValue &headerOut,
xiongziliang committed
941
								const string &contentOut){
942 943 944 945
	if(codeOut.empty()){
		sendNotFound(bClose);
		return;
	}
xiongziliang committed
946
	auto headerOther = makeHttpHeader(bClose,contentOut.size(),"text/plain");
947 948 949
    for (auto &pr : headerOther){
        //添加默认http头,默认http头不能覆盖用户自定义的头
        const_cast<KeyValue &>(headerOut).emplace(pr.first,pr.second);
950
    }
951
	sendResponse(codeOut.data(), headerOut, contentOut);
xzl committed
952
}
953 954

void HttpSession::sendNotFound(bool bClose) {
955
    GET_CONFIG(string,notFound,Http::kNotFound);
956
    sendResponse("404 Not Found", makeHttpHeader(bClose, notFound.size()), notFound);
xzl committed
957 958
}

959 960 961 962 963 964 965 966 967
void HttpSession::setSocketFlags(){
    GET_CONFIG(bool,ultraLowDelay,General::kUltraLowDelay);
    if(!ultraLowDelay) {
        //推流模式下,关闭TCP_NODELAY会增加推流端的延时,但是服务器性能将提高
        SockUtil::setNoDelay(_sock->rawFD(), false);
        //播放模式下,开启MSG_MORE会增加延时,但是能提高发送性能
        (*this) << SocketFlags(SOCKET_DEFAULE_FLAGS | FLAG_MORE);
    }
}
xiongziliang committed
968 969

void HttpSession::onWrite(const Buffer::Ptr &buffer) {
970
	_ticker.resetTime();
971 972 973 974 975 976 977 978 979 980 981
    if(!_flv_over_websocket){
        _ui64TotalBytes += buffer->size();
        send(buffer);
        return;
    }

    WebSocketHeader header;
    header._fin = true;
    header._reserved = 0;
    header._opcode = WebSocketHeader::BINARY;
    header._mask_flag = false;
982
    WebSocketSplitter::encode(header,buffer);
983 984
}

985 986 987
void HttpSession::onWebSocketEncodeData(const Buffer::Ptr &buffer){
    _ui64TotalBytes += buffer->size();
    send(buffer);
xiongziliang committed
988 989
}

xiongziliang committed
990
void HttpSession::onDetach() {
xiongziliang committed
991
	shutdown(SockException(Err_shutdown,"rtmp ring buffer detached"));
xiongziliang committed
992 993 994 995
}

std::shared_ptr<FlvMuxer> HttpSession::getSharedPtr(){
	return dynamic_pointer_cast<FlvMuxer>(shared_from_this());
996 997
}

xiongziliang committed
998
} /* namespace mediakit */