encryptor.cpp
Go to the documentation of this file.
1 /* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
2 /*
3  * Copyright (c) 2014-2024, Regents of the University of California
4  *
5  * NAC library is free software: you can redistribute it and/or modify it under the
6  * terms of the GNU Lesser General Public License as published by the Free Software
7  * Foundation, either version 3 of the License, or (at your option) any later version.
8  *
9  * NAC library is distributed in the hope that it will be useful, but WITHOUT ANY
10  * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
11  * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
12  *
13  * You should have received copies of the GNU General Public License and GNU Lesser
14  * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
15  * <http://www.gnu.org/licenses/>.
16  *
17  * See AUTHORS.md for complete list of NAC library authors and contributors.
18  */
19 
20 #include "encryptor.hpp"
21 
22 #include <ndn-cxx/security/transform/block-cipher.hpp>
23 #include <ndn-cxx/security/transform/buffer-source.hpp>
24 #include <ndn-cxx/security/transform/stream-sink.hpp>
25 #include <ndn-cxx/util/logger.hpp>
26 #include <ndn-cxx/util/random.hpp>
27 
28 #include <boost/lexical_cast.hpp>
29 
30 namespace ndn::nac {
31 
32 NDN_LOG_INIT(nac.Encryptor);
33 
34 constexpr size_t N_RETRIES = 3;
35 
36 Encryptor::Encryptor(const Name& accessPrefix,
37  const Name& ckPrefix, SigningInfo ckDataSigningInfo,
38  const ErrorCallback& onFailure,
39  Validator&, KeyChain& keyChain, Face& face)
40  : m_accessPrefix(accessPrefix)
41  , m_ckPrefix(ckPrefix)
42  , m_ckBits(AES_KEY_SIZE)
43  , m_ckDataSigningInfo(std::move(ckDataSigningInfo))
44  , m_isKekRetrievalInProgress(false)
45  , m_onFailure(onFailure)
46  , m_keyChain(keyChain)
47  , m_face(face)
48  , m_scheduler(face.getIoContext())
49 {
50  regenerateCk();
51 
52  auto serveFromIms = [this] (const Name&, const Interest& interest) {
53  auto data = m_ims.find(interest);
54  if (data != nullptr) {
55  NDN_LOG_DEBUG("Serving " << data->getName() << " from InMemoryStorage");
56  m_face.put(*data);
57  }
58  else {
59  NDN_LOG_DEBUG("Didn't find CK data for " << interest.getName());
60  // send NACK?
61  }
62  };
63 
64  auto handleError = [] (const Name& prefix, const std::string& msg) {
65  NDN_LOG_ERROR("Failed to register prefix " << prefix << ": " << msg);
66  };
67 
68  m_ckReg = m_face.setInterestFilter(Name(ckPrefix).append(CK), serveFromIms, handleError);
69 }
70 
72 {
73  m_kekPendingInterest.cancel();
74 }
75 
76 void
77 Encryptor::retryFetchingKek()
78 {
79  if (m_isKekRetrievalInProgress) {
80  return;
81  }
82 
83  NDN_LOG_DEBUG("Retrying fetching of KEK");
84  m_isKekRetrievalInProgress = true;
85  fetchKekAndPublishCkData([&] {
86  NDN_LOG_DEBUG("KEK retrieved and published");
87  m_isKekRetrievalInProgress = false;
88  },
89  [=] (const ErrorCode& code, const std::string& msg) {
90  NDN_LOG_ERROR("Failed to retrieved KEK: " + msg);
91  m_isKekRetrievalInProgress = false;
92  m_onFailure(code, msg);
93  },
94  N_RETRIES);
95 }
96 
97 void
99 {
100  m_ckName = m_ckPrefix;
101  m_ckName
102  .append(CK)
103  .appendVersion(); // version = ID of CK
104  NDN_LOG_DEBUG("Generating new CK: " << m_ckName);
105  random::generateSecureBytes(m_ckBits);
106 
107  // one implication: if CK updated before KEK fetched, KDK for the old CK will not be published
108  if (!m_kek) {
109  retryFetchingKek();
110  }
111  else {
112  makeAndPublishCkData(m_onFailure);
113  }
114 }
115 
117 Encryptor::encrypt(span<const uint8_t> data)
118 {
119  // Generate IV
120  auto iv = std::make_shared<Buffer>(AES_IV_SIZE);
121  random::generateSecureBytes(*iv);
122 
123  OBufferStream os;
124  security::transform::bufferSource(data)
125  >> security::transform::blockCipher(BlockCipherAlgorithm::AES_CBC,
126  CipherOperator::ENCRYPT,
127  m_ckBits, *iv)
128  >> security::transform::streamSink(os);
129 
130  EncryptedContent content;
131  content.setIv(iv);
132  content.setPayload(os.buf());
133  content.setKeyLocator(m_ckName);
134 
135  return content;
136 }
137 
138 void
139 Encryptor::fetchKekAndPublishCkData(const std::function<void()>& onReady,
140  const ErrorCallback& onFailure,
141  size_t nTriesLeft)
142 {
143  // interest for <access-prefix>/KEK to retrieve <access-prefix>/KEK/<key-id> KekData
144 
145  NDN_LOG_DEBUG("Fetching KEK " << Name(m_accessPrefix).append(KEK));
146 
147  auto kekInterest = Interest(Name(m_accessPrefix).append(KEK))
148  .setCanBePrefix(true)
149  .setMustBeFresh(true);
150  m_kekPendingInterest = m_face.expressInterest(kekInterest,
151  [=] (const Interest&, const Data& kek) {
152  // @todo verify if the key is legit
153  m_kek = kek;
154  if (makeAndPublishCkData(onFailure)) {
155  onReady();
156  }
157  // otherwise, failure has been already declared
158  },
159  [=] (const Interest& i, const lp::Nack& nack) {
160  if (nTriesLeft > 1) {
161  m_scheduler.schedule(RETRY_DELAY_AFTER_NACK, [=] {
162  fetchKekAndPublishCkData(onReady, onFailure, nTriesLeft - 1);
163  });
164  }
165  else {
166  onFailure(ErrorCode::KekRetrievalFailure, "Retrieval of KEK [" + i.getName().toUri() +
167  "] failed. Got NACK with reason " + boost::lexical_cast<std::string>(nack.getReason()));
168  NDN_LOG_DEBUG("Scheduling retry from NACK");
169  m_scheduler.schedule(RETRY_DELAY_KEK_RETRIEVAL, [this] { retryFetchingKek(); });
170  }
171  },
172  [=] (const Interest& i) {
173  if (nTriesLeft > 1) {
174  fetchKekAndPublishCkData(onReady, onFailure, nTriesLeft - 1);
175  }
176  else {
178  "Retrieval of KEK [" + i.getName().toUri() + "] timed out");
179  NDN_LOG_DEBUG("Scheduling retry after all timeouts");
180  m_scheduler.schedule(RETRY_DELAY_KEK_RETRIEVAL, [this] { retryFetchingKek(); });
181  }
182  });
183 }
184 
185 bool
186 Encryptor::makeAndPublishCkData(const ErrorCallback& onFailure)
187 {
188  try {
189  PublicKey kek;
190  kek.loadPkcs8(m_kek->getContent().value_bytes());
191 
192  EncryptedContent content;
193  content.setPayload(kek.encrypt(m_ckBits));
194 
195  auto ckData = std::make_shared<Data>(Name(m_ckName).append(ENCRYPTED_BY).append(m_kek->getName()));
196  ckData->setContent(content.wireEncode());
197  // FreshnessPeriod can serve as a soft access control for revoking access
198  ckData->setFreshnessPeriod(DEFAULT_CK_FRESHNESS_PERIOD);
199  m_keyChain.sign(*ckData, m_ckDataSigningInfo);
200  m_ims.insert(*ckData);
201 
202  NDN_LOG_DEBUG("Publishing CK data: " << ckData->getName());
203  return true;
204  }
205  catch (const std::runtime_error&) {
207  "Failed to encrypt generated CK with KEK " + m_kek->getName().toUri());
208  return false;
209  }
210 }
211 
212 } // namespace ndn::nac
EncryptedContent & setPayload(Block payload)
EncryptedContent & setIv(Block iv)
EncryptedContent & setKeyLocator(Name keyLocator)
Encryptor(const Name &accessPrefix, const Name &ckPrefix, SigningInfo ckDataSigningInfo, const ErrorCallback &onFailure, Validator &validator, KeyChain &keyChain, Face &face)
Definition: encryptor.cpp:36
EncryptedContent encrypt(span< const uint8_t > data)
Synchronously encrypt supplied data.
Definition: encryptor.cpp:117
void regenerateCk()
Create a new content key and publish the corresponding CK data.
Definition: encryptor.cpp:98
@ EncryptedContent
Definition: common.hpp:74
constexpr time::seconds DEFAULT_CK_FRESHNESS_PERIOD
Definition: common.hpp:93
std::function< void(const ErrorCode &, const std::string &)> ErrorCallback
Definition: common.hpp:117
constexpr size_t N_RETRIES
Definition: decryptor.cpp:34
constexpr size_t AES_IV_SIZE
Definition: common.hpp:89
const name::Component ENCRYPTED_BY
Definition: common.hpp:82
const name::Component CK
Definition: common.hpp:86
const name::Component KEK
Definition: common.hpp:84
constexpr time::seconds RETRY_DELAY_KEK_RETRIEVAL
Definition: common.hpp:96
constexpr size_t AES_KEY_SIZE
Definition: common.hpp:88
constexpr time::seconds RETRY_DELAY_AFTER_NACK
Definition: common.hpp:95