package com.planet_ink.coffee_mud.core.smtp; import com.planet_ink.coffee_mud.core.interfaces.*; import com.planet_ink.coffee_mud.core.*; import com.planet_ink.coffee_mud.core.collections.*; import com.planet_ink.coffee_mud.Abilities.interfaces.*; import com.planet_ink.coffee_mud.Areas.interfaces.*; import com.planet_ink.coffee_mud.Behaviors.interfaces.*; import com.planet_ink.coffee_mud.CharClasses.interfaces.*; import com.planet_ink.coffee_mud.Commands.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.*; import com.planet_ink.coffee_mud.Exits.interfaces.*; import com.planet_ink.coffee_mud.Items.interfaces.*; import com.planet_ink.coffee_mud.Libraries.interfaces.JournalsLibrary; import com.planet_ink.coffee_mud.Libraries.interfaces.PlayerLibrary.ThinPlayer; import com.planet_ink.coffee_mud.Locales.interfaces.*; import com.planet_ink.coffee_mud.MOBS.interfaces.*; import com.planet_ink.coffee_mud.Races.interfaces.*; import java.net.*; import java.util.*; import java.util.concurrent.atomic.AtomicInteger; import com.planet_ink.coffee_mud.core.exceptions.*; import java.io.*; /* Copyright 2004-2016 Bo Zimmerman Licensed 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. */ public class ProcessSMTPrequest implements Runnable { private final static String cr = "\r\n"; private final static String S_250 = "250 OK"; private final static long IDLE_TIMEOUT=10000; private static volatile AtomicInteger instanceCnt = new AtomicInteger(0); private Socket sock; private SMTPserver server=null; private StringBuffer data=null; protected String from=null; protected MOB fromM=null; protected String domain=null; protected String runnableName; protected boolean debug=false; protected Vector<String> to=null; public ProcessSMTPrequest(Socket a_sock, SMTPserver a_Server) { runnableName="SMTPrq"+(instanceCnt.addAndGet(1)); server = a_Server; sock = a_sock; } public String validLocalAccount(String s, boolean checkFROMcase) { final int x=s.indexOf('@'); String name=s; if(x>0) { name=s.substring(0,x).trim(); final String domain=s.substring(x+1).trim(); if(!domain.toUpperCase().endsWith(server.domainName().toUpperCase())) { if(server.mailboxName().length()>0) { name=CMLib.database().DBPlayerEmailSearch(s); if(name!=null) return name; if(checkFROMcase) // accounts cannot receive emails { final PlayerAccount A=CMLib.players().getLoadAccountByEmail(s); if(A!=null) return A.getAccountName(); } } return null; } } if(server.getAnEmailJournal(name)!=null) return server.getAnEmailJournal(name); if(server.mailboxName().length()>0) { if(CMLib.players().playerExists(name)) return CMStrings.capitalizeAndLower(name); if(checkFROMcase) // accounts cannot receive emails { if(CMLib.players().accountExists(name)) return CMStrings.capitalizeAndLower(name); } } return null; } public MOB getAccountMob(String s) { MOB M=CMLib.players().getPlayer(s); if(M!=null) return M; if(CMLib.players().playerExists(s)) return CMLib.players().getLoadPlayer(s); if(CMLib.players().accountExists(s)) { final PlayerAccount A=CMLib.players().getLoadAccount(s); if(A.numPlayers()==0) M=A.getAccountMob(); else { ThinPlayer tP=null; for(final Enumeration<ThinPlayer> e=A.getThinPlayers();e.hasMoreElements();) { final ThinPlayer P=e.nextElement(); if((tP==null)||(P.level()>tP.level())) tP=P; } if(tP!=null) M=CMLib.players().getLoadPlayer(tP.name()); } return M; } return null; } public void cleanHtml(String journal, StringBuilder finalData) { if(journal!= null) { // the input MUST be html -- text that only might be need not apply final JournalsLibrary.ForumJournal forum=CMLib.journals().getForumJournal(journal); if(forum!=null) CMStrings.stripHeadHtmlTags(finalData); else CMStrings.convertHtmlToText(finalData); } else CMStrings.convertHtmlToText(finalData); } @Override public void run() { BufferedReader sin = null; PrintWriter sout = null; int failures=0; debug = CMSecurity.isDebugging(CMSecurity.DbgFlag.SMTPSERVER); byte[] replyData = null; final byte[] lastReplyData = null; int msgsSent = 0; try { sock.setSoTimeout(100); sout=new PrintWriter(new BufferedWriter(new OutputStreamWriter(sock.getOutputStream(),"US-ASCII"))); sin=new BufferedReader(new InputStreamReader(sock.getInputStream(),"US-ASCII")); final String initialMsg = "220 ESMTP "+server.domainName()+" "+SMTPserver.ServerVersionString+"; "+CMLib.time().date2String(System.currentTimeMillis()); if(debug) Log.debugOut(runnableName,"Sent: "+initialMsg); sout.write(initialMsg+cr); sout.flush(); boolean quitFlag=false; boolean dataMode=false; final LinkedList<String> cmdQueue=new LinkedList<String>(); final LinkedList<byte[]> respQueue=new LinkedList<byte[]>(); long timeSinceLastChar=System.currentTimeMillis(); while(!quitFlag) { char lastc=(char)-1; char c=(char)-1; final StringBuffer input=new StringBuffer(""); while(!quitFlag) { lastc=c; try { c=(char)sin.read(); } catch(final java.net.SocketTimeoutException ioe) { c=65535; } if(c<0) { if(debug) Log.debugOut(runnableName,"Internal: EOF observed."); throw new IOException("reset by peer"); } if(c==65535) { final long ellapsed = System.currentTimeMillis()-timeSinceLastChar; if(ellapsed > IDLE_TIMEOUT) { quitFlag=true; if(debug) Log.debugOut(runnableName,"Internal: generated timeout: "+msgsSent+" msgs sent"); break; } else break; } if(sock.isClosed()) { if(debug) Log.debugOut(runnableName,"Internal: Noticed socket close."); quitFlag=true; break; } timeSinceLastChar=System.currentTimeMillis(); if((lastc==cr.charAt(0))&&(c==cr.charAt(1))) { cmdQueue.add(input.substring(0,input.length()-1)); input.setLength(0); continue; } input.append(c); if(input.length()>server.getMaxMsgSize()) { if(debug) { final StringBuilder str=new StringBuilder(""); for(int i=0;i<10 && i<input.length();i++) str.append((int)input.charAt(i)).append(","); Log.debugOut(runnableName,"Internal: 552 String exceeds size limit ("+server.getMaxMsgSize()+"): "+str.toString()); } //Log.errOut("SMTPR","Long request from "+sock.getInetAddress()); sout.write("552 String exceeds size limit. You are very bad!"+cr); sout.flush(); quitFlag=true; input.setLength(0); } } while(cmdQueue.size()>0) { final String s=cmdQueue.removeFirst(); String parm=""; final int cmdindex=s.indexOf(' '); String cmd=s.toUpperCase(); if(cmdindex>0) { cmd=s.substring(0,cmdindex).toUpperCase(); parm=s.substring(cmdindex+1); } if(debug) Log.debugOut(runnableName,"Input: "+cmd+" "+parm); if((dataMode)&&(s.equals("."))) { if(debug) Log.debugOut(runnableName,"End of data reached."); dataMode=false; boolean translateEqualSigns=false; /*When the SMTP server accepts a message either for relaying or for final delivery, it inserts a trace record (also referred to interchangeably as a "time stamp line" or "Received" line) at the top of the mail data. This trace record indicates the identity of the host that sent the message, the identity of the host that received the message (and is inserting this time stamp), and the date and time the message was received.*/ if(data.length()>=server.getMaxMsgSize()) replyData=("552 Message exceeds size limit."+cr).getBytes(); else { replyData=("250 Message accepted for delivery."+cr).getBytes(); msgsSent++; boolean startBuffering=false; StringBuilder finalData=new StringBuilder(""); char bodyType='t'; // h=html, t=text String subject=null; String boundry=null; final Map<Character,StringBuilder> dataBlocks=new Hashtable<Character,StringBuilder>(); // -1=waitForHeaderDone, // 0=waitForFirstHeaderDone, // 1=waitForBoundry, // 2=waitForContentTypeConfirmation int boundryState=-1; try { final BufferedReader lineR=new BufferedReader(new InputStreamReader(new ByteArrayInputStream(data.toString().getBytes()))); while(true) { String s2=lineR.readLine(); if(s2==null) break; String s2u=s2.toUpperCase(); if(debug) Log.debugOut(runnableName,"Header State="+boundryState+", "+s2); if((startBuffering)&&(boundry!=null)&&(s2.indexOf(boundry)>=0)) { if(debug) Log.debugOut(runnableName,"Multipart boundary "+boundry+" completed."); if(finalData.length()>0) dataBlocks.put(Character.valueOf(bodyType), finalData); finalData=new StringBuilder(""); boundryState=-1; startBuffering=false; continue; } else if(startBuffering) { boolean nextAppended=false; if(translateEqualSigns) { if(s2.indexOf('=')>=0) { final StringBuffer newStr=new StringBuffer(s2); for(int c1=0;c1<newStr.length()-2;c1++) if(newStr.charAt(c1)=='=') { if(("0123456789ABCDEF".indexOf(Character.toUpperCase(newStr.charAt(c1+1)))>=0) &&("0123456789ABCDEF".indexOf(Character.toUpperCase(newStr.charAt(c1+2)))>=0)) { final int x=(16*("0123456789ABCDEF".indexOf(Character.toUpperCase(newStr.charAt(c1+1))))) +"0123456789ABCDEF".indexOf(Character.toUpperCase(newStr.charAt(c1+2))); newStr.replace(c1,c1+3,""+((char)x)); } } s2=newStr.toString(); } if(s2!=null) { if(s2.endsWith("=")) { nextAppended=true; s2=s2.substring(0,s2.length()-1); } else if(s2.endsWith("=<BR>")) { nextAppended=true; s2=s2.substring(0,s2.length()-5); } } } if(nextAppended) finalData.append(s2); else finalData.append(s2+cr); } else if((s2.length()==0)&&(boundryState<0)) startBuffering=true; else if((s2.length()==0)&&(boundryState==0)) boundryState=1; else if(boundryState==1) { if(debug) Log.debugOut(runnableName,"Boundary "+boundry+" spotted -- state is now 2."); if(s2.indexOf(boundry)>=0) boundryState=2; continue; } else if((s2u.startsWith("SUBJECT: "))&&(boundryState<2)) { subject=s2.substring(9).trim(); if(debug) Log.debugOut(runnableName,"Subject="+subject); } else if((s2u.startsWith("CONTENT-TRANSFER-ENCODING: ")) ||(s2u.startsWith("CONTENT TRANSFER ENCODING: "))) { if(s2u.substring(27).trim().startsWith("QUOTED-PRINTABLE") ||s2u.substring(27).trim().startsWith("QUOTED PRINTABLE")) translateEqualSigns=true; else translateEqualSigns=false; if(debug) Log.debugOut(runnableName,"Transfer Equal Sign="+translateEqualSigns); } else if((s2u.startsWith("CONTENT TYPE: ")||s2u.startsWith("CONTENT-TYPE: "))) { if((boundryState<0)&&(s2u.substring(14).trim().startsWith("MULTIPART/"))) { String contentType=s2.substring(14).trim(); int y=s2u.indexOf(';'); if(y>=0) contentType=s2.substring(14,y).trim(); int x=s2u.indexOf("BOUNDARY="); for(int z=0;(z<5)&&(x<0);z++) { s2=lineR.readLine(); if(s2==null) break; s2u=s2.toUpperCase(); x=s2u.indexOf("BOUNDARY="); } if(x<0) { replyData=("552 Message content type '"+contentType+"' not accepted without boundry."+cr).getBytes(); subject=null; break; } if(s2!=null) { boundry=s2.substring(x+9).trim(); y=s2.indexOf(';'); if(y>=0) s2=s2.substring(0,y).trim(); if(boundry.startsWith("\"")&&boundry.endsWith("\"")) boundry=boundry.substring(1,boundry.length()-1).trim(); boundryState=0; } } else if(s2u.substring(14).trim().startsWith("TEXT/HTML")) { if(boundryState==2) boundryState=-1; bodyType='h'; } else if(!s2u.substring(14).trim().startsWith("TEXT/PLAIN")) { bodyType='t'; if(boundryState==2) boundryState=0; else { replyData=("552 Message content type '"+s2u.substring(14).trim()+"' not accepted."+cr).getBytes(); subject=null; break; } } else if(boundryState==2) boundryState=-1; } } } catch(final IOException e){} if((replyData!=null)&&(new String(replyData).startsWith("250"))) { if((finalData.length()==0)&&(!startBuffering)) { finalData=new StringBuilder(data.toString()); if(subject==null) subject=""; } if(dataBlocks.containsKey(Character.valueOf('h'))) { bodyType='h'; finalData=dataBlocks.get(Character.valueOf('h')); } else if(finalData.toString().trim().length()==0) { if(dataBlocks.size()>0) { bodyType=dataBlocks.keySet().iterator().next().charValue(); finalData=dataBlocks.get(Character.valueOf(bodyType)); } } if(subject!=null) { if(subject.toUpperCase().startsWith("MOTD") ||subject.toUpperCase().startsWith("MOTM") ||subject.toUpperCase().startsWith("MOTY")) { if((fromM==null)||(!CMSecurity.isAllowedAnywhere(fromM,CMSecurity.SecFlag.JOURNALS))) subject=subject.substring(4); } for(int i=0;i<to.size();i++) { final String journal=server.getAnEmailJournal(to.elementAt(i)); if(journal!=null) { String parentKey=""; final String fdat=finalData.toString().trim(); if((subject!=null) &&(!subject.trim().equalsIgnoreCase("subscribe")) &&(!subject.trim().equalsIgnoreCase("unsubscribe")) &&(!fdat.trim().equalsIgnoreCase("subscribe")) &&(!fdat.trim().equalsIgnoreCase("unsubscribe")) &&(server.isAForwardingJournal(journal))) { if(server.isASubscribeOnlyJournal(journal)) { if((fromM==null)||(!CMSecurity.isAllowedAnywhere(fromM,CMSecurity.SecFlag.JOURNALS))) { replyData=("552 Mailbox '"+journal+"' only accepts subscribe/unsubscribe."+cr).getBytes(); break; } } else { final Map<String, List<String>> lists=Resources.getCachedMultiLists("mailinglists.txt",true); List<String> mylist=null; if(lists!=null) mylist=lists.get(journal); if((mylist==null)||(!mylist.contains(from))) { if(debug) Log.debugOut(runnableName,from+" is not in mailing list for journal "+journal); replyData=("552 Mailbox '"+journal+"' only accepts messages from subscribers. Send an email with 'subscribe' as the subject."+cr).getBytes(); break; } final JournalsLibrary.ForumJournal forum=CMLib.journals().getForumJournal(journal); if((forum != null) &&(subject.trim().toUpperCase().startsWith("RE:")||subject.trim().toUpperCase().startsWith("RE "))) { String realSubject=subject.substring(3).trim(); if(realSubject.toUpperCase().startsWith("["+journal.toUpperCase()+"]")) realSubject=realSubject.substring(journal.length()+2).trim(); final List<JournalEntry> entries = CMLib.database().DBReadJournalPageMsgs(journal, null, realSubject, 0, 0); for(final JournalEntry entry : entries) { if(entry.subj().equalsIgnoreCase(realSubject)) { parentKey=entry.key(); break; } } } } } if(debug) Log.debugOut(runnableName,"Written: "+journal+"/"+from+"/ALL/"+bodyType); final StringBuilder finalFinalData=new StringBuilder(finalData); if(bodyType=='h') cleanHtml(journal, finalFinalData); CMLib.database().DBWriteJournalChild(journal, "",from, "ALL", parentKey, CMLib.coffeeFilter().simpleInFilter(new StringBuilder(subject)).toString(), CMLib.coffeeFilter().simpleInFilter(finalData).toString()); } else if(finalData.toString().trim().length()>0) { if(debug) Log.debugOut(runnableName,"Written: "+server.mailboxName()+"/"+from+"/"+to.elementAt(i)+"/"+bodyType); final StringBuilder finalFinalData=new StringBuilder(finalData); if(bodyType=='h') cleanHtml(journal, finalFinalData); CMLib.database().DBWriteJournal(server.mailboxName(), from, to.elementAt(i), CMLib.coffeeFilter().simpleInFilter(new StringBuilder(subject)).toString(), CMLib.coffeeFilter().simpleInFilter(finalFinalData).toString()); } } } } } } else if(cmd.equals("RSET")) { replyData=("250 Reset state"+cr).getBytes(); dataMode=false; from=null; fromM=null; to=null; data=null; } else if(dataMode) { if(data==null) data=new StringBuffer(""); if(data.length()<server.getMaxMsgSize()) data.append(s+cr); } else if(cmd.equals("HELP")) { parm=parm.toUpperCase(); if(parm.length()==0) { replyData=( "214-This is "+SMTPserver.ServerVersionString+cr+ "214-Topics:"+cr+ "214- HELO EHLO MAIL RCPT DATA"+cr+ "214- RSET NOOP QUIT HELP VRFY"+cr+ "214- EXPN VERB ETRN DSN"+cr+ "214-For more info use \"HELP <topic>\"."+cr+ "214-For local information send email to your local Archon."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("NOOP")) { replyData=( "214-NOOP"+cr+ "214- Do nothing."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("HELO")) { replyData=( "214-HELO <hostname>"+cr+ "214- Introduce yourself."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("EHLO")) { replyData=( "214-EHLO"+cr+ "214- Introduce yourself, and request extended SMTP mode."+cr+ "214-Possible replies include:"+cr+ "214- SEND Send as mail [RFC821]"+cr+ "214- SOML Send as mail or terminal [RFC821]"+cr+ "214- SAML Send as mail and terminal [RFC821]"+cr+ "214- EXPN Expand the mailing list [RFC821]"+cr+ "214- HELP Supply helpful information [RFC821]"+cr+ "214- TURN Turn the operation around [RFC821]"+cr+ "214- 8BITMIME Use 8-bit data [RFC1652]"+cr+ "214- SIZE Message size declaration [RFC1870]"+cr+ "214- VERB Verbose [Allman]"+cr+ "214- ONEX One message transaction only [Allman]"+cr+ "214- CHUNKING Chunking [RFC1830]"+cr+ "214- BINARYMIME Binary MIME [RFC1830]"+cr+ "214- PIPELINING Command Pipelining [RFC1854]"+cr+ "214- DSN Delivery Status Notification [RFC1891]"+cr+ "214- ETRN Remote Message Queue Starting [RFC1985]"+cr+ "214- XUSR Initial (user) submission [Allman]"+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("MAIL")) { replyData=( "214-MAIL FROM: <sender> [ <parameters> ]"+cr+ "214- Specifies the sender. Parameters are ESMTP extensions."+cr+ "214- See \"HELP DSN\" for details."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("DATA")) { replyData=( "214-DATA"+cr+ "214- Following text is collected as the message."+cr+ "214- End with a single dot."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("RSET")) { replyData=( "214-RSET"+cr+ "214- Resets the system."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("QUIT")) { replyData=( "214-QUIT"+cr+ "214- Exit SMTP."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("VRFY")) { replyData=( "214-VRFY <recipient>"+cr+ "214- Verify an address. If you want to see what it aliases"+cr+ "214- to, use EXPN instead."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("EXPN")) { replyData=( "214-EXPN <recipient>"+cr+ "214- Expand an address. If the address indicates a mailing"+cr+ "214- list, return the contents of that list."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("VERB")) { replyData=( "214-VERB"+cr+ "214- Not implemented in this server."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("ETRN")) { replyData=( "214-ETRN [ <hostname> | @<domain> | #<queuename> ]"+cr+ "214- Not implemented in this server."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("DSN")) { replyData=( "214-MAIL FROM: <sender> [ RET={ FULL | HDRS} ] [ ENVID=<envid> ]"+cr+ "214-RCPT TO: <recipient> [ NOTIFY={NEVER,SUCCESS,FAILURE,DELAY} ]"+cr+ "214- [ ORCPT=<recipient> ]"+cr+ "214- SMTP Delivery Status Notifications."+cr+ "214-Descriptions:"+cr+ "214- RET Return either the full message or only headers."+cr+ "214- ENVID Sender's \"envelope identifier\" for tracking."+cr+ "214- NOTIFY When to send a DSN. Multiple options are OK, comma-"+cr+ "214- delimited. NEVER must appear by itself."+cr+ "214- ORCPT Original recipient."+cr+ "214 End of HELP info"+cr).getBytes(); } else if(parm.equals("RCPT")) { replyData=( "214-RCPT TO: <recipient> [ <parameters> ]"+cr+ "214- Specifies the recipient. Can be used any number of times."+cr+ "214- Parameters are ESMTP extensions. See \"HELP DSN\" for details."+cr+ "214 End of HELP info"+cr).getBytes(); } else replyData=("504 Help topic \""+parm+"\" unknown"+cr).getBytes(); } else if(cmd.equals("NOOP")) replyData=(S_250+cr).getBytes(); else if(cmd.equals("HELO") ||cmd.equals("EHLO")) { if((domain!=null)&&(parm.trim().length()==0)) replyData=("503 "+sock.getLocalAddress().getHostName()+" Duplicate HELO/EHLO"+cr).getBytes(); else if(parm.trim().length()==0) replyData=("501 "+cmd+" requires domain address"+cr).getBytes(); else { domain=parm; if(cmd.equals("EHLO")) { replyData=("250-"+server.domainName()+" Ok."+cr +"250-8BITMIME"+cr +"250-SIZE "+server.getMaxMsgSize()+cr +"250-HELP"+cr +"250-ONEX"+cr +"250-PIPELINING"+cr +"250 DSN"+cr).getBytes(); } else replyData=("250 "+sock.getLocalAddress().getHostName()+" Hello "+sock.getInetAddress().getHostName()+" ["+sock.getInetAddress().getHostAddress()+"], pleased to meet you"+cr).getBytes(); } } else if(cmd.equals("MAIL")) { int x=parm.indexOf(':'); if(x<0) replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); else { final String to2=parm.substring(0,x).trim(); if(!to2.equalsIgnoreCase("from")) replyData=("500 Unrecognized command \""+cmd+"\""+cr).getBytes(); else { parm=parm.substring(x+1).trim(); String parmparms=""; boolean error=false; if(parm.startsWith("<")) { x=parm.indexOf('>'); if(x<0) { replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); error=true; } else { parmparms=parm.substring(x+1).trim(); parm=parm.substring(1,x); } } else if(parm.indexOf(' ')>=0) { replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); error=true; } if(parmparms.trim().length()>0) { if((parmparms.trim().toUpperCase().startsWith("SIZE=")) ||(!CMath.isNumber(parmparms.trim().toUpperCase().substring(5)))) { final int size=CMath.s_int(parmparms.trim().toUpperCase().substring(5)); if(size>server.getMaxMsgSize()) { replyData=("552 String exceeds size limit. But you were nice to tell me!"+cr).getBytes(); error=true; } } else { replyData=("502 Parameters not supported... \""+parmparms+"\""+cr).getBytes(); error=true; } } if(!error) { final String name=validLocalAccount(parm,true); if(name==null) { if((++failures)==3) { replyData=("421 Quit Fishing!"+cr).getBytes(); quitFlag=true; } else replyData=("551 Requested action not taken: User is not local."+cr).getBytes(); } else { replyData=("250 OK "+name+cr).getBytes(); from=name; fromM=getAccountMob(from); } } } } } else if(cmd.equals("DATA")) { if(from==null) replyData=("503 Need MAIL command"+cr).getBytes(); else if(to==null) replyData=("503 Need RCPT (recipient)"+cr).getBytes(); else { replyData=("354 Enter mail, end with \".\" on a line by itself"+cr).getBytes(); dataMode=true; } } else if(cmd.equals("QUIT")) { replyData=("221 "+server.domainName()+" closing connection"+cr).getBytes(); quitFlag=true; } else if(cmd.equals("VRFY")) replyData=("252 Cannot VRFY user; try RCPT to attempt delivery (or try finger)"+cr).getBytes(); else if(cmd.equals("EXPN")) replyData=("502 Sorry, we don't allow mailing lists"+cr).getBytes(); else if(cmd.equals("VERB")) replyData=("502 Verbose unavailable"+cr).getBytes(); else if(cmd.equals("ETRN")) replyData=("502 ETRN not implemented"+cr).getBytes(); else if(cmd.equals("RCPT")) { int x=parm.indexOf(':'); if(x<0) replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); else { final String to2=parm.substring(0,x).trim(); if(!to2.equalsIgnoreCase("to")) replyData=("500 Unrecognized command \""+cmd+"\""+cr).getBytes(); else { parm=parm.substring(x+1).trim(); String parmparms=""; boolean error=false; if(parm.startsWith("<")) { x=parm.indexOf('>'); if(x<0) { replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); error=true; } else { parmparms=parm.substring(x+1).trim(); parm=parm.substring(1,x); if(parmparms.trim().length()>0) { if((parmparms.trim().toUpperCase().startsWith("SIZE=")) ||(!CMath.isNumber(parmparms.trim().toUpperCase().substring(5)))) { final int size=CMath.s_int(parmparms.trim().toUpperCase().substring(5)); if(size>server.getMaxMsgSize()) replyData=("552 String exceeds size limit. But you were nice to tell me!"+cr).getBytes(); } //else replyData=("502 Parameters not supported... \""+parmparms+"\""+cr).getBytes(); // say nothing, see if that works. } } } else if(parm.indexOf(' ')>=0) { replyData=("501 Syntax error in \""+parm+"\""+cr).getBytes(); error=true; } if(parm.indexOf('@')<0) replyData=("550 "+parm+" user unknown."+cr).getBytes(); else if(!error) { final String name=validLocalAccount(parm,false); if(name==null) { if((++failures)==3) { replyData=("421 Quit Fishing!"+cr).getBytes(); quitFlag=true; } else replyData=("553 Requested action not taken: User is not local."+cr).getBytes(); } else { if(server.getAnEmailJournal(name)!=null) { boolean jerror=false; if(server.getJournalCriteria(name)!=null) { if(from==null) { replyData=("503 Need MAIL before RCPT"+cr).getBytes(); jerror=true; } else { if((fromM==null) ||(!CMLib.masking().maskCheck(server.getJournalCriteria(name),fromM,false))) { replyData=("552 User '"+from+"' may not send emails to '"+name+"'."+cr).getBytes(); jerror=true; } } } if(!jerror) { replyData=("250 OK "+name+cr).getBytes(); if(to==null) to=new Vector<String>(); if(!to.contains(name)) to.addElement(name); } } else if(CMLib.database().DBCountJournal(server.mailboxName(),null,name)>=server.getMaxMsgs()) replyData=("552 Mailbox '"+name+"' is full."+cr).getBytes(); else { replyData=("250 OK "+name+cr).getBytes(); if(to==null) to=new Vector<String>(); if(!to.contains(name)) to.addElement(name); } } } } } } else replyData=lastReplyData;//("500 Command Unrecognized: \""+cmd+"\""+cr).getBytes(); if ((replyData != null)) { respQueue.add(replyData); replyData=null; if((cmdQueue.size()==0)&&(respQueue.size()>0)) { // we should be looping through these .. why does ZD act so wierd?! //final byte [] resp=respQueue.getLast(); for(byte[] resp : respQueue) { if(debug) Log.debugOut(runnableName,"Reply: "+CMStrings.replaceAll(new String(resp),cr,"\\r\\n")); // must insert a blank line before message body sout.write(new String(resp)); sout.flush(); } respQueue.clear(); } } } } } catch (final java.net.SocketTimeoutException e2) { try { if (sout != null) { sout.write("421 You're taking too long. I'm outa here."+cr); sout.flush(); } } catch(final Exception e) { Log.errOut(runnableName,"Exception2: " + e.getMessage() ); } } catch (final Exception e) { final String errorMessage=e.getMessage(); final StringBuilder msg = new StringBuilder(errorMessage==null?"EMPTY e.getMessage()":errorMessage); final StackTraceElement[] ts = e.getStackTrace(); if(ts != null) for(final StackTraceElement t : ts) msg.append(" ").append(t.getFileName()).append("(").append(t.getLineNumber()).append(")"); Log.errOut(runnableName,"Exception: " + msg.toString() ); } try { if (sout != null) { sout.flush(); sout.close(); sout = null; } } catch (final Exception e) {} try { if (sin != null) { sin.close(); sin = null; } } catch (final Exception e) {} try { if (sock != null) { sock.close(); sock = null; } } catch (final Exception e) {} } }