All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.vertx.ext.mail.impl.SMTPConnection Maven / Gradle / Ivy

There is a newer version: 5.0.0.CR3
Show newest version
/*
 *  Copyright (c) 2011-2015 The original author or authors
 *
 *  All rights reserved. This program and the accompanying materials
 *  are made available under the terms of the Eclipse Public License v1.0
 *  and Apache License v2.0 which accompanies this distribution.
 *
 *       The Eclipse Public License is available at
 *       http://www.eclipse.org/legal/epl-v10.html
 *
 *       The Apache License v2.0 is available at
 *       http://www.opensource.org/licenses/apache2.0.php
 *
 *  You may elect to redistribute this code under either of these licenses.
 */

package io.vertx.ext.mail.impl;

import io.vertx.core.Context;
import io.vertx.core.Handler;
import io.vertx.core.Vertx;
import io.vertx.core.buffer.Buffer;
import io.vertx.core.impl.NoStackTraceThrowable;
import io.vertx.core.logging.Logger;
import io.vertx.core.logging.LoggerFactory;
import io.vertx.core.net.NetClient;
import io.vertx.core.net.NetSocket;
import io.vertx.ext.mail.MailConfig;

/**
 * SMTP connection to a server.
 * 

* Encapsulate the NetSocket connection and the data writing/reading * * @author Alexander Lehmann */ class SMTPConnection { private static final Logger log = LoggerFactory.getLogger(SMTPConnection.class); private final Vertx vertx; private NetSocket ns; private boolean socketClosed; private boolean socketShutDown; private Handler commandReplyHandler; private Handler errorHandler; private boolean broken; private boolean idle; private boolean doShutdown; private final NetClient client; private Capabilities capa = new Capabilities(); private final ConnectionLifeCycleListener listener; private Context context; SMTPConnection(Vertx vertx, NetClient client, ConnectionLifeCycleListener listener) { broken = true; idle = false; doShutdown = false; socketClosed = false; socketShutDown = false; this.client = client; this.listener = listener; this.vertx = vertx; } /** * @return the capabilities object */ Capabilities getCapa() { return capa; } /** * parse capabilities from the ehlo reply string * * @param message the capabilities to set */ void parseCapabilities(String message) { capa = new Capabilities(); capa.parseCapabilities(message); } void shutdown() { broken = true; commandReplyHandler = null; socketShutDown = true; if (ns != null) { ns.close(); ns = null; } } /* * write command without masking anything */ void write(String str, Handler commandResultHandler) { write(str, -1, commandResultHandler); } /* * write command masking everything after position blank */ void write(String str, int blank, Handler commandResultHandler) { this.commandReplyHandler = commandResultHandler; if (socketClosed) { log.debug("connection was closed by server"); handleError("connection was closed by server"); } else { if (ns != null) { if (log.isDebugEnabled()) { String logStr; if (blank >= 0) { StringBuilder sb = new StringBuilder(); for (int i = blank; i < str.length(); i++) { sb.append('*'); } logStr = str.substring(0, blank) + sb; } else { logStr = str; } // avoid logging large mail body if (logStr.length() < 1000) { log.debug("command: " + logStr); } else { log.debug("command: " + logStr.substring(0, 1000) + "..."); } } ns.write(str + "\r\n"); } else { log.debug("not sending command " + str + " since the netsocket is null"); } } } // write single line not expecting a reply void writeLine(String str, boolean mayLog) { if (mayLog) { log.debug(str); } ns.write(str + "\r\n"); } // write single line not expecting a reply, using drain handler void writeLineWithDrainHandler(String str, boolean mayLog, Handler handler) { if (mayLog) { log.debug(str); } if (ns.writeQueueFull()) { ns.drainHandler(v -> { // avoid getting confused by being called twice ns.drainHandler(null); ns.write(str + "\r\n"); handler.handle(null); }); } else { ns.write(str + "\r\n"); handler.handle(null); } } boolean writeQueueFull() { return ns.writeQueueFull(); } private void handleError(String message) { handleError(new NoStackTraceThrowable(message)); } private void handleError(Throwable throwable) { errorHandler.handle(throwable); } public void openConnection(MailConfig config, Handler initialReplyHandler, Handler errorHandler) { this.errorHandler = errorHandler; broken = false; idle = false; client.connect(config.getPort(), config.getHostname(), asyncResult -> { if (asyncResult.succeeded()) { context = Vertx.currentContext(); ns = asyncResult.result(); socketClosed = false; ns.exceptionHandler(e -> { // avoid returning two exceptions log.debug("exceptionHandler called"); if (!socketClosed && !socketShutDown && !idle && !broken) { setBroken(); log.debug("got an exception on the netsocket", e); handleError(e); } else { log.debug("not returning follow-up exception", e); } }); ns.closeHandler(v -> { log.debug("socket has been closed"); listener.connectionClosed(this); socketClosed = true; // avoid exception if we regularly shut down the socket on our side if (!socketShutDown && !idle && !broken) { setBroken(); log.debug("throwing: connection has been closed by the server"); handleError("connection has been closed by the server"); } else { if (socketShutDown || broken) { log.debug("close has been expected"); } else { log.debug("closed while connection has been idle (timeout on server?)"); } if (!broken) { setBroken(); } if (!socketShutDown) { shutdown(); listener.dataEnded(this); } } }); commandReplyHandler = initialReplyHandler; final Handler mlp = new MultilineParser(buffer -> { if (commandReplyHandler == null) { log.debug("dropping reply arriving after we stopped processing \"" + buffer.toString() + "\""); } else { // make sure we only call the handler once Handler currentHandler = commandReplyHandler; commandReplyHandler = null; currentHandler.handle(buffer.toString()); } }); ns.handler(mlp); } else { log.error("exception on connect", asyncResult.cause()); // notify the pool that the connection attempt didn't work so that the connection count is correct listener.connectionClosed(null); handleError(asyncResult.cause()); } }); } boolean isSsl() { return ns.isSsl(); } void upgradeToSsl(Handler handler) { ns.upgradeToSsl(handler); } public boolean isBroken() { return broken; } public boolean isIdle() { return idle; } public void returnToPool() { if (isIdle()) { log.info("state error: idle connection returned to pool"); handleError("state error: idle connection returned to pool"); } else { if (doShutdown) { log.debug("shutting connection down"); quitCloseConnection(); } else { log.debug("returning connection to pool"); commandReplyHandler = null; listener.dataEnded(this); log.debug("setting error handler to null"); errorHandler = null; } } } /** * send QUIT and close the connection, this operation waits for the success of the quit command but will close the * connection on exception as well */ void quitCloseConnection() { if (!socketShutDown) { context.runOnContext(v1 -> { log.debug("shutting down connection"); if (socketClosed) { log.debug("connection is already closed, only doing shutdown()"); shutdown(); } else { // set the connection to in use to avoid it being used by another getConnection operation useConnection(); new SMTPQuit(this, v -> { shutdown(); log.debug("connection is shut down"); }).start(); } }); } } /** * mark a connection as being used again */ void useConnection() { idle = false; } /** * mark a connection as free */ void setIdle() { idle = true; } /** * set error handler to a "local" handler to be reset later */ private Handler prevErrorHandler = null; public void setErrorHandler(Handler newHandler) { if (prevErrorHandler == null) { prevErrorHandler = errorHandler; } errorHandler = newHandler; } /** * reset error handler to previous */ public void resetErrorHandler() { errorHandler = prevErrorHandler; } /** * set connection to broken and shut it down */ public void setBroken() { if (!broken) { log.debug("setting connection to broken"); broken = true; commandReplyHandler = null; log.debug("closing connection"); shutdown(); listener.dataEnded(this); } else { log.debug("connection is already set to broken"); } } /** * if connection is still active, shut it down when the current * operation has finished */ public void setDoShutdown() { log.debug("will shut down connection after send operation finishes"); doShutdown = true; } /** * close the connection doing a QUIT command first */ public void close() { quitCloseConnection(); } /** * check if a connection is already closed (this is mostly for unit tests) */ boolean isClosed() { return socketClosed; } /** * get the context associated with this connection * * @return */ Context getContext() { return context; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy