/
ErrorLogPageFactory.cs
223 lines (180 loc) · 7.39 KB
/
ErrorLogPageFactory.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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
#region License, Terms and Author(s)
//
// ELMAH - Error Logging Modules and Handlers for ASP.NET
// Copyright (c) 2004-9 Atif Aziz. All rights reserved.
//
// Author(s):
//
// Atif Aziz, http://www.raboof.com
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
#endregion
namespace Elmah
{
#region Imports
using System;
using System.Collections;
using System.Web;
using CultureInfo = System.Globalization.CultureInfo;
using Encoding = System.Text.Encoding;
#endregion
/// <summary>
/// HTTP handler factory that dispenses handlers for rendering views and
/// resources needed to display the error log.
/// </summary>
public class ErrorLogPageFactory : IHttpHandlerFactory
{
private static readonly object _authorizationHandlersKey = new object();
private static readonly IRequestAuthorizationHandler[] _zeroAuthorizationHandlers = new IRequestAuthorizationHandler[0];
/// <summary>
/// Returns an object that implements the <see cref="IHttpHandler"/>
/// interface and which is responsible for serving the request.
/// </summary>
/// <returns>
/// A new <see cref="IHttpHandler"/> object that processes the request.
/// </returns>
public virtual IHttpHandler GetHandler(HttpContext context, string requestType, string url, string pathTranslated)
{
//
// The request resource is determined by the looking up the
// value of the PATH_INFO server variable.
//
string resource = context.Request.PathInfo.Length == 0 ? string.Empty :
context.Request.PathInfo.Substring(1).ToLower(CultureInfo.InvariantCulture);
IHttpHandler handler = FindHandler(resource);
if (handler == null)
throw new HttpException(404, "Resource not found.");
//
// Check if authorized then grant or deny request.
//
int authorized = IsAuthorized(context);
if (authorized == 0
|| (authorized < 0 // Compatibility case...
&& !HttpRequestSecurity.IsLocal(context.Request)
&& !SecurityConfiguration.Default.AllowRemoteAccess))
{
(new ManifestResourceHandler("RemoteAccessError.htm", "text/html")).ProcessRequest(context);
HttpResponse response = context.Response;
response.Status = "403 Forbidden";
response.End();
//
// HttpResponse.End docs say that it throws
// ThreadAbortException and so should never end up here but
// that's not been the observation in the debugger. So as a
// precautionary measure, bail out anyway.
//
return null;
}
return handler;
}
private static IHttpHandler FindHandler(string name)
{
Debug.Assert(name != null);
switch (name)
{
case "detail":
return new ErrorDetailPage();
case "html":
return new ErrorHtmlPage();
case "xml":
return new ErrorXmlHandler();
case "json":
return new ErrorJsonHandler();
case "rss":
return new ErrorRssHandler();
case "digestrss":
return new ErrorDigestRssHandler();
case "download":
return new ErrorLogDownloadHandler();
case "stylesheet":
return new ManifestResourceHandler("ErrorLog.css",
"text/css", Encoding.GetEncoding("Windows-1252"));
case "test":
throw new TestException();
case "about":
return new AboutPage();
default:
return name.Length == 0 ? new ErrorLogPage() : null;
}
}
/// <summary>
/// Enables the factory to reuse an existing handler instance.
/// </summary>
public virtual void ReleaseHandler(IHttpHandler handler)
{
}
/// <summary>
/// Determines if the request is authorized by objects implementing
/// <see cref="IRequestAuthorizationHandler" />.
/// </summary>
/// <returns>
/// Returns zero if unauthorized, a value greater than zero if
/// authorized otherwise a value less than zero if no handlers
/// were available to answer.
/// </returns>
private static int IsAuthorized(HttpContext context)
{
Debug.Assert(context != null);
int authorized = /* uninitialized */ -1;
IEnumerator authorizationHandlers = GetAuthorizationHandlers(context).GetEnumerator();
while (authorized != 0 && authorizationHandlers.MoveNext())
{
IRequestAuthorizationHandler authorizationHandler = (IRequestAuthorizationHandler)authorizationHandlers.Current;
authorized = authorizationHandler.Authorize(context) ? 1 : 0;
}
return authorized;
}
private static IList GetAuthorizationHandlers(HttpContext context)
{
Debug.Assert(context != null);
object key = _authorizationHandlersKey;
IList handlers = (IList)context.Items[key];
if (handlers == null)
{
const int capacity = 4;
ArrayList list = null;
HttpApplication application = context.ApplicationInstance;
if (application is IRequestAuthorizationHandler)
{
list = new ArrayList(capacity);
list.Add(application);
}
foreach (IHttpModule module in HttpModuleRegistry.GetModules(application))
{
if (module is IRequestAuthorizationHandler)
{
if (list == null)
list = new ArrayList(capacity);
list.Add(module);
}
}
context.Items[key] = handlers = ArrayList.ReadOnly(
list != null
? list.ToArray(typeof(IRequestAuthorizationHandler))
: _zeroAuthorizationHandlers);
}
return handlers;
}
internal static Uri GetRequestUrl(HttpContext context)
{
if (context == null) throw new ArgumentNullException("context");
Uri url = context.Items["ELMAH_REQUEST_URL"] as Uri;
return url != null ? url : context.Request.Url;
}
}
public interface IRequestAuthorizationHandler
{
bool Authorize(HttpContext context);
}
}