summaryrefslogtreecommitdiffstats
path: root/apps/http-proxy/src/ATSConnector.h
blob: 8d91b7b7bd768b16a7666c627d8a200136861801 (plain)
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
108
109
/*
 * Copyright (c) 2019 Cisco and/or its affiliates.
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at:
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#pragma once

#include <hicn/transport/core/packet.h>

#define ASIO_STANDALONE
#include <asio.hpp>
#include <deque>
#include <functional>

namespace transport {

using asio::ip::tcp;

typedef std::function<void(const uint8_t *data, std::size_t size, bool is_last,
                           bool headers)>
    ContentReceivedCallback;
typedef std::function<void()> OnReconnect;
typedef std::function<void()> ContentSentCallback;
typedef std::deque<
    std::pair<std::unique_ptr<utils::MemBuf>, ContentSentCallback>>
    BufferQueue;

class ATSConnector {
  static constexpr uint32_t buffer_size = 1024 * 512;

  enum class ConnectorState {
    CLOSED,
    CONNECTING,
    CONNECTED,
  };

 public:
  ATSConnector(asio::io_service &io_service, std::string &ip_address,
               std::string &port, ContentReceivedCallback receive_callback,
               OnReconnect on_reconnect_callback);

  ~ATSConnector();

  void send(const uint8_t *buffer, std::size_t len,
            ContentSentCallback &&content_sent = 0);

  void send(utils::MemBuf *buffer, ContentSentCallback &&content_sent);

  void close();

 private:
  void doConnect();

  void doReadHeader();

  void doReadBody(std::size_t body_size, std::size_t additional_bytes);

  // void handleReadChunked(std::error_code ec, std::size_t length,
  //                        std::size_t size);

  void doReadChunkedHeader();

  void doWrite();

  bool checkConnected();

 private:
  void handleRead(std::error_code ec, std::size_t length);
  void tryReconnection();
  void startConnectionTimer();
  void handleDeadline(const std::error_code &ec);

  asio::io_service &io_service_;
  asio::ip::tcp::socket socket_;
  asio::ip::tcp::resolver resolver_;
  asio::ip::tcp::resolver::iterator endpoint_iterator_;
  asio::steady_timer timer_;

  BufferQueue write_msgs_;

  asio::streambuf input_buffer_;

  bool is_reconnection_;
  bool data_available_;

  std::size_t content_length_;

  // Chunked encoding
  bool is_last_chunk_;
  bool chunked_;

  ContentReceivedCallback receive_callback_;
  OnReconnect on_reconnect_callback_;

  // Connector state
  ConnectorState state_;
};

}  // namespace transport