--- parser3/src/classes/mail.C 2003/07/24 11:31:20 1.89 +++ parser3/src/classes/mail.C 2023/09/26 20:49:06 1.136 @@ -1,12 +1,10 @@ /** @file Parser: @b mail parser class. - Copyright (c) 2001-2003 ArtLebedev Group (http://www.artlebedev.com) - Author: Alexandr Petrosian (http://paf.design.ru) + Copyright (c) 2001-2023 Art. Lebedev Studio (http://www.artlebedev.com) + Authors: Konstantin Morshnev , Alexandr Petrosian */ -static const char* IDENT_MAIL_C="$Date: 2003/07/24 11:31:20 $"; - #include "pa_config_includes.h" #include "pa_vmethod_frame.h" @@ -19,14 +17,14 @@ static const char* IDENT_MAIL_C="$Date: #include "pa_uue.h" #include "pa_vmail.h" -#ifdef _MSC_VER -# include "smtp/smtp.h" -#endif +#include "smtp.h" + +volatile const char * IDENT_MAIL_C="$Id: mail.C,v 1.136 2023/09/26 20:49:06 moko Exp $"; // defines #define MAIL_CLASS_NAME "mail" -#define MAIL_SENDMAIL_NAME "sendmail" +#define SENDMAIL_NAME "sendmail" // consts @@ -45,7 +43,7 @@ public: // global variable -DECLARE_CLASS_VAR(mail, 0/*fictive*/, new MMail); +DECLARE_CLASS_VAR(mail, new MMail); // defines for statics @@ -54,74 +52,78 @@ DECLARE_CLASS_VAR(mail, 0/*fictive*/, ne // statics static const String mail_name(MAIL_NAME); -static const String mail_sendmail_name(MAIL_SENDMAIL_NAME); +static const String mail_sendmail_name(SENDMAIL_NAME); // helpers -static void sendmail(Request& r, - const String& message, - const String* from, const String* to) { - const char* message_cstr=message.cstr(String::L_UNSPECIFIED); - Value* vmail_conf=static_cast(r.classes_conf.get(mail_base_class->name())); - +static void sendmail( + Value* +#ifndef WIN32 + vmail_conf +#endif + , Value* smtp_server_port, + const String& message, + const String* from, + const String* to, + const String* +#ifndef WIN32 + options +#endif + ) { const char* exception_type="email.format"; if(!from) // we use in sendmail -f {from} && SMTP MAIL from: {from} - throw Exception(exception_type, - 0, - "parameter does not specify 'from' header field"); + throw Exception(exception_type, 0, "parameter does not specify 'from' header field"); -#ifdef _MSC_VER - if(!to) // we use only in SMTP RCPT to: {to} - throw Exception(exception_type, - 0, - "parameter does not specify 'to' header field"); + const char* message_cstr=message.untaint_cstr(String::L_AS_IS); - SMTP smtp; - Value* server_port; - // $MAIN:MAIL.SMTP[mail.yourdomain.ru[:port]] - if(vmail_conf && - (server_port=vmail_conf->get_hash()->get(StringBody("SMTP")))) { - char* server=server_port->as_string().cstrm(); + if(smtp_server_port) { + if(!to) // we use only in SMTP RCPT to: {to} + throw Exception(exception_type, 0, "parameter does not specify 'to' header field"); + + SMTP smtp; + char* server=smtp_server_port->as_string().cstrm(); const char* port=rsplit(server, ':'); if(!port) port="25"; smtp.Send(server, port, message_cstr, from->cstrm(), to->cstrm()); - } else - throw Exception("parser.runtime", - 0, - "$"MAIN_CLASS_NAME":"MAIL_NAME".SMTP not defined"); + return; + } + +#ifdef WIN32 + // win32 without SMTP server configured + throw Exception(PARSER_RUNTIME, 0, "$" MAIN_CLASS_NAME ":" MAIL_NAME ".SMTP not defined"); #else // unix // $MAIN:MAIL.sendmail["/usr/sbin/sendmail -t -i -f postmaster"] default // $MAIN:MAIL.sendmail["/usr/lib/sendmail -t -i -f postmaster"] default - const String* sendmail_command; + String* sendmail_command=new String; if(vmail_conf) { #ifdef PA_FORCED_SENDMAIL - throw Exception("parser.runtime", + throw Exception(PARSER_RUNTIME, 0, - "Parser was configured with --with-sendmail="PA_FORCED_SENDMAIL + "Parser was configured with --with-sendmail=" PA_FORCED_SENDMAIL " key, to change sendmail you should reconfigure and recompie it"); #else if(Value* sendmail_value=vmail_conf->get_hash()->get(mail_sendmail_name)) - sendmail_command=&sendmail_value->as_string(); + *sendmail_command<as_string(); else - throw Exception("parser.runtime", - 0, - "$"MAIN_CLASS_NAME":"MAIL_NAME"."MAIL_SENDMAIL_NAME" not defined"); + throw Exception(PARSER_RUNTIME, 0, "$" MAIN_CLASS_NAME ":" MAIL_NAME "." SENDMAIL_NAME " not defined"); #endif } else { #ifdef PA_FORCED_SENDMAIL - sendmail_command=new String(PA_FORCED_SENDMAIL); + *sendmail_command<pos("postmaster"); @@ -139,9 +141,8 @@ static void sendmail(Request& r, if(after_file_spec==STRING_NOT_FOUND || after_file_spec==0) file_spec=sendmail_command; else { - size_t pos_after=after_file_spec; - file_spec=&sendmail_command->mid(0, pos_after++); - sendmail_command->split(argv, pos_after, " ", String::L_AS_IS); + file_spec=&sendmail_command->mid(0, after_file_spec); + sendmail_command->split(argv, after_file_spec+1, " ", String::L_AS_IS); } if(!file_executable(*file_spec)) @@ -151,12 +152,10 @@ static void sendmail(Request& r, #ifdef PA_FORCED_SENDMAIL " Use configure key \"--with-sendmail=appropriate sendmail command\"" #else - " Set $"MAIN_CLASS_NAME":"MAIL_NAME"."MAIL_SENDMAIL_NAME" to appropriate sendmail command" + " Set $" MAIN_CLASS_NAME ":" MAIL_NAME "." SENDMAIL_NAME " to appropriate sendmail command" #endif ); - - String in(message_cstr); String out; String err; PA_exec_result exec=pa_exec( // forced_allow #ifdef PA_FORCED_SENDMAIL @@ -167,40 +166,45 @@ static void sendmail(Request& r, , *file_spec, 0 /* pass env */, argv, - in); + String::C(message_cstr, strlen(message_cstr))); + if(exec.status || exec.err.length()) - throw Exception("email.send", - 0, - "'%s' reported problem: %s (%d)", - file_spec->cstr(), - exec.err.length()?exec.err.cstr():"UNKNOWN", - exec.status); -#endif + throw Exception("email.send", 0, "'%s' reported problem: %s (%d)", file_spec->cstr(), exec.err.length() ? exec.err.cstr() : "UNKNOWN", exec.status); +#endif //WIN32 } // methods static void _send(Request& r, MethodParams& params) { - Value& vhash=params.as_no_junction(0, "message must not be code"); - HashStringValue* hash=vhash.get_hash(); - if(!hash) - throw Exception("parser.runtime", - 0, - "message must be hash"); + HashStringValue* hash=params.as_hash(0, "message"); + if(!hash || !hash->count()) + return; + // todo@ check if enough options are specified. + // now ^mail:send[^hash::create[]] and ^mail:send[$.print-debug(1)] "work". + + const String* soptions=0; + if(Value* voptions=hash->get(MAIL_OPTIONS_NAME)) + soptions=&voptions->as_string(); + + bool print_debug=false; + if(Value* vdebug=hash->get(MAIL_DEBUG_NAME)) + print_debug=vdebug->as_bool(); + + Value* vmail_conf=static_cast(r.classes_conf.get(mail_class->type())); + Value* smtp_server_port=0; + if(vmail_conf) { + // $MAIN:MAIL.SMTP[mail.yourdomain.ru[:port]] + smtp_server_port=vmail_conf->get_hash()->get("SMTP"); + } const String* from=0; String* to=0; - const String& message= - GET_SELF(r, VMail).message_hash_to_string(r, hash, 0, from, -#ifdef WIN32 - true -#else - false -#endif - , to); + const String& message = GET_SELF(r, VMail).message_hash_to_string(r, hash, from, smtp_server_port ? true : false /*send by SMTP=strip to?*/, to); - //r.write_pass_lang(message); - sendmail(r, message, from, to); + if(print_debug) + r.write(message); + else + sendmail(vmail_conf, smtp_server_port, message, from, to, soptions); } // constructor & configurator @@ -213,12 +217,11 @@ MMail::MMail(): Methoded(MAIL_CLASS_NAME void MMail::configure_user(Request& r) { // $MAIN:MAIL[$SMTP[mail.design.ru]] - if(Value* mail_element=r.main_class.get_element(mail_name, r.main_class, false)) + if(Value* mail_element=r.main_class.get_element(mail_name)) { if(mail_element->get_hash()) - r.classes_conf.put(name(), mail_element); + r.classes_conf.put(type(), mail_element); else if( !mail_element->is_string() ) - throw Exception("parser.runtime", - 0, - "$" MAIL_CLASS_NAME ":" MAIL_NAME " is not hash"); + throw Exception(PARSER_RUNTIME, 0, "$" MAIL_CLASS_NAME ":" MAIL_NAME " is not hash"); + } }