/* Chrysalide - Outil d'analyse de fichiers binaires
 * security.c - configuration des paramètres liés à la sécurité
 *
 * Copyright (C) 2025 Cyrille Bagard
 *
 *  This file is part of Chrysalide.
 *
 *  Chrysalide is free software; you can redistribute it and/or modify
 *  it under the terms of the GNU General Public License as published by
 *  the Free Software Foundation; either version 3 of the License, or
 *  (at your option) any later version.
 *
 *  Chrysalide is distributed in the hope that it will be useful,
 *  but WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 *  GNU General Public License for more details.
 *
 *  You should have received a copy of the GNU General Public License
 *  along with Chrysalide.  If not, see <http://www.gnu.org/licenses/>.
 */


#include "security.h"


#include <i18n.h>


#include "security-int.h"
#include "../../../core/global.h"
#include "../../../core/logs.h"
#include "../../../gtkext/helpers.h"



/* ---------------------- GESTION DE L'ENSEMBLE DES PARAMETRES ---------------------- */


/* Procède à l'initialisation de classe des configurations. */
static void gtk_security_tweak_panel_class_init(GtkSecurityTweakPanelClass *);

/* Procède à l'initialisation des configurations de sécurité. */
static void gtk_security_tweak_panel_init(GtkSecurityTweakPanel *);

/* Supprime toutes les références externes. */
static void gtk_security_tweak_panel_dispose(GObject *);

/* Procède à la libération totale de la mémoire. */
static void gtk_security_tweak_panel_finalize(GObject *);



/* ---------------------- CONSERVATION SECURISEE DE PARAMETRES ---------------------- */


/* Initalise la partie relative au stockage sécurisé. */
static void gtk_security_tweak_panel_init_secret_storage(GtkSecurityTweakPanel *);

/* Supprime toutes les références externes. */
static void gtk_security_tweak_panel_dispose_secret_storage(GtkSecurityTweakPanel *);

/* Note le changement de verrouillage du stockage sécurisé. */
static void gtk_security_tweak_panel_on_secret_storage_lock_update(GSecretStorage *, GtkSecurityTweakPanel *);

/* Réagit à un changement dans l'édition d'un mot de passe. */
static void gtk_security_tweak_panel_on_new_passwords_changed(GtkEditable *, GtkSecurityTweakPanel *);

/* Réagit à une demande de création de mot de passe. */
static void gtk_security_tweak_panel_on_set_password_clicked(GtkButton *, GtkSecurityTweakPanel *);

/* Réagit à une demande de changement de mot de passe. */
static void gtk_security_tweak_panel_on_change_password_clicked(GtkButton *, GtkSecurityTweakPanel *);

/* Réagit à une demande de suppression de mot de passe. */
static void gtk_security_tweak_panel_on_remove_password_clicked(GtkButton *, GtkSecurityTweakPanel *);



/* ---------------------------------------------------------------------------------- */
/*                        GESTION DE L'ENSEMBLE DES PARAMETRES                        */
/* ---------------------------------------------------------------------------------- */


/* Indique le type du composant d'édition des paramètres de sécurité. */
G_DEFINE_TYPE(GtkSecurityTweakPanel, gtk_security_tweak_panel, GTK_TYPE_BOX);


/******************************************************************************
*                                                                             *
*  Paramètres  : class = classe GTK à initialiser.                            *
*                                                                             *
*  Description : Procède à l'initialisation de classe des configurations.     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_class_init(GtkSecurityTweakPanelClass *class)
{
    GObjectClass *object;                   /* Plus haut niveau équivalent */
    GtkWidgetClass *widget;                 /* Classe de haut niveau       */

    object = G_OBJECT_CLASS(class);

    object->dispose = gtk_security_tweak_panel_dispose;
    object->finalize = gtk_security_tweak_panel_finalize;

    widget = GTK_WIDGET_CLASS(class);

    gtk_widget_class_set_template_from_resource(widget, "/re/chrysalide/framework/gui/dialogs/prefs/security.ui");

    /* Stockage sécurisé */

    gtk_widget_class_bind_template_callback_full(widget, BUILDER_CB(gtk_security_tweak_panel_on_new_passwords_changed));
    gtk_widget_class_bind_template_callback_full(widget, BUILDER_CB(gtk_security_tweak_panel_on_set_password_clicked));
    gtk_widget_class_bind_template_callback_full(widget, BUILDER_CB(gtk_security_tweak_panel_on_change_password_clicked));
    gtk_widget_class_bind_template_callback_full(widget, BUILDER_CB(gtk_security_tweak_panel_on_remove_password_clicked));

    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, current_primary_passwd);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, new_primary_passwd);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, new_primary_passwd_lbl);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, new_primary_passwd_2);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, new_primary_passwd_2_lbl);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, set_password);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, change_password);
    gtk_widget_class_bind_template_child(widget, GtkSecurityTweakPanel, remove_password);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : panel = composant GTK à initialiser.                         *
*                                                                             *
*  Description : Procède à l'initialisation des configurations de sécurité.   *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_init(GtkSecurityTweakPanel *panel)
{
    gtk_widget_init_template(GTK_WIDGET(panel));

    gtk_security_tweak_panel_init_secret_storage(panel);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : object = instance d'objet GLib à traiter.                    *
*                                                                             *
*  Description : Supprime toutes les références externes.                     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_dispose(GObject *object)
{
    GtkSecurityTweakPanel *panel;           /* Version spécialisée         */

    panel = GTK_SECURITY_TWEAK_PANEL(object);

    gtk_security_tweak_panel_dispose_secret_storage(panel);

    gtk_widget_dispose_template(GTK_WIDGET(panel), GTK_TYPE_SECURITY_TWEAK_PANEL);

    G_OBJECT_CLASS(gtk_security_tweak_panel_parent_class)->dispose(object);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : object = instance d'objet GLib à traiter.                    *
*                                                                             *
*  Description : Procède à la libération totale de la mémoire.                *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_finalize(GObject *object)
{
    G_OBJECT_CLASS(gtk_security_tweak_panel_parent_class)->finalize(object);

}



/* ---------------------------------------------------------------------------------- */
/*                        CONSERVATION SECURISEE DE PARAMETRES                        */
/* ---------------------------------------------------------------------------------- */


/******************************************************************************
*                                                                             *
*  Paramètres  : panel = composant GTK à initialiser.                         *
*                                                                             *
*  Description : Initalise la partie relative au stockage sécurisé.           *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_init_secret_storage(GtkSecurityTweakPanel *panel)
{
    panel->storage = get_secret_storage();

    g_signal_connect(panel->storage, "lock-update",
                     G_CALLBACK(gtk_security_tweak_panel_on_secret_storage_lock_update), panel);

    gtk_security_tweak_panel_on_secret_storage_lock_update(panel->storage, panel);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : panel = instance d'objet GLib à traiter.                     *
*                                                                             *
*  Description : Supprime toutes les références externes.                     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_dispose_secret_storage(GtkSecurityTweakPanel *panel)
{
    if (panel->storage != NULL)
        g_signal_handlers_disconnect_by_func(panel->storage,
                                             gtk_security_tweak_panel_on_secret_storage_lock_update, panel);

    g_clear_object(&panel->storage);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : storage = gardien des secrets impliqué.                      *
*                panel   = support de la partie relative au stockage sécurisé.*
*                                                                             *
*  Description : Note le changement de verrouillage du stockage sécurisé.     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_on_secret_storage_lock_update(GSecretStorage *storage, GtkSecurityTweakPanel *panel)
{
    bool has_key;                           /* Existence d'une clef ?      */

    has_key = g_secret_storage_has_key(panel->storage);

    gtk_widget_set_visible(panel->new_primary_passwd, has_key);
    gtk_widget_set_visible(panel->new_primary_passwd_lbl, has_key);

    gtk_widget_set_visible(panel->new_primary_passwd_2, has_key);
    gtk_widget_set_visible(panel->new_primary_passwd_2_lbl, has_key);

    gtk_widget_set_visible(panel->set_password, !has_key);
    gtk_widget_set_visible(panel->change_password, has_key);
    gtk_widget_set_visible(panel->remove_password, has_key);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : editable = zone de saisie bouton GTK concerné par l'appel.   *
*                panel    = support de la partie liée au stockage sécurisé.   *
*                                                                             *
*  Description : Réagit à un changement dans l'édition d'un mot de passe.     *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_on_new_passwords_changed(GtkEditable *editable, GtkSecurityTweakPanel *panel)
{
    const char *new_passwd;                 /* Nouveau mot de passe #1     */
    const char *new_passwd_2;               /* Nouveau mot de passe #2     */
    bool status;                            /* Bilan de l'opération        */

    new_passwd = gtk_editable_get_text(GTK_EDITABLE(panel->new_primary_passwd));

    new_passwd_2 = gtk_editable_get_text(GTK_EDITABLE(panel->new_primary_passwd_2));

    status = (strcmp(new_passwd, new_passwd_2) == 0);

    gtk_widget_set_sensitive(panel->change_password, status);

}


/******************************************************************************
*                                                                             *
*  Paramètres  : button = bouton GTK concerné par l'appel.                    *
*                panel  = support de la partie relative au stockage sécurisé. *
*                                                                             *
*  Description : Réagit à une demande de création de mot de passe.            *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_on_set_password_clicked(GtkButton *button, GtkSecurityTweakPanel *panel)
{
    const char *passwd;                     /* Mot de passe à considérer   */
    bool status;                            /* Bilan de l'opération        */

    passwd = gtk_editable_get_text(GTK_EDITABLE(panel->current_primary_passwd));

    status = g_secret_storage_set_password(panel->storage, passwd);

    log_variadic_message(LMT_INFO, _("Setting password for secret storage: %s"),
                         status ? _("success") : _("failed"));

}


/******************************************************************************
*                                                                             *
*  Paramètres  : button = bouton GTK concerné par l'appel.                    *
*                panel  = support de la partie relative au stockage sécurisé. *
*                                                                             *
*  Description : Réagit à une demande de changement de mot de passe.          *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_on_change_password_clicked(GtkButton *button, GtkSecurityTweakPanel *panel)
{
    const char *old_passwd;                 /* Ancien mot de passe         */
    const char *new_passwd;                 /* Nouveau mot de passe        */
    bool status;                            /* Bilan de l'opération        */

    old_passwd = gtk_editable_get_text(GTK_EDITABLE(panel->current_primary_passwd));

    new_passwd = gtk_editable_get_text(GTK_EDITABLE(panel->new_primary_passwd));

    status = g_secret_storage_change_password(panel->storage, old_passwd, new_passwd);

    log_variadic_message(LMT_INFO, _("Changed password for secret storage: %s"),
                         status ? _("success") : _("failed"));

}


/******************************************************************************
*                                                                             *
*  Paramètres  : button = bouton GTK concerné par l'appel.                    *
*                panel  = support de la partie relative au stockage sécurisé. *
*                                                                             *
*  Description : Réagit à une demande de suppression de mot de passe.         *
*                                                                             *
*  Retour      : -                                                            *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static void gtk_security_tweak_panel_on_remove_password_clicked(GtkButton *button, GtkSecurityTweakPanel *panel)
{
    const char *passwd;                     /* Mot de passe à considérer   */
    bool status;                            /* Bilan de l'opération        */

    passwd = gtk_editable_get_text(GTK_EDITABLE(panel->current_primary_passwd));

    status = g_secret_storage_remove_password(panel->storage, passwd);

    log_variadic_message(LMT_INFO, _("Removed password for secret storage: %s"),
                         status ? _("success") : _("failed"));

}