Blob Blame History Raw
/* SPDX-License-Identifier: GPL-2.0-or-later */
/*
 * Copyright (C) 2013 Red Hat, Inc.
 */

/**
 * SECTION:nmt-password-dialog
 * @short_description: A password dialog
 *
 * #NmtPasswordDialog is the password dialog used to get connection
 * secrets when activating a connection.
 */

#include "libnm/nm-default-client.h"

#include "nmt-password-dialog.h"
#include "nm-secret-agent-simple.h"
#include "nmtui.h"

G_DEFINE_TYPE(NmtPasswordDialog, nmt_password_dialog, NMT_TYPE_NEWT_FORM)

#define NMT_PASSWORD_DIALOG_GET_PRIVATE(o) \
    (G_TYPE_INSTANCE_GET_PRIVATE((o), NMT_TYPE_PASSWORD_DIALOG, NmtPasswordDialogPrivate))

typedef struct {
    char *     request_id;
    char *     prompt;
    GPtrArray *secrets;
    GPtrArray *entries;

    NmtNewtWidget *ok, *cancel;
    NmtNewtWidget *last_entry;
    NmtNewtWidget *secret_grid;

    gboolean succeeded;
} NmtPasswordDialogPrivate;

enum {
    PROP_0,
    PROP_REQUEST_ID,
    PROP_PROMPT,
    PROP_SECRETS,

    LAST_PROP
};

/**
 * nmt_password_dialog_new:
 * @request_id: the request ID from the #NMSecretAgentSimple
 * @title: the dialog title
 * @prompt: the prompt text to display
 * @secrets: (element-type #NMSecretAgentSimpleSecret): the secrets requested
 *
 * Creates a new #NmtPasswordDialog to request passwords from
 * the user.
 *
 * Returns: a new #NmtPasswordDialog.
 */
NmtNewtForm *
nmt_password_dialog_new(const char *request_id,
                        const char *title,
                        const char *prompt,
                        GPtrArray * secrets)
{
    return g_object_new(NMT_TYPE_PASSWORD_DIALOG,
                        "request-id",
                        request_id,
                        "title",
                        title,
                        "prompt",
                        prompt,
                        "secrets",
                        secrets,
                        "escape-exits",
                        TRUE,
                        NULL);
}

static void
nmt_password_dialog_init(NmtPasswordDialog *dialog)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);

    priv->entries = g_ptr_array_new();
}

static void
maybe_save_input_and_exit(NmtNewtWidget *widget, gpointer dialog)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);
    int                       i;

    /* This gets invoked when the user types Return in the final entry,
     * but the form may not be fully valid in that case.
     */
    if (!nmt_newt_widget_get_valid(priv->secret_grid))
        return;

    priv->succeeded = TRUE;

    for (i = 0; i < priv->secrets->len; i++) {
        NMSecretAgentSimpleSecret *secret = priv->secrets->pdata[i];

        g_free(secret->value);
        g_object_get(priv->entries->pdata[i], "text", &secret->value, NULL);
    }

    nmt_newt_form_quit(nmt_newt_widget_get_form(widget));
}

static void
nmt_password_dialog_constructed(GObject *object)
{
    NmtPasswordDialog *       dialog = NMT_PASSWORD_DIALOG(object);
    NmtPasswordDialogPrivate *priv   = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);
    NmtNewtWidget *           widget;
    NmtNewtGrid *             grid, *secret_grid;
    NmtNewtButtonBox *        bbox;
    int                       i;

    widget = nmt_newt_grid_new();
    nmt_newt_form_set_content(NMT_NEWT_FORM(dialog), widget);
    grid = NMT_NEWT_GRID(widget);

    widget = nmt_newt_textbox_new(0, 60);
    nmt_newt_textbox_set_text(NMT_NEWT_TEXTBOX(widget), priv->prompt);
    nmt_newt_grid_add(grid, widget, 0, 0);

    widget = nmt_newt_grid_new();
    nmt_newt_grid_add(grid, widget, 0, 1);
    nmt_newt_widget_set_padding(widget, 0, 1, 0, 1);
    priv->secret_grid = widget;
    secret_grid       = NMT_NEWT_GRID(widget);

    for (i = 0; i < priv->secrets->len; i++) {
        NMSecretAgentSimpleSecret *secret = priv->secrets->pdata[i];
        NmtNewtEntryFlags          flags;

        widget = nmt_newt_label_new(secret->pretty_name);
        nmt_newt_grid_add(secret_grid, widget, 0, i);
        nmt_newt_widget_set_padding(widget, 4, 0, 1, 0);

        flags = NMT_NEWT_ENTRY_NONEMPTY;
        if (secret->is_secret)
            flags |= NMT_NEWT_ENTRY_PASSWORD;
        widget = nmt_newt_entry_new(30, flags);
        if (secret->value)
            nmt_newt_entry_set_text(NMT_NEWT_ENTRY(widget), secret->value);
        nmt_newt_grid_add(secret_grid, widget, 1, i);
        g_ptr_array_add(priv->entries, widget);

        if (i == priv->secrets->len - 1) {
            priv->last_entry = widget;
            g_signal_connect(widget, "activated", G_CALLBACK(maybe_save_input_and_exit), dialog);
        }
    }

    widget = nmt_newt_button_box_new(NMT_NEWT_BUTTON_BOX_HORIZONTAL);
    nmt_newt_grid_add(grid, widget, 0, 2);
    bbox = NMT_NEWT_BUTTON_BOX(widget);

    priv->cancel = nmt_newt_button_box_add_end(NMT_NEWT_BUTTON_BOX(bbox), _("Cancel"));
    nmt_newt_widget_set_exit_on_activate(priv->cancel, TRUE);

    priv->ok = nmt_newt_button_box_add_end(NMT_NEWT_BUTTON_BOX(bbox), _("OK"));
    g_signal_connect(priv->ok, "activated", G_CALLBACK(maybe_save_input_and_exit), dialog);
    g_object_bind_property(priv->secret_grid,
                           "valid",
                           priv->ok,
                           "sensitive",
                           G_BINDING_SYNC_CREATE);

    G_OBJECT_CLASS(nmt_password_dialog_parent_class)->constructed(object);
}

static void
nmt_password_dialog_finalize(GObject *object)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(object);

    g_free(priv->request_id);
    g_free(priv->prompt);
    nm_clear_pointer(&priv->entries, g_ptr_array_unref);
    nm_clear_pointer(&priv->secrets, g_ptr_array_unref);

    G_OBJECT_CLASS(nmt_password_dialog_parent_class)->finalize(object);
}

static void
nmt_password_dialog_set_property(GObject *     object,
                                 guint         prop_id,
                                 const GValue *value,
                                 GParamSpec *  pspec)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(object);

    switch (prop_id) {
    case PROP_REQUEST_ID:
        priv->request_id = g_value_dup_string(value);
        break;
    case PROP_PROMPT:
        priv->prompt = g_value_dup_string(value);
        break;
    case PROP_SECRETS:
        priv->secrets = g_value_dup_boxed(value);
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
        break;
    }
}

static void
nmt_password_dialog_get_property(GObject *object, guint prop_id, GValue *value, GParamSpec *pspec)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(object);

    switch (prop_id) {
    case PROP_REQUEST_ID:
        g_value_set_string(value, priv->request_id);
        break;
    case PROP_PROMPT:
        g_value_set_string(value, priv->prompt);
        break;
    case PROP_SECRETS:
        g_value_set_boxed(value, priv->secrets);
        break;
    default:
        G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
        break;
    }
}

static void
nmt_password_dialog_class_init(NmtPasswordDialogClass *dialog_class)
{
    GObjectClass *object_class = G_OBJECT_CLASS(dialog_class);

    g_type_class_add_private(dialog_class, sizeof(NmtPasswordDialogPrivate));

    /* virtual methods */
    object_class->constructed  = nmt_password_dialog_constructed;
    object_class->set_property = nmt_password_dialog_set_property;
    object_class->get_property = nmt_password_dialog_get_property;
    object_class->finalize     = nmt_password_dialog_finalize;

    /**
     * NmtPasswordDialog:request-id:
     *
     * The request ID from the #NMSecretAgentSimple
     */
    g_object_class_install_property(
        object_class,
        PROP_REQUEST_ID,
        g_param_spec_string("request-id",
                            "",
                            "",
                            NULL,
                            G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
    /**
     * NmtPasswordDialog:prompt:
     *
     * The prompt text.
     */
    g_object_class_install_property(
        object_class,
        PROP_PROMPT,
        g_param_spec_string("prompt",
                            "",
                            "",
                            NULL,
                            G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
    /**
     * NmtPasswordDialog:secrets:
     *
     * The array of request secrets
     *
     * Element-Type: #NMSecretAgentSimpleSecret.
     */
    g_object_class_install_property(
        object_class,
        PROP_SECRETS,
        g_param_spec_boxed("secrets",
                           "",
                           "",
                           G_TYPE_PTR_ARRAY,
                           G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS));
}

/**
 * nmt_password_dialog_succeeded:
 * @dialog: the #NmtPasswordDialog
 *
 * After the dialog has exited, returns %TRUE if the user clicked
 * "OK", %FALSE if "Cancel".
 *
 * Returns: whether the dialog succeeded.
 */
gboolean
nmt_password_dialog_succeeded(NmtPasswordDialog *dialog)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);

    return priv->succeeded;
}

/**
 * nmt_password_dialog_get_request_id:
 * @dialog: the #NmtPasswordDialog
 *
 * Gets the dialog's request ID.
 *
 * Returns: the dialog's request ID.
 */
const char *
nmt_password_dialog_get_request_id(NmtPasswordDialog *dialog)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);

    return priv->request_id;
}

/**
 * nmt_password_dialog_get_secrets:
 * @dialog: the #NmtPasswordDialog
 *
 * Gets the dialog's secrets array.
 *
 * Returns: (transfer none): the dialog's secrets array.
 */
GPtrArray *
nmt_password_dialog_get_secrets(NmtPasswordDialog *dialog)
{
    NmtPasswordDialogPrivate *priv = NMT_PASSWORD_DIALOG_GET_PRIVATE(dialog);

    return priv->secrets;
}