9
9
using Microsoft . AspNetCore . Hosting ;
10
10
using Microsoft . AspNetCore . Mvc . Testing ;
11
11
using Microsoft . AspNetCore . Builder ;
12
+ using Microsoft . AspNetCore . Http ;
12
13
using Serilog . Filters ;
13
14
using Serilog . AspNetCore . Tests . Support ;
14
15
@@ -65,7 +66,51 @@ public async Task RequestLoggingMiddlewareShouldEnrich()
65
66
Assert . True ( completionEvent . Properties . ContainsKey ( "Elapsed" ) ) ;
66
67
}
67
68
68
- WebApplicationFactory < TestStartup > Setup ( ILogger logger , bool dispose , Action < RequestLoggingOptions > configureOptions = null )
69
+ [ Fact ]
70
+ public async Task RequestLoggingMiddlewareShouldEnrichWithCollectedExceptionIfNoUnhandledException ( )
71
+ {
72
+ var diagnosticContextException = new Exception ( "Exception set in diagnostic context" ) ;
73
+ var ( sink , web ) = Setup ( options =>
74
+ {
75
+ options . EnrichDiagnosticContext += ( diagnosticContext , _ ) =>
76
+ {
77
+ diagnosticContext . SetException ( diagnosticContextException ) ;
78
+ } ;
79
+ } ) ;
80
+
81
+ await web . CreateClient ( ) . GetAsync ( "/resource" ) ;
82
+
83
+ var completionEvent = sink . Writes . First ( logEvent => Matching . FromSource < RequestLoggingMiddleware > ( ) ( logEvent ) ) ;
84
+
85
+ Assert . Same ( diagnosticContextException , completionEvent . Exception ) ;
86
+ }
87
+
88
+ [ Theory ]
89
+ [ InlineData ( false ) ]
90
+ [ InlineData ( true ) ]
91
+ public async Task RequestLoggingMiddlewareShouldEnrichWithUnhandledExceptionEvenIfExceptionIsSetInDiagnosticContext ( bool setExceptionInDiagnosticContext )
92
+ {
93
+ var diagnosticContextException = new Exception ( "Exception set in diagnostic context" ) ;
94
+ var unhandledException = new Exception ( "Unhandled exception thrown in API action" ) ;
95
+ var ( sink , web ) = Setup ( options =>
96
+ {
97
+ options . EnrichDiagnosticContext += ( diagnosticContext , _ ) =>
98
+ {
99
+ if ( setExceptionInDiagnosticContext )
100
+ diagnosticContext . SetException ( diagnosticContextException ) ;
101
+ } ;
102
+ } , actionCallback : _ => throw unhandledException ) ;
103
+
104
+ Func < Task > act = ( ) => web . CreateClient ( ) . GetAsync ( "/resource" ) ;
105
+
106
+ Exception thrownException = await Assert . ThrowsAsync < Exception > ( act ) ;
107
+ var completionEvent = sink . Writes . First ( logEvent => Matching . FromSource < RequestLoggingMiddleware > ( ) ( logEvent ) ) ;
108
+ Assert . Same ( unhandledException , completionEvent . Exception ) ;
109
+ Assert . Same ( unhandledException , thrownException ) ;
110
+ }
111
+
112
+ WebApplicationFactory < TestStartup > Setup ( ILogger logger , bool dispose , Action < RequestLoggingOptions > configureOptions = null ,
113
+ Action < HttpContext > actionCallback = null )
69
114
{
70
115
var web = _web . WithWebHostBuilder (
71
116
builder => builder
@@ -80,24 +125,29 @@ WebApplicationFactory<TestStartup> Setup(ILogger logger, bool dispose, Action<Re
80
125
. Configure ( app =>
81
126
{
82
127
app . UseSerilogRequestLogging ( configureOptions ) ;
83
- app . Run ( _ => Task . CompletedTask ) ; // 200 OK
128
+ app . Run ( ctx =>
129
+ {
130
+ actionCallback ? . Invoke ( ctx ) ;
131
+ return Task . CompletedTask ;
132
+ } ) ; // 200 OK
84
133
} )
85
134
. UseSerilog ( logger , dispose ) ) ;
86
135
87
136
return web ;
88
137
}
89
138
90
- ( SerilogSink , WebApplicationFactory < TestStartup > ) Setup ( Action < RequestLoggingOptions > configureOptions = null )
139
+ ( SerilogSink , WebApplicationFactory < TestStartup > ) Setup ( Action < RequestLoggingOptions > configureOptions = null ,
140
+ Action < HttpContext > actionCallback = null )
91
141
{
92
142
var sink = new SerilogSink ( ) ;
93
143
var logger = new LoggerConfiguration ( )
94
144
. Enrich . FromLogContext ( )
95
145
. WriteTo . Sink ( sink )
96
146
. CreateLogger ( ) ;
97
147
98
- var web = Setup ( logger , true , configureOptions ) ;
148
+ var web = Setup ( logger , true , configureOptions , actionCallback ) ;
99
149
100
150
return ( sink , web ) ;
101
151
}
102
152
}
103
- }
153
+ }
0 commit comments