-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: adds transfer rate extension (#96)
* feat: instead of trying to insert preferred behaviors at the top of the behavior list, use a comparer to resort the behaviors. This is more reliable in case of custom extensions messing with the behavior list. * feat: adds TransferRate extension that wraps the response content to introduce artificial slower responses
- Loading branch information
Showing
11 changed files
with
533 additions
and
17 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,149 @@ | ||
using System.Globalization; | ||
using MockHttp.IO; | ||
|
||
// ReSharper disable once CheckNamespace | ||
namespace MockHttp; | ||
|
||
/// <summary> | ||
/// Defines different types of bit rates to simulate a slow network. | ||
/// </summary> | ||
public sealed class BitRate | ||
{ | ||
private readonly Func<int> _factory; | ||
private readonly string _name; | ||
|
||
private BitRate(Func<int> factory, string name) | ||
{ | ||
_factory = factory; | ||
_name = name; | ||
} | ||
|
||
/// <inheritdoc /> | ||
public override string ToString() | ||
{ | ||
return $"{GetType().Name}.{_name}"; | ||
} | ||
|
||
/// <summary> | ||
/// 2G (mobile network) bit rate. (~64kbps). | ||
/// </summary> | ||
public static BitRate TwoG() | ||
{ | ||
return Create(64_000, nameof(TwoG)); | ||
} | ||
|
||
/// <summary> | ||
/// 3G (mobile network) bit rate. (~2Mbps) | ||
/// </summary> | ||
public static BitRate ThreeG() | ||
{ | ||
return Create(2_000_000, nameof(ThreeG)); | ||
} | ||
|
||
/// <summary> | ||
/// 4G (mobile network) bit rate. (~64Mbps) | ||
/// </summary> | ||
public static BitRate FourG() | ||
{ | ||
return Create(64_000_000, nameof(FourG)); | ||
} | ||
|
||
/// <summary> | ||
/// 5G (mobile network) bit rate. (~512Mbps) | ||
/// </summary> | ||
public static BitRate FiveG() | ||
{ | ||
return Create(512_000_000, nameof(FiveG)); | ||
} | ||
|
||
/// <summary> | ||
/// 10 Mbps. | ||
/// </summary> | ||
public static BitRate TenMegabit() | ||
{ | ||
return Create(10_000_000, nameof(TenMegabit)); | ||
} | ||
|
||
/// <summary> | ||
/// 100 Mbps. | ||
/// </summary> | ||
public static BitRate OneHundredMegabit() | ||
{ | ||
return Create(100_000_000, nameof(OneHundredMegabit)); | ||
} | ||
|
||
/// <summary> | ||
/// 1 Gbps. | ||
/// </summary> | ||
public static BitRate OneGigabit() | ||
{ | ||
return Create(1_000_000_000, nameof(OneGigabit)); | ||
} | ||
|
||
/// <summary> | ||
/// Converts a bit rate to an integer representing the bit rate in bits per second. | ||
/// </summary> | ||
/// <param name="bitRate"></param> | ||
/// <returns></returns> | ||
public static explicit operator int (BitRate bitRate) | ||
{ | ||
return ToInt32(bitRate); | ||
} | ||
|
||
/// <summary> | ||
/// Converts a bit rate to an integer representing the bit rate in bits per second. | ||
/// </summary> | ||
/// <param name="bitRate"></param> | ||
/// <returns></returns> | ||
public static explicit operator BitRate (int bitRate) | ||
{ | ||
return FromInt32(bitRate); | ||
} | ||
|
||
/// <summary> | ||
/// Converts a bit rate to an integer representing the bit rate in bits per second. | ||
/// </summary> | ||
/// <param name="bitRate">The bit rate.</param> | ||
/// <returns>The underlying bit rate value.</returns> | ||
public static int ToInt32(BitRate bitRate) | ||
{ | ||
// ReSharper disable once ConditionalAccessQualifierIsNonNullableAccordingToAPIContract | ||
return bitRate?._factory() ?? -1; | ||
} | ||
|
||
/// <summary> | ||
/// Convert an integer bit rate (in bits per second) to a <see cref="BitRate" />. | ||
/// </summary> | ||
/// <param name="bitRate">The bit rate.</param> | ||
/// <returns>The bit rate.</returns> | ||
public static BitRate FromInt32(int bitRate) | ||
{ | ||
return Create(bitRate, FormatBps(bitRate)); | ||
} | ||
|
||
private static BitRate Create(int bitRate, string name) | ||
{ | ||
if (bitRate <= 0) | ||
{ | ||
throw new ArgumentOutOfRangeException(nameof(bitRate)); | ||
} | ||
|
||
return new BitRate(() => bitRate, name); | ||
} | ||
|
||
private static string FormatBps(long value) | ||
{ | ||
return BpsToString().ToString(CultureInfo.InvariantCulture); | ||
|
||
FormattableString BpsToString() | ||
{ | ||
return value switch | ||
{ | ||
< 1_000 => $"Around({value}bps)", | ||
< 1_000_000 => $"Around({(double)value / 1_000:#.##}kbps)", | ||
< 1_000_000_000 => $"Around({(double)value / 1_000_000:#.##}Mbps)", | ||
_ => $"Around({(double)value / 1_000_000_000:#.##}Gbps)" | ||
}; | ||
} | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,65 @@ | ||
using System.Net; | ||
using MockHttp.IO; | ||
|
||
namespace MockHttp.Responses; | ||
|
||
internal sealed class TransferRateBehavior : IResponseBehavior | ||
{ | ||
private readonly int _bitRate; | ||
|
||
public TransferRateBehavior(int bitRate) | ||
{ | ||
if (bitRate < RateLimitedStream.MinBitRate) | ||
{ | ||
throw new ArgumentOutOfRangeException(nameof(bitRate), $"Bit rate must be higher than or equal to {RateLimitedStream.MinBitRate}."); | ||
} | ||
|
||
_bitRate = bitRate; | ||
} | ||
|
||
public async Task HandleAsync(MockHttpRequestContext requestContext, HttpResponseMessage responseMessage, ResponseHandlerDelegate next, CancellationToken cancellationToken) | ||
{ | ||
await next(requestContext, responseMessage, cancellationToken).ConfigureAwait(false); | ||
responseMessage.Content = new RateLimitedHttpContent(responseMessage.Content, _bitRate); | ||
} | ||
|
||
private sealed class RateLimitedHttpContent : HttpContent | ||
{ | ||
private readonly int _bitRate; | ||
private readonly HttpContent _originalContent; | ||
|
||
internal RateLimitedHttpContent(HttpContent originalContent, int bitRate) | ||
{ | ||
_originalContent = originalContent; | ||
_bitRate = bitRate; | ||
} | ||
|
||
protected override async Task SerializeToStreamAsync(Stream stream, TransportContext? context) | ||
{ | ||
Stream originalStream = await _originalContent.ReadAsStreamAsync().ConfigureAwait(false); | ||
var rateLimitedStream = new RateLimitedStream(originalStream, _bitRate); | ||
#if NETSTANDARD2_1_OR_GREATER || NET6_0_OR_GREATER | ||
await using (originalStream) | ||
await using (rateLimitedStream) | ||
#else | ||
using (originalStream) | ||
using (rateLimitedStream) | ||
#endif | ||
{ | ||
await rateLimitedStream.CopyToAsync(stream).ConfigureAwait(false); | ||
} | ||
} | ||
|
||
protected override bool TryComputeLength(out long length) | ||
{ | ||
long? contentLength = _originalContent.Headers.ContentLength; | ||
length = 0; | ||
if (contentLength.HasValue) | ||
{ | ||
length = contentLength.Value; | ||
} | ||
|
||
return contentLength.HasValue; | ||
} | ||
} | ||
} |
Oops, something went wrong.