5 // Copyright (c) 2007-2008 Jiri Moudry, Pascal Craponne
\r
7 // Permission is hereby granted, free of charge, to any person obtaining a copy
\r
8 // of this software and associated documentation files (the "Software"), to deal
\r
9 // in the Software without restriction, including without limitation the rights
\r
10 // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
\r
11 // copies of the Software, and to permit persons to whom the Software is
\r
12 // furnished to do so, subject to the following conditions:
\r
14 // The above copyright notice and this permission notice shall be included in
\r
15 // all copies or substantial portions of the Software.
\r
17 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
\r
18 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
\r
19 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
\r
20 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
\r
21 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
\r
22 // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
\r
27 using System.Collections.Generic;
\r
30 using DbLinq.Data.Linq;
\r
31 using DbLinq.Schema;
\r
32 using DbLinq.Schema.Dbml;
\r
33 using DbLinq.Schema.Dbml.Adapter;
\r
35 using Type = System.Type;
\r
38 using System.Data.Linq;
\r
41 namespace DbMetal.Generator.Implementation.CodeTextGenerator
\r
46 abstract partial class CodeGenerator : ICodeGenerator
\r
48 public abstract string LanguageCode { get; }
\r
49 public abstract string Extension { get; }
\r
51 protected class MassDisposer : IDisposable
\r
53 public IList<IDisposable> Disposables = new List<IDisposable>();
\r
55 public void Dispose()
\r
57 for (int index = Disposables.Count - 1; index > 0; index--)
\r
59 Disposables[index].Dispose();
\r
64 protected abstract CodeWriter CreateCodeWriter(TextWriter textWriter);
\r
66 public void Write(TextWriter textWriter, Database dbSchema, GenerationContext context)
\r
68 if (dbSchema == null || dbSchema.Tables == null)
\r
70 //Logger.Write(Level.Error, "CodeGenAll ERROR: incomplete dbSchema, cannot start generating code");
\r
74 context["namespace"] = string.IsNullOrEmpty(context.Parameters.Namespace)
\r
75 ? dbSchema.ContextNamespace
\r
76 : context.Parameters.Namespace;
\r
77 context["database"] = dbSchema.Name;
\r
78 context["generationTime"] = DateTime.Now.ToString("u");
\r
79 context["class"] = dbSchema.Class;
\r
81 using (var codeWriter = CreateCodeWriter(textWriter))
\r
83 WriteBanner(codeWriter, context);
\r
84 WriteUsings(codeWriter, context);
\r
86 string contextNamespace = context.Parameters.Namespace;
\r
87 if (string.IsNullOrEmpty(contextNamespace))
\r
88 contextNamespace = dbSchema.ContextNamespace;
\r
90 string entityNamespace = context.Parameters.Namespace;
\r
91 if (string.IsNullOrEmpty(entityNamespace))
\r
92 entityNamespace = dbSchema.EntityNamespace;
\r
94 if (contextNamespace == entityNamespace)
\r
96 using (WriteNamespace(codeWriter, contextNamespace))
\r
98 WriteDataContext(codeWriter, dbSchema, context);
\r
99 WriteClasses(codeWriter, dbSchema, context);
\r
104 using (WriteNamespace(codeWriter, contextNamespace))
\r
105 WriteDataContext(codeWriter, dbSchema, context);
\r
106 using (WriteNamespace(codeWriter, entityNamespace))
\r
107 WriteClasses(codeWriter, dbSchema, context);
\r
112 private void WriteBanner(CodeWriter writer, GenerationContext context)
\r
114 using (writer.WriteRegion(context.Evaluate("Auto-generated classes for ${database} database on ${generationTime}")))
\r
116 // http://www.network-science.de/ascii/
\r
117 // http://www.network-science.de/ascii/ascii.php?TEXT=MetalSequel&x=14&y=14&FONT=_all+fonts+with+your+text_&RICH=no&FORM=left&STRE=no&WIDT=80
\r
118 writer.WriteCommentLines(
\r
121 | _ \| |__ | \/ | ___| |_ __ _| |
\r
122 | | | | '_ \| |\/| |/ _ \ __/ _` | |
\r
123 | |_| | |_) | | | | __/ || (_| | |
\r
124 |____/|_.__/|_| |_|\___|\__\__,_|_|
\r
126 writer.WriteCommentLines(context.Evaluate("Auto-generated from ${database} on ${generationTime}"));
\r
127 writer.WriteCommentLines("Please visit http://linq.to/db for more information");
\r
131 private void WriteUsings(CodeWriter writer, GenerationContext context)
\r
133 writer.WriteUsingNamespace("System");
\r
134 writer.WriteUsingNamespace("System.Data");
\r
135 writer.WriteUsingNamespace("System.Data.Linq.Mapping");
\r
136 writer.WriteUsingNamespace("System.Diagnostics");
\r
137 writer.WriteUsingNamespace("System.Reflection");
\r
140 writer.WriteUsingNamespace("System.Data.Linq");
\r
142 writer.WriteUsingNamespace("DbLinq.Data.Linq");
\r
143 writer.WriteUsingNamespace("DbLinq.Vendor");
\r
146 // writer.WriteUsingNamespace("System");
\r
147 // writer.WriteUsingNamespace("System.Collections.Generic");
\r
148 // writer.WriteUsingNamespace("System.ComponentModel");
\r
149 // writer.WriteUsingNamespace("System.Data");
\r
150 // writer.WriteUsingNamespace("System.Data.Linq.Mapping");
\r
151 // writer.WriteUsingNamespace("System.Diagnostics");
\r
152 // writer.WriteUsingNamespace("System.Linq");
\r
153 // writer.WriteUsingNamespace("System.Reflection");
\r
154 // writer.WriteUsingNamespace("System.Text");
\r
156 // writer.WriteUsingNamespace("System.Data.Linq");
\r
158 // writer.WriteUsingNamespace("DbLinq.Data.Linq");
\r
159 // writer.WriteUsingNamespace("DbLinq.Data.Linq.Mapping");
\r
162 // now, we write usings required by implemented interfaces
\r
163 foreach (var implementation in context.Implementations())
\r
164 implementation.WriteHeader(writer, context);
\r
166 // write namespaces for members attributes
\r
167 foreach (var memberExposedAttribute in context.Parameters.MemberExposedAttributes)
\r
168 WriteUsingNamespace(writer, GetNamespace(memberExposedAttribute));
\r
170 // write namespaces for clases attributes
\r
171 foreach (var entityExposedAttribute in context.Parameters.EntityExposedAttributes)
\r
172 WriteUsingNamespace(writer, GetNamespace(entityExposedAttribute));
\r
174 writer.WriteLine();
\r
178 /// Writes a using, if given namespace is not null or empty
\r
180 /// <param name="writer"></param>
\r
181 /// <param name="nameSpace"></param>
\r
182 protected virtual void WriteUsingNamespace(CodeWriter writer, string nameSpace)
\r
184 if (!string.IsNullOrEmpty(nameSpace))
\r
185 writer.WriteUsingNamespace(nameSpace);
\r
188 protected virtual string GetNamespace(string fullName)
\r
190 var namePartIndex = fullName.LastIndexOf('.');
\r
191 // if we have a dot, we have a namespace
\r
192 if (namePartIndex < 0)
\r
194 return fullName.Substring(0, namePartIndex);
\r
197 private IDisposable WriteNamespace(CodeWriter writer, string nameSpace)
\r
199 if (!string.IsNullOrEmpty(nameSpace))
\r
200 return writer.WriteNamespace(nameSpace);
\r
204 private void WriteDataContext(CodeWriter writer, Database schema, GenerationContext context)
\r
206 if (schema.Tables.Count == 0)
\r
208 writer.WriteCommentLine("L69 no tables found");
\r
213 string contextBase = schema.BaseType;
\r
214 var contextBaseType = TypeLoader.Load(contextBase);
\r
215 // if we don't specify a base type, use the default
\r
216 if (string.IsNullOrEmpty(contextBase))
\r
218 contextBaseType = typeof(DataContext);
\r
220 // in all cases, get the literal type name from loaded type
\r
221 contextBase = writer.GetLiteralType(contextBaseType);
\r
223 var specifications = SpecificationDefinition.Partial;
\r
224 if (schema.AccessModifierSpecified)
\r
225 specifications |= GetSpecificationDefinition(schema.AccessModifier);
\r
227 specifications |= SpecificationDefinition.Public;
\r
228 if (schema.ModifierSpecified)
\r
229 specifications |= GetSpecificationDefinition(schema.Modifier);
\r
230 using (writer.WriteClass(specifications, schema.Class, contextBase))
\r
232 WriteDataContextCtors(writer, schema, contextBaseType, context);
\r
233 WriteDataContextTables(writer, schema, context);
\r
234 WriteDataContextProcedures(writer, schema, context);
\r
238 private void WriteDataContextTables(CodeWriter writer, Database schema, GenerationContext context)
\r
240 foreach (var table in schema.Tables)
\r
241 WriteDataContextTable(writer, table);
\r
242 writer.WriteLine();
\r
245 protected abstract void WriteDataContextTable(CodeWriter writer, Table table);
\r
247 protected virtual Type GetType(string literalType, bool canBeNull)
\r
249 bool isNullable = literalType.EndsWith("?");
\r
251 literalType = literalType.Substring(0, literalType.Length - 1);
\r
252 bool isArray = literalType.EndsWith("[]");
\r
254 literalType = literalType.Substring(0, literalType.Length - 2);
\r
255 Type type = GetSimpleType(literalType);
\r
259 type = type.MakeArrayType();
\r
261 type = typeof(Nullable<>).MakeGenericType(type);
\r
262 else if (canBeNull)
\r
264 if (type.IsValueType)
\r
265 type = typeof(Nullable<>).MakeGenericType(type);
\r
270 private Type GetSimpleType(string literalType)
\r
272 switch (literalType)
\r
275 return typeof(string);
\r
277 return typeof(long);
\r
279 return typeof(short);
\r
281 return typeof(int);
\r
283 return typeof(char);
\r
285 return typeof(byte);
\r
287 return typeof(float);
\r
289 return typeof(double);
\r
291 return typeof(decimal);
\r
293 return typeof(bool);
\r
295 return typeof(DateTime);
\r
297 return typeof(object);
\r
299 return Type.GetType(literalType);
\r
303 protected string GetAttributeShortName<T>()
\r
304 where T : Attribute
\r
306 string literalAttribute = typeof(T).Name;
\r
307 string end = "Attribute";
\r
308 if (literalAttribute.EndsWith(end))
\r
309 literalAttribute = literalAttribute.Substring(0, literalAttribute.Length - end.Length);
\r
310 return literalAttribute;
\r
313 protected AttributeDefinition NewAttributeDefinition<T>()
\r
314 where T : Attribute
\r
316 return new AttributeDefinition(GetAttributeShortName<T>());
\r
319 protected IDisposable WriteAttributes(CodeWriter writer, params AttributeDefinition[] definitions)
\r
321 var massDisposer = new MassDisposer();
\r
322 foreach (var definition in definitions)
\r
323 massDisposer.Disposables.Add(writer.WriteAttribute(definition));
\r
324 return massDisposer;
\r
327 protected IDisposable WriteAttributes(CodeWriter writer, params string[] definitions)
\r
329 var attributeDefinitions = new List<AttributeDefinition>();
\r
330 foreach (string definition in definitions)
\r
331 attributeDefinitions.Add(new AttributeDefinition(definition));
\r
332 return WriteAttributes(writer, attributeDefinitions.ToArray());
\r
335 protected virtual SpecificationDefinition GetSpecificationDefinition(AccessModifier accessModifier)
\r
337 switch (accessModifier)
\r
339 case AccessModifier.Public:
\r
340 return SpecificationDefinition.Public;
\r
341 case AccessModifier.Internal:
\r
342 return SpecificationDefinition.Internal;
\r
343 case AccessModifier.Protected:
\r
344 return SpecificationDefinition.Protected;
\r
345 case AccessModifier.ProtectedInternal:
\r
346 return SpecificationDefinition.Protected | SpecificationDefinition.Internal;
\r
347 case AccessModifier.Private:
\r
348 return SpecificationDefinition.Private;
\r
350 throw new ArgumentOutOfRangeException("accessModifier");
\r
354 protected virtual SpecificationDefinition GetSpecificationDefinition(ClassModifier classModifier)
\r
356 switch (classModifier)
\r
358 case ClassModifier.Sealed:
\r
359 return SpecificationDefinition.Sealed;
\r
360 case ClassModifier.Abstract:
\r
361 return SpecificationDefinition.Abstract;
\r
363 throw new ArgumentOutOfRangeException("classModifier");
\r
367 protected virtual SpecificationDefinition GetSpecificationDefinition(MemberModifier memberModifier)
\r
369 switch (memberModifier)
\r
371 case MemberModifier.Virtual:
\r
372 return SpecificationDefinition.Virtual;
\r
373 case MemberModifier.Override:
\r
374 return SpecificationDefinition.Override;
\r
375 case MemberModifier.New:
\r
376 return SpecificationDefinition.New;
\r
377 case MemberModifier.NewVirtual:
\r
378 return SpecificationDefinition.New | SpecificationDefinition.Virtual;
\r
380 throw new ArgumentOutOfRangeException("memberModifier");
\r
385 /// The "custom types" are types related to a class
\r
386 /// Currently, we only support enums (non-standard)
\r
388 /// <param name="writer"></param>
\r
389 /// <param name="table"></param>
\r
390 /// <param name="schema"></param>
\r
391 /// <param name="context"></param>
\r
392 protected virtual void WriteCustomTypes(CodeWriter writer, Table table, Database schema, GenerationContext context)
\r
394 // detect required custom types
\r
395 foreach (var column in table.Type.Columns)
\r
397 var extendedType = column.ExtendedType;
\r
398 var enumType = extendedType as EnumType;
\r
399 if (enumType != null)
\r
401 context.ExtendedTypes[column] = new GenerationContext.ExtendedTypeAndName
\r
403 Type = column.ExtendedType,
\r
409 var customTypesNames = new List<string>();
\r
411 // create names and avoid conflits
\r
412 foreach (var extendedTypePair in context.ExtendedTypes)
\r
414 if (extendedTypePair.Value.Table != table)
\r
417 if (string.IsNullOrEmpty(extendedTypePair.Value.Type.Name))
\r
419 string name = extendedTypePair.Key.Member + "Type";
\r
422 if ((from t in context.ExtendedTypes.Values where t.Type.Name == name select t).FirstOrDefault() == null)
\r
424 extendedTypePair.Value.Type.Name = name;
\r
427 // at 3rd loop, it will look ugly, however we will never go there
\r
428 name = extendedTypePair.Value.Table.Type.Name + name;
\r
431 customTypesNames.Add(extendedTypePair.Value.Type.Name);
\r
434 // write custom types
\r
435 if (customTypesNames.Count > 0)
\r
437 using (writer.WriteRegion(string.Format("Custom type definition for {0}", string.Join(", ", customTypesNames.ToArray()))))
\r
440 foreach (var extendedTypePair in context.ExtendedTypes)
\r
442 if (extendedTypePair.Value.Table != table)
\r
445 var extendedType = extendedTypePair.Value.Type;
\r
446 var enumValue = extendedType as EnumType;
\r
448 if (enumValue != null)
\r
450 writer.WriteEnum(GetSpecificationDefinition(extendedTypePair.Key.AccessModifier),
\r
451 enumValue.Name, enumValue);
\r