File: src\Analyzers\CSharp\Tests\MatchFolderAndNamespace\CSharpMatchFolderAndNamespaceTests.cs
Web Access
Project: src\src\CodeStyle\CSharp\Tests\Microsoft.CodeAnalysis.CSharp.CodeStyle.UnitTests.csproj (Microsoft.CodeAnalysis.CSharp.CodeStyle.UnitTests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using System.Collections.Generic;
using System.IO;
using System.Threading.Tasks;
using Microsoft.CodeAnalysis.CSharp.Analyzers.MatchFolderAndNamespace;
using Microsoft.CodeAnalysis.CSharp.CodeFixes.MatchFolderAndNamespace;
using Microsoft.CodeAnalysis.Editor.UnitTests.CodeActions;
using Microsoft.CodeAnalysis.Shared.Extensions;
using Roslyn.Test.Utilities;
using Xunit;
 
namespace Microsoft.CodeAnalysis.CSharp.Analyzers.UnitTests.MatchFolderAndNamespace;
 
using VerifyCS = CSharpCodeFixVerifier<
    CSharpMatchFolderAndNamespaceDiagnosticAnalyzer,
    CSharpChangeNamespaceToMatchFolderCodeFixProvider>;
 
public sealed class CSharpMatchFolderAndNamespaceTests
{
    private static readonly string Directory = "/0/";
 
    // DefaultNamespace gets exposed as RootNamespace in the build properties
    private const string DefaultNamespace = "Test.Root.Namespace";
 
    private static readonly string EditorConfig = $"""
        is_global=true
        build_property.ProjectDir = {Directory}
        build_property.RootNamespace = {DefaultNamespace}
        """;
 
    private static string CreateFolderPath(params string[] folders)
        => Path.Combine(Directory, Path.Combine(folders));
 
    private static Task RunTestAsync(string fileName, string fileContents, string? directory = null, string? editorConfig = null, string? fixedCode = null, string? defaultNamespace = null)
    {
        var filePath = Path.Combine(directory ?? Directory, fileName);
        fixedCode ??= fileContents;
 
        return RunTestAsync(
            new[] { (filePath, fileContents) },
            new[] { (filePath, fixedCode) },
            editorConfig,
            defaultNamespace);
    }
 
    private static Task RunTestAsync(IEnumerable<(string, string)> originalSources, IEnumerable<(string, string)>? fixedSources = null, string? editorconfig = null, string? defaultNamespace = null)
    {
        // When a namespace isn't provided we will fallback on our default
        defaultNamespace ??= DefaultNamespace;
 
        var testState = new VerifyCS.Test
        {
            EditorConfig = editorconfig ?? EditorConfig,
            CodeFixTestBehaviors = CodeAnalysis.Testing.CodeFixTestBehaviors.SkipFixAllInDocumentCheck,
            LanguageVersion = LanguageVersion.CSharp10,
        };
 
        foreach (var (fileName, content) in originalSources)
            testState.TestState.Sources.Add((fileName, content));
 
        fixedSources ??= [];
        foreach (var (fileName, content) in fixedSources)
            testState.FixedState.Sources.Add((fileName, content));
 
        // If empty string was provided as the namespace, then we will not set a default
        if (defaultNamespace.Length > 0)
        {
            testState.SolutionTransforms.Add((solution, projectId) =>
            {
                var project = solution.GetRequiredProject(projectId);
                return project.WithDefaultNamespace(defaultNamespace).Solution;
            });
        }
 
        return testState.RunAsync();
    }
 
    [Fact]
    public Task InvalidFolderName1_NoDiagnostic()
    {
        // No change namespace action because the folder name is not valid identifier
        var folder = CreateFolderPath(["3B", "C"]);
        var code =
            """
            namespace A.B
            {
                class Class1
                {
                }
            }
            """;
 
        return RunTestAsync(
            "File1.cs",
            code,
            directory: folder);
    }
 
    [Fact]
    public Task InvalidFolderName1_NoDiagnostic_FileScopedNamespace()
    {
        // No change namespace action because the folder name is not valid identifier
        var folder = CreateFolderPath(["3B", "C"]);
        var code =
            """
            namespace A.B;
 
            class Class1
            {
            }
            """;
 
        return RunTestAsync(
            "File1.cs",
            code,
            directory: folder);
    }
 
    [Fact]
    public Task InvalidFolderName2_NoDiagnostic()
    {
        // No change namespace action because the folder name is not valid identifier
        var folder = CreateFolderPath(["B.3C", "D"]);
        var code =
            """
            namespace A.B
            {
                class Class1
                {
                }
            }
            """;
 
        return RunTestAsync(
            "File1.cs",
            code,
            directory: folder);
    }
 
    [Fact]
    public Task InvalidFolderName3_NoDiagnostic()
    {
        // No change namespace action because the folder name is not valid identifier
        var folder = CreateFolderPath([".folder", "..subfolder", "name"]);
        var code =
            """
            namespace A.B
            {
                class Class1
                {
                }
            }
            """;
 
        return RunTestAsync(
            "File1.cs",
            code,
            directory: folder);
    }
 
    [Fact]
    public Task CaseInsensitiveMatch_NoDiagnostic()
    {
        var folder = CreateFolderPath(["A", "B"]);
        var code =
            $$"""
            namespace {{DefaultNamespace}}.a.b
            {
                class Class1
                {
                }
            }
            """;
 
        return RunTestAsync(
            "File1.cs",
            code,
            directory: folder);
    }
 
    [Fact]
    public async Task CodeStyleOptionIsFalse()
    {
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace A.B
            {
                class Class1
                {
                }
            }
            """;
 
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder,
            editorConfig: EditorConfig + """
 
            dotnet_style_namespace_match_folder = false
            """);
    }
 
    [Fact]
    public async Task SingleDocumentNoReference()
    {
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace [|A.B|]
            {
                class Class1
                {
                }
            }
            """;
 
        var fixedCode =
            $$"""
            namespace {{DefaultNamespace}}.B.C
            {
                class Class1
                {
                }
            }
            """;
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder,
            fixedCode: fixedCode);
    }
 
    [Fact]
    public async Task SingleDocumentNoReference_FileScopedNamespace()
    {
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace [|A.B|];
 
            class Class1
            {
            }
            """;
 
        var fixedCode =
            $$"""
            namespace {{DefaultNamespace}}.B.C;
 
            class Class1
            {
            }
            """;
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder,
            fixedCode: fixedCode);
    }
 
    [Fact]
    public async Task SingleDocumentNoReference_NoDefaultNamespace()
    {
        var editorConfig = $"""
            is_global=true
            build_property.ProjectDir = {Directory}
            """;
 
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace [|A.B|]
            {
                class Class1
                {
                }
            }
            """;
 
        var fixedCode =
            $$"""
            namespace B.C
            {
                class Class1
                {
                }
            }
            """;
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder,
            fixedCode: fixedCode,
            editorConfig: editorConfig,
            // passing empty string means that a default namespace isn't set on the test Project
            defaultNamespace: string.Empty);
    }
 
    [Fact]
    public async Task SingleDocumentNoReference_NoDefaultNamespace_FileScopedNamespace()
    {
        var editorConfig = $"""
            is_global=true
            build_property.ProjectDir = {Directory}
            """;
 
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace [|A.B|];
 
            class Class1
            {
            }
            """;
 
        var fixedCode =
            """
            namespace B.C;
 
            class Class1
            {
            }
            """;
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder,
            fixedCode: fixedCode,
            editorConfig: editorConfig,
            // passing empty string means that a default namespace isn't set on the test Project
            defaultNamespace: string.Empty);
    }
 
    [Fact]
    public async Task NamespaceWithSpaces_NoDiagnostic()
    {
        var folder = CreateFolderPath("A", "B");
        var code =
            $$"""
            namespace {{DefaultNamespace}}.A    .     B
            {
                class Class1
                {
                }
            }
            """;
 
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder);
    }
 
    [Fact]
    public async Task NestedNamespaces_NoDiagnostic()
    {
        // The code fix doesn't currently support nested namespaces for sync, so
        // diagnostic does not report.
 
        var folder = CreateFolderPath("B", "C");
        var code =
            """
            namespace A.B
            {
                namespace C.D
                {
                    class CDClass
                    {
                    }
                }
 
                class ABClass
                {
                }
            }
            """;
 
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder);
    }
 
    [Fact]
    public async Task PartialTypeWithMultipleDeclarations_NoDiagnostic()
    {
        // The code fix doesn't currently support nested namespaces for sync, so
        // diagnostic does not report.
 
        var folder = CreateFolderPath("B", "C");
        var code1 =
            """
            namespace A.B
            {
                partial class ABClass
                {
                    void M1() {}
                }
            }
            """;
 
        var code2 =
            """
            namespace A.B
            {
                partial class ABClass
                {
                    void M2() {}
                }
            }
            """;
 
        var sources = new[]
        {
            (Path.Combine(folder, "ABClass1.cs"), code1),
            (Path.Combine(folder, "ABClass2.cs"), code2),
        };
 
        await RunTestAsync(sources);
    }
 
    [Fact]
    public async Task FileNotInProjectFolder_NoDiagnostic()
    {
        // Default directory is Test\Directory for the project,
        // putting the file outside the directory should have no
        // diagnostic shown.
 
        var folder = Path.Combine("B", "C");
        var code =
            $$"""
            namespace A.B
            {
                class ABClass
                {
                }
            }
            """;
 
        await RunTestAsync(
            fileName: "Class1.cs",
            fileContents: code,
            directory: folder);
    }
 
    [Fact]
    public async Task SingleDocumentLocalReference()
    {
        var @namespace = "Bar.Baz";
 
        var folder = CreateFolderPath("A", "B", "C");
        var code =
            $$"""
            namespace [|{{@namespace}}|]
            {
                delegate void D1();
 
                interface Class1
                {
                    void M1();
                }
 
                class Class2 : {{@namespace}}.Class1
                {
                    {{@namespace}}.D1 d;
 
                    void {{@namespace}}.Class1.M1() { }
                }
            }
            """;
 
        var expected =
            $$"""
            namespace {{DefaultNamespace}}.A.B.C
            {
                delegate void D1();
 
                interface Class1
                {
                    void M1();
                }
 
                class Class2 : Class1
                {
                    D1 d;
 
                    void Class1.M1() { }
                }
            }
            """;
 
        await RunTestAsync(
            "Class1.cs",
            code,
            folder,
            fixedCode: expected);
    }
 
    [Fact]
    public async Task ChangeUsingsInMultipleContainers()
    {
        var declaredNamespace = "Bar.Baz";
 
        var folder = CreateFolderPath("A", "B", "C");
        var code1 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class1
                {
                }
            }
            """;
 
        var code2 =
            $$"""
            namespace NS1
            {
                using {{declaredNamespace}};
 
                class Class2
                {
                    Class1 c2;
                }
 
                namespace NS2
                {
                    using {{declaredNamespace}};
 
                    class Class2
                    {
                        Class1 c1;
                    }
                }
            }
            """;
 
        var fixed1 =
            $$"""
            namespace {{DefaultNamespace}}.A.B.C
            {
                class Class1
                {
                }
            }
            """;
 
        var fixed2 =
            $$"""
            namespace NS1
            {
                using {{DefaultNamespace}}.A.B.C;
 
                class Class2
                {
                    Class1 c2;
                }
 
                namespace NS2
                {
                    class Class2
                    {
                        Class1 c1;
                    }
                }
            }
            """;
 
        var originalSources = new[]
        {
            (Path.Combine(folder, "Class1.cs"), code1),
            ("Class2.cs", code2)
        };
 
        var fixedSources = new[]
        {
            (Path.Combine(folder, "Class1.cs"), fixed1),
            ("Class2.cs", fixed2)
        };
 
        await RunTestAsync(originalSources, fixedSources);
    }
 
    [Fact]
    public async Task DocumentAtRoot_NoDiagnostic()
    {
        var folder = CreateFolderPath();
 
        var code = $$"""
            namespace {{DefaultNamespace}}
            {
                class C { }
            }
            """;
 
        await RunTestAsync(
            "File1.cs",
            code,
            folder);
    }
 
    [Fact]
    public async Task DocumentAtRoot_ChangeNamespace()
    {
        var folder = CreateFolderPath();
 
        var code =
            $$"""
            namespace [|{{DefaultNamespace}}.Test|]
            {
                class C { }
            }
            """;
 
        var fixedCode =
            $$"""
            namespace {{DefaultNamespace}}
            {
                class C { }
            }
            """;
 
        await RunTestAsync(
            "File1.cs",
            code,
            folder,
            fixedCode: fixedCode);
    }
 
    [Fact]
    public async Task ChangeNamespace_WithAliasReferencesInOtherDocument()
    {
        var declaredNamespace = $"Bar.Baz";
 
        var folder = CreateFolderPath("A", "B", "C");
        var code1 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class1
                {
                }
            }
            """;
 
        var code2 = $$"""
            using System;
            using {{declaredNamespace}};
            using Class1Alias = {{declaredNamespace}}.Class1;
 
            namespace Foo
            {
                class RefClass
                {
                    private Class1Alias c1;
                }
            }
            """;
 
        var fixed1 =
            $$"""
            namespace {{DefaultNamespace}}.A.B.C
            {
                class Class1
                {
                }
            }
            """;
 
        var fixed2 =
            $$"""
            using System;
            using Class1Alias = {{DefaultNamespace}}.A.B.C.Class1;
 
            namespace Foo
            {
                class RefClass
                {
                    private Class1Alias c1;
                }
            }
            """;
 
        var originalSources = new[]
        {
            (Path.Combine(folder, "Class1.cs"), code1),
            ("Class2.cs", code2)
        };
 
        var fixedSources = new[]
        {
            (Path.Combine(folder, "Class1.cs"), fixed1),
            ("Class2.cs", fixed2)
        };
 
        await RunTestAsync(originalSources, fixedSources);
    }
 
    [Fact]
    public async Task FixAll()
    {
        var declaredNamespace = "Bar.Baz";
 
        var folder1 = CreateFolderPath("A", "B", "C");
        var fixedNamespace1 = $"{DefaultNamespace}.A.B.C";
 
        var folder2 = CreateFolderPath("Second", "Folder", "Path");
        var fixedNamespace2 = $"{DefaultNamespace}.Second.Folder.Path";
 
        var folder3 = CreateFolderPath("Third", "Folder", "Path");
        var fixedNamespace3 = $"{DefaultNamespace}.Third.Folder.Path";
 
        var code1 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class1
                {
                    Class2 C2 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var fixed1 =
            $$"""
            using {{fixedNamespace2}};
            using {{fixedNamespace3}};
 
            namespace {{fixedNamespace1}}
            {
                class Class1
                {
                    Class2 C2 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var code2 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class2
                {
                    Class1 C1 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var fixed2 =
            $$"""
            using {{fixedNamespace1}};
            using {{fixedNamespace3}};
 
            namespace {{fixedNamespace2}}
            {
                class Class2
                {
                    Class1 C1 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var code3 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class3
                {
                    Class1 C1 { get; }
                    Class2 C2 { get; }
                }
            }
            """;
 
        var fixed3 =
            $$"""
            using {{fixedNamespace1}};
            using {{fixedNamespace2}};
 
            namespace {{fixedNamespace3}}
            {
                class Class3
                {
                    Class1 C1 { get; }
                    Class2 C2 { get; }
                }
            }
            """;
 
        var sources = new[]
        {
            (Path.Combine(folder1, "Class1.cs"), code1),
            (Path.Combine(folder2, "Class2.cs"), code2),
            (Path.Combine(folder3, "Class3.cs"), code3),
        };
 
        var fixedSources = new[]
        {
            (Path.Combine(folder1, "Class1.cs"), fixed1),
            (Path.Combine(folder2, "Class2.cs"), fixed2),
            (Path.Combine(folder3, "Class3.cs"), fixed3),
        };
 
        await RunTestAsync(sources, fixedSources);
    }
 
    [Fact]
    public async Task FixAll_MultipleProjects()
    {
        var declaredNamespace = "Bar.Baz";
 
        var folder1 = CreateFolderPath("A", "B", "C");
        var fixedNamespace1 = $"{DefaultNamespace}.A.B.C";
 
        var folder2 = CreateFolderPath("Second", "Folder", "Path");
        var fixedNamespace2 = $"{DefaultNamespace}.Second.Folder.Path";
 
        var folder3 = CreateFolderPath("Third", "Folder", "Path");
        var fixedNamespace3 = $"{DefaultNamespace}.Third.Folder.Path";
 
        var code1 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                public class Class1
                {
                    Class2 C2 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var fixed1 =
            $$"""
            using {{fixedNamespace2}};
            using {{fixedNamespace3}};
 
            namespace {{fixedNamespace1}}
            {
                public class Class1
                {
                    Class2 C2 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var code2 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class2
                {
                    Class1 C1 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var fixed2 =
            $$"""
            using {{fixedNamespace1}};
            using {{fixedNamespace3}};
 
            namespace {{fixedNamespace2}}
            {
                class Class2
                {
                    Class1 C1 { get; }
                    Class3 C3 { get; }
                }
            }
            """;
 
        var code3 =
            $$"""
            namespace [|{{declaredNamespace}}|]
            {
                class Class3
                {
                    Class1 C1 { get; }
                    Class2 C2 { get; }
                }
            }
            """;
 
        var fixed3 =
            $$"""
            using {{fixedNamespace1}};
            using {{fixedNamespace2}};
 
            namespace {{fixedNamespace3}}
            {
                class Class3
                {
                    Class1 C1 { get; }
                    Class2 C2 { get; }
                }
            }
            """;
 
        var project2Directory = "/Project2/";
        var project2folder = Path.Combine(project2Directory, "A", "B", "C");
        var project2EditorConfig = $"""
            is_global=true
            build_property.ProjectDir = {project2Directory}
            build_property.RootNamespace = {DefaultNamespace}
            """;
 
        var project2Source =
            $$"""
            using {{declaredNamespace}};
 
            namespace [|Project2.Test|]
            {
                class P
                {
                    Class1 _c1;
                }
            }
            """;
 
        var project2FixedSource =
            $$"""
            namespace {{fixedNamespace1}}
            {
                class P
                {
                    Class1 _c1;
                }
            }
            """;
 
        var testState = new VerifyCS.Test
        {
            EditorConfig = EditorConfig,
            CodeFixTestBehaviors = CodeAnalysis.Testing.CodeFixTestBehaviors.SkipFixAllInDocumentCheck | CodeAnalysis.Testing.CodeFixTestBehaviors.SkipFixAllInProjectCheck,
            LanguageVersion = LanguageVersion.CSharp10,
            TestState =
            {
                Sources =
                {
                    (Path.Combine(folder1, "Class1.cs"), code1),
                    (Path.Combine(folder2, "Class2.cs"), code2),
                    (Path.Combine(folder3, "Class3.cs"), code3),
                },
                AdditionalProjects =
                {
                    ["Project2"] =
                    {
                        AdditionalProjectReferences = { "TestProject" },
                        Sources = { (Path.Combine(project2folder, "P.cs"), project2Source) },
                        AnalyzerConfigFiles = { (Path.Combine(project2Directory, ".editorconfig"), project2EditorConfig) },
                    },
                },
            },
            FixedState =
            {
                Sources =
                {
                    (Path.Combine(folder1, "Class1.cs"), fixed1),
                    (Path.Combine(folder2, "Class2.cs"), fixed2),
                    (Path.Combine(folder3, "Class3.cs"), fixed3),
                },
                AdditionalProjects =
                {
                    ["Project2"] =
                    {
                        AdditionalProjectReferences = { "TestProject" },
                        Sources = { (Path.Combine(project2folder, "P.cs"), project2FixedSource) },
                        AnalyzerConfigFiles = { (Path.Combine(project2Directory, ".editorconfig"), project2EditorConfig) },
                    }
                }
            }
        };
 
        testState.SolutionTransforms.Add((solution, projectId) =>
        {
            foreach (var id in solution.ProjectIds)
            {
                var project = solution.GetRequiredProject(id);
                solution = project.WithDefaultNamespace(DefaultNamespace).Solution;
            }
            return solution;
        });
 
        await testState.RunAsync();
    }
 
    [Fact, WorkItem("https://github.com/dotnet/roslyn/issues/58372")]
    public async Task InvalidProjectName_ChangeNamespace()
    {
        var defaultNamespace = "Invalid-Namespace";
        var editorConfig = $"""
            is_global=true
            build_property.ProjectDir = {Directory}
            build_property.RootNamespace = {defaultNamespace}
            """;
 
        var folder = CreateFolderPath(["B", "C"]);
        var code =
            """
            namespace [|A.B|]
            {
                class Class1
                {
                }
            }
            """;
 
        // The project name is invalid so the default namespace is not prepended
        var fixedCode =
            $$"""
            namespace B.C
            {
                class Class1
                {
                }
            }
            """;
 
        await RunTestAsync(
            "Class1.cs",
            fileContents: code,
            fixedCode: fixedCode,
            directory: folder,
            editorConfig: editorConfig,
            defaultNamespace: defaultNamespace);
    }
 
    [Fact]
    public async Task InvalidProjectName_DocumentAtRoot_ChangeNamespace()
    {
        var defaultNamespace = "Invalid-Namespace";
        var editorConfig = $"""
            is_global=true
            build_property.ProjectDir = {Directory}
            build_property.RootNamespace = {defaultNamespace}
            """;
 
        var folder = CreateFolderPath();
 
        var code =
            $$"""
            namespace Test.Code
            {
                class C { }
            }
            """;
 
        await RunTestAsync(
            "Class1.cs",
            fileContents: code,
            directory: folder,
            editorConfig: editorConfig,
            defaultNamespace: defaultNamespace);
    }
 
    [Fact]
    public async Task InvalidRootNamespace_DocumentAtRoot_ChangeNamespace()
    {
        var editorConfig = $"""
            is_global=true
            build_property.ProjectDir = {Directory}
            build_property.RootNamespace = Test.Code # not an editorconfig comment even though it looks like one
            """;
 
        var folder = CreateFolderPath();
 
        var code =
            $$"""
            namespace Test.Code
            {
                class C { }
            }
            """;
 
        await RunTestAsync(
            "Class1.cs",
            fileContents: code,
            directory: folder,
            editorConfig: editorConfig,
            defaultNamespace: "Invalid-Namespace");
    }
}