Adjusted Gzip stream to reduce response times.

pull/2/head
Taloth Saldono 9 years ago
parent 19aded7a15
commit 87d00abdf1

@ -5,6 +5,7 @@ using System.Linq;
using Nancy; using Nancy;
using Nancy.Bootstrapper; using Nancy.Bootstrapper;
using NLog; using NLog;
using NzbDrone.Common.Extensions;
namespace NzbDrone.Api.Extensions.Pipelines namespace NzbDrone.Api.Extensions.Pipelines
{ {
@ -21,50 +22,62 @@ namespace NzbDrone.Api.Extensions.Pipelines
public void Register(IPipelines pipelines) public void Register(IPipelines pipelines)
{ {
pipelines.AfterRequest.AddItemToEndOfPipeline(c => CompressResponse(c.Request, c.Response)); pipelines.AfterRequest.AddItemToEndOfPipeline(CompressResponse);
} }
private Response CompressResponse(Request request, Response response) private void CompressResponse(NancyContext context)
{ {
var request = context.Request;
var response = context.Response;
try try
{ {
if ( if (
!response.ContentType.Contains("image") !response.ContentType.Contains("image")
&& !response.ContentType.Contains("font") && !response.ContentType.Contains("font")
&& request.Headers.AcceptEncoding.Any(x => x.Contains("gzip")) && request.Headers.AcceptEncoding.Any(x => x.Contains("gzip"))
&& (!response.Headers.ContainsKey("Content-Encoding") || response.Headers["Content-Encoding"] != "gzip")) && !AlreadyGzipEncoded(response)
&& !ContentLengthIsTooSmall(response))
{ {
var data = new MemoryStream(); var contents = response.Contents;
response.Contents.Invoke(data);
data.Position = 0; response.Headers["Content-Encoding"] = "gzip";
if (data.Length < 1024) response.Contents = responseStream =>
{ {
response.Contents = stream => using (var gzip = new GZipStream(responseStream, CompressionMode.Compress, true))
using (var buffered = new BufferedStream(gzip, 8192))
{ {
data.CopyTo(stream); contents.Invoke(buffered);
stream.Flush();
};
} }
else
{
response.Headers["Content-Encoding"] = "gzip";
response.Contents = s =>
{
var gzip = new GZipStream(s, CompressionMode.Compress, true);
data.CopyTo(gzip);
gzip.Close();
}; };
} }
} }
return response;
}
catch (Exception ex) catch (Exception ex)
{ {
_logger.Error(ex, "Unable to gzip response"); _logger.Error(ex, "Unable to gzip response");
throw; throw;
} }
} }
private static bool ContentLengthIsTooSmall(Response response)
{
var contentLength = response.Headers.GetValueOrDefault("Content-Length");
if (contentLength != null && long.Parse(contentLength) < 1024)
{
return true;
}
return false;
}
private static bool AlreadyGzipEncoded(Response response)
{
var contentEncoding = response.Headers.GetValueOrDefault("Content-Encoding");
if (contentEncoding == "gzip")
{
return true;
}
return false;
}
} }
} }
Loading…
Cancel
Save