Reformat source code, new line length for comments (79), some trailing whitespace...
[jFCPlib.git] / src / main / java / net / pterodactylus / fcp / FcpConnectionHandler.java
1 /*
2  * jFCPlib - FcpConnectionHandler.java - Copyright © 2008 David Roden
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published by
6  * the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful,
10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
12  * GNU General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
17  */
18
19 package net.pterodactylus.fcp;
20
21 import java.io.IOException;
22 import java.io.InputStream;
23 import java.nio.ByteBuffer;
24 import java.nio.charset.Charset;
25 import java.util.logging.Level;
26 import java.util.logging.Logger;
27
28 import net.pterodactylus.util.logging.Logging;
29
30 /**
31  * Handles an FCP connection to a node.
32  *
33  * @author David ‘Bombe’ Roden <bombe@freenetproject.org>
34  */
35 class FcpConnectionHandler implements Runnable {
36
37         /** The logger. */
38         private static final Logger logger = Logging.getLogger(FcpConnectionHandler.class.getName());
39
40         /** The underlying connection. */
41         private final FcpConnection fcpConnection;
42
43         /** The input stream from the node. */
44         private final InputStream remoteInputStream;
45
46         /** Whether to stop the connection handler. */
47         private boolean shouldStop;
48
49         /** Whether the next read line feed should be ignored. */
50         private boolean ignoreNextLinefeed;
51
52         /**
53          * Creates a new connection handler that operates on the given connection
54          * and input stream.
55          *
56          * @param fcpConnection
57          *            The underlying FCP connection
58          * @param remoteInputStream
59          *            The input stream from the node
60          */
61         public FcpConnectionHandler(FcpConnection fcpConnection, InputStream remoteInputStream) {
62                 this.fcpConnection = fcpConnection;
63                 this.remoteInputStream = remoteInputStream;
64         }
65
66         /**
67          * {@inheritDoc}
68          */
69         @Override
70         public void run() {
71                 FcpMessage fcpMessage = null;
72                 Throwable throwable = null;
73                 while (true) {
74                         synchronized (this) {
75                                 if (shouldStop) {
76                                         break;
77                                 }
78                         }
79                         try {
80                                 String line = readLine();
81                                 logger.log(Level.FINEST, "read line: %1$s", line);
82                                 if (line == null) {
83                                         break;
84                                 }
85                                 if (line.length() == 0) {
86                                         continue;
87                                 }
88                                 line = line.trim();
89                                 if (fcpMessage == null) {
90                                         fcpMessage = new FcpMessage(line);
91                                         continue;
92                                 }
93                                 if ("EndMessage".equalsIgnoreCase(line) || "Data".equalsIgnoreCase(line)) {
94                                         fcpConnection.handleMessage(fcpMessage);
95                                         fcpMessage = null;
96                                 }
97                                 int equalSign = line.indexOf('=');
98                                 if (equalSign == -1) {
99                                         /* something's fishy! */
100                                         continue;
101                                 }
102                                 String field = line.substring(0, equalSign);
103                                 String value = line.substring(equalSign + 1);
104                                 assert fcpMessage != null: "fcp message is null";
105                                 fcpMessage.setField(field, value);
106                         } catch (IOException ioe1) {
107                                 throwable = ioe1;
108                                 break;
109                         }
110                 }
111                 fcpConnection.handleDisconnect(throwable);
112         }
113
114         /**
115          * Stops the connection handler.
116          */
117         public void stop() {
118                 synchronized (this) {
119                         shouldStop = true;
120                 }
121         }
122
123         //
124         // PRIVATE METHODS
125         //
126
127         /**
128          * Reads bytes from {@link #remoteInputStream} until ‘\r’ or ‘\n’ are
129          * encountered and decodes the read bytes using UTF-8.
130          *
131          * @return The decoded line
132          * @throws IOException
133          *             if an I/O error occurs
134          */
135         private String readLine() throws IOException {
136                 byte[] readBytes = new byte[512];
137                 int readIndex = 0;
138                 while (true) {
139                         int nextByte = remoteInputStream.read();
140                         if (nextByte == -1) {
141                                 if (readIndex == 0) {
142                                         return null;
143                                 }
144                                 break;
145                         }
146                         if (nextByte == 10) {
147                                 if (!ignoreNextLinefeed) {
148                                         break;
149                                 }
150                         }
151                         ignoreNextLinefeed = false;
152                         if (nextByte == 13) {
153                                 ignoreNextLinefeed = true;
154                                 break;
155                         }
156                         if (readIndex == readBytes.length) {
157                                 /* recopy & enlarge array */
158                                 byte[] newReadBytes = new byte[readBytes.length * 2];
159                                 System.arraycopy(readBytes, 0, newReadBytes, 0, readBytes.length);
160                                 readBytes = newReadBytes;
161                         }
162                         readBytes[readIndex++] = (byte) nextByte;
163                 }
164                 ByteBuffer byteBuffer = ByteBuffer.wrap(readBytes, 0, readIndex);
165                 return Charset.forName("UTF-8").decode(byteBuffer).toString();
166         }
167
168 }