Created
December 1, 2019 14:16
-
-
Save watahani/f94a8362a4cc075a67254eb403c9c1c7 to your computer and use it in GitHub Desktop.
fido2 sample
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
<!DOCTYPE html> | |
<html lang="en-US"> | |
<head> | |
<title>User details</title> | |
<meta http-equiv="X-UA-Compatible" content="IE=edge"> | |
<meta charset="utf-8"> | |
<meta name="locale" content="en-US"> | |
<meta name="ROBOTS" content="NONE, NOARCHIVE"> | |
<meta name="GOOGLEBOT" content="NOARCHIVE"> | |
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> | |
<meta name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=2.0, user-scalable=yes"> | |
<style id="common"> | |
.pageLevel,body{text-align:left}@charset "utf-8";#panel,.pageLevel,.panel li,label{display:block}button:disabled,button[disabled]{background-color:#696969}::-webkit-input-placeholder{color:#6d6d6d}:-moz-placeholder{color:#6d6d6d}::-moz-placeholder{color:#6d6d6d}:-ms-input-placeholder{color:#6d6d6d!important}.accountButton,button{-moz-user-select:none;user-select:none}.accountButton,a,button{cursor:pointer}h1,h2{font-weight:400}.panel li{list-style:none}#api ul li{display:inline;list-style-type:none;margin-left:0}html{background-color:#00abec}.normaltext,.smalltext,.tinytext{font-family:'Segoe UI',Segoe,SegoeUI-Regular-final,Tahoma,Helvetica,Arial,sans-serif}div#background_branding_container,div#background_page_overlay{width:100%;z-index:0;position:fixed;left:0;overflow:hidden;top:0}div#background_branding_container{background:#00abec;height:100%;-webkit-animation:fadeIn 1s;animation:fadeIn 1s}*,.panel,body{margin:0;padding:0}body{z-index:-999}.panel,.panel_layout,.panel_layout_row,body,html{height:100%}#panel{border-left:1px solid #FFF}a{background-color:transparent;color:#2872DD;text-decoration:none}a:hover{text-decoration:underline}button{width:auto;min-width:50px;height:32px;margin-top:2px;-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;background:#2672E6;border:1px solid #FFF;color:#fff;transition:background 1s ease 0s;font-size:100%;padding:0 2px}button:hover{background:#0F3E83;border:1px solid #3079ed;-moz-box-shadow:0 0 0;-webkit-box-shadow:0 0 0;box-shadow:0 0 0}img{border:0}form{height:auto;width:auto}h2{font-size:1em}.normaltext{font-size:.9em}input:focus,select:focus,textarea:focus{outline:0}.error{color:#a61e0c}.pageLevel{width:293px;margin-top:5px;font-size:1.1em;height:auto}.highlightError{border:1px solid #a61e0c!important}img#background_background_image{height:100%;width:100%}div#background_page_overlay{background:left top no-repeat fixed #F7F7F7;height:100%;opacity:1}.panel{background:#FFF;float:right;height:100%;overflow-x:hidden;overflow-y:auto;position:fixed;right:0;width:500px;z-index:1}.accountButton,.accountButton:hover{background-image:url();background-repeat:no-repeat}.inner_container{max-height:90%;min-height:90%;width:100%}#panel_center,#panel_left,#panel_right{display:inline-block;border:0;height:100%;margin:0}#panel_left{padding:0;width:50px}#panel_center{min-height:100%;padding:0;width:378px}.hide{opacity:0}.api_container{display:inline-block;padding-left:0;padding-top:120px;position:relative;transition:padding .6s ease 0s;width:100%;height:100%}select{height:28px}input[type=email],input[type=number],input[type=password],input[type=text]{-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;height:28px;width:300px;border:1px solid #696969;z-index:3;color:#000;padding:0 0 0 3px;-moz-box-shadow:0 0 0;-webkit-box-shadow:0 0 0;box-shadow:0 0 0;margin-right:3px}input[type=email]:focus,input[type=number]:focus,input[type=password]:focus,input[type=text]:focus{border:1px solid #4d90fe}.accountButton{border:1px solid #FFF;color:#FFF;margin-left:0;margin-right:2px;transition:background-color 1s ease 0s;-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;text-align:center;word-wrap:break-word;height:34px;width:158px;padding-left:30px;background-color:#505050}.accountButton:hover{background-color:#B9B9B9;border:1px solid #FFF;-moz-box-shadow:0 0 0;-webkit-box-shadow:0 0 0;box-shadow:0 0 0}#GoogleExchange:focus,.accountButton:focus{outline:gold solid 1px}#GitHubExchange,#GoogleExchange,#MicrosoftAccountExchange{outline:#AAA solid 1px;color:#000;background-color:#FFF}#MicrosoftAccountExchange{background-image:url();outline-offset:-2.5px}#GitHubExchange:hover,#GoogleExchange:hover,#MicrosoftAccountExchange:hover{background-color:#EEE}#GoogleExchange{background-image:url();outline-offset:-2.5px}#GitHubExchange{background-image:url();outline-offset:-2.5px;background-size:26px;background-position-x:5px;background-position-y:4px}#TwitterExchange{background-image:url();background-color:#60A9DD}#TwitterExchange:hover{background-color:#BFDCF1}#FacebookExchange{background-image:url();background-color:#3B5B9C}#FacebookExchange:hover{background-color:#B0BDD7}#LinkedInExchange{background-image:url();background-color:#0077B5}#LinkedInExchange:hover{background-color:#99CAE1}#AmazonExchange{background-image:url();background-color:#FFF;color:transparent}.working{background:url() no-repeat;height:10px;width:auto}@-webkit-keyframes fadeIn{from{opacity:0}to{opacity:1}}@media only screen and (min-height:250px) and (max-width:450px),(min-device-height:250px) and (max-device-width:450px){html{overflow:hidden}.tinytext{font-size:.6em}.smalltext{font-size:.7em}.normaltext{font-size:.8em}.bigtext{font-size:.9em}.gianttext{font-size:1.2em}div#background_branding_container{display:none;opacity:0;z-index:-999}div#background_page_overlay{display:none;z-index:-999}.panel_layout,.panel_layout_row{width:100%}#panel_left{width:40px}#panel_center{width:calc(100% - 80px)}.inner_container,.panel{width:100%}#panel_right{display:none}.panel{float:none;height:100%;margin:0;min-width:220px;overflow:auto;padding:0;z-index:1}}@media only screen and (max-width:300px),(max-device-width:300px){#panel_left{width:15px}} | |
</style> | |
<style>.div.buttons.verify label{display:none!important}.intro,.tiny{display:none}span.placeholder{color:#696969;margin:6px 0 0 -6px;padding-left:10px;width:380px;z-index:-1}.attrEntry{padding-top:8px}.attrEntry .error.itemLevel{display:none;color:#a61e0c;font-size:.9em}.attrEntry.validate .error.itemLevel.show,.attrEntry.validate .helpText.show,.helpText.show{display:block}.attrEntry input:invalid{outline:0}.attrEntry.validate input:invalid{outline:#a61e0c solid 1px;background-color:#fce8e8}.attrEntry .helpText{display:none;padding-bottom:10px;padding-top:10px;color:#a61e0c;font-size:.9em}.pageLevel{width:calc(100% - 35px)}.attrEntry input[type=text],.day,.dropdown_single,.month,.year,input[type=password]{-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;height:28px;margin:3px 0;border:1px solid #696969;z-index:3;background:0 0;color:#000;padding:0 0 0 3px;-moz-box-shadow:0 0 0;-webkit-box-shadow:0 0 0;box-shadow:0 0 0}.attrEntry .day{width:100px;margin-right:10px}.attrEntry .month{width:120px;margin-right:10px}.attrEntry .year{width:115px;margin-right:10px}.attrEntry .dropdown_single{width:355px;margin:0 100px 0 0}.attrEntry input[type=text],input[type=password],input[type=email]{width:calc(100% - 30px)!important;margin-right:0!important}.attrEntry input[type=text]:focus{border:1px solid #6B6B6B}.errorText{padding:20px;width:310px;color:#a61e0c!important;background-color:#fce8e8!important;font-size:1.1em;text-align:center}.buttons button{-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;-moz-user-select:none;background:#2672E6;border:1px solid #FFF;color:#FFF;cursor:pointer;margin-right:4px;margin-left:0;padding:6px 12px;transition:background 1s ease 0s;width:auto;height:auto;font-size:100%}.buttons button:disabled,button[disabled]{background-color:#696969}.buttons button:hover{background:#0F3E83;border:1px solid #FFF;-moz-box-shadow:0 0 0;-webkit-box-shadow:0 0 0;box-shadow:0 0 0}.button:focus{outline:gold solid 1px}.alert-modal,.verifying-modal{display:none}.verifying-modal{color:#fff;font-size:24px;display:none;font-family:'Segoe UI',Segoe,SegoeUI-Regular-final,Tahoma,Helvetica,Arial,sans-serif}@-moz-document url-prefix(){#AmazonExchange::-moz-focus-inner,#FacebookExchange::-moz-focus-inner,#GoogleExchange::-moz-focus-inner,#LinkedInExchange::-moz-focus-inner,#MicrosoftAccountExchange::-moz-focus-inner{padding-left:45px}#AmazonExchange,#FacebookExchange,#GoogleExchange,#LinkedInExchange,#MicrosoftAccountExchange{padding-left:0}}</style> | |
<style> | |
#challenge, #extension_fido_rawId, #clientDataJSON, #signature, #authenticatorData, #userHandle, | |
label[for=challenge], label[for=extension_fido_rawId], label[for=clientDataJSON], label[for=signature], label[for=authenticatorData], label[for=userHandle], | |
#readOnlyDisplayName, #readOnlyName, | |
label[for=readOnlyDisplayName], label[for=readOnlyName], | |
#objectId, #attestationObject, | |
label[for=objectId], label[for=attestationObject], | |
#continue, #cancel | |
{ | |
display: none; | |
} | |
#challenge, #readOnlyName, #readOnlyDisplayName, #objectId { | |
border-color: green; | |
background-color: #daeadd; | |
} | |
#fidoAuthn{ | |
padding: 5px 10px 5px 10px; | |
} | |
#attachmentPlatform, #attachmentCrossPlatform | |
{ | |
display: inline!important; | |
margin-left: 15px; | |
} | |
label[for=attachmentPlatform], label[for=attachmentCrossPlatform] | |
{ | |
display: inline!important; | |
margin-left: 5px; | |
} | |
</style> | |
</head> | |
<body> | |
<div id="background_branding_container"> | |
<img id="background_background_image" src="https://ads5m2.b2clogin.com/static/tenant/default/img/default_signin_illustration_optimized.png" alt="Illustration" /> | |
</div> | |
<p id="page_loaded_msg" role="alert" aria-live="assertive" tabindex="-1" aria-label="Page has been loaded successfully"></p> | |
<div class="panel" id="panel"> | |
<table class="panel_layout"> | |
<tbody> | |
<tr class="panel_layout_row"> | |
<td id="panel_left" /> | |
<td id="panel_center"> | |
<div class="inner_container"> | |
<div class="api_container normaltext"> | |
<div id="api"></div> | |
<span id="selectPlatform"> | |
<b>Select the type of the authenticator</b><br /> | |
<input type="radio" name="attachment" id="attachmentPlatform" value="platform" checked> | |
<label for="attachmentPlatform"> | |
Windows Hello (platform) | |
</label><span id="attachmentPlatformError" style="color: red;"> - Not available </span> | |
<input type="radio" name="attachment" id="attachmentCrossPlatform" value="cross-platform"> | |
<label for="attachmentCrossPlatform"> | |
External device (cross-platform) | |
</label> | |
</span> | |
<div id="lastCredentialsContainer"> | |
<input type="checkbox" id="lastCredentials"> Sign-in with last credentials used in this browser</input> | |
</div> | |
<br> | |
<button href="#" role="button" type="button" id="fidoAuthn"> | |
<i id="registerSpinner"></i><span id="fidoAuthnTitle">Register or sign-in</span></button> | |
<br /> | |
<div class="alert alert-primary hidden wordWrap" role="alert" id="status"> | |
</div> | |
<br /> | |
<a id="debugMode" href="#" style="font-size: .75em;">Debug</a> | |
<span id="spaceLine"> | |
<br /> | |
</span> | |
</div> | |
</div> | |
</td> | |
</tr> | |
</tbody> | |
</table> | |
</div> | |
<script> | |
"use strict"; | |
$(document).ready(function() { if (navigator.userAgent.match(/IEMobile\/10\.0/)) { var t = document.createElement("style"); t.appendChild(document.createTextNode("@-ms-viewport{width:auto!important}")), t.appendChild(document.createTextNode("@-ms-viewport{height:auto!important}")), document.getElementsByTagName("head")[0].appendChild(t) } if (navigator.userAgent.match(/MSIE 10/i)) { var e = $("#footer_links_container"); $(e).css("padding-top", "100px") } var o, i = $("#background_background_image"), n = function () { document.body.style.overflow = "hidden", ($(window).width() - 500) / $(window).height() < o ? (i.height($(window).height()), i.width("auto")) : (i.width($(window).width() - 500), i.height("auto")), document.body.style.overflow = "" }; $("<img>").attr("src", i.attr("src")).load(function () { o = this.width / this.height, n() }), $(window).resize(function () { n() }), "undefined" != typeof $("#MicrosoftAccountExchange") && $("#MicrosoftAccountExchange").text("Microsoft"), $("*").removeAttr("placeholder") });</script> | |
<script> | |
$(function () { | |
$("#debugMode").on('click', () => showHiddenFields()); | |
if ($("#signature").length) | |
{ | |
// Sing-in | |
$("#selectPlatform").hide(); | |
$("#fidoAuthn").text("Authenticate"); | |
$("#fidoAuthn").on('click', () => authenticateButtonClicked()); | |
if (!localStorage.getItem("credentialId")) | |
$("#lastCredentialsContainer").hide(); | |
else | |
{ | |
$("#lastCredentials").prop( "checked", true ); | |
} | |
} | |
else | |
{ | |
// Register | |
$("#attachmentPlatformError").hide(); | |
$("#lastCredentialsContainer").hide(); | |
$( "#spaceLine").insertBefore( $( "#attachmentCrossPlatform" ) ); | |
$("#fidoAuthn").text("Register"); | |
$("#fidoAuthn").on('click', () => registerButtonClicked()); | |
} | |
//Update UI to reflect availability of platform authenticator | |
if (PublicKeyCredential && typeof PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable !== "function") { | |
markPlatformAuthenticatorUnavailable(); | |
} else if (PublicKeyCredential && typeof PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable === "function") { | |
PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable().then(available => { | |
if (!available) { | |
markPlatformAuthenticatorUnavailable(); | |
} | |
}).catch(e=>{ | |
markPlatformAuthenticatorUnavailable(); | |
}); | |
} | |
}); | |
var isDebug = false; | |
function showHiddenFields() { | |
isDebug = true; | |
var claims = ['challenge', 'extension_fido_rawId', 'clientDataJSON', 'signature', 'authenticatorData', 'userHandle', 'readOnlyDisplayName', 'readOnlyName', 'objectId', 'attestationObject', 'continue']; | |
claims.forEach(function(item, index, array) { | |
if ($('#' + item).length >0 ) | |
$('#' + item).show() | |
if ($('label[for="' + item + '"]').length >0 ) | |
$('label[for="' + item + '"]').show() | |
}) | |
} | |
/** | |
* Marks platform authenticator as unavailable in UI | |
*/ | |
function markPlatformAuthenticatorUnavailable() { | |
$("#attachmentPlatformError").show(); | |
$("#attachmentPlatform").attr("disabled", true); | |
$("#attachmentCrossPlatform" ).prop( "checked", true ); | |
} | |
/** | |
* Disables all input controls and buttons on the page | |
*/ | |
function disableControls() { | |
$('#fidoAuthn').attr('disabled',''); | |
$("#status").addClass('hidden'); | |
} | |
/** | |
* Enables all input controls and buttons on the page | |
*/ | |
function enableControls() { | |
$('#fidoAuthn').removeAttr('disabled'); | |
$("#status").removeClass('hidden'); | |
} | |
/** | |
* Handler for create button being pressed | |
*/ | |
function registerButtonClicked() { | |
disableControls(); | |
$("#registerSpinner").removeClass("hidden"); | |
createCredential() | |
.then(credentialId => { | |
$("#registerSpinner").addClass("hidden"); | |
enableControls(); | |
}).catch(e => { | |
$("#status").text("Error: " + e); | |
$("#registerSpinner").addClass("hidden"); | |
enableControls(); | |
}); | |
} | |
/** | |
* Handler for get button being pressed | |
*/ | |
function authenticateButtonClicked() { | |
disableControls(); | |
$("#authenticateSpinner").removeClass("hidden"); | |
getAssertion() | |
.then(credential => { | |
$("#status").text("Successfully verified credential with ID: " + credential.id); | |
$("#authenticateSpinner").addClass("hidden"); | |
enableControls(); | |
}).catch(e => { | |
$("#status").text("Error: " + e); | |
$("#authenticateSpinner").addClass("hidden"); | |
enableControls(); | |
}); | |
} | |
/** | |
* Calls the .create() webauthn APIs and sends returns to server | |
* @param {ArrayBuffer} challenge challenge to use | |
* @return {any} server response object | |
*/ | |
function createCredential() { | |
console.log("createCredential started"); | |
if (!PublicKeyCredential || typeof PublicKeyCredential.isUserVerifyingPlatformAuthenticatorAvailable !== "function") | |
return Promise.reject("WebAuthn APIs are not available on this user agent."); | |
var attachment = $("input[name='attachment']:checked").val(); | |
var createCredentialOptions = { | |
rp: { | |
name: "WebAuthn Sample App", | |
icon: "https://example.com/rpIcon.png" | |
}, | |
user: { | |
id: stringToArrayBuffer($("#objectId").val()), | |
name: $("#readOnlyName").val(), | |
displayName: $("#readOnlyDisplayName").val(), | |
icon: "https://example.com/userIcon.png" | |
}, | |
pubKeyCredParams: [ | |
{ | |
//External authenticators support the ES256 algorithm | |
type: "public-key", | |
alg: -7 | |
}, | |
{ | |
//Windows Hello supports the RS256 algorithm | |
type: "public-key", | |
alg: -257 | |
} | |
], | |
authenticatorSelection: { | |
//Select authenticators that support username-less flows | |
requireResidentKey: true, | |
//Select authenticators that have a second factor (e.g. PIN, Bio) | |
userVerification: "required", | |
//Selects between bound or detachable authenticators | |
authenticatorAttachment: attachment | |
}, | |
//Since Edge shows UI, it is better to select larger timeout values | |
timeout: 50000, | |
//an opaque challenge that the authenticator signs over | |
challenge: stringToArrayBuffer($("#challenge").val()), | |
//prevent re-registration by specifying existing credentials here | |
excludeCredentials: [], | |
//specifies whether you need an attestation statement | |
attestation: "none" | |
}; | |
return navigator.credentials.create({ | |
publicKey: createCredentialOptions | |
}).then(rawAttestation => { | |
$("#extension_fido_rawId").val(base64encode(rawAttestation.rawId)); | |
$("#clientDataJSON").val(base64encode(rawAttestation.response.clientDataJSON)); | |
$("#attestationObject").val(base64encode(rawAttestation.response.attestationObject)); | |
localStorage.setItem("credentialId", $("#extension_fido_rawId").val()); | |
$("#status").text("Successfully created credential with ID: " + $("#extension_fido_rawId").val()); | |
if (!isDebug) | |
$("#continue").click(); | |
}); | |
} | |
/** | |
* Calls the .get() API and sends result to server to verify | |
* @param {ArrayBuffer} challenge | |
* @return {any} server response object | |
*/ | |
function getAssertion() { | |
if (!PublicKeyCredential) | |
return Promise.reject("WebAuthn APIs are not available on this user agent."); | |
var allowCredentials = []; | |
var getAssertionOptions = { | |
//an opaque challenge that the authenticator signs over | |
challenge: stringToArrayBuffer($("#challenge").val()), | |
//Since Edge shows UI, it is better to select larger timeout values | |
timeout: 50000 | |
}; | |
if ($("#lastCredentials").is(':checked')) | |
{ | |
var credentialId = localStorage.getItem("credentialId"); | |
if (!credentialId) | |
return Promise.reject("Please create a credential first"); | |
console.log("Allow credentials to sign-in: " + credentialId); | |
//specifies which credential IDs are allowed to authenticate the user | |
//if empty, any credential can authenticate the users | |
getAssertionOptions.allowCredentials = [{ | |
type: "public-key", | |
id: Uint8Array.from(atob(credentialId), c=>c.charCodeAt(0)).buffer | |
}] | |
} | |
else | |
{ | |
console.log("Allow credentials to sign-in is empty"); | |
} | |
return navigator.credentials.get({ | |
publicKey: getAssertionOptions | |
}).then(rawAssertion => { | |
$("#extension_fido_rawId").val(base64encode(rawAssertion.rawId)); | |
$("#clientDataJSON").val(base64encode(rawAssertion.response.clientDataJSON)); | |
$("#userHandle").val(base64encode(rawAssertion.response.userHandle)); | |
$("#signature").val(base64encode(rawAssertion.response.signature)); | |
$("#authenticatorData").val(base64encode(rawAssertion.response.authenticatorData)); | |
if (!isDebug) | |
$("#continue").click(); | |
return {"id": $("#extension_fido_rawId").val()} | |
}); | |
} | |
/** | |
* Base64 encodes an array buffer | |
* @param {ArrayBuffer} arrayBuffer | |
*/ | |
function base64encode(arrayBuffer) { | |
if (!arrayBuffer || arrayBuffer.length == 0) | |
return undefined; | |
return btoa(String.fromCharCode.apply(null, new Uint8Array(arrayBuffer))); | |
} | |
/** | |
* Converts an array buffer to a UTF-8 string | |
* @param {ArrayBuffer} arrayBuffer | |
* @returns {string} | |
*/ | |
function arrayBufferToString(arrayBuffer) { | |
return String.fromCharCode.apply(null, new Uint8Array(arrayBuffer)); | |
} | |
/** | |
* Converts a string to an ArrayBuffer | |
* @param {string} string string to convert | |
* @returns {ArrayBuffer} | |
*/ | |
function stringToArrayBuffer(str){ | |
return Uint8Array.from(str, c => c.charCodeAt(0)).buffer; | |
} | |
</script> | |
</body> | |
</html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment