Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/sources/HeuristicLab.Functions/ProgrammableFunction.cs @ 159

Last change on this file since 159 was 155, checked in by gkronber, 17 years ago

merged FunctionsAndStructIdRefactoring-branch (r142, r143, r144, r145, r146, r147, r148, r149, r152, r153) back into the trunk (ticket #112)

File size: 8.0 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2008 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;
23using System.Collections.Generic;
24using System.Text;
25using HeuristicLab.Core;
26using System.Diagnostics;
27using HeuristicLab.Constraints;
28using HeuristicLab.DataAnalysis;
29using System.Xml;
30using System.Reflection;
31using System.CodeDom;
32using System.CodeDom.Compiler;
33using Microsoft.CSharp;
34using System.IO;
35
36namespace HeuristicLab.Functions {
37  public class ProgrammableFunction : FunctionBase {
38    private MethodInfo evaluateMethod;
39
40    private string myDescription;
41    public override string Description {
42      get { return myDescription; }
43    }
44    private string myCode;
45    public string Code {
46      get { return myCode; }
47      set {
48        if(value != myCode) {
49          myCode = value;
50          evaluateMethod = null;
51          OnCodeChanged();
52        }
53      }
54    }
55
56    public ProgrammableFunction() : base() {
57      myCode = "return 0.0;";
58      myDescription = "A function that can be programmed for arbitrary needs.";
59      evaluateMethod = null;
60    }
61
62    public void SetDescription(string description) {
63      if(description == null)
64        throw new NullReferenceException("description must not be null");
65
66      if(description != myDescription) {
67        myDescription = description;
68        OnDescriptionChanged();
69      }
70    }
71
72    public void Compile() {
73      CodeNamespace ns = new CodeNamespace("HeuristicLab.Functions.CustomFunctions");
74      CodeTypeDeclaration typeDecl = new CodeTypeDeclaration("Function");
75      typeDecl.IsClass = true;
76      typeDecl.TypeAttributes = TypeAttributes.Public;
77
78      CodeMemberMethod method = new CodeMemberMethod();
79      method.Name = "Evaluate";
80      method.ReturnType = new CodeTypeReference(typeof(double));
81      method.Attributes = MemberAttributes.Public | MemberAttributes.Static;
82      method.Parameters.Add(new CodeParameterDeclarationExpression(typeof(IFunction), "function"));
83      method.Parameters.Add(new CodeParameterDeclarationExpression(typeof(Dataset), "dataset"));
84      method.Parameters.Add(new CodeParameterDeclarationExpression(typeof(int), "index"));
85      foreach(IVariableInfo info in VariableInfos)
86        method.Parameters.Add(new CodeParameterDeclarationExpression(info.DataType, info.FormalName));
87      string code = myCode;
88      method.Statements.Add(new CodeSnippetStatement(code));
89      typeDecl.Members.Add(method);
90
91      ns.Types.Add(typeDecl);
92      ns.Imports.Add(new CodeNamespaceImport("System"));
93      ns.Imports.Add(new CodeNamespaceImport("System.Collections.Generic"));
94      ns.Imports.Add(new CodeNamespaceImport("System.Text"));
95      ns.Imports.Add(new CodeNamespaceImport("HeuristicLab.Core"));
96      ns.Imports.Add(new CodeNamespaceImport("HeuristicLab.Functions"));
97      foreach(IVariableInfo variableInfo in VariableInfos)
98        ns.Imports.Add(new CodeNamespaceImport(variableInfo.DataType.Namespace));
99
100      CodeCompileUnit unit = new CodeCompileUnit();
101      unit.Namespaces.Add(ns);
102      CompilerParameters parameters = new CompilerParameters();
103      parameters.GenerateExecutable = false;
104      parameters.GenerateInMemory = true;
105      parameters.IncludeDebugInformation = false;
106      Assembly[] loadedAssemblies = AppDomain.CurrentDomain.GetAssemblies();
107      foreach(Assembly loadedAssembly in loadedAssemblies)
108        parameters.ReferencedAssemblies.Add(loadedAssembly.Location);
109      CodeDomProvider provider = new CSharpCodeProvider();
110      CompilerResults results = provider.CompileAssemblyFromDom(parameters, unit);
111
112      evaluateMethod = null;
113      if(results.Errors.HasErrors) {
114        StringWriter writer = new StringWriter();
115        CodeGeneratorOptions options = new CodeGeneratorOptions();
116        options.BlankLinesBetweenMembers = false;
117        options.ElseOnClosing = true;
118        options.IndentString = "  ";
119        provider.GenerateCodeFromCompileUnit(unit, writer, options);
120        writer.Flush();
121        string[] source = writer.ToString().Split(new string[] { "\r\n" }, StringSplitOptions.None);
122        StringBuilder builder = new StringBuilder();
123        for(int i = 0; i < source.Length; i++)
124          builder.AppendLine((i + 1).ToString("###") + "     " + source[i]);
125        builder.AppendLine();
126        builder.AppendLine();
127        builder.AppendLine();
128        foreach(CompilerError error in results.Errors) {
129          builder.Append("Line " + error.Line.ToString());
130          builder.Append(", Column " + error.Column.ToString());
131          builder.AppendLine(": " + error.ErrorText);
132        }
133        throw new Exception("Compile Errors:\n\n" + builder.ToString());
134      } else {
135        Assembly assembly = results.CompiledAssembly;
136        Type[] types = assembly.GetTypes();
137        evaluateMethod = types[0].GetMethod("Evaluate");
138      }
139    }
140
141    public override double Apply(Dataset dataset, int sampleIndex, double[] args) {
142      //if(evaluateMethod == null) {
143      //  Compile();
144      //}
145
146      //// collect parameters
147      //object[] parameters = new object[VariableInfos.Count + 3];
148      //parameters[0] = this;
149      //parameters[1] = dataset;
150      //parameters[2] = sampleIndex;
151      //int i = 3;
152      //// all local variables are available in the custom function
153      //foreach(IVariableInfo info in VariableInfos) {
154      //  if(info.Local) {
155      //    parameters[i] = this.GetVariable(info.ActualName);
156      //    i++;
157      //  }
158      //}
159      //return (double)evaluateMethod.Invoke(null, parameters);
160      return 0.0;
161    }
162
163    public override object Clone(IDictionary<Guid, object> clonedObjects) {
164      ProgrammableFunction clone = (ProgrammableFunction)base.Clone(clonedObjects);
165      clone.myDescription = Description;
166      clone.myCode = Code;
167      clone.evaluateMethod = evaluateMethod;
168      return clone;
169    }
170
171    public override void Accept(IFunctionVisitor visitor) {
172      visitor.Visit(this);
173    }
174
175    public event EventHandler DescriptionChanged;
176    protected virtual void OnDescriptionChanged() {
177      if(DescriptionChanged != null)
178        DescriptionChanged(this, new EventArgs());
179    }
180    public event EventHandler CodeChanged;
181    protected virtual void OnCodeChanged() {
182      if(CodeChanged != null)
183        CodeChanged(this, new EventArgs());
184    }
185
186    #region Persistence Methods
187    public override XmlNode GetXmlNode(string name, XmlDocument document, IDictionary<Guid, IStorable> persistedObjects) {
188      XmlNode node = base.GetXmlNode(name, document, persistedObjects);
189      XmlNode descriptionNode = document.CreateNode(XmlNodeType.Element, "Description", null);
190      descriptionNode.InnerText = myDescription;
191      node.AppendChild(descriptionNode);
192      XmlNode codeNode = document.CreateNode(XmlNodeType.Element, "Code", null);
193      codeNode.InnerText = myCode;
194      node.AppendChild(codeNode);
195      return node;
196    }
197    public override void Populate(XmlNode node, IDictionary<Guid, IStorable> restoredObjects) {
198      base.Populate(node, restoredObjects);
199      XmlNode descriptionNode = node.SelectSingleNode("Description");
200      myDescription = descriptionNode.InnerText;
201      XmlNode codeNode = node.SelectSingleNode("Code");
202      myCode = codeNode.InnerText;
203    }
204    #endregion
205  }
206}
Note: See TracBrowser for help on using the repository browser.