yubioath-flutter/qml/YubiKey.qml

163 lines
5.0 KiB
QML
Raw Normal View History

import QtQuick 2.0
import io.thp.pyotherside 1.4
2017-01-30 16:59:58 +03:00
// @disable-check M300
Python {
id: py
property int nDevices
property bool hasDevice
property string name
property string version
property string serial
property var features: []
property var connections: []
property var credentials: null
property int nextRefresh: 0
property var enabled: []
property bool ready: false
property var queue: []
Component.onCompleted: {
2017-01-30 16:59:58 +03:00
importModule('site', function () {
call('site.addsitedir', [appDir + '/pymodules'], function () {
addImportPath(urlPrefix + '/py')
importModule('yubikey', function () {
ready = true
2017-01-30 16:59:58 +03:00
do_call('yubikey.controller.get_features', [],
function (res) {
features = res
for (var i in queue) {
do_call(queue[i][0], queue[i][1],
queue[i][2])
}
queue = []
})
})
})
})
}
onError: {
console.log('Python error: ' + traceback)
}
function do_call(func, args, cb) {
if (!ready) {
queue.push([func, args, cb])
} else {
call(func, args.map(JSON.stringify), function (json) {
if (cb) {
cb(json ? JSON.parse(json) : undefined)
}
})
}
}
function refresh() {
do_call('yubikey.controller.count_devices', [], function (n) {
nDevices = n
if (nDevices == 1) {
do_call('yubikey.controller.refresh', [], function (dev) {
hasDevice = dev !== undefined && dev !== null
name = dev ? dev.name : ''
version = dev ? dev.version : ''
serial = dev ? dev.serial : ''
enabled = dev ? dev.enabled : []
connections = dev ? dev.connections : []
var now = Math.floor(Date.now() / 1000)
if (nextRefresh < now) {
2017-02-03 14:24:41 +03:00
refreshCredentials()
}
})
} else if (hasDevice) {
hasDevice = false
credentials = null
nextRefresh = 0
}
})
2017-01-30 16:59:58 +03:00
}
2017-02-03 14:24:41 +03:00
function refreshCredentials() {
var now = Math.floor(Date.now() / 1000)
2017-01-30 16:59:58 +03:00
if (enabled.indexOf('CCID') != -1) {
2017-02-03 14:24:41 +03:00
do_call('yubikey.controller.refresh_credentials', [now],
handleCredentials)
2017-01-30 16:59:58 +03:00
}
}
2017-02-01 18:27:45 +03:00
function calculate(credential) {
var now = Math.floor(Date.now() / 1000)
2017-02-03 14:24:41 +03:00
do_call('yubikey.controller.calculate', [credential, now],
updateCredential)
2017-02-01 18:27:45 +03:00
}
function updateCredential(cred) {
2017-02-02 13:08:02 +03:00
var result = []
for (var i = 0; i < credentials.length; i++) {
if (credentials[i].name === cred.name) {
result.push(cred)
} else {
result.push(credentials[i])
}
}
credentials = result
2017-02-01 18:27:45 +03:00
}
function handleCredentials(creds) {
2017-01-31 11:57:44 +03:00
var result = []
var minExpiration = (Date.now() / 1000) + 10000
2017-01-31 11:57:44 +03:00
for (var i = 0; i < creds.length; i++) {
2017-02-01 18:27:45 +03:00
var cred = creds[i]
// Update min expiration
if (cred.expiration && cred.expiration < minExpiration) {
minExpiration = cred.expiration
}
// Touch credentials should only be replaced by user
if (credentialExists(cred.name) && cred.touch) {
result.push(getCredential(cred.name))
continue
}
// HOTP credentials should only be replaced by user
if (credentialExists(cred.name) && cred.oath_type === 'hotp') {
result.push(getCredential(cred.name))
continue
}
// TOTP credentials should be updated
2017-01-31 11:57:44 +03:00
result.push(cred)
}
nextRefresh = minExpiration
credentials = result
2017-01-31 11:57:44 +03:00
}
function getCredential(name) {
for (var i = 0; i < credentials.length; i++) {
if (credentials[i].name === name) {
return credentials[i]
}
}
}
function credentialExists(name) {
if (credentials != null) {
for (var i = 0; i < credentials.length; i++) {
if (credentials[i].name === name) {
return true
}
}
}
return false
}
2017-02-03 14:24:41 +03:00
function addCredential(name, key, oathType, digits, algorithm, touch) {
do_call('yubikey.controller.add_credential',
[name, key, oathType, digits, algorithm, touch])
}
function set_mode(connections, cb) {
do_call('yubikey.controller.set_mode', [connections], cb)
}
}