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