-
Notifications
You must be signed in to change notification settings - Fork 89
docs: Add migration guide for moving off legacy serverpod_auth #585
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
Merged
Merged
Changes from all commits
Commits
Show all changes
4 commits
Select commit
Hold shift + click to select a range
3370c59
docs: Add migration guide for moving off legacy serverpod_auth
Zfinix 8090360
docs: Fix MD025 and tighten migration guide prose
Zfinix 96d674c
docs: Update migration guide for legacy auth to include new package d…
Zfinix 67e98dd
docs: Update links in migration guide for legacy auth to reflect new …
Zfinix File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,299 @@ | ||
| --- | ||
| description: Move a Serverpod 3.5 project off serverpod_auth_server onto the modular auth stack with email, Google, and Flutter session continuity. | ||
| sidebar_label: Migrate from legacy auth | ||
| --- | ||
|
|
||
| # Migrate from legacy serverpod_auth | ||
|
|
||
| This guide is for apps still running `serverpod_auth_server` on Serverpod 3.4 or earlier 3.5 betas. At the end, existing users sign in through the new modular auth stack with their old passwords and old sessions, and your legacy endpoints keep working until every client has rolled forward. Plan for about an hour, plus migration runtime. | ||
|
|
||
| ## Before you start | ||
|
|
||
| - A Serverpod 3.5.x project. If you are still on 3.4 or earlier, follow [Upgrade to 3.5](./upgrade-to-three-five) first. | ||
| - Dart SDK 3.8.0 or later. | ||
| - Flutter SDK 3.32.0 or later (only if you are migrating the Flutter app). | ||
| - Postgres 14 or later, or SQLite3. | ||
| - The four auth packages at `3.5.0-beta.9` (or the matching beta on pub.dev): `serverpod_auth_core`, `serverpod_auth_idp`, `serverpod_auth_bridge`, and `serverpod_auth_migration`. These are still beta and may receive breaking changes before 3.5 stable. | ||
| - Back up your production database. | ||
| - Commit your current state on a clean branch. | ||
| - Restore a copy of production data into a staging environment and rehearse this guide against it before running it for real. | ||
|
|
||
| ## Add the new auth packages | ||
|
|
||
| Add the new packages to each of your server, client, and Flutter `pubspec.yaml` files. Keep the legacy `serverpod_auth_*` packages installed; you will remove them after the migration completes. | ||
|
|
||
| In `<project>_server/pubspec.yaml`: | ||
|
|
||
| ```yaml | ||
| dependencies: | ||
| serverpod: 3.5.0-beta.9 | ||
| serverpod_auth_server: 3.5.0-beta.9 # legacy, keep during migration | ||
| serverpod_auth_core_server: 3.5.0-beta.9 | ||
| serverpod_auth_idp_server: 3.5.0-beta.9 | ||
| serverpod_auth_bridge_server: 3.5.0-beta.9 | ||
| serverpod_auth_migration_server: 3.5.0-beta.9 | ||
| ``` | ||
|
|
||
| In `<project>_client/pubspec.yaml`: | ||
|
|
||
| ```yaml | ||
| dependencies: | ||
| serverpod_client: 3.5.0-beta.9 | ||
| serverpod_auth_core_client: 3.5.0-beta.9 | ||
| serverpod_auth_idp_client: 3.5.0-beta.9 | ||
| serverpod_auth_bridge_client: 3.5.0-beta.9 | ||
| ``` | ||
|
|
||
| In `<project>_flutter/pubspec.yaml`: | ||
|
|
||
| ```yaml | ||
| dependencies: | ||
| serverpod_flutter: 3.5.0-beta.9 | ||
| serverpod_auth_core_flutter: 3.5.0-beta.9 | ||
| serverpod_auth_idp_flutter: 3.5.0-beta.9 | ||
| serverpod_auth_bridge_flutter: 3.5.0-beta.9 | ||
| ``` | ||
|
|
||
| `serverpod_auth_bridge_client` and `serverpod_auth_bridge_flutter` are required for the session import covered later under [Update the Flutter app](#update-the-flutter-app). | ||
|
|
||
| From each package directory, run: | ||
|
|
||
| ```bash | ||
| dart pub upgrade | ||
| serverpod generate | ||
| ``` | ||
|
|
||
| The project now builds with the new packages installed alongside the legacy ones. | ||
|
|
||
| ## Configure the server | ||
|
|
||
| In `<project>_server/lib/server.dart`, call `pod.initializeAuthServices` before `pod.start` with the modular identity providers, register `LegacySessionTokenManager` so existing tokens keep validating, and enable legacy client forwarding so old client builds keep working. | ||
|
|
||
| `serverpod_auth_bridge_server` exports its own `Endpoints` and `Protocol` classes from its generated code, which clash with your project's. Use a `show` clause to bring only the symbols you need into scope: | ||
|
|
||
| ```dart | ||
| import 'package:serverpod/serverpod.dart'; | ||
| import 'package:serverpod_auth_bridge_server/serverpod_auth_bridge_server.dart' | ||
| show LegacySessionTokenManager, LegacyClientSupport; | ||
| import 'package:serverpod_auth_idp_server/core.dart'; | ||
| import 'package:serverpod_auth_idp_server/providers/email.dart'; | ||
| import 'package:serverpod_auth_idp_server/providers/google.dart'; | ||
|
|
||
| import 'src/generated/protocol.dart'; | ||
| import 'src/generated/endpoints.dart'; | ||
|
|
||
| void run(List<String> args) async { | ||
| final pod = Serverpod( | ||
| args, | ||
| Protocol(), | ||
| Endpoints(), | ||
| ); | ||
|
|
||
| pod.initializeAuthServices( | ||
| tokenManagerBuilders: [ | ||
| JwtConfigFromPasswords(), | ||
| ServerSideSessionsConfigFromPasswords(), | ||
| const LegacySessionTokenManager(), | ||
| ], | ||
| identityProviderBuilders: [ | ||
| EmailIdpConfigFromPasswords(), | ||
| GoogleIdpConfigFromPasswords(), | ||
| ], | ||
| ); | ||
|
|
||
| pod.enableLegacyClientSupport(); | ||
|
|
||
| await pod.start(); | ||
| } | ||
| ``` | ||
|
|
||
| Put the password-importing email endpoint in its own file under `<project>_server/lib/src/endpoints/`. Importing the full bridge package alongside the project's `Endpoints` and `Protocol` from the same file confuses the `serverpod generate` endpoint scanner, so keep it isolated: | ||
|
|
||
| ```dart | ||
| // lib/src/endpoints/password_importing_email_idp_endpoint.dart | ||
| import 'package:serverpod/serverpod.dart'; | ||
| import 'package:serverpod_auth_bridge_server/serverpod_auth_bridge_server.dart'; | ||
| import 'package:serverpod_auth_idp_server/core.dart'; | ||
| import 'package:serverpod_auth_idp_server/providers/email.dart'; | ||
|
|
||
| class PasswordImportingEmailIdpEndpoint extends EmailIdpBaseEndpoint { | ||
| @override | ||
| Future<AuthSuccess> login( | ||
| Session session, { | ||
| required String email, | ||
| required String password, | ||
| }) async { | ||
| await AuthBackwardsCompatibility.importLegacyPasswordIfNeeded( | ||
| session, | ||
| email: email, | ||
| password: password, | ||
| ); | ||
| return super.login(session, email: email, password: password); | ||
| } | ||
| } | ||
| ``` | ||
|
|
||
| Add the required entries to `<project>_server/config/passwords.yaml` under the `development:` section (use distinct values per environment): | ||
|
|
||
| ```yaml | ||
| development: | ||
| # ... existing keys (database, redis, serviceSecret) ... | ||
| jwtRefreshTokenHashPepper: 'your-jwt-refresh-pepper' | ||
| jwtHmacSha512PrivateKey: 'your-hmac-private-key-at-least-64-bytes' | ||
| serverSideSessionKeyHashPepper: 'your-session-pepper' | ||
| emailSecretHashPepper: 'your-email-pepper' | ||
| ``` | ||
|
|
||
| See [Storing secrets](../concepts/authentication/setup#storing-secrets) for production handling and additional provider-specific secrets. | ||
|
|
||
| The server now starts with both legacy and modular endpoints mounted side by side, so existing legacy clients still work. | ||
|
|
||
| ## Run the migration | ||
|
|
||
| Create and apply the schema migrations for the new modular tables: | ||
|
|
||
| ```bash | ||
| serverpod create-migration --tag modular-auth | ||
| dart run bin/main.dart --apply-migrations | ||
| ``` | ||
|
|
||
| Then run the user migration once. The example below migrates every legacy user, but you can pass `maxUsers` to process in batches if your dataset is large. The `userMigration` callback fires once per migrated user so you can remap your own foreign keys from the legacy `int` user ID to the new `UuidValue` auth user ID inside the same transaction. | ||
|
|
||
| Put this helper in its own file so its imports do not collide with `server.dart`. `serverpod_auth_migration_server` exports its own `Endpoints` and `Protocol` classes, so use a `hide` clause when adding it elsewhere: | ||
|
|
||
| ```dart | ||
| // lib/src/auth_migration.dart | ||
| import 'package:serverpod/serverpod.dart'; | ||
| import 'package:serverpod_auth_idp_server/core.dart'; | ||
| import 'package:serverpod_auth_migration_server/serverpod_auth_migration_server.dart' | ||
| hide Endpoints, Protocol; | ||
|
|
||
| Future<void> runMigration(Serverpod pod) async { | ||
| AuthMigrations.config = AuthMigrationConfig( | ||
| emailIdp: AuthServices.instance.emailIdp, | ||
| ); | ||
|
|
||
| final session = await pod.createSession(); | ||
| try { | ||
| final migrated = await AuthMigrations.migrateUsers( | ||
| session, | ||
| userMigration: ( | ||
| session, { | ||
| required int oldUserId, | ||
| required UuidValue newAuthUserId, | ||
| Transaction? transaction, | ||
| }) async { | ||
| // Remap your own tables from oldUserId to newAuthUserId using transaction. | ||
| }, | ||
| ); | ||
| session.log('Migrated $migrated users.', level: LogLevel.info); | ||
| } finally { | ||
| await session.close(); | ||
| } | ||
| } | ||
| ``` | ||
|
|
||
| Run this as a one-off from a dedicated entry point, not on a request path. The simplest pattern is a script under `<project>_server/bin/migrate.dart` that imports the helper above and invokes it once on a `Serverpod` instance started with `--role maintenance`. Do not call `runMigration` from inside an endpoint or future call. | ||
|
|
||
| The migration is idempotent: re-running `migrateUsers` skips users that already have a row in `serverpod_auth_migration_migrated_user`. The returned count is "users selected this run," not "new users created." | ||
|
|
||
| ## Wire up sign-in for migrated users | ||
|
|
||
| For email accounts, the `PasswordImportingEmailIdpEndpoint` subclass from the server step calls `AuthBackwardsCompatibility.importLegacyPasswordIfNeeded` before delegating to the base implementation. The bridge upgrades the password hash on first login and removes the `LegacyEmailPassword` row. | ||
|
|
||
| For Google accounts, `AuthMigrations.migrateUsers` seeded the `serverpod_auth_bridge_external_user_id` table with each legacy user's stored identifier (a Google `sub` for newer rows, or an email address for older rows). Subclass `GoogleIdpBaseEndpoint` and call `AuthBackwardsCompatibility.importGoogleAccount` before the base login. The bridge looks up the legacy identifier by Google `sub` first, falls back to a case-insensitive email match, links the Google account to the existing `AuthUser`, and removes the bridge mapping. | ||
|
|
||
| Put this in its own file under `<project>_server/lib/src/endpoints/`, alongside the email endpoint: | ||
|
|
||
| ```dart | ||
| // lib/src/endpoints/google_linking_idp_endpoint.dart | ||
| import 'package:serverpod/serverpod.dart'; | ||
| import 'package:serverpod_auth_bridge_server/serverpod_auth_bridge_server.dart'; | ||
| import 'package:serverpod_auth_idp_server/core.dart'; | ||
| import 'package:serverpod_auth_idp_server/providers/google.dart'; | ||
|
|
||
| class GoogleLinkingIdpEndpoint extends GoogleIdpBaseEndpoint { | ||
| @override | ||
| Future<AuthSuccess> login( | ||
| Session session, { | ||
| required String idToken, | ||
| required String? accessToken, | ||
| }) async { | ||
| await AuthBackwardsCompatibility.importGoogleAccount( | ||
| session, | ||
| idToken: idToken, | ||
| accessToken: accessToken, | ||
| ); | ||
| return super.login(session, idToken: idToken, accessToken: accessToken); | ||
| } | ||
| } | ||
| ``` | ||
|
|
||
| A migrated user can now sign in with their old password or Google account and lands in their existing data. | ||
|
|
||
| ## Update the Flutter app | ||
|
|
||
| In `<project>_flutter/lib/main.dart`, swap the auth setup to use `FlutterAuthSessionManager` and call `initAndImportLegacySessionIfNeeded` before any sign-in UI renders. This exchanges any old auth key stored on the device for a new modular session so existing installs do not have to sign in again. | ||
|
|
||
| ```dart | ||
| import 'package:serverpod_auth_bridge_flutter/serverpod_auth_bridge_flutter.dart'; | ||
| import 'package:serverpod_auth_core_flutter/serverpod_auth_core_flutter.dart'; | ||
| import 'package:your_client/your_client.dart'; | ||
|
|
||
| late Client client; | ||
|
|
||
| Future<void> main() async { | ||
| WidgetsFlutterBinding.ensureInitialized(); | ||
|
|
||
| client = Client('https://api.example.com/') | ||
| ..authSessionManager = FlutterAuthSessionManager(); | ||
|
|
||
| await client.authSessionManager.initAndImportLegacySessionIfNeeded( | ||
| client.modules.serverpod_auth_bridge, | ||
| ); | ||
|
|
||
| runApp(const MyApp()); | ||
| } | ||
| ``` | ||
|
|
||
| This requires `serverpod_auth_bridge_client` and `serverpod_auth_bridge_flutter` in `<project>_flutter/pubspec.yaml` from [Add the new auth packages](#add-the-new-auth-packages). | ||
|
|
||
| Flutter clients that used the default legacy session storage carry the legacy session forward on first launch after the upgrade. If your project customized session storage, pass a `legacyStringGetter` to `initAndImportLegacySessionIfNeeded` that reads from your custom location. | ||
|
|
||
| ## Verify and clean up | ||
|
|
||
| ### Verify | ||
|
|
||
| - Query `SELECT count(*) FROM serverpod_auth_migration_migrated_user` and confirm it matches your legacy user count. | ||
| - Sign in with a known migrated account through the new IdP and confirm the `LegacyEmailPassword` row for that user is gone afterwards. | ||
| - Monitor `serverpod_auth_bridge_external_user_id` and the bridge's legacy-session table over the following days. The row counts should decrease as users sign in through the new stack. | ||
|
|
||
| ### While clients catch up | ||
|
|
||
| The legacy database tables stay untouched, so a rollback to the legacy stack still works while old client builds are still in the wild. `LegacySessionTokenManager` validates legacy session tokens against the bridge's stored sessions, and `pod.enableLegacyClientSupport()` keeps the legacy routes mounted so old apps continue to hit `serverpod_auth.*` endpoints. | ||
|
|
||
| ### When you are ready to remove legacy | ||
|
|
||
| Once the bridge tables are empty (or close enough that you accept the long tail), drop the legacy packages and the migration dependency. Remove `serverpod_auth_server`, `serverpod_auth_migration_server`, and `serverpod_auth_bridge_server` from `pubspec.yaml`, remove the `LegacySessionTokenManager` and `enableLegacyClientSupport` calls from `server.dart`, then drop the legacy database columns that still reference integer `userInfoId`. | ||
|
|
||
| ## Troubleshooting | ||
|
|
||
| | Symptom | Likely cause | What to do | | ||
| | --- | --- | --- | | ||
| | `migrateUsers` throws or rolls back | Migrations not applied, or the wrong `emailIdp` instance on `AuthMigrationConfig` | Apply all module migrations; set `AuthMigrations.config = AuthMigrationConfig(emailIdp: AuthServices.instance.emailIdp)` after `pod.initializeAuthServices`. | | ||
| | Migrated email user cannot log in with their old password | `importLegacyPasswordIfNeeded` is not on the login path | Confirm your email endpoint subclasses `EmailIdpBaseEndpoint` and calls `AuthBackwardsCompatibility.importLegacyPasswordIfNeeded` before `super.login`. | | ||
| | Flutter app prompts the user to sign in again after upgrade | Bridge client is not on the classpath, or the wrong module caller was passed | Add `serverpod_auth_bridge_client` to the client package; pass `client.modules.serverpod_auth_bridge` into `initAndImportLegacySessionIfNeeded`. | | ||
| | Duplicate Google `AuthUser` created on first sign-in | The legacy external user ID was never stored, or `importGoogleAccount` is not on the Google login path | Re-run `AuthMigrations.migrateUsers` for affected users (it backfills `serverpod_auth_bridge_external_user_id`); ensure the Google sign-in path calls `AuthBackwardsCompatibility.importGoogleAccount` first. | | ||
| | Authentication handler rejects modular tokens | `pod.initializeAuthServices` was not called, or token managers list is missing | Call `pod.initializeAuthServices(...)` before `pod.start()`; include `JwtConfigFromPasswords` (or `ServerSideSessionsConfigFromPasswords`) plus `LegacySessionTokenManager` in `tokenManagerBuilders`. | | ||
| | `serverpod generate` fails with "Endpoint analysis skipped due to invalid Dart syntax" or "The function 'Protocol' isn't defined" | The bridge or migration package exports its own `Endpoints` and `Protocol` classes that clash with your project's | Import the bridge with a `show` clause (`show LegacySessionTokenManager, LegacyClientSupport`) in `server.dart`, move the email endpoint subclass into its own file under `lib/src/endpoints/`, and use `hide Endpoints, Protocol` when importing the migration package in a helper file. | | ||
| | `PasswordNotFoundException: jwtRefreshTokenHashPepper was not found` on startup | `JwtConfigFromPasswords` requires several peppers and keys in `passwords.yaml` | Add `jwtRefreshTokenHashPepper`, `jwtHmacSha512PrivateKey`, `serverSideSessionKeyHashPepper`, and `emailSecretHashPepper` to each environment section. | | ||
|
|
||
| ## Still stuck? | ||
|
|
||
| Reach out on the [community page](../support). | ||
|
|
||
| ## Related | ||
|
|
||
| - [Upgrade to 3.5](./upgrade-to-three-five): do this first. | ||
| - [Authentication setup](../concepts/authentication/setup): modular configuration reference. | ||
| - [Database migrations](../concepts/database/migrations): creating and applying schema changes safely. | ||
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
Uh oh!
There was an error while loading. Please reload this page.