407 lines
15 KiB
HTML
407 lines
15 KiB
HTML
<!DOCTYPE html>
|
|
|
|
<html
|
|
lang="en-US"
|
|
class="light-style"
|
|
>
|
|
<head>
|
|
<meta charset="utf-8" />
|
|
<meta
|
|
name="viewport"
|
|
content="width=device-width, initial-scale=1.0, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0"
|
|
/>
|
|
|
|
<title>Devices - amiPro sample site</title>
|
|
|
|
<meta name="description" content="" />
|
|
|
|
<!-- Favicon -->
|
|
<link rel="icon" type="image/x-icon" href="files/favicon.ico" />
|
|
|
|
<!-- Fonts -->
|
|
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
|
<link
|
|
href="https://fonts.googleapis.com/css2?family=Public+Sans:ital,wght@0,300;0,400;0,500;0,600;0,700;1,300;1,400;1,500;1,600;1,700&display=swap"
|
|
rel="stylesheet"
|
|
/>
|
|
|
|
<!-- Icons. Uncomment required icon fonts -->
|
|
<link rel="stylesheet" href="files/boxicons.css?v=20230405" />
|
|
|
|
<!-- Core CSS -->
|
|
<link rel="stylesheet" href="files/core.css" class="template-customizer-core-css" />
|
|
<link rel="stylesheet" href="files/theme-default.css" class="template-customizer-theme-css" />
|
|
<link rel="stylesheet" href="files/demo.css" />
|
|
|
|
<!-- Vendors CSS -->
|
|
<link rel="stylesheet" href="files/perfect-scrollbar.css" />
|
|
|
|
<!-- Page CSS -->
|
|
|
|
<!-- Helpers -->
|
|
<script src="files/helpers.js"></script>
|
|
|
|
<!--! Template customizer & Theme config files MUST be included after core stylesheets and helpers.js in the <head> section -->
|
|
<!--? Config: Mandatory theme config file contain global vars & default theme options, Set your preferred theme option in this file. -->
|
|
<script src="files/config.js"></script>
|
|
|
|
<script src="files/jquery.js"></script>
|
|
<script src="files/popper.js"></script>
|
|
<script src="files/bootstrap.js"></script>
|
|
<script src="files/perfect-scrollbar.js"></script>
|
|
<script src="files/menu.js"></script>
|
|
<script src="files/main.js"></script>
|
|
|
|
<script src="files/ua-parser.js"></script>
|
|
|
|
<script src="files/amipro_utils.js?v=20230401402"></script>
|
|
<script src="files/dfido2-lib.js?v=20230918"></script>
|
|
|
|
<script>
|
|
|
|
var user_id, reg_session_id;
|
|
|
|
const i18n_messages = new Map();
|
|
|
|
var lang_map = new Map();
|
|
lang_map.set("en-US", "Add device");
|
|
lang_map.set("zh-CN", "添加设备");
|
|
lang_map.set("ja", "デバイスを追加");
|
|
i18n_messages.set("btn_add", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "My devices");
|
|
lang_map.set("zh-CN", "我的设备");
|
|
lang_map.set("ja", "マイデバイス");
|
|
i18n_messages.set("my_devices", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Device registered");
|
|
lang_map.set("zh-CN", "添加设备成功");
|
|
lang_map.set("ja", "デバイス登録完了。");
|
|
i18n_messages.set("msg_register_ok", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Device");
|
|
lang_map.set("zh-CN", "设备");
|
|
lang_map.set("ja", "デバイス");
|
|
i18n_messages.set("title_device", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Registered time");
|
|
lang_map.set("zh-CN", "添加时间");
|
|
lang_map.set("ja", "登録時間");
|
|
i18n_messages.set("title_time", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Actions");
|
|
lang_map.set("zh-CN", "操作");
|
|
lang_map.set("ja", "操作");
|
|
i18n_messages.set("title_act", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Delete");
|
|
lang_map.set("zh-CN", "删除");
|
|
lang_map.set("ja", "削除");
|
|
i18n_messages.set("title_del", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Contact us");
|
|
lang_map.set("zh-CN", "联系我们");
|
|
lang_map.set("ja", "お問い合わせ");
|
|
i18n_messages.set("title_contact", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Log out");
|
|
lang_map.set("zh-CN", "登出");
|
|
lang_map.set("ja", "ログアウト");
|
|
i18n_messages.set("title_logout", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "No devices, please add.");
|
|
lang_map.set("zh-CN", "无设备,请添加。");
|
|
lang_map.set("ja", "デバイスがなし、追加してください。");
|
|
i18n_messages.set("title_empty_list", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Deleted device.");
|
|
lang_map.set("zh-CN", "设备删除成功。");
|
|
lang_map.set("ja", "デバイスを削除しました。");
|
|
i18n_messages.set("msg_deldev_ok", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "Do you want to delete this device?");
|
|
lang_map.set("zh-CN", "确认删除此设备吗?");
|
|
lang_map.set("ja", "デバイスを削除しますか?");
|
|
i18n_messages.set("msg_confirm_deldev", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "The fido2 session is good.");
|
|
lang_map.set("zh-CN", "FIDO2会话正常。");
|
|
lang_map.set("ja", "FIDO2セッションは正常です。");
|
|
i18n_messages.set("msg_session_status_ok", lang_map);
|
|
|
|
lang_map = new Map();
|
|
lang_map.set("en-US", "The fido2 session is invalid. You can refuse accessing to your site's pages.");
|
|
lang_map.set("zh-CN", "FIDO2会话无效。您可以拒绝访问您网站的页面。");
|
|
lang_map.set("ja", "FIDO2セッションは無効です。あなたのサイトのページにアクセスすることを拒否できます。");
|
|
i18n_messages.set("msg_session_status_fail", lang_map);
|
|
|
|
window.onload = async function() {
|
|
setI18NText(i18n_messages)
|
|
|
|
|
|
setSessionStatus();
|
|
|
|
let url = new URL(window.location.href);
|
|
let params = url.searchParams;
|
|
reg_session_id = params.get("rid");
|
|
user_id = params.get("uid");
|
|
|
|
if(reg_session_id && 0 < reg_session_id.length){
|
|
const reg_unm_json = await getRegistrationUser(reg_session_id);
|
|
user_id = ''
|
|
if(reg_unm_json && reg_unm_json.status === 'ok'){
|
|
user_id = reg_unm_json.username;
|
|
await addDevice(reg_unm_json.displayname)
|
|
}else{
|
|
window.location.href = "login.html";
|
|
}
|
|
}
|
|
$('#user_id').html(user_id);
|
|
listDevices();
|
|
}
|
|
|
|
async function setSessionStatus(){
|
|
const sessionOk = await validSession();
|
|
if(sessionOk){
|
|
$('#session_status').html(getI18NText(i18n_messages, 'msg_session_status_ok'))
|
|
}else{
|
|
$('#session_status').html(getI18NText(i18n_messages, 'msg_session_status_fail'))
|
|
}
|
|
}
|
|
|
|
async function addDevice(display_name){
|
|
//You can change the display name or ask the user to input it, if the display_name is null.
|
|
const result = await registerFido2(user_id, display_name?display_name:'dis_'+user_id);
|
|
|
|
if(result.status === 'ok'){
|
|
//You can try to insert user_id into your user db now if this user_id does not exist in your db.
|
|
|
|
alert($('#msg_register_ok').html());
|
|
listDevices();
|
|
}else{
|
|
errProcessFido2(result)
|
|
}
|
|
|
|
setSessionStatus();
|
|
}
|
|
|
|
async function delDevice(device_id){
|
|
if(window.confirm(getI18NText(i18n_messages, 'msg_confirm_deldev'))){
|
|
const result = await delUserDeviceFido2(device_id)
|
|
if("ok" === result.status){
|
|
alert(getI18NText(i18n_messages, 'msg_deldev_ok'));
|
|
listDevices();
|
|
}else{
|
|
const msg = getI18NErrorMessage(result.errorMessage);
|
|
alert(msg?msg:result.errorMessage);
|
|
}
|
|
}
|
|
}
|
|
|
|
async function listDevices(){
|
|
const result = await listUserDevicesFido2()
|
|
$("#devices_list").html("");
|
|
if("ok" === result.status){
|
|
let lst_html = ""
|
|
|
|
for(let dev of result.devices){
|
|
var dev_desc = dev.desc
|
|
if(!dev_desc || 0 === dev_desc.length){
|
|
let parser = new UAParser(dev.userAgent);
|
|
|
|
if(parser.getOS().name){
|
|
dev_desc = parser.getDevice().model + ',' + parser.getOS().name + ',' + parser.getBrowser().name;
|
|
}else{
|
|
dev_desc = dev.userAgent;
|
|
}
|
|
}
|
|
|
|
var date = new Date(dev.registered_time);
|
|
|
|
lst_html += '<tr><td><strong>'+ dev_desc +'</strong></td><td>'+
|
|
date.toLocaleString()+'</td><td>'+
|
|
"<a style=\"padding-left: 0px;\" class=\"dropdown-item\" href=\"javascript:delDevice('"+dev.device_id+"');\">"+
|
|
'<i class="bx bx-trash me-1"></i><span id="title_del"> '+ getI18NText(i18n_messages, 'title_del') +'</span></a></div></td></tr>'
|
|
}
|
|
$("#devices_list").html(lst_html);
|
|
}else{
|
|
const msg = getI18NErrorMessage(result.errorMessage); //fido2LibErrMsgLanguages.japanese
|
|
alert(msg?msg:result.errorMessage);
|
|
}
|
|
}
|
|
|
|
</script>
|
|
</head>
|
|
|
|
<body>
|
|
<div id="msg_register_ok" style="display:none;">Device registered</div>
|
|
|
|
<!-- Layout wrapper -->
|
|
<div class="layout-wrapper layout-content-navbar">
|
|
<div class="layout-container">
|
|
<!-- Layout container -->
|
|
<div class="layout-page">
|
|
<!-- Navbar -->
|
|
|
|
<nav
|
|
class="layout-navbar container-xxl navbar navbar-expand-xl navbar-detached align-items-center bg-navbar-theme"
|
|
id="layout-navbar"
|
|
>
|
|
<div class="layout-menu-toggle navbar-nav align-items-xl-center me-3 me-xl-0 d-xl-none">
|
|
<a class="nav-item nav-link px-0 me-xl-4" href="javascript:void(0)">
|
|
<i class="bx bx-menu bx-sm"></i>
|
|
</a>
|
|
</div>
|
|
|
|
<div class="navbar-nav-right d-flex align-items-center" id="navbar-collapse">
|
|
<!-- Search -->
|
|
<div class="navbar-nav align-items-center">
|
|
<div class="nav-item d-flex align-items-center">
|
|
<i class="bx bx-user fs-4 lh-0"></i>
|
|
WELCOME <span id="session_status" style="padding-left: 20px;"></span>
|
|
</div>
|
|
</div>
|
|
<!-- /Search -->
|
|
|
|
<ul class="navbar-nav flex-row align-items-center ms-auto">
|
|
|
|
<!-- User -->
|
|
<li class="nav-item navbar-dropdown dropdown-user dropdown">
|
|
<a class="nav-link dropdown-toggle hide-arrow" href="javascript:void(0);" data-bs-toggle="dropdown">
|
|
<div class="avatar avatar-online">
|
|
<img src="files/avatar.png" alt class="w-px-40 h-auto rounded-circle" />
|
|
</div>
|
|
</a>
|
|
<ul class="dropdown-menu dropdown-menu-end">
|
|
<li>
|
|
<a class="dropdown-item" href="#">
|
|
<div class="d-flex">
|
|
<div class="flex-shrink-0 me-3">
|
|
<div class="avatar avatar-online">
|
|
<img src="files/avatar.png" alt class="w-px-40 h-auto rounded-circle" />
|
|
</div>
|
|
</div>
|
|
<div class="flex-grow-1">
|
|
<span class="fw-semibold d-block" id="user_id">User</span>
|
|
</div>
|
|
</div>
|
|
</a>
|
|
</li>
|
|
<li>
|
|
<div class="dropdown-divider"></div>
|
|
</li>
|
|
<li>
|
|
<a class="dropdown-item" href="login.html">
|
|
<i class="bx bx-power-off me-2"></i>
|
|
<span class="align-middle" id="title_logout">Log Out</span>
|
|
</a>
|
|
</li>
|
|
</ul>
|
|
</li>
|
|
<!--/ User -->
|
|
</ul>
|
|
</div>
|
|
</nav>
|
|
|
|
<!-- / Navbar -->
|
|
|
|
<!-- Content wrapper -->
|
|
<div class="content-wrapper">
|
|
<!-- Content -->
|
|
|
|
<div class="container-xxl flex-grow-1 container-p-y">
|
|
|
|
<!-- Basic Bootstrap Table -->
|
|
<div class="card">
|
|
<h5 class="card-header" id="my_devices">My devices</h5>
|
|
<button style="width:30%;float: right;margin-left: 66%;" class="btn btn-info" onclick="addDevice();"
|
|
id="btn_add">Add device</button>
|
|
<!-- button style="width:30%;float: right;margin-left: 66%;" class="btn btn-info" onclick="listDevices();"
|
|
id="btn_add">List</button -->
|
|
<div class="table-responsive text-nowrap">
|
|
<table class="table">
|
|
<thead>
|
|
<tr>
|
|
<th id="title_device">Device</th>
|
|
<th id="title_time">Registered time</th>
|
|
<th id="title_act">Actions</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody class="table-border-bottom-0" id="devices_list">
|
|
<tr>
|
|
<td colspan="3" style="text-align: center;" id="title_empty_list"></td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
<!--/ Basic Bootstrap Table -->
|
|
|
|
</div>
|
|
<!-- / Content -->
|
|
|
|
<!-- Footer -->
|
|
<footer class="content-footer footer bg-footer-theme">
|
|
<div class="container-xxl d-flex flex-wrap justify-content-between py-2 flex-md-row flex-column">
|
|
<div class="mb-2 mb-md-0">
|
|
©
|
|
<script>
|
|
document.write(new Date().getFullYear());
|
|
</script>
|
|
<a href="https://www.amipro.me" target="_blank" class="footer-link fw-bolder">amiPro</a>
|
|
</div>
|
|
<div>
|
|
|
|
<a
|
|
href="mailto:sales@amipro.me?subject=contact"
|
|
target="_blank"
|
|
class="footer-link me-4"
|
|
id="title_contact"
|
|
>Contact</a
|
|
>
|
|
</div>
|
|
</div>
|
|
</footer>
|
|
<!-- / Footer -->
|
|
|
|
<div class="content-backdrop fade"></div>
|
|
</div>
|
|
<!-- Content wrapper -->
|
|
</div>
|
|
<!-- / Layout page -->
|
|
</div>
|
|
|
|
<!-- Overlay -->
|
|
<div class="layout-overlay layout-menu-toggle"></div>
|
|
</div>
|
|
<!-- / Layout wrapper -->
|
|
|
|
<!-- div class="buy-now">
|
|
<a
|
|
href="https://themeselection.com/products/sneat-bootstrap-html-admin-template/"
|
|
target="_blank"
|
|
class="btn btn-danger btn-buy-now"
|
|
>Upgrade to Pro</a
|
|
>
|
|
</div -->
|
|
|
|
<!-- Page JS -->
|
|
|
|
<!-- Place this tag in your head or just before your close body tag. -->
|
|
<script async defer src="https://buttons.github.io/buttons.js"></script>
|
|
</body>
|
|
</html>
|