/* Chrysalide - Outil d'analyse de fichiers binaires
 * cluster.c - équivalent Python du fichier "glibext/gtkext/graph/cluster.c"
 *
 * Copyright (C) 2019 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 this program; if not, write to the Free Software
 *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */


#include "cluster.h"


#include <malloc.h>
#include <pygobject.h>


#include <i18n.h>
#include <gtkext/graph/cluster.h>
#include <plugins/dt.h>


#include "../../access.h"
#include "../../helpers.h"
#include "../../struct.h"
#include "../../analysis/binary.h"
#include "../../analysis/block.h"



/* Recherche le groupe de blocs avec un bloc donné comme chef. */
static PyObject *py_graph_cluster_find_by_block(PyObject *, PyObject *);

/* Recherche le groupe de blocs avec un composant comme chef. */
static PyObject *py_graph_cluster_find_by_widget(PyObject *, PyObject *);

/* Recherche le groupe de blocs avec une cible particulière. */
static PyObject *py_graph_cluster_find(PyObject *, PyObject *);

/* Construit un graphique à partir de blocs basiques. */
static PyObject *py_graph_cluster_bootstrap(PyObject *, PyObject *);

/* Collecte tous les chefs de file de blocs de code. */
static PyObject *py_graph_cluster_collect(PyObject *, PyObject *);

/* Collecte tous les liens de chefs de file de blocs de code. */
static PyObject *py_graph_cluster_collect_edges(PyObject *, PyObject *);

/* Fournit le bloc de code principal du groupe. */
static PyObject *py_graph_cluster_get_block(PyObject *, void *);

/* Fournit le composant graphique principal du groupe. */
static PyObject *py_graph_cluster_get_widget(PyObject *, void *);

/* Fournit l'emplacement prévu pour un chef de file de blocs. */
static PyObject *py_graph_cluster_get_allocation(PyObject *, void *);

/* Détermine l'emplacement requis d'un ensemble de blocs. */
static PyObject *py_graph_cluster_get_needed_alloc(PyObject *, void *);



/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Recherche le groupe de blocs avec un bloc donné comme chef.  *
*                                                                             *
*  Retour      : Groupe trouvé ou None en cas d'échec.                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_find_by_block(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Instance à retourner        */
    GCodeBlock *block;                      /* Bloc de code à retrouver    */
    int ret;                                /* Bilan de lecture des args.  */
    GGraphCluster *cluster;                 /* Ensemble mis en place       */
    GGraphCluster *found;                   /* Ensemble graphique trouvé   */

    ret = PyArg_ParseTuple(args, "O&",
                           convert_to_code_block, &block);
    if (!ret) return NULL;

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    found = g_graph_cluster_find_by_block(cluster, block);

    if (found != NULL)
    {
        result = pygobject_new(G_OBJECT(found));
        g_object_unref(G_OBJECT(found));
    }
    else
    {
        result = Py_None;
        Py_INCREF(result);
    }

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Recherche le groupe de blocs avec un composant comme chef.   *
*                                                                             *
*  Retour      : Groupe trouvé ou None en cas d'échec.                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_find_by_widget(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Instance à retourner        */
    PyObject *gtk_mod;                      /* Module Python Gtk           */
    PyObject *type;                         /* Module "GtkWidget"          */
    PyObject *widget_obj;                   /* Composant GTK en Python     */
    int ret;                                /* Bilan de lecture des args.  */
    GGraphCluster *cluster;                 /* Ensemble mis en place       */
    GtkWidget *widget;                      /* Composant GTK à retrouver   */
    GGraphCluster *found;                   /* Ensemble graphique trouvé   */

    gtk_mod = PyImport_ImportModule("gi.repository.Gtk");

    if (gtk_mod == NULL)
    {
        PyErr_SetString(PyExc_TypeError, "unable to find the Gtk Python module");
        return NULL;
    }

    type = PyObject_GetAttrString(gtk_mod, "Widget");

    Py_DECREF(gtk_mod);

    ret = PyArg_ParseTuple(args, "O!", type, &widget_obj);

    Py_DECREF(type);

    if (!ret) return NULL;

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    widget = GTK_WIDGET(pygobject_get(widget_obj));

    found = g_graph_cluster_find_by_widget(cluster, widget);

    if (found != NULL)
    {
        result = pygobject_new(G_OBJECT(found));
        g_object_unref(G_OBJECT(found));
    }
    else
    {
        result = Py_None;
        Py_INCREF(result);
    }

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Recherche le groupe de blocs avec une cible particulière.    *
*                                                                             *
*  Retour      : Groupe trouvé ou None en cas d'échec.                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_find(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Instance à retourner        */
    PyObject *block_or_widget;              /* Objet Python fourni         */
    int ret;                                /* Bilan de lecture des args.  */

    ret = PyArg_ParseTuple(args, "O", &block_or_widget);
    if (!ret) return NULL;

    ret = PyObject_IsInstance(block_or_widget, (PyObject *)get_python_code_block_type());

    if (ret == 1)
        result = py_graph_cluster_find_by_block(self, args);

    else
    {
        PyErr_Clear();
        result = py_graph_cluster_find_by_widget(self, args);
    }

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Construit un graphique à partir de blocs basiques.           *
*                                                                             *
*  Retour      : Structure mise en place.                                     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_bootstrap(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Instance à retourner        */
    GLoadedBinary *binary;                  /* Binaire chargé avec contenu */
    GBlockList *list;                       /* Liste de blocs de code      */
    int ret;                                /* Bilan de lecture des args.  */
    GGraphCluster *cluster;                 /* Ensemble mis en place       */

    ret = PyArg_ParseTuple(args, "O&O&",
                           convert_to_loaded_binary, &binary,
                           convert_to_block_list_with_ref, &list);
    if (!ret) return NULL;

    cluster = bootstrap_graph_cluster(binary, list, NULL);

    if (cluster != NULL)
    {
        result = pygobject_new(G_OBJECT(cluster));
        g_object_unref(G_OBJECT(cluster));
    }
    else
    {
        result = Py_None;
        Py_INCREF(result);
    }

    g_object_unref(G_OBJECT(list));

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Collecte tous les chefs de file de blocs de code.            *
*                                                                             *
*  Retour      : Liste de graphiques de blocs rassemblés.                     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_collect(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Liste à retourner           */
    GGraphCluster *root;                    /* Chef de file à analyser     */
    size_t count;                           /* Taille de la liste          */
    GGraphCluster **list;                   /* Liste constituée            */
    size_t i;                               /* Boucle de parcours          */
    PyObject *item;                         /* Instance à transmettre      */

    root = G_GRAPH_CLUSTER(pygobject_get(self));

    list = collect_graph_clusters(root, &count);

    result = PyTuple_New(count);

    for (i = 0; i < count; i++)
    {
        item = pygobject_new(G_OBJECT(list[i]));
        g_object_unref(G_OBJECT(list[i]));

        PyTuple_SetItem(result, i, item);

    }

    if (list != NULL)
        free(list);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self = NULL car méthode statique.                            *
*                args = arguments fournis pour l'appel.                       *
*                                                                             *
*  Description : Collecte tous les liens de chefs de file de blocs de code.   *
*                                                                             *
*  Retour      : Liste de liens graphiques de blocs rassemblés.               *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_collect_edges(PyObject *self, PyObject *args)
{
    PyObject *result;                       /* Liste à retourner           */
    GGraphCluster *root;                    /* Chef de file à analyser     */
    size_t count;                           /* Taille de la liste          */
    GGraphEdge **list;                      /* Liste constituée            */
    size_t i;                               /* Boucle de parcours          */
    PyObject *item;                         /* Instance à transmettre      */

    root = G_GRAPH_CLUSTER(pygobject_get(self));

    list = collect_graph_cluster_edges(root, &count);

    result = PyTuple_New(count);

    for (i = 0; i < count; i++)
    {
        item = pygobject_new(G_OBJECT(list[i]));
        g_object_unref(G_OBJECT(list[i]));

        PyTuple_SetItem(result, i, item);

    }

    if (list != NULL)
        free(list);

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self    = objet Python concerné par l'appel.                 *
*                closure = non utilisé ici.                                   *
*                                                                             *
*  Description : Fournit le bloc de code principal du groupe.                 *
*                                                                             *
*  Retour      : Bloc de code associé.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_get_block(PyObject *self, void *closure)
{
    PyObject *result;                       /* Construction à retourner    */
    GGraphCluster *cluster;                 /* Version GLib du type        */
    GCodeBlock *block;                      /* Bloc de code associé        */

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    block = g_graph_cluster_get_block(cluster);

    result = pygobject_new(G_OBJECT(block));

    g_object_unref(G_OBJECT(block));

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self    = objet Python concerné par l'appel.                 *
*                closure = non utilisé ici.                                   *
*                                                                             *
*  Description : Fournit le composant graphique principal du groupe.          *
*                                                                             *
*  Retour      : Composant graphique principal utilisé.                       *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_get_widget(PyObject *self, void *closure)
{
    PyObject *result;                       /* Construction à retourner    */
    GGraphCluster *cluster;                 /* Version GLib du type        */
    GtkWidget *widget;                      /* Composant graphique associé */

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    widget = g_graph_cluster_get_widget(cluster);

    result = new_pygobject_widget(widget);

    g_object_unref(G_OBJECT(widget));

    return result;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self    = objet Python concerné par l'appel.                 *
*                closure = non utilisé ici.                                   *
*                                                                             *
*  Description : Fournit l'emplacement prévu pour un chef de file de blocs.   *
*                                                                             *
*  Retour      : Emplacement idéal pour l'affichage.                          *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_get_allocation(PyObject *self, void *closure)
{
    PyObject *result;                       /* Construction à retourner    */
    GGraphCluster *cluster;                 /* Version GLib du type        */
    GtkAllocation alloc;                    /* Aire à convertir en Python  */
    PyTypeObject *base;                     /* Modèle d'objet à créer      */
    PyObject *attrib;                       /* Attribut à constituer       */
    int ret;                                /* Bilan d'une mise en place   */

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    g_graph_cluster_get_allocation(cluster, &alloc);

    base = get_python_py_struct_type();

    result = PyObject_CallFunction((PyObject *)base, NULL);
    assert(result != NULL);

#define TRANSLATE_ALLOC_FIELD(_n, _v)                       \
    do                                                      \
    {                                                       \
        attrib = PyLong_FromUnsignedLongLong(_v);           \
        ret = PyDict_SetItemString(result, _n, attrib);     \
        if (ret != 0) goto failed;                          \
    }                                                       \
    while (0);

    TRANSLATE_ALLOC_FIELD("x", alloc.x);
    TRANSLATE_ALLOC_FIELD("y", alloc.y);
    TRANSLATE_ALLOC_FIELD("width", alloc.width);
    TRANSLATE_ALLOC_FIELD("height", alloc.height);

    return result;

 failed:

    Py_DECREF(result);

    return NULL;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : self    = objet Python concerné par l'appel.                 *
*                closure = non utilisé ici.                                   *
*                                                                             *
*  Description : Détermine l'emplacement requis d'un ensemble de blocs.       *
*                                                                             *
*  Retour      : Emplacement idéal pour l'affichage.                          *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

static PyObject *py_graph_cluster_get_needed_alloc(PyObject *self, void *closure)
{
    PyObject *result;                       /* Construction à retourner    */
    GGraphCluster *cluster;                 /* Version GLib du type        */
    GtkAllocation alloc;                    /* Aire à convertir en Python  */
    PyTypeObject *base;                     /* Modèle d'objet à créer      */
    PyObject *attrib;                       /* Attribut à constituer       */
    int ret;                                /* Bilan d'une mise en place   */

    cluster = G_GRAPH_CLUSTER(pygobject_get(self));

    g_graph_cluster_compute_needed_alloc(cluster, &alloc);

    base = get_python_py_struct_type();

    result = PyObject_CallFunction((PyObject *)base, NULL);
    assert(result != NULL);

#define TRANSLATE_ALLOC_FIELD(_n, _v)                       \
    do                                                      \
    {                                                       \
        attrib = PyLong_FromUnsignedLongLong(_v);           \
        ret = PyDict_SetItemString(result, _n, attrib);     \
        if (ret != 0) goto failed;                          \
    }                                                       \
    while (0);

    TRANSLATE_ALLOC_FIELD("x", alloc.x);
    TRANSLATE_ALLOC_FIELD("y", alloc.y);
    TRANSLATE_ALLOC_FIELD("width", alloc.width);
    TRANSLATE_ALLOC_FIELD("height", alloc.height);

    return result;

 failed:

    Py_DECREF(result);

    return NULL;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : -                                                            *
*                                                                             *
*  Description : Fournit un accès à une définition de type à diffuser.        *
*                                                                             *
*  Retour      : Définition d'objet pour Python.                              *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

PyTypeObject *get_python_graph_cluster_type(void)
{
    static PyMethodDef py_graph_cluster_methods[] = {
        {
            "find_by_block", py_graph_cluster_find_by_block,
            METH_VARARGS,
            "find_by_block(block, /)\n--\n\nFind the cluster associated with a given code block."
        },
        {
            "find_by_widget", py_graph_cluster_find_by_widget,
            METH_VARARGS,
            "find_by_widget(widget, /)\n--\n\nFind the cluster associated with a given GTK widget."
        },
        {
            "find", py_graph_cluster_find,
            METH_VARARGS,
            "find(block_or_widget, /)\n--\n\nFind a cluster depending on the provided property."
            "\n"
            "Alias for find_by_block() or find_by_widget()."
        },
        {
            "bootstrap", py_graph_cluster_bootstrap,
            METH_VARARGS | METH_STATIC,
            "bootstrap(binary, list, /)\n--\n\nBuild a graph cluster from a binary and a list of code blocks."
        },
        {
            "collect_clusters", py_graph_cluster_collect,
            METH_NOARGS,
            "collect_clusters()\n--\n\nCollect all clusters involvded in a graph view clustering."
        },
        {
            "collect_edges", py_graph_cluster_collect_edges,
            METH_NOARGS,
            "collect_edges()\n--\n\nCollect all cluster edges involvded in a graph view clustering."
        },
        { NULL }
    };

    static PyGetSetDef py_graph_cluster_getseters[] = {
        {
            "block", py_graph_cluster_get_block, NULL,
            "Main code block linked to the cluster.", NULL
        },
        {
            "widget", py_graph_cluster_get_widget, NULL,
            "GTK widget built to display the code block linked to the cluster.", NULL
        },
        {
            "allocation", py_graph_cluster_get_allocation, NULL,
            "Area allocated for the cluster code block.", NULL
        },
        {
            "needed_alloc", py_graph_cluster_get_needed_alloc, NULL,
            "Area needed for the cluster code block and all its children.", NULL
        },
        { NULL }
    };

    static PyTypeObject py_graph_cluster_type = {

        PyVarObject_HEAD_INIT(NULL, 0)

        .tp_name        = "pychrysalide.gtkext.graph.GraphCluster",
        .tp_basicsize   = sizeof(PyGObject),

        .tp_flags       = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,

        .tp_doc         = "Graphical cluster used in the graph view.\n" \
                          "\n" \
                          "The aim of this object is to provide a read-only " \
                          "access to the information relative to graphical " \
                          "cluster contained in a layout.",

        .tp_methods     = py_graph_cluster_methods,
        .tp_getset      = py_graph_cluster_getseters,

        .tp_new         = no_python_constructor_allowed,

    };

    return &py_graph_cluster_type;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : module = module dont la définition est à compléter.          *
*                                                                             *
*  Description : Prend en charge l'objet 'pychrysalide.gtkext..GraphCluster'. *
*                                                                             *
*  Retour      : Bilan de l'opération.                                        *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

bool ensure_python_graph_cluster_is_registered(void)
{
    PyTypeObject *type;                     /* Type Python 'BinPortion'    */
    PyObject *module;                       /* Module à recompléter        */
    PyObject *dict;                         /* Dictionnaire du module      */

    type = get_python_graph_cluster_type();

    if (!PyType_HasFeature(type, Py_TPFLAGS_READY))
    {
        module = get_access_to_python_module("pychrysalide.gtkext.graph");

        dict = PyModule_GetDict(module);

        if (!register_class_for_pygobject(dict, G_TYPE_GRAPH_CLUSTER, type))
            return false;

    }

    return true;

}


/******************************************************************************
*                                                                             *
*  Paramètres  : arg = argument quelconque à tenter de convertir.             *
*                dst = destination des valeurs récupérées en cas de succès.   *
*                                                                             *
*  Description : Tente de convertir en ensemble de blocs de code.             *
*                                                                             *
*  Retour      : Bilan de l'opération, voire indications supplémentaires.     *
*                                                                             *
*  Remarques   : -                                                            *
*                                                                             *
******************************************************************************/

int convert_to_graph_cluster(PyObject *arg, void *dst)
{
    int result;                             /* Bilan à retourner           */

    result = PyObject_IsInstance(arg, (PyObject *)get_python_graph_cluster_type());

    switch (result)
    {
        case -1:
            /* L'exception est déjà fixée par Python */
            result = 0;
            break;

        case 0:
            PyErr_SetString(PyExc_TypeError, "unable to convert the provided argument to graph cluster");
            break;

        case 1:
            *((GGraphCluster **)dst) = G_GRAPH_CLUSTER(pygobject_get(arg));
            break;

        default:
            assert(false);
            break;

    }

    return result;

}