/TextMash/src/pl/olek/clojure/Remote.java

http://textmash.googlecode.com/ · Java · 222 lines · 151 code · 28 blank · 43 comment · 21 complexity · c26294df2ac99d17d269cb69b2fc9fee MD5 · raw file

  1. /**
  2. * TextMash - simple IDE for Clojure
  3. *
  4. * Copyright (C) 2010 Aleksander Naszko
  5. *
  6. * This program is free software: you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation, either version 3 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  18. */
  19. package pl.olek.clojure;
  20. import java.io.DataInputStream;
  21. import java.io.DataOutputStream;
  22. import java.io.EOFException;
  23. import java.io.File;
  24. import java.io.IOException;
  25. import java.io.OutputStream;
  26. import java.net.InetAddress;
  27. import java.net.ServerSocket;
  28. import java.net.Socket;
  29. import pl.olek.jruce.Daemon;
  30. import pl.olek.jruce.GroupSender;
  31. import pl.olek.textmash.Support;
  32. /**
  33. * @author anaszko
  34. */
  35. public class Remote {
  36. final static int PUT_INPUT = 2;
  37. final static int RUN_PROCESS = 3;
  38. final static int DESTROY_PROCESS = 4;
  39. // final static int TERMINATE_CONNECTOR = 5;
  40. Socket server;
  41. DataOutputStream dataOutput;
  42. public static String getPath(String path) {
  43. path = path.replaceAll("\\\\", "/");
  44. Configuration settings =
  45. new Configuration(Support.getInstance().getId());
  46. String projectPath = settings.get("project-classpath")[0];
  47. for(String customPath : projectPath.split(System.getProperty( "path.separator" ))) {
  48. // System.out.println("Checking " + path + " and " + customPath);
  49. if (path.startsWith(customPath)) {
  50. path = path.substring(customPath.length());
  51. }
  52. }
  53. if (path.startsWith("/")) {
  54. path = path.substring(1);
  55. }
  56. // System.out.println("After all " + path);
  57. return path;
  58. }
  59. public Remote(String host, int port) {
  60. try {
  61. InetAddress addr = InetAddress.getByName(host);
  62. server = new Socket(addr, port);
  63. dataOutput = new DataOutputStream(server.getOutputStream());
  64. } catch (Exception e) {
  65. throw new RuntimeException(e);
  66. }
  67. }
  68. public void run(String workingDir, String[] cmd) {
  69. try {
  70. dataOutput.writeInt(RUN_PROCESS);
  71. dataOutput.writeUTF(workingDir);
  72. dataOutput.writeInt(cmd.length);
  73. for (String s : cmd) {
  74. dataOutput.writeUTF(s);
  75. }
  76. dataOutput.flush();
  77. } catch (Exception e) {
  78. throw new RuntimeException(e);
  79. }
  80. }
  81. public void input(String cmd) {
  82. try {
  83. dataOutput.writeInt(PUT_INPUT);
  84. dataOutput.writeUTF(cmd);
  85. dataOutput.flush();
  86. } catch (Exception e) {
  87. throw new RuntimeException(e);
  88. }
  89. }
  90. public void destroy() {
  91. try {
  92. dataOutput.writeInt(DESTROY_PROCESS);
  93. dataOutput.flush();
  94. } catch (Exception e) {
  95. throw new RuntimeException(e);
  96. }
  97. }
  98. // public void terminate() {
  99. // try {
  100. // dataOutput.writeInt(TERMINATE_CONNECTOR);
  101. // dataOutput.flush();
  102. // } catch (Exception e) {
  103. // throw new RuntimeException(e);
  104. // }
  105. // }
  106. public void close() {
  107. try {
  108. dataOutput.close();
  109. server.close();
  110. } catch (Exception e) {
  111. throw new RuntimeException(e);
  112. }
  113. }
  114. static Process repl = null;
  115. public static void spawn(int port, final String name) {
  116. try {
  117. final ServerSocket server = new ServerSocket(port);
  118. GroupSender gp = new GroupSender("textmash", 5000) {
  119. @Override
  120. public void send(OutputStream output) throws Exception {
  121. DataOutputStream dat = new DataOutputStream(output);
  122. dat.writeUTF(name);
  123. dat.writeInt(server.getLocalPort());
  124. dat.flush();
  125. }
  126. };
  127. gp.fire();
  128. Socket cl;
  129. boolean running = true;
  130. while ((running) && (cl = server.accept()) != null) {
  131. final Socket client = cl;
  132. new Daemon() {
  133. // Process repl;
  134. @Override
  135. public void run() {
  136. // TODO Auto-generated method stub
  137. try {
  138. DataInputStream reader = new DataInputStream(client
  139. .getInputStream());
  140. int msg;
  141. try {
  142. for (;;) {
  143. msg = reader.readInt();
  144. if (msg == RUN_PROCESS) {
  145. String workingDir = reader.readUTF();
  146. String[] cmds = new String[reader
  147. .readInt()];
  148. for (int i = 0; i < cmds.length; ++i) {
  149. cmds[i] = reader.readUTF();
  150. }
  151. if (repl != null) {
  152. repl.destroy();
  153. }
  154. repl = new Process(
  155. new File(workingDir), cmds);
  156. // System.out.println("Working dir " + System.getProperties());
  157. // System.out.println("Working dir " + Arrays.toString(cmds));
  158. Stream.redirect(repl.getInput(),
  159. System.out);
  160. } else if (repl != null) {
  161. if (msg == PUT_INPUT) {
  162. repl
  163. .getOutput()
  164. .write(
  165. reader.readUTF()
  166. .getBytes());
  167. repl.getOutput().flush();
  168. } else if (msg == DESTROY_PROCESS) {
  169. repl.destroy();
  170. }
  171. }
  172. // else if (msg == TERMINATE_CONNECTOR) {
  173. // server.close();
  174. // gp.stop();
  175. // running = false;
  176. // break;
  177. // }
  178. }
  179. } catch (EOFException e) {
  180. }
  181. } catch (IOException e) {
  182. }
  183. }
  184. };
  185. }
  186. } catch (Exception e) {
  187. throw new RuntimeException(e);
  188. }
  189. }
  190. public static void main(final String[] args) {
  191. // port, id
  192. Remote.spawn(Integer.parseInt(args[0]), args[1]);
  193. }
  194. }