Program.cs 39 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. using System.Reflection;
  7. using System.Threading.Tasks;
  8. using System.Xml.Linq;
  9. namespace HomoIconize
  10. {
  11. static class Program
  12. {
  13. static void Main(string[] args)
  14. {
  15. Console.WriteLine("Auto-homoiconizer for Qbservable[Ex]");
  16. Console.WriteLine("------------------------------------");
  17. Console.WriteLine();
  18. var uri = new Uri(Assembly.GetEntryAssembly().CodeBase);
  19. var root = Path.Combine(Path.GetDirectoryName(uri.LocalPath), @"..\..\..\..\Source");
  20. if (!Directory.Exists(root))
  21. {
  22. Console.WriteLine("Error: Could not find directory \"" + root + "\"");
  23. return;
  24. }
  25. Process(root,
  26. "System.Reactive.Linq",
  27. "System.Reactive.Providers",
  28. @"Reactive\Linq\Qbservable.Generated.cs",
  29. "System.Reactive.Linq.Observable", "Qbservable", true);
  30. Console.WriteLine();
  31. Process(root,
  32. "System.Reactive.Experimental",
  33. "System.Reactive.Experimental",
  34. @"Reactive\Linq\QbservableEx.Generated.cs",
  35. "System.Reactive.Linq.ObservableEx", "QbservableEx");
  36. Console.WriteLine();
  37. Process(root,
  38. "System.Reactive.Observable.Aliases",
  39. "System.Reactive.Observable.Aliases",
  40. "Qbservable.Aliases.Generated.cs",
  41. "System.Reactive.Observable.Aliases.QueryLanguage", "QbservableAliases",
  42. includeAsync: false, createAliases: true);
  43. Console.WriteLine();
  44. Console.WriteLine("Processing complete, press enter to continue.");
  45. Console.ReadLine();
  46. }
  47. static void Process(string root, string sourceAssembly, string targetAssembly, string targetFile, string sourceTypeName, string targetTypeName, bool includeAsync = false, bool createAliases = false)
  48. {
  49. var rxRoot = Path.Combine(root, sourceAssembly);
  50. if (!Directory.Exists(rxRoot))
  51. {
  52. Console.WriteLine("Error: Could not find directory \"" + rxRoot + "\"");
  53. return;
  54. }
  55. var qbRoot = Path.Combine(root, targetAssembly);
  56. if (!Directory.Exists(qbRoot))
  57. {
  58. Console.WriteLine("Error: Could not find directory \"" + qbRoot + "\"");
  59. return;
  60. }
  61. var dll = Path.Combine(rxRoot, @"..\bin\debug40\" + sourceAssembly + ".dll");
  62. if (!File.Exists(dll))
  63. {
  64. Console.WriteLine("Error: Could not find file \"" + dll + "\"");
  65. return;
  66. }
  67. var xml = Path.Combine(rxRoot, @"..\bin\debug40\" + sourceAssembly + ".xml");
  68. if (!File.Exists(xml))
  69. {
  70. Console.WriteLine("Error: Could not find file \"" + xml + "\"");
  71. return;
  72. }
  73. var qbsgen = Path.Combine(qbRoot, targetFile);
  74. if (!File.Exists(qbsgen))
  75. {
  76. Console.WriteLine("Error: Could not find file \"" + qbsgen + "\"");
  77. return;
  78. }
  79. Generate(dll, xml, qbsgen, sourceTypeName, targetTypeName, includeAsync, createAliases);
  80. }
  81. // Prototype interface to break dependencies. Only used for ToString2 ultimately.
  82. interface IQbservable<T>
  83. {
  84. }
  85. static void Generate(string input, string xml, string output, string sourceTypeName, string targetTypeName, bool includeAsync, bool createAliases)
  86. {
  87. var docs = XDocument.Load(xml).Root.Element("members").Elements("member").ToDictionary(m => m.Attribute("name").Value, m => m);
  88. Console.WriteLine("Loading {0}...", input);
  89. var asm = Assembly.LoadFrom(input);
  90. var t = asm.GetType(sourceTypeName);
  91. _qbs = typeof(IQbservable<>); //asm.GetType("System.Reactive.Linq.IQbservable`1");
  92. Console.WriteLine("Checking {0}...", output);
  93. var attr = File.GetAttributes(output);
  94. if ((attr & FileAttributes.ReadOnly) == FileAttributes.ReadOnly)
  95. {
  96. Console.Write("Attempting to check out generated files... ");
  97. try
  98. {
  99. System.Diagnostics.Process.Start("tf.exe", "edit \"" + output + "\"").WaitForExit();
  100. }
  101. catch { /* no comment */ }
  102. attr = File.GetAttributes(output);
  103. if ((attr & FileAttributes.ReadOnly) == FileAttributes.ReadOnly)
  104. {
  105. Console.WriteLine("Failed.");
  106. Console.ForegroundColor = ConsoleColor.Yellow;
  107. Console.WriteLine("Making file writable. DON'T FORGET TO INCLUDE IN CHECK-IN!");
  108. Console.ResetColor();
  109. File.SetAttributes(output, attr & ~FileAttributes.ReadOnly);
  110. }
  111. else
  112. {
  113. Console.WriteLine("Succeeded.");
  114. }
  115. }
  116. Console.WriteLine("Deleting {0}...", output);
  117. File.Delete(output);
  118. Console.WriteLine("Creating {0}...", output);
  119. using (var fs = File.OpenWrite(output))
  120. {
  121. using (Out = new StreamWriter(fs))
  122. {
  123. Generate(t, docs, targetTypeName, includeAsync, createAliases);
  124. }
  125. }
  126. }
  127. static Type _qbs;
  128. static void Generate(Type t, IDictionary<string, XElement> docs, string typeName, bool includeAsync, bool createAliases)
  129. {
  130. WriteLine(
  131. @"/*
  132. * WARNING: Auto-generated file (" + DateTime.Now + @")
  133. * Run Rx's auto-homoiconizer tool to generate this file (in the HomoIcon directory).
  134. */
  135. ");
  136. WriteLine(
  137. @"#pragma warning disable 1591
  138. ");
  139. WriteLine(
  140. @"#if !NO_EXPRESSIONS
  141. ");
  142. WriteLine(
  143. @"using System;
  144. using System.Reactive.Concurrency;
  145. using System.Collections.Generic;
  146. using System.Reactive.Joins;
  147. using System.Linq;
  148. using System.Linq.Expressions;
  149. using System.Reflection;
  150. using System.Threading;
  151. using System.Reactive;
  152. using System.Reactive.Subjects;
  153. #if !NO_TPL
  154. using System.Threading.Tasks;
  155. #endif
  156. #if !NO_REMOTING
  157. using System.Runtime.Remoting.Lifetime;
  158. #endif
  159. ");
  160. WriteLine(
  161. @"namespace System.Reactive.Linq
  162. {");
  163. Indent();
  164. WriteLine(
  165. @"public static partial class " + typeName + @"
  166. {");
  167. Indent();
  168. var except = new[] { "ToAsync", "FromAsyncPattern", "And", "Then", "GetEnumerator", "get_Provider", "Wait", "ForEach", "ForEachAsync", "GetAwaiter", "First", "FirstOrDefault", "Last", "LastOrDefault", "Single", "SingleOrDefault", "Subscribe", "AsQbservable", "AsObservable", "ToEvent", "ToEventPattern" };
  169. 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)))
  170. {
  171. var docName = ToDocName(m);
  172. var xmlDoc = default(XElement);
  173. if (!docs.TryGetValue(docName, out xmlDoc))
  174. {
  175. Console.ForegroundColor = ConsoleColor.Yellow;
  176. Console.WriteLine("Missing XML documentation for {0}", docName);
  177. Console.ResetColor();
  178. }
  179. var p = m.GetParameters();
  180. if (m.Name == "When" && p.Length == 1 && p.Single().ParameterType.ToString().Contains("Plan"))
  181. continue;
  182. var funky = from pi in p
  183. let pt = pi.ParameterType
  184. where pt.IsGenericType
  185. let ptgtd = pt.GetGenericTypeDefinition()
  186. where ptgtd.Name.StartsWith("Func")
  187. where ptgtd.GetGenericArguments().Count() > 5
  188. select pi;
  189. //var isLargeArity = funky.Any();
  190. var hasTask = p.Any(pa => ContainsTask(pa.ParameterType));
  191. var ret = m.ReturnType;
  192. if (ret.IsGenericType)
  193. {
  194. var d = ret.GetGenericTypeDefinition();
  195. if (d.Name.StartsWith("IConnectableObservable") || d.Name.StartsWith("ListObservable"))
  196. continue;
  197. if (d != typeof(IObservable<>) && d != typeof(IEnumerable<>))
  198. throw new InvalidOperationException("Invalid return type for " + m.Name);
  199. }
  200. else
  201. throw new InvalidOperationException("Invalid return type for " + m.Name);
  202. ret = ret.Iconize();
  203. var hasProvider = true;
  204. if (p.Length > 0)
  205. {
  206. var f = p.First();
  207. if (f.ParameterType.IsGenericType)
  208. {
  209. var d = f.ParameterType.GetGenericTypeDefinition();
  210. if (d == typeof(IObservable<>)) // Check - e.g. Amb || d == typeof(IEnumerable<>))
  211. hasProvider = false;
  212. }
  213. }
  214. var nulls = new List<string>();
  215. var pars = new List<string>();
  216. var parNames = new List<string>();
  217. var ptps = new List<string>();
  218. var args = new List<string>();
  219. var firstArg = hasProvider ? "IQbservableProvider" : p.First().ParameterType.Iconize().ToString2();
  220. var firstName = hasProvider ? "provider" : p.First().Name;
  221. pars.Add("this " + firstArg + " " + firstName);
  222. ptps.Add(firstArg);
  223. if (!hasProvider)
  224. args.Add(firstName + ".Expression");
  225. else
  226. args.Add("Expression.Constant(provider, typeof(IQbservableProvider))");
  227. nulls.Add(firstName);
  228. parNames.Add(firstName);
  229. var rem = hasProvider ? p : p.Skip(1);
  230. var isCreateAsync = false;
  231. foreach (var q in rem)
  232. {
  233. var pt = q.ParameterType;
  234. if (pt.Name.StartsWith("Func") || pt.Name.StartsWith("Action"))
  235. {
  236. if (pt.Name.StartsWith("Func") && pt.GetGenericArguments().Last().Name.StartsWith("Task"))
  237. {
  238. isCreateAsync = true;
  239. }
  240. pt = typeof(Expression<>).MakeGenericType(pt);
  241. args.Add(q.Name);
  242. }
  243. else
  244. {
  245. var isObs = new Func<Type, bool>(tt => tt.IsGenericType && tt.GetGenericTypeDefinition() == typeof(IObservable<>));
  246. var isEnm = new Func<Type, bool>(tt => tt.IsGenericType && tt.GetGenericTypeDefinition() == typeof(IEnumerable<>));
  247. if (isObs(pt) || pt.IsArray && isObs(pt.GetElementType()) || isEnm(pt) || pt.IsArray && isEnm(pt.GetElementType()))
  248. args.Add("GetSourceExpression(" + q.Name + ")");
  249. else
  250. args.Add("Expression.Constant(" + q.Name + ", typeof(" + pt.ToString2() + "))");
  251. }
  252. var pts = pt.ToString2();
  253. var par = pts + " " + q.Name;
  254. if (q.IsDefined(typeof(ParamArrayAttribute), false))
  255. par = "params " + par;
  256. pars.Add(par);
  257. ptps.Add(pts);
  258. parNames.Add(q.Name);
  259. if (!q.ParameterType.IsValueType && !q.ParameterType.IsGenericParameter)
  260. nulls.Add(q.Name);
  261. }
  262. var factory = hasProvider ? "provider" : p.First().Name + ".Provider";
  263. var requiresQueryProvider = ret.GetGenericTypeDefinition() == typeof(IQueryable<>);
  264. if (requiresQueryProvider)
  265. factory = "((IQueryProvider)" + factory + ")";
  266. var genArgs = m.GetGenericArguments().Select(a => a.ToString2()).ToList();
  267. var g = genArgs.Count > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
  268. var name = m.Name;
  269. if (name == "ToEnumerable")
  270. name = "ToQueryable";
  271. var isExp = m.GetCustomAttributes(true).Where(a => a.GetType().Name.Equals("ExperimentalAttribute")).Any();
  272. if (isExp)
  273. WriteLine("#if !STABLE", true);
  274. var obsolete = m.GetCustomAttributes(typeof(ObsoleteAttribute), false).Cast<ObsoleteAttribute>().SingleOrDefault();
  275. var poundIf = false;
  276. if (name == "ObserveOn" || name == "SubscribeOn")
  277. {
  278. if (p.Last().ParameterType.Name == "DispatcherScheduler")
  279. {
  280. WriteLine("#if !MONO", true);
  281. poundIf = true;
  282. }
  283. if (p.Last().ParameterType.Name == "ControlScheduler")
  284. {
  285. WriteLine("#if DESKTOPCLR", true);
  286. poundIf = true;
  287. }
  288. }
  289. if (name == "ObserveOnDispatcher" || name == "SubscribeOnDispatcher")
  290. {
  291. WriteLine("#if !MONO", true);
  292. poundIf = true;
  293. }
  294. if (isCreateAsync || hasTask)
  295. {
  296. WriteLine("#if !NO_TPL", true);
  297. poundIf = true;
  298. }
  299. //if (name == "Remotable")
  300. //{
  301. // WriteLine("#if DESKTOPCLR", true);
  302. // poundIf = true;
  303. // if (nulls.Contains("lease"))
  304. // nulls.Remove("lease");
  305. //}
  306. //if (isLargeArity)
  307. //{
  308. // WriteLine("#if !NO_LARGEARITY", true);
  309. //}
  310. var isFep = m.Name == "FromEventPattern";
  311. var isGenFep = isFep && m.GetGenericArguments().Any(a => a.Name == "TEventArgs");
  312. var isNonGenFep = isFep && !isGenFep;
  313. for (var r = 0; r < (isNonGenFep ? 2 : 1); r++)
  314. {
  315. var retStr = ret.ToString2();
  316. if (isNonGenFep)
  317. {
  318. if (r == 0)
  319. {
  320. WriteLine("#if !NO_EVENTARGS_CONSTRAINT", true);
  321. }
  322. else if (r == 1)
  323. {
  324. WriteLine("#else", true);
  325. retStr = retStr.Replace("EventPattern<EventArgs>", "EventPattern<object>");
  326. }
  327. }
  328. if (xmlDoc != null)
  329. {
  330. foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
  331. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  332. if (hasProvider)
  333. WriteLine("/// <param name=\"provider\">Query provider used to construct the IQbservable&lt;T&gt; data source.</param>");
  334. foreach (var docLine in xmlDoc.Elements().Where(e => e.Name != "summary").SelectMany(e => e.ToString().Split('\n')))
  335. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  336. if (requiresQueryProvider)
  337. WriteLine("/// <remarks>This operator requires the source's <see cref=\"IQbservableProvider\"/> object (see <see cref=\"IQbservable.Provider\"/>) to implement <see cref=\"IQueryProvider\"/>.</remarks>");
  338. }
  339. if (isExp)
  340. WriteLine("[Experimental]");
  341. if (obsolete != null)
  342. WriteLine("[Obsolete(\"" + obsolete.Message + "\")]");
  343. WriteLine("public static " + retStr + " " + name + g + "(" + string.Join(", ", pars) + ")");
  344. if (isGenFep)
  345. {
  346. WriteLine("#if !NO_EVENTARGS_CONSTRAINT", true);
  347. Indent();
  348. WriteLine("where TEventArgs : EventArgs");
  349. Outdent();
  350. WriteLine("#endif", true);
  351. }
  352. else
  353. {
  354. var genCons = (from a in m.GetGenericArguments()
  355. from c in a.GetGenericParameterConstraints()
  356. select new { a, c })
  357. .ToList();
  358. if (genCons.Count > 0)
  359. {
  360. Indent();
  361. foreach (var gc in genCons)
  362. WriteLine("where " + gc.a.Name + " : " + gc.c.Name);
  363. Outdent();
  364. }
  365. }
  366. if (createAliases)
  367. {
  368. string underlying = "";
  369. switch (name)
  370. {
  371. case "Map":
  372. underlying = "Select";
  373. break;
  374. case "FlatMap":
  375. underlying = "SelectMany";
  376. break;
  377. case "Filter":
  378. underlying = "Where";
  379. break;
  380. }
  381. WriteLine("{");
  382. Indent();
  383. WriteLine("return Qbservable." + underlying + g + "(" + string.Join(", ", parNames) + ");");
  384. Outdent();
  385. WriteLine("}");
  386. continue;
  387. }
  388. WriteLine("{");
  389. Indent();
  390. foreach (var n in nulls)
  391. {
  392. WriteLine("if (" + n + " == null)");
  393. Indent();
  394. WriteLine("throw new ArgumentNullException(\"" + n + "\");");
  395. Outdent();
  396. }
  397. WriteLine("");
  398. var gArg = ret.GetGenericArguments().Single().ToString2();
  399. if (isNonGenFep && r == 1)
  400. {
  401. gArg = gArg.Replace("EventPattern<EventArgs>", "EventPattern<object>");
  402. }
  403. WriteLine("return " + factory + ".CreateQuery<" + gArg + ">(");
  404. Indent();
  405. WriteLine("Expression.Call(");
  406. Indent();
  407. WriteLine("null,");
  408. var cma = args.Count > 0 ? "," : "";
  409. WriteLine("#if CRIPPLED_REFLECTION", true);
  410. WriteLine("InfoOf(() => " + typeName + "." + name + g + "(" + string.Join(", ", ptps.Select(pt => "default(" + pt + ")")) + "))" + cma);
  411. WriteLine("#else", true);
  412. if (!m.IsGenericMethod)
  413. WriteLine("(MethodInfo)MethodInfo.GetCurrentMethod()" + cma);
  414. else
  415. WriteLine("((MethodInfo)MethodInfo.GetCurrentMethod()).MakeGenericMethod(" + string.Join(", ", m.GetGenericArguments().Select(ga => "typeof(" + ga.Name + ")").ToArray()) + ")" + cma);
  416. WriteLine("#endif", true);
  417. for (int j = 0; j < args.Count; j++)
  418. WriteLine(args[j] + (j < args.Count - 1 ? "," : ""));
  419. Outdent();
  420. WriteLine(")");
  421. Outdent();
  422. WriteLine(");");
  423. Outdent();
  424. WriteLine("}");
  425. if (isNonGenFep && r == 1)
  426. WriteLine("#endif", true);
  427. }
  428. if (poundIf)
  429. WriteLine("#endif", true);
  430. if (isExp)
  431. WriteLine("#endif", true);
  432. //if (isLargeArity)
  433. // WriteLine("#endif", true);
  434. WriteLine("");
  435. }
  436. if (includeAsync)
  437. {
  438. GenerateAsync(docs, typeName);
  439. }
  440. Outdent();
  441. WriteLine(
  442. @"}");
  443. Outdent();
  444. WriteLine(
  445. @"}
  446. ");
  447. WriteLine(
  448. @"#endif
  449. ");
  450. WriteLine(
  451. @"#pragma warning restore 1591
  452. ");
  453. }
  454. static bool ContainsTask(Type t)
  455. {
  456. if (t == typeof(Task))
  457. return true;
  458. if (t.IsGenericType)
  459. {
  460. if (t.GetGenericTypeDefinition() == typeof(Task<>))
  461. return true;
  462. return t.GetGenericArguments().Any(ContainsTask);
  463. }
  464. if (t.IsArray)
  465. return ContainsTask(t.GetElementType());
  466. return false;
  467. }
  468. static void GenerateAsync(IDictionary<string, XElement> docs, string typeName)
  469. {
  470. foreach (var ret in new[] { "Unit", "TResult" })
  471. {
  472. for (int i = 0; i <= 16; i++)
  473. {
  474. //if (i == 5)
  475. // WriteLine("#if !NO_LARGEARITY", true);
  476. foreach (var withScheduler in new[] { false, true })
  477. {
  478. var genArgs = default(string[]);
  479. var lamPars = default(string[]);
  480. if (i == 0)
  481. {
  482. genArgs = new string[0];
  483. lamPars = new string[0];
  484. }
  485. //else if (i == 1)
  486. //{
  487. // genArgs = new[] { "TSource" };
  488. // lamPars = new[] { "t" };
  489. //}
  490. else
  491. {
  492. genArgs = Enumerable.Range(1, i).Select(j => "TArg" + j).ToArray();
  493. lamPars = Enumerable.Range(1, i).Select(j => "t" + j).ToArray();
  494. }
  495. var fParam = ret == "Unit" ? "action" : "function";
  496. var gConst = ret == "Unit" ? "Action" : "Func";
  497. var retType = "Func<" + string.Join(", ", genArgs.Concat(new[] { "IQbservable<" + ret + ">" }).ToArray()) + ">";
  498. if (ret != "Unit")
  499. genArgs = genArgs.Concat(new[] { "TResult" }).ToArray();
  500. var docName = "M:System.Reactive.Linq.Observable.ToAsync";
  501. if (genArgs.Length > 0)
  502. docName += "``" + genArgs.Length;
  503. var docArg = ret == "Unit" ? "System.Action" : "System.Func";
  504. if (genArgs.Length > 0)
  505. docArg += "{" + string.Join(",", Enumerable.Range(0, genArgs.Length).Select(j => "``" + j)) + "}";
  506. docName += "(" + docArg + (withScheduler ? ",System.Reactive.Concurrency.IScheduler" : "") + ")";
  507. var xmlDoc = default(XElement);
  508. if (!docs.TryGetValue(docName, out xmlDoc))
  509. {
  510. Console.ForegroundColor = ConsoleColor.Yellow;
  511. Console.WriteLine("Missing XML documentation for {0}", docName);
  512. Console.ResetColor();
  513. }
  514. var actType = "Expression<" + gConst + (genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "") + ">";
  515. var genArgss = genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
  516. if (xmlDoc != null)
  517. {
  518. foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
  519. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  520. WriteLine("/// <param name=\"provider\">Query provider used to construct the IQbservable&lt;T&gt; data source.</param>");
  521. foreach (var docLine in xmlDoc.Elements().Where(e => e.Name != "summary").SelectMany(e => e.ToString().Split('\n')))
  522. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  523. }
  524. WriteLine("public static " + retType + " ToAsync" + genArgss + "(this IQbservableProvider provider, " + actType + " " + fParam + (withScheduler ? ", IScheduler scheduler" : "") + ")");
  525. WriteLine("{");
  526. Indent();
  527. WriteLine("if (provider == null)");
  528. Indent();
  529. WriteLine("throw new ArgumentNullException(\"provider\");");
  530. Outdent();
  531. WriteLine("if (" + fParam + " == null)");
  532. Indent();
  533. WriteLine("throw new ArgumentNullException(\"" + fParam + "\");");
  534. Outdent();
  535. if (withScheduler)
  536. {
  537. WriteLine("if (scheduler == null)");
  538. Indent();
  539. WriteLine("throw new ArgumentNullException(\"scheduler\");");
  540. Outdent();
  541. }
  542. WriteLine("");
  543. WriteLine("#if CRIPPLED_REFLECTION", true);
  544. var aprs = new List<string> { "IQbservableProvider", actType };
  545. if (withScheduler)
  546. aprs.Add("IScheduler");
  547. WriteLine("var m = InfoOf(() => " + typeName + ".ToAsync" + genArgss + "(" + string.Join(", ", aprs.Select(pt => "default(" + pt + ")")) + "));");
  548. WriteLine("#else", true);
  549. if (genArgs.Length == 0)
  550. WriteLine("var m = (MethodInfo)MethodInfo.GetCurrentMethod();");
  551. else
  552. WriteLine("var m = ((MethodInfo)MethodInfo.GetCurrentMethod()).MakeGenericMethod(" + string.Join(", ", genArgs.Select(a => "typeof(" + a + ")").ToArray()) + ");");
  553. WriteLine("#endif", true);
  554. WriteLine("return (" + string.Join(", ", lamPars) + ") => provider.CreateQuery<" + ret + ">(");
  555. Indent();
  556. WriteLine("Expression.Invoke(");
  557. Indent();
  558. WriteLine("Expression.Call(");
  559. Indent();
  560. WriteLine("null,");
  561. WriteLine("m,");
  562. WriteLine("Expression.Constant(provider, typeof(IQbservableProvider)),");
  563. WriteLine(fParam + (withScheduler ? "," : ""));
  564. if (withScheduler)
  565. WriteLine("Expression.Constant(scheduler, typeof(IScheduler))");
  566. Outdent();
  567. WriteLine(")" + (lamPars.Length > 0 ? "," : ""));
  568. var k = 0;
  569. foreach (var e in genArgs.Zip(lamPars, (g, l) => new { g, l }))
  570. {
  571. WriteLine("Expression.Constant(" + e.l + ", typeof(" + e.g + "))" + (k < i - 1 ? "," : ""));
  572. k++;
  573. }
  574. Outdent();
  575. WriteLine(")");
  576. Outdent();
  577. WriteLine(");");
  578. Outdent();
  579. WriteLine("}");
  580. WriteLine("");
  581. }
  582. //if (i == 16)
  583. // WriteLine("#endif", true);
  584. }
  585. }
  586. WriteLine("");
  587. foreach (var ret in new[] { "Unit", "TResult" })
  588. {
  589. for (int i = 0; i < 15; i++)
  590. {
  591. //if (i == 3)
  592. // WriteLine("#if !NO_LARGEARITY", true);
  593. var genArgs = default(string[]);
  594. var lamPars = default(string[]);
  595. if (i == 0)
  596. {
  597. genArgs = new string[0];
  598. lamPars = new string[0];
  599. }
  600. else
  601. {
  602. genArgs = Enumerable.Range(1, i).Select(j => "TArg" + j).ToArray();
  603. lamPars = Enumerable.Range(1, i).Select(j => "t" + j).ToArray();
  604. }
  605. var fParam = ret == "Unit" ? "action" : "function";
  606. var retType = "Func<" + string.Join(", ", genArgs.Concat(new[] { "IQbservable<" + ret + ">" }).ToArray()) + ">";
  607. var begType = "Expression<Func<" + string.Join(", ", genArgs.Concat(new[] { "AsyncCallback", "object", "IAsyncResult" }).ToArray()) + ">>";
  608. var endType = ret == "Unit" ? "Expression<Action<IAsyncResult>>" : "Expression<Func<IAsyncResult, TResult>>";
  609. if (ret != "Unit")
  610. genArgs = genArgs.Concat(new[] { "TResult" }).ToArray();
  611. var docName = "M:System.Reactive.Linq.Observable.FromAsyncPattern";
  612. if (genArgs.Length > 0)
  613. docName += "``" + genArgs.Length;
  614. if (ret == "Unit")
  615. {
  616. var docArgB = "System.Func{" + string.Join(",", Enumerable.Range(0, genArgs.Length).Select(j => "``" + j)) + (genArgs.Length > 0 ? "," : "") + "System.AsyncCallback,System.Object,System.IAsyncResult}";
  617. var docArgE = "System.Action{System.IAsyncResult}";
  618. docName += "(" + docArgB + "," + docArgE + ")";
  619. }
  620. else
  621. {
  622. var docArgB = "System.Func{" + string.Join(",", Enumerable.Range(0, genArgs.Length - 1).Select(j => "``" + j)) + (genArgs.Length > 1 ? "," : "") + "System.AsyncCallback,System.Object,System.IAsyncResult}";
  623. var docArgE = "System.Func{System.IAsyncResult,``" + (genArgs.Length - 1) + "}";
  624. docName += "(" + docArgB + "," + docArgE + ")";
  625. }
  626. var xmlDoc = default(XElement);
  627. if (!docs.TryGetValue(docName, out xmlDoc))
  628. {
  629. Console.ForegroundColor = ConsoleColor.Yellow;
  630. Console.WriteLine("Missing XML documentation for {0}", docName);
  631. Console.ResetColor();
  632. }
  633. var genArgss = genArgs.Length > 0 ? "<" + string.Join(", ", genArgs) + ">" : "";
  634. if (xmlDoc != null)
  635. {
  636. foreach (var docLine in xmlDoc.Element("summary").ToString().Split('\n'))
  637. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  638. WriteLine("/// <param name=\"provider\">Query provider used to construct the IQbservable&lt;T&gt; data source.</param>");
  639. foreach (var docLine in xmlDoc.Elements().Where(e => e.Name != "summary").SelectMany(e => e.ToString().Split('\n')))
  640. WriteLine("/// " + docLine.TrimStart().TrimEnd('\r'));
  641. }
  642. WriteLine("#if PREFERASYNC", true);
  643. WriteLine("[Obsolete(Constants_Linq.USE_TASK_FROMASYNCPATTERN)]");
  644. WriteLine("#endif", true);
  645. WriteLine("public static " + retType + " FromAsyncPattern" + genArgss + "(this IQbservableProvider provider, " + begType + " begin, " + endType + "end)");
  646. WriteLine("{");
  647. Indent();
  648. WriteLine("if (provider == null)");
  649. Indent();
  650. WriteLine("throw new ArgumentNullException(\"provider\");");
  651. Outdent();
  652. WriteLine("if (begin == null)");
  653. Indent();
  654. WriteLine("throw new ArgumentNullException(\"begin\");");
  655. Outdent();
  656. WriteLine("if (end == null)");
  657. Indent();
  658. WriteLine("throw new ArgumentNullException(\"end\");");
  659. Outdent();
  660. WriteLine("");
  661. WriteLine("#if CRIPPLED_REFLECTION", true);
  662. var aprs = new List<string> { "IQbservableProvider", begType, endType };
  663. WriteLine("var m = InfoOf(() => " + typeName + ".FromAsyncPattern" + genArgss + "(" + string.Join(", ", aprs.Select(pt => "default(" + pt + ")")) + "));");
  664. WriteLine("#else", true);
  665. if (genArgs.Length == 0)
  666. WriteLine("var m = (MethodInfo)MethodInfo.GetCurrentMethod();");
  667. else
  668. WriteLine("var m = ((MethodInfo)MethodInfo.GetCurrentMethod()).MakeGenericMethod(" + string.Join(", ", genArgs.Select(a => "typeof(" + a + ")").ToArray()) + ");");
  669. WriteLine("#endif", true);
  670. WriteLine("return (" + string.Join(", ", lamPars) + ") => provider.CreateQuery<" + ret + ">(");
  671. Indent();
  672. WriteLine("Expression.Invoke(");
  673. Indent();
  674. WriteLine("Expression.Call(");
  675. Indent();
  676. WriteLine("null,");
  677. WriteLine("m,");
  678. WriteLine("Expression.Constant(provider, typeof(IQbservableProvider)),");
  679. WriteLine("begin,");
  680. WriteLine("end");
  681. Outdent();
  682. WriteLine(")" + (lamPars.Length > 0 ? "," : ""));
  683. var k = 0;
  684. foreach (var e in genArgs.Zip(lamPars, (g, l) => new { g, l }))
  685. {
  686. WriteLine("Expression.Constant(" + e.l + ", typeof(" + e.g + "))" + (k < i - 1 ? "," : ""));
  687. k++;
  688. }
  689. Outdent();
  690. WriteLine(")");
  691. Outdent();
  692. WriteLine(");");
  693. Outdent();
  694. WriteLine("}");
  695. WriteLine("");
  696. //if (i == 14)
  697. // WriteLine("#endif", true);
  698. }
  699. }
  700. }
  701. static TextWriter Out { get; set; }
  702. static int _indent;
  703. static void WriteLine(string s, bool noIndent = false)
  704. {
  705. foreach (var t in s.Split('\n'))
  706. Out.WriteLine((noIndent ? "" : new string(' ', _indent * 4)) + t.TrimEnd('\r'));
  707. }
  708. static void Indent()
  709. {
  710. _indent++;
  711. }
  712. static void Outdent()
  713. {
  714. _indent--;
  715. }
  716. static Type Iconize(this Type type)
  717. {
  718. if (type.IsGenericType && !type.IsGenericTypeDefinition)
  719. {
  720. var g = type.GetGenericTypeDefinition();
  721. if (g == typeof(IObservable<>))
  722. {
  723. return _qbs.MakeGenericType(type.GetGenericArguments());
  724. }
  725. else if (g == typeof(IEnumerable<>))
  726. {
  727. return typeof(IQueryable<>).MakeGenericType(type.GetGenericArguments());
  728. }
  729. }
  730. return type;
  731. }
  732. static string ToString2(this Type type)
  733. {
  734. if (type == typeof(int))
  735. return "int";
  736. else if (type == typeof(uint))
  737. return "uint";
  738. else if (type == typeof(long))
  739. return "long";
  740. else if (type == typeof(ulong))
  741. return "ulong";
  742. else if (type == typeof(float))
  743. return "float";
  744. else if (type == typeof(double))
  745. return "double";
  746. else if (type == typeof(byte))
  747. return "byte";
  748. else if (type == typeof(sbyte))
  749. return "sbyte";
  750. else if (type == typeof(bool))
  751. return "bool";
  752. else if (type == typeof(short))
  753. return "short";
  754. else if (type == typeof(ushort))
  755. return "ushort";
  756. else if (type == typeof(string))
  757. return "string";
  758. else if (type == typeof(object))
  759. return "object";
  760. else if (type == typeof(void))
  761. return "void";
  762. else if (type == typeof(decimal))
  763. return "decimal";
  764. if (type.IsArray)
  765. return type.GetElementType().ToString2() + "[" + new string(',', type.GetArrayRank() - 1) + "]";
  766. if (type.IsGenericType)
  767. {
  768. if (!type.IsGenericTypeDefinition)
  769. {
  770. var g = type.GetGenericTypeDefinition();
  771. if (g == typeof(Nullable<>))
  772. return type.GetGenericArguments()[0].ToString2() + "?";
  773. else
  774. return g.ToString2() + "<" + string.Join(", ", type.GetGenericArguments().Select(t => t.ToString2()).ToArray()) + ">";
  775. }
  776. else
  777. {
  778. var s = type.Name;
  779. return s.Substring(0, s.LastIndexOf('`'));
  780. }
  781. }
  782. return type.Name;
  783. }
  784. static string ToDocName(MethodInfo method)
  785. {
  786. var name = "M:" + ToDocName(method.DeclaringType) + "." + method.Name;
  787. var genArgs = new Type[0];
  788. if (method.IsGenericMethod)
  789. {
  790. genArgs = method.GetGenericArguments();
  791. name += "``" + genArgs.Length;
  792. }
  793. var pars = method.GetParameters();
  794. if (pars.Length > 0)
  795. {
  796. name += "(" + string.Join(",", method.GetParameters().Select(p => ToDocName(p.ParameterType, genArgs))) + ")";
  797. }
  798. return name;
  799. }
  800. static string ToDocName(Type t, params Type[] genArgs)
  801. {
  802. var i = Array.IndexOf(genArgs, t);
  803. if (i >= 0)
  804. return "``" + i;
  805. if (t.IsArray)
  806. {
  807. return ToDocName(t.GetElementType(), genArgs) + "[]";
  808. }
  809. if (t.IsGenericType)
  810. {
  811. var def = t.GetGenericTypeDefinition();
  812. var name = def.FullName.Substring(0, def.FullName.LastIndexOf("`"));
  813. var args = t.GetGenericArguments();
  814. name += "{" + string.Join(",", args.Select(a => ToDocName(a, genArgs))) + "}";
  815. return name;
  816. }
  817. else
  818. {
  819. return t.FullName;
  820. }
  821. }
  822. }
  823. }