Jason 'XenoPhage' Frisvold said:
> I'll look into it and see what I come up with myself... If I'm
> successful, I'll be sure to send a patch to the list.. :)
Turns out this was nothing more than the moving of a single line of
code... :) I've attached a complete patch to this message. Standard
disclaimer applies, use at your own risk, if the sun melts down, it's not
my fault, etc..
I hope this helps everyone out there! This toaster is *GREAT* and I'm
just trying to do my part by giving some back!
---------------------------
Jason 'XenoPhage' Frisvold
Engine / Technology Programmer
[EMAIL PROTECTED]
RedHat Certified - RHCE # 803004140609871
MySQL Pro Certified - ID# 207171862
MySQL Core Certified - ID# 205982910
---------------------------
"Something mysterious is formed, born in the silent void. Waiting alone
and unmoving, it is at once still and yet in constant motion. It is the
source of all programs. I do not know its name, so I will call it the Tao
of Programming."
This patch is intended to be used in conjunction with Bill Shupp's large
qmail-toaster-greylisting patch. I tool the existing chkusr-mysql patch
and applied it to qmail, ensuring that chkusr is run before greylisting.
Kudos to Bill and all the others who have put forth the effort to make
such a great email toaster!
So as to not take away any credit due, Bill's original message is below...
Jason Frisvold
This patch was writtn by Antonio Nati ([EMAIL PROTECTED]), and is
only slightly modified by me. This version is suitable for use with mysql
and has tmda support as well. Please edit conf-mysql to match the location
of your libmysqlclient.a library.
Here's a link to the original web page:
http://www.interazioni.it/qmail/#qmail-smtpd
Bill Shupp
diff -urN netqmail-1.05/conf-mysql netqmail-1.05-mine/conf-mysql
--- netqmail-1.05/conf-mysql 1969-12-31 19:00:00.000000000 -0500
+++ netqmail-1.05-mine/conf-mysql 2004-04-30 17:16:34.000000000 -0400
@@ -0,0 +1,3 @@
+/usr/lib/mysql/libmysqlclient.a
+
+The FULL path do the libmysqlclient.a library
diff -urN netqmail-1.05/conf-vpopmail netqmail-1.05-mine/conf-vpopmail
--- netqmail-1.05/conf-vpopmail 1969-12-31 19:00:00.000000000 -0500
+++ netqmail-1.05-mine/conf-vpopmail 2004-04-30 17:16:49.000000000 -0400
@@ -0,0 +1,3 @@
+/home/vpopmail
+
+The home directory of the vpopmail user
diff -urN netqmail-1.05/Makefile netqmail-1.05-mine/Makefile
--- netqmail-1.05/Makefile 2004-04-30 21:32:45.000000000 -0400
+++ netqmail-1.05-mine/Makefile 2004-04-30 17:18:26.000000000 -0400
@@ -23,7 +23,7 @@
auto-ccld.sh: \
conf-cc conf-ld warn-auto.sh
( cat warn-auto.sh; \
- echo CC=\'`head -1 conf-cc`\'; \
+ echo CC=\'`head -1 conf-cc` -I`head -1 conf-vpopmail`/include\'; \
echo LD=\'`head -1 conf-ld`\' \
) > auto-ccld.sh
@@ -1584,8 +1584,10 @@
tls.o ssl_timeoutio.o ndelay.a -L/usr/local/ssl/lib -lssl -lcrypto \
received.o date822fmt.o now.o qmail.o cdb.a fd.a wait.a \
datetime.a getln.a open.a sig.a case.a env.a stralloc.a \
- alloc.a strerr.a substdio.a error.a str.a fs.a auto_qmail.o base64.o `cat \
- socket.lib` dns.o `cat dns.lib`
+ alloc.a strerr.a substdio.a error.a str.a fs.a auto_qmail.o base64.o \
+ `cat socket.lib` dns.o `cat dns.lib` -lcrypt \
+ `head -1 conf-vpopmail`/lib/libvpopmail.a \
+ `head -1 conf-mysql` -lz -lm
qmail-envelope-scanner: \
load qmail-envelope-scanner.o local_scan.o /usr/lib/mysql/libmysqlclient.a
diff -urN netqmail-1.05/qmail-smtpd.c netqmail-1.05-mine/qmail-smtpd.c
--- netqmail-1.05/qmail-smtpd.c 2004-04-30 21:32:45.000000000 -0400
+++ netqmail-1.05-mine/qmail-smtpd.c 2004-04-30 21:09:07.000000000 -0400
@@ -25,6 +25,17 @@
#include "timeoutread.h"
#include "timeoutwrite.h"
#include "commands.h"
+#include <pwd.h>
+#include <sys/types.h>
+#include <unistd.h>
+#include <dirent.h>
+#include <stdio.h>
+
+#include "open.h"
+#include <vpopmail.h>
+#include <vauth.h>
+#include <vpopmail_config.h>
+
#include "qregex.h"
#include "strerr.h"
@@ -313,6 +324,194 @@
return 0;
}
+void err_realrcpt() { out("550 sorry, no mailbox here by that name (#5.1.1 - chkusr)\r\n"); }
+
+int realrcpt_check()
+{
+ stralloc user = {0};
+ stralloc domain = {0};
+ stralloc domain_path = {0};
+ stralloc bounce_path = {0};
+ stralloc alias_name = {0};
+ stralloc alias_path = {0};
+ stralloc mailing_path = {0};
+ int count;
+ int retstat = 0;
+ struct vqpasswd *user_passwd = NULL;
+ int fd_file = -1;
+ int read_char;
+ DIR *dir_file = NULL;
+ int offset;
+ char read_buf[1024];
+
+/* if not local rcpthost we cannot control mailbox */
+
+ if (!addrallowed()) { return 1; }
+
+/* Set up our variables */
+
+/* Search the '@' character */
+ count = byte_rchr(addr.s,addr.len,'@');
+
+/*
+ * Give extra room to variables used often or used outside stralloc_x calls
+ * This should make all safer and even faster
+ * (when these fields are used by stralloc_x routines)
+*/
+ if (!stralloc_ready (&domain, 200)) die_nomem();
+ if (!stralloc_ready (&domain_path, 200)) die_nomem();
+
+ if (count < addr.len) {
+ if (!stralloc_copyb (&user, addr.s, count)) die_nomem();
+ if (!stralloc_0 (&user)) die_nomem();
+ if (!stralloc_copys (&domain, addr.s + count + 1)) die_nomem();
+ if (!stralloc_0 (&domain)) die_nomem();
+ }
+ else {
+ if (!stralloc_copys (&user, addr.s)) die_nomem();
+ if (!stralloc_0 (&user)) die_nomem();
+ if (!stralloc_copys (&domain, DEFAULT_DOMAIN)) die_nomem();
+ if (!stralloc_0 (&domain)) die_nomem();
+ }
+
+/* My personal control: continue only if a domain (default or not) is specified */
+
+ if (domain.len == 1)
+ return 0;
+
+ case_lowers (user.s);
+ case_lowers (domain.s);
+
+/* Check if domain is a real domain */
+
+ /* if (!stralloc_0 (&domain)) die_nomem();
+ vget_real_domain(domain.s, domain.a);
+
+ domain.len = strlen (domain.s);
+ if (domain.len > (domain.a - 1)) die_nomem(); */
+
+/* Let's get domain's real path */
+ vget_assign(domain.s, domain_path.s, 156, NULL, NULL);
+
+ domain_path.len = strlen (domain_path.s);
+
+/* Now Let's start the test suite */
+
+ switch (0) {
+
+ case 0:
+/* Check if domain has bouncing enabled */
+
+ /* Allocate room for bounce_path */
+ if (!stralloc_ready (&bounce_path, 200)) die_nomem();
+ if (!stralloc_copy (&bounce_path, &domain_path)) die_nomem();
+ if (!stralloc_cats (&bounce_path, "/.qmail-default")) die_nomem();
+ if (!stralloc_0 (&bounce_path)) die_nomem();
+
+ read_char = 0;
+ fd_file = open_read (bounce_path.s);
+ if (fd_file != -1) {
+ read_char = read (fd_file, read_buf, sizeof(read_buf) - 1);
+ close (fd_file);
+ if (read_char < 0) read_char = 0;
+ }
+ read_buf[read_char] = 0;
+
+ if ( strstr(read_buf, "bounce-no-mailbox") == NULL ) {
+ retstat = 1;
+ break;
+ }
+
+ case 1:
+/* User control: check the existance of a real user */
+
+ user_passwd = vauth_getpw (user.s, domain.s);
+ if (user_passwd == NULL) {
+ count = 0;
+ while ((count < (user.len -1)) && (user_passwd == NULL)) {
+ count += byte_chr(&user.s[count], user.len - count,'-');
+ if (count < user.len) {
+ if (!stralloc_copyb (&alias_name, user.s, count)) die_nomem();
+ if (!stralloc_0 (&alias_name)) die_nomem();
+ user_passwd = vauth_getpw (alias_name.s, domain.s);
+ ++count;
+ }
+ }
+ }
+ if (user_passwd != NULL) {
+
+ /* If user exists check if he has BOUNCE_MAIL flag set */
+
+ if (user_passwd->pw_gid & BOUNCE_MAIL)
+ retstat = 0;
+ else
+ retstat = 1;
+ break;
+ }
+
+ case 2:
+/* Check for aliases/forwards - valias*/
+
+ if (valias_select (user.s, domain.s) != NULL) {
+ retstat = 1;
+ break;
+ }
+
+ case 3:
+/* Check for aliases/forwards - .qmail.x files */
+
+ /* Allocate room for alias_path */
+ if (!stralloc_ready (&alias_path, 200)) die_nomem();
+ if (!stralloc_copy (&alias_name, &user)) die_nomem();
+
+ /* Change all '.' in ':' before continuing on aliases */
+ for (count = 0; count < alias_name.len; ++count)
+ if (*(alias_name.s + count) == '.') *(alias_name.s + count) = ':';
+
+ if (!stralloc_copy (&alias_path, &domain_path)) die_nomem();
+ if (!stralloc_cats (&alias_path, "/.qmail-")) die_nomem();
+ if (!stralloc_cats (&alias_path, alias_name.s)) die_nomem();
+ if (!stralloc_0 (&alias_path)) die_nomem();
+
+ /* access executes anyway as real (vpopmail:vchkpw), that's ok */
+ if (access (alias_path.s, F_OK) == 0) {
+ retstat = 1;
+ break;
+ }
+
+ case 4:
+/* Let's check for mailing lists */
+
+ /* Allocate room for mailing_path */
+ if (!stralloc_ready (&mailing_path, 300)) die_nomem();
+
+ /* Search for the outer '-' character */
+ for (offset = user.len - 1; offset > 0; --offset)
+ if (*(user.s + offset) == '-') {
+ if (!stralloc_copy (&mailing_path, &domain_path)) die_nomem();
+ if (!stralloc_cats (&mailing_path, "/")) die_nomem();
+ if (!stralloc_catb (&mailing_path, user.s, offset)) die_nomem();
+ if (!stralloc_cats (&mailing_path, "/mailinglist")) die_nomem();
+ if (!stralloc_0 (&mailing_path)) die_nomem();
+ /* access executes anyway as real (vpopmail:vchkpw), that's ok */
+ if (access (mailing_path.s, F_OK) == 0) {
+ retstat = 1;
+ break;
+ }
+ }
+
+/*
+ * Add this code if another case is following
+ if (retstat == 1)
+ break;
+*/
+
+ } /* end switch */
+
+ return retstat;
+}
+
+
int mfcheck()
{
stralloc sa = {0};
@@ -423,6 +622,7 @@
rcptcount = 0;
out("250 ok\r\n");
}
+
void smtp_rcpt(arg) char *arg; {
if (!seenmail) { err_wantmail(); return; }
if (!addrparse(arg)) { err_syntax(); return; }
@@ -437,6 +637,9 @@
err_bmt();
return;
}
+ // Check if the rcpt is real first
+ if (!realrcpt_check()) { err_realrcpt(); return; }
+ // Then check to see if they've been through the greylist
if (relayclient) {
--addr.len;
if (!stralloc_cats(&addr,relayclient)) die_nomem();