-
Notifications
You must be signed in to change notification settings - Fork 33
/
Copy pathRequestExtensions.Helpers.cs
110 lines (95 loc) · 4.07 KB
/
RequestExtensions.Helpers.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
using System;
using System.Collections.Generic;
using System.Reflection;
using System.Text;
namespace KubeClient.Http
{
using ValueProviders;
/// <summary>
/// Helper methods for <see cref="HttpRequest"/> / <see cref="IHttpRequest"/> extensions.
/// </summary>
public static partial class RequestExtensions
{
/// <summary>
/// Configure the request URI (and template status) in the request properties.
/// </summary>
/// <param name="requestProperties">
/// The request properties to modify.
/// </param>
/// <param name="requestUri">
/// The request URI.
/// </param>
static void SetUri(this IDictionary<string, object> requestProperties, Uri requestUri)
{
if (requestProperties == null)
throw new ArgumentNullException(nameof(requestProperties));
if (requestUri == null)
throw new ArgumentNullException(nameof(requestUri));
requestProperties[nameof(IHttpRequest.Uri)] = requestUri;
requestProperties[nameof(IHttpRequest.IsUriTemplate)] = UriTemplate.IsTemplate(requestUri);
}
/// <summary>
/// Ensure that the specified string is surrounted by quotes.
/// </summary>
/// <param name="str">
/// The string to examine.
/// </param>
/// <returns>
/// The string, with quotes prepended / appended as required.
/// </returns>
/// <remarks>
/// Some HTTP headers (such as If-Match) require their values to be quoted.
/// </remarks>
static string EnsureQuoted(string str)
{
if (str == null)
throw new ArgumentNullException(nameof(str));
if (str.Length == 0)
return "\"\"";
StringBuilder quotedStringBuilder = new StringBuilder(str);
if (quotedStringBuilder[0] != '\"')
quotedStringBuilder.Insert(0, '\"');
if (quotedStringBuilder[quotedStringBuilder.Length - 1] != '\"')
quotedStringBuilder.Append('\"');
return quotedStringBuilder.ToString();
}
/// <summary>
/// Convert the specified object's properties to deferred parameters.
/// </summary>
/// <typeparam name="TContext">
/// The type of object used by the request when resolving deferred template parameters.
/// </typeparam>
/// <typeparam name="TParameters">
/// The type of object whose properties will form the parameters.
/// </typeparam>
/// <param name="parameters">
/// The object whose properties will form the parameters.
/// </param>
/// <returns>
/// A sequence of key / value pairs representing the parameters.
/// </returns>
static IEnumerable<KeyValuePair<string, IValueProvider<TContext, string>>> CreateDeferredParameters<TContext, TParameters>(this TParameters parameters)
{
if (Equals(parameters, null))
throw new ArgumentNullException(nameof(parameters));
// TODO: Refactor PropertyInfo retrieval logic (move it out to an extension method).
// Yes yes yes, reflection might be "slow", but it's still blazingly fast compared to making a request over the network.
foreach (PropertyInfo property in typeof(TParameters).GetTypeInfo().DeclaredProperties)
{
// Ignore write-only properties.
if (!property.CanRead)
continue;
// Public instance properties only.
if (!property.GetMethod.IsPublic || property.GetMethod.IsStatic)
continue;
yield return new KeyValuePair<string, IValueProvider<TContext, string>>(
property.Name,
ValueProvider<TContext>.FromSelector(
context => property.GetValue(parameters)
)
.Convert().ValueToString()
);
}
}
}
}