【FileZilla】 从事件类型到消息类型的函数形参类型转化

本篇其实是前篇【Filezilla】 dispatch函数重载的例子-CSDN博客的一个补充,其中涉及到【FileZilla】事件调用机制代码解析-CSDN博客中的事件分发机制时钩子函数的参数传递怎么实现的。跟【FileZilla】sftp协议的数据传输上传和下载-CSDN博客同样,用事件是 CSftpEvent 即 send_event<T> () 或者说pending_events.emplace_back(handler, evt, deletable ) 调用时的T和evt,开始。

我们的问题是事件处理函数 CSftpControlSocket::OnSftpEvent(sftp_message const& message) 中的是怎么得到的消息message的?

// In event_loop.hpp
typedef std::deque<std::tuple<event_handler*, event_base*, bool>> Events;
    // Events::value_type ev{};

// In event.h
typedef fz::simple_event<sftp_event_type, sftp_message> CSftpEvent;
 
 
// In sftpcontrolsocket.cpp
void CSftpControlSocket::operator()(fz::event_base const& ev)
{
	if (fz::dispatch<fz::process_event, CSftpEvent, CSftpListEvent, SftpRateAvailableEvent>(ev, this,
		&CSftpControlSocket::OnProcessEvent,
		&CSftpControlSocket::OnSftpEvent,
		&CSftpControlSocket::OnSftpListEvent,
		&CSftpControlSocket::OnQuotaRequest)) {
		return;
	}
 
	CControlSocket::operator()(ev);
}

// In sftpcontrolsocket.cpp
void CSftpControlSocket::OnSftpEvent(sftp_message const& message)
{
    ......
    switch (message.type)
    {
        ......
        case sftpEvent::Recv:......break;
        case sftpEvent::Send:......break;
        case sftpEvent::Transfer:......break;
......
        case sftpEvent::io_nextbuf:......break;
        case sftpEvent::io_open:......break;
        case sftpEvent::io_size:......break;
        case sftpEvent::io_finalize:......break;
        default:......
    }
    ......
}

类似【FileZilla】事件调用机制代码解析-CSDN博客,在SftpInputParser::OnData()函数中 event_ = std::make_unique<CSftpEvent>() 随后调用事件入队列 owner_.send_event(event_.release()) ,以下式子逐步等价:

  • owner_.send_event(event_.release())
  • event_loop.send_event(this, event_.release(), true),this 指针指向子类CSftpControlSocket的实例
  • 实际调用:pending_events_.emplace_back(handler,evt,deletable),handler就是this指针、evt就是CSftpEvent,event_.release()就是从智能指针中释放裸指针,交出所有权

其中owner_是CSftpControlSocket类型,CSftpControlSocket继承自 CControlSocket ,它继承自event_handler,公有父类实现了 send_event() 方法。于是,当调用entry()现成调process_event()时,语句 (*std::get<0>(ev))(*std::get<1>(ev)) 时,相当于以下式子逐步“等价”:

  • (*std::get<0>(ev))(*std::get<1>(ev))
  •  (*handler)(evt)
  • handler->operator()(evt) 
  • CSftpControlSocket的实例->operator()( CSftpEvent * ev ),operator实际接收了event_base类型
  • CSftpControlSocket::operator()(fz::event_base const& ev)

于是从【FileZilla】sftp协议的数据传输上传和下载-CSDN博客中,我们知道事件分发后会调用CSftpControlSocket::OnSftpEvent(sftp_message const& message)。这时我们发现根据【Filezilla】 dispatch函数重载的例子-CSDN博客中dispatch重载的代码handler函数传入的参数是CSftpEvent * ev,如何变成了一个 sftp_message const& message ?

注意到CSftpEvent是一个分支上fz::simple_event模版类(在第一段代码里),而simple_event的解析如下。关键是因为 CSftpEvent ::v_ 是一个 std::tuple<sftp_message>fz::apply() 会自动解包为sftp_message。

template<typename UniqueType, typename...Values>
class simple_event final : public event_base
{
public:
	typedef UniqueType unique_type;
	typedef std::tuple<Values...> tuple_type;

	using event_base::event_base;

	template<typename First_Value, typename...Remaining_Values>
	explicit simple_event(First_Value&& value, Remaining_Values&& ...values)
	    : v_(std::forward<First_Value>(value), std::forward<Remaining_Values>(values)...)
	{
	}

	/// \brief Returns a unique id for the type such that can be used directly in derived_type.
	inline static size_t type() {
		// Exporting templates from DLLs is problematic to say the least. It breaks
		// ODR, so we use this trick that goes over the type name.
		static size_t const v = get_unique_type_id(typeid(UniqueType*));
		return v;
	}

	/// \brief Simply returns \ref type()
	virtual size_t derived_type() const override {
		return type();
	}

	/** \brief The event value, gets built from the arguments passed in the constructor.
	 *
	 * You don't need to access this member directly if you use the \ref dispatch mechanism.
	 */
	mutable tuple_type v_;
};

这是一个泛型事件类,它:

  • 继承自 event_base(这是 FileZilla 所有事件的基类);

  • 用模板参数构造出不同的事件类型;

  • std::tuple<Values...> 存储事件内容;

  • UniqueType 生成类型唯一标识。

你可以理解为:这是 C++ 中的 type-safe 事件封装器。

好啦,这就解释清楚了消息message是怎么来的,这对于sftp协议也是很重要的一点。在处理sftp事件时,FileZilla和fzsftp通信时利用了下面的函数:

static int fznotify(sftpEventTypes type)
{
	fprintf(stdout, "%c", (int)type + '0');
	fflush(stdout);
	return 0;
}
static int fznotify1(sftpEventTypes type, int data)
{
	fprintf(stdout, "%c%d\n", (int)type + '0', data);
	fflush(stdout);
	return 0;
}

其中的sftpEventType就是message.type的类型啦

enum class sftpEvent {
	Unknown = -1,
	Reply = 0,
	Done,
	Error,
	Verbose,
	Info,
	Status,
	Recv,
	Send,
	Listentry,
	AskHostkey,
	AskHostkeyChanged,
	AskHostkeyBetteralg,
	AskPassword,
	Transfer,
	RequestPreamble,
	RequestInstruction,
	UsedQuotaRecv,
	UsedQuotaSend,
	KexAlgorithm,
	KexHash,
	KexCurve,
	CipherClientToServer,
	CipherServerToClient,
	MacClientToServer,
	MacServerToClient,
	Hostkey,
	io_size,
	io_open,
	io_nextbuf,
	io_finalize,

	count
};

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Emilin Amy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值