communicator.h 2.9 KB
Newer Older
羽飞's avatar
羽飞 已提交
1
/* Copyright (c) 2021 OceanBase and/or its affiliates. All rights reserved.
羽飞's avatar
羽飞 已提交
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
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接收消息。
L
Longda Feng 已提交
31

羽飞's avatar
羽飞 已提交
32 33
 */
class Communicator {
L
Longda Feng 已提交
34
public:
羽飞's avatar
羽飞 已提交
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
  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;

  /**
   * 关联的会话信息
   */
L
Longda Feng 已提交
60 61 62 63
  Session *session() const
  {
    return session_;
  }
羽飞's avatar
羽飞 已提交
64 65 66 67

  /**
   * libevent使用的数据,参考server.cpp
   */
L
Longda Feng 已提交
68 69 70 71
  struct event &read_event()
  {
    return read_event_;
  }
羽飞's avatar
羽飞 已提交
72 73 74 75 76

  /**
   * 对端地址
   * 如果是unix socket,可能没有意义
   */
L
Longda Feng 已提交
77 78 79 80
  const char *addr() const
  {
    return addr_.c_str();
  }
羽飞's avatar
羽飞 已提交
81 82 83 84 85 86 87 88 89 90 91 92 93

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

/**
 * 与客户端进行通讯
 * 使用简单的文本通讯协议,每个消息使用'\0'结尾
 */
class PlainCommunicator : public Communicator {
L
Longda Feng 已提交
94
public:
羽飞's avatar
羽飞 已提交
95 96 97 98 99 100 101 102 103 104
  RC read_event(SessionEvent *&event) override;
  RC write_result(SessionEvent *event, bool &need_disconnect) override;

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

/**
 * 当前支持的通讯协议
 */
L
Longda Feng 已提交
105
enum class CommunicateProtocol {
羽飞's avatar
羽飞 已提交
106 107 108 109
  PLAIN,  //! 以'\0'结尾的协议
  MYSQL,  //! mysql通讯协议。具体实现参考 MysqlCommunicator
};

L
Longda Feng 已提交
110 111
class CommunicatorFactory {
public:
羽飞's avatar
羽飞 已提交
112 113
  Communicator *create(CommunicateProtocol protocol);
};