-
Notifications
You must be signed in to change notification settings - Fork 10.3k
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
Add InvokeNew, GetValue & SetValue to JS interop API #61246
base: main
Are you sure you want to change the base?
Conversation
e6be100
to
c6934f6
Compare
I have not benchmarked the effect of the change in how the interop configuration is passed (i.e. as JSON-serialized object). If needed, I can set up such benchmarks. |
@@ -14,6 +14,7 @@ import { addDispatchEventMiddleware } from './Rendering/WebRendererInteropMethod | |||
import { WebAssemblyComponentDescriptor, WebAssemblyServerOptions, discoverWebAssemblyPersistedState } from './Services/ComponentDescriptorDiscovery'; | |||
import { receiveDotNetDataStream } from './StreamingInterop'; | |||
import { WebAssemblyComponentAttacher } from './Platform/WebAssemblyComponentAttacher'; | |||
import { DotNet } from '@microsoft/dotnet-js-interop'; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Where is the source for this dotnet-js-interop
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's the Microsoft.JSInterop.JS
library. See https://github.com/dotnet/aspnetcore/blob/main/src/JSInterop/Microsoft.JSInterop.JS/src/package.json#L2
/// A JSON representation of the arguments. | ||
/// </summary> | ||
[StringSyntax(StringSyntaxAttribute.Json)] | ||
public required string? ArgsJson { get; init; } |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could this be JsonObject
, System.Object
or System.Object[]
so that we don't have to call JSON de/serialization twice ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I love JSInvocationInfoSourceGenerationContext
, makes trimming better. I wonder if that could also made work for application data types
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
- Good idea, I wanted to do that but forgot 🙂
- What do you mean by "application data types"?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
2. What do you mean by "application data types"?
I mean that the json code gen would ideally also generate serializer for class Weather
or whatever is the application payload passed as arguments by value. Or do we only support simple/value types here ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The existing API of InvokeAsync
takes object?[]? args
. This gets serialized to JSON using JsonSerializer.Serialize
with options that we set in the framework and don't give users way to customize them.
If I understand correctly how the source-generated features of System.Text.Json
work, it is a opt-in thing configured per type (in several possible ways). Do you propose we expose a way for the users to give us a JsonSerializerContext
for their types or something akin to that?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, but I don't know how or if that's easy. It could be separate enhancement, because this was not code generated so far. I wonder if making this JsonArray
would prevent it from being source-generated compatible.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree it would be interesting to investigate this further as a separate improvement. If we expose such optimization to the users it would be an opt-in thing, so the APIs that we're adding here (with default reflection-based serialization of user types) are still needed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are several reasons for which we don't do this:
- We consider JSON an implementation detail of the IJSRuntime interface that we reserve the right to change in the future.
- The serialization is used internally by the framework and we don't want any customization from the user to affect how we serialize and deserialize our internal payloads (for example JSObjectReferences, and so on).
- Serialization and deserialization need to match between .NET and JS and there's not a great story for doing this on the JS side, and would require us to expose matching JS APIs.
We already give you a way to customize serialization as much as you want by performing serialization yourself and passing down a byte[]
as an argument. We will marshall the byte[]
instance and you can do your own serialization and deserialization.
@@ -118,7 +118,7 @@ public TestJSRuntime(byte[] data = default) | |||
_data = data; | |||
} | |||
|
|||
public virtual ValueTask<TValue> InvokeAsync<[DynamicallyAccessedMembers(DynamicallyAccessedMemberTypes.PublicConstructors | DynamicallyAccessedMemberTypes.PublicFields | DynamicallyAccessedMemberTypes.PublicProperties)] TValue>(string identifier, CancellationToken cancellationToken, object[] args) | |||
public virtual ValueTask<TValue> InvokeAsync<TValue>(string identifier, CancellationToken cancellationToken, object[] args) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why did the DynamicallyAccessedMembers
go away ? They are helping with trimming, no? Do we need them on the new signatures too ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is trimming relevant in a test project? Anyway, this got removed by mistake. I thought I added it myself and removed it when reverting unintentional changes. I will add it back just in case.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We discussed this with Pavel. The attributes are used properly in all actual production code. They are not used in most existing mock implementations of IJSRuntime
which are found in various test projects. This one particular implementation in PullFromJSDataStreamTest
had the attribute and I removed it by accident.
I am going to keep it like that for consistency. (If anyone has a problem with that, then I argue we should add it to all test projects which is annoying because the JsonSerialized
alias is declared in shared/LinkerFlags.cs
which is not available in most of the test projects due to conflict in referenced projects.)
|
||
class JSObject { | ||
_cachedFunctions: Map<string, Function>; | ||
// TODO(oroztocil): Is it correct to cache functions/properties when they can change? | ||
_cachedMembers: Map<string, ObjectMemberDescriptor>; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
When is this garbage collected ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The lifetime of this cache (and the previously used cache for functions) is tied to the JSObject
instance which is itself held by a module-scoped object cachedJSObjectsById
. See:
const cachedJSObjectsById: { [id: number]: JSObject } = { |
The JSObject
reference can be deleted from the module-scoped cache using disposeJSObjectReference
function. This is called via interop from the DisposeAsync
method of the .NET JSObjectReference
instance. (We also call the JS dispose internally for some objects allocated by the JS framework.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is DisposeAsync
called by JSObjectReference
finalizer ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
JSObjectReference
does not define a finalizer.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Does that mean that if people don't call it explicitly this always leaks memory? Seems like no-go for me.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The caching is done for performance reasons. Let's not change things that aren't a problem today. The pattern has been to tell people to dispose their refs and this hasn't shown to be a problem in either direction. I would be more worried about DotNetObjectReference
if anything than JSObjectReference. In general, we care 100 times less about browser memory than server memory, as in one case it might only crash the tab, in the other it might crash the server. In both cases, there's a bug in the application not correctly releasing the interop references.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The features we just added for dealing with properties are making memory leak worse, right ?
Before this was only caching functions, typically finite number of them.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The features we just added for dealing with properties are making memory leak worse, right ?
Before this was only caching functions, typically finite number of them.
I don't think this is a problem in practice, even if users fail to properly dispose things on the C# side. I don't see a difference between functions and properties, your objects are also going to have a finite set of fields.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are two levels of client-side caching:
- The global
JSObject
cache (map from numeric IDs to object references) - this is necessary for theJSObjectReference
mechanism and we can't/don't want to change that at this point. - The per-
JSObject
cache of resolved functions - this is an internal implementation detail which is not necessary and I would argue does not make a difference performance-wise. It potentially saves a not very expensive client-side operation whose running time will be marginal compared to the network transfer. The downside is that the cache is never updated even though the things identified by the same names can change.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
When I said I would personally be for removing caching, I was only talking about the second kind.
However, I leave it up to you. We can come back to it if someone complains :-)
return property.parent; | ||
} | ||
|
||
// TODO(oroztocil): Do we want to throw on undefined property or return null? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There are few TODO(oroztocil)
still sprinkled in this PR
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, these are intentional and pertain to the questions below.
There are some question that I want to discuss in this review:
|
|
Non-authoritative, just my opinions
Correct.
Yes, recursive.
We are far beyond that point with Blazor interop.
No, stale data and memory leaks. I would say that functions as well need to be resolved on every call, because they could be redefined. Especially now, that we have the setter. |
Identifier = JSMethodIdentifier, | ||
CallType = JSCallType.FunctionCall, | ||
ResultType = JSCallResultType.JSVoidResult, | ||
ArgsJson = argsJson, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Including the arguments here is going to be a problem. As far as I can tell, JSInvocationInfo gets json serialized, so this means ArgsJson gets "doubly" serialized which is going to result in a bunch of additional unescaped characters.
I think we should keep the arguments outside of this struct for that reason. I believe the other implementation does it this way, doesn't it?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Could this be JsonObject
or JsonArray
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Regarding DotNetInvocationInfo
: It is not comparable because that DTO is only used internally on the .NET side. It is not sent over the interop bridge (in neither SignalR, Wasm interop, nor WebView IPC). Rather, individual arguments are transferred and only then packed into DotNetInvocationInfo
.
Meanwhile the JSInvocationInfo
is created by the .NET side, sent over the respective bridge, and read by the JS side. The motivation for this is to set up this communication once and then be able to modify JSInvocationInfo
without needing to go around and update all the layers of all the bridge implementations again.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I will check if passing args
as a JsonObject
or JsonArray
works well (and what does the over-the-wire message looks like). If there are issues with that, then I will pass it as a separate string argument that is serialized once separately.
src/Components/WebAssembly/JSInterop/src/WebAssemblyJSRuntime.cs
Outdated
Show resolved
Hide resolved
src/Components/WebAssembly/WebAssembly.Authentication/test/RemoteAuthenticatorCoreTests.cs
Outdated
Show resolved
Hide resolved
|
||
class JSObject { | ||
_cachedFunctions: Map<string, Function>; | ||
// TODO(oroztocil): Is it correct to cache functions/properties when they can change? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Functions could change too, couldn't they?
While it's potentially problematic if something on the path changes, I think we aren't doing anything different here, and this hasn't been a source of problems in the past.
const synchronousResultOrPromise = jsFunction(...(args || [])); | ||
resolve(synchronousResultOrPromise); | ||
const valueOrPromise = this.handleJSCall(targetInstanceId, identifier, callType, argsJson); | ||
resolve(valueOrPromise); | ||
}); | ||
|
||
// We only listen for a result if the caller wants to be notified about it | ||
if (asyncHandle) { | ||
// On completion, dispatch result back to .NET | ||
// Not using "await" because it codegens a lot of boilerplate |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Probably ok to use await
here since we don't transpile to ES5 anymore (I might be wrong)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Based on package.json
we're targeting ES2019. I'll look into refactoring this with await
.
@javiercn What should in your opinion happen if someone calls |
We shouldn't do anything and let the browser throw an error same as if you wrote it directly on JS |
src/JSInterop/Microsoft.JSInterop.JS/src/src/Microsoft.JSInterop.ts
Outdated
Show resolved
Hide resolved
const result = { parent, name: lastKey } as ObjectMemberDescriptor; | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Instead of creating an object here, you could define a function function getOrSet(value) { if(arguments.length === 0) { return parent[lastKey]; } else { parent[lastKey] = value; }
(or similar) and that way you can treat all options as a function and you don't have to change the cache.
if (!this.isWritableProperty(property.parent, property.name)) { | ||
throw new Error(`The property '${identifier}' is not writable.`); | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We should make these checks (if at all) when we search for the function/getter/setter as opposed to every time the developer tries to read/write the value. I also don't think we should make this check at all. If you try to write to a non-writable property the browser will already give you an error, so this is redundant work we are doing.
This is important because all these are hot paths.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
- We had checks for things that would throw in the browser already. E.g. we throw a custom error when we resolve a thing that is not a function while handling
InvokeAsync
interop call (even though it would be aTypeError
in JS anyway). - Some things are not error in JS but could be reasonably expected by a .NET developer to be an error. E.g. reading from a setter-only property is not an error but returns
undefined
. Writing into getter-only property is not an error and simply does nothing. Both of these seem like a bug that a user would never want to do intentionally. - Is it actually a hot path? This code runs only as many times as the user calls and interop method. The entire interop operation will be dominated by the interop bridge communication (especially if that happens over network). If they call
GetValue
in a loop, the application will be killed by all the messages, not by this piece of client-side code.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
However, if we keep caching resolved members, then it makes sense to do this check only when we resolve the property and not on every access. I can rewrite it like that 👍
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm fine if we want to keep the checks as long as we only perform them when we look up the value/function.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I can also simply remove the checks and let whatever happens, happen. I did it this way as it seemed to me we are trying to protect the .NET developer a bit from common JS quirks.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is it actually a hot path? This code runs only as many times as the user calls and interop method. The entire interop operation will be dominated by the interop bridge communication (especially if that happens over network). If they call GetValue in a loop, the application will be killed by all the messages, not by this piece of client-side code.
It's still part of the hot path, and there isn't a significant overhead when we are calling a function from webassembly.
For such fundamental operations like these it's important to "trim the fat" as much as possible as in many situations all these calls accumulate and "compound" over time
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I agree with performing the checks once when we resolve and cache the property (or the property access function as you want) but I would still argue for keeping the checks in for the .NET developer's sake. Is that ok with you or should I remove it wholesale?
} | ||
|
||
// For empty identifier, we return the object itself to support "dereferencing" JS object references. | ||
return property.parent[property.name]; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What does this mean "dereferncing" a JSObject reference. Can you provide a concrete example of where and how this is used?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Currently, when you have a JSObjectReference
in .NET that points to some JS object, there is no way how to directly get the state of that object. This special case is here to enable doing things like:
IJSObjectReference myObject = await JS.InvokeNewAsync(...);
// Do some work with the object like:
await myObject.InvokeAsync(...);
// Later
MyObjectModel data = await myObject.GetValue<MyObjectModel>();
As mentioned in my question #3 before I am not sure if to add that feature. It seems nice but I won't miss it if I remove it.
} | ||
} | ||
|
||
handleJSCall(targetInstanceId: number, identifier: string, callType: JSCallType, argsJson: string | null) { | ||
const args = parseJsonWithRevivers(this, argsJson) ?? []; | ||
const member = findObjectMember(identifier, targetInstanceId); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
findObjectMember
needs to receive the JSCallType
and create a function that can be cached and used to perform the expected invocation without requiring any additional check/validation.
We can keep JSCallType on the cache for matching purposes if we want to support being able to have the same "path" match multiple JSCallTypes but I don't think that's necessary.
For example, we don't need to support a developer calling a.b.c()
as a function and then also being able to call it as new a.b.c()
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am not sure how turning the access into a function is better but you're the boss.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The rest of the code becomes returnValue = func(args)
@@ -123,9 +138,11 @@ protected override void BeginInvokeJS(long asyncHandle, string identifier, strin | |||
} | |||
} | |||
|
|||
Log.BeginInvokeJS(_logger, asyncHandle, identifier); | |||
var invocationInfoJson = invocationInfo.ToJson(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We shouldn't serialize things to a big string here.
We should keep _clientProxy.SendAsync
(and the equivalent webassembly method) the same, unpacking the JSInvocationInfo and passing in the individual members.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So you want to keep passing individual arguments across the interop bridge? I.e. only add the new JSCallType
value to the already long list of arguments?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes, those are not part of the public API, so it's fine, we can change those whenever we want. We can't avoid serializing the arguments as a string, but we don't want to wrap that string into something else and serialize that too.
int resultType, | ||
string argsJson, | ||
[JSMarshalAs<JSType.Number>] long asyncHandle); | ||
public static partial string InvokeJSJson(string invocationInfoString); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We shouldn't stringify things here, we need to keep this API the same, including the additional info and passing down the individual members.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is the point of JSInvocationInfo
that you made me implement then?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The point is that whenever we need to add things to the public API we don't need to keep adding overloads to JSRuntime and other public classes.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I see, that makes sense. However, I thought it was also to not having to rewrite the three different transport channels whenever we add some parameter to the interop (which based on my recent experience is the part that takes the most work to do).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I haven't finished looking at it but looks like a good starting point. However, there are several things we need to cleanup/polish before this is ready to go. In general terms:
- The implementation needs to remain as allocation free as possible:
- JSInvocationInfo needs to be a struct.
- JSInvocationInfo is for the public API only.
- It should not be JSON serialized, internal APIs need to continue unpacking the parameters and passing them down to the JS side. Otherwise, we pay the cost of:
- One large string allocation for JSInvocationInfo (in addition to the one for the parameters which we can't avoid).
- The cost of JSON serialization (which is not negligible in these cases, even with SG).
- Double encoding of the arguments, since they are already serialized.
- It should not be JSON serialized, internal APIs need to continue unpacking the parameters and passing them down to the JS side. Otherwise, we pay the cost of:
/// <param name="identifier">An identifier for the constructor function to invoke. For example, the value <c>"someScope.SomeClass"</c> will invoke the constructor <c>someScope.SomeClass</c> on the target instance.</param> | ||
/// <param name="args">JSON-serializable arguments.</param> | ||
/// <returns>An <see cref="IJSObjectReference"/> instance that represents the created JS object.</returns> | ||
ValueTask<IJSObjectReference> InvokeNewAsync(string identifier, object?[]? args); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Can you provide an example for the case where a IJSObjectReference would invoke a member property using new
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For example when you load a module as IJSObjectReference
.
Also simply for consistency sake. There is no rule in JS that you can't have constructor functions in an object.
If you want it remove to save code, sure.
What level do you mean by public API? The user-facing API will not use it. The internal communication methods will not use it. It seems useless to pack arguments into some struct just to destructure them a few lines of executed code later. |
/// </summary> | ||
/// <param name="invocationInfo">Configuration of the interop call.</param> | ||
/// <returns>A JSON representation of the result.</returns> | ||
protected abstract string? InvokeJS(JSInvocationInfo invocationInfo); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This needs to be (JSInvocationInfo info, string argsJson)
]
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ok. It should be said that if we won't be serializing the JSInvocationInfo
and sending it to the JS side then it does not matter if we pass argsJson
at this level separately.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Fair enough
This is in JSRuntimeBase and its public API
|
Description
Adds three new methods to the JS interop API and refactors existing interop communication to pass arguments using a
JsInvocationInfo
object.The API design is based on conclusions in #31151 (comment).
Changes
ValueTask<IJSObjectReference> InvokeNewAsync(string identifier, object?[]? args)
ValueTask<TValue> GetValueAsync<TValue>(string identifier)
ValueTask SetValueAsync<TValue>(string identifier, TValue value)
IJSRuntime
andIJSObjectReference
and their implementations with the same scoping behavior as the existingInvokeAsync
method.CancellationToken
argument,TimeSpan
timeout argument, orparams object?[]? args
argument are also added to be consistent with the overloads ofInvokeAsync
.IJSObjectReference.GetValueAsync<TValue>()
that does not take and identifier and simply returns the value of the object (TValue
is here meant to be a model of the object's properties that the caller is interested in).IJSInProcessObjectReference InvokeNew(string identifier, params object?[]? args)
TValue GetValue<TValue>(string identifier)
void SetValue<TValue>(string identifier, TValue value)
IJSInProcessRuntime
andIJSInProcessObjectReference
and their implementations with the same scoping behavior as the existingInvoke
method.BeginInvokeJS
andInvokeJS
on the .NET side) that previously passed interop call configuration as separate arguments. The configuration is now packaged into aJSInvocationInfo
object which is serialized as JSON string (using a source generated JsonSerializer). Motivation for this change was to make the interop communication across the various supported channels better maintainable and extensible going forward.Microsoft.JSInterop.JS
library to support the new interop methods, including reading and modifying object properties and invoking constructor functions with thenew
keyword.Microsoft.JSInterop.JS
library.InteropTest
inMicrosoft.AspNetCore.Components.E2ETests
.DotNetToJSInterop.razor
component toBasicTestApp
intended for manual testing and debugging of the JS interop functionality on different platforms.