xref: /netbsd-src/external/bsd/atf/dist/tools/ui.cpp (revision d780102efefa02003390cc43ea410dbd0ebb4a85)
1*d780102eSjmmv //
2*d780102eSjmmv // Automated Testing Framework (atf)
3*d780102eSjmmv //
4*d780102eSjmmv // Copyright (c) 2007 The NetBSD Foundation, Inc.
5*d780102eSjmmv // All rights reserved.
6*d780102eSjmmv //
7*d780102eSjmmv // Redistribution and use in source and binary forms, with or without
8*d780102eSjmmv // modification, are permitted provided that the following conditions
9*d780102eSjmmv // are met:
10*d780102eSjmmv // 1. Redistributions of source code must retain the above copyright
11*d780102eSjmmv //    notice, this list of conditions and the following disclaimer.
12*d780102eSjmmv // 2. Redistributions in binary form must reproduce the above copyright
13*d780102eSjmmv //    notice, this list of conditions and the following disclaimer in the
14*d780102eSjmmv //    documentation and/or other materials provided with the distribution.
15*d780102eSjmmv //
16*d780102eSjmmv // THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND
17*d780102eSjmmv // CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
18*d780102eSjmmv // INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
19*d780102eSjmmv // MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20*d780102eSjmmv // IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY
21*d780102eSjmmv // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*d780102eSjmmv // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE
23*d780102eSjmmv // GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24*d780102eSjmmv // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER
25*d780102eSjmmv // IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
26*d780102eSjmmv // OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN
27*d780102eSjmmv // IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28*d780102eSjmmv //
29*d780102eSjmmv 
30*d780102eSjmmv extern "C" {
31*d780102eSjmmv #include <sys/ioctl.h>
32*d780102eSjmmv 
33*d780102eSjmmv #include <termios.h>
34*d780102eSjmmv #include <unistd.h>
35*d780102eSjmmv }
36*d780102eSjmmv 
37*d780102eSjmmv #include <cassert>
38*d780102eSjmmv #include <sstream>
39*d780102eSjmmv 
40*d780102eSjmmv #include "env.hpp"
41*d780102eSjmmv #include "text.hpp"
42*d780102eSjmmv #include "ui.hpp"
43*d780102eSjmmv 
44*d780102eSjmmv namespace impl = tools::ui;
45*d780102eSjmmv #define IMPL_NAME "tools::ui"
46*d780102eSjmmv 
47*d780102eSjmmv static
48*d780102eSjmmv size_t
terminal_width(void)49*d780102eSjmmv terminal_width(void)
50*d780102eSjmmv {
51*d780102eSjmmv     static bool done = false;
52*d780102eSjmmv     static size_t width = 0;
53*d780102eSjmmv 
54*d780102eSjmmv     if (!done) {
55*d780102eSjmmv         if (tools::env::has("COLUMNS")) {
56*d780102eSjmmv             const std::string cols = tools::env::get("COLUMNS");
57*d780102eSjmmv             if (cols.length() > 0) {
58*d780102eSjmmv                 width = tools::text::to_type< size_t >(cols);
59*d780102eSjmmv             }
60*d780102eSjmmv         } else {
61*d780102eSjmmv             struct winsize ws;
62*d780102eSjmmv             if (ioctl(STDOUT_FILENO, TIOCGWINSZ, &ws) != -1)
63*d780102eSjmmv                 width = ws.ws_col;
64*d780102eSjmmv         }
65*d780102eSjmmv 
66*d780102eSjmmv         if (width >= 80)
67*d780102eSjmmv             width -= 5;
68*d780102eSjmmv 
69*d780102eSjmmv         done = true;
70*d780102eSjmmv     }
71*d780102eSjmmv 
72*d780102eSjmmv     return width;
73*d780102eSjmmv }
74*d780102eSjmmv 
75*d780102eSjmmv static
76*d780102eSjmmv std::string
format_paragraph(const std::string & text,const std::string & tag,const bool first,const bool repeat,const size_t col)77*d780102eSjmmv format_paragraph(const std::string& text,
78*d780102eSjmmv                  const std::string& tag,
79*d780102eSjmmv                  const bool first,
80*d780102eSjmmv                  const bool repeat,
81*d780102eSjmmv                  const size_t col)
82*d780102eSjmmv {
83*d780102eSjmmv     assert(text.find('\n') == std::string::npos);
84*d780102eSjmmv 
85*d780102eSjmmv     const std::string pad(col - tag.length(), ' ');
86*d780102eSjmmv     const std::string fullpad(col, ' ');
87*d780102eSjmmv 
88*d780102eSjmmv     std::string formatted;
89*d780102eSjmmv     if (first || repeat)
90*d780102eSjmmv         formatted = tag + pad;
91*d780102eSjmmv     else
92*d780102eSjmmv         formatted = fullpad;
93*d780102eSjmmv     assert(formatted.length() == col);
94*d780102eSjmmv     size_t curcol = col;
95*d780102eSjmmv 
96*d780102eSjmmv     const size_t maxcol = terminal_width();
97*d780102eSjmmv 
98*d780102eSjmmv     std::vector< std::string > words = tools::text::split(text, " ");
99*d780102eSjmmv     for (std::vector< std::string >::const_iterator iter = words.begin();
100*d780102eSjmmv          iter != words.end(); iter++) {
101*d780102eSjmmv         const std::string& word = *iter;
102*d780102eSjmmv 
103*d780102eSjmmv         if (iter != words.begin() && maxcol > 0 &&
104*d780102eSjmmv             curcol + word.length() + 1 > maxcol) {
105*d780102eSjmmv             if (repeat)
106*d780102eSjmmv                 formatted += '\n' + tag + pad;
107*d780102eSjmmv             else
108*d780102eSjmmv                 formatted += '\n' + fullpad;
109*d780102eSjmmv             curcol = col;
110*d780102eSjmmv         } else if (iter != words.begin()) {
111*d780102eSjmmv             formatted += ' ';
112*d780102eSjmmv             curcol++;
113*d780102eSjmmv         }
114*d780102eSjmmv 
115*d780102eSjmmv         formatted += word;
116*d780102eSjmmv         curcol += word.length();
117*d780102eSjmmv     }
118*d780102eSjmmv 
119*d780102eSjmmv     return formatted;
120*d780102eSjmmv }
121*d780102eSjmmv 
122*d780102eSjmmv std::string
format_error(const std::string & prog_name,const std::string & error)123*d780102eSjmmv impl::format_error(const std::string& prog_name, const std::string& error)
124*d780102eSjmmv {
125*d780102eSjmmv     return format_text_with_tag("ERROR: " + error, prog_name + ": ", true);
126*d780102eSjmmv }
127*d780102eSjmmv 
128*d780102eSjmmv std::string
format_info(const std::string & prog_name,const std::string & msg)129*d780102eSjmmv impl::format_info(const std::string& prog_name, const std::string& msg)
130*d780102eSjmmv {
131*d780102eSjmmv     return format_text_with_tag(msg, prog_name + ": ", true);
132*d780102eSjmmv }
133*d780102eSjmmv 
134*d780102eSjmmv std::string
format_text(const std::string & text)135*d780102eSjmmv impl::format_text(const std::string& text)
136*d780102eSjmmv {
137*d780102eSjmmv     return format_text_with_tag(text, "", false, 0);
138*d780102eSjmmv }
139*d780102eSjmmv 
140*d780102eSjmmv std::string
format_text_with_tag(const std::string & text,const std::string & tag,bool repeat,size_t col)141*d780102eSjmmv impl::format_text_with_tag(const std::string& text, const std::string& tag,
142*d780102eSjmmv                            bool repeat, size_t col)
143*d780102eSjmmv {
144*d780102eSjmmv     assert(col == 0 || col >= tag.length());
145*d780102eSjmmv     if (col == 0)
146*d780102eSjmmv         col = tag.length();
147*d780102eSjmmv 
148*d780102eSjmmv     std::string formatted;
149*d780102eSjmmv 
150*d780102eSjmmv     std::vector< std::string > lines = tools::text::split(text, "\n");
151*d780102eSjmmv     for (std::vector< std::string >::const_iterator iter = lines.begin();
152*d780102eSjmmv          iter != lines.end(); iter++) {
153*d780102eSjmmv         const std::string& line = *iter;
154*d780102eSjmmv 
155*d780102eSjmmv         formatted += format_paragraph(line, tag, iter == lines.begin(),
156*d780102eSjmmv                                       repeat, col);
157*d780102eSjmmv         if (iter + 1 != lines.end()) {
158*d780102eSjmmv             if (repeat)
159*d780102eSjmmv                 formatted += "\n" + tag + "\n";
160*d780102eSjmmv             else
161*d780102eSjmmv                 formatted += "\n\n";
162*d780102eSjmmv         }
163*d780102eSjmmv     }
164*d780102eSjmmv 
165*d780102eSjmmv     return formatted;
166*d780102eSjmmv }
167*d780102eSjmmv 
168*d780102eSjmmv std::string
format_warning(const std::string & prog_name,const std::string & error)169*d780102eSjmmv impl::format_warning(const std::string& prog_name, const std::string& error)
170*d780102eSjmmv {
171*d780102eSjmmv     return format_text_with_tag("WARNING: " + error, prog_name + ": ", true);
172*d780102eSjmmv }
173