I’d like to add a user invitation flow to an application currently using AshAuthentication with the password strategy.
The flow would work something like:
An existing Admin user will create the initial User resource
AshAuthentication generates a token with ~48h expiry
Invitation email is sent to new user including token in a URL
User clicks and lands on a page allowing them to set their password
Once password set, user is authenticated and redirected to a page in the app
Is this something I can achieve with the password strategy? It’s fairly similar to triggering a password-reset, but has a few differences in the email content, token expiry and UI.
I’d suggest looking at the way that the password strategy handles the reset flow - it should be relatively simple to duplicate and modify it to support invites.
My only question is whether this should be a different strategy entirely - ie separate from the password strategy. It seems to me that you may want to invite users to sign up with other strategies also. My gut feeling is that you can use the token resource to store invites (you can store arbitrary data in the extra_data field) that way you can rely on the existing expiration and expunge logic. Perhaps there should also be a setting that disables registration without the invite token?
My only question is whether this should be a different strategy entirely - ie separate from the password strategy. It seems to me that you may want to invite users to sign up with other strategies also.
Yes, invitation is more of a registration concern than an authentication strategy.
Perhaps there should also be a setting that disables registration without the invite token?
That would be ideal
Riffing on the API for resettable something along the lines of:
strategies do
password :password do
identity_field :email
registration_enabled? false # Disable self-service registration, or perhaps `:invitation_only` here?
invitable do
accept_invitation_action_name :create_from_invite
generate_invitation_action_name :generate_invitation
generate_invitation_accept [
:email, :first_name, :last_name, :role # params accepted for invitation, stored with invite token
]
token_lifetime 48 # longer lived token
sender fn user_params, token, _opts ->
MyApp.InvitationEmail.send(
recipients: user_params,
token: token
)
end
end
resettable do
token_lifetime 1 # shorter lifetime on password reset token
sender fn user, token, _opts -> # accepts the full user resource
MyApp.ResetPasswordEmail.send(
recipients: user,
token: token
)
end
end
end
end
What about non-password strategies? ie allowing someone to accept an invite and then sign in with github for example.
Could a new top-level Section be added to the DSL?
invitation do
accept_invitation_action_name :create_from_invite
generate_invitation_action_name :generate_invitation
generate_invitation_accept [
:email, :first_name, :last_name, :role
]
token_lifetime 48
sender fn user_params, token, _opts ->
MyApp.InvitationEmail.send(
recipients: user_params,
token: token
)
end
end
authentication do
strategies do
password do
# ...
end
google do
# ...
end
github do
# ...
end
end
end
Generating and sending an invitation would store a Token record, then when the user lands on the /accept-invitation route, they would be presented with options to ‘Continue with Password’, or ‘Continue with Google’, ‘Continue with Github’, etc based on the available auth strategies.
From the users choice of auth strategy (and possibly some OAuth2/OIDC flow) the corresponding register_with_<strategy> action is called.
Email links to a custom UI presenting a “Set Your Password” message in place of the usual “Reset Password”
I had to work around having multiple password reset routes in the same scope by inlining the reset_route macro:
scope "/", MyAppWeb do
pipe_through([:browser, :browser_ash_authentication])
reset_route(
live_view: MyAppWeb.PasswordResetLive,
overrides: [MyAppWeb.AuthOverrides, AshAuthentication.Phoenix.Overrides.Default]
)
# Can't have multiple reset_route - inlining macro here to approximate it.
scope "/accept-invitation", alias: false do
live_session :accept_invitation,
session: %{
"overrides" => [MyAppWeb.AuthOverrides, AshAuthentication.Phoenix.Overrides.Default],
"otp_app" => nil
} do
live("/:token", MyAppWeb.AcceptInviteLive, :accept_invitation, as: :auth)
end
end
... more routes
end
The custom UI eventually calls the reset action with: