ndk-busybox/networking/httpd.c

1818 lines
48 KiB
C
Raw Normal View History

2003-01-05 04:01:56 +00:00
/*
* httpd implementation for busybox
*
* Copyright (C) 2002,2003 Glenn Engel <glenne@engel.org>
* Copyright (C) 2003 Vladimir Oleynik <dzo@simtreas.ru>
2003-01-05 04:01:56 +00:00
*
* simplify patch stolen from libbb without using strdup
2003-01-05 04:01:56 +00: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.
*
* 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.
*
* 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
*****************************************************************************
*
* Typical usage:
* for non root user
* httpd -p 8080 -h $HOME/public_html
* or for daemon start from rc script with uid=0:
* httpd -u www
* This is equivalent if www user have uid=80 to
* httpd -p 80 -u 80 -h /www -c /etc/httpd.conf -r "Web Server Authentication"
*
2003-01-05 04:01:56 +00:00
*
* When a url contains "cgi-bin" it is assumed to be a cgi script. The
* server changes directory to the location of the script and executes it
* after setting QUERY_STRING and other environment variables. If url args
* are included in the url or as a post, the args are placed into decoded
* environment variables. e.g. /cgi-bin/setup?foo=Hello%20World will set
* the $CGI_foo environment variable to "Hello World" while
* CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV enabled.
2003-01-05 04:01:56 +00:00
*
* The server can also be invoked as a url arg decoder and html text encoder
* as follows:
* foo=`httpd -d $foo` # decode "Hello%20World" as "Hello World"
* bar=`httpd -e "<Hello World>"` # encode as "%3CHello%20World%3E"
2003-01-05 04:01:56 +00:00
*
* httpd.conf has the following format:
A:172.20. # Allow any address that begins with 172.20
A:10.10. # Allow any address that begins with 10.10.
A:10.10 # Allow any address that previous set and 10.100-109.X.X
A:127.0.0.1 # Allow local loopback connections
D:* # Deny from other IP connections
/cgi-bin:foo:bar # Require user foo, pwd bar on urls starting with /cgi-bin/
/adm:admin:setup # Require user admin, pwd setup on urls starting with /adm/
/adm:toor:PaSsWd # or user toor, pwd PaSsWd on urls starting with /adm/
.au:audio/basic # additional mime type for audio.au files
A shortes path and D:from[^*] automaticaly sorting to top.
All longest path can`t reset user:password if shorted protect setted.
A/D may be as a/d or allow/deny - first char case unsensitive parsed only.
Each subdir can have config file.
For protect as user:pass current subdir and subpathes set from subdir config:
/:user:pass
if not, other subpathes for give effect must have path from httpd root
/current_subdir_path_from_httpd_root/subpath:user:pass
The Deny/Allow IP logic:
1. Allow all:
The config don`t set D: lines
2. Allow from setted only:
see the begin format example
3. Set deny, allow from other:
D:1.2.3. # deny from 1.2.3.0 - 1.2.3.255
D:2.3.4. # deny from 2.3.4.0 - 2.3.4.255
A:* # allow from other, this line not strongly require
A global and subdirs config merging logic:
allow rules reducing, deny lines strongled.
The algorithm combinations:
4. If current config have
A:from
D:*
subdir config A: lines skiping, D:from - moving top
5. If current config have
D:from
A:*
result config:
D:from current
D:from subdir
A:from subdir
and seting D:* if subdir config have this
If -c don`t setted, used httpd root config, else httpd root config skiped.
Exited with fault if can`t open start config.
For set wide open server, use -c /dev/null ;=)
2003-01-05 04:01:56 +00:00
*/
2003-01-05 04:01:56 +00:00
#include <stdio.h>
#include <ctype.h> /* for isspace */
#include <string.h>
2003-01-05 04:01:56 +00:00
#include <stdlib.h> /* for malloc */
#include <time.h>
#include <unistd.h> /* for close */
#include <signal.h>
#include <sys/types.h>
#include <sys/socket.h> /* for connect and socket*/
#include <netinet/in.h> /* for sockaddr_in */
#include <sys/time.h>
2003-01-05 04:01:56 +00:00
#include <sys/stat.h>
#include <sys/wait.h>
#include <fcntl.h> /* for open modes */
#include "busybox.h"
2003-01-05 04:01:56 +00:00
static const char httpdVersion[] = "busybox httpd/1.20 31-Jan-2003";
static const char default_patch_httpd_conf[] = "/etc";
static const char httpd_conf[] = "httpd.conf";
static const char home[] = "/www";
2003-01-05 04:01:56 +00:00
// Note: xfuncs are not used because we want the server to keep running
// if something bad happens due to a malformed user request.
// As a result, all memory allocation after daemonize
// is checked rigorously
//#define DEBUG 1
/* Configure options, disabled by default as nonstandart httpd feature */
//#define CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
//#define CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
//#define CONFIG_FEATURE_HTTPD_DECODE_URL_STR
/* disabled as not necessary feature */
//#define CONFIG_FEATURE_HTTPD_SET_REMOTE_PORT_TO_ENV
//#define CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
//#define CONFIG_FEATURE_HTTPD_SETUID
//#define CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP
/* If seted this you can use this server from internet superserver only */
//#define CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
/* You can use this server as standalone, require libbb.a for linking */
//#define HTTPD_STANDALONE
/* Config options, disable this for do very small module */
//#define CONFIG_FEATURE_HTTPD_CGI
//#define CONFIG_FEATURE_HTTPD_BASIC_AUTH
#ifdef HTTPD_STANDALONE
/* standalone, enable all features */
#undef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
/* unset config option for remove warning as redefined */
#undef CONFIG_FEATURE_HTTPD_BASIC_AUTH
#undef CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
#undef CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
#undef CONFIG_FEATURE_HTTPD_DECODE_URL_STR
#undef CONFIG_FEATURE_HTTPD_SET_REMOTE_PORT_TO_ENV
#undef CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
#undef CONFIG_FEATURE_HTTPD_CGI
#undef CONFIG_FEATURE_HTTPD_SETUID
#undef CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP
/* enable all features now */
2003-01-05 04:01:56 +00:00
#define CONFIG_FEATURE_HTTPD_BASIC_AUTH
#define CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
#define CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
#define CONFIG_FEATURE_HTTPD_DECODE_URL_STR
#define CONFIG_FEATURE_HTTPD_SET_REMOTE_PORT_TO_ENV
#define CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
#define CONFIG_FEATURE_HTTPD_CGI
#define CONFIG_FEATURE_HTTPD_SETUID
#define CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP
/* require from libbb.a for linking */
const char *applet_name = "httpd";
void show_usage(void)
2003-01-05 04:01:56 +00:00
{
fprintf(stderr, "Usage: %s [-p <port>] [-c configFile] [-d/-e <string>] "
"[-r realm] [-u user]\n", applet_name);
exit(1);
2003-01-05 04:01:56 +00:00
}
#endif
#ifdef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
#undef CONFIG_FEATURE_HTTPD_SETUID /* use inetd user.group config settings */
#undef CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP /* so is not daemon */
/* inetd set stderr to accepted socket and we can`t true see debug messages */
#undef DEBUG
#endif
/* CGI environ size */
#ifdef CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
#define ENVSIZE 50 /* set max 35 CGI_variable */
#else
#define ENVSIZE 15 /* minimal requires */
2003-01-05 04:01:56 +00:00
#endif
#define MAX_POST_SIZE (64*1024) /* 64k. Its Small? May be ;) */
#define MAX_MEMORY_BUFF 8192 /* IO buffer */
typedef struct HT_ACCESS {
char *after_colon;
struct HT_ACCESS *next;
char before_colon[1]; /* really bigger, must last */
} Htaccess;
typedef struct
{
#ifdef CONFIG_FEATURE_HTTPD_CGI
char *envp[ENVSIZE+1];
int envCount;
#endif
char buf[MAX_MEMORY_BUFF];
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
const char *realm;
#endif
const char *configFile;
char rmt_ip[16]; /* for set env REMOTE_ADDR */
unsigned port; /* server initial port and for
set env REMOTE_PORT */
const char *found_mime_type;
off_t ContentLength; /* -1 - unknown */
time_t last_mod;
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
int accepted_socket;
#define a_c_r config->accepted_socket
#define a_c_w config->accepted_socket
int debugHttpd; /* if seted, don`t stay daemon */
#else
#define a_c_r 0
#define a_c_w 1
#endif
Htaccess *Httpd_conf_parsed;
} HttpdConfig;
static HttpdConfig *config;
static const char request_GET[] = "GET"; /* size algorithic optimize */
2003-01-05 04:01:56 +00:00
static const char* const suffixTable [] = {
/* Warning: shorted equalent suffix in one line must be first */
2003-01-05 04:01:56 +00:00
".htm.html", "text/html",
".jpg.jpeg", "image/jpeg",
".gif", "image/gif",
".png", "image/png",
".txt.h.c.cc.cpp", "text/plain",
".css", "text/css",
".wav", "audio/wav",
".avi", "video/x-msvideo",
".qt.mov", "video/quicktime",
".mpe.mpeg", "video/mpeg",
".mid.midi", "audio/midi",
".mp3", "audio/mpeg",
#if 0 /* unpopular */
".au", "audio/basic",
".pac", "application/x-ns-proxy-autoconfig",
".vrml.wrl", "model/vrml",
#endif
0, "application/octet-stream" /* default */
2003-01-05 04:01:56 +00:00
};
typedef enum
{
HTTP_OK = 200,
HTTP_UNAUTHORIZED = 401, /* authentication needed, respond with auth hdr */
HTTP_NOT_FOUND = 404,
HTTP_NOT_IMPLEMENTED = 501, /* used for unrecognized requests */
HTTP_BAD_REQUEST = 400, /* malformed syntax */
HTTP_FORBIDDEN = 403,
2003-01-05 04:01:56 +00:00
HTTP_INTERNAL_SERVER_ERROR = 500,
#if 0 /* future use */
HTTP_CONTINUE = 100,
HTTP_SWITCHING_PROTOCOLS = 101,
HTTP_CREATED = 201,
HTTP_ACCEPTED = 202,
HTTP_NON_AUTHORITATIVE_INFO = 203,
HTTP_NO_CONTENT = 204,
HTTP_MULTIPLE_CHOICES = 300,
HTTP_MOVED_PERMANENTLY = 301,
HTTP_MOVED_TEMPORARILY = 302,
HTTP_NOT_MODIFIED = 304,
HTTP_PAYMENT_REQUIRED = 402,
HTTP_BAD_GATEWAY = 502,
HTTP_SERVICE_UNAVAILABLE = 503, /* overload, maintenance */
HTTP_RESPONSE_SETSIZE=0xffffffff
#endif
} HttpResponseNum;
typedef struct
{
HttpResponseNum type;
const char *name;
const char *info;
} HttpEnumString;
static const HttpEnumString httpResponseNames[] = {
{ HTTP_OK, "OK" },
{ HTTP_NOT_IMPLEMENTED, "Not Implemented",
2003-01-05 04:01:56 +00:00
"The requested method is not recognized by this server." },
{ HTTP_UNAUTHORIZED, "Unauthorized", "" },
{ HTTP_NOT_FOUND, "Not Found",
"The requested URL was not found on this server." },
{ HTTP_BAD_REQUEST, "Bad Request" ,
"Unsupported method." },
{ HTTP_FORBIDDEN, "Forbidden", "" },
2003-01-05 04:01:56 +00:00
{ HTTP_INTERNAL_SERVER_ERROR, "Internal Server Error"
"Internal Server Error" },
#if 0
{ HTTP_CREATED, "Created" },
{ HTTP_ACCEPTED, "Accepted" },
{ HTTP_NO_CONTENT, "No Content" },
{ HTTP_MULTIPLE_CHOICES, "Multiple Choices" },
{ HTTP_MOVED_PERMANENTLY, "Moved Permanently" },
{ HTTP_MOVED_TEMPORARILY, "Moved Temporarily" },
{ HTTP_NOT_MODIFIED, "Not Modified" },
{ HTTP_BAD_GATEWAY, "Bad Gateway", "" },
{ HTTP_SERVICE_UNAVAILABLE, "Service Unavailable", "" },
#endif
};
static const char RFC1123FMT[] = "%a, %d %b %Y %H:%M:%S GMT";
static const char Content_length[] = "Content-length:";
/*
* sotring to:
* .ext:mime/type
* /path:user:pass
* /path/subdir:user:pass
* D:from
* A:from
* D:*
*/
static int conf_sort(const void *p1, const void *p2)
{
const Htaccess *cl1 = *(const Htaccess **)p1;
const Htaccess *cl2 = *(const Htaccess **)p2;
char c1 = cl1->before_colon[0];
char c2 = cl2->before_colon[0];
int test;
#ifdef CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
/* .ext line up before other lines for simlify algorithm */
test = c2 == '.';
if(c1 == '.')
return -(!test);
if(test)
return test;
#endif
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
test = c1 == '/';
/* /path line up before A/D lines for simlify algorithm */
if(test) {
if(c2 != '/')
return -test;
/* a shortes path with user:pass must be first */
return strlen(cl1->before_colon) - strlen(cl2->before_colon);
} else if(c2 == '/')
return !test;
#endif
/* D:from must move top */
test = c2 == 'D' && cl2->after_colon[0] != 0;
if(c1 == 'D' && cl1->after_colon[0] != 0) {
return -(!test);
}
if(test)
return test;
/* next lines - A:from */
test = c2 == 'A' && cl2->after_colon[0] != 0;
if(c1 == 'A' && cl1->after_colon[0] != 0) {
return -(!test);
}
if(test)
return test;
/* end lines - D:* */
test = c2 == 'D' && cl2->after_colon[0] == 0;
if(c1 == 'D' && cl1->after_colon[0] == 0) {
return -(!test);
}
#ifdef DEBUG
if(!test)
error_msg_and_die("sort: can`t found compares!");
#endif
return test;
}
/* flag */
#define FIRST_PARSE 0
#define SUBDIR_PARSE 1
#define SIGNALED_PARSE 2
static void parse_conf(const char *path, int flag)
{
#define bc cur->before_colon[0]
#define ac cur->after_colon[0]
FILE *f;
Htaccess *prev;
Htaccess *cur;
const char *cf = config->configFile;
char buf[80];
char *p0 = NULL;
int deny_all = 0; /* default A:* */
int n = 0; /* count config lines */
if(flag == SUBDIR_PARSE || cf == NULL) {
cf = p0 = alloca(strlen(path) + sizeof(httpd_conf) + 2);
if(p0 == NULL) {
if(flag == FIRST_PARSE)
error_msg_and_die(memory_exhausted);
return;
}
sprintf(p0, "%s/%s", path, httpd_conf);
}
while((f = fopen(cf, "r")) == NULL) {
if(flag != FIRST_PARSE)
return; /* subdir config not found */
if(p0 == NULL) /* if -c option gived */
perror_msg_and_die("%s", cf);
p0 = NULL;
cf = httpd_conf; /* set -c ./httpd_conf */
}
prev = config->Httpd_conf_parsed;
if(flag != SUBDIR_PARSE) {
/* free previous setuped */
while( prev ) {
cur = prev;
prev = cur->next;
free(cur);
}
config->Httpd_conf_parsed = prev; /* eq NULL */
} else {
/* parse previous IP logic for merge */
for(cur = prev; cur; cur = cur->next) {
if(bc == 'D' && ac == 0)
deny_all++;
n++;
/* find last for set prev->next of merging */
if(cur != prev)
prev = prev->next;
}
}
/* This could stand some work */
while ( (p0 = fgets(buf, 80, f)) != NULL) {
char *p;
char *colon;
for(p = colon = p0; *p; p++) {
if(*p == '#') {
*p = 0;
break;
}
if(isspace(*p)) {
if(p != p0) {
*p = 0;
break;
}
p0++;
}
else if(*p == ':' && colon <= p0)
colon = p;
}
/* test for empty or strange line */
if (colon <= p0 || colon[1] == 0)
continue;
if(colon[1] == '*')
colon[1] = 0; /* Allow all */
if(*p0 == 'a')
*p0 = 'A';
if(*p0 == 'd')
*p0 = 'D';
if(*p0 != 'A' && *p0 != 'D'
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
&& *p0 != '/'
#endif
#ifdef CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
&& *p0 != '.'
#endif
)
continue;
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
if(*p0 == '/' && colon[1] == 0) {
/* skip /path:* */
continue;
}
#endif
if(*p0 == 'A' || *p0 == 'D') {
if(colon[1] == 0) {
if(*p0 == 'A' || deny_all != 0)
continue; /* skip default A:* or double D:* */
}
if(deny_all != 0 && *p0 == 'A')
continue; // if previous setted rule D:* skip all subdir A:
}
/* storing current config line */
cur = calloc(1, sizeof(Htaccess) + (p-p0));
if(cur) {
if(*(colon-1) == '/' && (colon-1) != p0)
colon[-1] = 0; // remove last / from /path/
cur->after_colon = strcpy(cur->before_colon, p0);
cur->after_colon += (colon-p0);
*cur->after_colon++ = 0;
if(prev == NULL) {
/* first line */
config->Httpd_conf_parsed = prev = cur;
} else {
prev->next = cur;
prev = cur;
}
n++;
}
}
fclose(f);
if(n > 1) {
/* sorting conf lines */
Htaccess ** pcur; /* array for qsort */
prev = config->Httpd_conf_parsed;
pcur = alloca((n + 1) * sizeof(Htaccess *));
if(pcur == NULL) {
if(flag == FIRST_PARSE)
error_msg_and_die(memory_exhausted);
return;
}
n = 0;
for(cur = prev; cur; cur = cur->next)
pcur[n++] = cur;
pcur[n] = NULL;
qsort(pcur, n, sizeof(Htaccess *), conf_sort);
/* storing sorted config */
config->Httpd_conf_parsed = *pcur;
for(cur = *pcur; cur; cur = cur->next) {
#ifdef DEBUG
error_msg("%s: %s:%s", cf, cur->before_colon, cur->after_colon);
#endif
cur->next = *++pcur;
}
}
}
#ifdef CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: encodeString()
*
* $Description: Given a string, html encode special characters.
* This is used for the -e command line option to provide an easy way
* for scripts to encode result data without confusing browsers. The
* returned string pointer is memory allocated by malloc().
*
* $Parameters:
* (const char *) string . . The first string to encode.
*
* $Return: (char *) . . . .. . . A pointer to the encoded string.
*
* $Errors: Returns a null string ("") if memory is not available.
*
****************************************************************************/
static char *encodeString(const char *string)
{
/* take the simple route and encode everything */
/* could possibly scan once to get length. */
int len = strlen(string);
char *out = malloc(len*5 +1);
2003-01-05 04:01:56 +00:00
char *p=out;
char ch;
2003-01-05 04:01:56 +00:00
if (!out) return "";
while ((ch = *string++)) {
2003-01-05 04:01:56 +00:00
// very simple check for what to encode
if (isalnum(ch)) *p++ = ch;
else p += sprintf(p, "&#%d", (unsigned char) ch);
2003-01-05 04:01:56 +00:00
}
*p=0;
return out;
}
#endif /* CONFIG_FEATURE_HTTPD_ENCODE_URL_STR */
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_DECODE_URL_STR
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: decodeString()
*
* $Description: Given a URL encoded string, convert it to plain ascii.
* Since decoding always makes strings smaller, the decode is done in-place.
* Thus, callers should strdup() the argument if they do not want the
* argument modified. The return is the original pointer, allowing this
* function to be easily used as arguments to other functions.
*
* $Parameters:
* (char *) string . . . The first string to decode.
*
* $Return: (char *) . . . . A pointer to the decoded string (same as input).
*
* $Errors: None
*
****************************************************************************/
static char *decodeString(char *string)
{
/* note that decoded string is always shorter than original */
char *orig = string;
char *ptr = string;
while (*ptr)
{
if (*ptr == '+') { *string++ = ' '; ptr++; }
2003-01-05 04:01:56 +00:00
else if (*ptr != '%') *string++ = *ptr++;
else {
2003-01-05 04:01:56 +00:00
unsigned int value;
sscanf(ptr+1, "%2X", &value);
2003-01-05 04:01:56 +00:00
*string++ = value;
ptr += 3;
}
}
*string = '\0';
return orig;
}
#endif /* CONFIG_FEATURE_HTTPD_DECODE_URL_STR */
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_CGI
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: addEnv()
*
* $Description: Add an enviornment variable setting to the global list.
* A NAME=VALUE string is allocated, filled, and added to the list of
* environment settings passed to the cgi execution script.
*
* $Parameters:
* (char *) name_before_underline - The first part environment variable name.
* (char *) name_after_underline - The second part environment variable name.
* (char *) value . . The value to which the env variable is set.
2003-01-05 04:01:56 +00:00
*
* $Return: (void)
2003-01-05 04:01:56 +00:00
*
* $Errors: Silently returns if the env runs out of space to hold the new item
*
****************************************************************************/
static void addEnv(const char *name_before_underline,
const char *name_after_underline, const char *value)
2003-01-05 04:01:56 +00:00
{
char *s;
if (config->envCount >= ENVSIZE)
return;
if (!value)
value = "";
s = malloc(strlen(name_before_underline) + strlen(name_after_underline) +
strlen(value) + 3);
if (s) {
const char *underline = *name_after_underline ? "_" : "";
sprintf(s,"%s%s%s=%s", name_before_underline, underline,
name_after_underline, value);
config->envp[config->envCount++] = s;
config->envp[config->envCount] = 0;
2003-01-05 04:01:56 +00:00
}
}
/* set environs SERVER_PORT and REMOTE_PORT */
static void addEnvPort(const char *port_name)
{
char buf[16];
sprintf(buf, "%u", config->port);
addEnv(port_name, "PORT", buf);
}
#endif /* CONFIG_FEATURE_HTTPD_CGI */
#ifdef CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: addEnvCgi
*
* $Description: Create environment variables given a URL encoded arg list.
* For each variable setting the URL encoded arg list, create a corresponding
* environment variable. URL encoded arguments have the form
* name1=value1&name2=value2&name3=&ignores
* from this example, name3 set empty value, tail without '=' skiping
2003-01-05 04:01:56 +00:00
*
* $Parameters:
* (char *) pargs . . . . A pointer to the URL encoded arguments.
*
* $Return: None
*
* $Errors: None
*
****************************************************************************/
static void addEnvCgi(const char *pargs)
{
char *args;
char *memargs;
2003-01-05 04:01:56 +00:00
if (pargs==0) return;
2003-01-05 04:01:56 +00:00
/* args are a list of name=value&name2=value2 sequences */
memargs = args = strdup(pargs);
while (args && *args) {
const char *name = args;
char *value = strchr(args, '=');
if (!value) /* &XXX without '=' */
break;
*value++ = 0;
args = strchr(value, '&');
if (args)
*args++ = 0;
addEnv("CGI", name, decodeString(value));
2003-01-05 04:01:56 +00:00
}
free(memargs);
2003-01-05 04:01:56 +00:00
}
#endif /* CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV */
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: decodeBase64()
*
> $Description: Decode a base 64 data stream as per rfc1521.
* Note that the rfc states that none base64 chars are to be ignored.
* Since the decode always results in a shorter size than the input, it is
* OK to pass the input arg as an output arg.
*
* $Parameter:
* (char *) Data . . . . A pointer to a base64 encoded string.
* Where to place the decoded data.
2003-01-05 04:01:56 +00:00
*
* $Return: void
2003-01-05 04:01:56 +00:00
*
* $Errors: None
*
****************************************************************************/
static void decodeBase64(char *Data)
2003-01-05 04:01:56 +00:00
{
int i = 0;
static const char base64ToBin[] =
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
const unsigned char *in = Data;
// The decoded size will be at most 3/4 the size of the encoded
2003-01-05 04:01:56 +00:00
unsigned long ch = 0;
while (*in) {
2003-01-05 04:01:56 +00:00
unsigned char conv = 0;
while (*in) {
const char *p64;
p64 = strchr(base64ToBin, *in++);
if(p64 == NULL)
continue;
conv = (p64 - base64ToBin);
2003-01-05 04:01:56 +00:00
break;
}
ch = (ch << 6) | conv;
i++;
if (i== 4) {
*Data++ = (char) (ch >> 16);
*Data++ = (char) (ch >> 8);
*Data++ = (char) ch;
2003-01-05 04:01:56 +00:00
i = 0;
}
}
*Data = 0;
2003-01-05 04:01:56 +00:00
}
#endif
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: openServer()
*
* $Description: create a listen server socket on the designated port.
*
* $Return: (int) . . . A connection socket. -1 for errors.
*
* $Errors: None
*
****************************************************************************/
static int openServer(void)
2003-01-05 04:01:56 +00:00
{
struct sockaddr_in lsocket;
int fd;
2003-01-05 04:01:56 +00:00
/* create the socket right now */
/* inet_addr() returns a value that is already in network order */
memset(&lsocket, 0, sizeof(lsocket));
lsocket.sin_family = AF_INET;
lsocket.sin_addr.s_addr = INADDR_ANY;
lsocket.sin_port = htons(config->port) ;
2003-01-05 04:01:56 +00:00
fd = socket(AF_INET, SOCK_STREAM, 0);
if (fd >= 0) {
2003-01-05 04:01:56 +00:00
/* tell the OS it's OK to reuse a previous address even though */
/* it may still be in a close down state. Allows bind to succeed. */
int on = 1;
2003-01-05 04:01:56 +00:00
#ifdef SO_REUSEPORT
setsockopt(fd, SOL_SOCKET, SO_REUSEPORT, (void *)&on, sizeof(on)) ;
2003-01-05 04:01:56 +00:00
#else
setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (void *)&on, sizeof(on)) ;
2003-01-05 04:01:56 +00:00
#endif
if (bind(fd, (struct sockaddr *)&lsocket, sizeof(lsocket)) == 0) {
2003-01-05 04:01:56 +00:00
listen(fd, 9);
signal(SIGCHLD, SIG_IGN); /* prevent zombie (defunct) processes */
} else {
perror_msg_and_die("bind");
2003-01-05 04:01:56 +00:00
}
} else {
perror_msg_and_die("create socket");
}
2003-01-05 04:01:56 +00:00
return fd;
}
#endif /* CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY */
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: sendHeaders()
*
* $Description: Create and send HTTP response headers.
* The arguments are combined and sent as one write operation. Note that
* IE will puke big-time if the headers are not sent in one packet and the
* second packet is delayed for any reason.
2003-01-05 04:01:56 +00:00
*
* $Parameter:
2003-01-05 04:01:56 +00:00
* (HttpResponseNum) responseNum . . . The result code to send.
*
* $Return: (int) . . . . writing errors
2003-01-05 04:01:56 +00:00
*
****************************************************************************/
static int sendHeaders(HttpResponseNum responseNum)
2003-01-05 04:01:56 +00:00
{
char *buf = config->buf;
2003-01-05 04:01:56 +00:00
const char *responseString = "";
const char *infoString = 0;
unsigned int i;
time_t timer = time(0);
char timeStr[80];
int len;
for (i = 0;
i < (sizeof(httpResponseNames)/sizeof(httpResponseNames[0])); i++) {
if (httpResponseNames[i].type == responseNum) {
responseString = httpResponseNames[i].name;
infoString = httpResponseNames[i].info;
break;
}
2003-01-05 04:01:56 +00:00
}
if (responseNum != HTTP_OK) {
config->found_mime_type = "text/html"; // error message is HTML
2003-01-05 04:01:56 +00:00
}
/* emit the current date */
strftime(timeStr, sizeof(timeStr), RFC1123FMT, gmtime(&timer));
len = sprintf(buf,
"HTTP/1.0 %d %s\nContent-type: %s\r\n"
"Date: %s\r\nConnection: close\r\n",
responseNum, responseString, config->found_mime_type, timeStr);
2003-01-05 04:01:56 +00:00
if (responseNum == HTTP_UNAUTHORIZED) {
len += sprintf(buf+len, "WWW-Authenticate: Basic realm=\"%s\"\r\n",
config->realm);
2003-01-05 04:01:56 +00:00
}
if (config->ContentLength != -1) { /* file */
strftime(timeStr, sizeof(timeStr), RFC1123FMT, gmtime(&config->last_mod));
len += sprintf(buf+len, "Last-Modified: %s\r\n%s %ld\r\n",
timeStr, Content_length, config->ContentLength);
2003-01-05 04:01:56 +00:00
}
strcat(buf, "\r\n");
len += 2;
if (infoString) {
len += sprintf(buf+len,
2003-01-05 04:01:56 +00:00
"<HEAD><TITLE>%d %s</TITLE></HEAD>\n"
"<BODY><H1>%d %s</H1>\n%s\n</BODY>\n",
responseNum, responseString,
responseNum, responseString, infoString);
2003-01-05 04:01:56 +00:00
}
#ifdef DEBUG
if (config->debugHttpd) fprintf(stderr, "Headers: '%s'", buf);
2003-01-05 04:01:56 +00:00
#endif
return full_write(a_c_w, buf, len);
2003-01-05 04:01:56 +00:00
}
/****************************************************************************
*
> $Function: getLine()
*
* $Description: Read from the socket until an end of line char found.
*
* Characters are read one at a time until an eol sequence is found.
*
* $Parameters:
* (char *) buf . . Where to place the read result.
*
* $Return: (int) . . . . number of characters read. -1 if error.
*
****************************************************************************/
static int getLine(char *buf)
2003-01-05 04:01:56 +00:00
{
int count = 0;
while (read(a_c_r, buf + count, 1) == 1) {
2003-01-05 04:01:56 +00:00
if (buf[count] == '\r') continue;
if (buf[count] == '\n') {
2003-01-05 04:01:56 +00:00
buf[count] = 0;
return count;
}
if(count < (MAX_MEMORY_BUFF-1)) /* check owerflow */
count++;
2003-01-05 04:01:56 +00:00
}
if (count) return count;
else return -1;
}
#ifdef CONFIG_FEATURE_HTTPD_CGI
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: sendCgi()
*
* $Description: Execute a CGI script and send it's stdout back
*
* Environment variables are set up and the script is invoked with pipes
* for stdin/stdout. If a post is being done the script is fed the POST
* data in addition to setting the QUERY_STRING variable (for GETs or POSTs).
*
* $Parameters:
* (const char *) url . . . The requested URL (with leading /).
* (const char *urlArgs). . Any URL arguments.
* (const char *body) . . . POST body contents.
* (int bodyLen) . . . . . Length of the post body.
* (const char *cookie) . . For set HTTP_COOKIE.
2003-01-05 04:01:56 +00:00
*
* $Return: (char *) . . . . A pointer to the decoded string (same as input).
*
* $Errors: None
*
****************************************************************************/
static int sendCgi(const char *url,
const char *request, const char *urlArgs,
const char *body, int bodyLen, const char *cookie)
2003-01-05 04:01:56 +00:00
{
int fromCgi[2]; /* pipe for reading data from CGI */
int toCgi[2]; /* pipe for sending data to CGI */
static char * argp[] = { 0, 0 };
int pid = 0;
int inFd;
2003-01-05 04:01:56 +00:00
int outFd;
int firstLine = 1;
2003-01-05 04:01:56 +00:00
do {
if (pipe(fromCgi) != 0) {
2003-01-05 04:01:56 +00:00
break;
}
if (pipe(toCgi) != 0) {
2003-01-05 04:01:56 +00:00
break;
}
pid = fork();
if (pid < 0) {
2003-01-05 04:01:56 +00:00
pid = 0;
break;
2003-01-05 04:01:56 +00:00
}
if (!pid) {
2003-01-05 04:01:56 +00:00
/* child process */
char *script;
char *purl = strdup( url );
char realpath_buff[MAXPATHLEN];
if(purl == NULL)
_exit(242);
inFd = toCgi[0];
outFd = fromCgi[1];
2003-01-05 04:01:56 +00:00
dup2(inFd, 0); // replace stdin with the pipe
dup2(outFd, 1); // replace stdout with the pipe
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
if (!config->debugHttpd)
#endif
dup2(outFd, 2); // replace stderr with the pipe
2003-01-05 04:01:56 +00:00
close(toCgi[0]);
close(toCgi[1]);
close(fromCgi[0]);
close(fromCgi[1]);
/*
* Find PATH_INFO.
2003-01-05 04:01:56 +00:00
*/
script = purl;
while((script = strchr( script + 1, '/' )) != NULL) {
/* have script.cgi/PATH_INFO or dirs/script.cgi[/PATH_INFO] */
struct stat sb;
*script = '\0';
if(is_directory(purl + 1, 1, &sb) == 0) {
/* not directory, found script.cgi/PATH_INFO */
*script = '/';
break;
2003-01-05 04:01:56 +00:00
}
*script = '/'; /* is directory, find next '/' */
}
addEnv("PATH", "INFO", script); /* set /PATH_INFO or NULL */
addEnv("PATH", "", getenv("PATH"));
addEnv("REQUEST", "METHOD", request);
if(urlArgs) {
char *uri = alloca(strlen(purl) + 2 + strlen(urlArgs));
if(uri)
sprintf(uri, "%s?%s", purl, urlArgs);
addEnv("REQUEST", "URI", uri);
} else {
addEnv("REQUEST", "URI", purl);
}
if(script != NULL)
*script = '\0'; /* reduce /PATH_INFO */
/* set SCRIPT_NAME as full path: /cgi-bin/dirs/script.cgi */
addEnv("SCRIPT_NAME", "", purl);
addEnv("QUERY_STRING", "", urlArgs);
addEnv("SERVER", "SOFTWARE", httpdVersion);
addEnv("SERVER", "PROTOCOL", "HTTP/1.0");
addEnv("GATEWAY_INTERFACE", "", "CGI/1.1");
#ifdef CONFIG_FEATURE_HTTPD_SET_REMOTE_PORT_TO_ENV
addEnv("REMOTE", "ADDR", config->rmt_ip);
addEnvPort("REMOTE");
#else
addEnv("REMOTE_ADDR", "", config->rmt_ip);
#endif
if(bodyLen) {
char sbl[32];
sprintf(sbl, "%d", bodyLen);
addEnv("CONTENT_LENGTH", "", sbl);
}
if(cookie)
addEnv("HTTP_COOKIE", "", cookie);
#ifdef CONFIG_FEATURE_HTTPD_SET_CGI_VARS_TO_ENV
if (request != request_GET) {
addEnvCgi(body);
} else {
addEnvCgi(urlArgs);
2003-01-05 04:01:56 +00:00
}
#endif
/* set execve argp[0] without path */
argp[0] = strrchr( purl, '/' ) + 1;
/* but script argp[0] must have absolute path and chdiring to this */
if(realpath(purl + 1, realpath_buff) != NULL) {
script = strrchr(realpath_buff, '/');
if(script) {
*script = '\0';
if(chdir(realpath_buff) == 0) {
*script = '/';
2003-01-05 04:01:56 +00:00
// now run the program. If it fails, use _exit() so no destructors
// get called and make a mess.
execve(realpath_buff, argp, config->envp);
}
}
}
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
config->accepted_socket = 1; /* send to stdout */
2003-01-05 04:01:56 +00:00
#endif
sendHeaders(HTTP_NOT_FOUND);
2003-01-05 04:01:56 +00:00
_exit(242);
} /* end child */
} while (0);
if (pid) {
2003-01-05 04:01:56 +00:00
/* parent process */
int status;
inFd = fromCgi[0];
outFd = toCgi[1];
2003-01-05 04:01:56 +00:00
close(fromCgi[1]);
close(toCgi[0]);
if (body) full_write(outFd, body, bodyLen);
2003-01-05 04:01:56 +00:00
close(outFd);
while (1) {
2003-01-05 04:01:56 +00:00
struct timeval timeout;
fd_set readSet;
char buf[160];
int nfound;
int count;
2003-01-05 04:01:56 +00:00
FD_ZERO(&readSet);
FD_SET(inFd, &readSet);
2003-01-05 04:01:56 +00:00
/* Now wait on the set of sockets! */
timeout.tv_sec = 0;
timeout.tv_usec = 10000;
nfound = select(inFd + 1, &readSet, 0, 0, &timeout);
2003-01-05 04:01:56 +00:00
if (nfound <= 0) {
if (waitpid(pid, &status, WNOHANG) > 0) {
close(inFd);
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd) {
2003-01-05 04:01:56 +00:00
if (WIFEXITED(status))
error_msg("piped has exited with status=%d", WEXITSTATUS(status));
2003-01-05 04:01:56 +00:00
if (WIFSIGNALED(status))
error_msg("piped has exited with signal=%d", WTERMSIG(status));
2003-01-05 04:01:56 +00:00
}
#endif
2003-01-05 04:01:56 +00:00
pid = -1;
break;
}
} else {
int s = a_c_w;
2003-01-05 04:01:56 +00:00
// There is something to read
count = full_read(inFd, buf, sizeof(buf)-1);
2003-01-05 04:01:56 +00:00
// If a read returns 0 at this point then some type of error has
// occurred. Bail now.
if (count == 0) break;
if (count > 0) {
if (firstLine) {
2003-01-05 04:01:56 +00:00
/* check to see if the user script added headers */
if (strncmp(buf, "HTTP/1.0 200 OK\n", 4) != 0) {
full_write(s, "HTTP/1.0 200 OK\n", 16);
2003-01-05 04:01:56 +00:00
}
if (strstr(buf, "ontent-") == 0) {
full_write(s, "Content-type: text/plain\n\n", 26);
2003-01-05 04:01:56 +00:00
}
firstLine=0;
}
full_write(s, buf, count);
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd)
fprintf(stderr, "cgi read %d bytes\n", count);
2003-01-05 04:01:56 +00:00
#endif
}
}
}
}
return 0;
}
#endif /* CONFIG_FEATURE_HTTPD_CGI */
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: sendFile()
*
* $Description: Send a file response to an HTTP request
*
* $Parameter:
2003-01-05 04:01:56 +00:00
* (const char *) url . . The URL requested.
* (char *) buf . . . . . The stack buffer.
2003-01-05 04:01:56 +00:00
*
* $Return: (int) . . . . . . Always 0.
*
****************************************************************************/
static int sendFile(const char *url, char *buf)
2003-01-05 04:01:56 +00:00
{
char * suffix;
2003-01-05 04:01:56 +00:00
int f;
const char * const * table;
const char * try_suffix;
2003-01-05 04:01:56 +00:00
suffix = strrchr(url, '.');
for (table = suffixTable; *table; table += 2)
if(suffix != NULL && (try_suffix = strstr(*table, suffix)) != 0) {
try_suffix += strlen(suffix);
if(*try_suffix == 0 || *try_suffix == '.')
break;
}
/* also, if not found, set default as "application/octet-stream"; */
config->found_mime_type = *(table+1);
#ifdef CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
if (suffix) {
Htaccess * cur;
for (cur = config->Httpd_conf_parsed; cur; cur = cur->next) {
if(strcmp(cur->before_colon, suffix) == 0) {
config->found_mime_type = cur->after_colon;
break;
}
}
2003-01-05 04:01:56 +00:00
}
#endif /* CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES */
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd)
fprintf(stderr, "Sending file '%s' Content-type: %s\n",
url, config->found_mime_type);
2003-01-05 04:01:56 +00:00
#endif
f = open(url, O_RDONLY);
if (f >= 0) {
int count;
sendHeaders(HTTP_OK);
while ((count = full_read(f, buf, MAX_MEMORY_BUFF)) > 0) {
full_write(a_c_w, buf, count);
}
close(f);
} else {
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd)
perror_msg("Unable to open '%s'", url);
2003-01-05 04:01:56 +00:00
#endif
sendHeaders(HTTP_NOT_FOUND);
2003-01-05 04:01:56 +00:00
}
2003-01-05 04:01:56 +00:00
return 0;
}
/****************************************************************************
*
> $Function: checkPerm()
*
* $Description: Check the permission file for access.
*
* If config file isn't present, everything is allowed.
* Entries are of the form you can see example from header source
2003-01-05 04:01:56 +00:00
*
* $Parameters:
* (const char *) path . . . . The file path or NULL for ip addresses.
2003-01-05 04:01:56 +00:00
* (const char *) request . . . User information to validate.
*
* $Return: (int) . . . . . . . . . 1 if request OK, 0 otherwise.
*
****************************************************************************/
2003-01-05 04:01:56 +00:00
static int checkPerm(const char *path, const char *request)
{
Htaccess * cur;
const char *p;
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
int ipaddr = path == NULL;
const char *prev = NULL;
#else
# define ipaddr 1
#endif
2003-01-05 04:01:56 +00:00
/* This could stand some work */
for (cur = config->Httpd_conf_parsed; cur; cur = cur->next) {
const char *p0 = cur->before_colon;
if(*p0 == 'A' || *p0 == 'D') {
if(!ipaddr)
continue;
} else {
if(ipaddr)
continue;
#ifdef CONFIG_FEATURE_HTTPD_CONFIG_WITH_MIME_TYPES
if(*p0 == '.')
continue;
#endif
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
if(prev != NULL && strcmp(prev, p0) != 0)
continue; /* find next identical */
#endif
}
p = cur->after_colon;
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd)
fprintf(stderr,"checkPerm: '%s' ? '%s'\n",
(ipaddr ? p : p0), request);
2003-01-05 04:01:56 +00:00
#endif
if(ipaddr) {
if(strncmp(p, request, strlen(p)) != 0)
continue;
return *p0 == 'A'; /* Allow/Deny */
2003-01-05 04:01:56 +00:00
}
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
else {
int l = strlen(p0);
if(strncmp(p0, path, l) == 0 &&
(l == 1 || path[l] == '/' || path[l] == 0)) {
/* path match found. Check request */
if (strcmp(p, request) == 0)
return 1; /* Ok */
/* unauthorized, but check next /path:user:password */
prev = p0;
}
}
#endif
} /* for */
#ifndef CONFIG_FEATURE_HTTPD_BASIC_AUTH
/* if uncofigured, return 1 - access from all */
return 1;
#else
return prev == NULL;
#endif
}
2003-01-05 04:01:56 +00:00
/****************************************************************************
*
> $Function: handleIncoming()
*
* $Description: Handle an incoming http request.
*
****************************************************************************/
static void handleIncoming(void)
2003-01-05 04:01:56 +00:00
{
char *buf = config->buf;
char *url;
char *purl;
int blank = -1;
2003-01-05 04:01:56 +00:00
char *urlArgs;
#ifdef CONFIG_FEATURE_HTTPD_CGI
const char *prequest = request_GET;
char *body = 0;
long length=0;
char *cookie = 0;
#endif
char *test;
struct stat sb;
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
int credentials = -1; /* if not requred this is Ok */
2003-01-05 04:01:56 +00:00
#endif
do {
int count;
if (getLine(buf) <= 0)
break; /* closed */
purl = strpbrk(buf, " \t");
if(purl == NULL) {
BAD_REQUEST:
sendHeaders(HTTP_BAD_REQUEST);
2003-01-05 04:01:56 +00:00
break;
}
*purl = 0;
#ifdef CONFIG_FEATURE_HTTPD_CGI
if(strcasecmp(buf, prequest) != 0) {
prequest = "POST";
if(strcasecmp(buf, prequest) != 0) {
sendHeaders(HTTP_NOT_IMPLEMENTED);
break;
}
}
#else
if(strcasecmp(buf, request_GET) != 0) {
sendHeaders(HTTP_NOT_IMPLEMENTED);
break;
2003-01-05 04:01:56 +00:00
}
#endif
*purl = ' ';
count = sscanf(purl, " %[^ ] HTTP/%d.%*d", buf, &blank);
2003-01-05 04:01:56 +00:00
if (count < 1 || buf[0] != '/') {
/* Garbled request/URL */
goto BAD_REQUEST;
}
url = alloca(strlen(buf) + 12); /* + sizeof("/index.html\0") */
if(url == NULL) {
sendHeaders(HTTP_INTERNAL_SERVER_ERROR);
break;
}
strcpy(url, buf);
2003-01-05 04:01:56 +00:00
/* extract url args if present */
urlArgs = strchr(url, '?');
if (urlArgs) {
*urlArgs++ = 0; /* next code can set '/' to this pointer,
but CGI script can`t be a directory */
}
/* algorithm stolen from libbb simplify_path(),
but don`t strdup and reducing trailing slash */
purl = test = url;
do {
if (*purl == '/') {
if (*test == '/') { /* skip duplicate (or initial) slash */
continue;
} else if (*test == '.') {
if (test[1] == '/' || test[1] == 0) { /* skip extra '.' */
continue;
} else if ((test[1] == '.') && (test[2] == '/' || test[2] == 0)) {
++test;
if (purl == url) {
/* protect out root */
goto BAD_REQUEST;
}
while (*--purl != '/'); /* omit previous dir */
continue;
}
}
}
*++purl = *test;
} while (*++test);
*++purl = 0; /* so keep last character */
test = purl; /* end ptr */
/* If URL is directory, adding '/' */
if(test[-1] != '/') {
if ( is_directory(url + 1, 1, &sb) ) {
*test++ = '/';
*test = 0;
purl = test; /* end ptr */
}
2003-01-05 04:01:56 +00:00
}
#ifdef DEBUG
if (config->debugHttpd)
fprintf(stderr, "url='%s', args=%s\n", url, urlArgs);
#endif
test = url;
while((test = strchr( test + 1, '/' )) != NULL) {
/* have path1/path2 */
*test = '\0';
if( is_directory(url + 1, 1, &sb) ) {
/* may be having subdir config */
parse_conf(url + 1, SUBDIR_PARSE);
}
*test = '/';
}
// read until blank line for HTTP version specified, else parse immediate
while (blank >= 0 && (count = getLine(buf)) > 0) {
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd) fprintf(stderr, "Header: '%s'\n", buf);
#endif
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_CGI
2003-01-05 04:01:56 +00:00
/* try and do our best to parse more lines */
if ((strncasecmp(buf, Content_length, 15) == 0)) {
if(prequest != request_GET)
length = strtol(buf + 15, 0, 0); // extra read only for POST
} else if ((strncasecmp(buf, "Cookie:", 7) == 0)) {
for(test = buf + 7; isspace(*test); test++)
;
cookie = strdup(test);
}
#endif
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
if (strncasecmp(buf, "Authorization:", 14) == 0) {
2003-01-05 04:01:56 +00:00
/* We only allow Basic credentials.
* It shows up as "Authorization: Basic <userid:password>" where
* the userid:password is base64 encoded.
*/
for(test = buf + 14; isspace(*test); test++)
;
if (strncasecmp(test, "Basic", 5) != 0)
continue;
test += 5; /* decodeBase64() skiping space self */
decodeBase64(test);
credentials = checkPerm(url, test);
2003-01-05 04:01:56 +00:00
}
#endif /* CONFIG_FEATURE_HTTPD_BASIC_AUTH */
} /* while extra header reading */
if (strcmp(strrchr(url, '/') + 1, httpd_conf) == 0 ||
checkPerm(NULL, config->rmt_ip) == 0) {
/* protect listing [/path]/httpd_conf or IP deny */
#ifdef CONFIG_FEATURE_HTTPD_CGI
FORBIDDEN: /* protect listing /cgi-bin */
#endif
sendHeaders(HTTP_FORBIDDEN);
break;
2003-01-05 04:01:56 +00:00
}
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
if (credentials <= 0 && checkPerm(url, ":") == 0) {
sendHeaders(HTTP_UNAUTHORIZED);
break;
2003-01-05 04:01:56 +00:00
}
#endif
2003-01-05 04:01:56 +00:00
test = url + 1; /* skip first '/' */
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_CGI
/* if strange Content-Length */
if (length < 0 || length > MAX_POST_SIZE)
break;
2003-01-05 04:01:56 +00:00
if (length > 0) {
body = malloc(length + 1);
if (body) {
length = full_read(a_c_r, body, length);
if(length < 0) // closed
length = 0;
body[length] = 0; // always null terminate for safety
urlArgs = body;
2003-01-05 04:01:56 +00:00
}
}
if (strncmp(test, "cgi-bin", 7) == 0) {
if(test[7] == 0 || (test[7] == '/' && test[8] == 0))
goto FORBIDDEN; // protect listing cgi-bin
sendCgi(url, prequest, urlArgs, body, length, cookie);
} else {
if (prequest != request_GET)
sendHeaders(HTTP_NOT_IMPLEMENTED);
else {
#endif /* CONFIG_FEATURE_HTTPD_CGI */
if(purl[-1] == '/')
strcpy(purl, "index.html");
if ( stat(test, &sb ) == 0 ) {
config->ContentLength = sb.st_size;
config->last_mod = sb.st_mtime;
}
sendFile(test, buf);
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
/* unset if non inetd looped */
config->ContentLength = -1;
#endif
#ifdef CONFIG_FEATURE_HTTPD_CGI
}
2003-01-05 04:01:56 +00:00
}
#endif
2003-01-05 04:01:56 +00:00
} while (0);
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
/* from inetd don`t looping: freeing, closing automatic from exit always */
# ifdef DEBUG
if (config->debugHttpd) fprintf(stderr, "closing socket\n");
# endif
# ifdef CONFIG_FEATURE_HTTPD_CGI
free(body);
free(cookie);
# endif
shutdown(a_c_w, SHUT_WR);
shutdown(a_c_r, SHUT_RD);
close(config->accepted_socket);
#endif /* CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY */
2003-01-05 04:01:56 +00:00
}
/****************************************************************************
*
> $Function: miniHttpd()
*
* $Description: The main http server function.
*
* Given an open socket fildes, listen for new connections and farm out
* the processing as a forked process.
*
* $Parameters:
* (int) server. . . The server socket fildes.
*
* $Return: (int) . . . . Always 0.
*
****************************************************************************/
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
2003-01-05 04:01:56 +00:00
static int miniHttpd(int server)
{
fd_set readfd, portfd;
int nfound;
2003-01-05 04:01:56 +00:00
FD_ZERO(&portfd);
FD_SET(server, &portfd);
2003-01-05 04:01:56 +00:00
/* copy the ports we are watching to the readfd set */
while (1) {
readfd = portfd;
2003-01-05 04:01:56 +00:00
/* Now wait INDEFINATELY on the set of sockets! */
nfound = select(server + 1, &readfd, 0, 0, 0);
switch (nfound) {
2003-01-05 04:01:56 +00:00
case 0:
/* select timeout error! */
break ;
case -1:
/* select error */
break;
default:
if (FD_ISSET(server, &readfd)) {
int on;
2003-01-05 04:01:56 +00:00
struct sockaddr_in fromAddr;
unsigned int addr;
2003-01-05 04:01:56 +00:00
socklen_t fromAddrLen = sizeof(fromAddr);
int s = accept(server,
(struct sockaddr *)&fromAddr, &fromAddrLen);
if (s < 0) {
2003-01-05 04:01:56 +00:00
continue;
}
config->accepted_socket = s;
2003-01-05 04:01:56 +00:00
addr = ntohl(fromAddr.sin_addr.s_addr);
sprintf(config->rmt_ip, "%u.%u.%u.%u",
(unsigned char)(addr >> 24),
(unsigned char)(addr >> 16),
(unsigned char)(addr >> 8),
addr & 0xff);
config->port = ntohs(fromAddr.sin_port);
2003-01-05 04:01:56 +00:00
#ifdef DEBUG
if (config->debugHttpd) {
error_msg("connection from IP=%s, port %u\n",
config->rmt_ip, config->port);
2003-01-05 04:01:56 +00:00
}
#endif
2003-01-05 04:01:56 +00:00
/* set the KEEPALIVE option to cull dead connections */
on = 1;
setsockopt(s, SOL_SOCKET, SO_KEEPALIVE, (void *)&on, sizeof (on));
2003-01-05 04:01:56 +00:00
if (config->debugHttpd || fork() == 0) {
2003-01-05 04:01:56 +00:00
/* This is the spawned thread */
handleIncoming();
if(!config->debugHttpd)
exit(0);
2003-01-05 04:01:56 +00:00
}
close(s);
}
}
} // while (1)
return 0;
}
#else
/* from inetd */
static int miniHttpd(void)
{
struct sockaddr_in fromAddrLen;
socklen_t sinlen = sizeof (struct sockaddr_in);
unsigned int addr;
getpeername (0, (struct sockaddr *)&fromAddrLen, &sinlen);
addr = ntohl(fromAddrLen.sin_addr.s_addr);
sprintf(config->rmt_ip, "%u.%u.%u.%u",
(unsigned char)(addr >> 24),
(unsigned char)(addr >> 16),
(unsigned char)(addr >> 8),
addr & 0xff);
config->port = ntohs(fromAddrLen.sin_port);
handleIncoming();
return 0;
}
#endif /* CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY */
#ifdef CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP
static void sighup_handler(int sig)
{
/* set and reset */
struct sigaction sa;
sa.sa_handler = sighup_handler;
sigemptyset(&sa.sa_mask);
sa.sa_flags = SA_RESTART;
sigaction(SIGHUP, &sa, NULL);
parse_conf(default_patch_httpd_conf,
sig == SIGHUP ? SIGNALED_PARSE : FIRST_PARSE);
}
#endif
#ifdef HTTPD_STANDALONE
int main(int argc, char *argv[])
#else
2003-01-05 04:01:56 +00:00
int httpd_main(int argc, char *argv[])
#endif
2003-01-05 04:01:56 +00:00
{
const char *home_httpd = home;
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
2003-01-05 04:01:56 +00:00
int server;
#endif
#ifdef CONFIG_FEATURE_HTTPD_SETUID
long uid = -1;
#endif
config = xcalloc(1, sizeof(*config));
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
config->realm = "Web Server Authentication";
#endif
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
config->port = 80;
#endif
config->ContentLength = -1;
2003-01-05 04:01:56 +00:00
/* check if user supplied a port number */
for (;;) {
int c = getopt( argc, argv, "c:h:"
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
"p:v"
#endif
#ifdef CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
"e:"
#endif
#ifdef CONFIG_FEATURE_HTTPD_DECODE_URL_STR
"d:"
#endif
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
"r:"
#endif
#ifdef CONFIG_FEATURE_HTTPD_SETUID
"u:"
2003-01-05 04:01:56 +00:00
#endif
);
if (c == EOF) break;
switch (c) {
case 'c':
config->configFile = optarg;
break;
case 'h':
home_httpd = optarg;
break;
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
2003-01-05 04:01:56 +00:00
case 'v':
config->debugHttpd = 1;
2003-01-05 04:01:56 +00:00
break;
case 'p':
config->port = atoi(optarg);
if(config->port <= 0 || config->port > 0xffff)
error_msg_and_die("invalid %s for -p", optarg);
2003-01-05 04:01:56 +00:00
break;
#endif
#ifdef CONFIG_FEATURE_HTTPD_DECODE_URL_STR
2003-01-05 04:01:56 +00:00
case 'd':
printf("%s", decodeString(optarg));
2003-01-05 04:01:56 +00:00
return 0;
#endif
#ifdef CONFIG_FEATURE_HTTPD_ENCODE_URL_STR
2003-01-05 04:01:56 +00:00
case 'e':
printf("%s", encodeString(optarg));
2003-01-05 04:01:56 +00:00
return 0;
#endif
#ifdef CONFIG_FEATURE_HTTPD_BASIC_AUTH
2003-01-05 04:01:56 +00:00
case 'r':
config->realm = optarg;
2003-01-05 04:01:56 +00:00
break;
#endif
#ifdef CONFIG_FEATURE_HTTPD_SETUID
case 'u':
{
char *e;
uid = strtol(optarg, &e, 0);
if(*e != '\0') {
/* not integer */
uid = my_getpwnam(optarg);
}
}
2003-01-05 04:01:56 +00:00
break;
#endif
2003-01-05 04:01:56 +00:00
default:
error_msg("%s", httpdVersion);
2003-01-05 04:01:56 +00:00
show_usage();
}
}
if(chdir(home_httpd)) {
perror_msg_and_die("can`t chdir to %s", home_httpd);
2003-01-05 04:01:56 +00:00
}
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
server = openServer();
# ifdef CONFIG_FEATURE_HTTPD_SETUID
/* drop privilegies */
if(uid > 0)
setuid(uid);
# endif
# ifdef CONFIG_FEATURE_HTTPD_CGI
addEnvPort("SERVER");
# endif
#endif
2003-01-05 04:01:56 +00:00
#ifdef CONFIG_FEATURE_HTTPD_RELOAD_CONFIG_SIGHUP
sighup_handler(0);
#else
parse_conf(default_patch_httpd_conf, FIRST_PARSE);
#endif
2003-01-05 04:01:56 +00:00
#ifndef CONFIG_FEATURE_HTTPD_USAGE_FROM_INETD_ONLY
if (!config->debugHttpd) {
if (daemon(1, 0) < 0) /* don`t change curent directory */
perror_msg_and_die("daemon");
}
return miniHttpd(server);
#else
return miniHttpd();
2003-01-05 04:01:56 +00:00
#endif
}