File: Rules\AttributesMustMatchTests.cs
Web Access
Project: ..\..\..\test\Microsoft.DotNet.ApiCompatibility.Tests\Microsoft.DotNet.ApiCompatibility.Tests.csproj (Microsoft.DotNet.ApiCompatibility.Tests)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
 
using Microsoft.CodeAnalysis;
using Microsoft.DotNet.ApiCompatibility.Tests;
using Microsoft.DotNet.ApiSymbolExtensions.Filtering;
using Microsoft.DotNet.ApiSymbolExtensions.Tests;
 
namespace Microsoft.DotNet.ApiCompatibility.Rules.Tests
{
    public class AttributesMustMatchTests
    {
        /*
         * Tests for:
         * - Types
         * - Fields
         * - Properties
         * - Methods
         * - Events
         * - ReturnValues
         * - Constructors
         * - Generic Parameters
         *
         * Grouped into:
         * - Type
         * - Member
         */
 
        private static readonly TestRuleFactory s_ruleFactory = new((settings, context) => new AttributesMustMatch(settings, context));
 
        public static TheoryData<string, string, CompatDifference[]> TypesCases => new()
        {
            // No change to type's attributes
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
new CompatDifference[] {}
            },
            // Attribute removed from type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "T:CompatTests.First:[T:System.SerializableAttribute]")
}
            },
            // Attribute changed on type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = false, B = 4)]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First:[T:CompatTests.FooAttribute]")
}
            },
            // Attribute repeated with additional arguments
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"")]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"")]
  [Foo(""T"")]
  public class First {}
}
",
new CompatDifference[] {}
            },
 
            // Attribute added to type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
new CompatDifference[] {}
            },
            // Attributes with array and type arguments
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(int[] arr, Type type) {}
  }
 
  [Foo(new int[] {1,2,3}, typeof(int))]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(int[] arr, Type type) {}
  }
 
  [Foo(new int[] {4,5,6}, typeof(bool))]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First:[T:CompatTests.FooAttribute]")
}
            },
            // Attributes on internal type arguments
            {
                @"
namespace CompatTests
{
  using System;
  using CompatTestsSecondNamespace;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(Type type) {}
  }
 
  [Foo(typeof(Bar))]
  public class First {}
}
 
namespace CompatTestsSecondNamespace {
  internal class Bar {}
}
",
                @"
namespace CompatTests
{
  using System;
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(Type type) {}
    public Type A;
  }
 
  internal class Bar {}
 
  [Foo(typeof(Bar))]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First:[T:CompatTests.FooAttribute]"),
}
            }
        };
 
        public static TheoryData<string, string, CompatDifference[]> MembersCases => new() {
            // Attributes on method
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public void F() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public void F() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
}
            },
            // Attributes on property
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public int F { get; }
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public int F { get; }
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "P:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "P:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
}
            },
            // Attributes on event
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public delegate void EventHandler(object sender, object e);
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public event EventHandler F;
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public delegate void EventHandler(object sender, object e);
 
    [Foo(""T"")]
    [Baz]
    public event EventHandler F;
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "E:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "E:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
}
            },
            // Attributes on constructor
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public First() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public First() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.#ctor:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.#ctor:[T:CompatTests.BarAttribute]"),
}
            },
            // Attributes on return type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [return: Foo(""S"", A = true, B = 3)]
    [return: Bar]
    public int F() => 0;
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [return: Foo(""T"")]
    [return: Baz]
    public int F() => 0;
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F->int:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F->int:[T:CompatTests.BarAttribute]"),
}
            },
            // Attributes on method parameter
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F([Bar] int v, [Foo(""S"", A = true, B = 0)] string s) {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F([Baz] int v, [Foo(""T"")] string s) {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F(System.Int32,System.String)$0:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F(System.Int32,System.String)$1:[T:CompatTests.FooAttribute]"),
 
}
            },
            // Attributes on type parameter of class
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First<[Bar] T1, [Foo(""S"", A = true, B = 0)] T2> {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First<[Baz] T1, [Foo(""T"")] T2> {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "T:CompatTests.First`2<0>:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First`2<1>:[T:CompatTests.FooAttribute]"),
 
}
            },
            // Attributes on type parameter of method
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F<[Bar] T1, [Foo(""S"", A = true, B = 0)] T2>() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F<[Baz] T1, [Foo(""T"")] T2>() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F``2<0>:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F``2<1>:[T:CompatTests.FooAttribute]"),
 
}
            }
        };
 
        public static TheoryData<string, string, CompatDifference[]> StrictMode => new()
        {
            // Attribute added to type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "T:CompatTests.First:[T:System.SerializableAttribute]")
}
            },
            // Attribute repeated with additional arguments
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"")]
  public class First {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"")]
  [Foo(""T"")]
  public class First {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First:[T:CompatTests.FooAttribute]")
}
            },
            // Attributes on method
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public void F() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public void F() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "M:CompatTests.First.F:[T:CompatTests.BazAttribute]")
}
            },
            // Attributes on property
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public int F { get; }
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public int F { get; }
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "P:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "P:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "P:CompatTests.First.F:[T:CompatTests.BazAttribute]")
}
            },
            // Attributes on event
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public delegate void EventHandler(object sender, object e);
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public event EventHandler F;
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public delegate void EventHandler(object sender, object e);
 
    [Foo(""T"")]
    [Baz]
    public event EventHandler F;
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "E:CompatTests.First.F:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "E:CompatTests.First.F:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "E:CompatTests.First.F:[T:CompatTests.BazAttribute]")
}
            },
            // Attributes on constructor
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""S"", A = true, B = 3)]
    [Bar]
    public First() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [Foo(""T"")]
    [Baz]
    public First() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.#ctor:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.#ctor:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "M:CompatTests.First.#ctor:[T:CompatTests.BazAttribute]")
}
            },
            // Attributes on return type
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [return: Foo(""S"", A = true, B = 3)]
    [return: Bar]
    public int F() => 0;
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    [return: Foo(""T"")]
    [return: Baz]
    public int F() => 0;
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F->int:[T:CompatTests.FooAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F->int:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "M:CompatTests.First.F->int:[T:CompatTests.BazAttribute]")
}
            },
            // Attributes on method parameter
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F([Bar] int v, [Foo(""S"", A = true, B = 0)] string s) {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F([Baz] int v, [Foo(""T"")] string s) {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F(System.Int32,System.String)$0:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "M:CompatTests.First.F(System.Int32,System.String)$0:[T:CompatTests.BazAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F(System.Int32,System.String)$1:[T:CompatTests.FooAttribute]"),
 
}
            },
            // Attributes on type parameter of class
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First<[Bar] T1, [Foo(""S"", A = true, B = 0)] T2> {}
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First<[Baz] T1, [Foo(""T"")] T2> {}
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "T:CompatTests.First`2<0>:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "T:CompatTests.First`2<0>:[T:CompatTests.BazAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "T:CompatTests.First`2<1>:[T:CompatTests.FooAttribute]"),
 
}
            },
            // Attributes on type parameter of method
            {
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F<[Bar] T1, [Foo(""S"", A = true, B = 0)] T2>() {}
  }
}
",
                @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A = false;
    public int B = 0;
  }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BarAttribute : Attribute { }
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class BazAttribute : Attribute { }
 
  public class First {
 
    public void F<[Baz] T1, [Foo(""T"")] T2>() {}
  }
}
",
new CompatDifference[] {
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotRemoveAttribute, "", DifferenceType.Removed, "M:CompatTests.First.F``2<0>:[T:CompatTests.BarAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotAddAttribute, "", DifferenceType.Added, "M:CompatTests.First.F``2<0>:[T:CompatTests.BazAttribute]"),
    CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.CannotChangeAttribute, "", DifferenceType.Changed, "M:CompatTests.First.F``2<1>:[T:CompatTests.FooAttribute]"),
 
}
            }
        };
 
        [Theory]
        [MemberData(nameof(TypesCases))]
        [MemberData(nameof(MembersCases))]
        public void EnsureDiagnosticIsReported(string leftSyntax, string rightSyntax, CompatDifference[] expected)
        {
            using TempDirectory root = new();
            string filePath = Path.Combine(root.DirPath, "exclusions.txt");
            File.Create(filePath).Dispose();
            IAssemblySymbol left = SymbolFactory.GetAssemblyFromSyntax(leftSyntax);
            IAssemblySymbol right = SymbolFactory.GetAssemblyFromSyntax(rightSyntax);
            ApiComparer differ = new(s_ruleFactory);
            differ.Settings.AttributeDataSymbolFilter = SymbolFilterFactory.GetFilterFromFiles([filePath], respectInternals: false);
 
            IEnumerable<CompatDifference> actual = differ.GetDifferences(left, right);
 
            Assert.Equal(expected, actual);
        }
 
        [Theory]
        [MemberData(nameof(StrictMode))]
        public void EnsureStrictModeReported(string leftSyntax, string rightSyntax, CompatDifference[] expected)
        {
            using TempDirectory root = new();
            string filePath = Path.Combine(root.DirPath, "exclusions.txt");
            File.Create(filePath).Dispose();
            IAssemblySymbol left = SymbolFactory.GetAssemblyFromSyntax(leftSyntax);
            IAssemblySymbol right = SymbolFactory.GetAssemblyFromSyntax(rightSyntax);
            ApiComparer differ = new(s_ruleFactory, new ApiComparerSettings(strictMode: true));
            differ.Settings.AttributeDataSymbolFilter = SymbolFilterFactory.GetFilterFromFiles([filePath], respectInternals: false);
 
            IEnumerable<CompatDifference> actual = differ.GetDifferences(left, right);
 
            Assert.Equal(expected, actual);
        }
 
        [Fact]
        public void TestExclusionsFilteredOut()
        {
            using TempDirectory root = new();
            string filePath = Path.Combine(root.DirPath, "exclusions.txt");
            File.WriteAllText(filePath, "T:System.SerializableAttribute");
            string leftSyntax = @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
";
            string rightSyntax = @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
 
  [Serializable]
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
";
            IAssemblySymbol left = SymbolFactory.GetAssemblyFromSyntax(leftSyntax);
            IAssemblySymbol right = SymbolFactory.GetAssemblyFromSyntax(rightSyntax);
            ApiComparer differ = new(s_ruleFactory);
            differ.Settings.AttributeDataSymbolFilter = SymbolFilterFactory.GetFilterFromFiles([filePath], respectInternals: false);
 
            IEnumerable<CompatDifference> actual = differ.GetDifferences(left, right);
 
            Assert.Empty(actual);
        }
 
        [Fact]
        public void AttributesExcludedButMembersValidated()
        {
            using TempDirectory root = new();
            string filePath = Path.Combine(root.DirPath, "exclusions.txt");
            File.WriteAllText(filePath, "T:CompatTests.FooAttribute");
            TestRuleFactory ruleFactory = new(
                (settings, context) => new AttributesMustMatch(settings, context),
                (settings, context) => new MembersMustExist(settings, context));
            string leftSyntax = @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
  }
  [Foo(""S"", A = true, B = 3)]
  public class First {}
}
";
            string rightSyntax = @"
namespace CompatTests
{
  using System;
 
  [AttributeUsage(AttributeTargets.All, AllowMultiple = true)]
  public class FooAttribute : Attribute {
    public FooAttribute(String s) {}
    public bool A;
    public int B;
    public string X;
  }
  [Foo(""T"", A = false, B = 4)]
  public class First {}
}
";
            IAssemblySymbol left = SymbolFactory.GetAssemblyFromSyntax(leftSyntax);
            IAssemblySymbol right = SymbolFactory.GetAssemblyFromSyntax(rightSyntax);
            ApiComparer differ = new(ruleFactory, new ApiComparerSettings(strictMode: true));
            differ.Settings.AttributeDataSymbolFilter = SymbolFilterFactory.GetFilterFromFiles([filePath], respectInternals: false);
 
            IEnumerable<CompatDifference> actual = differ.GetDifferences(left, right).ToArray();
 
            Assert.Equal(new[]
            {
                CompatDifference.CreateWithDefaultMetadata(DiagnosticIds.MemberMustExist, string.Empty, DifferenceType.Added, "F:CompatTests.FooAttribute.X"),
            }, actual);
        }
    }
}