You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			683 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			683 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //------------------------------------------------------------------------------
 | |
| // <copyright file="VirtualPath.cs" company="Microsoft">
 | |
| //     Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| // </copyright>
 | |
| //------------------------------------------------------------------------------
 | |
| 
 | |
| namespace System.Web {
 | |
| 
 | |
|     using System.Globalization;
 | |
|     using System.Collections;
 | |
|     using System.IO;
 | |
|     using System.Web.Util;
 | |
|     using System.Web.Hosting;
 | |
|     using System.Web.Caching;
 | |
|     using System.Security.Permissions;
 | |
|     using Microsoft.Win32;
 | |
| 
 | |
|     [Serializable]
 | |
|     internal sealed class VirtualPath : IComparable {
 | |
|         private string _appRelativeVirtualPath;
 | |
|         private string _virtualPath;
 | |
| 
 | |
|         // const masks into the BitVector32
 | |
|         private const int isWithinAppRootComputed           = 0x00000001;
 | |
|         private const int isWithinAppRoot                   = 0x00000002;
 | |
|         private const int appRelativeAttempted              = 0x00000004;
 | |
| 
 | |
|     #pragma warning disable 0649
 | |
|         private SimpleBitVector32 flags;
 | |
|     #pragma warning restore 0649
 | |
| 
 | |
| #if DBG
 | |
|         private static char[] s_illegalVirtualPathChars           = new char[] { '\0' };
 | |
| 
 | |
|         // Debug only method to check that the object is in a consistent state
 | |
|         private void ValidateState() {
 | |
| 
 | |
|             Debug.Assert(_virtualPath != null || _appRelativeVirtualPath != null);
 | |
| 
 | |
|             if (_virtualPath != null) {
 | |
|                 CheckValidVirtualPath(_virtualPath);
 | |
|             }
 | |
| 
 | |
|             if (_appRelativeVirtualPath != null) {
 | |
|                 Debug.Assert(UrlPath.IsAppRelativePath(_appRelativeVirtualPath));
 | |
|                 CheckValidVirtualPath(_appRelativeVirtualPath);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void CheckValidVirtualPath(string virtualPath) {
 | |
|             Debug.Assert(virtualPath.IndexOfAny(s_illegalVirtualPathChars) < 0);
 | |
|             Debug.Assert(virtualPath.IndexOf('\\') < 0);
 | |
|         }
 | |
| #endif
 | |
| 
 | |
|         internal static VirtualPath RootVirtualPath = VirtualPath.Create("/");
 | |
| 
 | |
|         private VirtualPath() { }
 | |
| 
 | |
|         // This is called to set the appropriate virtual path field when we already know
 | |
|         // that the path is generally well formed.
 | |
|         private VirtualPath(string virtualPath) {
 | |
|             if (UrlPath.IsAppRelativePath(virtualPath)) {
 | |
|                 _appRelativeVirtualPath = virtualPath;
 | |
|             }
 | |
|             else {
 | |
|                 _virtualPath = virtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         int IComparable.CompareTo(object obj) {
 | |
| 
 | |
|             VirtualPath virtualPath = obj as VirtualPath;
 | |
| 
 | |
|             // Make sure we're compared to another VirtualPath
 | |
|             if (virtualPath == null)
 | |
|                 throw new ArgumentException();
 | |
| 
 | |
|             // Check if it's the same object
 | |
|             if (virtualPath == this)
 | |
|                 return 0;
 | |
| 
 | |
|             return StringComparer.InvariantCultureIgnoreCase.Compare(
 | |
|                 this.VirtualPathString, virtualPath.VirtualPathString);
 | |
|         }
 | |
| 
 | |
|         public string VirtualPathString {
 | |
|             get {
 | |
|                 if (_virtualPath == null) {
 | |
|                     Debug.Assert(_appRelativeVirtualPath != null);
 | |
| 
 | |
|                     // This is not valid if we don't know the app path
 | |
|                     if (HttpRuntime.AppDomainAppVirtualPathObject == null) {
 | |
|                         throw new HttpException(SR.GetString(SR.VirtualPath_CantMakeAppAbsolute,
 | |
|                             _appRelativeVirtualPath));
 | |
|                     }
 | |
| 
 | |
|                     if (_appRelativeVirtualPath.Length == 1) {
 | |
|                         _virtualPath = HttpRuntime.AppDomainAppVirtualPath;
 | |
|                     }
 | |
|                     else {
 | |
|                         _virtualPath = HttpRuntime.AppDomainAppVirtualPathString +
 | |
|                             _appRelativeVirtualPath.Substring(2);
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 return _virtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal string VirtualPathStringNoTrailingSlash {
 | |
|             get {
 | |
|                 return UrlPath.RemoveSlashFromPathIfNeeded(VirtualPathString);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Return the virtual path string if we have it, otherwise null
 | |
|         internal string VirtualPathStringIfAvailable {
 | |
|             get {
 | |
|                 return _virtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal string AppRelativeVirtualPathStringOrNull {
 | |
|             get {
 | |
|                 if (_appRelativeVirtualPath == null) {
 | |
|                     Debug.Assert(_virtualPath != null);
 | |
| 
 | |
|                     // If we already tried to get it and couldn't, return null
 | |
|                     if (flags[appRelativeAttempted])
 | |
|                         return null;
 | |
| 
 | |
|                     // This is not valid if we don't know the app path
 | |
|                     if (HttpRuntime.AppDomainAppVirtualPathObject == null) {
 | |
|                         throw new HttpException(SR.GetString(SR.VirtualPath_CantMakeAppRelative, _virtualPath));
 | |
|                     }
 | |
| 
 | |
|                     _appRelativeVirtualPath = UrlPath.MakeVirtualPathAppRelativeOrNull(_virtualPath);
 | |
| 
 | |
|                     // Remember that we've attempted it
 | |
|                     flags[appRelativeAttempted] = true;
 | |
| 
 | |
|                     // It could be null if it's not under the app root
 | |
|                     if (_appRelativeVirtualPath == null)
 | |
|                         return null;
 | |
| #if DBG
 | |
|                     ValidateState();
 | |
| #endif
 | |
|                 }
 | |
| 
 | |
|                 return _appRelativeVirtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Return the app relative path if possible. Otherwise, settle for the absolute.
 | |
|         public string AppRelativeVirtualPathString {
 | |
|             get {
 | |
|                 string appRelativeVirtualPath = AppRelativeVirtualPathStringOrNull;
 | |
|                 return (appRelativeVirtualPath != null) ? appRelativeVirtualPath : _virtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // Return the app relative virtual path string if we have it, otherwise null
 | |
|         internal string AppRelativeVirtualPathStringIfAvailable {
 | |
|             get {
 | |
|                 return _appRelativeVirtualPath;
 | |
|             }
 | |
|         }
 | |
|         // Return the virtual string that's either app relative or not, depending on which
 | |
|         // one we already have internally.  If we have both, we return absolute
 | |
|         internal string VirtualPathStringWhicheverAvailable {
 | |
|             get {
 | |
|                 return _virtualPath != null ? _virtualPath : _appRelativeVirtualPath;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public string Extension {
 | |
|             get {
 | |
|                 return UrlPath.GetExtension(VirtualPathString);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public string FileName {
 | |
|             get {
 | |
|                 return UrlPath.GetFileName(VirtualPathStringNoTrailingSlash);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         // If it's relative, combine it with the app root
 | |
|         public VirtualPath CombineWithAppRoot() {
 | |
|             return HttpRuntime.AppDomainAppVirtualPathObject.Combine(this);
 | |
|         }
 | |
| 
 | |
|         public VirtualPath Combine(VirtualPath relativePath) {
 | |
|             if (relativePath == null)
 | |
|                 throw new ArgumentNullException("relativePath");
 | |
| 
 | |
|             // If it's not relative, return it unchanged
 | |
|             if (!relativePath.IsRelative)
 | |
|                 return relativePath;
 | |
| 
 | |
|             // The base of the combine should never be relative
 | |
|             FailIfRelativePath();
 | |
| 
 | |
|             // Get either _appRelativeVirtualPath or _virtualPath
 | |
|             string virtualPath = VirtualPathStringWhicheverAvailable;
 | |
| 
 | |
|             // Combine it with the relative
 | |
|             virtualPath = UrlPath.Combine(virtualPath, relativePath.VirtualPathString);
 | |
| 
 | |
|             // Set the appropriate virtual path in the new object
 | |
|             return new VirtualPath(virtualPath);
 | |
|         }
 | |
| 
 | |
|         // This simple version of combine should only be used when the relative
 | |
|         // path is known to be relative.  It's more efficient, but doesn't do any
 | |
|         // sanity checks.
 | |
|         internal VirtualPath SimpleCombine(string relativePath) {
 | |
|             return SimpleCombine(relativePath, false /*addTrailingSlash*/);
 | |
|         }
 | |
| 
 | |
|         internal VirtualPath SimpleCombineWithDir(string directoryName) {
 | |
|             return SimpleCombine(directoryName, true /*addTrailingSlash*/);
 | |
|         }
 | |
| 
 | |
|         private VirtualPath SimpleCombine(string filename, bool addTrailingSlash) {
 | |
| 
 | |
|             // The left part should always be a directory
 | |
|             Debug.Assert(HasTrailingSlash);
 | |
| 
 | |
|             // The right part should not start or end with a slash
 | |
|             Debug.Assert(filename[0] != '/' && !UrlPath.HasTrailingSlash(filename));
 | |
| 
 | |
|             // Use either _appRelativeVirtualPath or _virtualPath
 | |
|             string virtualPath = VirtualPathStringWhicheverAvailable + filename;
 | |
|             if (addTrailingSlash)
 | |
|                 virtualPath += "/";
 | |
| 
 | |
|             // Set the appropriate virtual path in the new object
 | |
|             VirtualPath combinedVirtualPath = new VirtualPath(virtualPath);
 | |
| 
 | |
|             // Copy some flags over to avoid having to recalculate them
 | |
|             combinedVirtualPath.CopyFlagsFrom(this, isWithinAppRootComputed | isWithinAppRoot | appRelativeAttempted);
 | |
| #if DBG
 | |
|             combinedVirtualPath.ValidateState();
 | |
| #endif
 | |
|             return combinedVirtualPath;
 | |
|         }
 | |
| 
 | |
|         public VirtualPath MakeRelative(VirtualPath toVirtualPath) {
 | |
|             VirtualPath resultVirtualPath = new VirtualPath();
 | |
| 
 | |
|             // Neither path can be relative
 | |
|             FailIfRelativePath();
 | |
|             toVirtualPath.FailIfRelativePath();
 | |
| 
 | |
|             // Set it directly since we know the slashes are already ok
 | |
|             resultVirtualPath._virtualPath = UrlPath.MakeRelative(this.VirtualPathString, toVirtualPath.VirtualPathString);
 | |
| #if DBG
 | |
|             resultVirtualPath.ValidateState();
 | |
| #endif
 | |
|             return resultVirtualPath;
 | |
|         }
 | |
| 
 | |
|         public string MapPath() {
 | |
|             return HostingEnvironment.MapPath(this);
 | |
|         }
 | |
| 
 | |
|         internal string MapPathInternal() {
 | |
|             return HostingEnvironment.MapPathInternal(this);
 | |
|         }
 | |
| 
 | |
|         internal string MapPathInternal(bool permitNull) {
 | |
|             return HostingEnvironment.MapPathInternal(this, permitNull);
 | |
|         }
 | |
| 
 | |
|         internal string MapPathInternal(VirtualPath baseVirtualDir, bool allowCrossAppMapping) {
 | |
|             return HostingEnvironment.MapPathInternal(this, baseVirtualDir, allowCrossAppMapping);
 | |
|         }
 | |
| 
 | |
|         ///////////// VirtualPathProvider wrapper methods /////////////
 | |
| 
 | |
|         public string GetFileHash(IEnumerable virtualPathDependencies) {
 | |
|             return HostingEnvironment.VirtualPathProvider.GetFileHash(this, virtualPathDependencies);
 | |
|         }
 | |
| 
 | |
|         public CacheDependency GetCacheDependency(IEnumerable virtualPathDependencies, DateTime utcStart) {
 | |
|             return HostingEnvironment.VirtualPathProvider.GetCacheDependency(
 | |
|                 this, virtualPathDependencies, utcStart);
 | |
|         }
 | |
| 
 | |
|         public bool FileExists() {
 | |
|             return HostingEnvironment.VirtualPathProvider.FileExists(this);
 | |
|         }
 | |
| 
 | |
|         public bool DirectoryExists() {
 | |
|             return HostingEnvironment.VirtualPathProvider.DirectoryExists(this);
 | |
|         }
 | |
| 
 | |
|         public VirtualFile GetFile() {
 | |
|             return HostingEnvironment.VirtualPathProvider.GetFile(this);
 | |
|         }
 | |
| 
 | |
|         public VirtualDirectory GetDirectory() {
 | |
|             Debug.Assert(this.HasTrailingSlash);
 | |
|             return HostingEnvironment.VirtualPathProvider.GetDirectory(this);
 | |
|         }
 | |
| 
 | |
|         public string GetCacheKey() {
 | |
|             return HostingEnvironment.VirtualPathProvider.GetCacheKey(this);
 | |
|         }
 | |
| 
 | |
|         public Stream OpenFile() {
 | |
|             return VirtualPathProvider.OpenFile(this);
 | |
|         }
 | |
| 
 | |
|         ///////////// end of VirtualPathProvider methods /////////////
 | |
| 
 | |
| 
 | |
|         internal bool HasTrailingSlash {
 | |
|             get {
 | |
|                 if (_virtualPath != null) {
 | |
|                     return UrlPath.HasTrailingSlash(_virtualPath);
 | |
|                 }
 | |
|                 else {
 | |
|                     return UrlPath.HasTrailingSlash(_appRelativeVirtualPath);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsWithinAppRoot {
 | |
|             get {
 | |
|                 // If we don't already know it, compute it and cache it
 | |
|                 if (!flags[isWithinAppRootComputed]) {
 | |
|                     if (HttpRuntime.AppDomainIdInternal == null) {
 | |
|                         Debug.Assert(false);
 | |
|                         return true;    // app domain not initialized
 | |
|                     }
 | |
| 
 | |
|                     if (flags[appRelativeAttempted]) {
 | |
|                         // If we already tried to get the app relative path, we can tell whether
 | |
|                         // it's in the app root by checking whether it's not null
 | |
|                         flags[isWithinAppRoot] = (_appRelativeVirtualPath != null);
 | |
|                     }
 | |
|                     else {
 | |
|                         flags[isWithinAppRoot] = UrlPath.IsEqualOrSubpath(HttpRuntime.AppDomainAppVirtualPathString,
 | |
|                             VirtualPathString);
 | |
|                     }
 | |
| 
 | |
|                     flags[isWithinAppRootComputed] = true;
 | |
|                 }
 | |
| 
 | |
|                 return flags[isWithinAppRoot];
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void FailIfNotWithinAppRoot() {
 | |
|             if (!this.IsWithinAppRoot) {
 | |
|                 throw new ArgumentException(SR.GetString(SR.Cross_app_not_allowed, this.VirtualPathString));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void FailIfRelativePath() {
 | |
|             if (this.IsRelative) {
 | |
|                 throw new ArgumentException(SR.GetString(SR.VirtualPath_AllowRelativePath, _virtualPath));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsRelative {
 | |
|             get {
 | |
|                 // Note that we don't need to check for "~/", since _virtualPath never contains
 | |
|                 // app relative paths (_appRelativeVirtualPath does)
 | |
|                 return _virtualPath != null && _virtualPath[0] != '/';
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsRoot {
 | |
|             get {
 | |
|                 return _virtualPath == "/";
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public VirtualPath Parent {
 | |
|             get {
 | |
|                 // Getting the parent doesn't make much sense on relative paths
 | |
|                 FailIfRelativePath();
 | |
| 
 | |
|                 // "/" doesn't have a parent, so return null
 | |
|                 if (IsRoot)
 | |
|                     return null;
 | |
| 
 | |
|                 // Get either _appRelativeVirtualPath or _virtualPath
 | |
|                 string virtualPath = VirtualPathStringWhicheverAvailable;
 | |
| 
 | |
|                 // Get rid of the ending slash, otherwise we end up with Parent("/app/sub/") == "/app/sub/"
 | |
|                 virtualPath = UrlPath.RemoveSlashFromPathIfNeeded(virtualPath);
 | |
| 
 | |
|                 // But if it's just "~", use the absolute path instead to get the parent
 | |
|                 if (virtualPath == "~")
 | |
|                     virtualPath = VirtualPathStringNoTrailingSlash;
 | |
| 
 | |
|                 int index = virtualPath.LastIndexOf('/');
 | |
|                 Debug.Assert(index >= 0);
 | |
| 
 | |
|                 // e.g. the parent of "/blah" is "/"
 | |
|                 if (index == 0)
 | |
|                     return RootVirtualPath;
 | |
| 
 | |
|                 // 
 | |
| 
 | |
|                 // Get the parent
 | |
|                 virtualPath = virtualPath.Substring(0, index + 1);
 | |
| 
 | |
|                 // Set the appropriate virtual path in the new object
 | |
|                 return new VirtualPath(virtualPath);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal static VirtualPath Combine(VirtualPath v1, VirtualPath v2) {
 | |
| 
 | |
|             // If the first is null, use the app root instead
 | |
|             if (v1 == null) {
 | |
|                 v1 = HttpRuntime.AppDomainAppVirtualPathObject;
 | |
|             }
 | |
| 
 | |
|             // If the first is still null, return the second, unless it's relative
 | |
|             if (v1 == null) {
 | |
|                 v2.FailIfRelativePath();
 | |
|                 return v2;
 | |
|             }
 | |
| 
 | |
|             return v1.Combine(v2);
 | |
|         }
 | |
| 
 | |
|         public static bool operator == (VirtualPath v1, VirtualPath v2) {
 | |
|             return VirtualPath.Equals(v1, v2);
 | |
|         }
 | |
|         
 | |
|         public static bool operator != (VirtualPath v1, VirtualPath v2) {
 | |
|             return !VirtualPath.Equals(v1, v2);
 | |
|         }
 | |
| 
 | |
|         public static bool Equals(VirtualPath v1, VirtualPath v2) {
 | |
| 
 | |
|             // Check if it's the same object
 | |
|             if ((Object)v1 == (Object)v2) {
 | |
|                 return true;
 | |
|             }
 | |
| 
 | |
|             if ((Object)v1 == null || (Object)v2 == null) {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             return EqualsHelper(v1, v2);
 | |
|         }
 | |
| 
 | |
|         public override bool Equals(object value) {
 | |
| 
 | |
|             if (value == null)
 | |
|                 return false;
 | |
| 
 | |
|             VirtualPath virtualPath = value as VirtualPath;
 | |
|             if ((object)virtualPath == null) {
 | |
|                 Debug.Assert(false);
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             return EqualsHelper(virtualPath, this);
 | |
|         }
 | |
| 
 | |
|         private static bool EqualsHelper(VirtualPath v1, VirtualPath v2) {
 | |
|             return StringComparer.InvariantCultureIgnoreCase.Compare(
 | |
|                 v1.VirtualPathString, v2.VirtualPathString) == 0;
 | |
|         }
 | |
| 
 | |
|         public override int GetHashCode() {
 | |
|             return StringComparer.InvariantCultureIgnoreCase.GetHashCode(VirtualPathString);
 | |
|         }
 | |
| 
 | |
|         public override String ToString() {
 | |
|             // If we only have the app relative path, and we don't know the app root, return
 | |
|             // the app relative path instead of accessing VirtualPathString, which would throw
 | |
|             if (_virtualPath == null && HttpRuntime.AppDomainAppVirtualPathObject == null) {
 | |
|                 Debug.Assert(_appRelativeVirtualPath != null);
 | |
|                 return _appRelativeVirtualPath;
 | |
|             }
 | |
| 
 | |
|             return VirtualPathString;
 | |
|         }
 | |
| 
 | |
|         // Copy a set of flags from another VirtualPath object
 | |
|         private void CopyFlagsFrom(VirtualPath virtualPath, int mask) {
 | |
|             flags.IntegerValue |= virtualPath.flags.IntegerValue & mask;
 | |
|         }
 | |
| 
 | |
|         internal static string GetVirtualPathString(VirtualPath virtualPath) {
 | |
|             return virtualPath == null ? null : virtualPath.VirtualPathString;
 | |
|         }
 | |
| 
 | |
|         internal static string GetVirtualPathStringNoTrailingSlash(VirtualPath virtualPath) {
 | |
|             return virtualPath == null ? null : virtualPath.VirtualPathStringNoTrailingSlash;
 | |
|         }
 | |
| 
 | |
|         internal static string GetAppRelativeVirtualPathString(VirtualPath virtualPath) {
 | |
|             return virtualPath == null ? null : virtualPath.AppRelativeVirtualPathString;
 | |
|         }
 | |
| 
 | |
|         // Same as GetAppRelativeVirtualPathString, but returns "" instead of null
 | |
|         internal static string GetAppRelativeVirtualPathStringOrEmpty(VirtualPath virtualPath) {
 | |
|             return virtualPath == null ? String.Empty : virtualPath.AppRelativeVirtualPathString;
 | |
|         }
 | |
| 
 | |
|         // Default Create method
 | |
|         public static VirtualPath Create(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAllPath);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateTrailingSlash(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAllPath | VirtualPathOptions.EnsureTrailingSlash);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateAllowNull(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAllPath | VirtualPathOptions.AllowNull);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateAbsolute(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateNonRelative(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowAppRelativePath);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateAbsoluteTrailingSlash(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.EnsureTrailingSlash);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateNonRelativeTrailingSlash(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowAppRelativePath |
 | |
|                 VirtualPathOptions.EnsureTrailingSlash);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateAbsoluteAllowNull(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowNull);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateNonRelativeAllowNull(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowAppRelativePath | VirtualPathOptions.AllowNull);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath CreateNonRelativeTrailingSlashAllowNull(string virtualPath) {
 | |
|             return Create(virtualPath, VirtualPathOptions.AllowAbsolutePath | VirtualPathOptions.AllowAppRelativePath |
 | |
|                 VirtualPathOptions.AllowNull | VirtualPathOptions.EnsureTrailingSlash);
 | |
|         }
 | |
| 
 | |
|         public static VirtualPath Create(string virtualPath, VirtualPathOptions options) {
 | |
|             
 | |
|             // Trim it first, so that blank strings (e.g. "  ") get treated as empty
 | |
|             if (virtualPath != null)
 | |
|                 virtualPath = virtualPath.Trim();
 | |
| 
 | |
|             // If it's empty, check whether we allow it
 | |
|             if (String.IsNullOrEmpty(virtualPath)) {
 | |
|                 if ((options & VirtualPathOptions.AllowNull) != 0)
 | |
|                     return null;
 | |
| 
 | |
|                 throw new ArgumentNullException("virtualPath");
 | |
|             }
 | |
| 
 | |
|             // Dev10 767308: optimize for normal paths, and scan once for
 | |
|             //     i) invalid chars
 | |
|             //    ii) slashes
 | |
|             //   iii) '.'
 | |
| 
 | |
|             bool slashes = false;
 | |
|             bool dot = false;
 | |
|             int len = virtualPath.Length;
 | |
|             unsafe {
 | |
|                 fixed (char * p = virtualPath) {
 | |
|                     for (int i = 0; i < len; i++) {
 | |
|                         switch (p[i]) {
 | |
|                             // need to fix slashes ?
 | |
|                             case '/':
 | |
|                                 if (i > 0 && p[i-1] == '/')
 | |
|                                     slashes = true;
 | |
|                                 break;
 | |
|                             case '\\':
 | |
|                                 slashes = true;
 | |
|                                 break;
 | |
|                             // contains "." or ".."
 | |
|                             case '.':
 | |
|                                 dot = true;
 | |
|                                 break;
 | |
|                             // invalid chars
 | |
|                             case '\0':
 | |
|                                 throw new HttpException(SR.GetString(SR.Invalid_vpath, virtualPath));
 | |
|                             default:
 | |
|                                 break;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (slashes) {
 | |
|                 // If we're supposed to fail on malformed path, then throw
 | |
|                 if ((options & VirtualPathOptions.FailIfMalformed) != 0) {
 | |
|                     throw new HttpException(SR.GetString(SR.Invalid_vpath, virtualPath));
 | |
|                 }
 | |
|                 // Flip ----lashes, and remove duplicate slashes                
 | |
|                 virtualPath = UrlPath.FixVirtualPathSlashes(virtualPath);
 | |
|             }
 | |
| 
 | |
|             // Make sure it ends with a trailing slash if requested
 | |
|             if ((options & VirtualPathOptions.EnsureTrailingSlash) != 0)
 | |
|                 virtualPath = UrlPath.AppendSlashToPathIfNeeded(virtualPath);
 | |
| 
 | |
|             VirtualPath virtualPathObject = new VirtualPath();
 | |
| 
 | |
|             if (UrlPath.IsAppRelativePath(virtualPath)) {
 | |
|                 
 | |
|                 if (dot)
 | |
|                     virtualPath = UrlPath.ReduceVirtualPath(virtualPath);
 | |
| 
 | |
|                 if (virtualPath[0] == UrlPath.appRelativeCharacter) {
 | |
|                     if ((options & VirtualPathOptions.AllowAppRelativePath) == 0) {
 | |
|                         throw new ArgumentException(SR.GetString(SR.VirtualPath_AllowAppRelativePath, virtualPath));
 | |
|                     }
 | |
| 
 | |
|                     virtualPathObject._appRelativeVirtualPath = virtualPath;
 | |
|                 }
 | |
|                 else {
 | |
|                     // It's possible for the path to become absolute after calling Reduce,
 | |
|                     // even though it started with "~/".  e.g. if the app is "/app" and the path is
 | |
|                     // "~/../hello.aspx", it becomes "/hello.aspx", which is absolute
 | |
| 
 | |
|                     if ((options & VirtualPathOptions.AllowAbsolutePath) == 0) {
 | |
|                         throw new ArgumentException(SR.GetString(SR.VirtualPath_AllowAbsolutePath, virtualPath));
 | |
|                     }
 | |
| 
 | |
|                     virtualPathObject._virtualPath = virtualPath;
 | |
|                 }
 | |
|             }
 | |
|             else {
 | |
|                 if (virtualPath[0] != '/') {
 | |
|                     if ((options & VirtualPathOptions.AllowRelativePath) == 0) {
 | |
|                         throw new ArgumentException(SR.GetString(SR.VirtualPath_AllowRelativePath, virtualPath));
 | |
|                     }
 | |
| 
 | |
|                     // Don't Reduce relative paths, since the Reduce method is broken (e.g. "../foo.aspx" --> "/foo.aspx!")
 | |
|                     // 
 | |
|                     virtualPathObject._virtualPath = virtualPath;
 | |
|                 }
 | |
|                 else {
 | |
|                     if ((options & VirtualPathOptions.AllowAbsolutePath) == 0) {
 | |
|                         throw new ArgumentException(SR.GetString(SR.VirtualPath_AllowAbsolutePath, virtualPath));
 | |
|                     }
 | |
| 
 | |
|                     if (dot)
 | |
|                         virtualPath = UrlPath.ReduceVirtualPath(virtualPath);
 | |
| 
 | |
|                     virtualPathObject._virtualPath = virtualPath;
 | |
|                 }
 | |
|             }
 | |
| #if DBG
 | |
|             virtualPathObject.ValidateState();
 | |
| #endif
 | |
|             return virtualPathObject;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     [Flags]
 | |
|     internal enum VirtualPathOptions {
 | |
|         AllowNull               = 0x00000001,
 | |
|         EnsureTrailingSlash     = 0x00000002,
 | |
|         AllowAbsolutePath       = 0x00000004,
 | |
|         AllowAppRelativePath    = 0x00000008,
 | |
|         AllowRelativePath       = 0x00000010,
 | |
|         FailIfMalformed         = 0x00000020,
 | |
| 
 | |
|         AllowAllPath = AllowAbsolutePath | AllowAppRelativePath | AllowRelativePath,
 | |
|     }
 | |
| }
 |