/src/net/bytemine/openvpn/ScpTool.java

http://github.com/bytemine/bytemine-manager · Java · 275 lines · 168 code · 62 blank · 45 comment · 34 complexity · e679674ea19c507bec4ddfcfce7cc980 MD5 · raw file

  1. /*************************************************************************
  2. * Written by / Copyright (C) 2009-2012 bytemine GmbH *
  3. * Author: Daniel Rauer E-Mail: rauer@bytemine.net *
  4. * *
  5. * http://www.bytemine.net/ *
  6. *************************************************************************/
  7. package net.bytemine.openvpn;
  8. import com.jcraft.jsch.Channel;
  9. import com.jcraft.jsch.ChannelExec;
  10. import com.jcraft.jsch.Session;
  11. import java.io.ByteArrayInputStream;
  12. import java.io.ByteArrayOutputStream;
  13. import java.io.IOException;
  14. import java.io.InputStream;
  15. import java.io.OutputStream;
  16. import java.net.ConnectException;
  17. import java.util.ResourceBundle;
  18. import java.util.logging.Level;
  19. import java.util.logging.Logger;
  20. import net.bytemine.manager.bean.Server;
  21. import net.bytemine.manager.i18n.ResourceBundleMgmt;
  22. import net.bytemine.openvpn.ssh.SSHConnector;
  23. import net.bytemine.openvpn.ssh.SSHSessionPool;
  24. /**
  25. * is able to copy files via scp
  26. *
  27. * @author Daniel Rauer
  28. */
  29. public class ScpTool {
  30. private static Logger logger = Logger.getLogger(ScpTool.class.getName());
  31. private ResourceBundle rb = ResourceBundleMgmt.getInstance().getUserBundle();
  32. private Server server;
  33. private Session sshSession;
  34. private SSHConnector sshConnector;
  35. ScpTool(Server server) throws Exception {
  36. this.server = server;
  37. sshConnector = new SSHConnector(this.server);
  38. }
  39. private Session getSession() throws ConnectException {
  40. sshSession = SSHSessionPool.getInstance().getSession(server.getHostname());
  41. if (sshSession == null || !sshSession.isConnected())
  42. this.sshSession = sshConnector.createSessionImmediately();
  43. return sshSession;
  44. }
  45. /**
  46. * Gets a file from a server via scp as OutputStream
  47. *
  48. * @param sourceFile The filename of the source file
  49. * @throws java.lang.Exception
  50. */
  51. byte[] getFromServer(String sourceFile) throws Exception {
  52. try {
  53. logger.info("ScpTool.getFrom start");
  54. ByteArrayOutputStream os = new ByteArrayOutputStream();
  55. // exec 'scp -f rfile'
  56. String command = "scp -f \"" + sourceFile + "\"";
  57. Channel channel = getSession().openChannel("exec");
  58. ((ChannelExec) channel).setCommand(command);
  59. OutputStream out = channel.getOutputStream();
  60. InputStream in = channel.getInputStream();
  61. channel.connect();
  62. byte[] buf = new byte[1024];
  63. // send '\0'
  64. buf[0] = 0;
  65. out.write(buf, 0, 1);
  66. out.flush();
  67. while (true) {
  68. int c = checkAck(in);
  69. if (c != 'C')
  70. break;
  71. // read '0644 '
  72. in.read(buf, 0, 5);
  73. long filesize = 0L;
  74. while (in.read(buf, 0, 1) >= 0 && buf[0] != ' ') {
  75. filesize = filesize * 10L + (long) (buf[0] - '0');
  76. }
  77. for (int i = 0; ; i++) {
  78. in.read(buf, i, 1);
  79. if (buf[i] == (byte) 0x0a) {
  80. break;
  81. }
  82. }
  83. // send '\0'
  84. buf[0] = 0;
  85. out.write(buf, 0, 1);
  86. out.flush();
  87. int pointer;
  88. while (true) {
  89. pointer = buf.length < filesize ? buf.length : (int) filesize;
  90. pointer = in.read(buf, 0, pointer);
  91. if (pointer < 0)
  92. break;
  93. os.write(buf, 0, pointer);
  94. filesize -= pointer;
  95. if (filesize == 0L)
  96. break;
  97. }
  98. os.close();
  99. // send '\0'
  100. buf[0] = 0;
  101. out.write(buf, 0, 1);
  102. out.flush();
  103. }
  104. channel.disconnect();
  105. logger.info("ScpTool.getFrom end");
  106. return os.toByteArray();
  107. } catch (Exception e) {
  108. logger.log(Level.SEVERE, e.getMessage(), e);
  109. throw new Exception(rb.getString("error.syncusers.scpFrom"));
  110. }
  111. }
  112. /**
  113. * Writes a String to a file on a server via scp
  114. *
  115. * @param content The content to write
  116. * @param targetFile The filename of the target file
  117. * @throws java.lang.Exception
  118. */
  119. void postToServer(String content, String targetFile) throws Exception {
  120. try {
  121. logger.info("ScpTool.postTo start");
  122. // exec 'scp -t rfile' remotely
  123. String command = "scp -t \"" + targetFile + "\"";
  124. Channel channel = getSession().openChannel("exec");
  125. ((ChannelExec) channel).setCommand(command);
  126. OutputStream out = channel.getOutputStream();
  127. InputStream in = channel.getInputStream();
  128. channel.connect();
  129. checkAck(in);
  130. long filesize = content.length();
  131. command = "C0644 " + filesize + " ";
  132. command += "file";
  133. command += "\n";
  134. out.write(command.getBytes());
  135. out.flush();
  136. checkAck(in);
  137. // send the content
  138. InputStream bais = new ByteArrayInputStream(content.getBytes());
  139. byte[] buf = new byte[1024];
  140. while (true) {
  141. int len = bais.read(buf, 0, buf.length);
  142. if (len <= 0)
  143. break;
  144. out.write(buf, 0, len);
  145. }
  146. bais.close();
  147. // send '\0'
  148. buf[0] = 0;
  149. out.write(buf, 0, 1);
  150. out.flush();
  151. checkAck(in);
  152. out.close();
  153. channel.disconnect();
  154. logger.info("ScpTool.postTo end");
  155. } catch (Exception e) {
  156. logger.log(Level.SEVERE, e.getMessage(), e);
  157. throw e;
  158. }
  159. }
  160. public void disconnectSession() {
  161. int i = 1;
  162. disconnectSSHSession(i, this.sshSession);
  163. SSHSessionPool.getInstance().removeSession(server.getHostname());
  164. }
  165. public static void disconnectSSHSession(int i, Session sshSession) {
  166. while (sshSession != null && sshSession.isConnected() && i < 100) {
  167. sshSession.disconnect();
  168. try {
  169. Thread.sleep(50);
  170. } catch (InterruptedException ignored) {
  171. }
  172. i++;
  173. }
  174. }
  175. /**
  176. * Checks if the server accepts a connection
  177. *
  178. * @param in
  179. * @return 0 for success
  180. * 1 for an error
  181. * 2 for a fatal error
  182. * @throws IOException
  183. */
  184. private static int checkAck(InputStream in) throws IOException {
  185. int b = in.read();
  186. // b may be 0 for success,
  187. // 1 for error,
  188. // 2 for fatal error,
  189. // -1
  190. if (b == 0 || b == -1)
  191. return b;
  192. CheckAcceptConnection(in, b, logger);
  193. return b;
  194. }
  195. public static void CheckAcceptConnection(InputStream in, int b, Logger logger) throws IOException {
  196. if (b == 1 || b == 2) {
  197. StringBuilder sb = new StringBuilder();
  198. int c;
  199. c = in.read();
  200. sb.append((char) c);
  201. while (c != '\n') {
  202. c = in.read();
  203. sb.append((char) c);
  204. }
  205. if (b == 1) {// error
  206. logger.severe("Error: " + sb.toString());
  207. throw new IOException(sb.toString());
  208. }// fatal error
  209. logger.severe("Fatal Error:" + sb.toString());
  210. throw new IOException(sb.toString());
  211. }
  212. }
  213. }