1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
|
/* ISC license. */
#include <string.h>
#include <skalibs/sgetopt.h>
#include <skalibs/types.h>
#include <skalibs/strerr2.h>
#include <skalibs/allreadwrite.h>
#include <skalibs/skamisc.h>
#define USAGE "s6-unquote [ -n ] [ -d delim ] string"
int main (int argc, char const *const *argv)
{
char const *delim = "\"" ;
char const *string ;
size_t len, delimlen ;
int nl = 1 ;
PROG = "s6-unquote" ;
{
subgetopt_t l = SUBGETOPT_ZERO ;
for (;;)
{
int opt = subgetopt_r(argc, argv, "nd:", &l) ;
if (opt == -1) break ;
switch (opt)
{
case 'n' : nl = 0 ; break ;
case 'd': delim = l.arg ; break ;
default : strerr_dieusage(100, USAGE) ;
}
}
argc -= l.ind ; argv += l.ind ;
}
if (!argc) strerr_dieusage(100, USAGE) ;
string = *argv ;
len = strlen(string) ;
delimlen = strlen(delim) ;
if (delimlen)
{
if (!len--) strerr_dief1x(100, "the empty string isn't a quoted string") ;
if (!memchr(delim, *string++, delimlen))
strerr_dief1x(100, "invalid starting quote character") ;
}
{
size_t r = 0, w = 0 ;
char buf[len+1] ;
if (!string_unquote_withdelim(buf, &w, string, len, &r, delim, delimlen))
{
char fmt[SIZE_FMT] ;
fmt[size_fmt(fmt, r + !!delimlen)] = 0 ;
strerr_diefu2sys(100, "unquote at character ", fmt) ;
}
if (delimlen)
{
if (r == len) strerr_dief1x(100, "no ending quote character") ;
else if (r < len - 1)
{
char fmtnum[SIZE_FMT] ;
char fmtden[SIZE_FMT] ;
fmtnum[size_fmt(fmtnum, r+1)] = 0 ;
fmtden[size_fmt(fmtden, len)] = 0 ;
strerr_warnw5x("found ending quote character at position ", fmtnum, "/", fmtden, "; ignoring remainder") ;
}
}
if (nl) buf[w++] = '\n' ;
if (allwrite(1, buf, w) < w)
strerr_diefu1sys(111, "write to stdout") ;
}
return 0 ;
}
|