paste: new applet
function old new delta paste_main - 493 +493 packed_usage 31019 31070 +51 applet_names 2569 2575 +6 applet_main 1484 1488 +4 ------------------------------------------------------------------------------ (add/remove: 2/0 grow/shrink: 3/0 up/down: 554/0) Total: 554 bytes Signed-off-by: Maxime Coste <mawww@kakoune.org> Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
parent
dba0dc1999
commit
d2383f57cd
3
AUTHORS
3
AUTHORS
@ -178,3 +178,6 @@ Mike Frysinger <vapier@gentoo.org>
|
||||
|
||||
Jie Zhang <jie.zhang@analog.com>
|
||||
fixed two bugs in msh and hush (exitcode of killed processes)
|
||||
|
||||
Maxime Coste <mawww@kakoune.org>
|
||||
paste implementation
|
||||
|
138
coreutils/paste.c
Normal file
138
coreutils/paste.c
Normal file
@ -0,0 +1,138 @@
|
||||
/* vi: set sw=4 ts=4: */
|
||||
/*
|
||||
* paste.c - implementation of the posix paste command
|
||||
*
|
||||
* Written by Maxime Coste <mawww@kakoune.org>
|
||||
*
|
||||
* Licensed under GPLv2 or later, see file LICENSE in this source tree.
|
||||
*/
|
||||
//config:config PASTE
|
||||
//config: bool "paste"
|
||||
//config: default y
|
||||
//config: help
|
||||
//config: paste is used to paste lines of different files together
|
||||
//config: and write the result to stdout
|
||||
|
||||
//applet:IF_PASTE(APPLET_NOEXEC(paste, paste, BB_DIR_USR_BIN, BB_SUID_DROP, paste))
|
||||
|
||||
//kbuild:lib-$(CONFIG_PASTE) += paste.o
|
||||
|
||||
//usage:#define paste_trivial_usage
|
||||
//usage: "[OPTIONS] [FILE]..."
|
||||
//usage:#define paste_full_usage "\n\n"
|
||||
//usage: "Paste lines from each input file, seperated with tab\n"
|
||||
//usage: "\n -d LIST Use delimiters from LIST, not tab"
|
||||
//usage: "\n -s Serial: one file at a time"
|
||||
//usage:
|
||||
//usage:#define paste_example_usage
|
||||
//usage: "# write out directory in four columns\n"
|
||||
//usage: "$ ls | paste - - - -\n"
|
||||
//usage: "# combine pairs of lines from a file into single lines\n"
|
||||
//usage: "$ paste -s -d '\\t\\n' file\n"
|
||||
|
||||
#include "libbb.h"
|
||||
|
||||
static void paste_files(FILE** files, int file_cnt, char* delims, int del_cnt)
|
||||
{
|
||||
char *line;
|
||||
char delim;
|
||||
int del_idx = 0;
|
||||
int active_files = file_cnt;
|
||||
int i;
|
||||
|
||||
while (active_files > 0) {
|
||||
for (i = 0; i < file_cnt; ++i) {
|
||||
if (files[i] == NULL)
|
||||
continue;
|
||||
|
||||
line = xmalloc_fgetline(files[i]);
|
||||
if (!line) {
|
||||
fclose_if_not_stdin(files[i]);
|
||||
files[i] = NULL;
|
||||
--active_files;
|
||||
continue;
|
||||
}
|
||||
fputs(line, stdout);
|
||||
free(line);
|
||||
delim = '\n';
|
||||
if (i != file_cnt - 1) {
|
||||
delim = delims[del_idx++];
|
||||
if (del_idx == del_cnt)
|
||||
del_idx = 0;
|
||||
}
|
||||
if (delim != '\0')
|
||||
fputc(delim, stdout);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void paste_files_separate(FILE** files, char* delims, int del_cnt)
|
||||
{
|
||||
char *line, *next_line;
|
||||
char delim;
|
||||
int del_idx = 0;
|
||||
int i;
|
||||
|
||||
for (i = 0; files[i]; ++i) {
|
||||
line = NULL;
|
||||
while ((next_line = xmalloc_fgetline(files[i])) != NULL) {
|
||||
if (line) {
|
||||
fputs(line, stdout);
|
||||
free(line);
|
||||
delim = delims[del_idx++];
|
||||
if (del_idx == del_cnt)
|
||||
del_idx = 0;
|
||||
if (delim != '\0')
|
||||
fputc(delim, stdout);
|
||||
}
|
||||
line = next_line;
|
||||
}
|
||||
if (line) {
|
||||
/* coreutils adds \n even if this is a final line
|
||||
* of the last file and it was not \n-terminated.
|
||||
*/
|
||||
printf("%s\n", line);
|
||||
free(line);
|
||||
}
|
||||
fclose_if_not_stdin(files[i]);
|
||||
}
|
||||
}
|
||||
|
||||
#define PASTE_OPT_DELIMITERS (1 << 0)
|
||||
#define PASTE_OPT_SEPARATE (1 << 1)
|
||||
|
||||
int paste_main(int argc, char **argv) MAIN_EXTERNALLY_VISIBLE;
|
||||
int paste_main(int argc UNUSED_PARAM, char **argv)
|
||||
{
|
||||
char *delims = (char*)"\t";
|
||||
int del_cnt = 1;
|
||||
unsigned opt;
|
||||
int i;
|
||||
|
||||
opt = getopt32(argv, "d:s", &delims);
|
||||
argv += optind;
|
||||
|
||||
if (opt & PASTE_OPT_DELIMITERS) {
|
||||
if (!delims[0])
|
||||
bb_error_msg_and_die("-d '' is not supported");
|
||||
/* unknown mappings are not changed: "\z" -> '\\' 'z' */
|
||||
/* trailing backslash, if any, is preserved */
|
||||
del_cnt = strcpy_and_process_escape_sequences(delims, delims) - delims;
|
||||
/* note: handle NUL properly (do not stop at it!): try -d'\t\0\t' */
|
||||
}
|
||||
|
||||
if (!argv[0])
|
||||
(--argv)[0] = (char*) "-";
|
||||
for (i = 0; argv[i]; ++i) {
|
||||
argv[i] = (void*) fopen_or_warn_stdin(argv[i]);
|
||||
if (!argv[i])
|
||||
xfunc_die();
|
||||
}
|
||||
|
||||
if (opt & PASTE_OPT_SEPARATE)
|
||||
paste_files_separate((FILE**)argv, delims, del_cnt);
|
||||
else
|
||||
paste_files((FILE**)argv, i, delims, del_cnt);
|
||||
|
||||
fflush_stdout_and_exit(0);
|
||||
}
|
@ -22,7 +22,7 @@ POSIX Tools supported only as shell built-ins (ash shell):
|
||||
POSIX Tools not supported:
|
||||
asa, at, batch, bc, c99, command, compress, csplit, ex, fc, file,
|
||||
gencat, getconf, iconv, join, link, locale, localedef, lp, m4,
|
||||
mailx, newgrp, nl, paste, pathchk, pax, pr, qalter, qdel, qhold, qmove,
|
||||
mailx, newgrp, nl, pathchk, pax, pr, qalter, qdel, qhold, qmove,
|
||||
qmsg, qrerun, qrls, qselect, qsig, qstat, qsub, tabs, talk, tput,
|
||||
tsort, unlink, uucp, uustat, uux
|
||||
|
||||
@ -469,6 +469,12 @@ od POSIX options
|
||||
-x | no | no |
|
||||
od Busybox specific options: None
|
||||
|
||||
paste POSIX options
|
||||
option | exists | compliant | remarks
|
||||
-d list | yes | yes |
|
||||
-s | yes | yes |
|
||||
paste Busybox specific options: None
|
||||
|
||||
patch POSIX options
|
||||
option | exists | compliant | remarks
|
||||
-D define | no | no |
|
||||
|
20
testsuite/paste/paste
Normal file
20
testsuite/paste/paste
Normal file
@ -0,0 +1,20 @@
|
||||
cat > foo <<EOF
|
||||
foo1
|
||||
foo2
|
||||
foo3
|
||||
EOF
|
||||
|
||||
cat > bar <<EOF
|
||||
bar1
|
||||
bar2
|
||||
bar3
|
||||
EOF
|
||||
|
||||
cat > baz <<EOF
|
||||
foo1 bar1
|
||||
foo2 bar2
|
||||
foo3 bar3
|
||||
EOF
|
||||
|
||||
busybox paste foo bar > qux
|
||||
diff -u baz qux
|
9
testsuite/paste/paste-back-cuted-lines
Normal file
9
testsuite/paste/paste-back-cuted-lines
Normal file
@ -0,0 +1,9 @@
|
||||
cat > foo <<EOF
|
||||
this is the first line
|
||||
this is the second line
|
||||
this is the third line
|
||||
EOF
|
||||
cut -b 1-13 -n foo > foo1
|
||||
cut -b 14- -n foo > foo2
|
||||
busybox paste -d '\0' foo1 foo2 > bar
|
||||
cmp foo bar
|
16
testsuite/paste/paste-multi-stdin
Normal file
16
testsuite/paste/paste-multi-stdin
Normal file
@ -0,0 +1,16 @@
|
||||
cat > foo <<EOF
|
||||
line1
|
||||
line2
|
||||
line3
|
||||
line4
|
||||
line5
|
||||
line6
|
||||
EOF
|
||||
|
||||
cat > bar <<EOF
|
||||
line1 line2 line3
|
||||
line4 line5 line6
|
||||
EOF
|
||||
|
||||
busybox paste - - - < foo > baz
|
||||
cmp bar baz
|
16
testsuite/paste/paste-pairs
Normal file
16
testsuite/paste/paste-pairs
Normal file
@ -0,0 +1,16 @@
|
||||
cat > foo <<EOF
|
||||
foo1
|
||||
bar1
|
||||
foo2
|
||||
bar2
|
||||
foo3
|
||||
EOF
|
||||
|
||||
cat > bar <<EOF
|
||||
foo1 bar1
|
||||
foo2 bar2
|
||||
foo3
|
||||
EOF
|
||||
|
||||
busybox paste -s -d "\t\n" foo > baz
|
||||
cmp bar baz
|
19
testsuite/paste/paste-separate
Normal file
19
testsuite/paste/paste-separate
Normal file
@ -0,0 +1,19 @@
|
||||
cat > foo <<EOF
|
||||
foo1
|
||||
foo2
|
||||
foo3
|
||||
EOF
|
||||
|
||||
cat > bar <<EOF
|
||||
bar1
|
||||
bar2
|
||||
bar3
|
||||
EOF
|
||||
|
||||
cat > baz <<EOF
|
||||
foo1 foo2 foo3
|
||||
bar1 bar2 bar3
|
||||
EOF
|
||||
|
||||
busybox paste -s foo bar > qux
|
||||
cmp baz qux
|
Loading…
x
Reference in New Issue
Block a user