Utils/Console: Add control char escaping to string formatter
[senf.git] / senf / Utils / Console / Traits.cc
1 // $Id$
2 //
3 // Copyright (C) 2008 
4 // Fraunhofer Institute for Open Communication Systems (FOKUS)
5 // Competence Center NETwork research (NET), St. Augustin, GERMANY
6 //     Stefan Bund <g0dil@berlios.de>
7 //
8 // This program is free software; you can redistribute it and/or modify
9 // it under the terms of the GNU General Public License as published by
10 // the Free Software Foundation; either version 2 of the License, or
11 // (at your option) any later version.
12 //
13 // This program is distributed in the hope that it will be useful,
14 // but WITHOUT ANY WARRANTY; without even the implied warranty of
15 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 // GNU General Public License for more details.
17 //
18 // You should have received a copy of the GNU General Public License
19 // along with this program; if not, write to the
20 // Free Software Foundation, Inc.,
21 // 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22
23 /** \file
24     \brief Traits non-inline non-template implementation */
25
26 #include "Traits.hh"
27 #include "Traits.ih"
28
29 // Custom includes
30 #include <boost/format.hpp>
31 #include <senf/Utils/senfassert.hh>
32
33 //#include "Traits.mpp"
34 #define prefix_
35 ///////////////////////////////cc.p////////////////////////////////////////
36
37 prefix_ std::string senf::console::ArgumentTraits<std::string>::str(std::string const & value)
38 {
39     if (! value.empty() && boost::algorithm::all(value, CommandParser::isWordChar))
40         return value;
41     else {
42         std::string rv (value);
43         for (std::string::size_type i (0); i < rv.size(); ++i)
44             if (rv[i] == '"' || rv[i] == '\\')
45                 rv.insert(i++,"\\");
46             else if (rv[i] < ' ' || rv[i] > 126) {
47                 rv.insert(i+1, (boost::format("x%02x") 
48                                 % unsigned(static_cast<unsigned char>(rv[i]))).str().c_str());
49                 rv[i] = '\\';
50                 i += 3;
51             }
52         
53         rv.insert(0,"\"");
54         rv.push_back('"');
55         return rv;
56     }
57 }
58
59 prefix_ long senf::console::detail::parseEnum(EnumTable const & table,
60                                               ParseCommandInfo::TokensRange const & tokens)
61 {
62     if (tokens.size() != 1)
63         throw SyntaxErrorException("parameter syntax error");
64
65     std::string sym (tokens.begin()[0].value());
66     boost::algorithm::to_lower(sym);
67     EnumTable::left_map::const_iterator i1 (table.left.lower_bound(sym));
68     EnumTable::left_map::const_iterator i2 (table.left.lower_bound(sym+"\xff"));
69     if (i1 == i2)
70         throw SyntaxErrorException("parameter syntax error: invalid enum value: ")
71             << tokens.begin()[0].value();
72     long v (i1->second);
73     if (boost::algorithm::to_lower_copy(i1->first) == sym)
74         return v;
75     ++i1;
76     if (i1 != i2)
77         throw SyntaxErrorException("parameter syntax error: ambiguous enum value: ")
78             << tokens.begin()[0].value();
79     return v;
80 }
81
82 prefix_ std::string senf::console::detail::formatEnum(EnumTable const & table, long value)
83 {
84     EnumTable::right_map::const_iterator i (table.right.find(value));
85     SENF_ASSERT( i != table.right.end() );
86     return i->second;
87 }
88
89 ///////////////////////////////cc.e////////////////////////////////////////
90 #undef prefix_
91 //#include "Traits.mpp"
92
93 \f
94 // Local Variables:
95 // mode: c++
96 // fill-column: 100
97 // comment-column: 40
98 // c-file-style: "senf"
99 // indent-tabs-mode: nil
100 // ispell-local-dictionary: "american"
101 // compile-command: "scons -u test"
102 // End: