summaryrefslogtreecommitdiff
path: root/src/analysis
diff options
context:
space:
mode:
Diffstat (limited to 'src/analysis')
-rw-r--r--src/analysis/binary.c59
-rw-r--r--src/analysis/binary.h6
-rw-r--r--src/analysis/db/cdb.c12
-rw-r--r--src/analysis/db/client.c3
-rw-r--r--src/analysis/db/collection-int.h10
-rw-r--r--src/analysis/db/collection.c625
-rw-r--r--src/analysis/db/collection.h31
-rw-r--r--src/analysis/db/item-int.h4
-rw-r--r--src/analysis/db/item.c171
-rw-r--r--src/analysis/db/item.h22
-rw-r--r--src/analysis/db/items/bookmark.c39
-rw-r--r--src/analysis/db/items/comment.c78
-rw-r--r--src/analysis/db/items/move.c40
-rw-r--r--src/analysis/db/items/switcher.c53
-rw-r--r--src/analysis/db/misc/timestamp.h4
-rw-r--r--src/analysis/db/protocol.h9
16 files changed, 520 insertions, 646 deletions
diff --git a/src/analysis/binary.c b/src/analysis/binary.c
index 822510c..05e42c7 100644
--- a/src/analysis/binary.c
+++ b/src/analysis/binary.c
@@ -1127,87 +1127,56 @@ bool _g_loaded_binary_add_to_collection(GLoadedBinary *binary, GDbItem *item, bo
}
+ g_object_unref(G_OBJECT(item));
+
}
g_object_unref(G_OBJECT(collec));
- g_object_unref(G_OBJECT(item));
return result;
}
+
+
/******************************************************************************
* *
-* Paramètres : binary = élément binaire à consulter. *
-* feature = fonctionnalité visée par la requête. *
-* item = élémnent à retirer d'un serveur de collection. *
-* lock = indique si le verrou d'écriture doit être posé. *
+* Paramètres : binary = élément binaire à consulter. *
+* timestamp = date du dernier élément à garder comme actif. *
* *
-* Description : Demande la suppression de modification dans une collection. *
+* Description : Active les éléments en amont d'un horodatage donné. XXXXXXXXXXXXXXXXX *
* *
-* Retour : Bilan partiel de l'opération demandée. *
+* Retour : true si la commande a bien été envoyée, false sinon. *
* *
* Remarques : - *
* *
******************************************************************************/
-bool _g_loaded_binary_remove_from_collection(GLoadedBinary *binary, DBFeatures feature, GDbItem *item, bool lock)
+bool g_loaded_binary_set_last_active(GLoadedBinary *binary, timestamp_t timestamp)
{
- bool result; /* Bilan à faire remonter */
- GDbCollection *collec; /* Collection visée au final */
- DBStorage storage; /* Forme d'enregistrement */
- GHubClient *client; /* Liaison à utiliser */
- packed_buffer out_pbuf; /* Tampon d'émission */
- SSL *tls_fd; /* Canal de communication SSL */
+ bool result; /* Bilan à retourner */
- collec = g_loaded_binary_find_collection(binary, feature);
- if (collec == NULL) return false;
- /* S'il n'y a pas besoin de sauvegarde... */
- if (g_db_item_get_flags(item) & DIF_VOLATILE)
- result = _g_db_collection_remove_item(collec, item, lock, true);
+ result = false;
- /* Sinon on envoie par le réseau ! */
- else
- {
- storage = g_loaded_binary_get_storage(binary, feature);
- if (storage == DBS_ALL_REMOTE)
- client = binary->remote;
- else
- client = NULL;
- if (client == NULL)
- client = binary->local;
- init_packed_buffer(&out_pbuf);
+ return result;
+
+}
- tls_fd = g_hub_client_get_ssl_fd(client);
- if (tls_fd == NULL)
- result = false;
- else
- {
- result = g_db_collection_pack(collec, &out_pbuf, DBA_REM_ITEM, item);
- if (result)
- result = ssl_send_packed_buffer(&out_pbuf, tls_fd);
- g_hub_client_put_ssl_fd(client, tls_fd);
- }
- exit_packed_buffer(&out_pbuf);
- }
- g_object_unref(G_OBJECT(collec));
- return result;
-}
diff --git a/src/analysis/binary.h b/src/analysis/binary.h
index dd3dfc6..b4c8652 100644
--- a/src/analysis/binary.h
+++ b/src/analysis/binary.h
@@ -131,12 +131,10 @@ bool _g_loaded_binary_add_to_collection(GLoadedBinary *, GDbItem *, bool);
#define g_loaded_binary_add_to_collection(b, i) \
_g_loaded_binary_add_to_collection(b, i, true)
-/* Demande la suppression de modification dans une collection. */
-bool _g_loaded_binary_remove_from_collection(GLoadedBinary *, DBFeatures, GDbItem *, bool);
-#define g_loaded_binary_remove_from_collection(b, f, i) \
- _g_loaded_binary_remove_from_collection(b, f, i, true)
+/* Active les éléments en amont d'un horodatage donné. */
+bool g_loaded_binary_set_last_active(GLoadedBinary *, timestamp_t);
diff --git a/src/analysis/db/cdb.c b/src/analysis/db/cdb.c
index a787a19..ca39a58 100644
--- a/src/analysis/db/cdb.c
+++ b/src/analysis/db/cdb.c
@@ -967,8 +967,16 @@ static void *g_cdb_archive_process(GCdbArchive *archive)
case DBC_SET_LAST_ACTIVE:
- status = update_activity_in_collections(archive->collections, &in_pbuf, archive->db);
- if (!status) goto gcap_bad_exchange;
+ init_packed_buffer(&out_pbuf);
+
+ status = update_activity_in_collections(archive->collections, \
+ &in_pbuf, &out_pbuf, archive->db);
+ if (!status) goto gcap_bad_reply;
+
+ status = ssl_send_packed_buffer(&out_pbuf, archive->clients[i].ssl_fd);
+ if (!status) goto gcap_bad_reply;
+
+ exit_packed_buffer(&out_pbuf);
break;
diff --git a/src/analysis/db/client.c b/src/analysis/db/client.c
index 744fa57..beb30c1 100644
--- a/src/analysis/db/client.c
+++ b/src/analysis/db/client.c
@@ -789,7 +789,7 @@ static void *g_hub_client_update(GHubClient *client)
status = extract_packed_buffer(&in_pbuf, &tmp8, sizeof(uint8_t), true);
if (!status) goto gdcu_bad_exchange;
- client->can_get_updates = (tmp8 == 0x0);
+ client->can_get_updates = (tmp8 == 0x1);
break;
}
@@ -797,6 +797,7 @@ static void *g_hub_client_update(GHubClient *client)
if (has_more_data_in_packed_buffer(&in_pbuf))
goto next_command;
+ client->can_get_updates = true;
continue;
gdcu_bad_exchange:
diff --git a/src/analysis/db/collection-int.h b/src/analysis/db/collection-int.h
index b3b83bb..c9d37bf 100644
--- a/src/analysis/db/collection-int.h
+++ b/src/analysis/db/collection-int.h
@@ -35,9 +35,6 @@
/* Crée la table associée à une collection d'éléments. */
typedef bool (* collec_create_db_table_fc) (const GDbCollection *, sqlite3 *);
-/* Décrit les colonnes utiles à un chargement de données. */
-typedef bool (* collec_setup_load_fc) (GDbCollection *, bound_value **, size_t *);
-
/* Charge les valeurs utiles pour une localisation. */
typedef bool (* collec_load_item) (GDbCollection *, const bound_value *, size_t);
@@ -58,8 +55,8 @@ struct _GDbCollection
/* Référence circulaire */
GLoadedBinary *binary; /* Binaire rattaché éventuel */
- GList *items; /* Eléments rassemblés */
- GList *sorted; /* Eléments triés */
+ GDbItem **items; /* Eléments rassemblés */
+ size_t count; /* Quantité de ces éléments */
GHashTable *last_items; /* Statuts courants d'éléments */
GRWLock params_access; /* Verrou de protection */
@@ -71,7 +68,6 @@ struct _GDbCollectionClass
GObjectClass parent; /* A laisser en premier */
collec_create_db_table_fc create_table; /* Création de la table en SQL */
- collec_setup_load_fc setup_load; /* Prépare le chargement */
collec_load_item load_item; /* Charge un élément */
collec_has_key_fc has_key; /* Recherche de présence */
@@ -79,6 +75,8 @@ struct _GDbCollectionClass
void (* content_extended) (GDbCollection *, GDbItem *);
+ void (* state_changed) (GDbCollection *, GDbItem *);
+
};
diff --git a/src/analysis/db/collection.c b/src/analysis/db/collection.c
index f728718..edb55d8 100644
--- a/src/analysis/db/collection.c
+++ b/src/analysis/db/collection.c
@@ -58,14 +58,23 @@ static void g_db_collection_finalize(GDbCollection *);
-/* --------------------- MANIPULATIONS AVEC UNE BASE DE DONNEES --------------------- */
+#define g_db_collection_find_by_timestamped(c, i) c->count
+
+#define g_db_collection_find_by_timestamp(c, i) c->count
+
+
+
+/* Ajoute un élément dans la liste des éléments actifs. */
+static void g_db_collection_set_last_item(GDbCollection *, GDbItem *, bool);
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool _g_db_collection_setup_load(GDbCollection *, bound_value **, size_t *);
+/* Retire un élément de la liste des éléments courants. */
+static void g_db_collection_unset_last_item(GDbCollection *, GDbItem *, size_t);
+
+
+
+/* --------------------- MANIPULATIONS AVEC UNE BASE DE DONNEES --------------------- */
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool g_db_collection_setup_load(GDbCollection *, bound_value **, size_t *);
/* Charge et intère un élément dans une collection. */
static bool g_db_collection_load_new_item(const bound_value *, size_t, GDbCollection *);
@@ -104,8 +113,6 @@ static void g_db_collection_class_init(GDbCollectionClass *klass)
object->dispose = (GObjectFinalizeFunc/* ! */)g_db_collection_dispose;
object->finalize = (GObjectFinalizeFunc)g_db_collection_finalize;
- klass->setup_load = (collec_setup_load_fc)_g_db_collection_setup_load;
-
g_signal_new("content-extended",
G_TYPE_DB_COLLECTION,
G_SIGNAL_RUN_LAST,
@@ -114,6 +121,14 @@ static void g_db_collection_class_init(GDbCollectionClass *klass)
g_cclosure_marshal_VOID__OBJECT,
G_TYPE_NONE, 1, G_TYPE_OBJECT);
+ g_signal_new("state-changed",
+ G_TYPE_DB_COLLECTION,
+ G_SIGNAL_RUN_LAST,
+ G_STRUCT_OFFSET(GDbCollectionClass, state_changed),
+ NULL, NULL,
+ g_cclosure_marshal_VOID__OBJECT,
+ G_TYPE_NONE, 1, G_TYPE_OBJECT);
+
}
@@ -133,6 +148,9 @@ static void g_db_collection_init(GDbCollection *collec)
{
collec->binary = NULL;
+ collec->items = NULL;
+ collec->count = 0;
+
collec->last_items = g_hash_table_new_full((GHashFunc)g_db_item_hash_key,
(GEqualFunc)g_db_item_cmp_key,
(GDestroyNotify)g_object_unref,
@@ -157,17 +175,10 @@ static void g_db_collection_init(GDbCollection *collec)
static void g_db_collection_dispose(GDbCollection *collec)
{
- if (collec->items != NULL)
- {
- g_list_free_full(collec->items, g_object_unref);
- collec->items = NULL;
- }
+ size_t i; /* Boucle de parcours */
- if (collec->sorted != NULL)
- {
- g_list_free_full(collec->sorted, g_object_unref);
- collec->sorted = NULL;
- }
+ for (i = 0; i < collec->count; i++)
+ g_clear_object(&collec->items[i]);
G_OBJECT_CLASS(g_db_collection_parent_class)->dispose(G_OBJECT(collec));
@@ -188,6 +199,9 @@ static void g_db_collection_dispose(GDbCollection *collec)
static void g_db_collection_finalize(GDbCollection *collec)
{
+ if (collec->items != NULL)
+ free(collec->items);
+
g_rw_lock_clear(&collec->params_access);
G_OBJECT_CLASS(g_db_collection_parent_class)->finalize(G_OBJECT(collec));
@@ -361,8 +375,6 @@ bool g_db_collection_unpack(GDbCollection *collec, packed_buffer *pbuf, sqlite3
bool result; /* Bilan à faire remonter */
DBAction action; /* Commande de la requête */
GDbItem *item; /* Définition d'élément visé */
- GList *found; /* Test de présence existante */
- timestamp_t inactive; /* Horodatage de désactivation */
result = _g_db_collection_unpack(collec, pbuf, &action, &item);
if (!result) return false;
@@ -379,7 +391,7 @@ bool g_db_collection_unpack(GDbCollection *collec, packed_buffer *pbuf, sqlite3
if (result)
{
- if (collec->binary != NULL && g_db_item_is_active(item))
+ if (collec->binary != NULL && !g_db_item_has_flag(item, DIF_DISABLED))
g_db_item_apply(item, collec->binary);
if (db != NULL)
@@ -389,50 +401,29 @@ bool g_db_collection_unpack(GDbCollection *collec, packed_buffer *pbuf, sqlite3
break;
- case DBA_REM_ITEM:
-
- g_db_collection_wlock(collec);
-
- found = g_list_find_custom(collec->items, item, (GCompareFunc)g_db_item_compare_with_timestamp);
-
- result = (found != NULL);
+ case DBA_CHANGE_STATE:
- if (result)
+ if (collec->binary != NULL)
{
- /* Côté client */
- if (db == NULL)
- result = _g_db_collection_remove_item(collec, item, false, true);
+ result = g_db_collection_update_item_state(collec, item);
- /* Côté serveur */
- else
+ if (result)
{
- if (g_db_item_is_active(G_DB_ITEM(found->data)))
- {
- inactive = _g_db_collection_compute_inactivity_timestamp(collec, false);
- result = _g_db_collection_update_item_activity(collec, item, &inactive, false);
- }
+ if (g_db_item_has_flag(item, DIF_DISABLED))
+ g_db_item_cancel(item, collec->binary);
+ else
+ g_db_item_apply(item, collec->binary);
}
}
- g_db_collection_wunlock(collec);
-
- break;
-
- case DBA_CHANGE_STATE:
-
- if (db == NULL)
- {
- if (g_db_item_is_active(item))
- result = g_db_collection_update_item_activity(collec, item, NULL);
- else
- {
- inactive = g_db_item_get_timestamp(item) + 1;
- result = g_db_collection_update_item_activity(collec, item, &inactive);
- }
- }
else
+ {
+ assert(db != NULL);
result = false;
+ }
+
+ g_object_unref(G_OBJECT(item));
break;
@@ -442,8 +433,6 @@ bool g_db_collection_unpack(GDbCollection *collec, packed_buffer *pbuf, sqlite3
}
- g_object_unref(G_OBJECT(item));
-
return result;
}
@@ -500,7 +489,7 @@ bool g_db_collection_pack(GDbCollection *collec, packed_buffer *pbuf, DBAction a
bool g_db_collection_pack_all_updates(GDbCollection *collec, packed_buffer *pbuf)
{
bool result; /* Bilan à renvoyer */
- GList *iter; /* Boucle de parcours */
+ size_t i; /* Boucle de parcours */
result = true;
@@ -509,13 +498,8 @@ bool g_db_collection_pack_all_updates(GDbCollection *collec, packed_buffer *pbuf
* g_cdb_archive_add_client().
*/
- for (iter = g_list_first(collec->items);
- iter != NULL && result;
- iter = g_list_next(iter))
- {
- result = g_db_collection_pack(collec, pbuf, DBA_ADD_ITEM, G_DB_ITEM(iter->data));
-
- }
+ for (i = 0; i < collec->count && result; i++)
+ result = g_db_collection_pack(collec, pbuf, DBA_ADD_ITEM, G_DB_ITEM(collec->items[i]));
return result;
@@ -560,6 +544,7 @@ void g_db_collection_lock_unlock(GDbCollection *collec, bool write, bool lock)
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à consulter. *
+* count = taille de la liste constituée. [OUT] *
* *
* Description : Renvoie la liste des éléments rassemblés. *
* *
@@ -569,14 +554,34 @@ void g_db_collection_lock_unlock(GDbCollection *collec, bool write, bool lock)
* *
******************************************************************************/
-GList *g_db_collection_get_items(const GDbCollection *collec)
+GDbItem **g_db_collection_get_items(const GDbCollection *collec, size_t *count)
{
+ GDbItem **result; /* Liste à retourner */
+ size_t i; /* Boucle de parcours */
+
/**
* Un verrou doit être posé !
* Il n'y a pas d'assert() possible pour le vérifier...
*/
- return collec->items;
+ *count = collec->count;
+
+ if (*count == 0)
+ result = NULL;
+
+ else
+ {
+ result = malloc(*count * sizeof(GDbItem *));
+
+ for (i = 0; i < *count; i++)
+ {
+ result[i] = collec->items[i];
+ g_object_ref(G_OBJECT(result[i]));
+ }
+
+ }
+
+ return result;
}
@@ -626,16 +631,16 @@ GDbItem *g_db_collection_has_key(GDbCollection *collec, ...)
bool g_db_collection_has_item(GDbCollection *collec, GDbItem *item)
{
bool result; /* Bilan à retourner */
- GList *found; /* Test de présence existante */
+ size_t index; /* Indice de l'élément trouvé */
/**
* Un verrou doit être posé !
* Il n'y a pas d'assert() possible pour le vérifier...
*/
- found = g_list_find_custom(collec->items, item, (GCompareFunc)g_db_item_compare_with_timestamp);
+ index = g_db_collection_find_by_timestamped(collec, item);
- result = (found != NULL);
+ result = (index < collec->count);
return result;
@@ -646,36 +651,46 @@ bool g_db_collection_has_item(GDbCollection *collec, GDbItem *item)
* *
* Paramètres : collec = ensemble d'éléments à considérer. *
* item = élément de collection à manipuler. *
-* lock = indique si le verrou d'écriture doit être posé. *
+* new = précise la nature de l'élément à insérer. *
* *
-* Description : Procède à l'ajout d'un nouvel élément dans la collection. *
+* Description : Ajoute un élément dans la liste des éléments actifs. *
* *
-* Retour : Bilan de l'exécution de l'opération. *
+* Retour : - *
* *
-* Remarques : L'appelant reste le propriétaire de l'object transféré. *
+* Remarques : *
* *
******************************************************************************/
-bool _g_db_collection_add_item(GDbCollection *collec, GDbItem *item, bool lock)
+static void g_db_collection_set_last_item(GDbCollection *collec, GDbItem *item, bool new)
{
- bool result; /* Bilan à faire remonter */
GDbItem *prev; /* Elément similaire précédent */
+ timestamp_t its; /* Horodatage #0 */
+ timestamp_t pts; /* Horodatage #1 */
- result = true;
-
- if (lock)
- g_db_collection_wlock(collec);
-
- g_object_ref(G_OBJECT(item));
- collec->items = g_list_append(collec->items, item);
-
- g_object_ref(G_OBJECT(item));
- collec->sorted = g_list_insert_sorted(collec->sorted, item, (GCompareFunc)g_db_item_compare_with_timestamp);
+ assert(!g_db_item_has_flag(item, DIF_DISABLED));
prev = g_hash_table_lookup(collec->last_items, item);
if (prev != NULL)
{
+ /**
+ * Dans le cas où le changement intervient sans contexte particulier,
+ * on s'assure de le pas remplacer un élément plus récent déjà en place
+ * par un élément nouvellement actif mais dépassé.
+ *
+ * Le code de g_db_collection_disable_at(), aux conséquences portant
+ * dans le serveur et le client, procède de manière à éviter cette
+ * situation par un ordre de parcours choisi.
+ *
+ * On insère néanmoins une petite sécurité.
+ */
+
+ its = g_db_item_get_timestamp(item);
+ pts = g_db_item_get_timestamp(prev);
+
+ if (timestamp_is_younger(its, pts))
+ goto already_up_to_date;
+
g_object_ref(G_OBJECT(prev));
if (g_db_item_get_flags(item) & DIF_ERASER)
@@ -683,7 +698,8 @@ bool _g_db_collection_add_item(GDbCollection *collec, GDbItem *item, bool lock)
else
{
- g_db_item_add_flag(item, DIF_UPDATED);
+ if (new)
+ g_db_item_add_flag(item, DIF_UPDATED);
g_object_ref(G_OBJECT(item));
g_object_ref(G_OBJECT(item));
@@ -698,19 +714,20 @@ bool _g_db_collection_add_item(GDbCollection *collec, GDbItem *item, bool lock)
else
{
- g_object_ref(G_OBJECT(item));
- g_object_ref(G_OBJECT(item));
+ if (g_db_item_get_flags(item) & DIF_ERASER)
+ {
+ g_object_ref(G_OBJECT(item));
+ g_object_ref(G_OBJECT(item));
- g_hash_table_add(collec->last_items, item);
+ g_hash_table_add(collec->last_items, item);
+
+ }
}
- g_signal_emit_by_name(collec, "content-extended", item);
+ already_up_to_date:
- if (lock)
- g_db_collection_wunlock(collec);
-
- return result;
+ ;
}
@@ -720,49 +737,30 @@ bool _g_db_collection_add_item(GDbCollection *collec, GDbItem *item, bool lock)
* Paramètres : collec = ensemble d'éléments à considérer. *
* item = élément de collection à manipuler. *
* lock = indique si le verrou d'écriture doit être posé. *
-* signal = émet un événement pour signaler le retrait. *
* *
-* Description : Procède au retrait d'un nouvel élément dans la collection. *
+* Description : Procède à l'ajout d'un nouvel élément dans la collection. *
* *
* Retour : Bilan de l'exécution de l'opération. *
* *
-* Remarques : L'appelant reste le propriétaire de l'object transféré. *
+* Remarques : L'appelant perd la propriété de l'object transféré. *
* *
******************************************************************************/
-bool _g_db_collection_remove_item(GDbCollection *collec, GDbItem *item, bool lock, bool signal)
+bool _g_db_collection_add_item(GDbCollection *collec, GDbItem *item, bool lock)
{
bool result; /* Bilan à faire remonter */
- GList *found; /* Test de présence existante */
- GDbItem *internal; /* Elément interne à modifier */
result = true;
if (lock)
g_db_collection_wlock(collec);
- found = g_list_find_custom(collec->sorted, item, (GCompareFunc)g_db_item_compare_with_timestamp);
-
- result = (found != NULL);
+ collec->items = realloc(collec->items, ++collec->count * sizeof(GDbItem *));
+ collec->items[collec->count - 1] = item;
- if (result)
- {
- internal = G_DB_ITEM(found->data);
+ g_db_collection_set_last_item(collec, item, true);
- collec->sorted = g_list_delete_link(collec->sorted, found);
-
- found = g_list_find(collec->items, item);
- assert(found != NULL);
-
- collec->items = g_list_delete_link(collec->items, found);
-
- if (signal)
- ;//g_signal_emit_by_name(collec, "content-changed", DBA_REM_ITEM, internal);
-
- g_object_unref(G_OBJECT(internal));
- g_object_unref(G_OBJECT(internal));
-
- }
+ g_signal_emit_by_name(collec, "content-extended", item);
if (lock)
g_db_collection_wunlock(collec);
@@ -775,47 +773,54 @@ bool _g_db_collection_remove_item(GDbCollection *collec, GDbItem *item, bool loc
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à considérer. *
-* lock = indique si le verrou d'écriture doit être posé. *
+* pbuf = paquet de données où venir inscrire les infos. *
* *
-* Description : Détermine l'horodatage le plus jeune pour une désactivation. *
+* Description : Procède au retrait des éléments désactivés de la collection. *
* *
-* Retour : Horodatage à utiliser pour une phase de désactivation. *
+* Retour : Bilan de l'exécution de l'opération. *
* *
-* Remarques : - *
+* Remarques : *
* *
******************************************************************************/
-timestamp_t _g_db_collection_compute_inactivity_timestamp(GDbCollection *collec, bool lock)
+bool g_db_collection_drop_disabled_items(GDbCollection *collec, packed_buffer *pbuf)
{
- timestamp_t result; /* Horodatage à retourner */
- GList *iter; /* Boucle de parcours */
- GDbItem *item; /* Elément interne à consulter */
- timestamp_t stamp; /* Horodatage de l'élément */
+ bool result; /* Bilan à retourner */
+ size_t i; /* Boucle de parcours */
+ GDbItem *item; /* Elément désactivé */
+#ifndef NDEBUG
+ GDbItem *enabled; /* Eventuel similarité active */
+#endif
- result = TIMESTAMP_ALL_ACTIVE;
+ /**
+ * Voie de suppression d'un élément côté serveur.
+ */
- if (lock)
- g_db_collection_wlock(collec);
+ result = true;
- for (iter = g_list_first(collec->items);
- iter != NULL;
- iter = g_list_next(iter))
+ g_db_collection_wlock(collec);
+
+ for (i = collec->count; i > 0 && result; i--)
{
- item = G_DB_ITEM(iter->data);
+ item = collec->items[i - 1];
- if (!g_db_item_is_active(item))
- {
- stamp = g_db_item_get_timestamp(item);
+ if (!g_db_item_has_flag(item, DIF_DISABLED))
+ break;
- if (timestamp_is_younger(stamp, result))
- result = stamp;
+#ifndef NDEBUG
+ enabled = g_hash_table_lookup(collec->last_items, item);
+ assert(enabled != item);
+#endif
- }
+ collec->items = realloc(collec->items, --collec->count * sizeof(GDbItem *));
+
+ result = g_db_collection_pack(collec, pbuf, DBA_REM_ITEM, item);
+
+ g_object_unref(G_OBJECT(item));
}
- if (lock)
- g_db_collection_wunlock(collec);
+ g_db_collection_wunlock(collec);
return result;
@@ -826,9 +831,8 @@ timestamp_t _g_db_collection_compute_inactivity_timestamp(GDbCollection *collec,
* *
* Paramètres : collec = ensemble d'éléments à considérer. *
* item = élément de collection à manipuler. *
-* lock = indique si le verrou d'écriture doit être posé. *
* *
-* Description : Met à jour le statut d'activité d'un élément de collection. *
+* Description : Procède au retrait d'un élément dans la collection. *
* *
* Retour : Bilan de l'exécution de l'opération. *
* *
@@ -836,31 +840,47 @@ timestamp_t _g_db_collection_compute_inactivity_timestamp(GDbCollection *collec,
* *
******************************************************************************/
-bool _g_db_collection_update_item_activity(GDbCollection *collec, GDbItem *item, timestamp_t *timestamp, bool lock)
+bool g_db_collection_remove_item(GDbCollection *collec, const GDbItem *item)
{
bool result; /* Bilan à faire remonter */
- GList *found; /* Test de présence existante */
- GDbItem *internal; /* Elément interne à modifier */
+ size_t found; /* Indice de l'élément concerné*/
+ GDbItem *disabled; /* Elément désactivé */
+#ifndef NDEBUG
+ GDbItem *enabled; /* Eventuel similarité active */
+#endif
- if (lock)
- g_db_collection_wlock(collec);
+ /**
+ * Voie de suppression d'un élément côté serveur.
+ */
- found = g_list_find_custom(collec->items, item, (GCompareFunc)g_db_item_compare_without_timestamp);
+ g_db_collection_wlock(collec);
- result = (found != NULL);
+ found = g_db_collection_find_by_timestamped(collec, item);
+
+ result = (found < collec->count);
+ assert(result);
if (result)
{
- internal = G_DB_ITEM(found->data);
+ disabled = collec->items[found];
+
+ assert(g_db_item_has_flag(disabled, DIF_DISABLED));
+
+#ifndef NDEBUG
+ enabled = g_hash_table_lookup(collec->last_items, disabled);
+ assert(enabled != disabled);
+#endif
+
+ memmove(collec->items + found, collec->items + found + 1,
+ (collec->count - found - 1) * sizeof(GDbItem *));
- result = g_db_item_set_activity(internal, collec->binary, timestamp);
+ collec->items = realloc(collec->items, --collec->count * sizeof(GDbItem *));
- //g_signal_emit_by_name(collec, "content-changed", DBA_CHANGE_STATE, internal);
+ g_object_unref(G_OBJECT(disabled));
}
- if (lock)
- g_db_collection_wunlock(collec);
+ g_db_collection_wunlock(collec);
return result;
@@ -869,157 +889,202 @@ bool _g_db_collection_update_item_activity(GDbCollection *collec, GDbItem *item,
/******************************************************************************
* *
-* Paramètres : collec = ensemble d'éléments à considérer. *
-* timestamp = date du dernier élément à garder comme actif. *
-* inactive = date du premier élément inactif rencontré. [OUT] *
-* db = base de données à mettre à jour. *
+* Paramètres : collec = ensemble d'éléments à considérer. *
+* item = élément de collection à manipuler. *
+* count = décompte des éléments actifs. *
* *
-* Description : Active les éléments en amont d'un horodatage donné. *
+* Description : Retire un élément de la liste des éléments courants. *
* *
* Retour : - *
* *
-* Remarques : - *
+* Remarques : *
* *
******************************************************************************/
-GList *g_db_collection_set_last_active(GDbCollection *collec, timestamp_t timestamp, timestamp_t *inactive, sqlite3 *db)
+static void g_db_collection_unset_last_item(GDbCollection *collec, GDbItem *item, size_t count)
{
- GList *result; /* Liste d'inactifs à renvoyer */
- GList *i; /* Parcours des éléments */
- GDbItem *item; /* Elément à traiter */
- timestamp_t stamp; /* Horodatage de l'élément */
+ GDbItem *prev; /* Elément similaire précédent */
+ GDbItem *old; /* Ancien élément similaire */
+ size_t i; /* Boucle de parcours */
- result = NULL;
+ assert(g_db_item_has_flag(item, DIF_DISABLED));
+
+ prev = g_hash_table_lookup(collec->last_items, item);
- for (i = g_list_first(collec->items); i != NULL; i = g_list_next(i))
+ if (prev == item)
{
- item = G_DB_ITEM(i->data);
- stamp = g_db_item_get_timestamp(item);
+ old = NULL;
- if (timestamp_is_younger(stamp, timestamp))
- {
- if (!g_db_item_is_active(item))
- {
- /* ... */g_db_item_set_activity(item, collec->binary, NULL);
- /* ... */g_db_collection_store_updated_item(collec, item, db);
- //g_signal_emit_by_name(collec, "content-changed", DBA_CHANGE_STATE, item);
- }
+ for (i = count; i > 0; i++)
+ if (g_db_item_cmp_key(collec->items[i - 1], item))
+ break;
+ else
+ old = collec->items[i - 1];
- }
+ if (old == NULL || g_db_item_get_flags(old) & DIF_ERASER)
+ g_hash_table_remove(collec->last_items, item);
else
{
- if (!g_db_item_is_active(item))
- {
- if (timestamp_is_younger(stamp, *inactive))
- *inactive = stamp;
- }
+ g_object_ref(G_OBJECT(old));
+ g_object_ref(G_OBJECT(old));
- else
- result = g_list_append(result, item);
+ g_hash_table_replace(collec->last_items, old, old);
}
}
- return result;
-
}
/******************************************************************************
* *
-* Paramètres : collec = ensemble d'éléments à considérer. *
-* item = élément à désactiver. *
-* timestamp = date du premier élément inactif rencontré. [OUT] *
+* Paramètres : binary = élément binaire à consulter. *
+* timestamp = date du dernier élément à garder comme actif. *
+* db = base de données à mettre à jour. *
+* pbuf = paquet de données où venir inscrire les infos. *
* *
* Description : Désactive les éléments en aval d'un horodatage donné. *
* *
-* Retour : true si l'élément a été traité, false sinon. *
+* Retour : true si la commande a bien été envoyée, false sinon. *
* *
* Remarques : - *
* *
******************************************************************************/
-bool g_db_collection_set_inactive(GDbCollection *collec, GDbItem *item, timestamp_t *timestamp)
+bool g_db_collection_disable_at(GDbCollection *collec, timestamp_t timestamp, sqlite3 *db, packed_buffer *pbuf)
{
bool result; /* Bilan à retourner */
+ size_t start; /* Début de la zone à changer */
+ size_t back; /* Début de restauration */
+ size_t i; /* Boucle de parcours */
+ GDbItem *item; /* Elément à traiter */
- /* Si cette collection n'est pas concernée, on ne bouge pas ! */
- if (G_OBJECT_TYPE(G_OBJECT(item)) != collec->type) return false;
+ result = true;
- result = g_db_item_set_activity(item, collec->binary, timestamp);
+ g_db_collection_wlock(collec);
- //g_signal_emit_by_name(collec, "content-changed", DBA_CHANGE_STATE, item);
+ start = g_db_collection_find_by_timestamp(collec, timestamp);
- return result;
+ /* Réactivation d'éléments ? */
-}
+ if (start > 0)
+ {
+ back = start;
+ for (i = start; i > 0; i--)
+ if (!g_db_item_has_flag(collec->items[i - 1], DIF_DISABLED))
+ break;
+ else
+ back--;
+ for (i = back; i < start && result; i++)
+ {
+ item = collec->items[i];
-/* ---------------------------------------------------------------------------------- */
-/* MANIPULATIONS AVEC UNE BASE DE DONNEES */
-/* ---------------------------------------------------------------------------------- */
+ g_db_item_remove_flag(item, DIF_DISABLED);
+ g_db_collection_store_updated_item(collec, item, db);
-/******************************************************************************
-* *
-* Paramètres : collec = ensemble d'éléments spectateur des opérations. *
-* db = accès à la base de données. *
-* *
-* Description : Crée la table d'élément dans une base de données. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
+ g_db_collection_set_last_item(collec, item, false);
-bool g_db_collection_create_db_table(const GDbCollection *collec, sqlite3 *db)
-{
- return G_DB_COLLECTION_GET_CLASS(collec)->create_table(collec, db);
+ result = g_db_collection_pack(collec, pbuf, DBA_CHANGE_STATE, item);
+
+ }
+
+ }
+
+ /* Désactivation des éléments en queue */
+
+ for (i = start; i < collec->count && result; i++)
+ {
+ item = collec->items[i];
+
+ if (g_db_item_has_flag(item, DIF_DISABLED))
+ break;
+
+ g_db_item_add_flag(item, DIF_DISABLED);
+
+ g_db_collection_store_updated_item(collec, item, db);
+
+ g_db_collection_unset_last_item(collec, item, start);
+
+ result = g_db_collection_pack(collec, pbuf, DBA_CHANGE_STATE, item);
+
+ }
+
+ g_db_collection_wunlock(collec);
+
+ return result;
}
/******************************************************************************
* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
+* Paramètres : collec = ensemble d'éléments à considérer. *
+* item = élément de collection à manipuler. *
* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
+* Description : Prend acte d'un changement d'état d'un élément de collection.*
* *
-* Retour : Bilan de l'opération. *
+* Retour : Bilan de l'exécution de l'opération. *
* *
-* Remarques : - *
+* Remarques : L'appelant reste le propriétaire de l'object transféré. *
* *
******************************************************************************/
-static bool _g_db_collection_setup_load(GDbCollection *collec, bound_value **values, size_t *count)
+bool g_db_collection_update_item_state(GDbCollection *collec, const GDbItem *item)
{
- if (!store_timestamp(NULL, "created", values, count))
- return false;
+ bool result; /* Bilan à faire remonter */
+ size_t index; /* Indice de l'élément visé */
+ GDbItem *changed; /* Elément à basculer */
+ DbItemFlags new; /* Nouvelles propriétés */
+
+ result = false;
+
+ g_db_collection_wlock(collec);
+
+ index = g_db_collection_find_by_timestamped(collec, item);
- if (!store_timestamp(NULL, "timestamp", values, count))
- return false;
+ result = (index < collec->count);
- if (!store_rle_string(NULL, "author", values, count))
- return false;
+ if (result)
+ {
+ changed = collec->items[index];
+
+ new = g_db_item_get_flags(item);
+
+ g_db_item_set_flags(changed, new);
+
+ if (new & DIF_DISABLED)
+ g_db_collection_unset_last_item(collec, changed, index);
+ else
+ g_db_collection_set_last_item(collec, changed, false);
+
+ g_signal_emit_by_name(collec, "state-changed", changed);
+
+ }
- return true;
+ g_db_collection_wunlock(collec);
+
+ return result;
}
+
+/* ---------------------------------------------------------------------------------- */
+/* MANIPULATIONS AVEC UNE BASE DE DONNEES */
+/* ---------------------------------------------------------------------------------- */
+
+
/******************************************************************************
* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
+* Paramètres : collec = ensemble d'éléments spectateur des opérations. *
+* db = accès à la base de données. *
* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
+* Description : Crée la table d'élément dans une base de données. *
* *
* Retour : Bilan de l'opération. *
* *
@@ -1027,12 +1092,9 @@ static bool _g_db_collection_setup_load(GDbCollection *collec, bound_value **val
* *
******************************************************************************/
-static bool g_db_collection_setup_load(GDbCollection *collec, bound_value **values, size_t *count)
+bool g_db_collection_create_db_table(const GDbCollection *collec, sqlite3 *db)
{
- *values = NULL;
- *count = 0;
-
- return G_DB_COLLECTION_GET_CLASS(collec)->setup_load(collec, values, count);
+ return G_DB_COLLECTION_GET_CLASS(collec)->create_table(collec, db);
}
@@ -1061,13 +1123,10 @@ static bool g_db_collection_load_new_item(const bound_value *values, size_t coun
result = g_db_item_load(new, values, count);
if (result)
- {
result = g_db_collection_add_item(G_DB_COLLECTION(collec), new);
-
+ else
g_object_unref(G_OBJECT(new));
- }
-
return result;
}
@@ -1089,10 +1148,15 @@ static bool g_db_collection_load_new_item(const bound_value *values, size_t coun
bool g_db_collection_load_all_items(GDbCollection *collec, sqlite3 *db)
{
bool result; /* Conclusion à faire remonter */
+ GDbItem *dummy; /* Interface vide */
bound_value *values; /* Champs de table à inclure */
size_t count; /* Nombre de ces champs */
- result = g_db_collection_setup_load(collec, &values, &count);
+ dummy = g_object_new(collec->type, NULL);
+
+ result = g_db_item_setup_load(dummy, &values, &count);
+
+ g_object_unref(G_OBJECT(dummy));
if (result)
result = load_db_values(db, collec->name, values, count, (db_load_cb)g_db_collection_load_new_item, collec);
@@ -1447,9 +1511,10 @@ bool pack_all_collection_updates(GList *list, packed_buffer *pbuf)
/******************************************************************************
* *
-* Paramètres : list = ensemble de collectons à traiter. *
-* pbuf = paquet de données où venir puiser les infos. *
-* db = base de données à mettre à jour. *
+* Paramètres : list = ensemble de collectons à traiter. *
+* inbuf = paquet de données où venir puiser les infos. *
+* outbuf = paquet de données où inscrire les mises à jour. *
+* db = base de données à mettre à jour. *
* *
* Description : Met à jour les statuts d'activité des éléments. *
* *
@@ -1459,18 +1524,14 @@ bool pack_all_collection_updates(GList *list, packed_buffer *pbuf)
* *
******************************************************************************/
-bool update_activity_in_collections(GList *list, packed_buffer *pbuf, sqlite3 *db)
+bool update_activity_in_collections(GList *list, packed_buffer *inbuf, packed_buffer *outbuf, sqlite3 *db)
{
bool result; /* Résultat global à renvoyer */
bool status; /* Bilan de lecture initiale */
timestamp_t timestamp; /* Horodatage de limite */
- timestamp_t inactive; /* Date du premier inactif */
- GList *remaining; /* Eléments restants à traiter */
- GList *c; /* Boucle de parcours #1 */
- GDbCollection *collec; /* Collection à manipuler */
- GList *got; /* Eléments restants à traiter */
- GList *i; /* Boucle de parcours #2 */
- GDbItem *item; /* Elément collecté à manipuler*/
+ GList *iter; /* Boucle de parcours */
+
+ result = true;
/**
* Cette procédure n'est appelée que depuis g_cdb_archive_process(),
@@ -1479,55 +1540,19 @@ bool update_activity_in_collections(GList *list, packed_buffer *pbuf, sqlite3 *d
* On a donc l'assurance d'un traitement global homgène des horodatages.
*/
- status = unpack_timestamp(&timestamp, pbuf);
+ status = unpack_timestamp(&timestamp, inbuf);
if (!status) return false;
- inactive = TIMESTAMP_ALL_ACTIVE;
-
- remaining = NULL;
-
- wlock_collections(list);
-
- for (c = g_list_first(list); c != NULL; c = g_list_next(c))
- {
- collec = G_DB_COLLECTION(c->data);
-
- got = g_db_collection_set_last_active(collec, timestamp, &inactive, db);
- remaining = g_list_concat(remaining, got);
-
- }
-
- gint sort_with_timestamp(GDbItem *a, GDbItem *b)
- {
- return g_db_item_cmp(a, b, false);
- }
- remaining = g_list_sort(remaining, (GCompareFunc)sort_with_timestamp);
- result = true;
-
- for (i = g_list_last(remaining); i != NULL && result; i = g_list_previous(i))
+ for (iter = g_list_first(list);
+ iter != NULL && result;
+ iter = g_list_next(iter))
{
- item = G_DB_ITEM(i->data);
-
- for (c = g_list_first(list); c != NULL && result; c = g_list_next(c))
- {
- collec = G_DB_COLLECTION(c->data);
-
- if (g_db_collection_set_inactive(collec, item, &inactive))
- {
- result = g_db_collection_store_updated_item(collec, item, db);
- break;
- }
-
- }
+ result = g_db_collection_disable_at(G_DB_COLLECTION(iter->data), timestamp, db, outbuf);
}
- wunlock_collections(list);
-
- g_list_free(remaining);
-
return result;
}
diff --git a/src/analysis/db/collection.h b/src/analysis/db/collection.h
index 60e5c26..9539229 100644
--- a/src/analysis/db/collection.h
+++ b/src/analysis/db/collection.h
@@ -93,7 +93,7 @@ void g_db_collection_lock_unlock(GDbCollection *, bool, bool);
#define g_db_collection_runlock(col) g_db_collection_lock_unlock(col, false, false);
/* Renvoie la liste des éléments rassemblés. */
-GList *g_db_collection_get_items(const GDbCollection *);
+GDbItem **g_db_collection_get_items(const GDbCollection *, size_t *);
/* Détermine si un élément est déjà présent ou non. */
GDbItem *g_db_collection_has_key(GDbCollection *, ...);
@@ -104,25 +104,24 @@ bool g_db_collection_has_item(GDbCollection *, GDbItem *);
/* Procède à l'ajout d'un nouvel élément dans la collection. */
bool _g_db_collection_add_item(GDbCollection *, GDbItem *, bool);
-/* Procède au retrait d'un nouvel élément dans la collection. */
-bool _g_db_collection_remove_item(GDbCollection *, GDbItem *, bool, bool);
+/* Procède au retrait des éléments désactivés de la collection. */
+bool g_db_collection_drop_disabled_items(GDbCollection *, packed_buffer *);
-/* Détermine l'horodatage le plus jeune pour une désactivation. */
-timestamp_t _g_db_collection_compute_inactivity_timestamp(GDbCollection *, bool);
+/* Procède au retrait d'un élément dans la collection. */
+bool g_db_collection_remove_item(GDbCollection *, const GDbItem *);
-/* Met à jour le statut d'activité d'un élément de collection. */
-bool _g_db_collection_update_item_activity(GDbCollection *, GDbItem *, timestamp_t *, bool);
+/* Désactive les éléments en aval d'un horodatage donné. */
+bool g_db_collection_disable_at(GDbCollection *, timestamp_t, sqlite3 *, packed_buffer *);
-#define g_db_collection_add_item(c, i) _g_db_collection_add_item(c, i, true)
-#define g_db_collection_remove_item(c, i) _g_db_collection_remove_item(c, i, true, true)
-#define g_db_collection_compute_inactivity_timestamp(c) _g_db_collection_compute_inactivity_timestamp(c, true)
-#define g_db_collection_update_item_activity(c, i, t) _g_db_collection_update_item_activity(c, i, t, true)
+/* Prend acte d'un changement d'état d'un élément de collection. */
+bool g_db_collection_update_item_state(GDbCollection *, const GDbItem *);
-/* Active les éléments en amont d'un horodatage donné. */
-GList *g_db_collection_set_last_active(GDbCollection *, timestamp_t, timestamp_t *, sqlite3 *);
-/* Désactive les éléments en aval d'un horodatage donné. */
-bool g_db_collection_set_inactive(GDbCollection *, GDbItem *, timestamp_t *);
+
+
+
+#define g_db_collection_add_item(c, i) _g_db_collection_add_item(c, i, true)
+//#define g_db_collection_remove_item(c, i) _g_db_collection_remove_item(c, i, true, true)
@@ -159,7 +158,7 @@ void lock_unlock_collections(GList *, bool, bool);
bool pack_all_collection_updates(GList *, packed_buffer *);
/* Met à jour les statuts d'activité des éléments. */
-bool update_activity_in_collections(GList *, packed_buffer *, sqlite3 *);
+bool update_activity_in_collections(GList *, packed_buffer *, packed_buffer *, sqlite3 *);
diff --git a/src/analysis/db/item-int.h b/src/analysis/db/item-int.h
index 8bba8a3..8c13cd9 100644
--- a/src/analysis/db/item-int.h
+++ b/src/analysis/db/item-int.h
@@ -111,8 +111,8 @@ struct _GDbItemClass
/* Définition du tronc commun pour les créations SQLite */
#define SQLITE_DB_ITEM_CREATE \
SQLITE_TIMESTAMP_CREATE("created") ", " \
- SQLITE_TIMESTAMP_CREATE("timestamp") ", " \
- SQLITE_RLESTR_CREATE("author")
+ SQLITE_RLESTR_CREATE("author") ", " \
+ "flags INTEGER"
diff --git a/src/analysis/db/item.c b/src/analysis/db/item.c
index efa2cdf..f9c4201 100644
--- a/src/analysis/db/item.c
+++ b/src/analysis/db/item.c
@@ -462,7 +462,7 @@ bool g_db_item_apply(GDbItem *item, GLoadedBinary *binary)
{
bool result; /* Bilan à faire remonter */
- assert(g_db_item_is_active(item));
+ assert(!g_db_item_has_flag(item, DIF_DISABLED));
result = G_DB_ITEM_GET_CLASS(item)->apply(item, binary);
@@ -491,7 +491,7 @@ bool g_db_item_cancel(GDbItem *item, GLoadedBinary *binary)
{
bool result; /* Bilan à faire remonter */
- assert(!g_db_item_is_active(item));
+ assert(g_db_item_has_flag(item, DIF_DISABLED));
result = G_DB_ITEM_GET_CLASS(item)->cancel(item, binary);
@@ -533,7 +533,7 @@ char *g_db_item_get_label(GDbItem *item)
* *
* Description : Fournit l'horodatage associé à l'élément de collection. *
* *
-* Retour : Date d'activité de l'élément. *
+* Retour : Date de création de l'élément. *
* *
* Remarques : - *
* *
@@ -541,56 +541,9 @@ char *g_db_item_get_label(GDbItem *item)
timestamp_t g_db_item_get_timestamp(const GDbItem *item)
{
- return item->timestamp;
+ timestamp_t result; /* Horodatage à retourner */
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : item = élément de collection à mettre à jour. *
-* binary = binaire chargé en mémoire à modifier. *
-* first = horodatage du premier élément désactivé ou NULL. *
-* *
-* Description : Active ou désactive un élément de collection en place. *
-* *
-* Retour : Bilan de la mise à jour de l'élément. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-bool g_db_item_set_activity(GDbItem *item, GLoadedBinary *binary, timestamp_t *first)
-{
- bool result; /* Bilan à faire remonter */
- bool active; /* Etat avant changement */
-
- active = g_db_item_is_active(item);
-
- /* Archivage de l'état */
-
- if (first == NULL)
- {
- assert(!active);
- item->timestamp = item->created;
- }
- else
- {
- assert(active);
- item->timestamp = --(*first);
- }
-
- /* Application de l'état */
-
- if (binary != NULL)
- {
- if (active)
- result = g_db_item_cancel(item, binary);
- else
- result = g_db_item_apply(item, binary);
- }
- else
- result = true;
+ result = item->created;
return result;
@@ -599,19 +552,20 @@ bool g_db_item_set_activity(GDbItem *item, GLoadedBinary *binary, timestamp_t *f
/******************************************************************************
* *
-* Paramètres : item = élément de collection à consulter. *
+* Paramètres : item = base d'éléments à mettre à jour. *
+* flag = type de propriété à traiter. *
* *
-* Description : Indique si l'élément est activé ou désactivé. *
+* Description : Applique un ensemble de propriétés à un élément. *
* *
-* Retour : Etat de l'activité de l'élément. *
+* Retour : - *
* *
* Remarques : - *
* *
******************************************************************************/
-bool g_db_item_is_active(const GDbItem *item)
+void g_db_item_set_flags(GDbItem *item, DbItemFlags flag)
{
- return (cmp_timestamp(&item->created, &item->timestamp) == 0);
+ g_atomic_int_set(&item->atomic_flags, flag);
}
@@ -684,6 +638,50 @@ DbItemFlags g_db_item_get_flags(const GDbItem *item)
}
+/******************************************************************************
+* *
+* Paramètres : item = élément de collection à mettre à jour. *
+* binary = binaire chargé en mémoire à modifier. *
+* *
+* Description : Active ou désactive un élément de collection en place. *
+* *
+* Retour : Bilan de la mise à jour de l'élément. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+bool g_db_item_switch_state(GDbItem *item, GLoadedBinary *binary)
+{
+ bool result; /* Bilan à faire remonter */
+ bool enabled; /* Etat avant changement */
+
+ enabled = !g_db_item_has_flag(item, DIF_DISABLED);
+
+ /* Archivage de l'état */
+
+ if (enabled)
+ g_db_item_add_flag(item, DIF_DISABLED);
+ else
+ g_db_item_remove_flag(item, DIF_DISABLED);
+
+ /* Application de l'état */
+
+ if (binary != NULL)
+ {
+ if (enabled)
+ result = g_db_item_cancel(item, binary);
+ else
+ result = g_db_item_apply(item, binary);
+ }
+ else
+ result = true;
+
+ return result;
+
+}
+
+
/* ---------------------------------------------------------------------------------- */
/* MANIPULATIONS AVEC UNE BASE DE DONNEES */
@@ -692,6 +690,30 @@ DbItemFlags g_db_item_get_flags(const GDbItem *item)
/******************************************************************************
* *
+* Paramètres : item = base d'éléments à consulter. *
+* values = tableau d'éléments à compléter. [OUT] *
+* count = nombre de descriptions renseignées. [OUT] *
+* *
+* Description : Décrit les colonnes utiles à un chargement de données. *
+* *
+* Retour : Bilan de l'opération. *
+* *
+* Remarques : - *
+* *
+******************************************************************************/
+
+bool g_db_item_setup_load(const GDbItem *item, bound_value **values, size_t *count)
+{
+ *values = NULL;
+ *count = 0;
+
+ return G_DB_ITEM_GET_CLASS(item)->store(NULL, values, count);
+
+}
+
+
+/******************************************************************************
+* *
* Paramètres : item = base d'éléments à charger depuis les réponses. *
* values = tableau d'éléments à consulter. *
* count = nombre de descriptions renseignées. *
@@ -707,14 +729,23 @@ DbItemFlags g_db_item_get_flags(const GDbItem *item)
static bool _g_db_item_load(GDbItem *item, const bound_value *values, size_t count)
{
bool result; /* Bilan global à retourner */
+ const bound_value *value; /* Valeur à intégrer */
result = load_timestamp(&item->created, "created", values, count);
if (result)
- result = load_timestamp(&item->timestamp, "timestamp", values, count);
+ result = load_rle_string(&item->author, "author", values, count);
if (result)
- result = load_rle_string(&item->author, "author", values, count);
+ {
+ value = find_bound_value(values, count, "flags");
+
+ result = (value != NULL && value->type == SQLITE_INTEGER);
+
+ if (result)
+ item->flags = value->integer;
+
+ }
return result;
@@ -763,6 +794,7 @@ bool g_db_item_load(GDbItem *item, const bound_value *values, size_t count)
static bool _g_db_item_store(const GDbItem *item, bound_value **values, size_t *count)
{
bool result; /* Bilan à retourner */
+ bound_value *value; /* Valeur à éditer / définir */
if (item == NULL)
result = store_timestamp(NULL, "created", values, count);
@@ -772,17 +804,26 @@ static bool _g_db_item_store(const GDbItem *item, bound_value **values, size_t *
if (result)
{
if (item == NULL)
- result = store_timestamp(NULL, "timestamp", values, count);
+ result = store_rle_string(NULL, "author", values, count);
else
- result = store_timestamp(&item->timestamp, "timestamp", values, count);
+ result = store_rle_string(&item->author, "author", values, count);
}
if (result)
{
- if (item == NULL)
- result = store_rle_string(NULL, "author", values, count);
- else
- result = store_rle_string(&item->author, "author", values, count);
+ *values = realloc(*values, ++(*count) * sizeof(bound_value));
+
+ value = &(*values)[*count - 1];
+
+ value->cname = "flags";
+ value->built_name = false;
+ value->type = SQLITE_INTEGER;
+
+ value->has_value = (item != NULL);
+
+ if (value->has_value)
+ value->integer = g_db_item_get_flags(item);
+
}
return result;
diff --git a/src/analysis/db/item.h b/src/analysis/db/item.h
index 395a56f..b735fb8 100644
--- a/src/analysis/db/item.h
+++ b/src/analysis/db/item.h
@@ -48,6 +48,7 @@ typedef enum _DbItemFlags
DIF_UPDATED = (1 << 1), /* Mise à jour de l'élément */
DIF_VOLATILE = (1 << 2), /* Abscence de sauvegarde */
DIF_BROKEN = (1 << 3), /* Application impossible */
+ DIF_DISABLED = (1 << 4), /* Désactivation forcée */
} DbItemFlags;
@@ -109,11 +110,8 @@ char *g_db_item_get_label(GDbItem *);
/* Fournit l'horodatage associé à l'élément de collection. */
timestamp_t g_db_item_get_timestamp(const GDbItem *);
-/* Active ou désactive un élément de collection en place. */
-bool g_db_item_set_activity(GDbItem *, GLoadedBinary *, timestamp_t *);
-
-/* Indique si l'élément est activé ou désactivé. */
-bool g_db_item_is_active(const GDbItem *);
+/* Applique un ensemble de propriétés à un élément. */
+void g_db_item_set_flags(GDbItem *, DbItemFlags);
/* Ajoute une propriété à un élément de base de données. */
void g_db_item_add_flag(GDbItem *, DbItemFlags);
@@ -124,11 +122,25 @@ void g_db_item_remove_flag(GDbItem *, DbItemFlags);
/* Indique les propriétés particulières appliquées à l'élément. */
DbItemFlags g_db_item_get_flags(const GDbItem *);
+#define g_db_item_has_flag(i, f) \
+ (g_db_item_get_flags(i) & f)
+
+
+//////////
+#define g_db_item_is_enabled(i) ((g_db_item_get_flags(i) & DIF_DISABLED) == 0)
+
+
+/* Active ou désactive un élément de collection en place. */
+bool g_db_item_switch_state(GDbItem *, GLoadedBinary *);
+
/* --------------------- MANIPULATIONS AVEC UNE BASE DE DONNEES --------------------- */
+/* Décrit les colonnes utiles à un chargement de données. */
+bool g_db_item_setup_load(const GDbItem *, bound_value **, size_t *);
+
/* Charge les valeurs utiles pour un élément de collection. */
bool g_db_item_load(GDbItem *, const bound_value *, size_t);
diff --git a/src/analysis/db/items/bookmark.c b/src/analysis/db/items/bookmark.c
index 3d66f6b..c1b0fec 100644
--- a/src/analysis/db/items/bookmark.c
+++ b/src/analysis/db/items/bookmark.c
@@ -141,9 +141,6 @@ static void g_bookmark_collection_finalize(GBookmarkCollection *);
/* Crée la table des signets dans une base de données. */
static bool g_bookmark_collection_create_db_table(const GBookmarkCollection *, sqlite3 *);
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool g_bookmark_collection_setup_load(GBookmarkCollection *, bound_value **, size_t *);
-
/* Détermine si un élément est déjà présent ou non. */
static GDbItem *g_bookmark_collection_has_key(GBookmarkCollection *, va_list);
@@ -796,7 +793,6 @@ static void g_bookmark_collection_class_init(GBookmarkCollectionClass *klass)
collec = G_DB_COLLECTION_CLASS(klass);
collec->create_table = (collec_create_db_table_fc)g_bookmark_collection_create_db_table;
- collec->setup_load = (collec_setup_load_fc)g_bookmark_collection_setup_load;
collec->has_key = (collec_has_key_fc)g_bookmark_collection_has_key;
}
@@ -934,39 +930,6 @@ static bool g_bookmark_collection_create_db_table(const GBookmarkCollection *col
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
-* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_bookmark_collection_setup_load(GBookmarkCollection *collec, bound_value **values, size_t *count)
-{
- bool status; /* Bilan d'une préparation */
-
- status = G_DB_COLLECTION_CLASS(g_bookmark_collection_parent_class)->setup_load(G_DB_COLLECTION(collec), \
- values, count);
- if (!status) return false;
-
- if (!store_vmpa(NULL, NULL, values, count))
- return false;
-
- if (!store_rle_string(NULL, "comment", values, count))
- return false;
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
* ap = clef identifiant de manière unique un élément. *
* *
* Description : Détermine si un élément est déjà présent ou non. *
@@ -989,6 +952,7 @@ static GDbItem *g_bookmark_collection_has_key(GBookmarkCollection *collec, va_li
ref = va_arg(ap, vmpa2t *);
+#if 0
items = g_db_collection_get_items(G_DB_COLLECTION(collec));
for (iter = g_list_first(items); iter != NULL && result == NULL; iter = g_list_next(iter))
@@ -1005,6 +969,7 @@ static GDbItem *g_bookmark_collection_has_key(GBookmarkCollection *collec, va_li
result = G_DB_ITEM(bm);
}
+#endif
return result;
diff --git a/src/analysis/db/items/comment.c b/src/analysis/db/items/comment.c
index a987464..a231749 100644
--- a/src/analysis/db/items/comment.c
+++ b/src/analysis/db/items/comment.c
@@ -183,9 +183,6 @@ static void g_comment_collection_finalize(GCommentCollection *);
/* Crée la table des commentaires dans une base de données. */
static bool g_comment_collection_create_db_table(const GCommentCollection *, sqlite3 *);
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool g_comment_collection_setup_load(GCommentCollection *, bound_value **, size_t *);
-
/* Détermine si un élément est déjà présent ou non. */
static GDbItem *g_comment_collection_has_key(GCommentCollection *, va_list);
@@ -995,7 +992,7 @@ static bool g_db_comment_load(GDbComment *comment, const bound_value *values, si
if (result)
{
- value = find_bound_value(values, count, "flags");
+ value = find_bound_value(values, count, "lflags");
result = (value != NULL && value->type == SQLITE_INTEGER);
if (result)
@@ -1072,16 +1069,20 @@ static bool g_db_comment_store(GDbComment *comment, bound_value **values, size_t
value = &(*values)[*count - 1];
- value->cname = "flags";
+ value->cname = "lflags";
value->built_name = false;
value->type = SQLITE_INTEGER;
- value->integer = comment->flags;
- value->delete = NULL;
value->has_value = (comment != NULL);
if (value->has_value)
{
+ value->integer = comment->flags;
+ value->delete = NULL;
+ }
+
+ if (value->has_value)
+ {
init_dynamic_rle_string(&text, g_db_comment_get_text(comment));
status = store_rle_string(&text, "text", values, count);
exit_rle_string(&text);
@@ -1514,7 +1515,6 @@ static void g_comment_collection_class_init(GCommentCollectionClass *klass)
collec = G_DB_COLLECTION_CLASS(klass);
collec->create_table = (collec_create_db_table_fc)g_comment_collection_create_db_table;
- collec->setup_load = (collec_setup_load_fc)g_comment_collection_setup_load;
collec->has_key = (collec_has_key_fc)g_comment_collection_has_key;
}
@@ -1626,7 +1626,7 @@ static bool g_comment_collection_create_db_table(const GCommentCollection *colle
sql = "CREATE TABLE Comments (" \
SQLITE_DB_ITEM_CREATE ", " \
"%s, " \
- "flags INTEGER, " \
+ "lflags INTEGER, " \
SQLITE_RLESTR_CREATE("text") ", " \
"inlined INTEGER, " \
"repeatable INTEGER" \
@@ -1655,64 +1655,6 @@ static bool g_comment_collection_create_db_table(const GCommentCollection *colle
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
-* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_comment_collection_setup_load(GCommentCollection *collec, bound_value **values, size_t *count)
-{
- bool status; /* Bilan d'une préparation */
- bound_value *value; /* Valeur à éditer / définir */
-
- status = G_DB_COLLECTION_CLASS(g_comment_collection_parent_class)->setup_load(G_DB_COLLECTION(collec), \
- values, count);
- if (!status) return false;
-
- if (!store_vmpa(NULL, NULL, values, count))
- return false;
-
- *count += 1;
- *values = realloc(*values, *count * sizeof(bound_value));
-
- value = &(*values)[*count - 1];
-
- value->cname = "flags";
- value->built_name = false;
- value->type = SQLITE_INTEGER;
-
- if (!store_rle_string(NULL, "text", values, count))
- return false;
-
- *count += 2;
- *values = realloc(*values, *count * sizeof(bound_value));
-
- value = &(*values)[*count - 2];
-
- value->cname = "inlined";
- value->built_name = false;
- value->type = SQLITE_BOOLEAN;
-
- value = &(*values)[*count - 1];
-
- value->cname = "repeatable";
- value->built_name = false;
- value->type = SQLITE_BOOLEAN;
-
- return true;
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
* ap = clef identifiant de manière unique un élément. *
* *
* Description : Détermine si un élément est déjà présent ou non. *
@@ -1735,6 +1677,7 @@ static GDbItem *g_comment_collection_has_key(GCommentCollection *collec, va_list
ref = va_arg(ap, vmpa2t *);
+#if 0
items = g_db_collection_get_items(G_DB_COLLECTION(collec));
for (iter = g_list_first(items); iter != NULL && result == NULL; iter = g_list_next(iter))
@@ -1751,6 +1694,7 @@ static GDbItem *g_comment_collection_has_key(GCommentCollection *collec, va_list
result = G_DB_ITEM(bm);
}
+#endif
return result;
diff --git a/src/analysis/db/items/move.c b/src/analysis/db/items/move.c
index a5d2773..afc3331 100644
--- a/src/analysis/db/items/move.c
+++ b/src/analysis/db/items/move.c
@@ -136,9 +136,6 @@ static void g_move_collection_finalize(GMoveCollection *);
/* Crée la table des basculements dans une base de données. */
static bool g_move_collection_create_db_table(const GMoveCollection *, sqlite3 *);
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool g_move_collection_setup_load(GMoveCollection *, bound_value **, size_t *);
-
/* Détermine si un élément est déjà présent ou non. */
static GDbItem *g_move_collection_has_key(GMoveCollection *, va_list);
@@ -638,7 +635,6 @@ static void g_move_collection_class_init(GMoveCollectionClass *klass)
collec = G_DB_COLLECTION_CLASS(klass);
collec->create_table = (collec_create_db_table_fc)g_move_collection_create_db_table;
- collec->setup_load = (collec_setup_load_fc)g_move_collection_setup_load;
collec->has_key = (collec_has_key_fc)g_move_collection_has_key;
}
@@ -779,40 +775,6 @@ static bool g_move_collection_create_db_table(const GMoveCollection *collec, sql
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
-* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_move_collection_setup_load(GMoveCollection *collec, bound_value **values, size_t *count)
-{
- bool status; /* Bilan d'une préparation */
-
- status = G_DB_COLLECTION_CLASS(g_move_collection_parent_class)->setup_load(G_DB_COLLECTION(collec), \
- values, count);
- if (!status) return false;
-
- if (!g_binary_cursor_store(NULL, "src", values, count))
- return false;
-
- if (!g_binary_cursor_store(NULL, "dest", values, count))
- return false;
-
- return true;
-
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
* ap = clef identifiant de manière unique un élément. *
* *
* Description : Détermine si un élément est déjà présent ou non. *
@@ -835,6 +797,7 @@ static GDbItem *g_move_collection_has_key(GMoveCollection *collec, va_list ap)
ref = va_arg(ap, const GLineCursor *);
+#if 0
items = g_db_collection_get_items(G_DB_COLLECTION(collec));
for (iter = g_list_first(items); iter != NULL && result == NULL; iter = g_list_next(iter))
@@ -851,6 +814,7 @@ static GDbItem *g_move_collection_has_key(GMoveCollection *collec, va_list ap)
result = G_DB_ITEM(bm);
}
+#endif
return result;
diff --git a/src/analysis/db/items/switcher.c b/src/analysis/db/items/switcher.c
index 043effd..94d4940 100644
--- a/src/analysis/db/items/switcher.c
+++ b/src/analysis/db/items/switcher.c
@@ -136,9 +136,6 @@ static void g_switcher_collection_finalize(GSwitcherCollection *);
/* Crée la table des basculements dans une base de données. */
static bool g_switcher_collection_create_db_table(const GSwitcherCollection *, sqlite3 *);
-/* Décrit les colonnes utiles à un chargement de données. */
-static bool g_switcher_collection_setup_load(GSwitcherCollection *, bound_value **, size_t *);
-
/* Détermine si un élément est déjà présent ou non. */
static GDbItem *g_switcher_collection_has_key(GSwitcherCollection *, va_list);
@@ -790,7 +787,6 @@ static void g_switcher_collection_class_init(GSwitcherCollectionClass *klass)
collec = G_DB_COLLECTION_CLASS(klass);
collec->create_table = (collec_create_db_table_fc)g_switcher_collection_create_db_table;
- collec->setup_load = (collec_setup_load_fc)g_switcher_collection_setup_load;
collec->has_key = (collec_has_key_fc)g_switcher_collection_has_key;
}
@@ -929,53 +925,6 @@ static bool g_switcher_collection_create_db_table(const GSwitcherCollection *col
/******************************************************************************
* *
* Paramètres : collec = ensemble d'éléments à consulter. *
-* values = tableau d'éléments à compléter. [OUT] *
-* count = nombre de descriptions renseignées. [OUT] *
-* *
-* Description : Décrit les colonnes utiles à un chargement de données. *
-* *
-* Retour : Bilan de l'opération. *
-* *
-* Remarques : - *
-* *
-******************************************************************************/
-
-static bool g_switcher_collection_setup_load(GSwitcherCollection *collec, bound_value **values, size_t *count)
-{
- bool status; /* Bilan d'une préparation */
- bound_value *value; /* Valeur à éditer / définir */
-
- status = G_DB_COLLECTION_CLASS(g_switcher_collection_parent_class)->setup_load(G_DB_COLLECTION(collec), \
- values, count);
- if (!status) return false;
-
- if (!store_vmpa(NULL, NULL, values, count))
- return false;
-
- *count += 2;
- *values = realloc(*values, *count * sizeof(bound_value));
-
- value = &(*values)[*count - 2];
-
- value->cname = "op_index";
- value->built_name = false;
- value->type = SQLITE_INTEGER;
-
- value = &(*values)[*count - 1];
-
- value->cname = "type";
- value->built_name = false;
- value->type = SQLITE_INTEGER;
-
- return true;
-
-
-}
-
-
-/******************************************************************************
-* *
-* Paramètres : collec = ensemble d'éléments à consulter. *
* ap = clef identifiant de manière unique un élément. *
* *
* Description : Détermine si un élément est déjà présent ou non. *
@@ -998,6 +947,7 @@ static GDbItem *g_switcher_collection_has_key(GSwitcherCollection *collec, va_li
ref = va_arg(ap, vmpa2t *);
+#if 0
items = g_db_collection_get_items(G_DB_COLLECTION(collec));
for (iter = g_list_first(items); iter != NULL && result == NULL; iter = g_list_next(iter))
@@ -1014,6 +964,7 @@ static GDbItem *g_switcher_collection_has_key(GSwitcherCollection *collec, va_li
result = G_DB_ITEM(bm);
}
+#endif
return result;
diff --git a/src/analysis/db/misc/timestamp.h b/src/analysis/db/misc/timestamp.h
index eba0fa7..4ff714e 100644
--- a/src/analysis/db/misc/timestamp.h
+++ b/src/analysis/db/misc/timestamp.h
@@ -39,8 +39,8 @@ typedef uint64_t timestamp_t;
/* Valeurs particulières */
-#define TIMESTAMP_ALL_ACTIVE 0
-#define TIMESTAMP_ALL_INACTIVE 1
+#define TIMESTAMP_ALL_ACTIVE 0 // REMME
+#define TIMESTAMP_ALL_INACTIVE 1 // REMME
/* Obtient un horodatage initialisé au moment même. */
diff --git a/src/analysis/db/protocol.h b/src/analysis/db/protocol.h
index 5c3eed6..d429530 100644
--- a/src/analysis/db/protocol.h
+++ b/src/analysis/db/protocol.h
@@ -95,7 +95,6 @@ typedef enum _DBAction
{
DBA_ADD_ITEM, /* Ajout d'un élément */
DBA_REM_ITEM, /* Suppression d'un élément */
-
DBA_CHANGE_STATE, /* Changement d'activité */
DBA_COUNT
@@ -186,16 +185,16 @@ typedef enum _DBCommand
* avec une série de paquets de la forme :
*
* [ Traitement de collection : DBC_COLLECTION ]
- * [ Action : DBC_SET_LAST_ACTIVE ]
+ * [ Action : DBA_CHANGE_STATE ]
* [ <élément dont le statut a évolué> ]
*
* Les traitements se réalisent dans :
- * - g_db_collection_set_last_active() pour la partie serveur.
- * - g_db_client_set_last_active() pour la partie client.
+ * - g_db_collection_disable_at() pour la partie serveur.
+ * - g_db_collection_update_item_state() pour la partie client.
*
*/
- DBC_SET_LAST_ACTIVE, /* Définition du dernier actif */
+ DBC_SET_LAST_ACTIVE, /* Définition du dernier actif */ // REMME
DBC_COUNT