annotate C++/Converter.cpp @ 245:3c12f0e8bbb9

Converter: add cstring missing Sockets: * Starts a PIMPL idiom to safely cast Socket with the future SocketSsl and be able to return a class of type "Socket" whilst the interface returns effectively a SocketSsl. * Rename some function to be more cute to use like getDomain -> domain(). * SocketListener now owns completely sockets and must be moved to it, the dedicated function select() always return a socket owned by SocketListener. * Operation close is not needed anymore but the function stays in case it's needed.
author David Demelier <markand@malikania.fr>
date Sun, 28 Sep 2014 21:15:26 +0200
parents 5bb5712d400f
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
215
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
1 /*
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
2 * Converter.cpp -- iconv based converter
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
3 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
4 * Copyright (c) 2013, 2014 David Demelier <markand@malikania.fr>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
5 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
6 * Permission to use, copy, modify, and/or distribute this software for any
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
7 * purpose with or without fee is hereby granted, provided that the above
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
8 * copyright notice and this permission notice appear in all copies.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
9 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
17 */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
18
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
19 #include <cerrno>
245
3c12f0e8bbb9 Converter: add cstring missing
David Demelier <markand@malikania.fr>
parents: 215
diff changeset
20 #include <cstring>
215
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
21 #include <string>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
22 #include <iterator>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
23 #include <memory>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
24 #include <string>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
25 #include <stdexcept>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
26 #include <vector>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
27
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
28 #include <iconv.h>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
29
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
30 #include "Converter.h"
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
31
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
32 struct Deleter {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
33 void operator()(iconv_t desc)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
34 {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
35 iconv_close(desc);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
36 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
37 };
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
38
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
39 using Iconv = std::unique_ptr<std::remove_pointer<iconv_t>::type, Deleter>;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
40
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
41 std::string Converter::convert(const char *from,
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
42 const char *to,
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
43 const std::string &input)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
44 {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
45 // No conversion if from and to are identical
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
46 if (std::strcmp(from, to) == 0)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
47 return input;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
48
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
49 // Try to open the conversion descriptor
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
50 auto cd = iconv_open(to, from);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
51
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
52 if (cd == (iconv_t)-1)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
53 throw std::invalid_argument(std::strerror(errno));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
54
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
55 Iconv cv(cd);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
56 std::size_t insize(input.size());
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
57 std::size_t outsize(insize);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
58 std::vector<char> result(insize + 1);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
59
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
60 auto *b = &input[0];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
61 auto *p = &result[0];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
62
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
63 while (insize > 0) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
64 /* Convert */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
65 auto r = iconv(cv.get(), &b, &insize, &p, &outsize);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
66
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
67 if (r == (size_t)-1) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
68 switch (errno) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
69 case EBADF:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
70 case EILSEQ:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
71 case EINVAL:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
72 throw std::invalid_argument(std::strerror(errno));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
73 case E2BIG:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
74 /*
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
75 * Here, we need to reallocate more data because the output
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
76 * string may need more space.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
77 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
78 * We use 16 as an optimistic value.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
79 */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
80
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
81 result.reserve(result.size() + 16 + 1);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
82 p = &result[result.size()];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
83 outsize += 16;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
84 default:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
85 break;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
86 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
87 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
88
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
89 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
90
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
91 return std::string(&result[0], (p - &result[0]));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
92 }