Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove InterceptableAttribute #68213

Merged
merged 4 commits into from
May 17, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
51 changes: 4 additions & 47 deletions docs/features/interceptors.md
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@ c.InterceptableMethod(1); // prints "interceptable 1"

class C
{
[Interceptable]
public void InterceptableMethod(int param)
{
Console.WriteLine($"interceptable {param}");
Expand Down Expand Up @@ -47,49 +46,6 @@ static class D
## Detailed design
[design]: #detailed-design

### InterceptableAttribute

A method can indicate that its calls can be *intercepted* by including `[Interceptable]` on its declaration.

PROTOTYPE(ic): For now, if a call is intercepted to a method which lacks this attribute, a warning is reported, and interception still occurs. This may be changed to an error in the future.

```cs
namespace System.Runtime.CompilerServices
{
[AttributeUsage(AttributeTargets.Method)]
public sealed class InterceptableAttribute : Attribute { }
}
```

#### PROTOTYPE(ic): Use an assembly attribute instead?
The main reason we want something like `[Interceptable]` is so that users can better reason about *which calls might be getting intercepted or not*. We don't think there's a meaningful *security* concern which is addressed by `[Interceptable]`--if something untrusted has the ability to add source files to your build, taking away the ability for them to intercept certain calls doesn't really make a difference.

It's been suggested that `[Interceptable]` on method declarations is the wrong point of control. Some SG authors will have good reason to want to intercept a call to a method from a library they don't own--for example, EF wanting to intercept calls to IQueryable extensions. It shouldn't be necessary for those authors to have to request the library authors to add an attribute before they can start doing it.

One option to address this is to remove `[Interceptable]` from the design and not provide any mechanism for limiting which calls can be intercepted or not.

Another option would be to expose an assembly attribute instead.

```cs
namespace System.Runtime.CompilerServices
{
/// <summary>If present, indicates the set of types whose methods can have their calls intercepted in the current compilation.</summary>
/// <remarks>When this attribute is not present, methods on any type can have their calls intercepted.</remarks>
[AttributeUsage(AttributeTargets.Assembly, AllowMultiple = false)]
public sealed class AllowInterception(params Type[] types) : Attribute { }
}

[assembly: AllowInterception(typeof(EndpointRouteBuilderExtensions))]
```

In order to intercept a classic extension method call, the extension type must be provided to this attribute, not the `this` parameter type.

We specify `AllowMultiple = false` here to imply that the attribute is supposed to be hand-authored by the user. It's not meant for generators to use this to *opt themselves in* to being able to intercept things--it's for the user to state once and comprehensively which things can be intercepted.

In such a scheme, it's not clear how often users would actually want to go to the trouble of writing the attribute--particularly since the process of writing and maintaining it would simply be to run source generators, review the "not interceptable" errors, and add the related types to this attribute. Is that useful to users in practice?

Another alternative here might be to use `AllowMultiple = true`, require in practice that generators produce these attributes (i.e. disallow intercepting in the current compilation if this attribute isn't used in it), and let users search for usages of the attribute to get a sense of which calls may be getting intercepted in their project.

### InterceptsLocationAttribute

A method indicates that it is an *interceptor* by adding one or more `[InterceptsLocation]` attributes. These attributes refer to the source locations of the calls it intercepts.
Expand All @@ -104,10 +60,12 @@ namespace System.Runtime.CompilerServices
}
```

Any "ordinary method" (i.e. with `MethodKind.Ordinary`) can have its calls intercepted.

`[InterceptsLocation]` attributes included in source are emitted to the resulting assembly, just like other custom attributes.

PROTOTYPE(ic): We may want to recognize `file class InterceptsLocationAttribute` as a valid declaration of the attribute, to allow generators to bring the attribute in without conflicting with other generators which may also be bringing the attribute in. See open question in [User opt-in](#user-opt-in).
https://github.com/dotnet/roslyn/issues/67079 is a bug which causes file-local source declarations of well-known attributes to be generally treated as known. When that bug is fixed, we may want to single out one or both of `InterceptableAttribute` and `InterceptsLocationAttribute` as "recognized, even though they are file-local".
https://github.com/dotnet/roslyn/issues/67079 is a bug which causes file-local source declarations of well-known attributes to be generally treated as known. When that bug is fixed, we may want to single out `InterceptsLocationAttribute` as "recognized, even though they are file-local".

#### File paths

Expand Down Expand Up @@ -146,7 +104,6 @@ using System.Runtime.CompilerServices;

class C
{
[Interceptable]
public static void InterceptableMethod<T1>(T1 t) => throw null!;
}

Expand All @@ -160,7 +117,7 @@ static class Program

static class D
{
[InterceptsLocation("Program.cs", 13, 11)]
[InterceptsLocation("Program.cs", 12, 11)]
public static void Interceptor1(object s) => throw null!;
}
```
Expand Down
6 changes: 0 additions & 6 deletions src/Compilers/CSharp/Portable/CSharpResources.resx
Original file line number Diff line number Diff line change
Expand Up @@ -7505,12 +7505,6 @@ To remove the warning, you can use /reference instead (set the Embed Interop Typ
<data name="ERR_BadCaseInSwitchArm" xml:space="preserve">
<value>A switch expression arm does not begin with a 'case' keyword.</value>
</data>
<data name="WRN_CallNotInterceptable" xml:space="preserve">
<value>Call to '{0}' is intercepted, but the method is not marked with 'System.Runtime.CompilerServices.InterceptableAttribute'.</value>
</data>
<data name="WRN_CallNotInterceptable_Title" xml:space="preserve">
<value>Call to a method is intercepted, but the method is not marked with 'System.Runtime.CompilerServices.InterceptableAttribute'.</value>
</data>
<data name="ERR_InterceptorCannotBeGeneric" xml:space="preserve">
<value>Method '{0}' cannot be used as an interceptor because it or its containing type has type parameters.</value>
</data>
Expand Down
1 change: 0 additions & 1 deletion src/Compilers/CSharp/Portable/Errors/ErrorCode.cs
Original file line number Diff line number Diff line change
Expand Up @@ -2193,7 +2193,6 @@ internal enum ErrorCode
ERR_UnsupportedPrimaryConstructorParameterCapturingRefAny = 9136,

// PROTOTYPE(ic): pack errors
WRN_CallNotInterceptable = 27000,
ERR_InterceptorCannotBeGeneric = 27001,
ERR_InterceptorPathNotInCompilation = 27002,
ERR_InterceptorPathNotInCompilationWithCandidate = 27003,
Expand Down
2 changes: 0 additions & 2 deletions src/Compilers/CSharp/Portable/Errors/ErrorFacts.cs
Original file line number Diff line number Diff line change
Expand Up @@ -532,7 +532,6 @@ internal static int GetWarningLevel(ErrorCode code)
case ErrorCode.WRN_ParamsArrayInLambdaOnly:
case ErrorCode.WRN_CapturedPrimaryConstructorParameterPassedToBase:
case ErrorCode.WRN_UnreadPrimaryConstructorParameter:
case ErrorCode.WRN_CallNotInterceptable:
case ErrorCode.WRN_InterceptorSignatureMismatch:
case ErrorCode.WRN_NullabilityMismatchInReturnTypeOnInterceptor:
case ErrorCode.WRN_NullabilityMismatchInParameterTypeOnInterceptor:
Expand Down Expand Up @@ -585,7 +584,6 @@ internal static bool IsBuildOnlyDiagnostic(ErrorCode code)
case ErrorCode.ERR_EncUpdateFailedDelegateTypeChanged:
case ErrorCode.ERR_CannotBeConvertedToUtf8:
case ErrorCode.ERR_FileTypeNonUniquePath:
case ErrorCode.WRN_CallNotInterceptable:
case ErrorCode.ERR_InterceptorSignatureMismatch:
case ErrorCode.ERR_InterceptorMustHaveMatchingThisParameter:
case ErrorCode.ERR_InterceptorMustNotHaveThisParameter:
Expand Down

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Original file line number Diff line number Diff line change
Expand Up @@ -154,13 +154,6 @@ private void InterceptCallAndAdjustArguments(
Debug.Assert(interceptableLocation != null);
Debug.Assert(interceptor.Arity == 0);

if (!method.IsInterceptable)
{
// PROTOTYPE(ic): Eventually we may want this to be an error.
// For now it's convenient to just warn so we can experiment with intercepting APIs that haven't yet been marked.
this._diagnostics.Add(ErrorCode.WRN_CallNotInterceptable, attributeLocation, method);
}

var containingMethod = this._factory.CurrentFunction;
Debug.Assert(containingMethod is not null);

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -149,21 +149,5 @@ public UnmanagedCallersOnlyAttributeData? UnmanagedCallersOnlyAttributeData
SetDataStored();
}
}

private bool _hasInterceptableAttribute;
public bool HasInterceptableAttribute
{
get
{
VerifySealed(expected: true);
return _hasInterceptableAttribute;
}
set
{
VerifySealed(expected: false);
_hasInterceptableAttribute = value;
SetDataStored();
}
}
}
}
2 changes: 0 additions & 2 deletions src/Compilers/CSharp/Portable/Symbols/ErrorMethodSymbol.cs
Original file line number Diff line number Diff line change
Expand Up @@ -276,8 +276,6 @@ internal override bool GenerateDebugInfo

internal sealed override bool IsNullableAnalysisEnabled() => false;

internal override bool IsInterceptable => false;

protected override bool HasSetsRequiredMembersImpl
{
get
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -831,7 +831,6 @@ public override bool IsVararg
public override FlowAnalysisAnnotations FlowAnalysisAnnotations => FlowAnalysisAnnotations.None;
internal override bool IsMetadataNewSlot(bool ignoreInterfaceImplementationChanges = false) => false;
internal override bool IsMetadataVirtual(bool ignoreInterfaceImplementationChanges = false) => false;
internal sealed override bool IsInterceptable => false;
internal sealed override UnmanagedCallersOnlyAttributeData? GetUnmanagedCallersOnlyAttributeData(bool forceComplete) => null;

internal override bool GenerateDebugInfo => throw ExceptionUtilities.Unreachable();
Expand Down
Loading