mirror of
https://github.com/System-End/identity-vault.git
synced 2026-04-19 20:55:11 +00:00
88 lines
3.4 KiB
JavaScript
88 lines
3.4 KiB
JavaScript
export function webauthnRegister() {
|
|
return {
|
|
nickname: '',
|
|
loading: false,
|
|
error: null,
|
|
browserSupported: true,
|
|
|
|
init() {
|
|
this.browserSupported = !!(
|
|
globalThis.PublicKeyCredential?.parseCreationOptionsFromJSON &&
|
|
navigator.credentials?.create
|
|
);
|
|
},
|
|
|
|
async getRegistrationOptions() {
|
|
const response = await fetch('/identity_webauthn_credentials/options', {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
'X-CSRF-Token': document.querySelector('meta[name="csrf-token"]')?.content
|
|
}
|
|
});
|
|
|
|
if (!response.ok) {
|
|
throw new Error('Failed to get registration options from server');
|
|
}
|
|
|
|
return await response.json();
|
|
},
|
|
|
|
async register() {
|
|
if (!this.nickname.trim()) {
|
|
this.error = 'Please enter a nickname for your passkey';
|
|
return;
|
|
}
|
|
|
|
this.loading = true;
|
|
this.error = null;
|
|
|
|
try {
|
|
const options = await this.getRegistrationOptions();
|
|
const publicKey = PublicKeyCredential.parseCreationOptionsFromJSON(options);
|
|
const credential = await navigator.credentials.create({ publicKey });
|
|
|
|
if (!credential) {
|
|
throw new Error('Credential creation failed');
|
|
}
|
|
|
|
const response = credential.response;
|
|
const toBase64Url = (buffer) => {
|
|
return btoa(String.fromCharCode(...new Uint8Array(buffer)))
|
|
.replace(/\+/g, '-').replace(/\//g, '_').replace(/=+$/, '');
|
|
};
|
|
const credentialJSON = {
|
|
id: credential.id,
|
|
rawId: toBase64Url(credential.rawId),
|
|
type: credential.type,
|
|
response: {
|
|
clientDataJSON: toBase64Url(response.clientDataJSON),
|
|
attestationObject: toBase64Url(response.attestationObject),
|
|
},
|
|
clientExtensionResults: credential.getClientExtensionResults(),
|
|
};
|
|
|
|
const credentialDataField = document.getElementById('registration-credential-data');
|
|
const nicknameField = document.getElementById('registration-nickname');
|
|
const form = document.getElementById('webauthn-registration-form');
|
|
|
|
credentialDataField.value = JSON.stringify(credentialJSON);
|
|
nicknameField.value = this.nickname;
|
|
form.submit();
|
|
} catch (error) {
|
|
console.error('Passkey registration error:', error);
|
|
if (error.name === 'NotAllowedError') {
|
|
this.error = 'Registration was cancelled or not allowed';
|
|
} else if (error.name === 'InvalidStateError') {
|
|
this.error = 'This passkey is already registered';
|
|
} else if (error.name === 'NotSupportedError') {
|
|
this.error = 'Passkeys are not supported on this device';
|
|
} else {
|
|
this.error = error.message || 'An unexpected error occurred';
|
|
}
|
|
|
|
this.loading = false;
|
|
}
|
|
}
|
|
};
|
|
}
|