blob: ea093d50f57a18b2b3e060a50c90b07f9bc6748e [file] [log] [blame]
Alexander Afanasyevc169a812014-05-20 20:37:29 -04001/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
Yingdi Yuf50098d2014-02-26 14:26:29 -08002/**
Junxiao Shi2d4be252016-08-17 04:29:00 +00003 * Copyright (c) 2013-2016 Regents of the University of California.
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07004 *
5 * This file is part of ndn-cxx library (NDN C++ library with eXperimental eXtensions).
Alexander Afanasyevdfa52c42014-04-24 21:10:11 -07006 *
Alexander Afanasyevc169a812014-05-20 20:37:29 -04007 * ndn-cxx library is free software: you can redistribute it and/or modify it under the
8 * terms of the GNU Lesser General Public License as published by the Free Software
9 * Foundation, either version 3 of the License, or (at your option) any later version.
10 *
11 * ndn-cxx library is distributed in the hope that it will be useful, but WITHOUT ANY
12 * WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A
13 * PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details.
14 *
15 * You should have received copies of the GNU General Public License and GNU Lesser
16 * General Public License along with ndn-cxx, e.g., in COPYING.md file. If not, see
17 * <http://www.gnu.org/licenses/>.
18 *
19 * See AUTHORS.md for complete list of ndn-cxx authors and contributors.
Yingdi Yuf50098d2014-02-26 14:26:29 -080020 */
21
22#ifndef NDN_UTIL_IO_HPP
23#define NDN_UTIL_IO_HPP
24
Yingdi Yuf50098d2014-02-26 14:26:29 -080025#include "../encoding/block.hpp"
26
Yingdi Yuf50098d2014-02-26 14:26:29 -080027#include <iostream>
28#include <fstream>
Yingdi Yuf50098d2014-02-26 14:26:29 -080029
Yingdi Yuf50098d2014-02-26 14:26:29 -080030namespace ndn {
31namespace io {
32
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -070033class Error : public std::runtime_error
34{
35public:
36 explicit
37 Error(const std::string& what)
38 : std::runtime_error(what)
39 {
40 }
41};
Yingdi Yuf50098d2014-02-26 14:26:29 -080042
Junxiao Shif2515a72016-08-19 02:21:59 +000043/** \brief indicates how a file or stream is encoded
44 */
Yingdi Yuf50098d2014-02-26 14:26:29 -080045enum IoEncoding {
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +000046 /** \brief binary without encoding
47 */
48 NO_ENCODING,
49
50 /** \brief base64 encoding
51 *
52 * \p save() inserts a newline after every 64 characters,
53 * \p load() can accept base64 text with or without newlines
54 */
55 BASE64,
56
57 /** \brief hexadecimal encoding
58 *
59 * \p save() uses uppercase letters A-F, \p load() can accept mixed-case
60 */
61 HEX
Yingdi Yuf50098d2014-02-26 14:26:29 -080062};
63
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +000064constexpr IoEncoding DEPRECATED(BASE_64) = BASE64;
65
Junxiao Shi160701a2016-08-30 11:35:25 +000066/** \brief loads a TLV block from a stream
67 * \return if success, the Block and true
68 * otherwise, a default-constructed Block and false
69 */
70optional<Block>
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +000071loadBlock(std::istream& is, IoEncoding encoding = BASE64);
Junxiao Shi160701a2016-08-30 11:35:25 +000072
Junxiao Shif2515a72016-08-19 02:21:59 +000073/** \brief loads a TLV element from a stream
74 * \tparam T type of TLV element; T must be WireDecodable,
75 * and must have a Error nested type
76 * \return the TLV element, or nullptr if an error occurs
77 */
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -070078template<typename T>
79shared_ptr<T>
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +000080load(std::istream& is, IoEncoding encoding = BASE64)
Yingdi Yuf50098d2014-02-26 14:26:29 -080081{
Junxiao Shi160701a2016-08-30 11:35:25 +000082 optional<Block> block = loadBlock(is, encoding);
83 if (!block) {
Junxiao Shi2d4be252016-08-17 04:29:00 +000084 return nullptr;
85 }
Junxiao Shif2515a72016-08-19 02:21:59 +000086
87 try {
88 auto obj = make_shared<T>();
Junxiao Shi160701a2016-08-30 11:35:25 +000089 obj->wireDecode(*block);
Junxiao Shif2515a72016-08-19 02:21:59 +000090 return obj;
91 }
92 catch (const typename T::Error& e) {
Junxiao Shi2d4be252016-08-17 04:29:00 +000093 return nullptr;
94 }
95 catch (const tlv::Error& e) {
96 return nullptr;
97 }
Yingdi Yuf50098d2014-02-26 14:26:29 -080098}
99
Junxiao Shif2515a72016-08-19 02:21:59 +0000100/** \brief loads a TLV element from a file
101 */
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700102template<typename T>
103shared_ptr<T>
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +0000104load(const std::string& filename, IoEncoding encoding = BASE64)
Yingdi Yuf50098d2014-02-26 14:26:29 -0800105{
Junxiao Shif2515a72016-08-19 02:21:59 +0000106 std::ifstream is(filename);
Yingdi Yuf50098d2014-02-26 14:26:29 -0800107 return load<T>(is, encoding);
108}
109
Junxiao Shi160701a2016-08-30 11:35:25 +0000110/** \brief saves a TLV block to a stream
111 * \throw Error error during saving
112 */
113void
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +0000114saveBlock(const Block& block, std::ostream& os, IoEncoding encoding = BASE64);
Junxiao Shi160701a2016-08-30 11:35:25 +0000115
Junxiao Shif2515a72016-08-19 02:21:59 +0000116/** \brief saves a TLV element to a stream
117 * \tparam T type of TLV element; T must be WireEncodable,
118 * and must have a Error nested type
119 * \throw Error error during encoding or saving
120 */
Alexander Afanasyevfdbfc6d2014-04-14 15:12:11 -0700121template<typename T>
Yingdi Yuf50098d2014-02-26 14:26:29 -0800122void
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +0000123save(const T& obj, std::ostream& os, IoEncoding encoding = BASE64)
Yingdi Yuf50098d2014-02-26 14:26:29 -0800124{
Junxiao Shif2515a72016-08-19 02:21:59 +0000125 Block block;
Junxiao Shi2d4be252016-08-17 04:29:00 +0000126 try {
Junxiao Shif2515a72016-08-19 02:21:59 +0000127 block = obj.wireEncode();
Junxiao Shi2d4be252016-08-17 04:29:00 +0000128 }
Junxiao Shif2515a72016-08-19 02:21:59 +0000129 catch (const typename T::Error& e) {
Junxiao Shi2d4be252016-08-17 04:29:00 +0000130 BOOST_THROW_EXCEPTION(Error(e.what()));
131 }
132 catch (const tlv::Error& e) {
133 BOOST_THROW_EXCEPTION(Error(e.what()));
134 }
Junxiao Shif2515a72016-08-19 02:21:59 +0000135
Junxiao Shi160701a2016-08-30 11:35:25 +0000136 saveBlock(block, os, encoding);
Yingdi Yuf50098d2014-02-26 14:26:29 -0800137}
138
Junxiao Shif2515a72016-08-19 02:21:59 +0000139/** \brief saves a TLV element to a file
140 */
Yingdi Yuf50098d2014-02-26 14:26:29 -0800141template<typename T>
142void
Junxiao Shi4ce0bcf2016-09-03 07:09:03 +0000143save(const T& obj, const std::string& filename, IoEncoding encoding = BASE64)
Yingdi Yuf50098d2014-02-26 14:26:29 -0800144{
Junxiao Shif2515a72016-08-19 02:21:59 +0000145 std::ofstream os(filename);
146 save(obj, os, encoding);
Yingdi Yuf50098d2014-02-26 14:26:29 -0800147}
148
149} // namespace io
150} // namespace ndn
151
152#endif // NDN_UTIL_IO_HPP