blob: 7def4de11e82f68643dab4243896dafc878caea6 [file] [log] [blame]
Maxime Costed2383f52017-03-23 17:35:20 +01001/* vi: set sw=4 ts=4: */
2/*
3 * paste.c - implementation of the posix paste command
4 *
5 * Written by Maxime Coste <mawww@kakoune.org>
6 *
7 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
8 */
9//config:config PASTE
Denys Vlasenkob097a842018-12-28 03:20:17 +010010//config: bool "paste (4.9 kb)"
Maxime Costed2383f52017-03-23 17:35:20 +010011//config: default y
12//config: help
Denys Vlasenko72089cf2017-07-21 09:50:55 +020013//config: paste is used to paste lines of different files together
14//config: and write the result to stdout
Maxime Costed2383f52017-03-23 17:35:20 +010015
16//applet:IF_PASTE(APPLET_NOEXEC(paste, paste, BB_DIR_USR_BIN, BB_SUID_DROP, paste))
17
18//kbuild:lib-$(CONFIG_PASTE) += paste.o
19
20//usage:#define paste_trivial_usage
Denys Vlasenko84d5edd2020-12-13 22:34:05 +010021//usage: "[-d LIST] [-s] [FILE]..."
Maxime Costed2383f52017-03-23 17:35:20 +010022//usage:#define paste_full_usage "\n\n"
Denys Vlasenkof1db95a2017-03-23 17:41:59 +010023//usage: "Paste lines from each input file, separated with tab\n"
Maxime Costed2383f52017-03-23 17:35:20 +010024//usage: "\n -d LIST Use delimiters from LIST, not tab"
25//usage: "\n -s Serial: one file at a time"
26//usage:
27//usage:#define paste_example_usage
28//usage: "# write out directory in four columns\n"
29//usage: "$ ls | paste - - - -\n"
30//usage: "# combine pairs of lines from a file into single lines\n"
31//usage: "$ paste -s -d '\\t\\n' file\n"
32
33#include "libbb.h"
34
35static void paste_files(FILE** files, int file_cnt, char* delims, int del_cnt)
36{
37 char *line;
38 char delim;
Maxime Costed2383f52017-03-23 17:35:20 +010039 int active_files = file_cnt;
40 int i;
41
42 while (active_files > 0) {
Denys Vlasenkof4a670a2017-03-23 17:58:32 +010043 int del_idx = 0;
44
Maxime Costed2383f52017-03-23 17:35:20 +010045 for (i = 0; i < file_cnt; ++i) {
46 if (files[i] == NULL)
47 continue;
48
49 line = xmalloc_fgetline(files[i]);
50 if (!line) {
51 fclose_if_not_stdin(files[i]);
52 files[i] = NULL;
53 --active_files;
54 continue;
55 }
Ron Yorstoncad3fc72021-02-03 20:47:14 +010056 fputs_stdout(line);
Maxime Costed2383f52017-03-23 17:35:20 +010057 free(line);
58 delim = '\n';
59 if (i != file_cnt - 1) {
60 delim = delims[del_idx++];
61 if (del_idx == del_cnt)
62 del_idx = 0;
63 }
64 if (delim != '\0')
65 fputc(delim, stdout);
66 }
67 }
68}
69
70static void paste_files_separate(FILE** files, char* delims, int del_cnt)
71{
72 char *line, *next_line;
73 char delim;
Maxime Costed2383f52017-03-23 17:35:20 +010074 int i;
75
76 for (i = 0; files[i]; ++i) {
Denys Vlasenkof4a670a2017-03-23 17:58:32 +010077 int del_idx = 0;
78
Maxime Costed2383f52017-03-23 17:35:20 +010079 line = NULL;
80 while ((next_line = xmalloc_fgetline(files[i])) != NULL) {
81 if (line) {
Ron Yorstoncad3fc72021-02-03 20:47:14 +010082 fputs_stdout(line);
Maxime Costed2383f52017-03-23 17:35:20 +010083 free(line);
84 delim = delims[del_idx++];
85 if (del_idx == del_cnt)
86 del_idx = 0;
87 if (delim != '\0')
88 fputc(delim, stdout);
89 }
90 line = next_line;
91 }
92 if (line) {
93 /* coreutils adds \n even if this is a final line
94 * of the last file and it was not \n-terminated.
95 */
96 printf("%s\n", line);
97 free(line);
98 }
99 fclose_if_not_stdin(files[i]);
100 }
101}
102
103#define PASTE_OPT_DELIMITERS (1 << 0)
104#define PASTE_OPT_SEPARATE (1 << 1)
105
106int paste_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
107int paste_main(int argc UNUSED_PARAM, char **argv)
108{
109 char *delims = (char*)"\t";
110 int del_cnt = 1;
111 unsigned opt;
112 int i;
113
114 opt = getopt32(argv, "d:s", &delims);
115 argv += optind;
116
117 if (opt & PASTE_OPT_DELIMITERS) {
118 if (!delims[0])
James Byrne69374872019-07-02 11:35:03 +0200119 bb_simple_error_msg_and_die("-d '' is not supported");
Maxime Costed2383f52017-03-23 17:35:20 +0100120 /* unknown mappings are not changed: "\z" -> '\\' 'z' */
121 /* trailing backslash, if any, is preserved */
122 del_cnt = strcpy_and_process_escape_sequences(delims, delims) - delims;
123 /* note: handle NUL properly (do not stop at it!): try -d'\t\0\t' */
124 }
125
126 if (!argv[0])
127 (--argv)[0] = (char*) "-";
128 for (i = 0; argv[i]; ++i) {
129 argv[i] = (void*) fopen_or_warn_stdin(argv[i]);
Denys Vlasenko876c1212017-03-24 15:00:12 +0100130 if (!argv[i])
Maxime Costed2383f52017-03-23 17:35:20 +0100131 xfunc_die();
132 }
133
134 if (opt & PASTE_OPT_SEPARATE)
135 paste_files_separate((FILE**)argv, delims, del_cnt);
136 else
137 paste_files((FILE**)argv, i, delims, del_cnt);
138
139 fflush_stdout_and_exit(0);
140}