autohit.call.modules.TolerantSmtpModule.java Source code

Java tutorial

Introduction

Here is the source code for autohit.call.modules.TolerantSmtpModule.java

Source

/**
 * AUTOHIT 2003
 * Copyright Erich P Gatejen (c) 1989,1997,2003,2004
 * 
 * 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
 *
 * Additional license information can be found in the documentation.
 * @author Erich P Gatejen
 */
package autohit.call.modules;

import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.Writer;

import org.apache.commons.net.io.Util;
import org.apache.commons.net.smtp.RelayPath;
import org.apache.commons.net.smtp.SMTPClient;
import org.apache.commons.net.smtp.SMTPConnectionClosedException;

import autohit.call.CallException;
import autohit.common.Constants;
import autohit.universe.UniverseException;
import autohit.vm.process.StringProcessors;

/**
 * Tolerant SMTP module. There is a client/per module at this time. This one
 * supports streaming and will not throw faults on every error. <code>
 * start(address,optional{port}) start an SMTP session<br>
 * login(optional{hostname}) login to peer<br>
 * sender(address) set the sender with address<br>
 * addsenderrelay(address) add to the sender relay path<br>
 * senderrelay() set the sender with sender relay path.  clear the accumulated relay path.<br>
 * recipient(address) add a recipient with address<br>
 * addrecipientrelay(address) add to the recipient relay path<br>
 * newrecipientrelay(address) start a new recipient relay path<br>
 * recipientrelay() add a recipient with recipient relay path.  clear the accumulated relay path.<br>
 * send(text) send message from text.<br>
 * senduni(uniobj) send message from universe object.<br>
 * senduniscrub(uniobj) send message from universe object.  scrub it first with variable replacements<br>
 * reset() reset the smtp state.<br>
 * done() complete a session.  It will logout and close.<br>
 * mailit(from,to,text,host,optional{port}) convenience method for sending small message.
 * mailituni(from,to,uniobj,host,optional{port}) convenience method for sending small message.
 * </code>
 * 
 * @author Erich P. Gatejen
 * @version 1.0 <i>Version History</i><code>EPG - Initial, branched from SimpleSmtpModule - 25 Dec03</code>
 */
public class TolerantSmtpModule extends Module {

    private final static String myNAME = "TolerantSmtp";
    private final static String TEMPFILE = "temp/tsmtpscrub";

    private final static String ERROR_STRING_FOR_FAILURE = "451";
    private final static int SMTP_ERROR_THRESHOLD = 300;
    // An SMTP error code

    /**
     * METHODS
     */
    private final static String method_START = "start";
    private final static String method_START_1_ADDRESS = "address";
    private final static String method_START_2_PORT = "port";
    private final static String method_LOGIN = "login";
    private final static String method_LOGIN_1_HOSTNAME = "hostname";
    private final static String method_SENDER = "sender";
    private final static String method_SENDER_1_ADDRESS = "address";
    private final static String method_ADDSENDERRELAY = "addsenderrelay";
    private final static String method_ADDSENDERRELAY_1_ADDRESS = "address";
    private final static String method_SENDERRELAY = "senderrelay";
    private final static String method_RECIPIENT = "recipient";
    private final static String method_RECIPIENT_1_ADDRESS = "address";
    private final static String method_ADDRECIPIENTRELAY = "addrecipientrelay";
    private final static String method_NEWRECIPIENTRELAY = "newrecipientrelay";
    private final static String method_ADDRECIPIENTRELAY_1_ADDRESS = "address";
    private final static String method_RECIPIENTRELAY = "recipientrelay";
    private final static String method_SEND = "send";
    private final static String method_SEND_1_TEXT = "text";
    private final static String method_SENDUNI = "senduni";
    private final static String method_SENDUNI_1_TEXT = "uniobj";
    private final static String method_SENDUNISCRUB = "senduniscrub";
    private final static String method_SENDUNISCRUB_1_TEXT = "uniobj";
    private final static String method_RESET = "reset";
    private final static String method_DONE = "done";
    private final static String method_MAILIT = "mailit";
    private final static String method_MAILIT_1_TO = "to";
    private final static String method_MAILIT_2_FROM = "from";
    private final static String method_MAILIT_3_TEXT = "text";
    private final static String method_MAILIT_4_HOSTNAME = "host";
    private final static String method_MAILIT_5_PORT = "port";
    private final static String method_MAILITUNI = "mailituni";
    private final static String method_MAILITUNI_1_TO = "to";
    private final static String method_MAILITUNI_2_FROM = "from";
    private final static String method_MAILITUNI_3_UNIOBJ = "uniobj";
    private final static String method_MAILITUNI_4_HOSTNAME = "host";
    private final static String method_MAILITUNI_5_PORT = "port";

    SMTPClient client;
    RelayPath senderrelay;
    RelayPath recipientrelay;
    boolean loggedIn;

    /**
     * Constructor
     */
    public TolerantSmtpModule() {

    }

    // IMPLEMENTORS

    /**
     * Execute a named method. You must implement this method. You can call any
     * of the helpers for data and services. The returned object better be a
     * string (for now).
     * 
     * @param name
     *            name of the method
     * @see autohit.common.NOPair
     * @throws CallException
     */
    public Object execute_chain(String name) throws CallException {

        Object response = Constants.EMPTY_LEFT;
        Object thingie;

        if (name.equals(method_START)) {
            String param1 = this.required(method_START_1_ADDRESS, name);
            String param2 = this.optional(method_START_2_PORT);
            this.start(param1, param2);

        } else if (name.equals(method_LOGIN)) {
            String param1 = this.optional(method_LOGIN_1_HOSTNAME);
            this.login(param1);

        } else if (name.equals(method_SENDER)) {
            String param1 = this.required(method_SENDER_1_ADDRESS, name);
            this.sender(param1);

        } else if (name.equals(method_ADDSENDERRELAY)) {
            String param1 = this.required(method_ADDSENDERRELAY_1_ADDRESS, name);
            this.addsenderrelay(param1);

        } else if (name.equals(method_SENDERRELAY)) {
            this.senderrelay();

        } else if (name.equals(method_RECIPIENT)) {
            String param1 = this.required(method_RECIPIENT_1_ADDRESS, name);
            this.recipient(param1);

        } else if (name.equals(method_ADDRECIPIENTRELAY)) {
            String param1 = this.required(method_ADDRECIPIENTRELAY_1_ADDRESS, name);
            this.addrecipientrelay(param1);

        } else if (name.equals(method_NEWRECIPIENTRELAY)) {
            this.newrecipientrelay();

        } else if (name.equals(method_RECIPIENTRELAY)) {
            this.recipientrelay();

        } else if (name.equals(method_SEND)) {
            String param1 = this.required(method_SEND_1_TEXT, name);
            response = this.send(param1);

        } else if (name.equals(method_SENDUNI)) {
            String param1 = this.required(method_SENDUNI_1_TEXT, name);
            response = this.senduni(param1);

        } else if (name.equals(method_SENDUNISCRUB)) {
            String param1 = this.required(method_SENDUNISCRUB_1_TEXT, name);
            response = this.senduniscrub(param1);

        } else if (name.equals(method_RESET)) {
            this.reset();

        } else if (name.equals(method_DONE)) {
            this.done();

        } else if (name.equals(method_MAILIT)) {
            String param1 = this.required(method_MAILIT_1_TO, name);
            String param2 = this.required(method_MAILIT_2_FROM, name);
            String param3 = this.required(method_MAILIT_3_TEXT, name);
            String param4 = this.required(method_MAILIT_4_HOSTNAME, name);
            String param5 = this.optional(method_MAILIT_5_PORT);
            response = this.mailit(param1, param2, param3, param4, param5);

        } else if (name.equals(method_MAILITUNI)) {
            String param1 = this.required(method_MAILITUNI_1_TO, name);
            String param2 = this.required(method_MAILITUNI_2_FROM, name);
            String param3 = this.required(method_MAILITUNI_3_UNIOBJ, name);
            String param4 = this.required(method_MAILITUNI_4_HOSTNAME, name);
            String param5 = this.optional(method_MAILITUNI_5_PORT);
            response = this.mailituni(param1, param2, param3, param4, param5);

        } else {
            error("Not a provided method.  method=" + name);
        }
        return response;
    }

    /**
     * Allow the subclass a chance to initialize. At a minium, an implementor
     * should create an empty method.
     * 
     * @throws CallException
     * @return the name
     */
    protected String instantiation_chain() throws CallException {
        client = null;
        loggedIn = false;
        return myNAME;
    }

    /**
     * Allow the subclass a chance to cleanup on free. At a minium, an
     * implementor should create an empty method.
     * 
     * @throws CallException
     */
    protected void free_chain() throws CallException {
        try {
            this.done();
        } catch (Exception e) {
            // don't care
        }
    }

    // PRIVATE IMPLEMENTATIONS

    /**
     * Start method. It will open a connection to an SMTP server/relay. If a
     * session is already started, it will report an error. If it cannot make
     * the connection, it will cause a fault.
     * 
     * @param addr
     *            the domain name address. Do not include protocol or port.
     * @param post
     *            this should be a parsable integer. If it is null, the default
     *            will be used.
     * @throws CallException
     */
    private void start(String addr, String port) throws CallException {

        SMTPClient candidate = null;

        // Already started?
        if (client != null) {
            this.error("Session already started.  Ignoring new start().");
            return;
        }
        // Passed a port number?
        int portNum = 0;
        if (port != null) {
            try {
                portNum = Integer.parseInt(port);
            } catch (Exception e) {
                this.fault("Malformed 'port' number.  It must be a parsable integer.  text=" + port);
            }
        }
        // Try and construct it
        try {
            candidate = new SMTPClient();
            if (port == null) {
                candidate.connect(addr);
            } else {
                candidate.connect(addr, portNum);
            }

        } catch (Exception ex) {
            this.error("Could not connect to host.  message=" + ex.getMessage());
            return;
        }

        // NO CODE AFTER THIS!
        this.log("Connection started.");
        loggedIn = false;
        client = candidate;
    }

    /**
     * Done method. Dispose of state and everything.
     */
    private void done() {

        // Brute force close. Don't care about errors
        if (client != null) {
            try {
                client.logout();
            } catch (Exception exx) {
            }
            try {
                client.disconnect();
            } catch (Exception exx) {
            }
        }
        senderrelay = null;
        client = null;
        loggedIn = false;
        this.log("Connection closed.");
        // NO NEW CODE BEFORE THIS LINE!
    }

    /**
     * Login method. It will fault if a session has not been started or has
     * expired. The hostname is optional; pass null if not used.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private void login(String hostname) throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {
            if (hostname == null) {
                client.login();
            } else {
                client.login(hostname);
            }

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Login failed with code=" + code + " reply=" + client.getReplyString());
            } else {
                loggedIn = true;
                if (this.isDebugging()) {
                    this.debug("Login complete.  code=" + code);
                }
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Cannot login.  Connection expired and closed itself.");
        } catch (Exception ex) {
            this.done();
            this.error("Cannot login due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Sender method. It will fault if a session has not been started or has
     * expired. It will set the sender. Subsequent calls will overwrite the
     * value. The address should be a valid email address.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private void sender(String s) throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {
            client.setSender(s);

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Sender failed with code=" + code + " reply=" + client.getReplyString());
            } else if (this.isDebugging()) {
                this.debug("Sender complete.  code=" + code);
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Cannot set sender.  Connection expired and closed itself.");
        } catch (Exception ex) {
            this.done();
            this.error("Cannot set sender due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Add sender relay leg method. It will fault if a session has not been
     * started or has expired. It will start accumulating a sender relay path.
     * You must call this at least once, if you are going to use senderrelay().
     * The first call should contain a complete email address at the root of
     * the relay chain. The accumulation will be reset after done() or other
     * connection ending event.
     * 
     * @param leg
     *            relay leg.
     * @throws CallException
     */
    private void addsenderrelay(String leg) throws CallException {

        // Is it started?
        if (client == null) {
            this.fault("Session not start()'ed.");
        }

        try {
            if (senderrelay == null) {
                senderrelay = new RelayPath(leg);
            } else {
                senderrelay.addRelay(leg);
            }

        } catch (Exception ex) {
            this.done();
            this.error("Cannot add sender relay due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Sender relay method. It will fault if a session has not been started or
     * has expired. It will set the sender as the accumulated relay. You must
     * have called addsenderrelay() at least once or you will get an error.
     * Subsequent calls will overwrite the value.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private void senderrelay() throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        // Is there a relay?
        if (senderrelay == null) {
            this.error(
                    "Sender relay not ready.  You need to call addsenderrelay() at least once before this method.");
            return;
        }

        try {
            client.setSender(senderrelay);

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Set sender relay failed with code=" + code + " reply=" + client.getReplyString());
            } else if (this.isDebugging()) {
                this.debug("Set sender relay complete.  code=" + code);
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Cannot set sender.  Connection expired and closed itself.");
        } catch (Exception ex) {
            this.done();
            this.error("Cannot set sender due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Recipient method. It will fault if a session has not been started or has
     * expired. It will add a recipient. Subsequent calls will add to the list
     * of recipients. The address should be a valid email address. The only way
     * to clear the list is to call reset() and start over.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private void recipient(String s) throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {
            client.addRecipient(s);

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Add recipient failed with code=" + code + " reply=" + client.getReplyString());
            } else if (this.isDebugging()) {
                this.debug("Add recipient complete.  code=" + code);
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Cannot add recipient.  Connection expired and closed itself.");
        } catch (Exception ex) {
            this.done();
            this.error("Cannot add recipient due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Add recipeint relay leg method. It will fault if a session has not been
     * started or has expired. It will start accumulating a recipeint relay
     * path. You must call this at least once, if you are going to use
     * recipient relay(). The first call should contain a complete email
     * address at the root of the relay chain. The accumulation will be reset
     * after done(), a call to new recipientrelay(), or some other connection
     * ending event.
     * 
     * @param leg
     *            relay leg.
     * @throws CallException
     */
    private void addrecipientrelay(String leg) throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {
            if (recipientrelay == null) {
                recipientrelay = new RelayPath(leg);
            } else {
                recipientrelay.addRelay(leg);
            }
        } catch (Exception ex) {
            this.done();
            this.error("Cannot add recipient relay due to exception.  message=" + ex.getMessage());
        }
    }

    /**
     * Recipient relay method. It will fault if a session has not been started
     * or has expired. It will set the sender as the accumulated relay. You
     * must have called addrecipientrelay() at least once or you will get an
     * error. Subsequent calls will overwrite the value.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private void recipientrelay() throws CallException {

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        // Is there a relay?
        if (recipientrelay == null) {
            this.error(
                    "Recipient relay not ready.  You need to call addrecipientrelay() at least once before this method.");
            return;
        }

        try {
            client.addRecipient(recipientrelay);

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Add recipient relay failed with code=" + code + " reply=" + client.getReplyString());
            } else if (this.isDebugging()) {
                this.debug("Add recipient relay complete.  code=" + code);
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Cannot add recipient.  Connection expired and closed itself.");
        } catch (Exception ex) {
            this.done();
            this.error("Cannot add recipient due to exception.  message=" + ex.getMessage());
        }

    }

    /**
     * Clear the recipeint relay accumulation. Do this if you want to start on
     * a new recipeint relay. This will never report any kind of error.
     * 
     * @param leg
     *            relay leg.
     * @throws CallException
     */
    private void newrecipientrelay() {
        recipientrelay = null;
    }

    /**
     * Send the message in the text. Returns the SMTP reply code.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private String send(String text) throws CallException {

        String result = ERROR_STRING_FOR_FAILURE;

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        // Is there something to send? PARANOID
        if (text == null) {
            this.error("Nothing to send.");
            return result;
        }

        try {

            client.sendShortMessageData(text);
            //client.completePendingCommand(); // don't care if it was ok.
            result = Integer.toString(client.getReplyCode());

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Message send FAILED.  code=" + code + " reply=" + client.getReplyString());
            } else if (this.isDebugging()) {
                this.debug("Message send complete.  code=" + code + " reply=" + client.getReplyString());
            }

        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Send failed.  Connection expired and closed itself." + " last reply="
                    + client.getReplyString());
        } catch (Exception ex) {
            this.done();
            this.error("Send failed due to exception.  message=" + ex.getMessage());
        }
        return result;
    }

    /**
     * Send the message in a Universe Object. Returns the SMTP reply code.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private String senduni(String uniobject) throws CallException {

        String result = ERROR_STRING_FOR_FAILURE;

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {
            // get the hoses
            InputStream unio = visUniverse.getStream(uniobject);
            BufferedReader bin = new BufferedReader(new InputStreamReader(unio));
            Writer mwriter = client.sendMessageData();

            // and pipe them together
            if (mwriter != null) {
                Util.copyReader(bin, mwriter);
                mwriter.close();
                unio.close();
                client.completePendingCommand(); // don't care if it was ok.

                // what happened?
                int code = client.getReplyCode();
                if (code >= SMTP_ERROR_THRESHOLD) {
                    this.error("Message send FAILED (from Universe).  code=" + code + " reply="
                            + client.getReplyString());
                } else if (this.isDebugging()) {
                    this.debug("Message send complete (from Universe).  code=");
                }
                result = Integer.toString(code);

            } else {
                this.log("Message send FAILED (from Universe) because SMTP connection was completely ready.  reply="
                        + client.getReplyString());
            }

        } catch (UniverseException uex) {
            this.fault("Could not send universe object due to Universe problem.  code=" + uex.numeric + "  message="
                    + uex.getMessage(), uex);
        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Send failed (from Universe).  Connection expired and closed itself.  Last reply="
                    + client.getReplyString());
        } catch (Exception ex) {
            this.done();
            this.error("Send failed (from Universe) due to exception.  message=" + ex.getMessage());
        }
        return result;
    }

    /**
     * Send the message in a Universe Object. Returns the SMTP reply code. It
     * will run a variable replace on it before sending it.
     * 
     * @param hostname
     *            hostname to use instead of localhost.
     * @throws CallException
     */
    private String senduniscrub(String uniobject) throws CallException {

        String result = ERROR_STRING_FOR_FAILURE;
        String tempObj = null;

        // Is it started?
        if (client == null) {
            this.error("Session not start()'ed.");
        }

        try {

            // Process it
            try {
                tempObj = visUniverse.reserveUnique(TEMPFILE);
                OutputStream os = visUniverse.putStream(tempObj);
                StringProcessors.evalStreams2Core(visUniverse.getStream(uniobject), os, visCore);
                os.close();

            } catch (UniverseException ue) {
                throw ue;
            } catch (Exception e) {
                this.fault(
                        "Send failed (from Universe).  Could not create scrubbed intermediary tempfile.  message="
                                + e.getMessage());
            }

            // get the hoses
            InputStream unio = visUniverse.getStream(tempObj);
            BufferedReader bin = new BufferedReader(new InputStreamReader(unio));
            Writer mwriter = client.sendMessageData();

            // and pipe them together
            if (mwriter != null) {
                Util.copyReader(bin, mwriter);
                mwriter.close();
                unio.close();
                client.completePendingCommand(); // don't care if it was ok.

                // what happened?
                int code = client.getReplyCode();
                if (code >= SMTP_ERROR_THRESHOLD) {
                    this.error("Message send FAILED (from Universe).  code=" + code + " reply="
                            + client.getReplyString());
                } else if (this.isDebugging()) {
                    this.debug("Message send complete (from Universe).  code=");
                }
                result = Integer.toString(code);

            } else {
                this.log("Message send FAILED (from Universe) because SMTP connection was completely ready.  reply="
                        + client.getReplyString());
            }

        } catch (CallException cex) {
            throw cex;
        } catch (UniverseException uex) {
            this.fault("Could not send universe object due to Universe problem.  code=" + uex.numeric + "  message="
                    + uex.getMessage(), uex);
        } catch (SMTPConnectionClosedException ex) {
            this.done();
            this.error("Send failed (from Universe).  Connection expired and closed itself.  Last reply="
                    + client.getReplyString());
        } catch (Exception ex) {
            this.done();
            this.error("Send failed (from Universe) due to exception.  message=" + ex.getMessage());
        } finally {
            if ((!this.isDebugging()) && (tempObj != null)) {
                try {
                    visUniverse.remove(tempObj);
                } catch (Exception ee) {
                    // Don't care
                }
            }
        }
        return result;
    }

    /**
     * Reset method. It will never give an error, even if not start()'ed.
     * 
     * @throws CallException
     */
    private void reset() throws CallException {
        try {
            if (client != null)
                client.reset();

            // what happened?
            int code = client.getReplyCode();
            if (code >= SMTP_ERROR_THRESHOLD) {
                this.error("Reset FAILED with code=" + code + " reply=" + client.getReplyString());
                this.done();
            } else if (this.isDebugging()) {
                this.debug("Reset complete.  code=" + code);
            }

        } catch (Exception ex) {
        }
    }

    /**
     * Mailit method. A complete transaction wrapped into a convenient method.
     * It will start a session using host/port parameters. If a session is
     * already started, host/port will be ignored and the current session will
     * be used.
     * 
     * @param to
     *            TO address.
     * @param from
     *            FROM address.
     * @param text
     *            to send as the message
     * @param host
     *            address of smpt server or relay
     * @param port
     *            optional port
     * @throws CallException
     */
    private String mailit(String to, String from, String text, String host, String port) throws CallException {

        String response = null;

        // Is it started?
        if (client == null) {
            this.start(host, port);
        }

        if (client == null)
            return "XXX";

        try {

            if (loggedIn == false)
                this.login(null);
            this.reset();
            this.sender(from);
            this.recipient(to);
            response = this.send(text);

        } catch (CallException ce) {
            this.done();
            throw ce;
        } catch (Exception ee) {
            this.done();
            this.fault("Serious exception caused FAULT while sending.  message=" + ee.getMessage());
        }
        return response;
    }

    /**
     * Mailituni method. A complete transaction wrapped into a convenient
     * method. It will start a session using host/port parameters. If a session
     * is already started, host/port will be ignored and the current session
     * will be used.
     * 
     * @param to
     *            TO address.
     * @param from
     *            FROM address.
     * @param uniobj
     *            to send as the message from universe
     * @param host
     *            address of smpt server or relay
     * @param port
     *            optional port
     * @throws CallException
     */
    private String mailituni(String to, String from, String uniobj, String host, String port) throws CallException {

        String response = null;

        // Is it started?
        if (client == null) {
            this.start(host, port);
        }

        if (client == null)
            return "XXX";

        try {
            if (loggedIn == false)
                this.login(null);
            this.reset();
            this.sender(from);
            this.recipient(to);
            response = this.senduniscrub(uniobj);

        } catch (CallException ce) {
            this.done();
            throw ce;
        } catch (Exception ee) {
            this.done();
            this.fault("Serious exception caused FAULT while sending.  message=" + ee.getMessage());
        }
        return response;
    }
}