/tags/1.0.0/java/kouchat/src/net/usikkert/kouchat/net/FileSender.java

http://kouchat.googlecode.com/ · Java · 414 lines · 236 code · 60 blank · 118 comment · 23 complexity · 591850e32a0aa78d5b0a2aa42e2ec75d MD5 · raw file

  1. /***************************************************************************
  2. * Copyright 2006-2009 by Christian Ihle *
  3. * kontakt@usikkert.net *
  4. * *
  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 2 of the License, or *
  8. * (at your option) any later version. *
  9. * *
  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. *
  14. * *
  15. * You should have received a copy of the GNU General Public License *
  16. * along with this program; if not, write to the *
  17. * Free Software Foundation, Inc., *
  18. * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. *
  19. ***************************************************************************/
  20. package net.usikkert.kouchat.net;
  21. import java.io.File;
  22. import java.io.FileInputStream;
  23. import java.io.IOException;
  24. import java.io.OutputStream;
  25. import java.net.InetAddress;
  26. import java.net.Socket;
  27. import java.net.UnknownHostException;
  28. import java.util.logging.Level;
  29. import java.util.logging.Logger;
  30. import net.usikkert.kouchat.event.FileTransferListener;
  31. import net.usikkert.kouchat.misc.User;
  32. import net.usikkert.kouchat.util.ByteCounter;
  33. /**
  34. * This is a class for sending files to other users.
  35. *
  36. * <p>To send a file, the user at the other end needs to
  37. * open a server socket so this client can connect.</p>
  38. *
  39. * @author Christian Ihle
  40. */
  41. public class FileSender implements FileTransfer
  42. {
  43. /** The logger. */
  44. private static final Logger LOG = Logger.getLogger( FileSender.class.getName() );
  45. /** The user to send a file to. */
  46. private final User user;
  47. /** The file to send to the user. */
  48. private final File file;
  49. /** Keeps count of the transfer speed. */
  50. private final ByteCounter bCounter;
  51. /** Percent of the file transferred. */
  52. private int percent;
  53. /** Number of bytes transferred. */
  54. private long transferred;
  55. /** If the file was successfully sent. */
  56. private boolean sent;
  57. /** If the file transfer is canceled. */
  58. private boolean cancel;
  59. /** If still waiting for the file transfer to begin. */
  60. private boolean waiting;
  61. /** The file transfer listener. */
  62. private FileTransferListener listener;
  63. /** The input stream from the file. */
  64. private FileInputStream fis;
  65. /** The output stream to the other user. */
  66. private OutputStream os;
  67. /** The socket connection to the other user. */
  68. private Socket sock;
  69. /**
  70. * Constructor. Creates a new file sender.
  71. *
  72. * @param user The user to send the file to.
  73. * @param file The file to send.
  74. */
  75. public FileSender( final User user, final File file )
  76. {
  77. this.user = user;
  78. this.file = file;
  79. bCounter = new ByteCounter();
  80. waiting = true;
  81. }
  82. /**
  83. * Connects to the user at the specified port and transfers the file
  84. * to that user.
  85. *
  86. * @param port The port to use when connecting to the user.
  87. * @return If the file transfer was successful.
  88. */
  89. public boolean transfer( final int port )
  90. {
  91. if ( !cancel )
  92. {
  93. listener.statusConnecting();
  94. waiting = false;
  95. sent = false;
  96. try
  97. {
  98. int counter = 0;
  99. while ( sock == null && counter < 10 )
  100. {
  101. counter++;
  102. try
  103. {
  104. sock = new Socket( InetAddress.getByName( user.getIpAddress() ), port );
  105. }
  106. catch ( final UnknownHostException e )
  107. {
  108. LOG.log( Level.SEVERE, e.toString(), e );
  109. }
  110. catch ( final IOException e )
  111. {
  112. LOG.log( Level.SEVERE, e.toString(), e );
  113. }
  114. try
  115. {
  116. Thread.sleep( 100 );
  117. }
  118. catch ( final InterruptedException e )
  119. {
  120. LOG.log( Level.SEVERE, e.toString(), e );
  121. }
  122. }
  123. if ( sock != null && !cancel )
  124. {
  125. listener.statusTransferring();
  126. fis = new FileInputStream( file );
  127. os = sock.getOutputStream();
  128. byte[] b = new byte[1024];
  129. transferred = 0;
  130. percent = 0;
  131. int tmpTransferred = 0;
  132. int tmpPercent = 0;
  133. int transCounter = 0;
  134. bCounter.prepare();
  135. while ( !cancel && ( tmpTransferred = fis.read( b ) ) != -1 )
  136. {
  137. os.write( b, 0, tmpTransferred );
  138. transferred += tmpTransferred;
  139. percent = (int) ( ( transferred * 100 ) / file.length() );
  140. bCounter.addBytes( tmpTransferred );
  141. transCounter++;
  142. if ( percent > tmpPercent || transCounter >= 250 )
  143. {
  144. transCounter = 0;
  145. tmpPercent = percent;
  146. listener.transferUpdate();
  147. }
  148. }
  149. if ( !cancel && transferred == file.length() )
  150. {
  151. sent = true;
  152. listener.statusCompleted();
  153. }
  154. else
  155. {
  156. listener.statusFailed();
  157. }
  158. }
  159. else
  160. {
  161. listener.statusFailed();
  162. }
  163. }
  164. catch ( final UnknownHostException e )
  165. {
  166. LOG.log( Level.SEVERE, e.toString(), e );
  167. listener.statusFailed();
  168. }
  169. catch ( final IOException e )
  170. {
  171. LOG.log( Level.SEVERE, e.toString() );
  172. listener.statusFailed();
  173. }
  174. finally
  175. {
  176. stopSender();
  177. cleanupConnections();
  178. }
  179. }
  180. return sent;
  181. }
  182. /**
  183. * Sets all connections to null.
  184. */
  185. private void cleanupConnections()
  186. {
  187. fis = null;
  188. os = null;
  189. sock = null;
  190. }
  191. /**
  192. * Closes the connection to the user.
  193. */
  194. private void stopSender()
  195. {
  196. try
  197. {
  198. if ( fis != null )
  199. fis.close();
  200. }
  201. catch ( final IOException e )
  202. {
  203. LOG.log( Level.SEVERE, e.toString(), e );
  204. }
  205. try
  206. {
  207. if ( os != null )
  208. os.flush();
  209. }
  210. catch ( final IOException e )
  211. {
  212. LOG.log( Level.SEVERE, e.toString(), e );
  213. }
  214. try
  215. {
  216. if ( os != null )
  217. os.close();
  218. }
  219. catch ( final IOException e )
  220. {
  221. LOG.log( Level.SEVERE, e.toString(), e );
  222. }
  223. try
  224. {
  225. if ( sock != null )
  226. sock.close();
  227. }
  228. catch ( final IOException e )
  229. {
  230. LOG.log( Level.SEVERE, e.toString(), e );
  231. }
  232. }
  233. /**
  234. * Checks if the file transfer has been canceled.
  235. *
  236. * @return If the file transfer has been canceled.
  237. */
  238. @Override
  239. public boolean isCanceled()
  240. {
  241. return cancel;
  242. }
  243. /**
  244. * Cancels the file transfer.
  245. */
  246. @Override
  247. public void cancel()
  248. {
  249. cancel = true;
  250. stopSender();
  251. listener.statusFailed();
  252. }
  253. /**
  254. * Checks if the file transfer is complete.
  255. *
  256. * @return If the file transfer is complete.
  257. */
  258. @Override
  259. public boolean isTransferred()
  260. {
  261. return sent;
  262. }
  263. /**
  264. * The percent of the file transfer that is completed.
  265. *
  266. * @return Percent completed.
  267. */
  268. @Override
  269. public int getPercent()
  270. {
  271. return percent;
  272. }
  273. /**
  274. * The other user, which receives a file.
  275. *
  276. * @return The other user.
  277. */
  278. @Override
  279. public User getUser()
  280. {
  281. return user;
  282. }
  283. /**
  284. * Number of bytes transferred.
  285. *
  286. * @return Bytes transferred.
  287. */
  288. @Override
  289. public long getTransferred()
  290. {
  291. return transferred;
  292. }
  293. /**
  294. * Gets the size of the file being transferred, in bytes.
  295. *
  296. * @return The file size.
  297. */
  298. @Override
  299. public long getFileSize()
  300. {
  301. return file.length();
  302. }
  303. /**
  304. * Gets the direction, which is send.
  305. *
  306. * @return Send, the direction of the file transfer.
  307. */
  308. @Override
  309. public Direction getDirection()
  310. {
  311. return Direction.SEND;
  312. }
  313. /**
  314. * Gets the number of bytes transferred per second.
  315. *
  316. * @return The speed in bytes per second.
  317. */
  318. @Override
  319. public long getSpeed()
  320. {
  321. return bCounter.getBytesPerSec();
  322. }
  323. /**
  324. * Gets the file that is being transferred.
  325. *
  326. * @return The file.
  327. */
  328. @Override
  329. public File getFile()
  330. {
  331. return file;
  332. }
  333. /**
  334. * If still waiting for the file transfer to begin.
  335. *
  336. * @return If waiting or not.
  337. */
  338. public boolean isWaiting()
  339. {
  340. return waiting;
  341. }
  342. /**
  343. * Registers a file transfer listener, which will receive updates
  344. * when certain events happen in the progression of the file transfer.
  345. *
  346. * @param listener The listener to register.
  347. */
  348. @Override
  349. public void registerListener( final FileTransferListener listener )
  350. {
  351. this.listener = listener;
  352. listener.statusWaiting();
  353. }
  354. }