xref: /freebsd/lib/libfetch/fetch.c (revision f1951fd745b894fe6586c298874af98544a5e272)
1 /*-
2  * SPDX-License-Identifier: BSD-3-Clause
3  *
4  * Copyright (c) 1998-2004 Dag-Erling Smørgrav
5  * All rights reserved.
6  *
7  * Redistribution and use in source and binary forms, with or without
8  * modification, are permitted provided that the following conditions
9  * are met:
10  * 1. Redistributions of source code must retain the above copyright
11  *    notice, this list of conditions and the following disclaimer
12  *    in this position and unchanged.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  * 3. The name of the author may not be used to endorse or promote products
17  *    derived from this software without specific prior written permission
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20  * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21  * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22  * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24  * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30 
31 #include <sys/cdefs.h>
32 __FBSDID("$FreeBSD$");
33 
34 #include <sys/param.h>
35 #include <sys/errno.h>
36 
37 #include <ctype.h>
38 #include <stdio.h>
39 #include <stdlib.h>
40 #include <string.h>
41 
42 #include "fetch.h"
43 #include "common.h"
44 
45 auth_t	 fetchAuthMethod;
46 int	 fetchLastErrCode;
47 char	 fetchLastErrString[MAXERRSTRING];
48 int	 fetchTimeout;
49 int	 fetchRestartCalls = 1;
50 int	 fetchDebug;
51 
52 
53 /*** Local data **************************************************************/
54 
55 /*
56  * Error messages for parser errors
57  */
58 #define URL_MALFORMED		1
59 #define URL_BAD_SCHEME		2
60 #define URL_BAD_PORT		3
61 static struct fetcherr url_errlist[] = {
62 	{ URL_MALFORMED,	FETCH_URL,	"Malformed URL" },
63 	{ URL_BAD_SCHEME,	FETCH_URL,	"Invalid URL scheme" },
64 	{ URL_BAD_PORT,		FETCH_URL,	"Invalid server port" },
65 	{ -1,			FETCH_UNKNOWN,	"Unknown parser error" }
66 };
67 
68 
69 /*** Public API **************************************************************/
70 
71 /*
72  * Select the appropriate protocol for the URL scheme, and return a
73  * read-only stream connected to the document referenced by the URL.
74  * Also fill out the struct url_stat.
75  */
76 FILE *
77 fetchXGet(struct url *URL, struct url_stat *us, const char *flags)
78 {
79 
80 	if (us != NULL) {
81 		us->size = -1;
82 		us->atime = us->mtime = 0;
83 	}
84 	if (strcasecmp(URL->scheme, SCHEME_FILE) == 0)
85 		return (fetchXGetFile(URL, us, flags));
86 	else if (strcasecmp(URL->scheme, SCHEME_FTP) == 0)
87 		return (fetchXGetFTP(URL, us, flags));
88 	else if (strcasecmp(URL->scheme, SCHEME_HTTP) == 0)
89 		return (fetchXGetHTTP(URL, us, flags));
90 	else if (strcasecmp(URL->scheme, SCHEME_HTTPS) == 0)
91 		return (fetchXGetHTTP(URL, us, flags));
92 	url_seterr(URL_BAD_SCHEME);
93 	return (NULL);
94 }
95 
96 /*
97  * Select the appropriate protocol for the URL scheme, and return a
98  * read-only stream connected to the document referenced by the URL.
99  */
100 FILE *
101 fetchGet(struct url *URL, const char *flags)
102 {
103 	return (fetchXGet(URL, NULL, flags));
104 }
105 
106 /*
107  * Select the appropriate protocol for the URL scheme, and return a
108  * write-only stream connected to the document referenced by the URL.
109  */
110 FILE *
111 fetchPut(struct url *URL, const char *flags)
112 {
113 
114 	if (strcasecmp(URL->scheme, SCHEME_FILE) == 0)
115 		return (fetchPutFile(URL, flags));
116 	else if (strcasecmp(URL->scheme, SCHEME_FTP) == 0)
117 		return (fetchPutFTP(URL, flags));
118 	else if (strcasecmp(URL->scheme, SCHEME_HTTP) == 0)
119 		return (fetchPutHTTP(URL, flags));
120 	else if (strcasecmp(URL->scheme, SCHEME_HTTPS) == 0)
121 		return (fetchPutHTTP(URL, flags));
122 	url_seterr(URL_BAD_SCHEME);
123 	return (NULL);
124 }
125 
126 /*
127  * Select the appropriate protocol for the URL scheme, and return the
128  * size of the document referenced by the URL if it exists.
129  */
130 int
131 fetchStat(struct url *URL, struct url_stat *us, const char *flags)
132 {
133 
134 	if (us != NULL) {
135 		us->size = -1;
136 		us->atime = us->mtime = 0;
137 	}
138 	if (strcasecmp(URL->scheme, SCHEME_FILE) == 0)
139 		return (fetchStatFile(URL, us, flags));
140 	else if (strcasecmp(URL->scheme, SCHEME_FTP) == 0)
141 		return (fetchStatFTP(URL, us, flags));
142 	else if (strcasecmp(URL->scheme, SCHEME_HTTP) == 0)
143 		return (fetchStatHTTP(URL, us, flags));
144 	else if (strcasecmp(URL->scheme, SCHEME_HTTPS) == 0)
145 		return (fetchStatHTTP(URL, us, flags));
146 	url_seterr(URL_BAD_SCHEME);
147 	return (-1);
148 }
149 
150 /*
151  * Select the appropriate protocol for the URL scheme, and return a
152  * list of files in the directory pointed to by the URL.
153  */
154 struct url_ent *
155 fetchList(struct url *URL, const char *flags)
156 {
157 
158 	if (strcasecmp(URL->scheme, SCHEME_FILE) == 0)
159 		return (fetchListFile(URL, flags));
160 	else if (strcasecmp(URL->scheme, SCHEME_FTP) == 0)
161 		return (fetchListFTP(URL, flags));
162 	else if (strcasecmp(URL->scheme, SCHEME_HTTP) == 0)
163 		return (fetchListHTTP(URL, flags));
164 	else if (strcasecmp(URL->scheme, SCHEME_HTTPS) == 0)
165 		return (fetchListHTTP(URL, flags));
166 	url_seterr(URL_BAD_SCHEME);
167 	return (NULL);
168 }
169 
170 /*
171  * Attempt to parse the given URL; if successful, call fetchXGet().
172  */
173 FILE *
174 fetchXGetURL(const char *URL, struct url_stat *us, const char *flags)
175 {
176 	struct url *u;
177 	FILE *f;
178 
179 	if ((u = fetchParseURL(URL)) == NULL)
180 		return (NULL);
181 
182 	f = fetchXGet(u, us, flags);
183 
184 	fetchFreeURL(u);
185 	return (f);
186 }
187 
188 /*
189  * Attempt to parse the given URL; if successful, call fetchGet().
190  */
191 FILE *
192 fetchGetURL(const char *URL, const char *flags)
193 {
194 	return (fetchXGetURL(URL, NULL, flags));
195 }
196 
197 /*
198  * Attempt to parse the given URL; if successful, call fetchPut().
199  */
200 FILE *
201 fetchPutURL(const char *URL, const char *flags)
202 {
203 	struct url *u;
204 	FILE *f;
205 
206 	if ((u = fetchParseURL(URL)) == NULL)
207 		return (NULL);
208 
209 	f = fetchPut(u, flags);
210 
211 	fetchFreeURL(u);
212 	return (f);
213 }
214 
215 /*
216  * Attempt to parse the given URL; if successful, call fetchStat().
217  */
218 int
219 fetchStatURL(const char *URL, struct url_stat *us, const char *flags)
220 {
221 	struct url *u;
222 	int s;
223 
224 	if ((u = fetchParseURL(URL)) == NULL)
225 		return (-1);
226 
227 	s = fetchStat(u, us, flags);
228 
229 	fetchFreeURL(u);
230 	return (s);
231 }
232 
233 /*
234  * Attempt to parse the given URL; if successful, call fetchList().
235  */
236 struct url_ent *
237 fetchListURL(const char *URL, const char *flags)
238 {
239 	struct url *u;
240 	struct url_ent *ue;
241 
242 	if ((u = fetchParseURL(URL)) == NULL)
243 		return (NULL);
244 
245 	ue = fetchList(u, flags);
246 
247 	fetchFreeURL(u);
248 	return (ue);
249 }
250 
251 /*
252  * Make a URL
253  */
254 struct url *
255 fetchMakeURL(const char *scheme, const char *host, int port, const char *doc,
256     const char *user, const char *pwd)
257 {
258 	struct url *u;
259 
260 	if (!scheme || (!host && !doc)) {
261 		url_seterr(URL_MALFORMED);
262 		return (NULL);
263 	}
264 
265 	if (port < 0 || port > 65535) {
266 		url_seterr(URL_BAD_PORT);
267 		return (NULL);
268 	}
269 
270 	/* allocate struct url */
271 	if ((u = calloc(1, sizeof(*u))) == NULL) {
272 		fetch_syserr();
273 		return (NULL);
274 	}
275 	u->netrcfd = -1;
276 
277 	if ((u->doc = strdup(doc ? doc : "/")) == NULL) {
278 		fetch_syserr();
279 		free(u);
280 		return (NULL);
281 	}
282 
283 #define seturl(x) snprintf(u->x, sizeof(u->x), "%s", x)
284 	seturl(scheme);
285 	seturl(host);
286 	seturl(user);
287 	seturl(pwd);
288 #undef seturl
289 	u->port = port;
290 
291 	return (u);
292 }
293 
294 /*
295  * Return value of the given hex digit.
296  */
297 static int
298 fetch_hexval(char ch)
299 {
300 
301 	if (ch >= '0' && ch <= '9')
302 		return (ch - '0');
303 	else if (ch >= 'a' && ch <= 'f')
304 		return (ch - 'a' + 10);
305 	else if (ch >= 'A' && ch <= 'F')
306 		return (ch - 'A' + 10);
307 	return (-1);
308 }
309 
310 /*
311  * Decode percent-encoded URL component from src into dst, stopping at end
312  * of string, or at @ or : separators.  Returns a pointer to the unhandled
313  * part of the input string (null terminator, @, or :).  No terminator is
314  * written to dst (it is the caller's responsibility).
315  */
316 static const char *
317 fetch_pctdecode(char *dst, const char *src, size_t dlen)
318 {
319 	int d1, d2;
320 	char c;
321 	const char *s;
322 
323 	for (s = src; *s != '\0' && *s != '@' && *s != ':'; s++) {
324 		if (s[0] == '%' && (d1 = fetch_hexval(s[1])) >= 0 &&
325 		    (d2 = fetch_hexval(s[2])) >= 0 && (d1 > 0 || d2 > 0)) {
326 			c = d1 << 4 | d2;
327 			s += 2;
328 		} else {
329 			c = *s;
330 		}
331 		if (dlen-- > 0)
332 			*dst++ = c;
333 	}
334 	return (s);
335 }
336 
337 /*
338  * Split an URL into components. URL syntax is:
339  * [method:/][/[user[:pwd]@]host[:port]/][document]
340  * This almost, but not quite, RFC1738 URL syntax.
341  */
342 struct url *
343 fetchParseURL(const char *URL)
344 {
345 	char *doc;
346 	const char *p, *q;
347 	struct url *u;
348 	int i;
349 
350 	/* allocate struct url */
351 	if ((u = calloc(1, sizeof(*u))) == NULL) {
352 		fetch_syserr();
353 		return (NULL);
354 	}
355 	u->netrcfd = -1;
356 
357 	/* scheme name */
358 	if ((p = strstr(URL, ":/"))) {
359 		snprintf(u->scheme, URL_SCHEMELEN+1,
360 		    "%.*s", (int)(p - URL), URL);
361 		URL = ++p;
362 		/*
363 		 * Only one slash: no host, leave slash as part of document
364 		 * Two slashes: host follows, strip slashes
365 		 */
366 		if (URL[1] == '/')
367 			URL = (p += 2);
368 	} else {
369 		p = URL;
370 	}
371 	if (!*URL || *URL == '/' || *URL == '.' ||
372 	    (u->scheme[0] == '\0' &&
373 		strchr(URL, '/') == NULL && strchr(URL, ':') == NULL))
374 		goto nohost;
375 
376 	p = strpbrk(URL, "/@");
377 	if (p && *p == '@') {
378 		/* username */
379 		q = fetch_pctdecode(u->user, URL, URL_USERLEN);
380 
381 		/* password */
382 		if (*q == ':')
383 			q = fetch_pctdecode(u->pwd, q + 1, URL_PWDLEN);
384 
385 		p++;
386 	} else {
387 		p = URL;
388 	}
389 
390 	/* hostname */
391 	if (*p == '[' && (q = strchr(p + 1, ']')) != NULL &&
392 	    (*++q == '\0' || *q == '/' || *q == ':')) {
393 		if ((i = q - p) > MAXHOSTNAMELEN)
394 			i = MAXHOSTNAMELEN;
395 		strncpy(u->host, p, i);
396 		p = q;
397 	} else {
398 		for (i = 0; *p && (*p != '/') && (*p != ':'); p++)
399 			if (i < MAXHOSTNAMELEN)
400 				u->host[i++] = *p;
401 	}
402 
403 	/* port */
404 	if (*p == ':') {
405 		for (q = ++p; *q && (*q != '/'); q++)
406 			if (isdigit((unsigned char)*q))
407 				u->port = u->port * 10 + (*q - '0');
408 			else {
409 				/* invalid port */
410 				url_seterr(URL_BAD_PORT);
411 				goto ouch;
412 			}
413 		p = q;
414 	}
415 
416 nohost:
417 	/* document */
418 	if (!*p)
419 		p = "/";
420 
421 	if (strcasecmp(u->scheme, SCHEME_HTTP) == 0 ||
422 	    strcasecmp(u->scheme, SCHEME_HTTPS) == 0) {
423 		const char hexnums[] = "0123456789abcdef";
424 
425 		/* percent-escape whitespace. */
426 		if ((doc = malloc(strlen(p) * 3 + 1)) == NULL) {
427 			fetch_syserr();
428 			goto ouch;
429 		}
430 		u->doc = doc;
431 		while (*p != '\0') {
432 			if (!isspace((unsigned char)*p)) {
433 				*doc++ = *p++;
434 			} else {
435 				*doc++ = '%';
436 				*doc++ = hexnums[((unsigned int)*p) >> 4];
437 				*doc++ = hexnums[((unsigned int)*p) & 0xf];
438 				p++;
439 			}
440 		}
441 		*doc = '\0';
442 	} else if ((u->doc = strdup(p)) == NULL) {
443 		fetch_syserr();
444 		goto ouch;
445 	}
446 
447 	DEBUGF("scheme:   \"%s\"\n"
448 	    "user:     \"%s\"\n"
449 	    "password: \"%s\"\n"
450 	    "host:     \"%s\"\n"
451 	    "port:     \"%d\"\n"
452 	    "document: \"%s\"\n",
453 	    u->scheme, u->user, u->pwd,
454 	    u->host, u->port, u->doc);
455 
456 	return (u);
457 
458 ouch:
459 	free(u);
460 	return (NULL);
461 }
462 
463 /*
464  * Free a URL
465  */
466 void
467 fetchFreeURL(struct url *u)
468 {
469 	free(u->doc);
470 	free(u);
471 }
472