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

feat: webauthn-1 #1078

Merged
merged 107 commits into from
Feb 28, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
107 commits
Select commit Hold shift + click to select a range
7dae379
feat: new dependency: webauthn4j
tamassoltesz Dec 5, 2024
b338903
feat: add tables for webauthn
tamassoltesz Dec 5, 2024
6e2bb33
fix: typo fixes
tamassoltesz Dec 5, 2024
4dcf4ae
feat: webauthn options
tamassoltesz Dec 17, 2024
b98d991
feat: registercredentials wip
tamassoltesz Jan 14, 2025
342a9b0
feat: passkeys register credentials wip
tamassoltesz Jan 14, 2025
83d325a
feat: recipe user sign up
tamassoltesz Jan 15, 2025
2a2cb7f
recipe user creation wip
tamassoltesz Jan 15, 2025
d76e24a
sign up recipe user
tamassoltesz Jan 15, 2025
38d82ea
feat: register credentials
tamassoltesz Jan 16, 2025
5a030c1
fix: temp
sattvikc Jan 21, 2025
f189cb9
feat: webauthn support wip
tamassoltesz Jan 27, 2025
83f3edc
feat: webauthn support wip
tamassoltesz Jan 28, 2025
a514928
merging
tamassoltesz Jan 28, 2025
a81c448
feat: webauthn support wip
tamassoltesz Jan 28, 2025
525066f
feat: getuserinfolist draft
tamassoltesz Jan 28, 2025
8c3fc92
feat: get user by account info - webauthn support
tamassoltesz Jan 29, 2025
c2a54c1
Merge branch 'feat/webauthn-1' into feat/webauthn-account-recovery
sattvikc Jan 29, 2025
4de299c
fix: generate account recovery token api
sattvikc Jan 29, 2025
599ff94
feat: get user by account info - webauthn support
tamassoltesz Jan 29, 2025
1f597a2
feat: signup with credentialsregister
tamassoltesz Jan 29, 2025
5088fc6
fix: fixes for tests
tamassoltesz Jan 30, 2025
d0e54b6
fix: fixes for tests
tamassoltesz Jan 30, 2025
508e898
feat: get generated options api
tamassoltesz Jan 30, 2025
0c682d8
feat: webauthn sign in
tamassoltesz Jan 31, 2025
5e0e3f7
fix: account recovery
sattvikc Jan 31, 2025
3537f3c
fix: fixes for tests
tamassoltesz Jan 31, 2025
3fb318e
fix: fixing id name in response
tamassoltesz Jan 31, 2025
1b17359
fix: fixing id encoding in response
tamassoltesz Jan 31, 2025
bf78593
fix: base64 url encode the challenge insted of base64 encode
tamassoltesz Jan 31, 2025
c83ba7d
fix: account recovery impl
sattvikc Jan 31, 2025
c42ab45
fix: base64 encoding changes
tamassoltesz Jan 31, 2025
8a41351
fix: fixes for tests
tamassoltesz Jan 31, 2025
716cfb7
fix: fixing sql issues and encoding issues
tamassoltesz Jan 31, 2025
87834d6
fix: fixes for tests
tamassoltesz Jan 31, 2025
c1b7548
fix: integration fix for signup
tamassoltesz Feb 3, 2025
ca0b250
fix: webauthn flow test stub
tamassoltesz Feb 3, 2025
60e6c21
merging base branch
tamassoltesz Feb 3, 2025
cef33eb
Merge pull request #1104 from supertokens/feat/webauthn-account-recovery
tamassoltesz Feb 3, 2025
ff624a7
fix: fixes for sdk tests
tamassoltesz Feb 3, 2025
2de0ab9
feat: add webauthn recover account apis to webserver
tamassoltesz Feb 3, 2025
4777ad2
feat: crud apis addition
tamassoltesz Feb 4, 2025
62158fc
feat: remove options api
tamassoltesz Feb 4, 2025
4dc4f4b
fix: additional field in the sign in options response
tamassoltesz Feb 4, 2025
2c8cf80
fix: reworked error handling
tamassoltesz Feb 5, 2025
c2dc3e2
fix: fixing GET api not to expect json body
tamassoltesz Feb 5, 2025
4ea4128
fix: sign in + options check
tamassoltesz Feb 5, 2025
2a87ee8
fix: more descriptive error messages for credentialsRegister
tamassoltesz Feb 5, 2025
353d512
fix: typo fixes
tamassoltesz Feb 5, 2025
d169d8b
fix: fixes for tests
tamassoltesz Feb 5, 2025
30db781
fix: not letting dependencies exception to leak out
tamassoltesz Feb 5, 2025
9f0acde
feat: clean up expired data cron
tamassoltesz Feb 5, 2025
151e879
fix: changing recovery token consume
tamassoltesz Feb 6, 2025
5c7d617
fix: fixing loginmethod collection
tamassoltesz Feb 6, 2025
9dd928a
fix: signin fixes
tamassoltesz Feb 6, 2025
189a751
fix: webauthn sign in fixes
tamassoltesz Feb 6, 2025
462440c
fix: add recipeUserId in signIn response
tamassoltesz Feb 7, 2025
9e15707
feat: enable credentials listing api
tamassoltesz Feb 7, 2025
1eda959
feat: extending user listing with webauthn
tamassoltesz Feb 7, 2025
6983db1
fix: don't use the counter at signin check
tamassoltesz Feb 7, 2025
459763a
fix: small fixes
tamassoltesz Feb 7, 2025
98193d0
fix: setting UV and RK to false
tamassoltesz Feb 7, 2025
05daa48
feat: saving userVerification and userPresence values
tamassoltesz Feb 10, 2025
9d1853e
fix: change a bunch of error messages for sdk integration
tamassoltesz Feb 10, 2025
16b1f3d
fix: change a bunch of error messages for sdk integration
tamassoltesz Feb 11, 2025
cb73003
fix: include userVerification and userPresence in options response
tamassoltesz Feb 11, 2025
7860a84
fix: error messages changes
tamassoltesz Feb 12, 2025
8e94554
fix: refactor exceptions
tamassoltesz Feb 12, 2025
f14e3d5
feat: get credential api
tamassoltesz Feb 12, 2025
d54e20f
fix: options generation no longer throws invalid options error as per…
tamassoltesz Feb 12, 2025
6d01f11
fix: more error handling for sign in
tamassoltesz Feb 12, 2025
97aa1f2
fix: rename methods for better readability
tamassoltesz Feb 12, 2025
0652f0c
fix: throw the right exception
tamassoltesz Feb 12, 2025
a27263e
ci: experiment with a GHA to publish test/dev images
porcellus Feb 13, 2025
2c610a3
ci: experiment with publishing dev docker images
porcellus Feb 13, 2025
9bed944
ci: experiment with publishing dev docker images
porcellus Feb 13, 2025
990d2eb
ci: experiment with a GHA to publish test/dev images
porcellus Feb 14, 2025
f9ea13d
Merge pull request #1106 from supertokens/ci/publish_docker
tamassoltesz Feb 14, 2025
0b2b31a
fix: options validation
tamassoltesz Feb 14, 2025
e109bf8
fix: options validation rpId doesn't have to be an url
tamassoltesz Feb 14, 2025
6f2452e
fix: additional validation
tamassoltesz Feb 14, 2025
655e650
fix: fixes for various sdk tests
tamassoltesz Feb 17, 2025
b349d53
merging base
tamassoltesz Feb 18, 2025
81422a7
fix: Dockerfile setupTestEnv --local
tamassoltesz Feb 18, 2025
96796c9
ci: remove arm64 build from dev-docker
porcellus Feb 18, 2025
66657f5
ci: fix dev-docker build
porcellus Feb 18, 2025
e3bcc00
fix: add webauth4jn-test dependency
tamassoltesz Feb 18, 2025
870f340
fix: fixing email verification query for webauthn
tamassoltesz Feb 19, 2025
d9d343b
fix: authenticator mocking and example usage
tamassoltesz Feb 19, 2025
76af6a0
fix: sem ver and few test fixes
sattvikc Feb 20, 2025
7d5c193
fix: test fixes
sattvikc Feb 20, 2025
ebe37f4
Merge pull request #1112 from supertokens/feat/webauthn/version-changes
tamassoltesz Feb 20, 2025
03e1b20
fix: cdi version increment in webauthn test
tamassoltesz Feb 20, 2025
ca19822
fix: webauthn signIn should load all loginmethods of the user
tamassoltesz Feb 20, 2025
2d03d2a
fix: fixing table locked issue with in memory db
tamassoltesz Feb 21, 2025
20b4f64
ci: fix dev-docker build
porcellus Feb 21, 2025
aa17e68
fix: remove unnecessary logging
tamassoltesz Feb 21, 2025
64045ef
ci: fix dev docker image publish
porcellus Feb 24, 2025
47e48db
fix: add tests and fixes
tamassoltesz Feb 25, 2025
764a558
fix: add tests and fixes for email update
tamassoltesz Feb 26, 2025
91c0be3
fix: additional tests and fixes related to useridmapping
tamassoltesz Feb 27, 2025
28aa0d4
fix: add null check
tamassoltesz Feb 27, 2025
acc99c4
fix: add test
tamassoltesz Feb 27, 2025
239c769
fix: additional indexes for performance optimization
tamassoltesz Feb 27, 2025
1fd36da
ci: fix dev-docker build
porcellus Feb 28, 2025
24cd34f
Merge branch '10.0' into feat/webauthn-1
porcellus Feb 28, 2025
efb2708
fix: self-review fixes
tamassoltesz Feb 28, 2025
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
4 changes: 2 additions & 2 deletions .github/helpers/Dockerfile
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ RUN mkdir -p supertokens-core
COPY ./ supertokens-core
RUN echo "org.gradle.vfs.watch=false" >> ./gradle.properties
RUN ./loadModules
RUN ./utils/setupTestEnv --cicd
RUN ./utils/setupTestEnv --local


FROM debian:bookworm-slim
Expand Down Expand Up @@ -70,5 +70,5 @@ RUN echo "$(md5sum /usr/lib/supertokens/config.yaml | awk '{ print $1 }')" >> /C
RUN ln -s /usr/local/bin/docker-entrypoint.sh /entrypoint.sh # backwards compat
EXPOSE 3567
USER "supertokens"
CMD ["/usr/lib/supertokens/jre/bin/java", "-classpath", "/usr/lib/supertokens/core/*:/usr/lib/supertokens/plugin-interface/*:/usr/lib/supertokens/ee/*", "io.supertokens.Main", "/usr/lib/supertokens", "DEV", "host=0.0.0.0", "test_mode"]
CMD ["/usr/lib/supertokens/jre/bin/java", "-classpath", "/usr/lib/supertokens/core/*:/usr/lib/supertokens/plugin-interface/*:/usr/lib/supertokens/ee/*", "io.supertokens.Main", "/usr/lib/supertokens", "DEV", "host=0.0.0.0", "test_mode", "tempDirLocation=/usr/lib/supertokens/temp", "configFile=/usr/lib/supertokens/temp/config.yaml"]
ENTRYPOINT ["/usr/local/bin/docker-entrypoint.sh"]
5 changes: 2 additions & 3 deletions .github/workflows/publish-dev-docker.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ on:
- "**"
jobs:
docker:
runs-on: ubuntu-24.04-arm
runs-on: ubuntu-latest
steps:
- name: set tag
id: set_tag
Expand Down Expand Up @@ -38,6 +38,5 @@ jobs:
uses: docker/build-push-action@v6
with:
push: true
platforms: linux/amd64,linux/arm64
tags: ${{ vars.DOCKERHUB_USERNAME }}/supertokens-core:dev-branch-${{ steps.set_tag.outputs.TAG }}
tags: supertokens/supertokens-core:dev-branch-${{ steps.set_tag.outputs.TAG }}
file: .github/helpers/Dockerfile
5 changes: 4 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -47,4 +47,7 @@ local.properties
*.iml
ee/bin
addDevTag
addReleaseTag
addReleaseTag

install-linux.sh
install-windows.bat
7 changes: 6 additions & 1 deletion build.gradle
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,12 @@ dependencies {
// https://mvnrepository.com/artifact/com.googlecode.libphonenumber/libphonenumber/
implementation group: 'com.googlecode.libphonenumber', name: 'libphonenumber', version: '8.13.25'

// https://mvnrepository.com/artifact/com.webauthn4j/webauthn4j-core
implementation group: 'com.webauthn4j', name: 'webauthn4j-core', version: '0.28.5.RELEASE'

// https://mvnrepository.com/artifact/com.webauthn4j/webauthn4j-test
implementation group: 'com.webauthn4j', name: 'webauthn4j-test', version: '0.28.5.RELEASE'

compileOnly project(":supertokens-plugin-interface")
testImplementation project(":supertokens-plugin-interface")

Expand All @@ -86,7 +92,6 @@ dependencies {
testImplementation group: 'org.reflections', name: 'reflections', version: '0.9.10'

testImplementation 'com.tngtech.archunit:archunit-junit4:0.22.0'

}

application {
Expand Down
4 changes: 4 additions & 0 deletions config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -174,3 +174,7 @@ core_config_version: 0
# (DIFFERENT_ACROSS_APPS | OPTIONAL | Default: number of available processor cores) int value. If specified,
# the supertokens core will use the specified number of threads to complete the migration of users.
# bulk_migration_parallelism:

# (DIFFERENT_ACROSS_APPS | OPTIONAL | Default: 300000) long value. Time in milliseconds for how long a webauthn
# account recovery token is valid for.
# webauthn_recover_account_token_lifetime:
3 changes: 2 additions & 1 deletion coreDriverInterfaceSupported.json
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
"4.0",
"5.0",
"5.1",
"5.2"
"5.2",
"5.3"
]
}
3 changes: 3 additions & 0 deletions devConfig.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -175,3 +175,6 @@ disable_telemetry: true
# the supertokens core will use the specified number of threads to complete the migration of users.
# bulk_migration_parallelism:

# (DIFFERENT_ACROSS_APPS | OPTIONAL | Default: 300000) long value. Time in milliseconds for how long a webauthn
# account recovery token is valid for.
# webauthn_recover_account_token_lifetime:
10 changes: 10 additions & 0 deletions implementationDependencies.json
Original file line number Diff line number Diff line change
Expand Up @@ -115,6 +115,16 @@
"jar": "https://repo1.maven.org/maven2/com/googlecode/libphonenumber/libphonenumber/8.13.25/libphonenumber-8.13.25.jar",
"name": "Libphonenumber 8.13.25",
"src": "https://repo1.maven.org/maven2/com/googlecode/libphonenumber/libphonenumber/8.13.25/libphonenumber-8.13.25-sources.jar"
},
{
"jar": "https://repo1.maven.org/maven2/com/webauthn4j/webauthn4j-core/0.28.3.RELEASE/webauthn4j-core-0.28.5.RELEASE.jar",
"name": "webauthn4j-core 0.28.3.RELEASE",
"src": "https://repo1.maven.org/maven2/com/webauthn4j/webauthn4j-core/0.28.3.RELEASE/webauthn4j-core-0.28.5.RELEASE-sources.jar"
},
{
"jar": "https://repo1.maven.org/maven2/com/webauthn4j/webauthn4j-test/0.28.5.RELEASE/webauthn4j-test-0.28.5.RELEASE.jar",
"name": "webauthn4j-test 0.28.5.RELEASE",
"src": "https://repo1.maven.org/maven2/com/webauthn4j/webauthn4j-test/0.28.5.RELEASE/webauthn4j-test-0.28.5.RELEASE-sources.jar"
}
]
}
2 changes: 1 addition & 1 deletion pluginInterfaceSupported.json
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
{
"_comment": "contains a list of plugin interfaces branch names that this core supports",
"versions": [
"7.0"
"feat/webauthn-1"
]
}
3 changes: 3 additions & 0 deletions src/main/java/io/supertokens/Main.java
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,7 @@
import io.supertokens.cronjobs.Cronjobs;
import io.supertokens.cronjobs.bulkimport.ProcessBulkImportUsers;
import io.supertokens.cronjobs.cleanupOAuthSessionsAndChallenges.CleanupOAuthSessionsAndChallenges;
import io.supertokens.cronjobs.cleanupWebauthnExpiredData.CleanUpWebauthnExpiredDataCron;
import io.supertokens.cronjobs.deleteExpiredAccessTokenSigningKeys.DeleteExpiredAccessTokenSigningKeys;
import io.supertokens.cronjobs.deleteExpiredDashboardSessions.DeleteExpiredDashboardSessions;
import io.supertokens.cronjobs.deleteExpiredEmailVerificationTokens.DeleteExpiredEmailVerificationTokens;
Expand Down Expand Up @@ -265,6 +266,8 @@ private void init() throws IOException, StorageQueryException {

Cronjobs.addCronjob(this, CleanupOAuthSessionsAndChallenges.init(this, uniqueUserPoolIdsTenants));

Cronjobs.addCronjob(this, CleanUpWebauthnExpiredDataCron.init(this, uniqueUserPoolIdsTenants));

// this is to ensure tenantInfos are in sync for the new cron job as well
MultitenancyHelper.getInstance(this).refreshCronjobs();

Expand Down
25 changes: 24 additions & 1 deletion src/main/java/io/supertokens/authRecipe/AuthRecipe.java
Original file line number Diff line number Diff line change
Expand Up @@ -1037,12 +1037,27 @@ public static List<CreatePrimaryUserBulkResult> createPrimaryUsers(Main main,
}
}

@TestOnly
public static AuthRecipeUserInfo[] getUsersByAccountInfo(TenantIdentifier tenantIdentifier,
Storage storage,
boolean doUnionOfAccountInfo, String email,
String phoneNumber, String thirdPartyId,
String thirdPartyUserId) throws StorageQueryException {
return getUsersByAccountInfo(tenantIdentifier,
storage,
doUnionOfAccountInfo, email,
phoneNumber, thirdPartyId,
thirdPartyUserId,
null);
}


public static AuthRecipeUserInfo[] getUsersByAccountInfo(TenantIdentifier tenantIdentifier,
Storage storage,
boolean doUnionOfAccountInfo, String email,
String phoneNumber, String thirdPartyId,
String thirdPartyUserId)
String thirdPartyUserId,
String webauthnCredentialId)
throws StorageQueryException {
Set<AuthRecipeUserInfo> result = new HashSet<>();

Expand All @@ -1064,6 +1079,14 @@ public static AuthRecipeUserInfo[] getUsersByAccountInfo(TenantIdentifier tenant
}
}

if(webauthnCredentialId != null){
AuthRecipeUserInfo user = StorageUtils.getAuthRecipeStorage(storage)
.getPrimaryUserByWebauthNCredentialId(tenantIdentifier, webauthnCredentialId);
if (user != null) {
result.add(user);
}
}

if (doUnionOfAccountInfo) {
AuthRecipeUserInfo[] finalResult = result.toArray(new AuthRecipeUserInfo[0]);
return Arrays.stream(finalResult).sorted((o1, o2) -> {
Expand Down
13 changes: 13 additions & 0 deletions src/main/java/io/supertokens/config/CoreConfig.java
Original file line number Diff line number Diff line change
Expand Up @@ -350,6 +350,11 @@ public class CoreConfig {
"migration of users. (Default: number of available processor cores).")
private int bulk_migration_parallelism = Runtime.getRuntime().availableProcessors();

@NotConflictingInApp
@JsonProperty
@ConfigDescription("Time in milliseconds for how long a webauthn account recovery token is valid for. [Default: 300000 (5 mins)]")
private long webauthn_recover_account_token_lifetime = 300000;

@IgnoreForAnnotationCheck
private static boolean disableOAuthValidationForTest = false;

Expand Down Expand Up @@ -589,6 +594,10 @@ public int getBulkMigrationParallelism() {
return bulk_migration_parallelism;
}

public long getWebauthnRecoverAccountTokenLifetime() {
return webauthn_recover_account_token_lifetime;
}

private String getConfigFileLocation(Main main) {
return new File(CLIOptions.get(main).getConfigFilePath() == null
? CLIOptions.get(main).getInstallationPath() + "config.yaml"
Expand Down Expand Up @@ -786,6 +795,10 @@ void normalizeAndValidate(Main main, boolean includeConfigFilePath) throws Inval
throw new InvalidConfigException("Provided bulk_migration_parallelism must be >= 1");
}

if (webauthn_recover_account_token_lifetime <= 0) {
throw new InvalidConfigException("Provided webauthn_recover_account_token_lifetime must be > 0");
}

for (String fieldId : CoreConfig.getValidFields()) {
try {
Field field = CoreConfig.class.getDeclaredField(fieldId);
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,76 @@
/*
* Copyright (c) 2025, VRAI Labs and/or its affiliates. All rights reserved.
*
* This software is licensed under the Apache License, Version 2.0 (the
* "License") as published by the Apache Software Foundation.
*
* You may not use this file except in compliance with the License. You may
* obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/

package io.supertokens.cronjobs.cleanupWebauthnExpiredData;

import io.supertokens.Main;
import io.supertokens.cronjobs.CronTask;
import io.supertokens.cronjobs.CronTaskTest;
import io.supertokens.pluginInterface.STORAGE_TYPE;
import io.supertokens.pluginInterface.Storage;
import io.supertokens.pluginInterface.StorageUtils;
import io.supertokens.pluginInterface.multitenancy.TenantIdentifier;
import io.supertokens.pluginInterface.webauthn.WebAuthNStorage;

import java.util.List;

public class CleanUpWebauthnExpiredDataCron extends CronTask {

public static final String RESOURCE_KEY = "io.supertokens.cronjobs.cleanupWebauthnExpiredData" +
".CleanUpWebauthnExpiredDataCron";

private CleanUpWebauthnExpiredDataCron(Main main, List<List<TenantIdentifier>> tenantsInfo) {
super("CleanUpWebauthnExpiredDataCron", main, tenantsInfo, true);
}

public static CleanUpWebauthnExpiredDataCron init(Main main, List<List<TenantIdentifier>> tenantsInfo) {
return (CleanUpWebauthnExpiredDataCron) main.getResourceDistributor()
.setResource(new TenantIdentifier(null, null, null), RESOURCE_KEY,
new CleanUpWebauthnExpiredDataCron(main, tenantsInfo));
}

@Override
protected void doTaskPerStorage(Storage storage) throws Exception {
if (storage.getType() != STORAGE_TYPE.SQL) {
return;
}

WebAuthNStorage webAuthNStorage = StorageUtils.getWebAuthNStorage(storage);
webAuthNStorage.deleteExpiredAccountRecoveryTokens();
webAuthNStorage.deleteExpiredGeneratedOptions();
}

@Override
public int getIntervalTimeSeconds() {
if (Main.isTesting) {
Integer interval = CronTaskTest.getInstance(main).getIntervalInSeconds(RESOURCE_KEY);
if (interval != null) {
return interval;
}
}
// Every 24 hours.
return 24 * 3600;
}

@Override
public int getInitialWaitTimeSeconds() {
if (!Main.isTesting) {
return getIntervalTimeSeconds();
} else {
return 0;
}
}
}
Loading
Loading