Fix for #567 "mailinglist gateway does not recover after database outage".
3 * see AUTHORS for the list of contributors
5 * This program is free software: you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation, either version 3 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program. If not, see <http://www.gnu.org/licenses/>.
19 package org.sonews.daemon.command;
21 import java.io.IOException;
22 import java.io.ByteArrayInputStream;
23 import java.io.ByteArrayOutputStream;
24 import java.sql.SQLException;
25 import java.util.Arrays;
26 import javax.mail.MessagingException;
27 import javax.mail.internet.AddressException;
28 import javax.mail.internet.InternetHeaders;
29 import org.sonews.config.Config;
30 import org.sonews.util.Log;
31 import org.sonews.mlgw.Dispatcher;
32 import org.sonews.storage.Article;
33 import org.sonews.storage.Group;
34 import org.sonews.daemon.NNTPConnection;
35 import org.sonews.storage.Headers;
36 import org.sonews.storage.StorageBackendException;
37 import org.sonews.storage.StorageManager;
38 import org.sonews.feed.FeedManager;
39 import org.sonews.util.Stats;
42 * Implementation of the POST command. This command requires multiple lines
43 * from the client, so the handling of asynchronous reading is a little tricky
45 * @author Christian Lins
48 public class PostCommand implements Command
51 private final Article article = new Article();
52 private int lineCount = 0;
53 private long bodySize = 0;
54 private InternetHeaders headers = null;
55 private long maxBodySize =
56 Config.inst().get(Config.ARTICLE_MAXSIZE, 128) * 1024L; // Size in bytes
57 private PostState state = PostState.WaitForLineOne;
58 private final ByteArrayOutputStream bufBody = new ByteArrayOutputStream();
59 private final StringBuilder strHead = new StringBuilder();
62 public String[] getSupportedCommandStrings()
64 return new String[]{"POST"};
68 public boolean hasFinished()
70 return this.state == PostState.Finished;
74 public String impliedCapability()
80 public boolean isStateful()
86 * Process the given line String. line.trim() was called by NNTPConnection.
88 * @throws java.io.IOException
89 * @throws java.sql.SQLException
91 @Override // TODO: Refactor this method to reduce complexity!
92 public void processLine(NNTPConnection conn, String line, byte[] raw)
93 throws IOException, StorageBackendException
99 if(line.equalsIgnoreCase("POST"))
101 conn.println("340 send article to be posted. End with <CR-LF>.<CR-LF>");
102 state = PostState.ReadingHeaders;
106 conn.println("500 invalid command usage");
112 strHead.append(line);
113 strHead.append(NNTPConnection.NEWLINE);
115 if("".equals(line) || ".".equals(line))
117 // we finally met the blank line
118 // separating headers from body
122 // Parse the header using the InternetHeader class from JavaMail API
123 headers = new InternetHeaders(
124 new ByteArrayInputStream(strHead.toString().trim()
125 .getBytes(conn.getCurrentCharset())));
127 // add the header entries for the article
128 article.setHeaders(headers);
130 catch (MessagingException e)
133 conn.println("500 posting failed - invalid header");
134 state = PostState.Finished;
138 // Change charset for reading body;
139 // for multipart messages UTF-8 is returned
140 //conn.setCurrentCharset(article.getBodyCharset());
142 state = PostState.ReadingBody;
146 // Post an article without body
147 postArticle(conn, article);
148 state = PostState.Finished;
157 // Set some headers needed for Over command
158 headers.setHeader(Headers.LINES, Integer.toString(lineCount));
159 headers.setHeader(Headers.BYTES, Long.toString(bodySize));
161 byte[] body = bufBody.toByteArray();
164 // Remove trailing CRLF
165 body = Arrays.copyOf(body, body.length - 2);
167 article.setBody(body); // set the article body
169 postArticle(conn, article);
170 state = PostState.Finished;
174 bodySize += line.length() + 1;
177 // Add line to body buffer
178 bufBody.write(raw, 0, raw.length);
179 bufBody.write(NNTPConnection.NEWLINE.getBytes());
181 if(bodySize > maxBodySize)
183 conn.println("500 article is too long");
184 state = PostState.Finished;
192 // Should never happen
193 Log.get().severe("PostCommand::processLine(): already finished...");
199 * Article is a control message and needs special handling.
202 private void controlMessage(NNTPConnection conn, Article article)
205 String[] ctrl = article.getHeader(Headers.CONTROL)[0].split(" ");
206 if(ctrl.length == 2) // "cancel <mid>"
210 StorageManager.current().delete(ctrl[1]);
212 // Move cancel message to "control" group
213 article.setHeader(Headers.NEWSGROUPS, "control");
214 StorageManager.current().addArticle(article);
215 conn.println("240 article cancelled");
217 catch(StorageBackendException ex)
219 Log.get().severe(ex.toString());
220 conn.println("500 internal server error");
225 conn.println("441 unknown control header");
229 private void supersedeMessage(NNTPConnection conn, Article article)
234 String oldMsg = article.getHeader(Headers.SUPERSEDES)[0];
235 StorageManager.current().delete(oldMsg);
236 StorageManager.current().addArticle(article);
237 conn.println("240 article replaced");
239 catch(StorageBackendException ex)
241 Log.get().severe(ex.toString());
242 conn.println("500 internal server error");
246 private void postArticle(NNTPConnection conn, Article article)
249 if(article.getHeader(Headers.CONTROL)[0].length() > 0)
251 controlMessage(conn, article);
253 else if(article.getHeader(Headers.SUPERSEDES)[0].length() > 0)
255 supersedeMessage(conn, article);
257 else // Post the article regularily
259 // Circle check; note that Path can already contain the hostname here
260 String host = Config.inst().get(Config.HOSTNAME, "localhost");
261 if(article.getHeader(Headers.PATH)[0].indexOf(host + "!", 1) > 0)
263 Log.get().info(article.getMessageID() + " skipped for host " + host);
264 conn.println("441 I know this article already");
268 // Try to create the article in the database or post it to
269 // appropriate mailing list
272 boolean success = false;
273 String[] groupnames = article.getHeader(Headers.NEWSGROUPS)[0].split(",");
274 for(String groupname : groupnames)
276 Group group = StorageManager.current().getGroup(groupname);
277 if(group != null && !group.isDeleted())
279 if(group.isMailingList() && !conn.isLocalConnection())
281 // Send to mailing list; the Dispatcher writes
282 // statistics to database
283 Dispatcher.toList(article, group.getName());
289 if(!StorageManager.current().isArticleExisting(article.getMessageID()))
291 StorageManager.current().addArticle(article);
293 // Log this posting to statistics
294 Stats.getInstance().mailPosted(
295 article.getHeader(Headers.NEWSGROUPS)[0]);
304 conn.println("240 article posted ok");
305 FeedManager.queueForPush(article);
309 conn.println("441 newsgroup not found");
312 catch(AddressException ex)
314 Log.get().warning(ex.getMessage());
315 conn.println("441 invalid sender address");
317 catch(MessagingException ex)
319 // A MessageException is thrown when the sender email address is
320 // invalid or something is wrong with the SMTP server.
321 System.err.println(ex.getLocalizedMessage());
322 conn.println("441 " + ex.getClass().getCanonicalName() + ": " + ex.getLocalizedMessage());
324 catch(StorageBackendException ex)
326 ex.printStackTrace();
327 conn.println("500 internal server error");