Program.cs 38 KB

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