Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.ObjectModel;
  4. using System.ComponentModel;
  5. using System.Diagnostics.CodeAnalysis;
  6. using System.Globalization;
  7. using System.IO;
  8. using System.Linq;
  9. using System.Net.Http;
  10. using System.Net.Http.Formatting;
  11. using System.Net.Http.Headers;
  12. using System.Web.Http.Description;
  13. using System.Xml.Linq;
  14. using Newtonsoft.Json;
  15. namespace WebAPiUtils_test.Areas.HelpPage
  16. {
  17. /// <summary>
  18. /// This class will generate the samples for the help page.
  19. /// </summary>
  20. public class HelpPageSampleGenerator
  21. {
  22. /// <summary>
  23. /// Initializes a new instance of the <see cref="HelpPageSampleGenerator"/> class.
  24. /// </summary>
  25. public HelpPageSampleGenerator()
  26. {
  27. ActualHttpMessageTypes = new Dictionary<HelpPageSampleKey, Type>();
  28. ActionSamples = new Dictionary<HelpPageSampleKey, object>();
  29. SampleObjects = new Dictionary<Type, object>();
  30. }
  31. /// <summary>
  32. /// Gets CLR types that are used as the content of <see cref="HttpRequestMessage"/> or <see cref="HttpResponseMessage"/>.
  33. /// </summary>
  34. public IDictionary<HelpPageSampleKey, Type> ActualHttpMessageTypes { get; internal set; }
  35. /// <summary>
  36. /// Gets the objects that are used directly as samples for certain actions.
  37. /// </summary>
  38. public IDictionary<HelpPageSampleKey, object> ActionSamples { get; internal set; }
  39. /// <summary>
  40. /// Gets the objects that are serialized as samples by the supported formatters.
  41. /// </summary>
  42. public IDictionary<Type, object> SampleObjects { get; internal set; }
  43. /// <summary>
  44. /// Gets the request body samples for a given <see cref="ApiDescription"/>.
  45. /// </summary>
  46. /// <param name="api">The <see cref="ApiDescription"/>.</param>
  47. /// <returns>The samples keyed by media type.</returns>
  48. public IDictionary<MediaTypeHeaderValue, object> GetSampleRequests(ApiDescription api)
  49. {
  50. return GetSample(api, SampleDirection.Request);
  51. }
  52. /// <summary>
  53. /// Gets the response body samples for a given <see cref="ApiDescription"/>.
  54. /// </summary>
  55. /// <param name="api">The <see cref="ApiDescription"/>.</param>
  56. /// <returns>The samples keyed by media type.</returns>
  57. public IDictionary<MediaTypeHeaderValue, object> GetSampleResponses(ApiDescription api)
  58. {
  59. return GetSample(api, SampleDirection.Response);
  60. }
  61. /// <summary>
  62. /// Gets the request or response body samples.
  63. /// </summary>
  64. /// <param name="api">The <see cref="ApiDescription"/>.</param>
  65. /// <param name="sampleDirection">The value indicating whether the sample is for a request or for a response.</param>
  66. /// <returns>The samples keyed by media type.</returns>
  67. public virtual IDictionary<MediaTypeHeaderValue, object> GetSample(ApiDescription api, SampleDirection sampleDirection)
  68. {
  69. if (api == null)
  70. {
  71. throw new ArgumentNullException("api");
  72. }
  73. string controllerName = api.ActionDescriptor.ControllerDescriptor.ControllerName;
  74. string actionName = api.ActionDescriptor.ActionName;
  75. IEnumerable<string> parameterNames = api.ParameterDescriptions.Select(p => p.Name);
  76. Collection<MediaTypeFormatter> formatters;
  77. Type type = ResolveType(api, controllerName, actionName, parameterNames, sampleDirection, out formatters);
  78. var samples = new Dictionary<MediaTypeHeaderValue, object>();
  79. // Use the samples provided directly for actions
  80. var actionSamples = GetAllActionSamples(controllerName, actionName, parameterNames, sampleDirection);
  81. foreach (var actionSample in actionSamples)
  82. {
  83. samples.Add(actionSample.Key.MediaType, WrapSampleIfString(actionSample.Value));
  84. }
  85. // Do the sample generation based on formatters only if an action doesn't return an HttpResponseMessage.
  86. // Here we cannot rely on formatters because we don't know what's in the HttpResponseMessage, it might not even use formatters.
  87. if (type != null && !typeof(HttpResponseMessage).IsAssignableFrom(type))
  88. {
  89. object sampleObject = GetSampleObject(type);
  90. foreach (var formatter in formatters)
  91. {
  92. foreach (MediaTypeHeaderValue mediaType in formatter.SupportedMediaTypes)
  93. {
  94. if (!samples.ContainsKey(mediaType))
  95. {
  96. object sample = GetActionSample(controllerName, actionName, parameterNames, type, formatter, mediaType, sampleDirection);
  97. // If no sample found, try generate sample using formatter and sample object
  98. if (sample == null && sampleObject != null)
  99. {
  100. sample = WriteSampleObjectUsingFormatter(formatter, sampleObject, type, mediaType);
  101. }
  102. samples.Add(mediaType, WrapSampleIfString(sample));
  103. }
  104. }
  105. }
  106. }
  107. return samples;
  108. }
  109. /// <summary>
  110. /// Search for samples that are provided directly through <see cref="ActionSamples"/>.
  111. /// </summary>
  112. /// <param name="controllerName">Name of the controller.</param>
  113. /// <param name="actionName">Name of the action.</param>
  114. /// <param name="parameterNames">The parameter names.</param>
  115. /// <param name="type">The CLR type.</param>
  116. /// <param name="formatter">The formatter.</param>
  117. /// <param name="mediaType">The media type.</param>
  118. /// <param name="sampleDirection">The value indicating whether the sample is for a request or for a response.</param>
  119. /// <returns>The sample that matches the parameters.</returns>
  120. public virtual object GetActionSample(string controllerName, string actionName, IEnumerable<string> parameterNames, Type type, MediaTypeFormatter formatter, MediaTypeHeaderValue mediaType, SampleDirection sampleDirection)
  121. {
  122. object sample;
  123. // First, try get sample provided for a specific mediaType, controllerName, actionName and parameterNames.
  124. // If not found, try get the sample provided for a specific mediaType, controllerName and actionName regardless of the parameterNames
  125. // If still not found, try get the sample provided for a specific type and mediaType
  126. if (ActionSamples.TryGetValue(new HelpPageSampleKey(mediaType, sampleDirection, controllerName, actionName, parameterNames), out sample) ||
  127. ActionSamples.TryGetValue(new HelpPageSampleKey(mediaType, sampleDirection, controllerName, actionName, new[] { "*" }), out sample) ||
  128. ActionSamples.TryGetValue(new HelpPageSampleKey(mediaType, type), out sample))
  129. {
  130. return sample;
  131. }
  132. return null;
  133. }
  134. /// <summary>
  135. /// Gets the sample object that will be serialized by the formatters.
  136. /// First, it will look at the <see cref="SampleObjects"/>. If no sample object is found, it will try to create one using <see cref="ObjectGenerator"/>.
  137. /// </summary>
  138. /// <param name="type">The type.</param>
  139. /// <returns>The sample object.</returns>
  140. public virtual object GetSampleObject(Type type)
  141. {
  142. object sampleObject;
  143. if (!SampleObjects.TryGetValue(type, out sampleObject))
  144. {
  145. // Try create a default sample object
  146. ObjectGenerator objectGenerator = new ObjectGenerator();
  147. sampleObject = objectGenerator.GenerateObject(type);
  148. }
  149. return sampleObject;
  150. }
  151. /// <summary>
  152. /// Resolves the type of the action parameter or return value when <see cref="HttpRequestMessage"/> or <see cref="HttpResponseMessage"/> is used.
  153. /// </summary>
  154. /// <param name="api">The <see cref="ApiDescription"/>.</param>
  155. /// <param name="controllerName">Name of the controller.</param>
  156. /// <param name="actionName">Name of the action.</param>
  157. /// <param name="parameterNames">The parameter names.</param>
  158. /// <param name="sampleDirection">The value indicating whether the sample is for a request or a response.</param>
  159. /// <param name="formatters">The formatters.</param>
  160. [SuppressMessage("Microsoft.Design", "CA1021:AvoidOutParameters", Justification = "This is only used in advanced scenarios.")]
  161. public virtual Type ResolveType(ApiDescription api, string controllerName, string actionName, IEnumerable<string> parameterNames, SampleDirection sampleDirection, out Collection<MediaTypeFormatter> formatters)
  162. {
  163. if (!Enum.IsDefined(typeof(SampleDirection), sampleDirection))
  164. {
  165. throw new InvalidEnumArgumentException("sampleDirection", (int)sampleDirection, typeof(SampleDirection));
  166. }
  167. if (api == null)
  168. {
  169. throw new ArgumentNullException("api");
  170. }
  171. Type type;
  172. if (ActualHttpMessageTypes.TryGetValue(new HelpPageSampleKey(sampleDirection, controllerName, actionName, parameterNames), out type) ||
  173. ActualHttpMessageTypes.TryGetValue(new HelpPageSampleKey(sampleDirection, controllerName, actionName, new[] { "*" }), out type))
  174. {
  175. // Re-compute the supported formatters based on type
  176. Collection<MediaTypeFormatter> newFormatters = new Collection<MediaTypeFormatter>();
  177. foreach (var formatter in api.ActionDescriptor.Configuration.Formatters)
  178. {
  179. if (IsFormatSupported(sampleDirection, formatter, type))
  180. {
  181. newFormatters.Add(formatter);
  182. }
  183. }
  184. formatters = newFormatters;
  185. }
  186. else
  187. {
  188. switch (sampleDirection)
  189. {
  190. case SampleDirection.Request:
  191. ApiParameterDescription requestBodyParameter = api.ParameterDescriptions.FirstOrDefault(p => p.Source == ApiParameterSource.FromBody);
  192. type = requestBodyParameter == null ? null : requestBodyParameter.ParameterDescriptor.ParameterType;
  193. formatters = api.SupportedRequestBodyFormatters;
  194. break;
  195. case SampleDirection.Response:
  196. default:
  197. type = api.ResponseDescription.ResponseType ?? api.ResponseDescription.DeclaredType;
  198. formatters = api.SupportedResponseFormatters;
  199. break;
  200. }
  201. }
  202. return type;
  203. }
  204. /// <summary>
  205. /// Writes the sample object using formatter.
  206. /// </summary>
  207. /// <param name="formatter">The formatter.</param>
  208. /// <param name="value">The value.</param>
  209. /// <param name="type">The type.</param>
  210. /// <param name="mediaType">Type of the media.</param>
  211. /// <returns></returns>
  212. [SuppressMessage("Microsoft.Design", "CA1031:DoNotCatchGeneralExceptionTypes", Justification = "The exception is recorded as InvalidSample.")]
  213. public virtual object WriteSampleObjectUsingFormatter(MediaTypeFormatter formatter, object value, Type type, MediaTypeHeaderValue mediaType)
  214. {
  215. if (formatter == null)
  216. {
  217. throw new ArgumentNullException("formatter");
  218. }
  219. if (mediaType == null)
  220. {
  221. throw new ArgumentNullException("mediaType");
  222. }
  223. object sample = String.Empty;
  224. MemoryStream ms = null;
  225. HttpContent content = null;
  226. try
  227. {
  228. if (formatter.CanWriteType(type))
  229. {
  230. ms = new MemoryStream();
  231. content = new ObjectContent(type, value, formatter, mediaType);
  232. formatter.WriteToStreamAsync(type, value, ms, content, null).Wait();
  233. ms.Position = 0;
  234. StreamReader reader = new StreamReader(ms);
  235. string serializedSampleString = reader.ReadToEnd();
  236. if (mediaType.MediaType.ToUpperInvariant().Contains("XML"))
  237. {
  238. serializedSampleString = TryFormatXml(serializedSampleString);
  239. }
  240. else if (mediaType.MediaType.ToUpperInvariant().Contains("JSON"))
  241. {
  242. serializedSampleString = TryFormatJson(serializedSampleString);
  243. }
  244. sample = new TextSample(serializedSampleString);
  245. }
  246. else
  247. {
  248. sample = new InvalidSample(String.Format(
  249. CultureInfo.CurrentCulture,
  250. "Failed to generate the sample for media type '{0}'. Cannot use formatter '{1}' to write type '{2}'.",
  251. mediaType,
  252. formatter.GetType().Name,
  253. type.Name));
  254. }
  255. }
  256. catch (Exception e)
  257. {
  258. sample = new InvalidSample(String.Format(
  259. CultureInfo.CurrentCulture,
  260. "An exception has occurred while using the formatter '{0}' to generate sample for media type '{1}'. Exception message: {2}",
  261. formatter.GetType().Name,
  262. mediaType.MediaType,
  263. e.Message));
  264. }
  265. finally
  266. {
  267. if (ms != null)
  268. {
  269. ms.Dispose();
  270. }
  271. if (content != null)
  272. {
  273. content.Dispose();
  274. }
  275. }
  276. return sample;
  277. }
  278. [SuppressMessage("Microsoft.Design", "CA1031:DoNotCatchGeneralExceptionTypes", Justification = "Handling the failure by returning the original string.")]
  279. private static string TryFormatJson(string str)
  280. {
  281. try
  282. {
  283. object parsedJson = JsonConvert.DeserializeObject(str);
  284. return JsonConvert.SerializeObject(parsedJson, Formatting.Indented);
  285. }
  286. catch
  287. {
  288. // can't parse JSON, return the original string
  289. return str;
  290. }
  291. }
  292. [SuppressMessage("Microsoft.Design", "CA1031:DoNotCatchGeneralExceptionTypes", Justification = "Handling the failure by returning the original string.")]
  293. private static string TryFormatXml(string str)
  294. {
  295. try
  296. {
  297. XDocument xml = XDocument.Parse(str);
  298. return xml.ToString();
  299. }
  300. catch
  301. {
  302. // can't parse XML, return the original string
  303. return str;
  304. }
  305. }
  306. private static bool IsFormatSupported(SampleDirection sampleDirection, MediaTypeFormatter formatter, Type type)
  307. {
  308. switch (sampleDirection)
  309. {
  310. case SampleDirection.Request:
  311. return formatter.CanReadType(type);
  312. case SampleDirection.Response:
  313. return formatter.CanWriteType(type);
  314. }
  315. return false;
  316. }
  317. private IEnumerable<KeyValuePair<HelpPageSampleKey, object>> GetAllActionSamples(string controllerName, string actionName, IEnumerable<string> parameterNames, SampleDirection sampleDirection)
  318. {
  319. HashSet<string> parameterNamesSet = new HashSet<string>(parameterNames, StringComparer.OrdinalIgnoreCase);
  320. foreach (var sample in ActionSamples)
  321. {
  322. HelpPageSampleKey sampleKey = sample.Key;
  323. if (String.Equals(controllerName, sampleKey.ControllerName, StringComparison.OrdinalIgnoreCase) &&
  324. String.Equals(actionName, sampleKey.ActionName, StringComparison.OrdinalIgnoreCase) &&
  325. (sampleKey.ParameterNames.SetEquals(new[] { "*" }) || parameterNamesSet.SetEquals(sampleKey.ParameterNames)) &&
  326. sampleDirection == sampleKey.SampleDirection)
  327. {
  328. yield return sample;
  329. }
  330. }
  331. }
  332. private static object WrapSampleIfString(object sample)
  333. {
  334. string stringSample = sample as string;
  335. if (stringSample != null)
  336. {
  337. return new TextSample(stringSample);
  338. }
  339. return sample;
  340. }
  341. }
  342. }