sksp2024-mcu/libucw/ucw/utils/ucw-urltool.c

146 lines
3 KiB
C
Raw Normal View History

2024-09-14 21:50:33 +02:00
/*
* LibUCW Utilities -- URL Handling Tool
*
* (c) 2004 Martin Mares <mj@ucw.cz>
*/
#include <ucw/lib.h>
#include <ucw/getopt.h>
#include <ucw/url.h>
#include <ucw/fastbuf.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
static byte *base_url;
static struct url base;
static uint opt_split = 0, opt_normalize = 0, opt_forgive = 0;
static struct fastbuf *fout;
static uint err_count;
static void
process_url(byte *url)
{
byte buf1[MAX_URL_SIZE], buf2[MAX_URL_SIZE], buf3[MAX_URL_SIZE], buf4[MAX_URL_SIZE];
int e;
struct url ur;
if ((e = url_deescape(url, buf1)) || (e = url_split(buf1, &ur, buf2)))
goto error;
if ((base_url || opt_normalize) && (e = url_normalize(&ur, &base)))
goto error;
if (opt_normalize && (e = url_canonicalize(&ur)))
goto error;
if (opt_split)
{
if (ur.protocol)
bprintf(fout, "protocol=%s\n", ur.protocol);
if (ur.user)
bprintf(fout, "user=%s\n", ur.user);
if (ur.pass)
bprintf(fout, "pass=%s\n", ur.pass);
if (ur.host)
bprintf(fout, "host=%s\n", ur.host);
if (ur.port != ~0U)
bprintf(fout, "port=%d\n", ur.port);
if (ur.rest)
bprintf(fout, "rest=%s\n", ur.rest);
bputc(fout, '\n');
}
else
{
if ((e = url_pack(&ur, buf3)) || (e = url_enescape(buf3, buf4)))
goto error;
bprintf(fout, "%s\n", buf4);
}
return;
error:
msg(L_ERROR, "%s: %s", url, url_error(e));
err_count++;
}
static char *shortopts = CF_SHORT_OPTS "b:fns";
static struct option longopts[] =
{
CF_LONG_OPTS
{ "base", 1, 0, 'b' },
{ "forgive", 0, 0, 'f' },
{ "normalize", 0, 0, 'n' },
{ "split", 0, 0, 's' },
{ NULL, 0, 0, 0 }
};
static char *help = "\
Usage: ucw-urltool [<options>] <operations> [<URL's>]\n\
\n\
Options:\n"
CF_USAGE "\
-b, --base <URL>\tInput URL's are relative to this base\n\
-f, --forgive\t\tReturn exit status 0 even if there were errors\n\
\n\
Operations:\n\
-s, --split\t\tSplit a given URL to components\n\
-n, --normalize\t\tNormalize given URL\n\
";
static void NONRET
usage(byte *msg)
{
if (msg)
{
fputs(msg, stderr);
fputc('\n', stderr);
}
fputs(help, stderr);
exit(1);
}
int
main(int argc, char **argv)
{
int opt, err;
byte *base_url = NULL;
byte basebuf1[MAX_URL_SIZE], basebuf2[MAX_URL_SIZE];
log_init(argv[0]);
while ((opt = cf_getopt(argc, argv, shortopts, longopts, NULL)) >= 0)
switch (opt)
{
case 'b':
base_url = optarg;
err = url_canon_split(base_url, basebuf1, basebuf2, &base);
if (err)
die("Invalid base URL: %s", url_error(err));
break;
case 's':
opt_split = 1;
break;
case 'n':
opt_normalize = 1;
break;
case 'f':
opt_forgive = 1;
break;
default:
usage("Invalid option");
}
fout = bfdopen_shared(1, 4096);
if (optind >= argc)
{
struct fastbuf *fin = bfdopen_shared(0, 4096);
byte url[MAX_URL_SIZE];
while (bgets(fin, url, sizeof(url)))
process_url(url);
bclose(fin);
}
else
while (optind < argc)
process_url(argv[optind++]);
bclose(fout);
return (err_count && !opt_forgive);
}