29 #include <ndn-cxx/tag.hpp>
30 #include <ndn-cxx/security/certificate-fetcher-offline.hpp>
31 #include <ndn-cxx/security/certificate-request.hpp>
32 #include <ndn-cxx/security/validation-policy.hpp>
33 #include <ndn-cxx/security/validation-policy-accept-all.hpp>
34 #include <ndn-cxx/security/validation-policy-command-interest.hpp>
35 #include <ndn-cxx/security/validator.hpp>
36 #include <ndn-cxx/util/io.hpp>
38 #include <boost/filesystem.hpp>
40 namespace security = ndn::security;
54 static optional<std::string>
57 shared_ptr<SignerTag> signerTag = interest.getTag<
SignerTag>();
58 if (signerTag ==
nullptr) {
62 return signerTag->get().toUri();
68 class CommandAuthenticatorValidationPolicy final :
public security::ValidationPolicy
72 checkPolicy(
const Interest& interest,
const shared_ptr<security::ValidationState>& state,
73 const ValidationContinuation& continueValidation)
final
75 Name klName = getKeyLocatorName(interest, *state);
76 if (!state->getOutcome()) {
83 auto state1 = dynamic_pointer_cast<security::InterestValidationState>(state);
84 state1->getOriginalInterest().setTag(make_shared<SignerTag>(klName));
86 continueValidation(make_shared<security::CertificateRequest>(klName), state);
90 checkPolicy(
const Data&,
const shared_ptr<security::ValidationState>&,
91 const ValidationContinuation&)
final
95 BOOST_ASSERT_MSG(
false,
"Data should not be passed to this policy");
99 shared_ptr<CommandAuthenticator>
105 CommandAuthenticator::CommandAuthenticator() =
default;
111 processConfig(std::forward<decltype(args)>(args)...);
116 CommandAuthenticator::processConfig(
const ConfigSection& section,
bool isDryRun,
const std::string& filename)
120 for (
auto& kv : m_validators) {
121 kv.second = make_shared<security::Validator>(
122 make_unique<security::ValidationPolicyCommandInterest>(make_unique<CommandAuthenticatorValidationPolicy>()),
123 make_unique<security::CertificateFetcherOffline>());
127 if (section.empty()) {
128 NDN_THROW(ConfigFile::Error(
"'authorize' is missing under 'authorizations'"));
131 int authSectionIndex = 0;
132 for (
const auto& kv : section) {
133 if (kv.first !=
"authorize") {
134 NDN_THROW(ConfigFile::Error(
"'" + kv.first +
"' section is not permitted under 'authorizations'"));
138 std::string certfile;
140 certfile = authSection.get<std::string>(
"certfile");
142 catch (
const boost::property_tree::ptree_error&) {
143 NDN_THROW(ConfigFile::Error(
"'certfile' is missing under authorize[" +
144 to_string(authSectionIndex) +
"]"));
148 shared_ptr<security::Certificate> cert;
149 if (certfile ==
"any") {
151 NFD_LOG_WARN(
"'certfile any' is intended for demo purposes only and "
152 "SHOULD NOT be used in production environments");
155 using namespace boost::filesystem;
156 path certfilePath = absolute(certfile, path(filename).parent_path());
157 cert = ndn::io::load<security::Certificate>(certfilePath.string());
158 if (cert ==
nullptr) {
159 NDN_THROW(ConfigFile::Error(
"cannot load certfile " + certfilePath.string() +
160 " for authorize[" + to_string(authSectionIndex) +
"]"));
166 privSection = &authSection.get_child(
"privileges");
168 catch (
const boost::property_tree::ptree_error&) {
169 NDN_THROW(ConfigFile::Error(
"'privileges' is missing under authorize[" +
170 to_string(authSectionIndex) +
"]"));
173 if (privSection->empty()) {
174 NFD_LOG_WARN(
"No privileges granted to certificate " << certfile);
176 for (
const auto& kv : *privSection) {
177 const std::string& module = kv.first;
178 auto found = m_validators.find(module);
179 if (found == m_validators.end()) {
180 NDN_THROW(ConfigFile::Error(
"unknown module '" + module +
181 "' under authorize[" + to_string(authSectionIndex) +
"]"));
189 found->second = make_shared<security::Validator>(make_unique<security::ValidationPolicyAcceptAll>(),
190 make_unique<security::CertificateFetcherOffline>());
191 NFD_LOG_INFO(
"authorize module=" << module <<
" signer=any");
194 const Name& keyName = cert->getKeyName();
195 security::Certificate certCopy = *cert;
196 found->second->loadAnchor(certfile, std::move(certCopy));
197 NFD_LOG_INFO(
"authorize module=" << module <<
" signer=" << keyName <<
" certfile=" << certfile);
205 ndn::mgmt::Authorization
208 m_validators[module];
210 auto self = this->shared_from_this();
211 return [=] (
const Name&,
const Interest& interest,
212 const ndn::mgmt::ControlParameters*,
213 const ndn::mgmt::AcceptContinuation& accept,
214 const ndn::mgmt::RejectContinuation& reject) {
215 auto validator =
self->m_validators.at(module);
217 auto successCb = [accept, validator] (
const Interest& interest1) {
219 BOOST_ASSERT(signer1 ||
220 dynamic_cast<security::ValidationPolicyAcceptAll*
>(&validator->getPolicy()) !=
nullptr);
221 std::string signer = signer1.value_or(
"*");
222 NFD_LOG_DEBUG(
"accept " << interest1.getName() <<
" signer=" << signer);
226 auto failureCb = [reject] (
const Interest& interest1,
const security::ValidationError& err) {
227 using ndn::mgmt::RejectReply;
228 RejectReply reply = RejectReply::STATUS403;
229 switch (err.getCode()) {
230 case security::ValidationError::NO_SIGNATURE:
231 case security::ValidationError::INVALID_KEY_LOCATOR:
232 reply = RejectReply::SILENT;
234 case security::ValidationError::POLICY_ERROR:
235 if (interest1.getName().size() < ndn::command_interest::MIN_SIZE) {
236 reply = RejectReply::SILENT;
240 NFD_LOG_DEBUG(
"reject " << interest1.getName() <<
" signer=" <<
246 validator->validate(interest, successCb, failureCb);
249 NFD_LOG_DEBUG(
"reject " << interest.getName() <<
" signer=" <<
251 reject(ndn::mgmt::RejectReply::STATUS403);
Provides ControlCommand authorization according to NFD configuration file.
ndn::mgmt::Authorization makeAuthorization(const std::string &module, const std::string &verb)
void setConfigFile(ConfigFile &configFile)
static shared_ptr< CommandAuthenticator > create()
configuration file parsing utility
void addSectionHandler(const std::string §ionName, ConfigSectionHandler subscriber)
setup notification of configuration file sections
#define NFD_LOG_INIT(name)
Copyright (c) 2014-2015, Regents of the University of California, Arizona Board of Regents,...
ndn::SimpleTag< Name, 20 > SignerTag
an Interest tag to indicate command signer
static optional< std::string > getSignerFromTag(const Interest &interest)
obtain signer from SignerTag attached to Interest, if available
boost::property_tree::ptree ConfigSection
a config file section