WebRtcSession.cpp 3.32 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13
/*
 * Copyright (c) 2016 The ZLMediaKit project authors. All Rights Reserved.
 *
 * This file is part of ZLMediaKit(https://github.com/xia-chu/ZLMediaKit).
 *
 * 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.
 */

#include "WebRtcSession.h"
#include "Util/util.h"

14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
static string getUserName(const Buffer::Ptr &buffer) {
    auto buf = buffer->data();
    auto len = buffer->size();
    if (!RTC::StunPacket::IsStun((const uint8_t *) buf, len)) {
        return "";
    }
    std::unique_ptr<RTC::StunPacket> packet(RTC::StunPacket::Parse((const uint8_t *) buf, len));
    if (!packet) {
        return "";
    }
    if (packet->GetClass() != RTC::StunPacket::Class::REQUEST ||
        packet->GetMethod() != RTC::StunPacket::Method::BINDING) {
        return "";
    }
    //收到binding request请求
    auto vec = split(packet->GetUsername(), ":");
    return vec[0];
}

ziyue committed
33
EventPoller::Ptr WebRtcSession::queryPoller(const Buffer::Ptr &buffer) {
34 35 36 37
    auto user_name = getUserName(buffer);
    if (user_name.empty()) {
        return nullptr;
    }
ziyue committed
38
    auto ret = WebRtcTransportManager::Instance().getItem(user_name);
39 40 41
    return ret ? ret->getPoller() : nullptr;
}

ziyue committed
42 43 44 45 46 47 48 49 50 51 52
////////////////////////////////////////////////////////////////////////////////

WebRtcSession::WebRtcSession(const Socket::Ptr &sock) : UdpSession(sock) {
    socklen_t addr_len = sizeof(_peer_addr);
    getpeername(sock->rawFD(), &_peer_addr, &addr_len);
}

WebRtcSession::~WebRtcSession() {
    InfoP(this);
}

53
void WebRtcSession::onRecv(const Buffer::Ptr &buffer) {
ziyue committed
54 55 56
    if (_find_transport) {
        //只允许寻找一次transport
        _find_transport = false;
57 58
        auto user_name = getUserName(buffer);
        _identifier = user_name + '-' + to_string(reinterpret_cast<uint64_t>(this));
ziyue committed
59
        auto transport = WebRtcTransportManager::Instance().getItem(user_name);
60 61 62
        CHECK(transport && transport->getPoller()->isCurrentThread());
        transport->setSession(shared_from_this());
        _transport = std::move(transport);
63
        InfoP(this);
64
    }
65
    _ticker.resetTime();
ziyue committed
66
    CHECK(_transport);
ziyue committed
67
    _transport->inputSockData(buffer->data(), buffer->size(), &_peer_addr);
68 69 70
}

void WebRtcSession::onError(const SockException &err) {
71 72 73 74
    //udp链接超时,但是rtc链接不一定超时,因为可能存在udp链接迁移的情况
    //在udp链接迁移时,新的WebRtcSession对象将接管WebRtcTransport对象的生命周期
    //本WebRtcSession对象将在超时后自动销毁
    WarnP(this) << err.what();
ziyue committed
75

ziyue committed
76 77 78
    if (!_transport) {
        return;
    }
ziyue committed
79
    auto transport = std::move(_transport);
80
    getPoller()->async([transport] {
ziyue committed
81 82
        //延时减引用,防止使用transport对象时,销毁对象
    }, false);
83 84 85
}

void WebRtcSession::onManager() {
86 87 88 89 90 91 92 93 94
    GET_CONFIG(float, timeoutSec, RTC::kTimeOutSec);
    if (!_transport && _ticker.createdTime() > timeoutSec * 1000) {
        shutdown(SockException(Err_timeout, "illegal webrtc connection"));
        return;
    }
    if (_ticker.elapsedTime() > timeoutSec * 1000) {
        shutdown(SockException(Err_timeout, "webrtc connection timeout"));
        return;
    }
95
}
96 97 98 99 100

std::string WebRtcSession::getIdentifier() const {
    return _identifier;
}