diff --git a/apps/susimail/src/src/i2p/susi/webmail/WebMail.java b/apps/susimail/src/src/i2p/susi/webmail/WebMail.java index 06f4871c4684dd5d5ce7d7a1a92a3072612b4f72..db022e6ffbab2dd61eca89ba49d5f2def9781eb3 100644 --- a/apps/susimail/src/src/i2p/susi/webmail/WebMail.java +++ b/apps/susimail/src/src/i2p/susi/webmail/WebMail.java @@ -415,10 +415,10 @@ public class WebMail extends HttpServlet } if( mailPart.multipart ) { - if( mailPart.type.compareTo( "multipart/alternative" ) == 0 ) { + if( mailPart.type.equals("multipart/alternative")) { MailPart chosen = null; for( MailPart subPart : mailPart.parts ) { - if( subPart.type != null && subPart.type.compareTo( "text/plain" ) == 0 ) + if( subPart.type != null && subPart.type.equals("text/plain")) chosen = subPart; } if( chosen != null ) { @@ -454,7 +454,7 @@ public class WebMail extends HttpServlet showBody = true; } if( showBody == false && mailPart.type != null ) { - if( mailPart.type.compareTo( "text/plain" ) == 0 ) { + if( mailPart.type.equals("text/plain")) { showBody = true; } else @@ -559,7 +559,7 @@ public class WebMail extends HttpServlet String pop3Port = request.getParameter( POP3 ); String smtpPort = request.getParameter( SMTP ); String fixedPorts = Config.getProperty( CONFIG_PORTS_FIXED, "true" ); - if( fixedPorts.compareToIgnoreCase( "false" ) != 0 ) { + if( !fixedPorts.equalsIgnoreCase("false")) { host = Config.getProperty( CONFIG_HOST, DEFAULT_HOST ); pop3Port = Config.getProperty( CONFIG_PORTS_POP3, "" + DEFAULT_POP3PORT ); smtpPort = Config.getProperty( CONFIG_PORTS_SMTP, "" + DEFAULT_SMTPPORT ); @@ -885,7 +885,7 @@ public class WebMail extends HttpServlet private static int getCheckedMessage(RequestWrapper request) { for( Enumeration<String> e = request.getParameterNames(); e.hasMoreElements(); ) { String parameter = e.nextElement(); - if( parameter.startsWith( "check" ) && request.getParameter( parameter ).compareTo( "1" ) == 0 ) { + if( parameter.startsWith( "check" ) && request.getParameter( parameter ).equals("1")) { String number = parameter.substring( 5 ); try { int n = Integer.parseInt( number ); @@ -972,7 +972,7 @@ public class WebMail extends HttpServlet else if( sessionObject.attachments != null && buttonPressed( request, DELETE_ATTACHMENT ) ) { for( Enumeration<String> e = request.getParameterNames(); e.hasMoreElements(); ) { String parameter = e.nextElement(); - if( parameter.startsWith( "check" ) && request.getParameter( parameter ).compareTo( "1" ) == 0 ) { + if( parameter.startsWith( "check" ) && request.getParameter( parameter ).equals("1")) { String number = parameter.substring( 5 ); try { int n = Integer.parseInt( number ); @@ -1119,7 +1119,7 @@ public class WebMail extends HttpServlet if( buttonPressed( request, REALLYDELETE ) ) { for( Enumeration<String> e = request.getParameterNames(); e.hasMoreElements(); ) { String parameter = e.nextElement(); - if( parameter.startsWith( "check" ) && request.getParameter( parameter ).compareTo( "1" ) == 0 ) { + if( parameter.startsWith( "check" ) && request.getParameter( parameter ).equals("1")) { String number = parameter.substring( 5 ); try { int n = Integer.parseInt( number ); @@ -1170,11 +1170,11 @@ public class WebMail extends HttpServlet { String str = request.getParameter( sort_id ); if( str != null ) { - if( str.compareToIgnoreCase( "up" ) == 0 ) { + if( str.equalsIgnoreCase("up")) { sessionObject.folder.setSortingDirection( Folder.UP ); sessionObject.folder.sortBy( sort_id ); } - if( str.compareToIgnoreCase( "down" ) == 0 ) { + if( str.equalsIgnoreCase("down")) { sessionObject.folder.setSortingDirection( Folder.DOWN ); sessionObject.folder.sortBy( sort_id ); } @@ -1471,7 +1471,7 @@ public class WebMail extends HttpServlet String prop = Config.getProperty( CONFIG_SENDER_FIXED, "true" ); String domain = Config.getProperty( CONFIG_SENDER_DOMAIN, "mail.i2p" ); - if( prop.compareToIgnoreCase( "false" ) != 0 ) { + if( !prop.equalsIgnoreCase("false")) { from = "<" + sessionObject.user + "@" + domain + ">"; } ArrayList<String> toList = new ArrayList<String>(); @@ -1503,7 +1503,7 @@ public class WebMail extends HttpServlet String bccToSelf = request.getParameter( NEW_BCC_TO_SELF ); - if( bccToSelf != null && bccToSelf.compareTo( "1" ) == 0 ) + if( bccToSelf != null && bccToSelf.equals("1")) recipients.add( sender ); if( toList.isEmpty() ) { @@ -1619,7 +1619,7 @@ public class WebMail extends HttpServlet String from = request.getParameter( NEW_FROM ); String fixed = Config.getProperty( CONFIG_SENDER_FIXED, "true" ); - if( from == null || fixed.compareToIgnoreCase( "false" ) != 0 ) { + if( from == null || !fixed.equalsIgnoreCase("false")) { String domain = Config.getProperty( CONFIG_SENDER_DOMAIN, "mail.i2p" ); from = "<" + sessionObject.user + "@" + domain + ">"; } @@ -1637,12 +1637,12 @@ public class WebMail extends HttpServlet out.println( "<table cellspacing=\"0\" cellpadding=\"5\">\n" + "<tr><td colspan=\"2\" align=\"center\"><hr></td></tr>\n" + - "<tr><td align=\"right\">" + _("From:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_FROM + "\" value=\"" + from + "\" " + ( fixed.compareToIgnoreCase( "false" ) != 0 ? "disabled" : "" ) +"></td></tr>\n" + + "<tr><td align=\"right\">" + _("From:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_FROM + "\" value=\"" + from + "\" " + ( !fixed.equalsIgnoreCase("false") ? "disabled" : "" ) +"></td></tr>\n" + "<tr><td align=\"right\">" + _("To:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_TO + "\" value=\"" + to + "\"></td></tr>\n" + "<tr><td align=\"right\">" + _("Cc:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_CC + "\" value=\"" + cc + "\"></td></tr>\n" + "<tr><td align=\"right\">" + _("Bcc:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_BCC + "\" value=\"" + bcc + "\"></td></tr>\n" + "<tr><td align=\"right\">" + _("Subject:") + "</td><td align=\"left\"><input type=\"text\" size=\"80\" name=\"" + NEW_SUBJECT + "\" value=\"" + subject + "\"></td></tr>\n" + - "<tr><td> </td><td align=\"left\"><input type=\"checkbox\" class=\"optbox\" name=\"" + NEW_BCC_TO_SELF + "\" value=\"1\"" + ( bccToSelf.compareToIgnoreCase( "false" ) != 0 ? "checked" : "" )+ ">" + _("Bcc to self") + "</td></tr>\n" + + "<tr><td> </td><td align=\"left\"><input type=\"checkbox\" class=\"optbox\" name=\"" + NEW_BCC_TO_SELF + "\" value=\"1\"" + ( !bccToSelf.equalsIgnoreCase("false") ? "checked" : "" )+ ">" + _("Bcc to self") + "</td></tr>\n" + "<tr><td colspan=\"2\" align=\"center\"><textarea cols=\"" + Config.getProperty( CONFIG_COMPOSER_COLS, 80 )+ "\" rows=\"" + Config.getProperty( CONFIG_COMPOSER_ROWS, 10 )+ "\" name=\"" + NEW_TEXT + "\">" + text + "</textarea>" + "<tr><td colspan=\"2\" align=\"center\"><hr></td></tr>\n" + "<tr><td align=\"right\">" + _("New Attachment:") + "</td><td align=\"left\"><input type=\"file\" size=\"50%\" name=\"" + NEW_FILENAME + "\" value=\"\"><input type=\"submit\" name=\"" + NEW_UPLOAD + "\" value=\"" + _("Upload File") + "\"></td></tr>" ); @@ -1668,7 +1668,7 @@ public class WebMail extends HttpServlet private static void showLogin( PrintWriter out ) { String fixedPorts = Config.getProperty( CONFIG_PORTS_FIXED, "true" ); - boolean fixed = fixedPorts.compareToIgnoreCase( "false" ) != 0; + boolean fixed = !fixedPorts.equalsIgnoreCase("false"); String host = Config.getProperty( CONFIG_HOST, DEFAULT_HOST ); String pop3 = Config.getProperty( CONFIG_PORTS_POP3, "" + DEFAULT_POP3PORT ); String smtp = Config.getProperty( CONFIG_PORTS_SMTP, "" + DEFAULT_SMTPPORT ); @@ -1747,7 +1747,7 @@ public class WebMail extends HttpServlet boolean idChecked = false; String checkId = sessionObject.pageChanged ? null : request.getParameter( "check" + i ); - if( checkId != null && checkId.compareTo( "1" ) == 0 ) + if( checkId != null && checkId.equals("1")) idChecked = true; if( sessionObject.markAll ) @@ -1765,7 +1765,7 @@ public class WebMail extends HttpServlet ( idChecked ? "checked" : "" ) + ">" + "</td><td>" + link + mail.shortSender + "</a></td><td> </td><td>" + link + mail.shortSubject + "</a></td><td> </td><td>" + // don't let date get split across lines - mail.localFormattedDate.replace(" ", " ") + "</td><td> </td><td>" + + mail.localFormattedDate.replace(" ", " ") + "</td><td> </td><td align=\"right\">" + DataHelper.formatSize2(mail.size) + "B</td></tr>" ); bg = 1 - bg; i++; diff --git a/apps/susimail/src/src/i2p/susi/webmail/smtp/SMTPClient.java b/apps/susimail/src/src/i2p/susi/webmail/smtp/SMTPClient.java index c6403f04517e5cf19ca5879480d5bbdd7fe0a941..ea21b2b4a9272029b365f0b4ec0e2fe8aff14a1f 100644 --- a/apps/susimail/src/src/i2p/susi/webmail/smtp/SMTPClient.java +++ b/apps/susimail/src/src/i2p/susi/webmail/smtp/SMTPClient.java @@ -24,6 +24,7 @@ package i2p.susi.webmail.smtp; import i2p.susi.debug.Debug; +import i2p.susi.webmail.Messages; import i2p.susi.webmail.encoding.Encoding; import i2p.susi.webmail.encoding.EncodingException; import i2p.susi.webmail.encoding.EncodingFactory; @@ -32,6 +33,10 @@ import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.net.Socket; +import java.util.ArrayList; +import java.util.List; + +import net.i2p.data.DataHelper; /** * @author susi @@ -42,6 +47,7 @@ public class SMTPClient { private final byte buffer[]; public String error; private String lastResponse; + private boolean supportsPipelining; private static final Encoding base64; @@ -72,61 +78,126 @@ public class SMTPClient { * @since 0.9.13 */ private int sendCmd(String cmd, boolean shouldWait) + { + if( socket == null ) + return 0; + try { + if (cmd != null) + sendCmdNoWait(cmd); + if (!shouldWait) + return 100; + socket.getOutputStream().flush(); + return getResult(); + } catch (IOException e) { + error += "IOException occured.<br>"; + return 0; + } + } + + /** + * Does not flush, wait, or read + * + * @param cmd non-null + * @since 0.9.13 + */ + private void sendCmdNoWait(String cmd) throws IOException { Debug.debug( Debug.DEBUG, "SMTP sendCmd(" + cmd +")" ); if( socket == null ) - return 0; + throw new IOException("no socket"); + OutputStream out = socket.getOutputStream(); + cmd += "\r\n"; + out.write( cmd.getBytes() ); + } + + /** + * Pipeline if supported + * + * @param cmds non-null + * @return number of successful commands + * @since 0.9.13 + */ + private int sendCmds(List<SendExpect> cmds) + { + int rv = 0; + if (supportsPipelining) { + Debug.debug(Debug.DEBUG, "SMTP pipelining " + cmds.size() + " commands"); + try { + for (SendExpect cmd : cmds) { + sendCmdNoWait(cmd.send); + } + socket.getOutputStream().flush(); + } catch (IOException ioe) { + return 0; + } + for (SendExpect cmd : cmds) { + int r = getResult(); + // stop only on EOF + if (r == 0) + break; + if (r == cmd.expect) + rv++; + } + } else { + for (SendExpect cmd : cmds) { + int r = sendCmd(cmd.send); + // stop at first error + if (r != cmd.expect) + break; + rv++; + } + } + Debug.debug(Debug.DEBUG, "SMTP success in " + rv + " of " + cmds.size() + " commands"); + return rv; + } + /** + * @return result code or 0 for failure + * @since 0.9.13 + */ + private int getResult() { + return getFullResult().result; + } + + /** + * @return result code and string, all lines combined with \r separators, + * first 3 bytes are the ASCII return code or "000" for failure + * Result and Result.recv non null + * @since 0.9.13 + */ + private Result getFullResult() { int result = 0; - lastResponse = ""; - + StringBuilder fullResponse = new StringBuilder(512); try { InputStream in = socket.getInputStream(); - OutputStream out = socket.getOutputStream(); - - if( cmd != null ) { - cmd += "\r\n"; - out.write( cmd.getBytes() ); - } - if (!shouldWait) - return 100; - out.flush(); - String str = ""; - boolean doContinue = true; - while( doContinue ) { - if( in.available() > 0 ) { - int read = in.read( buffer ); - str += new String( buffer, 0, read ); - lastResponse += str; - while( true ) { - int i = str.indexOf( "\r\n" ); - if( i == -1 ) - break; - if( result == 0 ) { - try { - result = Integer.parseInt( str.substring( 0, 3 ) ); - } - catch( NumberFormatException nfe ) { - result = 0; - doContinue = false; - break; - } - } - if( str.substring( 3, 4 ).compareTo( " " ) == 0 ) { - doContinue = false; - break; - } - str = str.substring( i + 2 ); + StringBuilder buf = new StringBuilder(128); + while (DataHelper.readLine(in, buf)) { + Debug.debug(Debug.DEBUG, "SMTP rcv \"" + buf.toString().trim() + '"'); + int len = buf.length(); + if (len < 4) { + result = 0; + break; // huh? no nnn\r? + } + if( result == 0 ) { + try { + String r = buf.substring(0, 3); + result = Integer.parseInt(r); + } catch ( NumberFormatException nfe ) { + break; } } + fullResponse.append(buf.substring(4)); + if (buf.charAt(3) == ' ') + break; + buf.setLength(0); } - } - catch (IOException e) { + } catch (IOException e) { error += "IOException occured.<br>"; result = 0; } - return result; + lastResponse = fullResponse.toString(); + return new Result(result, lastResponse); } /** @@ -139,60 +210,118 @@ public class SMTPClient { try { socket = new Socket( host, port ); - } - catch (Exception e) { - error += "Cannot connect: " + e.getMessage() + "<br>"; + } catch (Exception e) { + error += _("Cannot connect") + ": " + e.getMessage() + "<br>"; ok = false; } try { - if( ok && sendCmd( null ) == 220 && - sendCmd( "EHLO localhost" ) == 250 && - sendCmd( "AUTH LOGIN" ) == 334 && - sendCmd( base64.encode( user ) ) == 334 && - sendCmd( base64.encode( pass ) ) == 235 && - sendCmd( "MAIL FROM: " + sender ) == 250 ) { - + // SMTP ref: RFC 821 + // Pipelining ref: RFC 2920 + // AUTH ref: RFC 4954 + if (ok) { + int result = sendCmd(null); + if (result != 220) { + error += _("Server refused connection") + " (" + result + ")<br>"; + ok = false; + } + } + if (ok) { + sendCmdNoWait( "EHLO localhost" ); + socket.getOutputStream().flush(); + Result r = getFullResult(); + if (r.result == 250) { + supportsPipelining = r.recv.contains("PIPELINING"); + } else { + error += _("Server refused connection") + " (" + r.result + ")<br>"; + ok = false; + } + } + if (ok) { + // RFC 4954 says AUTH must be the last but let's assume + // that includes the user/pass on following lines + List<SendExpect> cmds = new ArrayList<SendExpect>(); + cmds.add(new SendExpect("AUTH LOGIN", 334)); + cmds.add(new SendExpect(base64.encode(user), 334)); + cmds.add(new SendExpect(base64.encode(pass), 235)); + if (sendCmds(cmds) != 3) { + error += _("Login failed") + "<br>"; + ok = false; + } + } + if (ok) { + List<SendExpect> cmds = new ArrayList<SendExpect>(); + cmds.add(new SendExpect("MAIL FROM: " + sender, 250)); for( int i = 0; i < recipients.length; i++ ) { - if( sendCmd( "RCPT TO: " + recipients[i] ) != 250 ) { - ok = false; - } + cmds.add(new SendExpect("RCPT TO: " + recipients[i], 250)); } - if( ok ) { - if( sendCmd( "DATA" ) == 354 ) { - if( body.indexOf( "\r\n.\r\n" ) != -1 ) - body = body.replaceAll( "\r\n.\r\n", "\r\n..\r\n" ); - body += "\r\n.\r\n"; - try { - socket.getOutputStream().write( body.getBytes() ); - if( sendCmd( null ) == 250 ) { - mailSent = true; - } - } - catch (Exception e) { - ok = false; - error += "Error while sending mail: " + e.getMessage() + "<br>"; - } - } + cmds.add(new SendExpect("DATA", 354)); + if (sendCmds(cmds) != cmds.size()) { + // TODO which recipient? + error += _("Mail rejected") + "<br>"; + ok = false; } } + if (ok) { + if( body.indexOf( "\r\n.\r\n" ) != -1 ) + body = body.replaceAll( "\r\n.\r\n", "\r\n..\r\n" ); + socket.getOutputStream().write( body.getBytes() ); + socket.getOutputStream().write("\r\n.\r\n".getBytes() ); + int result = sendCmd(null); + if (result == 250) + mailSent = true; + else + error += _("Error sending mail") + " (" + result + ")<br>"; + } + } catch (IOException e) { + error += _("Error sending mail") + ": " + e.getMessage() + "<br>"; + } catch (EncodingException e) { - ok = false; error += e.getMessage(); } if( !mailSent && lastResponse.length() > 0 ) { - String[] lines = lastResponse.split( "\r\n" ); + String[] lines = lastResponse.split( "\r" ); for( int i = 0; i < lines.length; i++ ) error += lines[i] + "<br>"; } - sendCmd("QUIT", false ); + sendCmd("QUIT", false); if( socket != null ) { try { socket.close(); - } - catch (IOException e1) { - // ignore - } + } catch (IOException e1) {} } return mailSent; } + + /** + * A command to send and a result code to expect + * @since 0.9.13 + */ + private static class SendExpect { + public final String send; + public final int expect; + + public SendExpect(String s, int e) { + send = s; + expect = e; + } + } + + /** + * A result string and code + * @since 0.9.13 + */ + private static class Result { + public final int result; + public final String recv; + + public Result(int r, String t) { + result = r; + recv = t; + } + } + + /** translate */ + private static String _(String s) { + return Messages.getString(s); + } }