@ -8,24 +8,31 @@ using MediaBrowser.Server.Implementations.HttpServer.SocketSharp;
using ServiceStack ;
using ServiceStack.Host ;
using ServiceStack.Host.Handlers ;
using ServiceStack.Logging ;
using ServiceStack.Web ;
using System ;
using System.Collections.Generic ;
using System.IO ;
using System.Linq ;
using System.Net.Security ;
using System.Net.Sockets ;
using System.Reflection ;
using System.Security.Cryptography.X509Certificates ;
using System.Threading ;
using System.Threading.Tasks ;
using Emby.Common.Implementations.Net ;
using Emby.Server.Implementations.HttpServer ;
using Emby.Server.Implementations.HttpServer.SocketSharp ;
using MediaBrowser.Common.Net ;
using MediaBrowser.Common.Security ;
using MediaBrowser.Controller ;
using MediaBrowser.Model.Cryptography ;
using MediaBrowser.Model.Extensions ;
using MediaBrowser.Model.IO ;
using MediaBrowser.Model.Net ;
using MediaBrowser.Model.Services ;
using ServiceStack.Api.Swagger ;
using MediaBrowser.Model.Text ;
using SocketHttpListener.Net ;
using SocketHttpListener.Primitives ;
namespace MediaBrowser.Server.Implementations.HttpServer
{
@ -49,21 +56,28 @@ namespace MediaBrowser.Server.Implementations.HttpServer
private readonly IServerConfigurationManager _config ;
private readonly INetworkManager _networkManager ;
private readonly IMemoryStream Provider _memoryStreamProvider ;
private readonly IMemoryStream Factory _memoryStreamProvider ;
private readonly IServerApplicationHost _appHost ;
private readonly ITextEncoding _textEncoding ;
private readonly ISocketFactory _socketFactory ;
private readonly ICryptoProvider _cryptoProvider ;
public HttpListenerHost ( IServerApplicationHost applicationHost ,
ILogManager logManager ,
IServerConfigurationManager config ,
string serviceName ,
string defaultRedirectPath , INetworkManager networkManager , IMemoryStream Provider memoryStreamProvider , params Assembly [ ] assembliesWithServices )
: base ( serviceName , assembliesWithServices )
string defaultRedirectPath , INetworkManager networkManager , IMemoryStream Factory memoryStreamProvider , ITextEncoding textEncoding , ISocketFactory socketFactory , ICryptoProvider cryptoProvider )
: base ( serviceName , new Assembly [ ] { } )
{
_appHost = applicationHost ;
DefaultRedirectPath = defaultRedirectPath ;
_networkManager = networkManager ;
_memoryStreamProvider = memoryStreamProvider ;
_textEncoding = textEncoding ;
_socketFactory = socketFactory ;
_cryptoProvider = cryptoProvider ;
_config = config ;
_logger = logManager . GetLogger ( "HttpServer" ) ;
@ -73,10 +87,9 @@ namespace MediaBrowser.Server.Implementations.HttpServer
public string GlobalResponse { get ; set ; }
public override void Configure ( Container container )
public override void Configure ( )
{
HostConfig . Instance . DefaultRedirectPath = DefaultRedirectPath ;
HostConfig . Instance . LogUnobservedTaskExceptions = false ;
HostConfig . Instance . MapExceptionToStatusCode = new Dictionary < Type , int >
{
@ -93,33 +106,11 @@ namespace MediaBrowser.Server.Implementations.HttpServer
{ typeof ( NotSupportedException ) , 500 }
} ;
HostConfig . Instance . GlobalResponseHeaders = new Dictionary < string , string > ( ) ;
HostConfig . Instance . DebugMode = false ;
HostConfig . Instance . LogFactory = LogManager . LogFactory ;
HostConfig . Instance . AllowJsonpRequests = false ;
// The Markdown feature causes slow startup times (5 mins+) on cold boots for some users
// Custom format allows images
HostConfig . Instance . EnableFeatures = Feature . Html | Feature . Json | Feature . Xml | Feature . CustomFormat ;
container . Adapter = _containerAdapter ;
Plugins . Add ( new SwaggerFeature ( ) ) ;
Plugins . Add ( new CorsFeature ( allowedHeaders : "Content-Type, Authorization, Range, X-MediaBrowser-Token, X-Emby-Authorization" ) ) ;
//Plugins.Add(new AuthFeature(() => new AuthUserSession(), new IAuthProvider[] {
// new SessionAuthProvider(_containerAdapter.Resolve<ISessionContext>()),
//}));
//PreRequestFilters.Add((httpReq, httpRes) =>
//{
// //Handles Request and closes Responses after emitting global HTTP Headers
// if (string.Equals(httpReq.Verb, "OPTIONS", StringComparison.OrdinalIgnoreCase))
// {
// httpRes.EndRequest(); //add a 'using ServiceStack;'
// }
//});
Container . Adapter = _containerAdapter ;
var requestFilters = _appHost . GetExports < IRequestFilter > ( ) . ToList ( ) ;
foreach ( var filter in requestFilters )
@ -130,11 +121,12 @@ namespace MediaBrowser.Server.Implementations.HttpServer
HostContext . GlobalResponseFilters . Add ( new ResponseFilter ( _logger ) . FilterResponse ) ;
}
p ublic override void OnAfterInit ( )
p rotected override ILogger Logger
{
SetAppDomainData ( ) ;
base . OnAfterInit ( ) ;
get
{
return _logger ;
}
}
public override void OnConfigLoad ( )
@ -153,23 +145,6 @@ namespace MediaBrowser.Server.Implementations.HttpServer
return new ServiceController ( this , ( ) = > types ) ;
}
public virtual void SetAppDomainData ( )
{
//Required for Mono to resolve VirtualPathUtility and Url.Content urls
var domain = Thread . GetDomain ( ) ; // or AppDomain.Current
domain . SetData ( ".appDomain" , "1" ) ;
domain . SetData ( ".appVPath" , "/" ) ;
domain . SetData ( ".appPath" , domain . BaseDirectory ) ;
if ( string . IsNullOrEmpty ( domain . GetData ( ".appId" ) as string ) )
{
domain . SetData ( ".appId" , "1" ) ;
}
if ( string . IsNullOrEmpty ( domain . GetData ( ".domainId" ) as string ) )
{
domain . SetData ( ".domainId" , "1" ) ;
}
}
public override ServiceStackHost Start ( string listeningAtUrlBase )
{
StartListener ( ) ;
@ -207,7 +182,35 @@ namespace MediaBrowser.Server.Implementations.HttpServer
private IHttpListener GetListener ( )
{
return new WebSocketSharpListener ( _logger , CertificatePath , _memoryStreamProvider ) ;
var cert = ! string . IsNullOrWhiteSpace ( CertificatePath ) & & File . Exists ( CertificatePath )
? GetCert ( CertificatePath ) :
null ;
var enableDualMode = Environment . OSVersion . Platform = = PlatformID . Win32NT ;
return new WebSocketSharpListener ( _logger , cert , _memoryStreamProvider , _textEncoding , _networkManager , _socketFactory , _cryptoProvider , new StreamFactory ( ) , enableDualMode , GetRequest ) ;
}
public static ICertificate GetCert ( string certificateLocation )
{
X509Certificate2 localCert = new X509Certificate2 ( certificateLocation ) ;
//localCert.PrivateKey = PrivateKey.CreateFromFile(pvk_file).RSA;
if ( localCert . PrivateKey = = null )
{
//throw new FileNotFoundException("Secure requested, no private key included", certificateLocation);
return null ;
}
return new Certificate ( localCert ) ;
}
private IHttpRequest GetRequest ( HttpListenerContext httpContext )
{
var operationName = httpContext . Request . GetOperationName ( ) ;
var req = new WebSocketSharpRequest ( httpContext , operationName , _logger , _memoryStreamProvider ) ;
return req ;
}
private void OnWebSocketConnecting ( WebSocketConnectingEventArgs args )
@ -259,11 +262,11 @@ namespace MediaBrowser.Server.Implementations.HttpServer
var contentType = httpReq . ResponseContentType ;
var serializer = HostContext. ContentTypes . GetResponseSerializer ( contentType ) ;
var serializer = ContentTypes. Instance . GetResponseSerializer ( contentType ) ;
if ( serializer = = null )
{
contentType = HostContext . Config . DefaultContentType ;
serializer = HostContext. ContentTypes . GetResponseSerializer ( contentType ) ;
serializer = ContentTypes. Instance . GetResponseSerializer ( contentType ) ;
}
var httpError = ex as IHttpError ;
@ -411,171 +414,170 @@ namespace MediaBrowser.Server.Implementations.HttpServer
protected async Task RequestHandler ( IHttpRequest httpReq , Uri url )
{
var date = DateTime . Now ;
var httpRes = httpReq . Response ;
bool enableLog = false ;
string urlToLog = null ;
string remoteIp = null ;
if ( _disposed )
try
{
httpRes . StatusCode = 503 ;
httpRes . Close ( ) ;
return ;
}
if ( _disposed )
{
httpRes . StatusCode = 503 ;
return ;
}
if ( ! ValidateHost ( url ) )
{
httpRes . StatusCode = 400 ;
httpRes . ContentType = "text/plain" ;
httpRes . Write ( "Invalid host" ) ;
if ( ! ValidateHost ( url ) )
{
httpRes . StatusCode = 400 ;
httpRes . ContentType = "text/plain" ;
httpRes . Write ( "Invalid host" ) ;
return ;
}
httpRes . Close ( ) ;
return ;
}
if ( string . Equals ( httpReq . Verb , "OPTIONS" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . StatusCode = 200 ;
httpRes . AddHeader ( "Access-Control-Allow-Origin" , "*" ) ;
httpRes . AddHeader ( "Access-Control-Allow-Methods" , "GET, POST, PUT, DELETE, PATCH, OPTIONS" ) ;
httpRes . AddHeader ( "Access-Control-Allow-Headers" ,
"Content-Type, Authorization, Range, X-MediaBrowser-Token, X-Emby-Authorization" ) ;
httpRes . ContentType = "text/html" ;
return ;
}
if ( string . Equals ( httpReq . Verb , "OPTIONS" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . StatusCode = 200 ;
httpRes . AddHeader ( "Access-Control-Allow-Origin" , "*" ) ;
httpRes . AddHeader ( "Access-Control-Allow-Methods" , "GET, POST, PUT, DELETE, PATCH, OPTIONS" ) ;
httpRes . AddHeader ( "Access-Control-Allow-Headers" , "Content-Type, Authorization, Range, X-MediaBrowser-Token, X-Emby-Authorization" ) ;
httpRes . ContentType = "text/html" ;
var operationName = httpReq . OperationName ;
var localPath = url . LocalPath ;
httpRes . Close ( ) ;
}
var urlString = url . OriginalString ;
enableLog = EnableLogging ( urlString , localPath ) ;
urlToLog = urlString ;
var operationName = httpReq . OperationName ;
var localPath = url . LocalPath ;
if ( enableLog )
{
urlToLog = GetUrlToLog ( urlString ) ;
remoteIp = httpReq . RemoteIp ;
var urlString = url . OriginalString ;
var enableLog = EnableLogging ( urlString , localPath ) ;
var urlToLog = urlString ;
LoggerUtils . LogRequest ( _logger , urlToLog , httpReq . HttpMethod , httpReq . UserAgent ) ;
}
if ( enableLog )
{
urlToLog = GetUrlToLog ( urlString ) ;
LoggerUtils . LogRequest ( _logger , urlToLog , httpReq . HttpMethod , httpReq . UserAgent ) ;
}
if ( string . Equals ( localPath , "/emby/" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser/" , StringComparison . OrdinalIgnoreCase ) )
{
RedirectToUrl ( httpRes , DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/emby" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser" , StringComparison . OrdinalIgnoreCase ) )
{
RedirectToUrl ( httpRes , "emby/" + DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/emby/" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser/" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/emby" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( "emby/" + DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/mediabrowser/" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser" , StringComparison . OrdinalIgnoreCase ) | |
localPath . IndexOf ( "mediabrowser/web" , StringComparison . OrdinalIgnoreCase ) ! = - 1 )
{
httpRes . StatusCode = 200 ;
httpRes . ContentType = "text/html" ;
var newUrl = urlString . Replace ( "mediabrowser" , "emby" , StringComparison . OrdinalIgnoreCase )
. Replace ( "/dashboard/" , "/web/" , StringComparison . OrdinalIgnoreCase ) ;
if ( string . Equals ( localPath, "/mediabrowser/" , StringComparison . OrdinalIgnoreCase ) | |
string . Equals ( localPath , "/mediabrowser" , StringComparison . OrdinalIgnoreCase ) | |
localPath . IndexOf ( "mediabrowser/web" , StringComparison . OrdinalIgnoreCase ) ! = - 1 )
{
httpRes . StatusCode = 200 ;
httpRes . ContentType = "text/html" ;
var newUrl = urlString . Replace ( "mediabrowser" , "emby" , StringComparison . OrdinalIgnoreCase )
. Replace ( "/dashboard/" , "/web/" , StringComparison . OrdinalIgnoreCase ) ;
if ( ! string . Equals ( newUrl , urlString , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . Write (
"<!doctype html><html><head><title>Emby</title></head><body>Please update your Emby bookmark to <a href=\"" +
newUrl + "\">" + newUrl + "</a></body></html>" ) ;
return ;
}
}
if ( ! string . Equals ( newUrl , urlString , StringComparison . OrdinalIgnoreCase ) )
if ( localPath . IndexOf ( "dashboard/" , StringComparison . OrdinalIgnoreCase ) ! = - 1 & &
localPath . IndexOf ( "web/dashboard" , StringComparison . OrdinalIgnoreCase ) = = - 1 )
{
httpRes . Write ( "<!doctype html><html><head><title>Emby</title></head><body>Please update your Emby bookmark to <a href=\"" + newUrl + "\">" + newUrl + "</a></body></html>" ) ;
httpRes . StatusCode = 200 ;
httpRes . ContentType = "text/html" ;
var newUrl = urlString . Replace ( "mediabrowser" , "emby" , StringComparison . OrdinalIgnoreCase )
. Replace ( "/dashboard/" , "/web/" , StringComparison . OrdinalIgnoreCase ) ;
httpRes . Close ( ) ;
return ;
if ( ! string . Equals ( newUrl , urlString , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . Write (
"<!doctype html><html><head><title>Emby</title></head><body>Please update your Emby bookmark to <a href=\"" +
newUrl + "\">" + newUrl + "</a></body></html>" ) ;
return ;
}
}
}
if ( localPath . IndexOf ( "dashboard/" , StringComparison . OrdinalIgnoreCase ) ! = - 1 & &
localPath . IndexOf ( "web/dashboard" , StringComparison . OrdinalIgnoreCase ) = = - 1 )
{
httpRes . StatusCode = 200 ;
httpRes . ContentType = "text/html" ;
var newUrl = urlString . Replace ( "mediabrowser" , "emby" , StringComparison . OrdinalIgnoreCase )
. Replace ( "/dashboard/" , "/web/" , StringComparison . OrdinalIgnoreCase ) ;
if ( string . Equals ( localPath , "/web" , StringComparison . OrdinalIgnoreCase ) )
{
RedirectToUrl ( httpRes , DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/web/" , StringComparison . OrdinalIgnoreCase ) )
{
RedirectToUrl ( httpRes , "../" + DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/" , StringComparison . OrdinalIgnoreCase ) )
{
RedirectToUrl ( httpRes , DefaultRedirectPath ) ;
return ;
}
if ( string . IsNullOrEmpty ( localPath ) )
{
RedirectToUrl ( httpRes , "/" + DefaultRedirectPath ) ;
return ;
}
if ( ! string . Equals ( newUrl , urlString , StringComparison . OrdinalIgnoreCase ) )
if ( string . Equals ( localPath, "/emby/pin" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . Write ( "<!doctype html><html><head><title>Emby</title></head><body>Please update your Emby bookmark to <a href=\"" + newUrl + "\">" + newUrl + "</a></body></html>" ) ;
RedirectToUrl ( httpRes , "web/pin.html" ) ;
return ;
}
httpRes . Close ( ) ;
if ( ! string . IsNullOrWhiteSpace ( GlobalResponse ) )
{
httpRes . StatusCode = 503 ;
httpRes . ContentType = "text/html" ;
httpRes . Write ( GlobalResponse ) ;
return ;
}
}
if ( string . Equals ( localPath , "/web" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/web/" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( "../" + DefaultRedirectPath ) ;
return ;
}
if ( string . Equals ( localPath , "/" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( DefaultRedirectPath ) ;
return ;
}
if ( string . IsNullOrEmpty ( localPath ) )
{
httpRes . RedirectToUrl ( "/" + DefaultRedirectPath ) ;
return ;
}
var handler = HttpHandlerFactory . GetHandler ( httpReq ) ;
if ( string . Equals ( localPath , "/emby/pin" , StringComparison . OrdinalIgnoreCase ) )
{
httpRes . RedirectToUrl ( "web/pin.html" ) ;
return ;
if ( handler ! = null )
{
await handler . ProcessRequestAsync ( httpReq , httpRes , operationName ) . ConfigureAwait ( false ) ;
}
}
if ( ! string . IsNullOrWhiteSpace ( GlobalResponse ) )
catch ( Exception ex )
{
httpRes . StatusCode = 503 ;
httpRes . ContentType = "text/html" ;
httpRes . Write ( GlobalResponse ) ;
httpRes . Close ( ) ;
return ;
ErrorHandler ( ex , httpReq ) ;
}
var handler = HttpHandlerFactory . GetHandler ( httpReq ) ;
var remoteIp = httpReq . RemoteIp ;
var serviceStackHandler = handler as IServiceStackHandler ;
if ( serviceStackHandler ! = null )
finally
{
var restHandler = serviceStackHandler as RestHandler ;
if ( restHandler ! = null )
{
httpReq . OperationName = operationName = restHandler . RestPath . RequestType . GetOperationName ( ) ;
}
httpRes . Close ( ) ;
try
{
await serviceStackHandler . ProcessRequestAsync ( httpReq , httpRes , operationName ) . ConfigureAwait ( false ) ;
}
finally
if ( enableLog )
{
httpRes . Close ( ) ;
var statusCode = httpRes . StatusCode ;
var duration = DateTime . Now - date ;
if ( enableLog )
{
LoggerUtils . LogResponse ( _logger , statusCode , urlToLog , remoteIp , duration ) ;
}
LoggerUtils . LogResponse ( _logger , statusCode , urlToLog , remoteIp , duration ) ;
}
}
else
{
httpRes . Close ( ) ;
}
}
public static void RedirectToUrl ( IResponse httpRes , string url )
{
httpRes . StatusCode = 302 ;
httpRes . AddHeader ( HttpHeaders . Location , url ) ;
httpRes . EndRequest ( ) ;
}
/// <summary>
/// Adds the rest handlers.
/// </summary>
@ -653,15 +655,6 @@ namespace MediaBrowser.Server.Implementations.HttpServer
return "mediabrowser/" + path ;
}
/// <summary>
/// Releases the specified instance.
/// </summary>
/// <param name="instance">The instance.</param>
public override void Release ( object instance )
{
// Leave this empty so SS doesn't try to dispose our objects
}
private bool _disposed ;
private readonly object _disposeLock = new object ( ) ;
protected virtual void Dispose ( bool disposing )
@ -696,4 +689,37 @@ namespace MediaBrowser.Server.Implementations.HttpServer
Start ( UrlPrefixes . First ( ) ) ;
}
}
public class StreamFactory : IStreamFactory
{
public Stream CreateNetworkStream ( ISocket socket , bool ownsSocket )
{
var netSocket = ( NetSocket ) socket ;
return new NetworkStream ( netSocket . Socket , ownsSocket ) ;
}
public Task AuthenticateSslStreamAsServer ( Stream stream , ICertificate certificate )
{
var sslStream = ( SslStream ) stream ;
var cert = ( Certificate ) certificate ;
return sslStream . AuthenticateAsServerAsync ( cert . X509Certificate ) ;
}
public Stream CreateSslStream ( Stream innerStream , bool leaveInnerStreamOpen )
{
return new SslStream ( innerStream , leaveInnerStreamOpen ) ;
}
}
public class Certificate : ICertificate
{
public Certificate ( X509Certificate x509Certificate )
{
X509Certificate = x509Certificate ;
}
public X509Certificate X509Certificate { get ; private set ; }
}
}