wok-next rev 11352
busybox/httpd: add support for system passwords
author | Pascal Bellard <pascal.bellard@slitaz.org> |
---|---|
date | Tue Nov 29 18:11:22 2011 +0100 (2011-11-29) |
parents | 05ee700100bf |
children | c2af9b1a616f |
files | busybox/receipt busybox/stuff/busybox-1.18-httpd.u |
line diff
1.1 --- a/busybox/receipt Tue Nov 29 11:53:47 2011 +0000 1.2 +++ b/busybox/receipt Tue Nov 29 18:11:22 2011 +0100 1.3 @@ -28,6 +28,7 @@ 1.4 printable.u 1.5 cmdline.u 1.6 conspy.u 1.7 +httpd.u 1.8 EOT 1.9 cp $stuff/$PACKAGE-${VERSION%.*}.config .config 1.10 }
2.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 2.2 +++ b/busybox/stuff/busybox-1.18-httpd.u Tue Nov 29 18:11:22 2011 +0100 2.3 @@ -0,0 +1,218 @@ 2.4 +Add support for system passwords 2.5 +--- busybox-1.18.2/networking/httpd.c 2.6 ++++ busybox-1.18.2/networking/httpd.c 2.7 +@@ -53,6 +53,8 @@ 2.8 + * /cgi-bin:foo:bar # Require user foo, pwd bar on urls starting with /cgi-bin/ 2.9 + * /adm:admin:setup # Require user admin, pwd setup on urls starting with /adm/ 2.10 + * /adm:toor:PaSsWd # or user toor, pwd PaSsWd on urls starting with /adm/ 2.11 ++ * /adm:root:* # or user root, pwd from /etc/passwd on urls starting with /adm/ 2.12 ++ * /wiki:*:* # or any user from /etc/passwd with according pwd on urls starting with /wiki/ 2.13 + * .au:audio/basic # additional mime type for audio.au files 2.14 + * *.php:/path/php # running cgi.php scripts through an interpreter 2.15 + * 2.16 +@@ -96,6 +98,14 @@ 2.17 + */ 2.18 + 2.19 + #include "libbb.h" 2.20 ++#if ENABLE_PAM 2.21 ++/* PAM may include <locale.h>. We may need to undefine bbox's stub define: */ 2.22 ++# undef setlocale 2.23 ++/* For some obscure reason, PAM is not in pam/xxx, but in security/xxx. 2.24 ++ * Apparently they like to confuse people. */ 2.25 ++# include <security/pam_appl.h> 2.26 ++# include <security/pam_misc.h> 2.27 ++#endif 2.28 + #if ENABLE_FEATURE_HTTPD_USE_SENDFILE 2.29 + #include <sys/sendfile.h> 2.30 + #endif 2.31 +@@ -1663,6 +1673,56 @@ 2.32 + } 2.33 + 2.34 + #if ENABLE_FEATURE_HTTPD_BASIC_AUTH 2.35 ++ 2.36 ++# if ENABLE_FEATURE_HTTPD_AUTH_MD5 && ENABLE_PAM 2.37 ++struct pam_userinfo { 2.38 ++ const char *name; 2.39 ++ const char *pw; 2.40 ++}; 2.41 ++ 2.42 ++static int pam_talker(int num_msg, 2.43 ++ const struct pam_message **msg, 2.44 ++ struct pam_response **resp, 2.45 ++ void *appdata_ptr) 2.46 ++{ 2.47 ++ int i; 2.48 ++ struct pam_userinfo *userinfo = (struct pam_userinfo *) appdata_ptr; 2.49 ++ struct pam_response *response; 2.50 ++ 2.51 ++ if (!resp || !msg || !userinfo) 2.52 ++ return PAM_CONV_ERR; 2.53 ++ 2.54 ++ /* allocate memory to store response */ 2.55 ++ response = xzalloc(num_msg * sizeof(*response)); 2.56 ++ 2.57 ++ /* copy values */ 2.58 ++ for (i = 0; i < num_msg; i++) { 2.59 ++ const char *s; 2.60 ++ 2.61 ++ switch (msg[i]->msg_style) { 2.62 ++ case PAM_PROMPT_ECHO_ON: 2.63 ++ s = userinfo->name; 2.64 ++ break; 2.65 ++ case PAM_PROMPT_ECHO_OFF: 2.66 ++ s = userinfo->pw; 2.67 ++ break; 2.68 ++ case PAM_ERROR_MSG: 2.69 ++ case PAM_TEXT_INFO: 2.70 ++ s = ""; 2.71 ++ break; 2.72 ++ default: 2.73 ++ free(response); 2.74 ++ return PAM_CONV_ERR; 2.75 ++ } 2.76 ++ response[i].resp = xstrdup(s); 2.77 ++ if (PAM_SUCCESS != 0) 2.78 ++ response[i].resp_retcode = PAM_SUCCESS; 2.79 ++ } 2.80 ++ *resp = response; 2.81 ++ return PAM_SUCCESS; 2.82 ++} 2.83 ++# endif 2.84 ++ 2.85 + /* 2.86 + * Config file entries are of the form "/<path>:<user>:<passwd>". 2.87 + * If config file has no prefix match for path, access is allowed. 2.88 +@@ -1672,7 +1732,7 @@ 2.89 + * 2.90 + * Returns 1 if user_and_passwd is OK. 2.91 + */ 2.92 +-static int check_user_passwd(const char *path, const char *user_and_passwd) 2.93 ++static int check_user_passwd(const char *path, char *user_and_passwd) 2.94 + { 2.95 + Htaccess *cur; 2.96 + const char *prev = NULL; 2.97 +@@ -1680,6 +1740,7 @@ 2.98 + for (cur = g_auth; cur; cur = cur->next) { 2.99 + const char *dir_prefix; 2.100 + size_t len; 2.101 ++ int r; 2.102 + 2.103 + dir_prefix = cur->before_colon; 2.104 + 2.105 +@@ -1704,36 +1765,96 @@ 2.106 + prev = dir_prefix; 2.107 + 2.108 + if (ENABLE_FEATURE_HTTPD_AUTH_MD5) { 2.109 +- char *md5_passwd; 2.110 ++ char *colon_after_user; 2.111 ++ const char *passwd; 2.112 ++# if ENABLE_FEATURE_SHADOWPASSWDS && !ENABLE_PAM 2.113 ++ char buffer[256]; /* will store passwd */ 2.114 ++# endif 2.115 + 2.116 +- md5_passwd = strchr(cur->after_colon, ':'); 2.117 +- if (md5_passwd && md5_passwd[1] == '$' && md5_passwd[2] == '1' 2.118 +- && md5_passwd[3] == '$' && md5_passwd[4] 2.119 +- ) { 2.120 +- char *encrypted; 2.121 +- int r, user_len_p1; 2.122 ++ colon_after_user = strchr(user_and_passwd, ':'); 2.123 ++ if (!colon_after_user) 2.124 ++ goto bad_input; 2.125 ++ passwd = strchr(cur->after_colon, ':'); 2.126 ++ if (!passwd) 2.127 ++ goto bad_input; 2.128 ++ passwd++; 2.129 ++ if (passwd[0] == '*') { 2.130 ++# if ENABLE_PAM 2.131 ++ struct pam_userinfo userinfo; 2.132 ++ struct pam_conv conv_info = { &pam_talker, (void *) &userinfo }; 2.133 ++ pam_handle_t *pamh; 2.134 + 2.135 +- md5_passwd++; 2.136 +- user_len_p1 = md5_passwd - cur->after_colon; 2.137 +- /* comparing "user:" */ 2.138 +- if (strncmp(cur->after_colon, user_and_passwd, user_len_p1) != 0) { 2.139 ++ /* compare "user:" */ 2.140 ++ if (cur->after_colon[0] != '*' 2.141 ++ && strncmp(cur->after_colon, user_and_passwd, colon_after_user - user_and_passwd + 1) != 0 2.142 ++ ) { 2.143 + continue; 2.144 + } 2.145 ++ /* this cfg entry is '*' or matches username from peer */ 2.146 ++ *colon_after_user = '\0'; 2.147 ++ userinfo.name = user_and_passwd; 2.148 ++ userinfo.pw = colon_after_user + 1; 2.149 ++ r = pam_start("httpd", user_and_passwd, &conv_info, &pamh) != PAM_SUCCESS; 2.150 ++ if (r == 0) { 2.151 ++ r = pam_authenticate(pamh, PAM_DISALLOW_NULL_AUTHTOK) != PAM_SUCCESS 2.152 ++ || pam_acct_mgmt(pamh, PAM_DISALLOW_NULL_AUTHTOK) != PAM_SUCCESS 2.153 ++ ; 2.154 ++ pam_end(pamh, PAM_SUCCESS); 2.155 ++ } 2.156 ++ *colon_after_user = ':'; 2.157 ++ goto end_check_passwd; 2.158 ++# else 2.159 ++# if ENABLE_FEATURE_SHADOWPASSWDS 2.160 ++ /* Using _r function to avoid pulling in static buffers */ 2.161 ++ struct spwd spw; 2.162 ++# endif 2.163 ++ struct passwd *pw; 2.164 + 2.165 +- encrypted = pw_encrypt( 2.166 +- user_and_passwd + user_len_p1 /* cleartext pwd from user */, 2.167 +- md5_passwd /*salt */, 1 /* cleanup */); 2.168 +- r = strcmp(encrypted, md5_passwd); 2.169 +- free(encrypted); 2.170 +- if (r == 0) 2.171 +- goto set_remoteuser_var; /* Ok */ 2.172 ++ *colon_after_user = '\0'; 2.173 ++ pw = getpwnam(user_and_passwd); 2.174 ++ *colon_after_user = ':'; 2.175 ++ if (!pw || !pw->pw_passwd) 2.176 ++ continue; 2.177 ++ passwd = pw->pw_passwd; 2.178 ++# if ENABLE_FEATURE_SHADOWPASSWDS 2.179 ++ if ((passwd[0] == 'x' || passwd[0] == '*') && !passwd[1]) { 2.180 ++ /* getspnam_r may return 0 yet set result to NULL. 2.181 ++ * At least glibc 2.4 does this. Be extra paranoid here. */ 2.182 ++ struct spwd *result = NULL; 2.183 ++ r = getspnam_r(pw->pw_name, &spw, buffer, sizeof(buffer), &result); 2.184 ++ if (r == 0 && result) 2.185 ++ passwd = result->sp_pwdp; /* note: passwd is located into buffer ! */ 2.186 ++ } 2.187 ++# endif 2.188 ++# endif /* ENABLE_PAM */ 2.189 ++ } 2.190 ++ 2.191 ++ /* compare "user:" */ 2.192 ++ if (cur->after_colon[0] != '*' 2.193 ++ && strncmp(cur->after_colon, user_and_passwd, colon_after_user - user_and_passwd + 1) != 0 2.194 ++ ) { 2.195 + continue; 2.196 + } 2.197 ++ /* this cfg entry is '*' or matches username from peer */ 2.198 ++ 2.199 ++ /* encrypt pwd from peer and check match with local one */ 2.200 ++ { 2.201 ++ char *encrypted = pw_encrypt( 2.202 ++ /* pwd: */ colon_after_user + 1, 2.203 ++ /* salt: */ passwd, 2.204 ++ /* cleanup: */ 0 2.205 ++ ); 2.206 ++ r = strcmp(encrypted, passwd); 2.207 ++ free(encrypted); 2.208 ++ goto end_check_passwd; 2.209 ++ } 2.210 ++ bad_input: ; 2.211 + } 2.212 + 2.213 + /* Comparing plaintext "user:pass" in one go */ 2.214 +- if (strcmp(cur->after_colon, user_and_passwd) == 0) { 2.215 +- set_remoteuser_var: 2.216 ++ end_check_passwd: 2.217 ++ r = strcmp(cur->after_colon, user_and_passwd); 2.218 ++ if (r == 0) { 2.219 + remoteuser = xstrndup(user_and_passwd, 2.220 + strchrnul(user_and_passwd, ':') - user_and_passwd); 2.221 + return 1; /* Ok */