diff --git a/CHANGELOG.md b/CHANGELOG.md
index fbe52a8e2a..0d88858870 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,6 +1,7 @@
## vNext (TBD)
### Enhancements
+* Added the `App.EmailPasswordAuth.RetryCustomConfirmationAsync` method to be able to run again the confirmation function on the server for a given email. (Issue [#3463](https://github.com/realm/realm-dotnet/issues/3463))
* Added `User.Changed` event that can be used to notify subscribers that something about the user changed - typically this would be the user state or the access token. (Issue [#3429](https://github.com/realm/realm-dotnet/issues/3429))
* Added support for customizing the ignore attribute applied on certain generated properties of Realm models. The configuration option is called `realm.custom_ignore_attribute` and can be set in a global configuration file (more information about global configuration files can be found in the [.NET documentation](https://learn.microsoft.com/en-us/dotnet/fundamentals/code-analysis/configuration-files)). The Realm generator will treat this as an opaque string, that will be appended to the `IgnoreDataMember` and `XmlIgnore` attributes already applied on these members. The attributes must be fully qualified unless the namespace they reside in is added to a global usings file. For example, this is how you would add `JsonIgnore` from `System.Text.Json`:
diff --git a/Realm/Realm/Handles/AppHandle.EmailPassword.cs b/Realm/Realm/Handles/AppHandle.EmailPassword.cs
index 9917f77e86..cd23cbf375 100644
--- a/Realm/Realm/Handles/AppHandle.EmailPassword.cs
+++ b/Realm/Realm/Handles/AppHandle.EmailPassword.cs
@@ -43,6 +43,11 @@ public static extern void resend_confirmation_email(AppHandle app,
[MarshalAs(UnmanagedType.LPWStr)] string email, IntPtr email_len,
IntPtr tcs_ptr, out NativeException ex);
+ [DllImport(InteropConfig.DLL_NAME, EntryPoint = "shared_app_email_retry_custom_confirmation", CallingConvention = CallingConvention.Cdecl)]
+ public static extern void retry_custom_comfirmation(AppHandle app,
+ [MarshalAs(UnmanagedType.LPWStr)] string email, IntPtr email_len,
+ IntPtr tcs_ptr, out NativeException ex);
+
[DllImport(InteropConfig.DLL_NAME, EntryPoint = "shared_app_email_send_reset_password_email", CallingConvention = CallingConvention.Cdecl)]
public static extern void send_reset_password_email(AppHandle app,
[MarshalAs(UnmanagedType.LPWStr)] string email, IntPtr email_len,
@@ -125,6 +130,23 @@ public async Task ResendConfirmationEmailAsync(string email)
}
}
+ public async Task RetryCustomConfirmationAsync(string email)
+ {
+ var tcs = new TaskCompletionSource();
+ var tcsHandle = GCHandle.Alloc(tcs);
+
+ try
+ {
+ EmailNativeMethods.retry_custom_comfirmation(_appHandle, email, (IntPtr)email.Length, GCHandle.ToIntPtr(tcsHandle), out var ex);
+ ex.ThrowIfNecessary();
+ await tcs.Task;
+ }
+ finally
+ {
+ tcsHandle.Free();
+ }
+ }
+
public async Task SendResetPasswordEmailAsync(string username)
{
var tcs = new TaskCompletionSource();
diff --git a/Realm/Realm/Sync/App.cs b/Realm/Realm/Sync/App.cs
index 53abd96088..febfbb525b 100644
--- a/Realm/Realm/Sync/App.cs
+++ b/Realm/Realm/Sync/App.cs
@@ -408,6 +408,21 @@ public Task ResendConfirmationEmailAsync(string email)
return _app.Handle.EmailPassword.ResendConfirmationEmailAsync(email);
}
+ ///
+ /// Rerun the custom confirmation function for the given mail.
+ ///
+ /// The email of the user.
+ ///
+ /// An awaitable representing the asynchronous request to the server that the custom confirmation function is run again. Successful
+ /// completion indicates that the user has been confirmed on the server.
+ ///
+ public Task RetryCustomConfirmationAsync(string email)
+ {
+ Argument.NotNullOrEmpty(email, nameof(email));
+
+ return _app.Handle.EmailPassword.RetryCustomConfirmationAsync(email);
+ }
+
///
/// Sends a password reset email to the specified address.
///
diff --git a/Tests/Realm.Tests/Sync/SyncTestHelpers.cs b/Tests/Realm.Tests/Sync/SyncTestHelpers.cs
index 43b2890311..fae2c57766 100644
--- a/Tests/Realm.Tests/Sync/SyncTestHelpers.cs
+++ b/Tests/Realm.Tests/Sync/SyncTestHelpers.cs
@@ -92,6 +92,8 @@ public static void RunBaasTestAsync(Func testFunc, int timeout = 30000)
public static string GetVerifiedUsername() => $"realm_tests_do_autoverify-{Guid.NewGuid()}";
+ public static string GetUnconfirmedUsername() => $"realm_tests_do_not_confirm-{Guid.NewGuid()}@g.it";
+
public static async Task TriggerClientResetOnServer(SyncConfigurationBase config)
{
var userId = config.User.Id;
diff --git a/Tests/Realm.Tests/Sync/UserManagementTests.cs b/Tests/Realm.Tests/Sync/UserManagementTests.cs
index 401e9d3a67..9ca3b86ee0 100644
--- a/Tests/Realm.Tests/Sync/UserManagementTests.cs
+++ b/Tests/Realm.Tests/Sync/UserManagementTests.cs
@@ -400,6 +400,42 @@ public void User_LinkCredentials_WhenInUse_Throws()
});
}
+ [Test]
+ public void User_RetryCustomConfirmationAsync_WorksInAllScenarios()
+ {
+ SyncTestHelpers.RunBaasTestAsync(async () =>
+ {
+ // Standard case
+ var unconfirmedMail = SyncTestHelpers.GetUnconfirmedUsername();
+ var credentials = Credentials.EmailPassword(unconfirmedMail, SyncTestHelpers.DefaultPassword);
+
+ // The first time the confirmation function is called we return "pending", so the user needs to be confirmed.
+ // At the same time we save the user email in a collection.
+ await DefaultApp.EmailPasswordAuth.RegisterUserAsync(unconfirmedMail, SyncTestHelpers.DefaultPassword).Timeout(10_000, detail: "Failed to register user");
+
+ var ex3 = await TestHelpers.AssertThrows(() => DefaultApp.LogInAsync(credentials));
+ Assert.That(ex3.Message, Does.Contain("confirmation required"));
+
+ // The second time we call the confirmation function we find the email we saved in the collection and return "success", so the user
+ // gets confirmed and can log in.
+ await DefaultApp.EmailPasswordAuth.RetryCustomConfirmationAsync(unconfirmedMail);
+ var user = await DefaultApp.LogInAsync(credentials);
+ Assert.That(user.State, Is.EqualTo(UserState.LoggedIn));
+
+ // Logged in user case
+ var loggedInUser = await GetUserAsync();
+ var ex = await TestHelpers.AssertThrows(() => DefaultApp.EmailPasswordAuth.RetryCustomConfirmationAsync(loggedInUser.Profile.Email!));
+ Assert.That(ex.StatusCode, Is.EqualTo(HttpStatusCode.BadRequest));
+ Assert.That(ex.Message, Does.Contain("already confirmed"));
+
+ // Unknown user case
+ var invalidEmail = "test@gmail.com";
+ var ex2 = await TestHelpers.AssertThrows(() => DefaultApp.EmailPasswordAuth.RetryCustomConfirmationAsync(invalidEmail));
+ Assert.That(ex2.StatusCode, Is.EqualTo(HttpStatusCode.NotFound));
+ Assert.That(ex2.Message, Does.Contain("user not found"));
+ });
+ }
+
[Test]
public void User_JWT_LogsInAndReadsDataFromToken()
{
diff --git a/Tools/DeployApps/BaasClient.cs b/Tools/DeployApps/BaasClient.cs
index fa3ece4eee..dc2998369c 100644
--- a/Tools/DeployApps/BaasClient.cs
+++ b/Tools/DeployApps/BaasClient.cs
@@ -88,12 +88,33 @@ public class FunctionReturn
}
private const string ConfirmFuncSource =
- @"exports = ({ token, tokenId, username }) => {
+ @"exports = async function ({ token, tokenId, username }) {
// process the confirm token, tokenId and username
if (username.includes(""realm_tests_do_autoverify"")) {
- return { status: 'success' }
+ return { status: 'success' };
}
- // do not confirm the user
+
+ if (username.includes(""realm_tests_do_not_confirm"")) {
+ const mongodb = context.services.get('BackingDB');
+ let collection = mongodb.db('test_db').collection('not_confirmed');
+ let result = await collection.findOne({'email': username});
+
+ if(result === null)
+ {
+ let newVal = {
+ 'email': username,
+ 'token': token,
+ 'tokenId': tokenId,
+ }
+
+ await collection.insertOne(newVal);
+ return { status: 'pending' };
+ }
+
+ return { status: 'success' };
+ }
+
+ // fail the user confirmation
return { status: 'fail' };
};";
diff --git a/wrappers/src/app_cs.cpp b/wrappers/src/app_cs.cpp
index 8539af5a49..a0e0abeb18 100644
--- a/wrappers/src/app_cs.cpp
+++ b/wrappers/src/app_cs.cpp
@@ -368,6 +368,14 @@ extern "C" {
});
}
+ REALM_EXPORT void shared_app_email_retry_custom_confirmation(SharedApp& app, uint16_t* email_buf, size_t email_len, void* tcs_ptr, NativeException::Marshallable& ex)
+ {
+ handle_errors(ex, [&]() {
+ Utf16StringAccessor email(email_buf, email_len);
+ app->provider_client().retry_custom_confirmation(email, get_callback_handler(tcs_ptr));
+ });
+ }
+
REALM_EXPORT void shared_app_email_send_reset_password_email(SharedApp& app, uint16_t* email_buf, size_t email_len, void* tcs_ptr, NativeException::Marshallable& ex)
{
handle_errors(ex, [&]() {