Free cookie consent management tool by TermsFeed Policy Generator

source: branches/Robocode.TrunkInt/HeuristicLab.Problems.Robocode/3.3/Interpreter.cs @ 9844

Last change on this file since 9844 was 9790, checked in by ascheibe, 11 years ago

#2069

  • added license headers
  • corrected version information
  • fixed formatting
File size: 19.6 KB
RevLine 
[9790]1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2013 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
4 *
5 * This file is part of HeuristicLab.
6 *
7 * HeuristicLab is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * HeuristicLab is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
19 */
20#endregion
21
22using System;
[9601]23using System.Diagnostics;
24using System.IO;
[9565]25using System.Linq;
26using System.Text;
27using System.Xml;
28using HeuristicLab.Core;
[9601]29using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
[9565]30
[9601]31namespace HeuristicLab.Problems.Robocode {
32  public class Interpreter {
[9565]33
[9601]34    private struct TankStats {
35      public int moves, shots;
36      public TankStats(int moves, int shots) {
37        this.moves = moves;
38        this.shots = shots;
39      }
40    }
41    static string chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZ";
[9565]42
[9601]43    public static string BuildTankProgram(ISymbolicExpressionTree tree) {
44      XmlDocument solutionCandidateRoot = new XmlDocument();
45      solutionCandidateRoot.Load("../tank.xml");
[9565]46
[9601]47      TankStats tankStats = EvaluateTankProgram(tree.Root, solutionCandidateRoot.FirstChild, null);
48      return solutionCandidateRoot.InnerText;
49    }
50
51    public static double EvaluateTankProgram(ISymbolicExpressionTree tree, XmlNode solutionCandidateRoot, string path) {
52      //TankStats tankStats = EvaluateTankProgram(tree.Root, solutionCandidateRoot, null);
53      //Evaluator.tankStatLog += "\r\n" + tankStats.shots + " / " + tankStats.moves;
54      string interpretedProgram = InterpretProgramTree(tree.Root);
55      Random random = new Random();
56      string formattedPath = path.Replace("/", "\\");
57      string outputname = "";
58      string output = "";
59      bool noError = true;
60      do {
61        try {
62          outputname = new string(Enumerable.Repeat(chars, 8).Select(s => s[random.Next(s.Length)]).ToArray());
63          output = /*solutionCandidateRoot.InnerText*/interpretedProgram.Replace("class output", "class " + outputname);
64          File.WriteAllText(path + "/robots/Evaluation/" + outputname + ".java", output, System.Text.Encoding.Default);
65          noError = true;
[9565]66        }
[9601]67        catch (Exception e) { noError = false; }
68      } while (!noError);
[9565]69
[9601]70      //string path = System.IO.Path.GetDirectoryName(System.Reflection.Assembly.GetExecutingAssembly().Location);
71      //path.Replace("\\", "\\\\");
72      //path.Replace(" ", "\\ ");
73      #region Compile code file
[9565]74
[9601]75      ProcessStartInfo javaCompileInfo = new ProcessStartInfo();
76      javaCompileInfo.FileName = "cmd.exe";
77      javaCompileInfo.Arguments = "/C javac -cp " + formattedPath + "\\libs\\robocode.jar " + formattedPath + "\\robots\\Evaluation\\" + outputname + ".java";
78      javaCompileInfo.RedirectStandardOutput = true;
79      javaCompileInfo.RedirectStandardError = true;
80      javaCompileInfo.UseShellExecute = false;
81      javaCompileInfo.CreateNoWindow = true;
[9565]82
[9601]83      Process javaCompile = new Process();
84      javaCompile.StartInfo = javaCompileInfo;
85      javaCompile.Start();
[9565]86
[9601]87      /*Evaluator.tankStatLog = "=====================================================\r\n";
88      Evaluator.tankStatLog += "Tree Compilation Output: \r\n";
89      Evaluator.tankStatLog += "=====================================================\r\n";*/
[9565]90
[9601]91      string cmdOutput = javaCompile.StandardOutput.ReadToEnd();
92      cmdOutput += javaCompile.StandardError.ReadToEnd();
[9612]93
[9601]94      //Evaluator.tankStatLog += cmdOutput;
95      //Console.WriteLine(cmdOutput);
96      //Console.ReadLine();
[9565]97
[9601]98      /*Evaluator.tankStatLog = "=====================================================\r\n";
99      Evaluator.tankStatLog += "End of Tree Compilation Output: \r\n";
100      Evaluator.tankStatLog += "=====================================================\r\n";*/
[9565]101
[9601]102      javaCompile.WaitForExit();
[9612]103      if (javaCompile.ExitCode != 0) {
104        return -1000.0;
105      }
[9601]106      #endregion
[9565]107
108
[9601]109      try { File.Delete(path + "/robots/robot.database"); }
110      catch (Exception e) { }
[9565]111
[9601]112      #region evaluate code
[9565]113
[9601]114      ProcessStartInfo evaluateCodeInfo = new ProcessStartInfo();
115      evaluateCodeInfo.FileName = "cmd.exe";
116      //javaCompileInfo.Arguments = "/C javac -cp C:\\robocode\\libs\\robocode.jar \"" + path + "\\Spaced Up\\output.java\"";
117      evaluateCodeInfo.Arguments = "/C java -classpath " + formattedPath + "\\libs;" + formattedPath + "\\libs\\robocode.core-1.8.1.0.jar;" + formattedPath + "\\libs\\robocode.jar;" + formattedPath + "\\libs\\picocontainer-2.14.2.jar BattleRunner Evaluation." + outputname + "* false";
118      //Console.WriteLine(javaCompileInfo.Arguments);
119      evaluateCodeInfo.RedirectStandardOutput = true;
120      evaluateCodeInfo.RedirectStandardError = true;
121      evaluateCodeInfo.UseShellExecute = false;
122      evaluateCodeInfo.CreateNoWindow = true;
[9565]123
[9601]124      Process evaluateCode = new Process();
125      evaluateCode.StartInfo = evaluateCodeInfo;
126      evaluateCode.Start();
[9565]127
[9601]128      StringBuilder q = new StringBuilder();
129      string scoreString = "";
[9612]130      evaluateCode.WaitForExit();
131
132      try {
[9601]133        scoreString = evaluateCode.StandardOutput.ReadToEnd().Split(
134        new string[] { "\r\n" }, StringSplitOptions.RemoveEmptyEntries).Last();
135        q.Append("Output: " + scoreString + "\r\n");
136        q.Append("Error: " + evaluateCode.StandardError.ReadToEnd() + "\r\n");
[9612]137        cmdOutput += q.ToString();
[9790]138      }
139      catch {
[9612]140        return -1000.0;
[9601]141      }
[9565]142
[9601]143      #endregion
[9565]144
[9601]145      double evaluation = -900;
146      try {
147        File.Delete(path + "/robots/Evaluation/" + outputname + ".java");
148        File.Delete(path + "/robots/Evaluation/" + outputname + ".class");
149        evaluation = Double.Parse(scoreString);
150      }
151      catch (Exception e) { evaluation = -900; }
152      Random rand = new Random();
153      evaluation += rand.NextDouble() / 100;
[9565]154
[9601]155      /*Evaluator.tankStatLog = "=====================================================\r\n";
156      Evaluator.tankStatLog += "START OF TREE - Score: " + evaluation + " (" + tankStats.shots + "/" + tankStats.moves + ")" + "\r\n" + " Compilation Output: " + cmdOutput + "\r\n";
157      Evaluator.tankStatLog += "=====================================================\r\n";
[9565]158
[9601]159      Evaluator.tankStatLog += solutionCandidateRoot.InnerText + "\r\n";
[9565]160
[9601]161      Evaluator.tankStatLog += "=====================================================\r\n";
162      Evaluator.tankStatLog += "END OF TREE\r\n";
163      Evaluator.tankStatLog += "=====================================================\r\n";
[9565]164
[9601]165      File.AppendAllText("../TankLog.txt", Evaluator.tankStatLog);*/
166      return evaluation;
167    }
[9565]168
[9601]169    public static double EvaluateTankProgram(ISymbolicExpressionTree tree, ScopeList coevolutionIndividuals, XmlNode solutionCandidateRoot, string path) {
170      Random random = new Random();
171      string formattedPath = path.Replace("/", "\\");
[9565]172
[9601]173      // Interpret and compile main tree
174      //XmlNode mainTreeNode = solutionCandidateRoot.Clone();
175      //EvaluateTankProgram(tree.Root, mainTreeNode, null);
176      string interpretedMainTree = InterpretProgramTree(tree.Root);
177      string mainTreeName = writeTreeToFile(interpretedMainTree, path, random);
178      compileCodefile(/*mainTreeName,*/ mainTreeName, formattedPath);
[9565]179
[9601]180      // Interpret and compile coevolutionIndividuals
181      String[] coevolutionIndividualsNames = new String[coevolutionIndividuals.Count];
182      for (int i = 0; i < coevolutionIndividuals.Count; i++) {
183        //XmlNode individualTreeNode = solutionCandidateRoot.Clone();
184        //EvaluateTankProgram(((ISymbolicExpressionTree)coevolutionIndividuals[i].Variables.ToArray()[0].Value).Root, individualTreeNode, null);
185        string interpretedIndividual = InterpretProgramTree(((ISymbolicExpressionTree)coevolutionIndividuals[i].Variables.ToArray()[0].Value).Root);
186        coevolutionIndividualsNames[i] = writeTreeToFile(/*individualTreeNode,*/ interpretedIndividual, path, random);
187        compileCodefile(coevolutionIndividualsNames[i], formattedPath);
188      }
[9565]189
[9601]190      // Force Robocode to update its robot database by deleting the existing one
191      try { File.Delete(path + "/robots/robot.database"); }
192      catch (Exception e) { }
[9565]193
[9601]194      // Run evaluator
195      double evaluation = runEvaluator(mainTreeName, coevolutionIndividualsNames, formattedPath);
196      try {
197        File.Delete(path + "/robots/Evaluation/" + mainTreeName + ".java");
198        File.Delete(path + "/robots/Evaluation/" + mainTreeName + ".class");
199        foreach (string s in coevolutionIndividualsNames) {
200          File.Delete(path + "/robots/Evaluation/" + s + ".java");
201          File.Delete(path + "/robots/Evaluation/" + s + ".class");
[9565]202        }
[9601]203      }
204      catch (Exception e) { }
205      Random rand = new Random();
206      evaluation += rand.NextDouble() / 100;
[9565]207
[9601]208      return evaluation;
209    }
[9565]210
[9601]211    private static string writeTreeToFile(/*XmlNode node,*/ string interpretedTree, string path, Random random) {
212      string outputName = "";
213      string outputString = "";
214      bool noError = true;
215      do {
216        try {
217          outputName = new string(Enumerable.Repeat(chars, 8).Select(s => s[random.Next(s.Length)]).ToArray());
218          outputString = /*node.InnerText*/interpretedTree.Replace("class output", "class " + outputName);
219          File.WriteAllText(path + "/robots/Evaluation/" + outputName + ".java", outputString, System.Text.Encoding.Default);
220          noError = true;
[9565]221        }
[9601]222        catch (Exception e) { noError = false; }
223      } while (!noError);
[9565]224
[9601]225      return outputName;
226    }
[9565]227
[9601]228    private static bool compileCodefile(string filename, string formattedPath) {
229      ProcessStartInfo javaCompileInfo = new ProcessStartInfo();
230      javaCompileInfo.FileName = "cmd.exe";
231      javaCompileInfo.Arguments = "/C javac -cp " + formattedPath + "\\libs\\robocode.jar " + formattedPath + "\\robots\\Evaluation\\" + filename + ".java";
232      javaCompileInfo.RedirectStandardOutput = true;
233      javaCompileInfo.RedirectStandardError = true;
234      javaCompileInfo.UseShellExecute = false;
235      javaCompileInfo.CreateNoWindow = true;
[9565]236
[9601]237      Process javaCompile = new Process();
238      javaCompile.StartInfo = javaCompileInfo;
239      javaCompile.Start();
[9565]240
[9601]241      string cmdOutput = javaCompile.StandardOutput.ReadToEnd();
242      cmdOutput += javaCompile.StandardError.ReadToEnd();
[9565]243
[9601]244      javaCompile.WaitForExit();
[9565]245
[9601]246      if (cmdOutput.Equals(""))
247        return true;
248      return false;
249    }
[9565]250
[9601]251    private static double runEvaluator(string mainTreeFilename, string[] coevolutionIndividualsNames, string formattedPath) {
252      ProcessStartInfo evaluateCodeInfo = new ProcessStartInfo();
253      evaluateCodeInfo.FileName = "cmd.exe";
254      //javaCompileInfo.Arguments = "/C javac -cp C:\\robocode\\libs\\robocode.jar \"" + path + "\\Spaced Up\\output.java\"";
255      evaluateCodeInfo.Arguments = "/C java -classpath " + formattedPath + "\\libs;" + formattedPath + "\\libs\\robocode.core-1.8.1.0.jar;" + formattedPath + "\\libs\\robocode.jar;" + formattedPath + "\\libs\\picocontainer-2.14.2.jar BattleRunner Evaluation." + mainTreeFilename + "* ";
256      if (coevolutionIndividualsNames != null) {
257        foreach (string s in coevolutionIndividualsNames)
258          evaluateCodeInfo.Arguments += "Evaluation." + s + "* ";
259      } else
260        evaluateCodeInfo.Arguments += "false";
261      //Console.WriteLine(javaCompileInfo.Arguments);
262      evaluateCodeInfo.RedirectStandardOutput = true;
263      evaluateCodeInfo.RedirectStandardError = true;
264      evaluateCodeInfo.UseShellExecute = false;
265      evaluateCodeInfo.CreateNoWindow = true;
[9565]266
[9601]267      Process evaluateCode = new Process();
268      evaluateCode.StartInfo = evaluateCodeInfo;
269      evaluateCode.Start();
[9565]270
[9601]271      StringBuilder q = new StringBuilder();
272      string scoreString = "";
273      while (!evaluateCode.HasExited) {
274        scoreString = evaluateCode.StandardOutput.ReadToEnd().Split(
275        new string[] { "\r\n" }, StringSplitOptions.RemoveEmptyEntries).Last();
276        q.Append("Output: " + scoreString + "\r\n");
277        q.Append("Error: " + evaluateCode.StandardError.ReadToEnd() + "\r\n");
278      }
279      string cmdOutput = q.ToString();
280      try {
281        double evaluation = Double.Parse(scoreString);
282        return evaluation;
283      }
284      catch (Exception e) { }
[9565]285
[9601]286      return -900;
287    }
[9565]288
[9601]289    private static TankStats EvaluateTankProgram(ISymbolicExpressionTreeNode node, XmlNode docNode, string method) {
290      TankStats tankStats = new TankStats(0, 0);
[9565]291
[9601]292      //string log = "Doing node: \r\n";
293      /*if (node.Symbol is ProgramRootSymbol ||
294          node.Symbol is StartSymbol)
295          tankStats = EvaluateTankProgram(node.GetSubtree(0), ref docNode, null);
296      else*/
297      if (node.Symbol is Constant)
298        docNode[method]["code"].InnerText += " " + ((ConstantTreeNode)node).Value + " ";
299      else if (node.Symbol is ShotPower)
300        docNode[method]["code"].InnerText += " " + ((ShotPowerTreeNode)node).Value + " ";
301      else if (node.Symbol is LogicalValue)
302        docNode[method]["code"].InnerText += " " + ((BooleanTreeNode)node).Value.ToString().ToLower() + " ";
303      else if (node.Symbol is Block) {
304        docNode[method]["code"].InnerText += "{\r\n";
305        foreach (ISymbolicExpressionTreeNode n in node.Subtrees)
306          EvaluateTankProgram(n, docNode, method);
307        docNode[method]["code"].InnerText += "}\r\n";
308      } else if (node.Symbol is IfStatement) {
309        docNode[method]["code"].InnerText += "if(";
310        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
311        docNode[method]["code"].InnerText += ")\r\n";
312        EvaluateTankProgram(node.GetSubtree(1), docNode, method);
313        if (node.SubtreeCount == 3)
314          EvaluateTankProgram(node.GetSubtree(2), docNode, method);
315      } else if (node.Symbol is ElseStatement) {
316        docNode[method]["code"].InnerText += "else\r\n";
317        if (node.SubtreeCount == 1)
318          EvaluateTankProgram(node.GetSubtree(0), docNode, method);
319      } else if (node.Symbol is WhileLoop) {
320        docNode[method]["code"].InnerText += "While(";
321        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
322        docNode[method]["code"].InnerText += ")\r\n";
323        EvaluateTankProgram(node.GetSubtree(1), docNode, method);
324      } else if (node.Symbol is Ahead) {
325        docNode[method]["code"].InnerText += "setAhead(";
326        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
327        docNode[method]["code"].InnerText += ");\r\n";
328        tankStats.moves = 1;
329      } else if (node.Symbol is Back) {
330        docNode[method]["code"].InnerText += "setBack(";
331        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
332        docNode[method]["code"].InnerText += ");\r\n";
333        tankStats.moves = 1;
334      } else if (node.Symbol is SetAdjustGunForRobotTurn) {
335        docNode[method]["code"].InnerText += "setAdjustGunForRobotTurn(";
336        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
337        docNode[method]["code"].InnerText += ");\r\n";
338        tankStats.moves = 1;
339      } else if (node.Symbol is SetAdjustRadarForGunTurn) {
340        docNode[method]["code"].InnerText += "setAdjustRadarForGunTurn(";
341        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
342        docNode[method]["code"].InnerText += ");\r\n";
343        tankStats.moves = 1;
344      } else if (node.Symbol is SetAdjustRadarForRobotTurn) {
345        docNode[method]["code"].InnerText += "setAdjustRadarForRobotTurn(";
346        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
347        docNode[method]["code"].InnerText += ");\r\n";
348        tankStats.moves = 1;
349      } else if (node.Symbol is TurnGunLeft) {
350        docNode[method]["code"].InnerText += "setTurnGunLeft(";
351        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
352        docNode[method]["code"].InnerText += ");\r\n";
353        tankStats.moves = 1;
354      } else if (node.Symbol is TurnGunRight) {
355        docNode[method]["code"].InnerText += "setTurnGunRight(";
356        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
357        docNode[method]["code"].InnerText += ");\r\n";
358        tankStats.moves = 1;
359      } else if (node.Symbol is TurnLeft) {
360        docNode[method]["code"].InnerText += "setTurnLeft(";
361        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
362        docNode[method]["code"].InnerText += ");\r\n";
363        tankStats.moves = 1;
364      } else if (node.Symbol is TurnRadarLeft) {
365        docNode[method]["code"].InnerText += "setTurnRadarLeft(";
366        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
367        docNode[method]["code"].InnerText += ");\r\n";
368        tankStats.moves = 1;
369      } else if (node.Symbol is TurnRadarRight) {
370        docNode[method]["code"].InnerText += "setTurnRadarRight(";
371        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
372        docNode[method]["code"].InnerText += ");\r\n";
373        tankStats.moves = 1;
374      } else if (node.Symbol is TurnRight) {
375        docNode[method]["code"].InnerText += "setTurnRight(";
376        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
377        docNode[method]["code"].InnerText += ");\r\n";
378        tankStats.moves = 1;
379      } else if (node.Symbol is Fire) {
380        docNode[method]["code"].InnerText += "setFire(";
381        EvaluateTankProgram(node.GetSubtree(0), docNode, method);
382        docNode[method]["code"].InnerText += ");\r\n";
383        tankStats.shots = 1;
384      } else if (node.Symbol is Run ||
385            node.Symbol is OnBulletHit ||
386            node.Symbol is OnBulletMissed ||
387            node.Symbol is OnHitByBullet ||
388            node.Symbol is OnHitRobot ||
389            node.Symbol is OnHitWall ||
390            node.Symbol is OnScannedRobot) {
391        tankStats = new TankStats();
392        foreach (SymbolicExpressionTreeNode n in node.Subtrees) {
393          string methodName = node.Symbol.GetType().ToString().Split('.').Last();
394          try {
395            var tempStats = EvaluateTankProgram(n, docNode, methodName);
396            tankStats.moves += tempStats.moves;
397            tankStats.shots += tempStats.shots;
398          }
399          catch (Exception e) {
400            //log += "NULL EXCEPTION ON:::::::: -" + methodName + "-\r\n";
401            //foreach (XmlNode f in docNode[methodName])
402            //    log += f.Name + "\r\n";
403          }
[9565]404        }
[9601]405      } else if (node.Symbol is Tank) {
406        foreach (SymbolicExpressionTreeNode n in node.Subtrees) {
407          try {
408            var tempStats = EvaluateTankProgram(n, docNode, method);
409            tankStats.moves += tempStats.moves;
410            tankStats.shots += tempStats.shots;
411          }
412          catch (Exception e) { }
413        }
414      } else
415        tankStats = EvaluateTankProgram(node.GetSubtree(0), docNode, null);
[9565]416
[9601]417      //log += tankStats.shots + " / " + tankStats.moves + "\r\n";
418      //File.AppendAllText("../TankLog.txt", log);
419
420      return tankStats;
[9565]421    }
[9601]422
423    public static string InterpretProgramTree(ISymbolicExpressionTreeNode node) {
424      var tankNode = node;
425      while (!(tankNode.Symbol is Tank))
426        tankNode = tankNode.GetSubtree(0);
427      return ((CodeNode)tankNode.Symbol).Interpret(tankNode, tankNode.Subtrees);
428    }
429  }
[9565]430}
Note: See TracBrowser for help on using the repository browser.