| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606 |
- //
- // System.Web.Compilation.AssemblyBuilder
- //
- // Authors:
- // Chris Toshok ([email protected])
- // Gonzalo Paniagua Javier ([email protected])
- // Marek Habersack ([email protected])
- //
- // (C) 2006-2008 Novell, Inc (http://www.novell.com)
- //
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- #if NET_2_0
- using System;
- using System.CodeDom;
- using System.CodeDom.Compiler;
- using System.Collections.Generic;
- using System.Collections.Specialized;
- using System.IO;
- using System.Reflection;
- using System.Web.Configuration;
- using System.Web.Util;
- using System.Web.Hosting;
- namespace System.Web.Compilation {
- internal class CompileUnitPartialType
- {
- public readonly CodeCompileUnit Unit;
- public readonly CodeNamespace ParentNamespace;
- public readonly CodeTypeDeclaration PartialType;
- string typeName;
-
- public string TypeName {
- get {
- if (typeName == null) {
- if (ParentNamespace == null || PartialType == null)
- return null;
-
- typeName = ParentNamespace.Name;
- if (String.IsNullOrEmpty (typeName))
- typeName = PartialType.Name;
- else
- typeName += "." + PartialType.Name;
- }
- return typeName;
- }
- }
-
- public CompileUnitPartialType (CodeCompileUnit unit, CodeNamespace parentNamespace, CodeTypeDeclaration type)
- {
- this.Unit = unit;
- this.ParentNamespace = parentNamespace;
- this.PartialType = type;
- }
- }
-
- public class AssemblyBuilder {
- const string DEFAULT_ASSEMBLY_BASE_NAME = "App_Web_";
- const int COPY_BUFFER_SIZE = 8192;
-
- static bool KeepFiles = (Environment.GetEnvironmentVariable ("MONO_ASPNET_NODELETE") != null);
-
- CodeDomProvider provider;
- CompilerParameters parameters;
- Dictionary <string, bool> code_files;
- Dictionary <string, List <CompileUnitPartialType>> partial_types;
- List <CodeCompileUnit> units;
- List <string> source_files;
- List <Assembly> referenced_assemblies;
- Dictionary <string, string> resource_files;
- TempFileCollection temp_files;
- string outputFilesPrefix;
- string outputAssemblyPrefix;
- string outputAssemblyName;
-
- internal AssemblyBuilder (CodeDomProvider provider)
- : this (null, provider, DEFAULT_ASSEMBLY_BASE_NAME)
- {}
- internal AssemblyBuilder (CodeDomProvider provider, string assemblyBaseName)
- : this (null, provider, assemblyBaseName)
- {}
- internal AssemblyBuilder (VirtualPath virtualPath, CodeDomProvider provider)
- : this (virtualPath, provider, DEFAULT_ASSEMBLY_BASE_NAME)
- {}
-
- internal AssemblyBuilder (VirtualPath virtualPath, CodeDomProvider provider, string assemblyBaseName)
- {
- this.provider = provider;
- this.outputFilesPrefix = assemblyBaseName ?? DEFAULT_ASSEMBLY_BASE_NAME;
-
- units = new List <CodeCompileUnit> ();
- CompilationSection section;
- if (virtualPath != null)
- section = (CompilationSection) WebConfigurationManager.GetSection ("system.web/compilation", virtualPath.Absolute);
- else
- section = (CompilationSection) WebConfigurationManager.GetSection ("system.web/compilation");
- string tempdir = section.TempDirectory;
- if (String.IsNullOrEmpty (tempdir))
- tempdir = AppDomain.CurrentDomain.SetupInformation.DynamicBase;
- if (!KeepFiles)
- KeepFiles = section.Debug;
-
- temp_files = new TempFileCollection (tempdir, KeepFiles);
- }
- internal string OutputFilesPrefix {
- get {
- if (outputFilesPrefix == null)
- outputFilesPrefix = DEFAULT_ASSEMBLY_BASE_NAME;
- return outputFilesPrefix;
- }
- set {
- if (String.IsNullOrEmpty (value))
- outputFilesPrefix = DEFAULT_ASSEMBLY_BASE_NAME;
- else
- outputFilesPrefix = value;
- outputAssemblyPrefix = null;
- outputAssemblyName = null;
- }
- }
-
- internal string OutputAssemblyPrefix {
- get {
- if (outputAssemblyPrefix == null) {
- string basePath = temp_files.BasePath;
- string baseName = Path.GetFileName (basePath);
- string baseDir = Path.GetDirectoryName (basePath);
- outputAssemblyPrefix = Path.Combine (baseDir, String.Concat (OutputFilesPrefix, baseName));
- }
- return outputAssemblyPrefix;
- }
- }
- internal string OutputAssemblyName {
- get {
- if (outputAssemblyName == null)
- outputAssemblyName = OutputAssemblyPrefix + ".dll";
- return outputAssemblyName;
- }
- }
-
- internal TempFileCollection TempFiles {
- get { return temp_files; }
- }
- internal CompilerParameters CompilerOptions {
- get { return parameters; }
- set { parameters = value; }
- }
-
- internal CodeCompileUnit [] GetUnitsAsArray ()
- {
- CodeCompileUnit [] result = new CodeCompileUnit [units.Count];
- units.CopyTo (result, 0);
- return result;
- }
- internal List <CodeCompileUnit> Units {
- get {
- if (units == null)
- units = new List <CodeCompileUnit> ();
- return units;
- }
- }
-
- internal Dictionary <string, List <CompileUnitPartialType>> PartialTypes {
- get {
- if (partial_types == null)
- partial_types = new Dictionary <string, List <CompileUnitPartialType>> ();
- return partial_types;
- }
- }
-
- Dictionary <string, bool> CodeFiles {
- get {
- if (code_files == null)
- code_files = new Dictionary <string, bool> ();
- return code_files;
- }
- }
-
- List <string> SourceFiles {
- get {
- if (source_files == null)
- source_files = new List <string> ();
- return source_files;
- }
- }
- Dictionary <string, string> ResourceFiles {
- get {
- if (resource_files == null)
- resource_files = new Dictionary <string, string> ();
- return resource_files;
- }
- }
- public void AddAssemblyReference (Assembly a)
- {
- if (a == null)
- throw new ArgumentNullException ("a");
- List <Assembly> assemblies = ReferencedAssemblies;
-
- if (assemblies.Contains (a))
- return;
-
- assemblies.Add (a);
- }
- internal void AddAssemblyReference (string assemblyLocation)
- {
- try {
- Assembly asm = Assembly.LoadFrom (assemblyLocation);
- if (asm == null)
- return;
- AddAssemblyReference (asm);
- } catch {
- // ignore, it will come up later
- }
- }
-
- internal void AddAssemblyReference (List <Assembly> asmlist)
- {
- if (asmlist == null)
- return;
-
- foreach (Assembly a in asmlist) {
- if (a == null)
- continue;
- AddAssemblyReference (a);
- }
- }
-
- internal void AddCodeCompileUnit (CodeCompileUnit compileUnit)
- {
- if (compileUnit == null)
- throw new ArgumentNullException ("compileUnit");
- units.Add (CheckForPartialTypes (compileUnit));
- }
-
- public void AddCodeCompileUnit (BuildProvider buildProvider, CodeCompileUnit compileUnit)
- {
- if (buildProvider == null)
- throw new ArgumentNullException ("buildProvider");
- if (compileUnit == null)
- throw new ArgumentNullException ("compileUnit");
- units.Add (CheckForPartialTypes (compileUnit));
- }
- public TextWriter CreateCodeFile (BuildProvider buildProvider)
- {
- if (buildProvider == null)
- throw new ArgumentNullException ("buildProvider");
- // Generate a file name with the correct source language extension
- string filename = GetTempFilePhysicalPath (provider.FileExtension);
- SourceFiles.Add (filename);
- return new StreamWriter (File.OpenWrite (filename));
- }
- internal void AddCodeFile (string path)
- {
- AddCodeFile (path, null, false);
- }
- internal void AddCodeFile (string path, BuildProvider bp)
- {
- AddCodeFile (path, bp, false);
- }
-
- internal void AddCodeFile (string path, BuildProvider bp, bool isVirtual)
- {
- if (String.IsNullOrEmpty (path))
- return;
- Dictionary <string, bool> codeFiles = CodeFiles;
- if (codeFiles.ContainsKey (path))
- return;
-
- codeFiles.Add (path, true);
-
- string extension = Path.GetExtension (path);
- if (extension == null || extension.Length == 0)
- return; // maybe better to throw an exception here?
- extension = extension.Substring (1);
- string filename = GetTempFilePhysicalPath (extension);
- if (isVirtual) {
- VirtualFile vf = HostingEnvironment.VirtualPathProvider.GetFile (path);
- if (vf == null)
- throw new HttpException (404, "Virtual file '" + path + "' does not exist.");
-
- using (FileStream f = new FileStream (filename, FileMode.Create, FileAccess.Write)) {
- using (Stream s = vf.Open ()) {
- byte[] input = new byte [COPY_BUFFER_SIZE];
- int retval;
- while ((retval = s.Read (input, 0, COPY_BUFFER_SIZE)) > 0)
- f.Write (input, 0, retval);
- }
- }
- } else
- File.Copy (path, filename);
-
- SourceFiles.Add (filename);
- }
-
- public Stream CreateEmbeddedResource (BuildProvider buildProvider, string name)
- {
- if (buildProvider == null)
- throw new ArgumentNullException ("buildProvider");
- if (name == null || name == "")
- throw new ArgumentNullException ("name");
- string filename = GetTempFilePhysicalPath ("resource");
- Stream stream = File.OpenWrite (filename);
- ResourceFiles [name] = filename;
- return stream;
- }
- [MonoTODO ("Not implemented, does nothing")]
- public void GenerateTypeFactory (string typeName)
- {
- // Do nothing by now.
- }
- public string GetTempFilePhysicalPath (string extension)
- {
- if (extension == null)
- throw new ArgumentNullException ("extension");
- string newFileName = OutputAssemblyPrefix + "_" + temp_files.Count + "." + extension;
- temp_files.AddFile (newFileName, KeepFiles);
- return newFileName;
- }
- public CodeDomProvider CodeDomProvider {
- get { return provider; }
- }
- List <Assembly> ReferencedAssemblies {
- get {
- if (referenced_assemblies == null)
- referenced_assemblies = new List <Assembly> ();
- return referenced_assemblies;
- }
- }
-
- CodeCompileUnit CheckForPartialTypes (CodeCompileUnit compileUnit)
- {
- if (compileUnit == null)
- return null;
- CodeTypeDeclarationCollection types;
- CompileUnitPartialType partialType;
- string partialTypeName;
- List <CompileUnitPartialType> tmp;
- Dictionary <string, List <CompileUnitPartialType>> partialTypes = PartialTypes;
-
- foreach (CodeNamespace ns in compileUnit.Namespaces) {
- if (ns == null)
- continue;
- types = ns.Types;
- if (types == null || types.Count == 0)
- continue;
- foreach (CodeTypeDeclaration type in types) {
- if (type == null)
- continue;
- if (type.IsPartial) {
- partialType = new CompileUnitPartialType (compileUnit, ns, type);
- partialTypeName = partialType.TypeName;
-
- if (!partialTypes.TryGetValue (partialTypeName, out tmp)) {
- tmp = new List <CompileUnitPartialType> (1);
- partialTypes.Add (partialTypeName, tmp);
- }
- tmp.Add (partialType);
- }
- }
- }
-
- return compileUnit;
- }
-
- void ProcessPartialTypes ()
- {
- Dictionary <string, List <CompileUnitPartialType>> partialTypes = PartialTypes;
- if (partialTypes.Count == 0)
- return;
-
- foreach (KeyValuePair <string, List <CompileUnitPartialType>> kvp in partialTypes)
- ProcessType (kvp.Value);
- }
- void ProcessType (List <CompileUnitPartialType> typeList)
- {
- CompileUnitPartialType[] types = new CompileUnitPartialType [typeList.Count];
- int counter = 0;
-
- foreach (CompileUnitPartialType type in typeList) {
- if (counter == 0) {
- types [0] = type;
- counter++;
- continue;
- }
- for (int i = 0; i < counter; i++)
- CompareTypes (types [i], type);
- types [counter++] = type;
- }
- }
- void CompareTypes (CompileUnitPartialType source, CompileUnitPartialType target)
- {
- CodeTypeDeclaration sourceType = source.PartialType;
- CodeTypeMemberCollection targetMembers = target.PartialType.Members;
- List <CodeTypeMember> membersToRemove = new List <CodeTypeMember> ();
-
- foreach (CodeTypeMember member in targetMembers) {
- if (TypeHasMember (sourceType, member))
- membersToRemove.Add (member);
- }
- foreach (CodeTypeMember member in membersToRemove)
- targetMembers.Remove (member);
- }
- bool TypeHasMember (CodeTypeDeclaration type, CodeMemberMethod member)
- {
- if (type == null || member == null)
- return false;
- CodeMemberMethod method = FindMemberByName (type, member.Name) as CodeMemberMethod;
- if (method == null)
- return false;
- if (method.Parameters.Count != member.Parameters.Count)
- return false;
-
- return true;
- }
- bool TypeHasMember (CodeTypeDeclaration type, CodeTypeMember member)
- {
- if (type == null || member == null)
- return false;
- return (FindMemberByName (type, member.Name) != null);
- }
- CodeTypeMember FindMemberByName (CodeTypeDeclaration type, string name)
- {
- foreach (CodeTypeMember m in type.Members) {
- if (m == null || m.Name != name)
- continue;
- return m;
- }
- return null;
- }
-
- internal CompilerResults BuildAssembly (VirtualPath virtualPath)
- {
- return BuildAssembly (virtualPath, CompilerOptions);
- }
-
- internal CompilerResults BuildAssembly (CompilerParameters options)
- {
- return BuildAssembly (null, options);
- }
-
- internal CompilerResults BuildAssembly (VirtualPath virtualPath, CompilerParameters options)
- {
- if (options == null)
- throw new ArgumentNullException ("options");
- options.TempFiles = temp_files;
- if (options.OutputAssembly == null)
- options.OutputAssembly = OutputAssemblyName;
- ProcessPartialTypes ();
-
- CompilerResults results;
- CodeCompileUnit [] units = GetUnitsAsArray ();
- // Since we may have some source files and some code
- // units, we generate code from all of them and then
- // compile the assembly from the set of temporary source
- // files. This also facilates possible debugging for the
- // end user, since they get the code beforehand.
- List <string> files = SourceFiles;
- Dictionary <string, string> resources = ResourceFiles;
- if (units.Length == 0 && files.Count == 0 && resources.Count == 0 && options.EmbeddedResources.Count == 0)
- return null;
-
- string filename;
- StreamWriter sw = null;
-
- foreach (CodeCompileUnit unit in units) {
- filename = GetTempFilePhysicalPath (provider.FileExtension);
- try {
- sw = new StreamWriter (File.OpenWrite (filename));
- provider.GenerateCodeFromCompileUnit (unit, sw, null);
- files.Add (filename);
- } catch {
- throw;
- } finally {
- if (sw != null) {
- sw.Flush ();
- sw.Close ();
- }
- }
- }
- foreach (KeyValuePair <string, string> de in resources)
- options.EmbeddedResources.Add (de.Value);
- foreach (Assembly refasm in ReferencedAssemblies)
- options.ReferencedAssemblies.Add (refasm.Location);
-
- results = provider.CompileAssemblyFromFile (options, files.ToArray ());
- if (results.NativeCompilerReturnValue != 0) {
- string fileText = null;
- try {
- using (StreamReader sr = File.OpenText (results.Errors [0].FileName)) {
- fileText = sr.ReadToEnd ();
- }
- } catch (Exception) {}
-
- #if DEBUG
- Console.WriteLine ("Compilation failed. Errors:");
- foreach (CompilerError err in results.Errors)
- Console.WriteLine (err);
- #endif
-
- throw new CompilationException (virtualPath.Original, results, fileText);
- }
-
- Assembly assembly = results.CompiledAssembly;
- if (assembly == null) {
- if (!File.Exists (options.OutputAssembly)) {
- results.TempFiles.Delete ();
- throw new CompilationException (virtualPath.Original, results.Errors,
- "No assembly returned after compilation!?");
- }
- try {
- results.CompiledAssembly = Assembly.LoadFrom (options.OutputAssembly);
- } catch (Exception ex) {
- results.TempFiles.Delete ();
- throw new HttpException ("Unable to load compiled assembly", ex);
- }
- }
- if (!KeepFiles)
- results.TempFiles.Delete ();
- return results;
- }
- }
- }
- #endif
|