Package org.apache.james.smtpserver

Source Code of org.apache.james.smtpserver.DataLineJamesMessageHookHandler$MailToMailEnvelopeWrapper

/****************************************************************
* Licensed to the Apache Software Foundation (ASF) under one   *
* or more contributor license agreements.  See the NOTICE file *
* distributed with this work for additional information        *
* regarding copyright ownership.  The ASF licenses this file   *
* to you under the Apache License, Version 2.0 (the            *
* "License"); you may not use this file except in compliance   *
* with the License.  You may obtain a copy of the License at   *
*                                                              *
*   http://www.apache.org/licenses/LICENSE-2.0                 *
*                                                              *
* Unless required by applicable law or agreed to in writing,   *
* software distributed under the License is distributed on an  *
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY       *
* KIND, either express or implied.  See the License for the    *
* specific language governing permissions and limitations      *
* under the License.                                           *
****************************************************************/



package org.apache.james.smtpserver;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;

import javax.annotation.Resource;
import javax.mail.MessagingException;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.apache.james.core.MailImpl;
import org.apache.james.core.MimeMessageCopyOnWriteProxy;
import org.apache.james.core.MimeMessageInputStreamSource;
import org.apache.james.lifecycle.LifecycleUtil;
import org.apache.james.lifecycle.LogEnabled;
import org.apache.james.protocols.api.ExtensibleHandler;
import org.apache.james.protocols.api.LineHandler;
import org.apache.james.protocols.api.WiringException;
import org.apache.james.protocols.smtp.MailEnvelope;
import org.apache.james.protocols.smtp.SMTPResponse;
import org.apache.james.protocols.smtp.SMTPRetCode;
import org.apache.james.protocols.smtp.SMTPSession;
import org.apache.james.protocols.smtp.core.AbstractHookableCmdHandler;
import org.apache.james.protocols.smtp.core.DataLineFilter;
import org.apache.james.protocols.smtp.dsn.DSNStatus;
import org.apache.james.protocols.smtp.hook.HookResult;
import org.apache.james.protocols.smtp.hook.HookResultHook;
import org.apache.james.protocols.smtp.hook.MessageHook;
import org.apache.james.services.MailServer;
import org.apache.mailet.Mail;
import org.apache.mailet.MailAddress;

/**
*
* Handles the calling of JamesMessageHooks
*
*/
public final class DataLineJamesMessageHookHandler implements DataLineFilter, ExtensibleHandler, LogEnabled {

    /** This log is the fall back shared by all instances */
    private static final Log FALLBACK_LOG = LogFactory.getLog(DataLineJamesMessageHookHandler.class);
   
    /** Non context specific log should only be used when no context specific log is available */
    private Log serviceLog = FALLBACK_LOG;
   
    private List<JamesMessageHook> messageHandlers;
   
    private List<HookResultHook> rHooks;
   
    private MailServer mailServer;

  private List<MessageHook> mHandlers;
   
    /**
     * Gets the mail server.
     * @return the mailServer
     */
    public final MailServer getMailServer() {
        return mailServer;
    }

    /**
     * Sets the mail server.
     * @param mailServer the mailServer to set
     */
    @Resource(name="mailserver")
    public final void setMailServer(MailServer mailServer) {
        this.mailServer = mailServer;
    }
   

    /*
     * (non-Javadoc)
     * @see org.apache.james.smtpserver.protocol.core.DataLineFilter#onLine(org.apache.james.smtpserver.protocol.SMTPSession, byte[], org.apache.james.api.protocol.LineHandler)
     */
    public void onLine(SMTPSession session, byte[] line, LineHandler<SMTPSession> next) {
        MimeMessageInputStreamSource mmiss = (MimeMessageInputStreamSource) session.getState().get(SMTPConstants.DATA_MIMEMESSAGE_STREAMSOURCE);

        try {
            OutputStream out = mmiss.getWritableOutputStream();

            // 46 is "."
            // Stream terminated
            if (line.length == 3 && line[0] == 46) {
                out.flush();
                out.close();
               
                List recipientCollection = (List) session.getState().get(SMTPSession.RCPT_LIST);
                MailImpl mail =
                    new MailImpl(mailServer.getId(),
                                 (MailAddress) session.getState().get(SMTPSession.SENDER),
                                 recipientCollection);
               
                // store mail in the session so we can be sure it get disposed later
                session.getState().put(SMTPConstants.MAIL, mail);
               
                MimeMessageCopyOnWriteProxy mimeMessageCopyOnWriteProxy = null;
                try {
                    mimeMessageCopyOnWriteProxy = new MimeMessageCopyOnWriteProxy(mmiss);
                    mail.setMessage(mimeMessageCopyOnWriteProxy);
                   
                    processExtensions(session, mail);
                   
                    session.popLineHandler();
                    //next.onLine(session, line);
   
                } catch (MessagingException e) {
                    // TODO probably return a temporary problem
                    session.getLogger().info("Unexpected error handling DATA stream",e);
                    session.writeResponse(new SMTPResponse(SMTPRetCode.LOCAL_ERROR, "Unexpected error handling DATA stream."));
                } finally {
                    LifecycleUtil.dispose(mimeMessageCopyOnWriteProxy);
                    LifecycleUtil.dispose(mmiss);
                    LifecycleUtil.dispose(mail);
                }
   
               
            // DotStuffing.
            } else if (line[0] == 46 && line[1] == 46) {
                out.write(line,1,line.length-1);
            // Standard write
            } else {
                // TODO: maybe we should handle the Header/Body recognition here
                // and if needed let a filter to cache the headers to apply some
                // transformation before writing them to output.
                out.write(line);
            }
            out.flush();
        } catch (IOException e) {
            LifecycleUtil.dispose(mmiss);

            SMTPResponse response;
            response = new SMTPResponse(SMTPRetCode.LOCAL_ERROR,DSNStatus.getStatus(DSNStatus.TRANSIENT,
                            DSNStatus.UNDEFINED_STATUS) + " Error processing message: " + e.getMessage());
           
            session.getLogger().error(
                    "Unknown error occurred while processing DATA.", e);
            session.writeResponse(response);
            return;
       
    }

  /**
   * @param session
   */
  private void processExtensions(SMTPSession session, Mail mail) {
    if (mail != null && messageHandlers != null) {
      try {
        for (int i = 0; i < mHandlers.size(); i++) {
          MessageHook rawHandler = mHandlers.get(i);
          session.getLogger().debug(
              "executing james message handler " + rawHandler);
          HookResult hRes = rawHandler.onMessage(session,
              new MailToMailEnvelopeWrapper(mail));

          if (rHooks != null) {
            for (int i2 = 0; i2 < rHooks.size(); i2++) {
              Object rHook = rHooks.get(i2);
              session.getLogger()
                  .debug("executing hook " + rHook);
              hRes = ((HookResultHook) rHook).onHookResult(
                  session, hRes, rawHandler);
            }
          }

          SMTPResponse response = AbstractHookableCmdHandler
              .calcDefaultSMTPResponse(hRes);

          // if the response is received, stop processing of command
          // handlers
          if (response != null) {
            session.writeResponse(response);
            return;
          }
        }

        int count = messageHandlers.size();
        for (int i = 0; i < count; i++) {
          Object rawHandler = messageHandlers.get(i);
          session.getLogger().debug(
              "executing james message handler " + rawHandler);
          HookResult hRes = ((JamesMessageHook) rawHandler)
              .onMessage(session, (Mail) mail);

          if (rHooks != null) {
            for (int i2 = 0; i2 < rHooks.size(); i2++) {
              Object rHook = rHooks.get(i2);
              session.getLogger()
                  .debug("executing hook " + rHook);
              hRes = ((HookResultHook) rHook).onHookResult(
                  session, hRes, rawHandler);
            }
          }

          SMTPResponse response = AbstractHookableCmdHandler
              .calcDefaultSMTPResponse(hRes);

          // if the response is received, stop processing of command
          // handlers
          if (response != null) {
            session.writeResponse(response);
            break;
          }
        }
      } finally {
        // Dispose the mail object and remove it
        if (mail != null) {
            LifecycleUtil.dispose(mail);
          mail = null;
        }
        // do the clean up
        session.resetState();
      }
    }
  }

   
    /**
     * @see org.apache.james.api.protocol.ExtensibleHandler#wireExtensions(java.lang.Class, java.util.List)
     */
    public void wireExtensions(Class interfaceName, List extension) throws WiringException {
        if (JamesMessageHook.class.equals(interfaceName)) {
            this.messageHandlers = extension;
            if (messageHandlers.size() == 0) {
                if (serviceLog.isErrorEnabled()) {
                    serviceLog.error("No messageHandler configured. Check that SendMailHandler is configured in the SMTPHandlerChain");
                }
                throw new WiringException("No messageHandler configured");
            }
        } else if (MessageHook.class.equals(interfaceName)) {
          this.mHandlers = extension;
        } else if (HookResultHook.class.equals(interfaceName)) {

            this.rHooks = extension;
        }
    }

    /**
     * @see org.apache.james.api.protocol.ExtensibleHandler#getMarkerInterfaces()
     */
    public List<Class<?>> getMarkerInterfaces() {
        List<Class<?>> classes = new LinkedList<Class<?>>();
        classes.add(JamesMessageHook.class);
        classes.add(MessageHook.class);
        classes.add(HookResultHook.class);
        return classes;
    }

    /**
     * Sets the service log.
     * Where available, a context sensitive log should be used.
     * @param Log not null
     */
    public void setLog(Log log) {
        this.serviceLog = log;
    }
   
    private class MailToMailEnvelopeWrapper implements MailEnvelope {
      private Mail mail;
      public MailToMailEnvelopeWrapper(Mail mail) {
        this.mail = mail;
      }
     
      /**
       * @see org.apache.james.protocols.smtp.MailEnvelope#getMessageInputStream()
       */
    public InputStream getMessageInputStream() throws Exception {
      return mail.getMessage().getInputStream();
    }
   
    /**
     * Return just null. Not sure if this is a good idea ..
     */
    public OutputStream getMessageOutputStream() {
      return null;
    }

    /**
     * @see org.apache.james.protocols.smtp.MailEnvelope#getRecipients()
     */
    public List<MailAddress> getRecipients() {
      return new ArrayList<MailAddress>(mail.getRecipients());
    }

    /**
     * (non-Javadoc)
     * @see org.apache.james.protocols.smtp.MailEnvelope#getSender()
     */
    public MailAddress getSender() {
      return mail.getSender();
    }
   
    /**
     * @see org.apache.james.protocols.smtp.MailEnvelope#getSize()
     */
    public int getSize() {
      try {
        return new Long(mail.getMessageSize()).intValue();
      } catch (MessagingException e) {
        return -1;
      }
    }

    /**
     * (non-Javadoc)
     * @see org.apache.james.protocols.smtp.MailEnvelope#setRecipients(java.util.List)
     */
    public void setRecipients(List<MailAddress> recipientCollection) {
      mail.setRecipients(recipientCollection);
    }
     
    }
}
TOP

Related Classes of org.apache.james.smtpserver.DataLineJamesMessageHookHandler$MailToMailEnvelopeWrapper

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.