blob: 5edbd576fb63c2ad97d2098a4927133cfedbdd4d [file] [log] [blame]
Alexander Afanasyeva9034b02014-01-26 18:32:02 -08001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Junxiao Shi2d491752017-07-14 21:32:05 +00002/*
Davide Pesaventod91fe6d2023-10-04 21:40:02 -04003 * Copyright (c) 2014-2023, Regents of the University of California,
Chengyu Fan4381fb62015-01-14 11:37:04 -07004 * Arizona Board of Regents,
5 * Colorado State University,
6 * University Pierre & Marie Curie, Sorbonne University,
7 * Washington University in St. Louis,
8 * Beijing Institute of Technology,
9 * The University of Memphis.
Alexander Afanasyev9bcbc7c2014-04-06 19:37:37 -070010 *
11 * This file is part of NFD (Named Data Networking Forwarding Daemon).
12 * See AUTHORS.md for complete list of NFD authors and contributors.
13 *
14 * NFD is free software: you can redistribute it and/or modify it under the terms
15 * of the GNU General Public License as published by the Free Software Foundation,
16 * either version 3 of the License, or (at your option) any later version.
17 *
18 * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
19 * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
20 * PURPOSE. See the GNU General Public License for more details.
21 *
22 * You should have received a copy of the GNU General Public License along with
23 * NFD, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
Steve DiBenedettoef04f272014-06-04 14:28:31 -060024 */
Alexander Afanasyeva9034b02014-01-26 18:32:02 -080025
Alexander Afanasyev0eb70652014-02-27 18:35:07 -080026#include "tcp-factory.hpp"
Alexander Afanasyeva9034b02014-01-26 18:32:02 -080027
Davide Pesaventoa9b09b62022-06-04 14:07:25 -040028#include <boost/lexical_cast.hpp>
29
Davide Pesaventoe422f9e2022-06-03 01:30:23 -040030namespace nfd::face {
Alexander Afanasyeva9034b02014-01-26 18:32:02 -080031
Davide Pesavento1d7e7af2015-10-10 23:54:08 +020032namespace ip = boost::asio::ip;
Alexander Afanasyev70aaf8a2014-12-13 00:44:22 -080033
Davide Pesaventoa3148082018-04-12 18:21:54 -040034NFD_LOG_INIT(TcpFactory);
Junxiao Shib47247d2017-01-24 15:09:16 +000035NFD_REGISTER_PROTOCOL_FACTORY(TcpFactory);
36
37const std::string&
Davide Pesaventod27841b2018-11-13 00:22:24 -050038TcpFactory::getId() noexcept
Junxiao Shib47247d2017-01-24 15:09:16 +000039{
40 static std::string id("tcp");
41 return id;
42}
Steve DiBenedettoca53ac62014-03-27 19:58:40 -060043
44void
Davide Pesaventod27841b2018-11-13 00:22:24 -050045TcpFactory::doProcessConfig(OptionalConfigSection configSection,
46 FaceSystem::ConfigContext& context)
Steve DiBenedettoca53ac62014-03-27 19:58:40 -060047{
Junxiao Shi38b24c72017-01-05 02:59:31 +000048 // tcp
49 // {
50 // listen yes
51 // port 6363
52 // enable_v4 yes
53 // enable_v6 yes
54 // }
55
Eric Newberry0c841642018-01-17 15:01:00 -070056 m_wantCongestionMarking = context.generalConfig.wantCongestionMarking;
57
Junxiao Shi38b24c72017-01-05 02:59:31 +000058 if (!configSection) {
59 if (!context.isDryRun && !m_channels.empty()) {
Davide Pesaventod27841b2018-11-13 00:22:24 -050060 NFD_LOG_WARN("Cannot disable TCP channels after initialization");
Junxiao Shi38b24c72017-01-05 02:59:31 +000061 }
62 return;
Davide Pesavento1d7e7af2015-10-10 23:54:08 +020063 }
Steve DiBenedettoca53ac62014-03-27 19:58:40 -060064
Junxiao Shi38b24c72017-01-05 02:59:31 +000065 bool wantListen = true;
66 uint16_t port = 6363;
67 bool enableV4 = true;
68 bool enableV6 = true;
Alexander Afanasyevded17422018-04-03 19:00:23 -040069 IpAddressPredicate local;
70 bool isLocalConfigured = false;
Steve DiBenedettoca53ac62014-03-27 19:58:40 -060071
Junxiao Shi38b24c72017-01-05 02:59:31 +000072 for (const auto& pair : *configSection) {
73 const std::string& key = pair.first;
74
75 if (key == "listen") {
76 wantListen = ConfigFile::parseYesNo(pair, "face_system.tcp");
77 }
78 else if (key == "port") {
79 port = ConfigFile::parseNumber<uint16_t>(pair, "face_system.tcp");
80 }
81 else if (key == "enable_v4") {
82 enableV4 = ConfigFile::parseYesNo(pair, "face_system.tcp");
83 }
84 else if (key == "enable_v6") {
85 enableV6 = ConfigFile::parseYesNo(pair, "face_system.tcp");
86 }
Alexander Afanasyevded17422018-04-03 19:00:23 -040087 else if (key == "local") {
88 isLocalConfigured = true;
89 for (const auto& localPair : pair.second) {
90 const std::string& localKey = localPair.first;
91 if (localKey == "whitelist") {
92 local.parseWhitelist(localPair.second);
93 }
94 else if (localKey == "blacklist") {
95 local.parseBlacklist(localPair.second);
96 }
97 else {
Davide Pesavento19779d82019-02-14 13:40:04 -050098 NDN_THROW(ConfigFile::Error("Unrecognized option face_system.tcp.local." + localKey));
Alexander Afanasyevded17422018-04-03 19:00:23 -040099 }
100 }
101 }
Junxiao Shi38b24c72017-01-05 02:59:31 +0000102 else {
Davide Pesavento19779d82019-02-14 13:40:04 -0500103 NDN_THROW(ConfigFile::Error("Unrecognized option face_system.tcp." + key));
Steve DiBenedettoca53ac62014-03-27 19:58:40 -0600104 }
Davide Pesaventob499a602014-11-18 22:36:56 +0100105 }
Alexander Afanasyevded17422018-04-03 19:00:23 -0400106 if (!isLocalConfigured) {
107 local.assign({{"subnet", "127.0.0.0/8"}, {"subnet", "::1/128"}}, {});
108 }
Steve DiBenedettoca53ac62014-03-27 19:58:40 -0600109
Junxiao Shi38b24c72017-01-05 02:59:31 +0000110 if (!enableV4 && !enableV6) {
Davide Pesavento19779d82019-02-14 13:40:04 -0500111 NDN_THROW(ConfigFile::Error(
Junxiao Shi38b24c72017-01-05 02:59:31 +0000112 "IPv4 and IPv6 TCP channels have been disabled. Remove face_system.tcp section to disable "
113 "TCP channels or enable at least one channel type."));
114 }
115
Davide Pesaventod27841b2018-11-13 00:22:24 -0500116 if (context.isDryRun) {
117 return;
Davide Pesaventob499a602014-11-18 22:36:56 +0100118 }
Davide Pesaventod27841b2018-11-13 00:22:24 -0500119
120 providedSchemes.insert("tcp");
121
122 if (enableV4) {
123 tcp::Endpoint endpoint(ip::tcp::v4(), port);
124 auto v4Channel = this->createChannel(endpoint);
125 if (wantListen && !v4Channel->isListening()) {
126 v4Channel->listen(this->addFace, nullptr);
127 }
128 providedSchemes.insert("tcp4");
129 }
130 else if (providedSchemes.count("tcp4") > 0) {
131 NFD_LOG_WARN("Cannot close tcp4 channel after its creation");
132 }
133
134 if (enableV6) {
135 tcp::Endpoint endpoint(ip::tcp::v6(), port);
136 auto v6Channel = this->createChannel(endpoint);
137 if (wantListen && !v6Channel->isListening()) {
138 v6Channel->listen(this->addFace, nullptr);
139 }
140 providedSchemes.insert("tcp6");
141 }
142 else if (providedSchemes.count("tcp6") > 0) {
143 NFD_LOG_WARN("Cannot close tcp6 channel after its creation");
144 }
145
146 m_local = std::move(local);
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800147}
148
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800149void
Davide Pesaventod27841b2018-11-13 00:22:24 -0500150TcpFactory::doCreateFace(const CreateFaceRequest& req,
151 const FaceCreatedCallback& onCreated,
152 const FaceCreationFailedCallback& onFailure)
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800153{
Davide Pesavento15b55052018-01-27 19:09:28 -0500154 if (req.localUri) {
Davide Pesaventoc5a9f812023-10-17 18:01:52 -0400155 NFD_LOG_TRACE("createFace: unsupported LocalUri");
Eric Newberry78e32b02017-04-01 14:34:44 +0000156 onFailure(406, "Unicast TCP faces cannot be created with a LocalUri");
157 return;
158 }
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200159
Davide Pesavento15b55052018-01-27 19:09:28 -0500160 if (req.params.persistency == ndn::nfd::FACE_PERSISTENCY_ON_DEMAND) {
Davide Pesaventoc5a9f812023-10-17 18:01:52 -0400161 NFD_LOG_TRACE("createFace: unsupported FacePersistency");
Davide Pesaventoa3c9ddb2017-04-10 22:15:24 -0400162 onFailure(406, "Outgoing TCP faces do not support on-demand persistency");
Eric Newberry42602412016-08-27 09:33:18 -0700163 return;
Yukai Tu7c90e6d2015-07-11 12:21:46 +0800164 }
165
Davide Pesaventod91fe6d2023-10-04 21:40:02 -0400166 tcp::Endpoint endpoint(ip::make_address(req.remoteUri.getHost()),
Davide Pesavento15b55052018-01-27 19:09:28 -0500167 boost::lexical_cast<uint16_t>(req.remoteUri.getPort()));
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800168
Davide Pesavento46afec42017-05-28 14:28:47 -0400169 // a canonical tcp4/tcp6 FaceUri cannot have a multicast address
170 BOOST_ASSERT(!endpoint.address().is_multicast());
Eric Newberry42602412016-08-27 09:33:18 -0700171
Davide Pesavento15b55052018-01-27 19:09:28 -0500172 if (req.params.wantLocalFields && !endpoint.address().is_loopback()) {
Davide Pesaventoc5a9f812023-10-17 18:01:52 -0400173 NFD_LOG_TRACE("createFace: cannot create non-local face with local fields enabled");
Eric Newberryf40551a2016-09-05 15:41:16 -0700174 onFailure(406, "Local fields can only be enabled on faces with local scope");
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200175 return;
176 }
Steve DiBenedettoca53ac62014-03-27 19:58:40 -0600177
Eric Newberry812d6152018-06-06 15:06:01 -0700178 if (req.params.mtu) {
Davide Pesaventoc5a9f812023-10-17 18:01:52 -0400179 NFD_LOG_TRACE("createFace: cannot create TCP face with overridden MTU");
Eric Newberry812d6152018-06-06 15:06:01 -0700180 onFailure(406, "TCP faces do not support MTU overrides");
181 return;
182 }
183
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800184 // very simple logic for now
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200185 for (const auto& i : m_channels) {
186 if ((i.first.address().is_v4() && endpoint.address().is_v4()) ||
187 (i.first.address().is_v6() && endpoint.address().is_v6())) {
Davide Pesavento15b55052018-01-27 19:09:28 -0500188 i.second->connect(endpoint, req.params, onCreated, onFailure);
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200189 return;
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800190 }
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200191 }
192
Davide Pesavento46afec42017-05-28 14:28:47 -0400193 NFD_LOG_TRACE("No channels available to connect to " << endpoint);
Eric Newberryf40551a2016-09-05 15:41:16 -0700194 onFailure(504, "No channels available to connect");
Alexander Afanasyevd6655302014-02-28 08:41:28 -0800195}
196
Junxiao Shi38b24c72017-01-05 02:59:31 +0000197shared_ptr<TcpChannel>
198TcpFactory::createChannel(const tcp::Endpoint& endpoint)
199{
Davide Pesavento4b89a6e2017-10-07 15:29:50 -0400200 auto it = m_channels.find(endpoint);
201 if (it != m_channels.end())
202 return it->second;
Junxiao Shi38b24c72017-01-05 02:59:31 +0000203
Davide Pesavento412c9822021-07-02 00:21:05 -0400204 auto channel = make_shared<TcpChannel>(endpoint, m_wantCongestionMarking, [this] (auto&&... args) {
205 return determineFaceScopeFromAddresses(std::forward<decltype(args)>(args)...);
206 });
Junxiao Shi38b24c72017-01-05 02:59:31 +0000207 m_channels[endpoint] = channel;
Junxiao Shi38b24c72017-01-05 02:59:31 +0000208 return channel;
209}
210
Davide Pesavento1d7e7af2015-10-10 23:54:08 +0200211std::vector<shared_ptr<const Channel>>
Davide Pesaventod27841b2018-11-13 00:22:24 -0500212TcpFactory::doGetChannels() const
Steve DiBenedettoef04f272014-06-04 14:28:31 -0600213{
Junxiao Shi38b24c72017-01-05 02:59:31 +0000214 return getChannelsFromMap(m_channels);
Steve DiBenedettoef04f272014-06-04 14:28:31 -0600215}
216
Alexander Afanasyevded17422018-04-03 19:00:23 -0400217ndn::nfd::FaceScope
Davide Pesaventoc0df94e2023-10-08 19:26:55 -0400218TcpFactory::determineFaceScopeFromAddresses(const ip::address& localAddress,
219 const ip::address& remoteAddress) const
Alexander Afanasyevded17422018-04-03 19:00:23 -0400220{
221 if (m_local(localAddress) && m_local(remoteAddress)) {
222 return ndn::nfd::FACE_SCOPE_LOCAL;
223 }
224 return ndn::nfd::FACE_SCOPE_NON_LOCAL;
225}
226
Davide Pesaventoe422f9e2022-06-03 01:30:23 -0400227} // namespace nfd::face