/* SPDX-License-Identifier: LGPL-2.1-or-later */ #include #include #include #include #include #include #include #include "alloc-util.h" #include "ansi-color.h" #include "fd-util.h" #include "fs-util.h" #include "macro.h" #include "path-util.h" #include "strv.h" #include "terminal-util.h" #include "tests.h" #include "tmpfile-util.h" #define LOREM_IPSUM "Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor " \ "incididunt ut labore et dolore magna aliqua. Ut enim ad minim veniam, quis nostrud exercitation " \ "ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in reprehenderit " \ "in voluptate velit esse cillum dolore eu fugiat nulla pariatur. Excepteur sint occaecat cupidatat " \ "non proident, sunt in culpa qui officia deserunt mollit anim id est laborum." TEST(colors_enabled) { log_info("colors_enabled: %s", yes_no(colors_enabled())); } TEST(default_term_for_tty) { puts(default_term_for_tty("/dev/tty23")); puts(default_term_for_tty("/dev/ttyS23")); puts(default_term_for_tty("/dev/tty0")); puts(default_term_for_tty("/dev/pty0")); puts(default_term_for_tty("/dev/pts/0")); puts(default_term_for_tty("/dev/console")); puts(default_term_for_tty("tty23")); puts(default_term_for_tty("ttyS23")); puts(default_term_for_tty("tty0")); puts(default_term_for_tty("pty0")); puts(default_term_for_tty("pts/0")); puts(default_term_for_tty("console")); } TEST(read_one_char) { _cleanup_fclose_ FILE *file = NULL; char r; bool need_nl; _cleanup_(unlink_tempfilep) char name[] = "/tmp/test-read_one_char.XXXXXX"; assert_se(fmkostemp_safe(name, "r+", &file) == 0); assert_se(fputs("c\n", file) >= 0); rewind(file); assert_se(read_one_char(file, &r, 1000000, &need_nl) >= 0); assert_se(!need_nl); assert_se(r == 'c'); assert_se(read_one_char(file, &r, 1000000, &need_nl) < 0); rewind(file); assert_se(fputs("foobar\n", file) >= 0); rewind(file); assert_se(read_one_char(file, &r, 1000000, &need_nl) < 0); rewind(file); assert_se(fputs("\n", file) >= 0); rewind(file); assert_se(read_one_char(file, &r, 1000000, &need_nl) < 0); } TEST(getttyname_malloc) { _cleanup_free_ char *ttyname = NULL; _cleanup_close_ int master = -EBADF; assert_se((master = posix_openpt(O_RDWR|O_NOCTTY)) >= 0); assert_se(getttyname_malloc(master, &ttyname) >= 0); log_info("ttyname = %s", ttyname); assert_se(PATH_IN_SET(ttyname, "ptmx", "pts/ptmx")); } typedef struct { const char *name; const char* (*func)(void); } Color; static const Color colors[] = { { "normal", ansi_normal }, { "highlight", ansi_highlight }, { "black", ansi_black }, { "red", ansi_red }, { "green", ansi_green }, { "yellow", ansi_yellow }, { "blue", ansi_blue }, { "magenta", ansi_magenta }, { "cyan", ansi_cyan }, { "white", ansi_white }, { "grey", ansi_grey }, { "bright-black", ansi_bright_black }, { "bright-red", ansi_bright_red }, { "bright-green", ansi_bright_green }, { "bright-yellow", ansi_bright_yellow }, { "bright-blue", ansi_bright_blue }, { "bright-magenta", ansi_bright_magenta }, { "bright-cyan", ansi_bright_cyan }, { "bright-white", ansi_bright_white }, { "highlight-black", ansi_highlight_black }, { "highlight-red", ansi_highlight_red }, { "highlight-green", ansi_highlight_green }, { "highlight-yellow (original)", _ansi_highlight_yellow }, { "highlight-yellow (replacement)", ansi_highlight_yellow }, { "highlight-blue", ansi_highlight_blue }, { "highlight-magenta", ansi_highlight_magenta }, { "highlight-cyan", ansi_highlight_cyan }, { "highlight-white", ansi_highlight_white }, { "highlight-grey", ansi_highlight_grey }, { "underline", ansi_underline }, { "highlight-underline", ansi_highlight_underline }, { "highlight-red-underline", ansi_highlight_red_underline }, { "highlight-green-underline", ansi_highlight_green_underline }, { "highlight-yellow-underline", ansi_highlight_yellow_underline }, { "highlight-blue-underline", ansi_highlight_blue_underline }, { "highlight-magenta-underline", ansi_highlight_magenta_underline }, { "highlight-grey-underline", ansi_highlight_grey_underline }, }; TEST(colors) { FOREACH_ELEMENT(color, colors) printf("<%s%s%s>\n", colors->func(), color->name, ansi_normal()); } TEST(text) { for (size_t i = 0; !streq(colors[i].name, "underline"); i++) { bool blwh = strstr(colors[i].name, "black") || strstr(colors[i].name, "white"); printf("\n" "Testing color %s%s\n%s%s%s\n", colors[i].name, blwh ? "" : ", this text should be readable", colors[i].func(), LOREM_IPSUM, ansi_normal()); } } TEST(get_ctty) { _cleanup_free_ char *ctty = NULL; struct stat st; dev_t devnr; int r; r = get_ctty(0, &devnr, &ctty); if (r < 0) { log_notice_errno(r, "Apparently called without a controlling TTY, cutting get_ctty() test short: %m"); return; } /* In almost all cases STDIN will match our controlling TTY. Let's verify that and then compare paths */ ASSERT_OK_ERRNO(fstat(STDIN_FILENO, &st)); if (S_ISCHR(st.st_mode) && st.st_rdev == devnr) { _cleanup_free_ char *stdin_name = NULL; assert_se(getttyname_malloc(STDIN_FILENO, &stdin_name) >= 0); assert_se(path_equal(stdin_name, ctty)); } else log_notice("Not invoked with stdin == ctty, cutting get_ctty() test short"); } TEST(get_default_background_color) { double red, green, blue; int r; r = get_default_background_color(&red, &green, &blue); if (r < 0) log_notice_errno(r, "Can't get terminal default background color: %m"); else log_notice("R=%g G=%g B=%g", red, green, blue); } TEST(terminal_get_size_by_dsr) { unsigned rows, columns; int r; r = terminal_get_size_by_dsr(STDIN_FILENO, STDOUT_FILENO, &rows, &columns); if (r < 0) log_notice_errno(r, "Can't get screen dimensions via DSR: %m"); else { log_notice("terminal size via DSR: rows=%u columns=%u", rows, columns); struct winsize ws = {}; if (ioctl(STDIN_FILENO, TIOCGWINSZ, &ws) < 0) log_warning_errno(errno, "Can't get terminal size via ioctl, ignoring: %m"); else log_notice("terminal size via ioctl: rows=%u columns=%u", ws.ws_row, ws.ws_col); } } TEST(terminal_fix_size) { int r; r = terminal_fix_size(STDIN_FILENO, STDOUT_FILENO); if (r < 0) log_warning_errno(r, "Failed to fix terminal size: %m"); else if (r == 0) log_notice("Not fixing terminal size, nothing to do."); else log_notice("Fixed terminal size."); } TEST(terminal_is_pty_fd) { _cleanup_close_ int fd1 = -EBADF, fd2 = -EBADF; int r; fd1 = openpt_allocate(O_RDWR, /* ret_peer_path= */ NULL); assert_se(fd1 >= 0); assert_se(terminal_is_pty_fd(fd1) > 0); fd2 = pty_open_peer(fd1, O_RDWR|O_CLOEXEC|O_NOCTTY); assert_se(fd2 >= 0); assert_se(terminal_is_pty_fd(fd2) > 0); fd1 = safe_close(fd1); fd2 = safe_close(fd2); fd1 = open("/dev/null", O_RDONLY|O_CLOEXEC); assert_se(fd1 >= 0); assert_se(terminal_is_pty_fd(fd1) == 0); /* In container managers real tty devices might be weird, avoid them. */ r = path_is_read_only_fs("/sys"); if (r != 0) return; FOREACH_STRING(p, "/dev/ttyS0", "/dev/tty1") { _cleanup_close_ int tfd = -EBADF; tfd = open_terminal(p, O_CLOEXEC|O_NOCTTY|O_RDONLY|O_NONBLOCK); if (tfd == -ENOENT) continue; if (tfd < 0) { log_notice_errno(tfd, "Failed to open '%s', skipping: %m", p); continue; } assert_se(terminal_is_pty_fd(tfd) <= 0); } } static void test_get_color_mode_with_env(const char *key, const char *val, ColorMode expected) { ASSERT_OK_ERRNO(setenv(key, val, true)); reset_terminal_feature_caches(); log_info("get_color_mode($%s=%s): %s", key, val, color_mode_to_string(get_color_mode())); ASSERT_EQ(get_color_mode(), expected); } TEST(get_color_mode) { log_info("get_color_mode(default): %s", color_mode_to_string(get_color_mode())); ASSERT_OK(get_color_mode()); test_get_color_mode_with_env("SYSTEMD_COLORS", "0", COLOR_OFF); test_get_color_mode_with_env("SYSTEMD_COLORS", "no", COLOR_OFF); test_get_color_mode_with_env("SYSTEMD_COLORS", "16", COLOR_16); test_get_color_mode_with_env("SYSTEMD_COLORS", "256", COLOR_256); test_get_color_mode_with_env("SYSTEMD_COLORS", "1", COLOR_24BIT); test_get_color_mode_with_env("SYSTEMD_COLORS", "yes", COLOR_24BIT); test_get_color_mode_with_env("SYSTEMD_COLORS", "24bit", COLOR_24BIT); ASSERT_OK_ERRNO(setenv("NO_COLOR", "1", true)); test_get_color_mode_with_env("SYSTEMD_COLORS", "42", COLOR_OFF); test_get_color_mode_with_env("SYSTEMD_COLORS", "invalid", COLOR_OFF); ASSERT_OK_ERRNO(unsetenv("NO_COLOR")); ASSERT_OK_ERRNO(unsetenv("SYSTEMD_COLORS")); test_get_color_mode_with_env("COLORTERM", "truecolor", terminal_is_dumb() ? COLOR_OFF : COLOR_24BIT); test_get_color_mode_with_env("COLORTERM", "24bit", terminal_is_dumb() ? COLOR_OFF : COLOR_24BIT); test_get_color_mode_with_env("COLORTERM", "invalid", terminal_is_dumb() ? COLOR_OFF : COLOR_256); test_get_color_mode_with_env("COLORTERM", "42", terminal_is_dumb() ? COLOR_OFF : COLOR_256); ASSERT_OK_ERRNO(unsetenv("COLORTERM")); reset_terminal_feature_caches(); } TEST(terminal_reset_defensive) { int r; r = terminal_reset_defensive(STDOUT_FILENO, /* switch_to_text= */ false); if (r < 0) log_notice_errno(r, "Failed to reset terminal: %m"); } TEST(pty_open_peer) { _cleanup_close_ int pty_fd = -EBADF, peer_fd = -EBADF; _cleanup_free_ char *pty_path = NULL; pty_fd = openpt_allocate(O_RDWR|O_NOCTTY|O_CLOEXEC|O_NONBLOCK, &pty_path); assert(pty_fd >= 0); assert(pty_path); peer_fd = pty_open_peer(pty_fd, O_RDWR|O_NOCTTY|O_CLOEXEC); assert(peer_fd >= 0); static const char x[] = { 'x', '\n' }; assert(write(pty_fd, x, sizeof(x)) == 2); char buf[3]; assert(read(peer_fd, &buf, sizeof(buf)) == sizeof(x)); assert(buf[0] == x[0]); assert(buf[1] == x[1]); } DEFINE_TEST_MAIN(LOG_INFO);