123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920 |
- using System;
- using System.Collections.Generic;
- using System.Globalization;
- using System.IO;
- using System.Linq;
- using System.Linq.Expressions;
- using System.Reflection;
- using System.Threading.Tasks;
- using System.Xml.Linq;
- namespace HomoIconize
- {
- static class Program
- {
- static void Main(string[] args)
- {
- Console.WriteLine("Auto-homoiconizer for Qbservable[Ex]");
- Console.WriteLine("------------------------------------");
- Console.WriteLine();
- var uri = new Uri(Assembly.GetEntryAssembly().CodeBase);
- var root = Path.Combine(Path.GetDirectoryName(uri.LocalPath), @"..\..\..\..\Source\src");
- if (!Directory.Exists(root))
- {
- Console.WriteLine("Error: Could not find directory \"" + root + "\"");
- return;
- }
- Process(root,
- "System.Reactive",
- @"System.Reactive\Linq\Qbservable.Generated.cs",
- "System.Reactive.Linq.Observable", "Qbservable",
- includeAsync: true, exludeFromCodeCoverage:true);
- Console.WriteLine();
- Process(root,
- "System.Reactive",
- @"System.Reactive\Linq\QbservableEx.Generated.cs",
- "System.Reactive.Linq.ObservableEx", "QbservableEx");
- Console.WriteLine();
- Process(root,
- "System.Reactive.Observable.Aliases",
- @"System.Reactive.Observable.Aliases\Qbservable.Aliases.Generated.cs",
- "System.Reactive.Observable.Aliases.QueryLanguage", "QbservableAliases",
- includeAsync: false, createAliases: true, exludeFromCodeCoverage: true);
- Console.WriteLine();
- Console.WriteLine("Processing complete, press enter to continue.");
- Console.ReadLine();
- }
- static void Process(string root, string sourceAssembly, string targetFile, string sourceTypeName, string targetTypeName, bool includeAsync = false, bool createAliases = false, bool exludeFromCodeCoverage = false)
- {
- var rxRoot = Path.Combine(root, sourceAssembly);
- if (!Directory.Exists(rxRoot))
- {
- Console.WriteLine("Error: Could not find directory \"" + rxRoot + "\"");
- return;
- }
- var dll = Path.Combine(rxRoot, @"bin\debug\netstandard2.0\" + sourceAssembly + ".dll");
- if (!File.Exists(dll))
- {
- Console.WriteLine("Error: Could not find file \"" + dll + "\"");
- return;
- }
- var xml = Path.Combine(rxRoot, @"bin\debug\netstandard2.0\" + sourceAssembly + ".xml");
- if (!File.Exists(xml))
- {
- Console.WriteLine("Error: Could not find file \"" + xml + "\"");
- return;
- }
- var qbsgen = Path.Combine(root, targetFile);
- if (!File.Exists(qbsgen))
- {
- Console.WriteLine("Error: Could not find file \"" + qbsgen + "\"");
- return;
- }
- Generate(dll, xml, qbsgen, sourceTypeName, targetTypeName, includeAsync, createAliases, exludeFromCodeCoverage);
- }
- // Prototype interface to break dependencies. Only used for ToString2 ultimately.
- interface IQbservable<T>
- {
- }
- static void Generate(string input, string xml, string output, string sourceTypeName, string targetTypeName, bool includeAsync, bool createAliases, bool exludeFromCodeCoverage)
- {
- var docs = XDocument.Load(xml).Root.Element("members").Elements("member").ToDictionary(m => m.Attribute("name").Value, m => m);
- Console.WriteLine("Loading {0}...", input);
- var asm = Assembly.LoadFrom(input);
- var t = asm.GetType(sourceTypeName);
- _qbs = typeof(IQbservable<>);
- Console.WriteLine("Checking {0}...", output);
- var attr = File.GetAttributes(output);
- if ((attr & FileAttributes.ReadOnly) == FileAttributes.ReadOnly)
- {
- Console.Write("Attempting to check out generated files... ");
- try
- {
- System.Diagnostics.Process.Start("tf.exe", "edit \"" + output + "\"").WaitForExit();
- }
- catch { /* no comment */ }
- attr = File.GetAttributes(output);
- if ((attr & FileAttributes.ReadOnly) == FileAttributes.ReadOnly)
- {
- Console.WriteLine("Failed.");
- Console.ForegroundColor = ConsoleColor.Yellow;
- Console.WriteLine("Making file writable. DON'T FORGET TO INCLUDE IN CHECK-IN!");
- Console.ResetColor();
- File.SetAttributes(output, attr & ~FileAttributes.ReadOnly);
- }
- else
- {
- Console.WriteLine("Succeeded.");
- }
- }
- Console.WriteLine("Deleting {0}...", output);
- File.Delete(output);
- Console.WriteLine("Creating {0}...", output);
- using (var fs = File.OpenWrite(output))
- {
- using (Out = new StreamWriter(fs))
- {
- Generate(t, docs, targetTypeName, includeAsync, createAliases, exludeFromCodeCoverage);
- }
- }
- }
- static Type _qbs;
- static void Generate(Type t, IDictionary<string, XElement> docs, string typeName, bool includeAsync, bool createAliases, bool exludeFromCodeCoverage)
- {
- WriteLine(
- @"/*
- * WARNING: Auto-generated file (" + DateTime.Now.ToString(CultureInfo.InvariantCulture) + @")
- * Run Rx's auto-homoiconizer tool to generate this file (in the HomoIcon directory).
- */
- ");
- WriteLine(
- @"#nullable enable
- #pragma warning disable 1591
- ");
- WriteLine(
- @"using System.Collections.Generic;");
- if (exludeFromCodeCoverage)
- WriteLine("using System.Diagnostics.CodeAnalysis;");
- WriteLine(
- @"using System.Linq;
- using System.Linq.Expressions;
- using System.Reactive.Concurrency;
- using System.Reactive.Subjects;
- using System.Reflection;
- using System.Threading;
- using System.Threading.Tasks;
- ");
- WriteLine(
- @"namespace System.Reactive.Linq
- {");
- Indent();
- if (exludeFromCodeCoverage)
- WriteLine("[ExcludeFromCodeCoverage]");
- WriteLine(
- @"public static partial class " + typeName + @"
- {");
- Indent();
- var except = new[] { "ToAsync", "FromAsyncPattern", "And", "Then", "GetEnumerator", "get_Provider", "Wait", "ForEach", "ForEachAsync", "GetAwaiter", "RunAsync", "First", "FirstOrDefault", "Last", "LastOrDefault", "Single", "SingleOrDefault", "Subscribe", "AsQbservable", "AsObservable", "ToEvent", "ToEventPattern" };
- foreach (var m in t.GetMethods(BindingFlags.Public | BindingFlags.Static).OrderBy(m => m.Name).ThenBy(m => !m.IsGenericMethod ? "" : string.Join(",", m.GetGenericArguments().Select(p => p.Name))).ThenBy(m => string.Join(",", m.GetParameters().Select(p => p.Name + ":" + p.ParameterType.FullName))).Where(m => !except.Contains(m.Name)))
- {
- var docName = ToDocName(m);
- var xmlDoc = default(XElement);
- if (!docs.TryGetValue(docName, out xmlDoc))
- {
- Console.ForegroundColor = ConsoleColor.Yellow;
- Console.WriteLine("Missing XML documentation for {0}", docName);
- Console.ResetColor();
- }
- var p = m.GetParameters();
- if (m.Name == "When" && p.Length == 1 && p.Single().ParameterType.ToString().Contains("Plan"))
- continue;
- var funky = from pi in p
- let pt = pi.ParameterType
- where pt.IsGenericType
- let ptgtd = pt.GetGenericTypeDefinition()
- where ptgtd.Name.StartsWith("Func")
- where ptgtd.GetGenericArguments().Count() > 5
- select pi;
- var ret = m.ReturnType;
- if (ret.IsGenericType)
- {
- var d = ret.GetGenericTypeDefinition();
- if (d.Name.StartsWith("IConnectableObservable") || d.Name.StartsWith("ListObservable"))
- continue;
- if (d != typeof(IObservable<>) && d != typeof(IEnumerable<>))
- throw new InvalidOperationException("Invalid return type for " + m.Name);
- }
- else
- throw new InvalidOperationException("Invalid return type for " + m.Name);
- ret = ret.Iconize();
- var hasProvider = true;
- if (p.Length > 0)
- {
- var f = p.First();
- if (f.ParameterType.IsGenericType)
- {
- var d = f.ParameterType.GetGenericTypeDefinition();
- if (d == typeof(IObservable<>)) // Check - e.g. Amb || d == typeof(IEnumerable<>))
- hasProvider = false;
- }
- }
- var nulls = new List<string>();
- var pars = new List<string>();
- var parNames = new List<string>();
- var ptps = new List<string>();
- var args = new List<string>();
- var firstArg = hasProvider ? "IQbservableProvider" : p.First().ParameterType.Iconize().ToString2();
- var firstName = hasProvider ? "provider" : p.First().Name;
- pars.Add("this " + firstArg + " " + firstName);
- ptps.Add(firstArg);
- if (!hasProvider)
- args.Add(firstName + ".Expression");
- else
- args.Add("Expression.Constant(provider, typeof(IQbservableProvider))");
- nulls.Add(firstName);
- parNames.Add(firstName);
- var rem = hasProvider ? p : p.Skip(1);
- var isCreateAsync = false;
- foreach (var q in rem)
- {
- var pt = q.ParameterType;
- if (pt.Name.StartsWith("Func") || pt.Name.StartsWith("Action"))
- {
- if (pt.Name.StartsWith("Func") && pt.GetGenericArguments().Last().Name.StartsWith("Task"))
- {
- isCreateAsync = true;
- }
- pt = typeof(Expression<>).MakeGenericType(pt);
- args.Add(q.Name);
- }
- else
- {
- var isObs = new Func<Type, bool>(tt => tt.IsGenericType && tt.GetGenericTypeDefinition() == typeof(IObservable<>));
- var isEnm = new Func<Type, bool>(tt => tt.IsGenericType && tt.GetGenericTypeDefinition() == typeof(IEnumerable<>));
- if (isObs(pt) || pt.IsArray && isObs(pt.GetElementType()) || isEnm(pt) || pt.IsArray && isEnm(pt.GetElementType()))
- args.Add("GetSourceExpression(" + q.Name + ")");
- else
- args.Add("Expression.Constant(" + q.Name + ", typeof(" + pt.ToString2() + "))");
- }
- var pts = pt.ToString2();
- var par = pts + " " + q.Name;
- if (q.IsDefined(typeof(ParamArrayAttribute), false))
- par = "params " + par;
- pars.Add(par);
- ptps.Add(pts);
- parNames.Add(q.Name);
- if (!q.ParameterType.IsValueType && !q.ParameterType.IsGenericParameter)
- nulls.Add(q.Name);
- }
- var factory = hasProvider ? "provider" : p.First().Name + ".Provider";
- var requiresQueryProvider = ret.GetGenericTypeDefinition() == typeof(IQueryable<>);
- if (requiresQueryProvider)
- factory = "((IQueryProvider)" + factory + ")";
-
- var genArgs = m.GetGenericArguments().Select(a => a.ToString2()).ToList();
- var g = genArgs.Count > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
- var name = m.Name;
- if (name == "ToEnumerable")
- name = "ToQueryable";
- var isExp = m.GetCustomAttributes(true).Where(a => a.GetType().Name.Equals("ExperimentalAttribute")).Any();
- if (isExp)
- WriteLine("#if !STABLE", true);
- var obsolete = m.GetCustomAttributes(typeof(ObsoleteAttribute), false).Cast<ObsoleteAttribute>().SingleOrDefault();
- var poundIf = false;
- if (name == "ObserveOn" || name == "SubscribeOn")
- {
- if (p.Last().ParameterType.Name == "DispatcherScheduler")
- {
- WriteLine("#if !MONO", true);
- poundIf = true;
- }
- if (p.Last().ParameterType.Name == "ControlScheduler")
- {
- WriteLine("#if DESKTOPCLR", true);
- poundIf = true;
- }
- }
- if (name == "ObserveOnDispatcher" || name == "SubscribeOnDispatcher")
- {
- WriteLine("#if !MONO", true);
- poundIf = true;
- }
- {
- var retStr = ret.ToString2();
- if (xmlDoc != null)
- {
- SimplifyCrefAttribute(xmlDoc);
- foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- if (hasProvider)
- WriteLine("/// <param name=\"provider\">Query provider used to construct the <see cref=\"IQbservable{T}\"/> data source.</param>");
- var rest = xmlDoc
- .Elements()
- .Where(e => e.Name != "summary")
- .SelectMany(e => e.ToString().Split('\n'));
- foreach (var docLine in rest)
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- if (requiresQueryProvider)
- WriteLine("/// <remarks>This operator requires the source's <see cref=\"IQbservableProvider\"/> object (see <see cref=\"IQbservable.Provider\"/>) to implement <see cref=\"IQueryProvider\"/>.</remarks>");
- }
- if (isExp)
- WriteLine("[Experimental]");
- if (obsolete != null)
- WriteLine("[Obsolete(\"" + obsolete.Message + "\")]");
- WriteLine("public static " + retStr + " " + name + g + "(" + string.Join(", ", pars) + ")");
- var genCons = (from a in m.GetGenericArguments()
- from c in a.GetGenericParameterConstraints()
- select new { a, c })
- .ToList();
- if (genCons.Count > 0)
- {
- Indent();
- foreach (var gc in genCons)
- WriteLine("where " + gc.a.Name + " : " + gc.c.Name);
- Outdent();
- }
- if (createAliases)
- {
- string underlying = "";
- switch (name)
- {
- case "Map":
- underlying = "Select";
- break;
- case "FlatMap":
- underlying = "SelectMany";
- break;
- case "Filter":
- underlying = "Where";
- break;
- }
- WriteLine("{");
- Indent();
- WriteLine("return Qbservable." + underlying + g + "(" + string.Join(", ", parNames) + ");");
- Outdent();
- WriteLine("}");
- WriteLine("");
- continue;
- }
- WriteLine("{");
- Indent();
- foreach (var n in nulls)
- {
- WriteLine("if (" + n + " == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(" + n + "));");
- Outdent();
- }
- WriteLine("");
- var gArg = ret.GetGenericArguments().Single().ToString2();
- WriteLine("return " + factory + ".CreateQuery<" + gArg + ">(");
- Indent();
- WriteLine("Expression.Call(");
- Indent();
- WriteLine("null,");
- var cma = args.Count > 0 ? "," : "";
- if (!m.IsGenericMethod)
- WriteLine("(MethodInfo)MethodInfo.GetCurrentMethod()!" + cma);
- else
- WriteLine("((MethodInfo)MethodInfo.GetCurrentMethod()!).MakeGenericMethod(" + string.Join(", ", m.GetGenericArguments().Select(ga => "typeof(" + ga.Name + ")").ToArray()) + ")" + cma);
- for (int j = 0; j < args.Count; j++)
- WriteLine(args[j] + (j < args.Count - 1 ? "," : ""));
- Outdent();
- WriteLine(")");
- Outdent();
- WriteLine(");");
- Outdent();
- WriteLine("}");
- }
- if (poundIf)
- WriteLine("#endif", true);
- if (isExp)
- WriteLine("#endif", true);
- WriteLine("");
- }
- if (includeAsync)
- {
- GenerateAsync(docs, typeName);
- }
- Outdent();
- WriteLine(
- @"}");
- Outdent();
- WriteLine(
- @"}
- ");
- WriteLine(
- @"#pragma warning restore 1591
- ");
- }
- private static void SimplifyCrefAttribute(XElement parent)
- {
- foreach (var element in parent.Descendants())
- {
- var att = element.Attribute("cref");
- if (att != null)
- {
- element.Attribute("cref").SetValue(SimplyfyDocType(att.Value as string));
- }
- }
- }
- private static string SimplyfyDocType(string cref)
- {
- var v = cref.Replace("T:System.", "")
- .Replace("F:System.", "");
- switch (v)
- {
- case "Double": return "double";
- // case "Decimal": return "decimal";
- case "Int32": return "int";
- case "Inte16": return "short";
- case "Int64": return "long";
- // case "Single": return "float";
- case "IObservable`1": return "IObservable{T}";
- }
- if (v.Contains(".") || v.Contains("`"))
- return cref;
- return v;
- }
- static bool ContainsTask(Type t)
- {
- if (t == typeof(Task))
- return true;
- if (t.IsGenericType)
- {
- if (t.GetGenericTypeDefinition() == typeof(Task<>))
- return true;
- return t.GetGenericArguments().Any(ContainsTask);
- }
- if (t.IsArray)
- return ContainsTask(t.GetElementType());
- return false;
- }
- static void GenerateAsync(IDictionary<string, XElement> docs, string typeName)
- {
- foreach (var ret in new[] { "Unit", "TResult" })
- {
- for (int i = 0; i <= 16; i++)
- {
- foreach (var withScheduler in new[] { false, true })
- {
- var genArgs = default(string[]);
- var lamPars = default(string[]);
- if (i == 0)
- {
- genArgs = new string[0];
- lamPars = new string[0];
- }
- else
- {
- genArgs = Enumerable.Range(1, i).Select(j => "TArg" + j).ToArray();
- lamPars = Enumerable.Range(1, i).Select(j => "t" + j).ToArray();
- }
- var fParam = ret == "Unit" ? "action" : "function";
- var gConst = ret == "Unit" ? "Action" : "Func";
- var retType = "Func<" + string.Join(", ", genArgs.Concat(new[] { "IQbservable<" + ret + ">" }).ToArray()) + ">";
- if (ret != "Unit")
- genArgs = genArgs.Concat(new[] { "TResult" }).ToArray();
- var docName = "M:System.Reactive.Linq.Observable.ToAsync";
- if (genArgs.Length > 0)
- docName += "``" + genArgs.Length;
- var docArg = ret == "Unit" ? "System.Action" : "System.Func";
- if (genArgs.Length > 0)
- docArg += "{" + string.Join(",", Enumerable.Range(0, genArgs.Length).Select(j => "``" + j)) + "}";
- docName += "(" + docArg + (withScheduler ? ",System.Reactive.Concurrency.IScheduler" : "") + ")";
- var xmlDoc = default(XElement);
- if (!docs.TryGetValue(docName, out xmlDoc))
- {
- Console.ForegroundColor = ConsoleColor.Yellow;
- Console.WriteLine("Missing XML documentation for {0}", docName);
- Console.ResetColor();
- }
- var actType = "Expression<" + gConst + (genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "") + ">";
- var genArgss = genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
- if (xmlDoc != null)
- {
- SimplifyCrefAttribute(xmlDoc);
- foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- WriteLine("/// <param name=\"provider\">Query provider used to construct the <see cref=\"IQbservable{T}\"/> data source.</param>");
- foreach (var docLine in xmlDoc.Elements().Where(e => e.Name != "summary").SelectMany(e => e.ToString().Split('\n')))
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- }
- WriteLine("public static " + retType + " ToAsync" + genArgss + "(this IQbservableProvider provider, " + actType + " " + fParam + (withScheduler ? ", IScheduler scheduler" : "") + ")");
- WriteLine("{");
- Indent();
- WriteLine("if (provider == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(provider));");
- Outdent();
- WriteLine("if (" + fParam + " == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(" + fParam + "));");
- Outdent();
- if (withScheduler)
- {
- WriteLine("if (scheduler == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(scheduler));");
- Outdent();
- }
- WriteLine("");
- if (genArgs.Length == 0)
- WriteLine("var m = (MethodInfo)MethodInfo.GetCurrentMethod()!;");
- else
- WriteLine("var m = ((MethodInfo)MethodInfo.GetCurrentMethod()!).MakeGenericMethod(" + string.Join(", ", genArgs.Select(a => "typeof(" + a + ")").ToArray()) + ");");
- WriteLine("return (" + string.Join(", ", lamPars) + ") => provider.CreateQuery<" + ret + ">(");
- Indent();
- WriteLine("Expression.Invoke(");
- Indent();
- WriteLine("Expression.Call(");
- Indent();
- WriteLine("null,");
- WriteLine("m,");
- WriteLine("Expression.Constant(provider, typeof(IQbservableProvider)),");
- WriteLine(fParam + (withScheduler ? "," : ""));
- if (withScheduler)
- WriteLine("Expression.Constant(scheduler, typeof(IScheduler))");
- Outdent();
- WriteLine(")" + (lamPars.Length > 0 ? "," : ""));
- var k = 0;
- foreach (var e in genArgs.Zip(lamPars, (g, l) => new { g, l }))
- {
- WriteLine("Expression.Constant(" + e.l + ", typeof(" + e.g + "))" + (k < i - 1 ? "," : ""));
- k++;
- }
- Outdent();
- WriteLine(")");
- Outdent();
- WriteLine(");");
- Outdent();
- WriteLine("}");
- WriteLine("");
- }
- }
- }
- WriteLine("");
- foreach (var ret in new[] { "Unit", "TResult" })
- {
- for (int i = 0; i < 15; i++)
- {
- var genArgs = default(string[]);
- var lamPars = default(string[]);
- if (i == 0)
- {
- genArgs = new string[0];
- lamPars = new string[0];
- }
- else
- {
- genArgs = Enumerable.Range(1, i).Select(j => "TArg" + j).ToArray();
- lamPars = Enumerable.Range(1, i).Select(j => "t" + j).ToArray();
- }
- var fParam = ret == "Unit" ? "action" : "function";
- var retType = "Func<" + string.Join(", ", genArgs.Concat(new[] { "IQbservable<" + ret + ">" }).ToArray()) + ">";
- var begType = "Expression<Func<" + string.Join(", ", genArgs.Concat(new[] { "AsyncCallback", "object", "IAsyncResult" }).ToArray()) + ">>";
- var endType = ret == "Unit" ? "Expression<Action<IAsyncResult>>" : "Expression<Func<IAsyncResult, TResult>>";
- if (ret != "Unit")
- genArgs = genArgs.Concat(new[] { "TResult" }).ToArray();
- var docName = "M:System.Reactive.Linq.Observable.FromAsyncPattern";
- if (genArgs.Length > 0)
- docName += "``" + genArgs.Length;
- if (ret == "Unit")
- {
- var docArgB = "System.Func{" + string.Join(",", Enumerable.Range(0, genArgs.Length).Select(j => "``" + j)) + (genArgs.Length > 0 ? "," : "") + "System.AsyncCallback,System.Object,System.IAsyncResult}";
- var docArgE = "System.Action{System.IAsyncResult}";
- docName += "(" + docArgB + "," + docArgE + ")";
- }
- else
- {
- var docArgB = "System.Func{" + string.Join(",", Enumerable.Range(0, genArgs.Length - 1).Select(j => "``" + j)) + (genArgs.Length > 1 ? "," : "") + "System.AsyncCallback,System.Object,System.IAsyncResult}";
- var docArgE = "System.Func{System.IAsyncResult,``" + (genArgs.Length - 1) + "}";
- docName += "(" + docArgB + "," + docArgE + ")";
- }
- var xmlDoc = default(XElement);
- if (!docs.TryGetValue(docName, out xmlDoc))
- {
- Console.ForegroundColor = ConsoleColor.Yellow;
- Console.WriteLine("Missing XML documentation for {0}", docName);
- Console.ResetColor();
- }
- var genArgss = genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
- if (xmlDoc != null)
- {
- SimplifyCrefAttribute(xmlDoc);
- foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- WriteLine("/// <param name=\"provider\">Query provider used to construct the <see cref=\"IQbservable{T}\"/> data source.</param>");
- foreach (var docLine in xmlDoc.Elements().Where(e => e.Name != "summary").SelectMany(e => e.ToString().Split('\n')))
- WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
- }
- WriteLine("#if PREFERASYNC", true);
- WriteLine("[Obsolete(Constants_Linq.USE_TASK_FROMASYNCPATTERN)]");
- WriteLine("#endif", true);
- WriteLine("public static " + retType + " FromAsyncPattern" + genArgss + "(this IQbservableProvider provider, " + begType + " begin, " + endType + " end)");
- WriteLine("{");
- Indent();
- WriteLine("if (provider == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(provider));");
- Outdent();
- WriteLine("if (begin == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(begin));");
- Outdent();
- WriteLine("if (end == null)");
- Indent();
- WriteLine("throw new ArgumentNullException(nameof(end));");
- Outdent();
- WriteLine("");
- if (genArgs.Length == 0)
- WriteLine("var m = (MethodInfo)MethodInfo.GetCurrentMethod()!;");
- else
- WriteLine("var m = ((MethodInfo)MethodInfo.GetCurrentMethod()!).MakeGenericMethod(" + string.Join(", ", genArgs.Select(a => "typeof(" + a + ")").ToArray()) + ");");
- WriteLine("return (" + string.Join(", ", lamPars) + ") => provider.CreateQuery<" + ret + ">(");
- Indent();
- WriteLine("Expression.Invoke(");
- Indent();
- WriteLine("Expression.Call(");
- Indent();
- WriteLine("null,");
- WriteLine("m,");
- WriteLine("Expression.Constant(provider, typeof(IQbservableProvider)),");
- WriteLine("begin,");
- WriteLine("end");
- Outdent();
- WriteLine(")" + (lamPars.Length > 0 ? "," : ""));
- var k = 0;
- foreach (var e in genArgs.Zip(lamPars, (g, l) => new { g, l }))
- {
- WriteLine("Expression.Constant(" + e.l + ", typeof(" + e.g + "))" + (k < i - 1 ? "," : ""));
- k++;
- }
- Outdent();
- WriteLine(")");
- Outdent();
- WriteLine(");");
- Outdent();
- WriteLine("}");
- WriteLine("");
- }
- }
- }
- static TextWriter Out { get; set; }
- static int _indent;
- static void WriteLine(string s, bool noIndent = false)
- {
- foreach (var t in s.Split('\n'))
- {
- var indentspace = noIndent || string.IsNullOrWhiteSpace(t)
- ? ""
- : new string(' ', _indent * 4);
- Out.WriteLine(indentspace + t.TrimEnd('\r'));
- }
- }
- static void Indent()
- {
- _indent++;
- }
- static void Outdent()
- {
- _indent--;
- }
- static Type Iconize(this Type type)
- {
- if (type.IsGenericType && !type.IsGenericTypeDefinition)
- {
- var g = type.GetGenericTypeDefinition();
- if (g == typeof(IObservable<>))
- {
- return _qbs.MakeGenericType(type.GetGenericArguments());
- }
- else if (g == typeof(IEnumerable<>))
- {
- return typeof(IQueryable<>).MakeGenericType(type.GetGenericArguments());
- }
- }
- return type;
- }
- static string ToString2(this Type type)
- {
- if (type == typeof(int))
- return "int";
- else if (type == typeof(uint))
- return "uint";
- else if (type == typeof(long))
- return "long";
- else if (type == typeof(ulong))
- return "ulong";
- else if (type == typeof(float))
- return "float";
- else if (type == typeof(double))
- return "double";
- else if (type == typeof(byte))
- return "byte";
- else if (type == typeof(sbyte))
- return "sbyte";
- else if (type == typeof(bool))
- return "bool";
- else if (type == typeof(short))
- return "short";
- else if (type == typeof(ushort))
- return "ushort";
- else if (type == typeof(string))
- return "string";
- else if (type == typeof(object))
- return "object";
- else if (type == typeof(void))
- return "void";
- else if (type == typeof(decimal))
- return "decimal";
- if (type.IsArray)
- return type.GetElementType().ToString2() + "[" + new string(',', type.GetArrayRank() - 1) + "]";
- if (type.IsGenericType)
- {
- if (!type.IsGenericTypeDefinition)
- {
- var g = type.GetGenericTypeDefinition();
- if (g == typeof(Nullable<>))
- return type.GetGenericArguments()[0].ToString2() + "?";
- else
- return g.ToString2() + "<" + string.Join(", ", type.GetGenericArguments().Select(t => t.ToString2()).ToArray()) + ">";
- }
- else
- {
- var s = type.Name;
- return s.Substring(0, s.LastIndexOf('`'));
- }
- }
- return type.Name;
- }
- static string ToDocName(MethodInfo method)
- {
- var name = "M:" + ToDocName(method.DeclaringType) + "." + method.Name;
- var genArgs = new Type[0];
- if (method.IsGenericMethod)
- {
- genArgs = method.GetGenericArguments();
- name += "``" + genArgs.Length;
- }
- var pars = method.GetParameters();
- if (pars.Length > 0)
- {
- name += "(" + string.Join(",", method.GetParameters().Select(p => ToDocName(p.ParameterType, genArgs))) + ")";
- }
- return name;
- }
- static string ToDocName(Type t, params Type[] genArgs)
- {
- var i = Array.IndexOf(genArgs, t);
- if (i >= 0)
- return "``" + i;
- if (t.IsArray)
- {
- return ToDocName(t.GetElementType(), genArgs) + "[]";
- }
- if (t.IsGenericType)
- {
- var def = t.GetGenericTypeDefinition();
- var name = def.FullName.Substring(0, def.FullName.LastIndexOf("`"));
- var args = t.GetGenericArguments();
- name += "{" + string.Join(",", args.Select(a => ToDocName(a, genArgs))) + "}";
- return name;
- }
- else
- {
- return t.FullName;
- }
- }
- }
- }
|