dotnet-installer/test/Microsoft.DotNet.Tools.Tests.Utilities/Mock/FileSystemMockBuilder.cs
William Li b0ee5db411
consume bring your own shim(byos) (#9018)
If there are shims packaged by convention in nupkg. Shim Repository will simply copy it to the right location.

The query interface ToolPackageInstance will be in charge of finding the shim folder and filter the right RID. Shim Repository will pick the right file after the folder is located since Shim Repository knows the shim name and it also book keep the files at uninstallation.
During development, due to the wrong adapter level. The mock duplicated too much logic. So, I corrected the abstraction level to lower (only create shim). And replaced the existing mock with a much smaller one without any atomic control and file move, copy logic. At the same time. The chmod, which is a IO action, causes problem during tests. So I added adapter layer to it and put it in Util.
2018-04-10 15:42:50 -07:00

267 lines
8 KiB
C#

// Copyright (c) .NET Foundation and contributors. All rights reserved.
// Licensed under the MIT license. See LICENSE file in the project root for full license information.
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using Microsoft.DotNet.Tools.Test.Utilities.Mock;
using Microsoft.Extensions.EnvironmentAbstractions;
namespace Microsoft.Extensions.DependencyModel.Tests
{
class FileSystemMockBuilder
{
private Dictionary<string, string> _files = new Dictionary<string, string>();
public string TemporaryFolder { get; set; }
internal static IFileSystem Empty { get; } = Create().Build();
public static FileSystemMockBuilder Create()
{
return new FileSystemMockBuilder();
}
public FileSystemMockBuilder AddFile(string name, string content = "")
{
_files.Add(name, content);
return this;
}
public FileSystemMockBuilder AddFiles(string basePath, params string[] files)
{
foreach (var file in files)
{
AddFile(Path.Combine(basePath, file));
}
return this;
}
internal IFileSystem Build()
{
return new FileSystemMock(_files, TemporaryFolder);
}
private class FileSystemMock : IFileSystem
{
public FileSystemMock(Dictionary<string, string> files, string temporaryFolder)
{
File = new FileMock(files);
Directory = new DirectoryMock(files, temporaryFolder);
}
public IFile File { get; }
public IDirectory Directory { get; }
}
private class FileMock : IFile
{
private Dictionary<string, string> _files;
public FileMock(Dictionary<string, string> files)
{
_files = files;
}
public bool Exists(string path)
{
return _files.ContainsKey(path);
}
public string ReadAllText(string path)
{
string text;
if (!_files.TryGetValue(path, out text))
{
throw new FileNotFoundException(path);
}
return text;
}
public Stream OpenRead(string path)
{
return new MemoryStream(Encoding.UTF8.GetBytes(ReadAllText(path)));
}
public Stream OpenFile(
string path,
FileMode fileMode,
FileAccess fileAccess,
FileShare fileShare,
int bufferSize,
FileOptions fileOptions)
{
throw new NotImplementedException();
}
public void CreateEmptyFile(string path)
{
_files.Add(path, string.Empty);
}
public void WriteAllText(string path, string content)
{
_files[path] = content;
}
public void Move(string source, string destination)
{
if (!Exists(source))
{
throw new FileNotFoundException("source does not exist.");
}
if (Exists(destination))
{
throw new IOException("destination exists.");
}
var content = _files[source];
_files.Remove(source);
_files[destination] = content;
}
public void Delete(string path)
{
if (!Exists(path))
{
return;
}
_files.Remove(path);
}
public void Copy(string source, string destination)
{
if (!Exists(source))
{
throw new FileNotFoundException("source does not exist.");
}
if (Exists(destination))
{
throw new IOException("destination exists.");
}
_files[destination] = _files[source];
}
}
private class DirectoryMock : IDirectory
{
private Dictionary<string, string> _files;
private readonly TemporaryDirectoryMock _temporaryDirectory;
public DirectoryMock(Dictionary<string, string> files, string temporaryDirectory)
{
_files = files;
_temporaryDirectory = new TemporaryDirectoryMock(temporaryDirectory);
}
public ITemporaryDirectory CreateTemporaryDirectory()
{
return _temporaryDirectory;
}
public IEnumerable<string> EnumerateFiles(string path, string searchPattern)
{
throw new NotImplementedException();
}
public IEnumerable<string> EnumerateFileSystemEntries(string path)
{
foreach (var entry in _files.Keys.Where(k => Path.GetDirectoryName(k) == path))
{
yield return entry;
}
}
public IEnumerable<string> EnumerateFileSystemEntries(string path, string searchPattern)
{
if (searchPattern != "*")
{
throw new NotImplementedException();
}
return EnumerateFileSystemEntries(path);
}
public string GetDirectoryFullName(string path)
{
throw new NotImplementedException();
}
public bool Exists(string path)
{
return _files.Keys.Any(k => k.StartsWith(path));
}
public void CreateDirectory(string path)
{
var current = path;
while (!string.IsNullOrEmpty(current))
{
_files[current] = current;
current = Path.GetDirectoryName(current);
}
}
public void Delete(string path, bool recursive)
{
if (!recursive && Exists(path) == true)
{
if (_files.Keys.Where(k => k.StartsWith(path)).Count() > 1)
{
throw new IOException("The directory is not empty");
}
}
foreach (var k in _files.Keys.Where(k => k.StartsWith(path)).ToList())
{
_files.Remove(k);
}
}
public void Move(string source, string destination)
{
if (!Exists(source))
{
throw new IOException("The source directory does not exist.");
}
if (Exists(destination))
{
throw new IOException("The destination already exists.");
}
foreach (var kvp in _files.Where(kvp => kvp.Key.StartsWith(source)).ToList())
{
var newKey = destination + kvp.Key.Substring(source.Length);
var newValue = kvp.Value.StartsWith(source) ?
destination + kvp.Value.Substring(source.Length) :
kvp.Value;
_files.Add(newKey, newValue);
_files.Remove(kvp.Key);
}
}
}
private class TemporaryDirectoryMock : ITemporaryDirectoryMock
{
public bool DisposedTemporaryDirectory { get; private set; }
public TemporaryDirectoryMock(string temporaryDirectory)
{
DirectoryPath = temporaryDirectory;
}
public string DirectoryPath { get; }
public void Dispose()
{
DisposedTemporaryDirectory = true;
}
}
}
}