/* * Copyright (c) 2017-2019 Cisco and/or its affiliates. * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #pragma once #include #include #include #include namespace transport { namespace core { template class ManifestInline : public Manifest> { using ManifestBase = Manifest>; using Hash = typename FormatTraits::Hash; using HashType = typename FormatTraits::HashType; using Suffix = typename FormatTraits::Suffix; using SuffixList = typename FormatTraits::SuffixList; using HashEntry = std::pair>; public: ManifestInline() : ManifestBase() {} ManifestInline(const core::Name &name, std::size_t signature_size = 0) : ManifestBase(name, signature_size) {} template ManifestInline(T &&base) : ManifestBase(std::forward(base)) {} static TRANSPORT_ALWAYS_INLINE ManifestInline *createManifest( const core::Name &manifest_name, ManifestVersion version, ManifestType type, auth::CryptoHashType algorithm, bool is_last, const Name &base_name, NextSegmentCalculationStrategy strategy, std::size_t signature_size) { auto manifest = new ManifestInline(manifest_name, signature_size); manifest->setVersion(version); manifest->setManifestType(type); manifest->setHashAlgorithm(algorithm); manifest->setFinalManifest(is_last); manifest->setBaseName(base_name); manifest->setNextSegmentCalculationStrategy(strategy); return manifest; } ManifestInline &encodeImpl() { ManifestBase::encoder_.encode(); return *this; } ManifestInline &decodeImpl() { base_name_ = ManifestBase::decoder_.getBaseName(); next_segment_strategy_ = ManifestBase::decoder_.getNextSegmentCalculationStrategy(); suffix_hash_map_ = ManifestBase::decoder_.getSuffixHashList(); return *this; } std::size_t estimateManifestSizeImpl(std::size_t additional_entries = 0) { return ManifestBase::encoder_.estimateSerializedLength(additional_entries); } ManifestInline &setBaseName(const Name &name) { base_name_ = name; ManifestBase::encoder_.setBaseName(base_name_); return *this; } const Name &getBaseName() { return base_name_; } ManifestInline &addSuffixHash(Suffix suffix, const Hash &hash) { ManifestBase::encoder_.addSuffixAndHash(suffix, hash); return *this; } // Call this function only after the decode function! const SuffixList &getSuffixList() { return suffix_hash_map_; } ManifestInline &setNextSegmentCalculationStrategy( NextSegmentCalculationStrategy strategy) { next_segment_strategy_ = strategy; ManifestBase::encoder_.setNextSegmentCalculationStrategy( next_segment_strategy_); return *this; } NextSegmentCalculationStrategy getNextSegmentCalculationStrategy() { return next_segment_strategy_; } // Convert several manifests into a single map from suffixes to packet hashes. // All manifests must have been decoded beforehand. static std::unordered_map getSuffixMap( const std::vector &manifests) { std::unordered_map suffix_map; for (auto manifest_ptr : manifests) { HashType hash_algorithm = manifest_ptr->getHashAlgorithm(); SuffixList suffix_list = manifest_ptr->getSuffixList(); for (auto it = suffix_list.begin(); it != suffix_list.end(); ++it) { std::vector hash( it->second, it->second + auth::hash_size_map[hash_algorithm]); suffix_map[it->first] = {hash_algorithm, hash}; } } return suffix_map; } static std::unordered_map getSuffixMap( ManifestInline *manifest) { return getSuffixMap(std::vector{manifest}); } private: core::Name base_name_; NextSegmentCalculationStrategy next_segment_strategy_; SuffixList suffix_hash_map_; }; } // namespace core } // namespace transport