Loading...
Searching...
No Matches
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
30namespace ndn::nac {
31
32NDN_LOG_INIT(nac.Encryptor);
33
34constexpr size_t N_RETRIES = 3;
35
36Encryptor::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{
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
76void
77Encryptor::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
97void
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
117Encryptor::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
138void
139Encryptor::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
185bool
186Encryptor::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.
void regenerateCk()
Create a new content key and publish the corresponding CK data.
Definition encryptor.cpp:98
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