communicator.h 2.9 KB
Newer Older
羽飞's avatar
羽飞 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 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 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107
/* Copyright (c) 2021 Xie Meiyi(xiemeiyi@hust.edu.cn) and OceanBase and/or its affiliates. All rights reserved.
miniob is licensed under Mulan PSL v2.
You can use this software according to the terms and conditions of the Mulan PSL v2.
You may obtain a copy of Mulan PSL v2 at:
         http://license.coscl.org.cn/MulanPSL2
THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
See the Mulan PSL v2 for more details. */

//
// Created by Wangyunlai on 2022/11/17.
//

#pragma once

#include <string>
#include <event.h>
#include "rc.h"

struct ConnectionContext;
class SessionEvent;
class Session;

/**
 * 负责与客户端通讯
 *
 * 在listener接收到一个新的连接(参考 server.cpp::accept), 就创建一个Communicator对象。
 * 并调用init进行初始化。
 * 在server中监听到某个连接有新的消息,就通过Communicator::read_event接收消息。
 
 */
class Communicator {
public: 
  virtual ~Communicator();

  /**
   * 接收到一个新的连接时,进行初始化
   */
  virtual RC init(int fd, Session *session, const std::string &addr);

  /**
   * 监听到有新的数据到达,调用此函数进行接收消息
   * 如果需要创建新的任务来处理,那么就创建一个SessionEvent 对象并通过event参数返回。
   */
  virtual RC read_event(SessionEvent *&event) = 0;

  /**
   * 在任务处理完成后,通过此接口将结果返回给客户端
   * @param event 任务数据,包括处理的结果
   * @param need_disconnect 是否需要断开连接
   * @return 处理结果。即使返回不是SUCCESS,也不能直接断开连接,需要通过need_disconnect来判断
   *         是否需要断开连接
   */
  virtual RC write_result(SessionEvent *event, bool &need_disconnect) = 0;

  /**
   * 关联的会话信息
   */
  Session *session() const { return session_; }

  /**
   * libevent使用的数据,参考server.cpp
   */
  struct event &read_event() { return read_event_; }

  /**
   * 对端地址
   * 如果是unix socket,可能没有意义
   */
  const char *addr() const { return addr_.c_str(); }

protected:
  Session *session_ = nullptr;
  int fd_ = -1;
  struct event read_event_;
  std::string addr_;
};

/**
 * 与客户端进行通讯
 * 使用简单的文本通讯协议,每个消息使用'\0'结尾
 */
class PlainCommunicator : public Communicator {
public: 
  RC read_event(SessionEvent *&event) override;
  RC write_result(SessionEvent *event, bool &need_disconnect) override;

private:
  RC write_state(SessionEvent *event, bool &need_disconnect);

};

/**
 * 当前支持的通讯协议
 */
enum class CommunicateProtocol
{
  PLAIN,  //! 以'\0'结尾的协议
  MYSQL,  //! mysql通讯协议。具体实现参考 MysqlCommunicator
};

class CommunicatorFactory
{
public: 
  Communicator *create(CommunicateProtocol protocol);
};