233 lines
8.2 KiB
C#
233 lines
8.2 KiB
C#
// Copyright (c) .NET Foundation. All rights reserved.
|
|
// Licensed under the Apache License, Version 2.0. See License.txt in the project root for license information.
|
|
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Linq;
|
|
|
|
namespace Microsoft.AspNetCore.Razor.Language
|
|
{
|
|
public class VirtualRazorProjectFileSystem : RazorProjectFileSystem
|
|
{
|
|
private readonly DirectoryNode _root = new DirectoryNode("/");
|
|
|
|
public override IEnumerable<RazorProjectItem> EnumerateItems(string basePath)
|
|
{
|
|
basePath = NormalizeAndEnsureValidPath(basePath);
|
|
var directory = _root.GetDirectory(basePath);
|
|
return directory?.EnumerateItems() ?? Enumerable.Empty<RazorProjectItem>();
|
|
}
|
|
|
|
public override RazorProjectItem GetItem(string path)
|
|
{
|
|
path = NormalizeAndEnsureValidPath(path);
|
|
return _root.GetItem(path) ?? new NotFoundProjectItem(string.Empty, path);
|
|
}
|
|
|
|
public void Add(RazorProjectItem projectItem)
|
|
{
|
|
if (projectItem == null)
|
|
{
|
|
throw new ArgumentNullException(nameof(projectItem));
|
|
}
|
|
|
|
var filePath = NormalizeAndEnsureValidPath(projectItem.FilePath);
|
|
_root.AddFile(new FileNode(filePath, projectItem));
|
|
}
|
|
|
|
// Internal for testing
|
|
[DebuggerDisplay("{Path}")]
|
|
internal class DirectoryNode
|
|
{
|
|
public DirectoryNode(string path)
|
|
{
|
|
Path = path;
|
|
}
|
|
|
|
public string Path { get; }
|
|
|
|
public List<DirectoryNode> Directories { get; } = new List<DirectoryNode>();
|
|
|
|
public List<FileNode> Files { get; } = new List<FileNode>();
|
|
|
|
public void AddFile(FileNode fileNode)
|
|
{
|
|
var filePath = fileNode.Path;
|
|
if (!filePath.StartsWith(Path, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
throw new InvalidOperationException($"File {fileNode.Path} does not belong to {Path}.");
|
|
}
|
|
|
|
// Look for the first / that appears in the path after the current directory path.
|
|
var directoryPath = GetDirectoryPath(filePath);
|
|
var directory = GetOrAddDirectory(this, directoryPath, createIfNotExists: true);
|
|
Debug.Assert(directory != null);
|
|
directory.Files.Add(fileNode);
|
|
}
|
|
|
|
public DirectoryNode GetDirectory(string path)
|
|
{
|
|
if (!path.StartsWith(Path, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
throw new InvalidOperationException($"File {path} does not belong to {Path}.");
|
|
}
|
|
|
|
return GetOrAddDirectory(this, path);
|
|
}
|
|
|
|
public IEnumerable<RazorProjectItem> EnumerateItems()
|
|
{
|
|
foreach (var file in Files)
|
|
{
|
|
yield return file.ProjectItem;
|
|
}
|
|
|
|
foreach (var directory in Directories)
|
|
{
|
|
foreach (var file in directory.EnumerateItems())
|
|
{
|
|
yield return file;
|
|
}
|
|
}
|
|
}
|
|
|
|
public RazorProjectItem GetItem(string path)
|
|
{
|
|
if (!path.StartsWith(Path, StringComparison.OrdinalIgnoreCase))
|
|
{
|
|
throw new InvalidOperationException($"File {path} does not belong to {Path}.");
|
|
}
|
|
|
|
var directoryPath = GetDirectoryPath(path);
|
|
var directory = GetOrAddDirectory(this, directoryPath);
|
|
if (directory == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
foreach (var file in directory.Files)
|
|
{
|
|
var filePath = file.Path;
|
|
var directoryLength = directory.Path.Length;
|
|
|
|
// path, filePath -> /Views/Home/Index.cshtml
|
|
// directory.Path -> /Views/Home/
|
|
// We only need to match the file name portion since we've already matched the directory segment.
|
|
if (string.Compare(path, directoryLength, filePath, directoryLength, path.Length - directoryLength, StringComparison.OrdinalIgnoreCase) == 0)
|
|
{
|
|
return file.ProjectItem;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private static string GetDirectoryPath(string path)
|
|
{
|
|
// /dir1/dir2/file.cshtml -> /dir1/dir2/
|
|
var fileNameIndex = path.LastIndexOf('/');
|
|
if (fileNameIndex == -1)
|
|
{
|
|
return path;
|
|
}
|
|
|
|
return path.Substring(0, fileNameIndex + 1);
|
|
}
|
|
|
|
private static DirectoryNode GetOrAddDirectory(
|
|
DirectoryNode directory,
|
|
string path,
|
|
bool createIfNotExists = false)
|
|
{
|
|
Debug.Assert(!string.IsNullOrEmpty(path));
|
|
if (path[path.Length - 1] != '/')
|
|
{
|
|
path += '/';
|
|
}
|
|
|
|
int index;
|
|
while ((index = path.IndexOf('/', directory.Path.Length)) != -1 && index != path.Length)
|
|
{
|
|
var subDirectory = FindSubDirectory(directory, path);
|
|
|
|
if (subDirectory == null)
|
|
{
|
|
if (createIfNotExists)
|
|
{
|
|
var directoryPath = path.Substring(0, index + 1); // + 1 to include trailing slash
|
|
subDirectory = new DirectoryNode(directoryPath);
|
|
directory.Directories.Add(subDirectory);
|
|
}
|
|
else
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
|
|
directory = subDirectory;
|
|
}
|
|
|
|
return directory;
|
|
}
|
|
|
|
private static DirectoryNode FindSubDirectory(DirectoryNode parentDirectory, string path)
|
|
{
|
|
for (var i = 0; i < parentDirectory.Directories.Count; i++)
|
|
{
|
|
// ParentDirectory.Path -> /Views/Home/
|
|
// CurrentDirectory.Path -> /Views/Home/SubDir/
|
|
// Path -> /Views/Home/SubDir/MorePath/File.cshtml
|
|
// Each invocation of FindSubDirectory returns the immediate subdirectory along the path to the file.
|
|
|
|
var currentDirectory = parentDirectory.Directories[i];
|
|
var directoryPath = currentDirectory.Path;
|
|
var startIndex = parentDirectory.Path.Length;
|
|
var directoryNameLength = directoryPath.Length - startIndex;
|
|
|
|
if (string.Compare(path, startIndex, directoryPath, startIndex, directoryPath.Length - startIndex, StringComparison.OrdinalIgnoreCase) == 0)
|
|
{
|
|
return currentDirectory;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|
|
|
|
// Internal for testing
|
|
[DebuggerDisplay("{Path}")]
|
|
internal struct FileNode
|
|
{
|
|
public FileNode(string path, RazorProjectItem projectItem)
|
|
{
|
|
Path = path;
|
|
ProjectItem = projectItem;
|
|
}
|
|
|
|
public string Path { get; }
|
|
|
|
public RazorProjectItem ProjectItem { get; }
|
|
}
|
|
|
|
private class NotFoundProjectItem : RazorProjectItem
|
|
{
|
|
public NotFoundProjectItem(string basePath, string path)
|
|
{
|
|
BasePath = basePath;
|
|
FilePath = path;
|
|
}
|
|
|
|
public override string BasePath { get; }
|
|
|
|
public override string FilePath { get; }
|
|
|
|
public override bool Exists => false;
|
|
|
|
public override string PhysicalPath => throw new NotSupportedException();
|
|
|
|
public override Stream Read() => throw new NotSupportedException();
|
|
}
|
|
}
|
|
} |