/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */ /** * Copyright (c) 2014-2017, Regents of the University of California, * Arizona Board of Regents, * Colorado State University, * University Pierre & Marie Curie, Sorbonne University, * Washington University in St. Louis, * Beijing Institute of Technology, * The University of Memphis. * * This file is part of NFD (Named Data Networking Forwarding Daemon). * See AUTHORS.md for complete list of NFD authors and contributors. * * NFD is free software: you can redistribute it and/or modify it under the terms * of the GNU General Public License as published by the Free Software Foundation, * either version 3 of the License, or (at your option) any later version. * * NFD is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; * without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR * PURPOSE. See the GNU General Public License for more details. * * You should have received a copy of the GNU General Public License along with * NFD, e.g., in COPYING.md file. If not, see . */ #include "websocket-factory.hpp" #include "core/logger.hpp" namespace nfd { namespace face { namespace ip = boost::asio::ip; NFD_LOG_INIT("WebSocketFactory"); NFD_REGISTER_PROTOCOL_FACTORY(WebSocketFactory); const std::string& WebSocketFactory::getId() { static std::string id("websocket"); return id; } void WebSocketFactory::processConfig(OptionalConfigSection configSection, FaceSystem::ConfigContext& context) { // websocket // { // listen yes // port 9696 // enable_v4 yes // enable_v6 yes // } bool wantListen = false; uint16_t port = 9696; bool enableV4 = true; bool enableV6 = true; if (configSection) { wantListen = true; for (const auto& pair : *configSection) { const std::string& key = pair.first; if (key == "listen") { wantListen = ConfigFile::parseYesNo(pair, "face_system.websocket"); } else if (key == "port") { port = ConfigFile::parseNumber(pair, "face_system.websocket"); } else if (key == "enable_v4") { enableV4 = ConfigFile::parseYesNo(pair, "face_system.websocket"); } else if (key == "enable_v6") { enableV6 = ConfigFile::parseYesNo(pair, "face_system.websocket"); } else { BOOST_THROW_EXCEPTION(ConfigFile::Error("Unrecognized option face_system.websocket." + key)); } } } if (!enableV4 && !enableV6) { BOOST_THROW_EXCEPTION(ConfigFile::Error( "IPv4 and IPv6 WebSocket channels have been disabled. Remove face_system.websocket section " "to disable WebSocket channels or enable at least one channel type.")); } if (!enableV4 && enableV6) { // websocketpp's IPv6 socket always accepts IPv4 connections. BOOST_THROW_EXCEPTION(ConfigFile::Error("NFD does not allow pure IPv6 WebSocket channel.")); } if (!context.isDryRun) { if (!wantListen) { if (!m_channels.empty()) { NFD_LOG_WARN("Cannot close WebSocket channel after initialization"); } return; } BOOST_ASSERT(enableV4); websocket::Endpoint endpoint(enableV6 ? ip::tcp::v6() : ip::tcp::v4(), port); auto channel = this->createChannel(endpoint); if (!channel->isListening()) { channel->listen(context.addFace); if (m_channels.size() > 1) { NFD_LOG_WARN("Adding WebSocket channel for new endpoint; cannot close existing channels"); } } } } void WebSocketFactory::createFace(const FaceUri& uri, ndn::nfd::FacePersistency persistency, bool wantLocalFieldsEnabled, const FaceCreatedCallback& onCreated, const FaceCreationFailedCallback& onFailure) { onFailure(406, "Unsupported protocol"); } shared_ptr WebSocketFactory::createChannel(const websocket::Endpoint& endpoint) { auto channel = findChannel(endpoint); if (channel) return channel; channel = make_shared(endpoint); m_channels[endpoint] = channel; return channel; } shared_ptr WebSocketFactory::createChannel(const std::string& localIp, const std::string& localPort) { websocket::Endpoint endpoint(ip::address::from_string(localIp), boost::lexical_cast(localPort)); return createChannel(endpoint); } std::vector> WebSocketFactory::getChannels() const { return getChannelsFromMap(m_channels); } shared_ptr WebSocketFactory::findChannel(const websocket::Endpoint& endpoint) const { auto i = m_channels.find(endpoint); if (i != m_channels.end()) return i->second; else return nullptr; } } // namespace face } // namespace nfd