2008-05-24 16:05:49 +08:00
|
|
|
/* tinyproxy - A fast light-weight HTTP proxy
|
|
|
|
* Copyright (C) 2003 Steven Young <sdyoung@miranda.org>
|
2003-03-15 06:45:59 +08:00
|
|
|
*
|
2008-05-24 16:05:49 +08:00
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
2003-03-15 06:45:59 +08:00
|
|
|
*
|
2008-05-24 16:05:49 +08:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2003-03-15 06:45:59 +08:00
|
|
|
*
|
2008-05-24 16:05:49 +08:00
|
|
|
* You should have received a copy of the GNU General Public License along
|
|
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* This file contains source code for the handling and display of
|
|
|
|
* HTML error pages with variable substitution.
|
2003-03-14 05:25:06 +08:00
|
|
|
*/
|
2003-03-15 06:45:59 +08:00
|
|
|
|
2009-08-07 06:12:53 +08:00
|
|
|
#include "main.h"
|
2003-03-14 05:25:06 +08:00
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "buffer.h"
|
|
|
|
#include "conns.h"
|
|
|
|
#include "heap.h"
|
2008-05-24 16:17:14 +08:00
|
|
|
#include "html-error.h"
|
2003-03-14 05:25:06 +08:00
|
|
|
#include "network.h"
|
|
|
|
#include "utils.h"
|
2009-12-08 05:23:31 +08:00
|
|
|
#include "conf.h"
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2003-03-15 06:45:59 +08:00
|
|
|
/*
|
|
|
|
* Add an error number -> filename mapping to the errorpages list.
|
|
|
|
*/
|
2008-12-08 21:39:44 +08:00
|
|
|
#define ERRORNUM_BUFSIZE 8 /* this is more than required */
|
2003-08-01 08:14:34 +08:00
|
|
|
#define ERRPAGES_BUCKETCOUNT 16
|
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
int add_new_errorpage (char *filepath, unsigned int errornum)
|
2005-08-15 11:54:31 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
char errornbuf[ERRORNUM_BUFSIZE];
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
config->errorpages = hashmap_create (ERRPAGES_BUCKETCOUNT);
|
|
|
|
if (!config->errorpages)
|
2009-09-15 03:41:25 +08:00
|
|
|
return (-1);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
snprintf (errornbuf, ERRORNUM_BUFSIZE, "%u", errornum);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
if (hashmap_insert (config->errorpages, errornbuf,
|
2009-09-15 03:41:25 +08:00
|
|
|
filepath, strlen (filepath) + 1) < 0)
|
|
|
|
return (-1);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
return (0);
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
2003-03-15 06:45:59 +08:00
|
|
|
/*
|
|
|
|
* Get the file appropriate for a given error.
|
|
|
|
*/
|
2009-09-15 03:41:25 +08:00
|
|
|
static char *get_html_file (unsigned int errornum)
|
2005-08-15 11:54:31 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
hashmap_iter result_iter;
|
|
|
|
char errornbuf[ERRORNUM_BUFSIZE];
|
|
|
|
char *key;
|
2010-01-26 02:10:03 +08:00
|
|
|
char *val;
|
2003-08-01 08:14:34 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
assert (errornum >= 100 && errornum < 1000);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
if (!config->errorpages)
|
|
|
|
return (config->errorpage_undef);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
snprintf (errornbuf, ERRORNUM_BUFSIZE, "%u", errornum);
|
2003-08-01 08:14:34 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
result_iter = hashmap_find (config->errorpages, errornbuf);
|
2003-08-01 08:14:34 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
if (hashmap_is_end (config->errorpages, result_iter))
|
|
|
|
return (config->errorpage_undef);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2020-01-16 00:09:41 +08:00
|
|
|
if (hashmap_return_entry (config->errorpages, result_iter,
|
2009-09-15 03:41:25 +08:00
|
|
|
&key, (void **) &val) < 0)
|
2020-01-16 00:09:41 +08:00
|
|
|
return (config->errorpage_undef);
|
2003-08-01 08:14:34 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
return (val);
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
2020-09-07 11:32:13 +08:00
|
|
|
static void varsubst_sendline(struct conn_s *connptr, regex_t *re, char *p) {
|
|
|
|
int fd = connptr->client_fd;
|
|
|
|
while(*p) {
|
|
|
|
regmatch_t match;
|
|
|
|
char varname[32+1], *varval;
|
|
|
|
size_t l;
|
|
|
|
int st = regexec(re, p, 1, &match, 0);
|
|
|
|
if(st == 0) {
|
|
|
|
if(match.rm_so > 0) safe_write(fd, p, match.rm_so);
|
|
|
|
l = match.rm_eo - match.rm_so;
|
|
|
|
assert(l>2 && l-2 < sizeof(varname));
|
|
|
|
p += match.rm_so;
|
|
|
|
memcpy(varname, p+1, l-2);
|
|
|
|
varname[l-2] = 0;
|
|
|
|
varval = lookup_variable(connptr->error_variables, varname);
|
|
|
|
if(varval) write_message(fd, "%s", varval);
|
|
|
|
else if(varval && !*varval) write_message(fd, "(unknown)");
|
|
|
|
else safe_write(fd, p, l);
|
|
|
|
p += l;
|
|
|
|
} else {
|
|
|
|
write_message(fd, "%s", p);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-03-15 06:45:59 +08:00
|
|
|
/*
|
|
|
|
* Send an already-opened file to the client with variable substitution.
|
|
|
|
*/
|
2009-10-02 15:31:32 +08:00
|
|
|
int
|
|
|
|
send_html_file (FILE *infile, struct conn_s *connptr)
|
2005-08-15 11:54:31 +08:00
|
|
|
{
|
2020-09-07 11:32:13 +08:00
|
|
|
regex_t re;
|
|
|
|
char *inbuf = safemalloc (4096);
|
|
|
|
(void) regcomp(&re, "{[a-z]\\{1,32\\}}", 0);
|
|
|
|
|
|
|
|
while (fgets (inbuf, 4096, infile)) {
|
|
|
|
varsubst_sendline(connptr, &re, inbuf);
|
2008-12-08 21:39:44 +08:00
|
|
|
}
|
2009-10-02 15:31:32 +08:00
|
|
|
|
2020-09-07 11:32:13 +08:00
|
|
|
regfree (&re);
|
2009-10-02 15:31:32 +08:00
|
|
|
safefree (inbuf);
|
2020-09-07 11:32:13 +08:00
|
|
|
return 1;
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
int send_http_headers (struct conn_s *connptr, int code, const char *message)
|
2005-08-15 11:54:31 +08:00
|
|
|
{
|
2018-01-28 12:32:59 +08:00
|
|
|
const char headers[] =
|
2009-09-15 03:41:25 +08:00
|
|
|
"HTTP/1.0 %d %s\r\n"
|
|
|
|
"Server: %s/%s\r\n"
|
2018-01-28 12:32:59 +08:00
|
|
|
"Content-Type: text/html\r\n"
|
|
|
|
"%s"
|
|
|
|
"Connection: close\r\n" "\r\n";
|
|
|
|
|
2019-06-14 08:18:17 +08:00
|
|
|
const char p_auth_str[] =
|
2018-01-28 12:32:59 +08:00
|
|
|
"Proxy-Authenticate: Basic realm=\""
|
|
|
|
PACKAGE_NAME "\"\r\n";
|
|
|
|
|
2019-06-14 08:18:17 +08:00
|
|
|
const char w_auth_str[] =
|
|
|
|
"WWW-Authenticate: Basic realm=\""
|
|
|
|
PACKAGE_NAME "\"\r\n";
|
|
|
|
|
2018-01-28 12:32:59 +08:00
|
|
|
/* according to rfc7235, the 407 error must be accompanied by
|
|
|
|
a Proxy-Authenticate header field. */
|
2019-06-14 08:18:17 +08:00
|
|
|
const char *add = code == 407 ? p_auth_str : (code == 401 ? w_auth_str : "");
|
2005-08-15 11:54:31 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
return (write_message (connptr->client_fd, headers,
|
2018-01-28 12:32:59 +08:00
|
|
|
code, message, PACKAGE, VERSION,
|
|
|
|
add));
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Display an error to the client.
|
|
|
|
*/
|
2009-09-15 03:41:25 +08:00
|
|
|
int send_http_error_message (struct conn_s *connptr)
|
2003-03-14 05:25:06 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
char *error_file;
|
|
|
|
FILE *infile;
|
|
|
|
int ret;
|
|
|
|
const char *fallback_error =
|
|
|
|
"<?xml version=\"1.0\" encoding=\"UTF-8\" ?>\n"
|
|
|
|
"<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.1//EN\" "
|
|
|
|
"\"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd\">\n"
|
|
|
|
"<html>\n"
|
|
|
|
"<head><title>%d %s</title></head>\n"
|
|
|
|
"<body>\n"
|
|
|
|
"<h1>%s</h1>\n"
|
|
|
|
"<p>%s</p>\n"
|
|
|
|
"<hr />\n"
|
|
|
|
"<p><em>Generated by %s version %s.</em></p>\n" "</body>\n"
|
|
|
|
"</html>\n";
|
|
|
|
|
|
|
|
send_http_headers (connptr, connptr->error_number,
|
|
|
|
connptr->error_string);
|
|
|
|
|
|
|
|
error_file = get_html_file (connptr->error_number);
|
|
|
|
if (!(infile = fopen (error_file, "r"))) {
|
2016-12-28 01:18:16 +08:00
|
|
|
char *detail = lookup_variable (connptr->error_variables, "detail");
|
2009-09-15 03:41:25 +08:00
|
|
|
return (write_message (connptr->client_fd, fallback_error,
|
|
|
|
connptr->error_number,
|
|
|
|
connptr->error_string,
|
|
|
|
connptr->error_string,
|
|
|
|
detail, PACKAGE, VERSION));
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = send_html_file (infile, connptr);
|
|
|
|
fclose (infile);
|
|
|
|
return (ret);
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
2008-05-24 16:05:49 +08:00
|
|
|
/*
|
2003-03-14 05:25:06 +08:00
|
|
|
* Add a key -> value mapping for HTML file substitution.
|
|
|
|
*/
|
2003-08-01 08:14:34 +08:00
|
|
|
|
|
|
|
#define ERRVAR_BUCKETCOUNT 16
|
|
|
|
|
2005-08-15 11:54:31 +08:00
|
|
|
int
|
2009-08-07 14:41:36 +08:00
|
|
|
add_error_variable (struct conn_s *connptr, const char *key, const char *val)
|
2003-03-14 05:25:06 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
if (!connptr->error_variables)
|
|
|
|
if (!
|
|
|
|
(connptr->error_variables =
|
|
|
|
hashmap_create (ERRVAR_BUCKETCOUNT)))
|
|
|
|
return (-1);
|
|
|
|
|
|
|
|
return hashmap_insert (connptr->error_variables, key, val,
|
|
|
|
strlen (val) + 1);
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
2008-06-10 12:21:23 +08:00
|
|
|
#define ADD_VAR_RET(x, y) \
|
2008-07-14 19:43:06 +08:00
|
|
|
do { \
|
|
|
|
if (y == NULL) \
|
|
|
|
break; \
|
2008-06-10 12:21:23 +08:00
|
|
|
if (add_error_variable(connptr, x, y) < 0) \
|
|
|
|
return -1; \
|
|
|
|
} while (0)
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2003-03-15 06:45:59 +08:00
|
|
|
/*
|
|
|
|
* Set some standard variables used by all HTML pages
|
|
|
|
*/
|
2009-09-15 03:41:25 +08:00
|
|
|
int add_standard_vars (struct conn_s *connptr)
|
2005-08-15 11:54:31 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
char errnobuf[16];
|
|
|
|
char timebuf[30];
|
|
|
|
time_t global_time;
|
|
|
|
|
|
|
|
snprintf (errnobuf, sizeof errnobuf, "%d", connptr->error_number);
|
|
|
|
ADD_VAR_RET ("errno", errnobuf);
|
|
|
|
|
|
|
|
ADD_VAR_RET ("cause", connptr->error_string);
|
|
|
|
ADD_VAR_RET ("request", connptr->request_line);
|
|
|
|
ADD_VAR_RET ("clientip", connptr->client_ip_addr);
|
|
|
|
|
|
|
|
/* The following value parts are all non-NULL and will
|
|
|
|
* trigger warnings in ADD_VAR_RET(), so we use
|
|
|
|
* add_error_variable() directly.
|
|
|
|
*/
|
|
|
|
|
|
|
|
global_time = time (NULL);
|
|
|
|
strftime (timebuf, sizeof (timebuf), "%a, %d %b %Y %H:%M:%S GMT",
|
|
|
|
gmtime (&global_time));
|
|
|
|
add_error_variable (connptr, "date", timebuf);
|
|
|
|
|
|
|
|
add_error_variable (connptr, "website",
|
2016-01-04 05:16:21 +08:00
|
|
|
"https://tinyproxy.github.io/");
|
2009-09-15 03:41:25 +08:00
|
|
|
add_error_variable (connptr, "version", VERSION);
|
|
|
|
add_error_variable (connptr, "package", PACKAGE);
|
|
|
|
|
|
|
|
return (0);
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add the error information to the conn structure.
|
|
|
|
*/
|
|
|
|
int
|
2009-08-07 14:48:51 +08:00
|
|
|
indicate_http_error (struct conn_s *connptr, int number,
|
|
|
|
const char *message, ...)
|
2003-03-14 05:25:06 +08:00
|
|
|
{
|
2009-09-15 03:41:25 +08:00
|
|
|
va_list ap;
|
|
|
|
char *key, *val;
|
2005-08-15 11:54:31 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
va_start (ap, message);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
while ((key = va_arg (ap, char *))) {
|
|
|
|
val = va_arg (ap, char *);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
if (add_error_variable (connptr, key, val) == -1) {
|
|
|
|
va_end (ap);
|
|
|
|
return (-1);
|
|
|
|
}
|
2008-12-08 21:39:44 +08:00
|
|
|
}
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
connptr->error_number = number;
|
|
|
|
connptr->error_string = safestrdup (message);
|
2003-03-14 05:25:06 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
va_end (ap);
|
2003-07-15 01:42:43 +08:00
|
|
|
|
2009-09-15 03:41:25 +08:00
|
|
|
return (add_standard_vars (connptr));
|
2003-03-14 05:25:06 +08:00
|
|
|
}
|