util: auth: Introduce virAuthAskCredential

The helper uses the user-provided auth callbacks to ask the user. The
helper encapsulates the steps we do to query the user in few places into
a common helper which can be then used further.

Signed-off-by: Peter Krempa <pkrempa@redhat.com>
Reviewed-by: Jonathon Jongsma <jjongsma@redhat.com>
This commit is contained in:
Peter Krempa 2022-12-08 15:55:53 +01:00
parent 7fb0c7418e
commit 95f4879e97
3 changed files with 75 additions and 0 deletions

View File

@ -1850,6 +1850,8 @@ virAuditSend;
# util/virauth.h
virAuthAskCredential;
virAuthConnectCredentialFree;
virAuthGetConfigFilePath;
virAuthGetConfigFilePathURI;
virAuthGetPassword;

View File

@ -31,6 +31,7 @@
#include "virerror.h"
#include "configmake.h"
#include "virauthconfig.h"
#include "virsecureerase.h"
#define VIR_FROM_THIS VIR_FROM_AUTH
@ -283,3 +284,68 @@ virAuthGetPassword(virConnectPtr conn,
return virAuthGetPasswordPath(path, auth, servicename, username, hostname);
}
void
virAuthConnectCredentialFree(virConnectCredential *cred)
{
if (cred->result) {
virSecureErase(cred->result, cred->resultlen);
g_free(cred->result);
}
g_free(cred);
}
/**
* virAuthAskCredential:
* @auth: authentication callback data
* @prompt: question string to ask the user
* @echo: false if user's reply should be considered sensitive and not echoed
*
* Invoke the authentication callback for the connection @auth and ask the user
* the question in @prompt. If @echo is false user's reply should be collected
* as sensitive (user's input not printed on screen).
*/
virConnectCredential *
virAuthAskCredential(virConnectAuthPtr auth,
const char *prompt,
bool echo)
{
g_autoptr(virConnectCredential) ret = g_new0(virConnectCredential, 1);
size_t i;
ret->type = -1;
for (i = 0; i < auth->ncredtype; ++i) {
int type = auth->credtype[i];
if (echo) {
if (type == VIR_CRED_ECHOPROMPT) {
ret->type = type;
break;
}
} else {
if (type == VIR_CRED_PASSPHRASE ||
type == VIR_CRED_NOECHOPROMPT) {
ret->type = type;
break;
}
}
}
if (ret->type == -1) {
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
_("no suitable callback authentication callback was found"));
return NULL;
}
ret->prompt = prompt;
if (auth->cb(ret, 1, auth->cbdata) < 0) {
virReportError(VIR_ERR_OPERATION_FAILED, "%s",
_("failed to retrieve user response for authentication callback"));
return NULL;
}
return g_steal_pointer(&ret);
}

View File

@ -52,3 +52,10 @@ char * virAuthGetPasswordPath(const char *path,
const char *servicename,
const char *username,
const char *hostname);
virConnectCredential *virAuthAskCredential(virConnectAuthPtr auth,
const char *prompt,
bool echo);
void virAuthConnectCredentialFree(virConnectCredential *cred);
G_DEFINE_AUTOPTR_CLEANUP_FUNC(virConnectCredential, virAuthConnectCredentialFree);