Owin.Compression (Deflate / GZip) module ("middleware") for the Microsoft OWIN pipeline. It can be used with .NET Full, .NET Core, .NET Standard, .NET6.0, and so on. It also works with Selfhost and AspNetCore (e.g. with Kestrel, which is OWIN based server).
It compresses the web request responses to make the transfer smaller, and it supports eTag caching.
The default compression used is deflate, then gzip, as deflate should be faster.
- When the server reads the content before compression, it calculates a hash-code over it.
- The hash-code is sent as ETag response header to the client with the response
- The next time the client asks for the same resource, it sends an If-None-Match header in the request with the same value.
- After the server reads the content before the compression, it calculates a hash-code over it. If it matches the If-None-Match of the request, the server can skip the compression and skip the sending and just send http status code 304 to the client which means "use what you have, it's not modified since".
This example demonstrates using MapCompressionModule-function defined in this sample library.
using System;
using Owin;
[assembly: Microsoft.Owin.OwinStartup(typeof(MyServer.MyWebStartup))]
namespace MyServer
{
class MyWebStartup
{
public void Configuration(Owin.IAppBuilder app)
{
// This will compress the whole request, if you want to use e.g. Microsoft.Owin.StaticFiles server:
// app.UseCompressionModule()
var settings = OwinCompression.DefaultCompressionSettingsWithPath("c:\\temp\\"); //" server path
//or var settings = new CompressionSettings( ... )
app.MapCompressionModule("/zipped", settings);
}
}
class Program
{
static void Main(string[] args)
{
Microsoft.Owin.Hosting.WebApp.Start<MyWebStartup>("http://*:8080"); //" run on localhost.
Console.WriteLine("Server started... Press enter to exit.");
Console.ReadLine();
}
}
}
|
And now your files are smaller than with e.g. just Microsoft.Owin.StaticFiles -library server:
Even though the browser sees everything as plain text, the traffic is actually transferred in compressed format.
You can monitor the traffic with e.g. Fiddler.
Running on OWIN Self-Host (Microsoft.Owin.Hosting) with static files server (Microsoft.Owin.StaticFiles)
and compressing only the ".json"-responses (and files) on-the-fly, with only gzip and not deflate:
using System;
using Owin;
[assembly: Microsoft.Owin.OwinStartup(typeof(MyServer.MyWebStartup))]
namespace MyServer
{
class MyWebStartup
{
public void Configuration(Owin.IAppBuilder app)
{
var settings = new CompressionSettings(
serverPath: "",
allowUnknonwnFiletypes: false,
allowRootDirectories: false,
cacheExpireTime: Microsoft.FSharp.Core.FSharpOption<DateTimeOffset>.None,
allowedExtensionAndMimeTypes:
new[] { Tuple.Create(".json", "application/json") },
minimumSizeToCompress: 1000,
streamingDisabled: false,
deflateDisabled: true
);
app.UseCompressionModule(settings);
}
}
class Program
{
static void Main(string[] args)
{
Microsoft.Owin.Hosting.WebApp.Start<MyWebStartup>("http://*:8080");
Console.WriteLine("Server started... Press enter to exit.");
Console.ReadLine();
}
}
}
|
Running on OWIN Self-Host (Microsoft.Owin.Hosting) with static files server (Microsoft.Owin.StaticFiles)
and compressing all the responses (and files) on-the-fly. This example is in F-Sharp (and can be run with F#-interactive):
#r "Owin.dll"
#r "Microsoft.Owin.dll"
#r "Microsoft.Owin.FileSystems.dll"
#r "Microsoft.Owin.Hosting.dll"
#r "Microsoft.Owin.StaticFiles.dll"
#r "System.Configuration.dll"
#r "Owin.Compression.dll"
open Owin
open System
module Examples =
type MyStartup() =
member __.Configuration(app:Owin.IAppBuilder) =
let app1 = app.UseCompressionModule()
app1.UseFileServer "/." |> ignore
()
let server = Microsoft.Owin.Hosting.WebApp.Start<MyStartup> "http://*:6000"
Console.WriteLine "Press Enter to stop & quit."
Console.ReadLine() |> ignore
server.Dispose()
Running on ASP.NET Core web API on .NET 6.0. You can use C# but this example is in F#
just because of the shorter syntax. The full project is available in tests-folder of this project:
open System
open Microsoft.AspNetCore.Builder
open Microsoft.Extensions.DependencyInjection
open Microsoft.Extensions.Hosting
open Owin
module Program =
[<EntryPoint>]
let main args =
let builder = WebApplication.CreateBuilder args
builder.Services.AddControllers() |> ignore
let app = builder.Build()
let compressionSetting =
{OwinCompression.DefaultCompressionSettings with
CacheExpireTime = Some (DateTimeOffset.Now.AddDays 7.)
AllowUnknonwnFiletypes = true
StreamingDisabled = true
}
(app :> IApplicationBuilder).UseCompressionModule(compressionSetting) |> ignore
app.MapControllers() |> ignore
app.Run()
0
https://github.com/Thorium/Owin.Compression/tree/master/tests/Aspnet.Core.WebAPI.Test
More complete examples can be found here.
The library comes with comprehensible documentation.
It can include tutorials automatically generated from *.fsx
files in the content folder.
The API reference is automatically generated from Markdown comments in the library implementation.
Tutorial contains a further explanation of this sample library.
-
API Reference contains automatically generated documentation for all types, modules
and functions in the library. This includes additional brief samples on using most of the
functions.
The project is hosted on GitHub where you can report issues, fork
the project and submit pull requests. If you're adding a new public API, please also
consider adding samples that can be turned into documentation. You might
also want to read the library design notes to understand how it works.
The library is available under a Public Domain license, which allows modification and
redistribution for both commercial and non-commercial purposes. For more information see the
License file in the GitHub repository.
namespace Owin
namespace System
module Examples
from Index
Multiple items
type MyStartup =
new: unit -> MyStartup
member Configuration: app: IAppBuilder -> unit
--------------------
new: unit -> MyStartup
val app: IAppBuilder
type IAppBuilder =
member Build: returnType: Type -> obj
member New: unit -> IAppBuilder
member Use: middleware: obj * [<ParamArray>] args: obj[] -> IAppBuilder
member Properties: IDictionary<string,obj>
val app1: obj
val ignore: value: 'T -> unit
<summary>Ignore the passed value. This is often used to throw away results of a computation.</summary>
<param name="value">The value to ignore.</param>
<example id="min-example"><code lang="fsharp">
ignore 55555 // Evaluates to ()
</code></example>
val server: obj
namespace Microsoft
type Console =
static member Beep: unit -> unit + 1 overload
static member Clear: unit -> unit
static member GetCursorPosition: unit -> struct (int * int)
static member MoveBufferArea: sourceLeft: int * sourceTop: int * sourceWidth: int * sourceHeight: int * targetLeft: int * targetTop: int -> unit + 1 overload
static member OpenStandardError: unit -> Stream + 1 overload
static member OpenStandardInput: unit -> Stream + 1 overload
static member OpenStandardOutput: unit -> Stream + 1 overload
static member Read: unit -> int
static member ReadKey: unit -> ConsoleKeyInfo + 1 overload
static member ReadLine: unit -> string
...
<summary>Represents the standard input, output, and error streams for console applications. This class cannot be inherited.</summary>
Console.WriteLine() : unit
(+0 other overloads)
Console.WriteLine(value: uint64) : unit
(+0 other overloads)
Console.WriteLine(value: uint32) : unit
(+0 other overloads)
Console.WriteLine(value: string) : unit
(+0 other overloads)
Console.WriteLine(value: float32) : unit
(+0 other overloads)
Console.WriteLine(value: obj) : unit
(+0 other overloads)
Console.WriteLine(value: int64) : unit
(+0 other overloads)
Console.WriteLine(value: int) : unit
(+0 other overloads)
Console.WriteLine(value: float) : unit
(+0 other overloads)
Console.WriteLine(value: decimal) : unit
(+0 other overloads)
Console.ReadLine() : string
Multiple items
type EntryPointAttribute =
inherit Attribute
new: unit -> EntryPointAttribute
<summary>Adding this attribute to a function indicates it is the entrypoint for an application.
If this attribute is not specified for an EXE then the initialization implicit in the
module bindings in the last file in the compilation sequence are used as the entrypoint.</summary>
<category>Attributes</category>
--------------------
new: unit -> EntryPointAttribute
val main: args: string[] -> int
val args: string[]
val builder: obj
val app: obj
val compressionSetting: obj
union case Option.Some: Value: 'T -> Option<'T>
<summary>The representation of "Value of type 'T"</summary>
<param name="Value">The input value.</param>
<returns>An option representing the value.</returns>
Multiple items
[<Struct>]
type DateTimeOffset =
new: date: DateOnly * time: TimeOnly * offset: TimeSpan -> unit + 8 overloads
member Add: timeSpan: TimeSpan -> DateTimeOffset
member AddDays: days: float -> DateTimeOffset
member AddHours: hours: float -> DateTimeOffset
member AddMicroseconds: microseconds: float -> DateTimeOffset
member AddMilliseconds: milliseconds: float -> DateTimeOffset
member AddMinutes: minutes: float -> DateTimeOffset
member AddMonths: months: int -> DateTimeOffset
member AddSeconds: seconds: float -> DateTimeOffset
member AddTicks: ticks: int64 -> DateTimeOffset
...
<summary>Represents a point in time, typically expressed as a date and time of day, relative to Coordinated Universal Time (UTC).</summary>
--------------------
DateTimeOffset ()
DateTimeOffset(dateTime: DateTime) : DateTimeOffset
DateTimeOffset(dateTime: DateTime, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(ticks: int64, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(date: DateOnly, time: TimeOnly, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(year: int, month: int, day: int, hour: int, minute: int, second: int, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(year: int, month: int, day: int, hour: int, minute: int, second: int, millisecond: int, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(year: int, month: int, day: int, hour: int, minute: int, second: int, millisecond: int, calendar: Globalization.Calendar, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(year: int, month: int, day: int, hour: int, minute: int, second: int, millisecond: int, microsecond: int, offset: TimeSpan) : DateTimeOffset
DateTimeOffset(year: int, month: int, day: int, hour: int, minute: int, second: int, millisecond: int, microsecond: int, calendar: Globalization.Calendar, offset: TimeSpan) : DateTimeOffset
property DateTimeOffset.Now: DateTimeOffset with get
<summary>Gets a <see cref="T:System.DateTimeOffset" /> object that is set to the current date and time on the current computer, with the offset set to the local time's offset from Coordinated Universal Time (UTC).</summary>
<returns>A <see cref="T:System.DateTimeOffset" /> object whose date and time is the current local time and whose offset is the local time zone's offset from Coordinated Universal Time (UTC).</returns>
DateTimeOffset.AddDays(days: float) : DateTimeOffset