Skip to content

Commit 5e768d1

Browse files
authored
fix: Throw correct exception when using File.Replace with case-only changes on MacOS (#638)
* Add tests for case-only changes in File.Copy, File.Move and File.Replace * Ensure correct casing in `Directory.Move` tests * Throw correct exception on MacOS
1 parent aa39bb6 commit 5e768d1

File tree

6 files changed

+76
-1
lines changed

6 files changed

+76
-1
lines changed

Source/Testably.Abstractions.Testing/Helpers/ExceptionFactory.cs

+4
Original file line numberDiff line numberDiff line change
@@ -203,6 +203,10 @@ internal static IOException ProcessCannotAccessTheFile(string path, int hResult)
203203
$"The process cannot access the file '{path}' because it is being used by another process.",
204204
hResult);
205205

206+
internal static IOException ReplaceSourceMustBeDifferentThanDestination(
207+
string sourcePath, string destinationPath)
208+
=> new($"The source '{sourcePath}' and destination '{destinationPath}' are the same file.", -2146232800);
209+
206210
#pragma warning disable MA0015 // Specify the parameter name
207211
internal static ArgumentException SearchPatternCannotContainTwoDots()
208212
=> new(

Source/Testably.Abstractions.Testing/Storage/InMemoryStorage.cs

+6
Original file line numberDiff line numberDiff line change
@@ -441,6 +441,12 @@ public IStorageContainer GetOrCreateContainer(
441441
throw ExceptionFactory.AccessToPathDenied(source.FullPath);
442442
}
443443

444+
if (_fileSystem.Execute.IsMac &&
445+
source.FullPath.Equals(destination.FullPath, StringComparison.OrdinalIgnoreCase))
446+
{
447+
throw ExceptionFactory.ReplaceSourceMustBeDifferentThanDestination(source.FullPath, destination.FullPath);
448+
}
449+
444450
using (_ = sourceContainer.RequestAccess(
445451
FileAccess.ReadWrite,
446452
FileShare.None,

Tests/Testably.Abstractions.Tests/FileSystem/Directory/MoveTests.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,7 @@ public void Move_CaseOnlyChange_ShouldMoveDirectoryWithContent(string path)
2929
FileSystem.Directory.Exists(source).Should().Be(!Test.RunsOnLinux);
3030
FileSystem.Should().HaveDirectory(destination);
3131
FileSystem.Directory.GetDirectories(".").Should()
32-
.ContainSingle(d => d.Contains(destination));
32+
.ContainSingle(d => d.Contains(destination, StringComparison.Ordinal));
3333
FileSystem.Directory.GetFiles(destination, initialized[1].Name)
3434
.Should().ContainSingle();
3535
FileSystem.Directory.GetDirectories(destination, initialized[2].Name)

Tests/Testably.Abstractions.Tests/FileSystem/File/CopyTests.cs

+27
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,33 @@ public abstract partial class CopyTests<TFileSystem>
77
: FileSystemTestBase<TFileSystem>
88
where TFileSystem : IFileSystem
99
{
10+
[SkippableTheory]
11+
[AutoData]
12+
public void Copy_CaseOnlyChange_ShouldThrowIOException_ExceptOnLinux(
13+
string name, string contents)
14+
{
15+
string sourceName = name.ToLowerInvariant();
16+
string destinationName = name.ToUpperInvariant();
17+
FileSystem.File.WriteAllText(sourceName, contents);
18+
19+
Exception? exception = Record.Exception(() =>
20+
{
21+
FileSystem.File.Copy(sourceName, destinationName);
22+
});
23+
24+
if (Test.RunsOnLinux)
25+
{
26+
exception.Should().BeNull();
27+
FileSystem.File.Exists(sourceName).Should().BeTrue();
28+
FileSystem.File.Exists(destinationName).Should().BeTrue();
29+
}
30+
else
31+
{
32+
exception.Should()
33+
.BeException<IOException>(hResult: Test.RunsOnWindows ? -2147024816 : 17);
34+
}
35+
}
36+
1037
[SkippableTheory]
1138
[AutoData]
1239
public void

Tests/Testably.Abstractions.Tests/FileSystem/File/MoveTests.cs

+2
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,8 @@ public void Move_CaseOnlyChange_ShouldMoveFileWithContent(
2626

2727
FileSystem.Should().HaveFile(destinationName)
2828
.Which.HasContent(contents);
29+
FileSystem.Directory.GetFiles(".").Should()
30+
.ContainSingle(d => d.Contains(destinationName, StringComparison.Ordinal));
2931
}
3032

3133
[SkippableTheory]

Tests/Testably.Abstractions.Tests/FileSystem/File/ReplaceTests.cs

+36
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,42 @@ public abstract partial class ReplaceTests<TFileSystem>
77
: FileSystemTestBase<TFileSystem>
88
where TFileSystem : IFileSystem
99
{
10+
[SkippableTheory]
11+
[AutoData]
12+
public void Replace_CaseOnlyChange_ShouldThrowIOException(
13+
string name, string contents)
14+
{
15+
string sourceName = name.ToLowerInvariant();
16+
string destinationName = name.ToUpperInvariant();
17+
FileSystem.File.WriteAllText(sourceName, contents);
18+
FileSystem.File.WriteAllText(destinationName, "other-content");
19+
20+
Exception? exception = Record.Exception(() =>
21+
{
22+
FileSystem.File.Replace(sourceName, destinationName, null);
23+
});
24+
25+
26+
if (Test.RunsOnLinux)
27+
{
28+
exception.Should().BeNull();
29+
FileSystem.File.Exists(sourceName).Should().BeFalse();
30+
FileSystem.File.Exists(destinationName).Should().BeTrue();
31+
}
32+
else if (Test.RunsOnMac)
33+
{
34+
exception.Should().BeException<IOException>(
35+
hResult: -2146232800,
36+
messageContains: $"The source '{FileSystem.Path.GetFullPath(sourceName)}' and destination '{FileSystem.Path.GetFullPath(destinationName)}' are the same file");
37+
}
38+
else
39+
{
40+
exception.Should().BeException<IOException>(
41+
hResult: -2147024864,
42+
messageContains: "The process cannot access the file");
43+
}
44+
}
45+
1046
[SkippableTheory]
1147
[AutoData]
1248
public void

0 commit comments

Comments
 (0)