Utils/Logger: Add 'message' console command
[senf.git] / Utils / Logger / Target.ih
1 // $Id$
2 //
3 // Copyright (C) 2007
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 Target internal header */
25
26 #ifndef IH_SENF_Utils_Logger_Target_
27 #define IH_SENF_Utils_Logger_Target_ 1
28
29 // Custom includes
30 #include <memory>
31 #include <boost/type_traits/is_same.hpp>
32 #include <boost/static_assert.hpp>
33 #include "../Console/LazyDirectory.hh"
34
35 ///////////////////////////////ih.p////////////////////////////////////////
36
37 namespace senf {
38 namespace log {
39 namespace detail {
40
41     /** \brief Internal: Target registry */
42     class TargetRegistry
43         : public senf::singleton<TargetRegistry>
44     {
45     public:
46         enum Level { 
47             VERBOSE = senf::log::VERBOSE::value, 
48             NOTICE = senf::log::NOTICE::value, 
49             MESSAGE = senf::log::MESSAGE::value, 
50             IMPORTANT = senf::log::IMPORTANT::value, 
51             CRITICAL = senf::log::CRITICAL::value, 
52             FATAL = senf::log::FATAL::value
53         };
54
55         using senf::singleton<TargetRegistry>::instance;
56
57         void write(StreamBase const & stream, AreaBase const & area, unsigned level, 
58                    std::string const & msg);
59
60         void routed();
61         bool fallbackRouting();
62
63     private:
64         TargetRegistry();
65         
66         void registerTarget(Target * target, std::string const & name);
67         void unregisterTarget(Target * target);
68
69         void consoleAreas(std::ostream & os);
70         void consoleStreams(std::ostream & os);
71         void consoleWrite(std::string const & stream, std::string const & area, Level level,
72                           std::string const & msg);
73
74         typedef std::set<Target *> Targets;
75         Targets targets_;
76
77         bool fallbackRouting_;
78
79         console::LazyDirectory consoleDir_;
80         
81         friend class senf::log::Target;
82         friend class senf::singleton<TargetRegistry>;
83     };
84
85     /** \brief Internal: Write log message */
86     template <class Stream, class Area, class Level>
87     void write(std::string const & msg);
88
89 #ifndef DOXYGEN
90
91     // This code takes the routing target template arguments in any order and sorts them 
92     // by type (Stream, Area and Level).
93
94     senf::mpl::rv<0u> RouteParameterCheck_(...);
95     senf::mpl::rv<1u> RouteParameterCheck_(StreamBase *);
96     senf::mpl::rv<2u> RouteParameterCheck_(AreaBase *);
97     template <class T> senf::mpl::rv<3u> RouteParameterCheck_(T*, typename T::SENFLogArea * = 0);
98     senf::mpl::rv<4u> RouteParameterCheck_(LevelBase *);
99
100     // For g++ 4.0 (at least) we need to provide the fully scoped name for this default value.
101     // no idea why. It works without the scope in 4.1
102     template < class T, class A2, class A1,
103                unsigned type = SENF_MPL_RV( senf::log::detail::RouteParameterCheck_(static_cast<T*>(0)) ) >
104     struct RouteParameters
105     {};
106
107     template <class A2, class A1>
108     struct RouteParameters<mpl::nil,A2,A1,0u>
109         : public RouteParameters<A2,A1,mpl::nil>
110     {};
111
112     struct NilLevel {
113         static unsigned const value = NONE::value;
114     };
115
116     template <>
117     struct RouteParameters<mpl::nil,mpl::nil,mpl::nil,0u>
118     {
119         typedef mpl::nil Stream;
120         typedef mpl::nil Area;
121         typedef NilLevel Level;
122     };
123
124     template <class T, class A2, class A1>
125     struct RouteParameters<T,A2,A1,1u>
126         : public RouteParameters<A2,A1,mpl::nil>
127     {
128         typedef RouteParameters<A2,A1,mpl::nil> base;
129         BOOST_STATIC_ASSERT(( boost::is_same<typename base::Stream, mpl::nil>::value ));
130         typedef T Stream;
131     };
132
133     template <class T, class A2, class A1>
134     struct RouteParameters<T,A2,A1,2u>
135         : public RouteParameters<A2,A1,mpl::nil>
136     {
137         typedef RouteParameters<A2,A1,mpl::nil> base;
138         BOOST_STATIC_ASSERT(( boost::is_same<typename base::Area, mpl::nil>::value ));
139         typedef T Area;
140     };
141
142     template <class T, class A2, class A1>
143     struct RouteParameters<T,A2,A1,3u>
144         : public RouteParameters<A2,A1,mpl::nil>
145     {
146         typedef RouteParameters<A2,A1,mpl::nil> base;
147         BOOST_STATIC_ASSERT(( boost::is_same<typename base::Area, mpl::nil>::value ));
148         typedef typename T::SENFLogArea Area;
149     };
150
151     template <class T, class A2, class A1>
152     struct RouteParameters<T,A2,A1,4u>
153         : public RouteParameters<A2,A1,mpl::nil>
154     {
155         typedef RouteParameters<A2,A1,mpl::nil> base;
156         BOOST_STATIC_ASSERT(( boost::is_same<typename base::Level, NilLevel>::value ));
157         typedef T Level;
158     };
159
160     template <class T, class RV>
161     struct InstanceP
162     {
163         static RV * value() { return & T::instance(); }
164     };
165
166     template <class RV>
167     struct InstanceP<mpl::nil, RV>
168     {
169         static RV * value() { return 0; }
170     };
171
172 #endif
173
174 }}}
175
176 ///////////////////////////////ih.e////////////////////////////////////////
177 #endif
178
179 \f
180 // Local Variables:
181 // mode: c++
182 // fill-column: 100
183 // comment-column: 40
184 // c-file-style: "senf"
185 // indent-tabs-mode: nil
186 // ispell-local-dictionary: "american"
187 // compile-command: "scons -u test"
188 // End: