Annotation of parser3/src/classes/mail.C, revision 1.88.2.9

1.1       paf         1: /** @file
                      2:        Parser: @b mail parser class.
                      3: 
1.88.2.1  paf         4:        Copyright (c) 2001-2003 ArtLebedev Group (http://www.artlebedev.com)
1.54      paf         5:        Author: Alexandr Petrosian <paf@design.ru> (http://paf.design.ru)
1.73      paf         6: */
1.1       paf         7: 
1.88.2.9! paf         8: static const char* IDENT_MAIL_C="$Date: 2003/02/17 07:18:57 $";
1.1       paf         9: 
                     10: #include "pa_config_includes.h"
1.88.2.4  paf        11: #include "pa_vmethod_frame.h"
1.1       paf        12: 
                     13: #include "pa_common.h"
                     14: #include "pa_request.h"
1.6       paf        15: #include "pa_vfile.h"
1.12      paf        16: #include "pa_exec.h"
1.45      paf        17: #include "pa_charsets.h"
                     18: #include "pa_charset.h"
1.67      paf        19: #include "pa_uue.h"
1.50      paf        20: 
                     21: #ifdef _MSC_VER
                     22: #      include "smtp/smtp.h"
                     23: #endif
1.4       paf        24: 
1.23      paf        25: // defines
1.1       paf        26: 
1.23      paf        27: #define MAIL_CLASS_NAME "mail"
                     28: 
1.67      paf        29: // consts
1.25      paf        30: 
1.67      paf        31: const int ATTACHMENT_WEIGHT=100;
1.25      paf        32: 
1.23      paf        33: // class
                     34: 
1.88.2.9! paf        35: class MMail: public Methoded {
1.24      paf        36: public: // Methoded
1.67      paf        37:        bool used_directly() { return false; }
1.68      paf        38:        void configure_user(Request& r);
1.7       paf        39: 
1.88.2.6  paf        40: public:
                     41:        MMail();
1.1       paf        42: };
1.70      paf        43: 
1.88.2.6  paf        44: // global variable
1.22      paf        45: 
1.88.2.8  paf        46: MethodedPtr mail_class(0);
1.88.2.6  paf        47: MethodedPtr mail_base_class(new MMail);
1.47      paf        48: 
1.88.2.6  paf        49: // defines for statics
1.2       paf        50: 
1.88.2.6  paf        51: #define MAIL_NAME "MAIL"
1.20      paf        52: 
1.88.2.6  paf        53: // statics
                     54:        
                     55: static StringPtr mail_name(new String(MAIL_NAME));
1.2       paf        56: 
1.88.2.6  paf        57: // helpers
1.2       paf        58: 
1.88.2.6  paf        59: static void sendmail(Request& r, StringPtr method_name, 
                     60:                                         StringPtr message, 
                     61:                                         StringPtr from, StringPtr to) {
1.4       paf        62:        Pool& pool=r.pool();
                     63: 
1.88.2.6  paf        64:        CharPtr message_cstr=message->cstr(String::UL_UNSPECIFIED);
1.88.2.7  paf        65:        Value* vmail_conf=static_cast<Value*>(r.classes_conf.get(mail_base_class->name()).get());
1.12      paf        66: 
1.88.2.6  paf        67:        const char *exception_type="email.format";
1.85      paf        68:        if(!from) // we use in sendmail -f {from} && SMTP MAIL from: {from}
1.75      paf        69:                throw Exception(exception_type,
1.88.2.5  paf        70:                        method_name,
1.70      paf        71:                        "parameter does not specify 'from' header field");
1.85      paf        72: 
                     73: #ifdef _MSC_VER
                     74:        if(!to) // we use only in SMTP RCPT to: {to}
1.75      paf        75:                throw Exception(exception_type,
1.88.2.5  paf        76:                        method_name,
1.70      paf        77:                        "parameter does not specify 'to' header field");
1.4       paf        78: 
1.88.2.6  paf        79:        SMTPPtr smtp(new SMTP(method_name));
                     80:        ValuePtr server_port;
1.29      parser     81:        // $MAIN:MAIL.SMTP[mail.yourdomain.ru[:port]]
1.88.2.6  paf        82:        if(vmail_conf && 
                     83:                (server_port=vmail_conf->get_hash(method_name)->get(StringPtr(new String("SMTP"))))) {
                     84:                CharPtr server=server_port->as_string(&pool)->cstr();
                     85:                const char *port=rsplit(server, ':');
1.4       paf        86:                if(!port)
1.11      paf        87:                        port="25";
1.4       paf        88: 
1.88.2.6  paf        89:                smtp->Send(server, port, message_cstr, from->cstr(), to->cstr());
1.4       paf        90:        } else
1.60      paf        91:                throw Exception("parser.runtime",
1.88.2.5  paf        92:                        method_name,
1.13      paf        93:                        "$"MAIN_CLASS_NAME":"MAIL_NAME".SMTP not defined");
1.4       paf        94: #else
1.12      paf        95:        // unix
1.70      paf        96:        // $MAIN:MAIL.sendmail["/usr/sbin/sendmail -t -i -f postmaster"] default
                     97:        // $MAIN:MAIL.sendmail["/usr/lib/sendmail -t -i  -f postmaster"] default
1.12      paf        98: 
1.55      paf        99:        const String *sendmail_command;
1.88.2.6  paf       100:        const char *sendmailkey_cstr="sendmail";
1.86      paf       101:        if(mail_conf) {
1.55      paf       102: #ifdef PA_FORCED_SENDMAIL
1.86      paf       103:                throw Exception("parser.runtime",
1.88.2.5  paf       104:                        method_name,
1.86      paf       105:                        "Parser was configured with --with-sendmail="PA_FORCED_SENDMAIL
1.87      paf       106:                        " key, to change sendmail you should reconfigure and recompie it");
1.55      paf       107: #else
1.51      paf       108:                if(Value *sendmail_value=static_cast<Value *>(mail_conf->get(String(pool, sendmailkey_cstr))))
1.52      paf       109:                        sendmail_command=&sendmail_value->as_string();
1.51      paf       110:                else
1.60      paf       111:                        throw Exception("parser.runtime",
1.88.2.5  paf       112:                                method_name,
1.51      paf       113:                                "$"MAIN_CLASS_NAME":"MAIL_NAME".%s not defined", 
                    114:                                sendmailkey_cstr);
1.86      paf       115: #endif
1.51      paf       116:        } else {
1.86      paf       117: #ifdef PA_FORCED_SENDMAIL
                    118:                sendmail_command=new(pool) String(pool, PA_FORCED_SENDMAIL);
                    119: #else
1.52      paf       120:                String *test=new(pool) String(pool, "/usr/sbin/sendmail");
                    121:                if(!file_executable(*test))
                    122:                        test=new(pool) String(pool, "/usr/lib/sendmail");
1.70      paf       123:                test->APPEND_CONST(" -t -i -f postmaster");
1.52      paf       124:                sendmail_command=test;
1.86      paf       125: #endif
1.51      paf       126:        }
                    127: 
1.70      paf       128:        // we know sendmail_command here, should replace "postmaster" with "$from" from message
                    129:        int at_postmaster=sendmail_command->pos("postmaster");
                    130:        if(at_postmaster>0) {
                    131:                String& reconstructed=sendmail_command->mid(0, at_postmaster);
1.71      paf       132:                reconstructed << *from;
                    133:                reconstructed << sendmail_command->mid(at_postmaster+10/*postmaster*/, sendmail_command->size());
1.70      paf       134:                sendmail_command=&reconstructed;
                    135:        }
                    136: 
                    137:        // execute it
1.51      paf       138:        Array argv(pool);
                    139:        const String *file_spec;
1.52      paf       140:        int after_file_spec=sendmail_command->pos(" ", 1);
1.51      paf       141:        if(after_file_spec<=0)
1.52      paf       142:                file_spec=sendmail_command;
1.51      paf       143:        else {
1.52      paf       144:                size_t pos_after=after_file_spec;
                    145:                file_spec=&sendmail_command->mid(0, pos_after++);
                    146:                sendmail_command->split(argv, &pos_after, " ", 1, String::UL_AS_IS);
1.36      parser    147:        }
1.51      paf       148: 
1.52      paf       149:        if(!file_executable(*file_spec))
1.75      paf       150:                throw Exception("email.send",
1.55      paf       151:                        file_spec, 
                    152:                        "is not executable."
                    153: #ifdef PA_FORCED_SENDMAIL
1.59      paf       154:                        " Use configure key \"--with-sendmail=appropriate sendmail command\""
1.58      paf       155: #else
1.70      paf       156:                        " Set $"MAIN_CLASS_NAME":"MAIL_NAME".%s to appropriate sendmail command", 
1.55      paf       157:                                sendmailkey_cstr
                    158: #endif
                    159:                );
                    160: 
1.51      paf       161: 
1.67      paf       162:        String in(pool, message_cstr); String out(pool); String err(pool);
1.56      paf       163:        int exit_status=pa_exec(
                    164:                // forced_allow
                    165: #ifdef PA_FORCED_SENDMAIL
                    166:                true
                    167: #else
                    168:                false
                    169: #endif
1.57      paf       170:                , *file_spec,
1.51      paf       171:                0/*default env*/,
                    172:                &argv,
                    173:                in, out, err);
                    174:        if(exit_status || err.size())
1.75      paf       175:                throw Exception("email.send",
1.88.2.5  paf       176:                        method_name,
1.51      paf       177:                        "'%s' reported problem: %s (%d)",
                    178:                                file_spec->cstr(),
                    179:                                err.size()?err.cstr():"UNKNOWN", 
                    180:                                exit_status);
1.4       paf       181: #endif
                    182: }
                    183: 
1.7       paf       184: // methods
                    185: 
1.88.2.3  paf       186: static void _send(Request& r, StringPtr method_name, MethodParams& params) {
1.1       paf       187:        Pool& pool=r.pool();
                    188: 
1.88.2.6  paf       189:        ValuePtr vhash=params.as_no_junction(0, "message must not be code");
                    190:        HashStringValue* hash=vhash->get_hash(method_name);
1.1       paf       191:        if(!hash)
1.60      paf       192:                throw Exception("parser.runtime",
1.88.2.5  paf       193:                        method_name,
1.1       paf       194:                        "message must be hash");
                    195: 
1.88.2.6  paf       196:        StringPtr from(0);
                    197:        StringPtr to(0);
                    198:        StringPtr message=
                    199:                GET_SELF(r, VMail).message_hash_to_string(r, method_name, hash, 0, from, to);
1.1       paf       200: 
1.67      paf       201:        //r.write_pass_lang(message);
                    202:        sendmail(r, method_name, message, from, to);
1.6       paf       203: }
                    204: 
1.24      paf       205: // constructor & configurator
1.23      paf       206: 
1.88.2.6  paf       207: MMail::MMail(): Methoded(MAIL_CLASS_NAME) {
1.27      paf       208:        // ^mail:send{hash}
1.23      paf       209:        add_native_method("send", Method::CT_STATIC, _send, 1, 1);
1.68      paf       210: }
                    211: 
                    212: void MMail::configure_user(Request& r) {
                    213:        Pool& pool=r.pool();
                    214: 
                    215:        // $MAIN:MAIL[$SMTP[mail.design.ru]]
1.88.2.6  paf       216:        if(ValuePtr mail_element=r.main_class->get_element(mail_name, *r.main_class, false))
                    217:                if(mail_element->get_hash(Exception::undefined_source))
                    218:                        r.classes_conf.put(name(), mail_element);
1.68      paf       219:                else
1.69      paf       220:                        if( !mail_element->is_string() )
                    221:                                throw Exception("parser.runtime",
1.88.2.6  paf       222:                                        Exception::undefined_source,
1.69      paf       223:                                        "$" MAIL_CLASS_NAME ":" MAIL_NAME " is not hash");
1.1       paf       224: }

E-mail: