Pages : 1
#1 Le 04/01/2006, à 20:28
- Bzh
Easy-Sauvegarde
Bonjour !
Mettant tellement galéré pour configurer mon écran et les accents, que je me suis décidé de me lancer dans la création d'un petit utilitaire permettant de sauvegarder et restaurer la config.
Langage: C/GTK
Il sauvegarde pour l'instant quatres fichiers:
1) /etc/X11/xorg.conf
2) /etc/environment
3) /etc/locale.gen
4) /etc/apt/sources.list
Je compte bien le développer afin de pouvoir sauvegarder d'autre paramettre. Ajouter aussi une vérification md5 et pourquoi pas de compresser le fichier généré.
Voici une copie d'écran:
Et là le code source:
//Inclusion des librairies
#include <stdio.h>
#include <string.h>
#include <gtk/gtk.h>
#include <stdlib.h>
//Déclaration des constantes des noms des fichiers
#define CHEMIN_XORG "/etc/X11/xorg.conf"
#define CHEMIN_ENVIRONMENT "/etc/environment"
#define CHEMIN_LOCAL "/etc/locale.gen"
#define CHEMIN_SOURCES "/etc/apt/sources.list"
//Déclaration des différents textes présent dans le programme
#define TEXTE_1 TEXTE_7" vous permettra de sauvegarder,\nen un clique, toute la configuration de votre installation\nUbuntu et, ainsi, la restaurer lors d'une prochaine\nréinstallation."
#define TEXTE_2 "Cet utilitaire,developpé en C/Gtk, permet d'exporter la configuration systeme de votre ordinateur afin de la restaurer lorsque vous le réinstallerez.\n\nVoici la listes des fichiers sauvegardés:\n1) "CHEMIN_XORG"\n2) "CHEMIN_ENVIRONMENT"\n3) "CHEMIN_LOCAL"\n4) "CHEMIN_SOURCES"\n\nVersion: 0.0.2\nDate: 06/01/06"
#define TEXTE_3 "Où voulez-vous sauvegarder votre configuration ?"
#define TEXTE_4 "Sauvegarder"
#define TEXTE_5 "Restaurer"
#define TEXTE_6 "À propos"
#define TEXTE_7 "Easy-Sauvegarde"
#define TEXTE_8 "La sauvegarde s'est corretement déroulée"
#define TEXTE_9 "Ouvrez le dossier où se trouve la sauvegarde..."
#define TEXTE_10 "La restauration de votre configuration s'est bien déroulée.\n\nUne copie des fichiés originaux a été faite dans leurs répertoires."
#define NOM_FICHIER_SAUVEGARDE "ConfigSauvegarde"
#define ERREUR_1 "Impossible d'ouvrir le fichier. Vérifiez que vous êtes bien en administrateur."
#define ERREUR_2 "Attention:\nLe fichier de sauvegarde ne peut être réutilisé car il comporte des erreurs. Il a sans doute du être modifié entre temps."
#define ERREUR_3 "Attention:\nUn des fichiers n'a pas été trouvé sur votre ordinateur. La sauvegarde ne peut pas être faite."
#define ERREUR_4 "Attention:\nAucun fichier correspondant n'a été trouvé dans ce répertoire.\n\nLe fichier ce nomme "NOM_FICHIER_SAUVEGARDE"."
//Déclaration des fonctions
void sauvegarde();
void restauration();
void a_propos();
void erreur(int erreur);
////////////////////////////////////////////////////////////////////////////////
//Fonction principale
int main(int argc, char **argv)
{
//Déclaration des Widget
GtkWidget *pWindow;
GtkWidget *pVBox;
GtkWidget *pHBox;
GtkWidget *pLabel;
GtkWidget *pButton[2];
GtkWidget *pFrame;
//Initiation de GTK
gtk_init(&argc,&argv);
//Création de notre fenêtre principale
pWindow = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(pWindow), TEXTE_7 );
gtk_window_set_default_size(GTK_WINDOW(pWindow), 385, 160);
gtk_window_set_position(GTK_WINDOW(pWindow), GTK_WIN_POS_CENTER);
g_signal_connect(G_OBJECT(pWindow), "destroy", G_CALLBACK(gtk_main_quit), NULL);
gtk_container_set_border_width(GTK_CONTAINER (pWindow), 10);
//Déclaration des labels
pLabel = gtk_label_new( TEXTE_1 );
//Déclaration des bouttons
pButton[0] = gtk_button_new_with_label( TEXTE_4 );
pButton[1] = gtk_button_new_with_label( TEXTE_5 );
pButton[2] = gtk_button_new_with_label( TEXTE_6 );
//Association des actions aux boutons
g_signal_connect(G_OBJECT( pButton[0] ), "clicked", sauvegarde, NULL);
g_signal_connect(G_OBJECT( pButton[1] ), "clicked", restauration, NULL);
g_signal_connect(G_OBJECT( pButton[2] ), "clicked", a_propos, NULL);
//Construction des conteneurs
pVBox = gtk_vbox_new(FALSE, 0);
pHBox = gtk_hbox_new(FALSE, 3);
//Construction des frames
pFrame = gtk_frame_new("Avant propos:");
//Ajout dans les conteneurs
gtk_container_add(GTK_CONTAINER(pWindow), pVBox);
gtk_box_pack_start(GTK_BOX(pVBox), pFrame, false, false, 0);
gtk_container_add(GTK_CONTAINER( pFrame ), pLabel );
gtk_box_pack_end(GTK_BOX(pVBox), pHBox, false, false, 0);
gtk_box_pack_start(GTK_BOX(pHBox), pButton[0], false, false, 10);
gtk_box_pack_start(GTK_BOX(pHBox), pButton[1], false, false, 10);
gtk_box_pack_end(GTK_BOX(pHBox), pButton[2], false, false, 10);
//Affichage de notre fenêtre
gtk_widget_show_all(pWindow);
//Lancement de notre boucle
gtk_main();
return 0;
}
////////////////////////////////////////////////////////////////////////////////
//Fonction de sauvegarde
void sauvegarde(){
//Déclaration des Widget et des variables
GtkWidget *pFileSelection;
GtkWidget *pDialog;
gchar *sChemin;
char adresse[512];
char buffer;
int resultat_choix;
//Création de la fenêtre du choix de l'emplacement pour la sauvegarde
pFileSelection = gtk_file_chooser_dialog_new( TEXTE_3 ,
GTK_WINDOW(NULL),
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_OK,
NULL);
//Affichage de la fenêtre
resultat_choix = gtk_dialog_run(GTK_DIALOG(pFileSelection));
//Traitement du retour
if( resultat_choix == GTK_RESPONSE_OK ){
//Récupération du chemin
sChemin = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(pFileSelection));
//Déclaration des pointeurs FILE
FILE *fichier_destination;
FILE *fichier_lecture;
//Création de l'adresse complète
strcpy(adresse,sChemin);
strcat(adresse,"/"NOM_FICHIER_SAUVEGARDE);
//Libération de la mémoire aloué pour sChemin
g_free(sChemin);
//Ouverture du fichier de destination
fichier_destination = fopen(adresse,"w");
//Ouverture de xorg en simple lecture
fichier_lecture = fopen(CHEMIN_XORG,"r");
//Vérification d'aucune erreur
if( fichier_destination == false || fichier_lecture == false){ fclose(fichier_destination);
remove(adresse);
erreur(3) ; }
//Ecriture premières lignes pour les informations
if( fputs("Ce fichier a été généré par \""TEXTE_7"\".\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("Ne surtout pas le modifier. Cela pourrait abimer votre configuration\n\n\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs(CHEMIN_XORG,fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("\n==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
//Copie du fichier xorg
while( ( buffer = (char)getc(fichier_lecture) ) != EOF ){ if( putc(buffer,fichier_destination) == EOF ){ erreur(1) ; } }
//Changement du fichier en lecture
fclose(fichier_lecture);
fichier_lecture = fopen( CHEMIN_ENVIRONMENT ,"r");
//Vérification d'aucune erreur
if( fichier_lecture == false){ fclose(fichier_destination);
remove(adresse);
erreur(3) ; }
//Indication de changement de fichier
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs( CHEMIN_ENVIRONMENT ,fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("\n==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
//Copie du fichier ENVIRONMENT
while( ( buffer = (char)getc(fichier_lecture) ) != EOF ){ if( putc(buffer,fichier_destination) == EOF ){ erreur(1) ; } }
//Changement du fichier en lecture
fclose(fichier_lecture);
fichier_lecture = fopen( CHEMIN_LOCAL ,"r");
//Vérification d'aucune erreur
if( fichier_lecture == false){ fclose(fichier_destination);
remove(adresse);
erreur(3) ; }
//Indication de changement de fichier
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs( CHEMIN_LOCAL ,fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("\n==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
//Copie du fichier ENVIRONMENT
while( ( buffer = (char)getc(fichier_lecture) ) != EOF ){ if( putc(buffer,fichier_destination) == EOF ){ erreur(1) ; } }
//Changement du fichier en lecture
fclose(fichier_lecture);
fichier_lecture = fopen( CHEMIN_SOURCES ,"r");
//Vérification d'aucune erreur
if( fichier_lecture == false){ fclose(fichier_destination);
remove(adresse);
erreur(3) ; }
//Indication de changement de fichier
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs( CHEMIN_SOURCES ,fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("\n==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
//Copie du fichier SOURCES
while( ( buffer = (char)getc(fichier_lecture) ) != EOF ){ if( putc(buffer,fichier_destination) == EOF ){ erreur(1) ; } }
//Indication de la fin de la sauvegarde
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("Fin\n",fichier_destination) == EOF ){ erreur(1) ; }
if( fputs("==============================================================================\n",fichier_destination) == EOF ){ erreur(1) ; }
//Suppresion des pointeurs FILE
fclose(fichier_destination);
fclose(fichier_lecture);
//Affichage du message de confirmation
pDialog = gtk_message_dialog_new(GTK_WINDOW(pFileSelection),
GTK_DIALOG_MODAL,
GTK_MESSAGE_INFO,
GTK_BUTTONS_OK,
TEXTE_8 );
gtk_dialog_run(GTK_DIALOG(pDialog));
gtk_widget_destroy(pDialog);
}
//Destruction de la fenêtre
gtk_widget_destroy(pFileSelection);
return;
}
////////////////////////////////////////////////////////////////////////////////
//Fonction A propos
void restauration(){
//Déclaration des Widget et des variables
GtkWidget *pFileSelection;
GtkWidget *pDialog;
gchar *sChemin;
char adresse[512];
char buffer;
int resultat_choix;
int compteur = 0;
char *memoire;
char *debut;
char *fin;
//Création de la fenêtre du choix de l'emplacement pour la sauvegarde
pFileSelection = gtk_file_chooser_dialog_new( TEXTE_9 ,
GTK_WINDOW(NULL),
GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_OK,
NULL);
//Affichage de la fenêtre
resultat_choix = gtk_dialog_run(GTK_DIALOG(pFileSelection));
//Traitement du retour
if( resultat_choix == GTK_RESPONSE_OK ){
//Récupération du chemin
sChemin = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(pFileSelection));
//Déclaration des pointeurs FILE
FILE *fichier;
//Création de l'adresse complète
strcpy(adresse,sChemin);
strcat(adresse,"/"NOM_FICHIER_SAUVEGARDE);
//Libération de la mémoire aloué pour sChemin
g_free(sChemin);
//Ouverture du fichier en lecture
fichier = fopen( adresse , "r" );
//Vérification d'aucune erreur
if( fichier == false){ erreur(4) ; }
//Récupération de la taille du fichier afin de le charger en mémoire
while( getc( fichier ) != EOF ){ compteur++ ; }
//Rembobinage du fichier
rewind(fichier);
//Allocation de la mémoire nécessaire
memoire = (char*)malloc(compteur + 1);
//On charge le fichier en mémoire
compteur = 0;
while( ( buffer = (char)getc( fichier ) ) != EOF ){
*(memoire + compteur ) = buffer;
compteur++;
}
//Fermeture du fichier_lecture
fclose( fichier );
//Sauvegarde des fichiers avant installation
if( 0 != rename(CHEMIN_LOCAL, CHEMIN_LOCAL".bak") ){ erreur(1) ; }
if( 0 != rename(CHEMIN_ENVIRONMENT, CHEMIN_ENVIRONMENT".bak") ){ erreur(1) ; }
if( 0 != rename(CHEMIN_XORG, CHEMIN_XORG".bak") ){ erreur(1) ; }
if( 0 != rename(CHEMIN_SOURCES, CHEMIN_SOURCES".bak") ){ erreur(1) ; }
////////////////////////////////////////////////////////////////////////////////
//Restauration du fichier sources.list
//Récupération du fichier CHEMIN_SOURCES
debut = strstr( memoire, "==============================================================================\n"CHEMIN_SOURCES"\n==============================================================================\n");
fin = strstr( memoire, "==============================================================================\nFin\n==============================================================================\n");
*fin = '\0';
//Vérification rapide (pas optimisé) de l'intégrité des données
if( debut == NULL || fin == NULL ){ erreur(2) ; }
//Ecriture du fichier CHEMIN_SOURCES
compteur = strlen("==============================================================================\n"CHEMIN_SOURCES"\n==============================================================================\n");
debut = debut + compteur;
fichier = fopen( CHEMIN_SOURCES, "w" );
fputs( debut, fichier );
fclose(fichier);
////////////////////////////////////////////////////////////////////////////////
//Restauration du fichier locale.gen
//Récupération du fichier CHEMIN_LOCAL
debut = strstr( memoire, "==============================================================================\n"CHEMIN_LOCAL"\n==============================================================================\n");
fin = strstr( memoire, "==============================================================================\n"CHEMIN_SOURCES"\n==============================================================================\n");
*fin = '\0';
//Vérification rapide (pas optimisé) de l'intégrité des données
if( debut == NULL || fin == NULL ){ erreur(2) ; }
//Ecriture du fichier CHEMIN_LOCAL
compteur = strlen("==============================================================================\n"CHEMIN_LOCAL"\n==============================================================================\n");
debut = debut + compteur;
fichier = fopen( CHEMIN_LOCAL, "w" );
fputs( debut, fichier );
fclose(fichier);
////////////////////////////////////////////////////////////////////////////////
//Restauration du fichier environment
//Récupération du fichier CHEMIN_ENVIRONMENT
debut = strstr( memoire, "==============================================================================\n"CHEMIN_ENVIRONMENT"\n==============================================================================\n");
fin = strstr( memoire, "==============================================================================\n"CHEMIN_LOCAL"\n==============================================================================\n");
*fin = '\0';
//Vérification rapide (pas optimisé) de l'intégrité des données
if( debut == NULL || fin == NULL ){ erreur(2) ; }
//Ecriture du fichier CHEMIN_ENVIRONMENT
compteur = strlen("==============================================================================\n"CHEMIN_ENVIRONMENT"\n==============================================================================\n");
debut = debut + compteur;
fichier = fopen( CHEMIN_ENVIRONMENT, "w" );
fputs( debut, fichier );
fclose(fichier);
////////////////////////////////////////////////////////////////////////////////
//Restauration du fichier xorg.conf
//Récupération du fichier CHEMIN_XORG
debut = strstr( memoire, "==============================================================================\n"CHEMIN_XORG"\n==============================================================================\n");
fin = strstr( memoire, "==============================================================================\n"CHEMIN_ENVIRONMENT"\n==============================================================================\n");
*fin = '\0';
//Vérification rapide (pas optimisé) de l'intégrité des données
if( debut == NULL || fin == NULL ){ erreur(2) ; }
//Ecriture du fichier CHEMIN_XORG
compteur = strlen("==============================================================================\n"CHEMIN_XORG"\n==============================================================================\n");
debut = debut + compteur;
fichier = fopen( CHEMIN_XORG, "w" );
fputs( debut, fichier );
fclose(fichier);
////////////////////////////////////////////////////////////////////////////////
//Libération de la mémoire
free(memoire);
//Affichage du message de confirmation
pDialog = gtk_message_dialog_new(GTK_WINDOW(pFileSelection),
GTK_DIALOG_MODAL,
GTK_MESSAGE_INFO,
GTK_BUTTONS_OK,
TEXTE_10);
gtk_dialog_run(GTK_DIALOG(pDialog));
gtk_widget_destroy(pDialog);
}
//Destruction de la fenêtre
gtk_widget_destroy(pFileSelection);
}
////////////////////////////////////////////////////////////////////////////////
//Fonction A propos
void a_propos(){
//Déclaration des Widget
GtkWidget *pDialog;
//Affichage puis destruction de la fenêtre
pDialog = gtk_message_dialog_new(GTK_WINDOW(NULL),
GTK_DIALOG_MODAL,
GTK_MESSAGE_INFO,
GTK_BUTTONS_OK,
TEXTE_2);
gtk_dialog_run(GTK_DIALOG(pDialog));
gtk_widget_destroy(pDialog);
}
////////////////////////////////////////////////////////////////////////////////
//Fonction Gestion des erreurs
void erreur(int erreur){
GtkWidget *pDialog;
char message[256];
//Chargement du message d'erreur
if( erreur == 1 ){ strcpy(message,ERREUR_1); }
if( erreur == 2 ){ strcpy(message,ERREUR_2); }
if( erreur == 3 ){ strcpy(message,ERREUR_3); }
if( erreur == 4 ){ strcpy(message,ERREUR_4); }
//Affichage puis destruction de la fenêtre
pDialog = gtk_message_dialog_new(GTK_WINDOW(NULL),
GTK_DIALOG_MODAL,
GTK_MESSAGE_INFO,
GTK_BUTTONS_OK,
message );
gtk_dialog_run(GTK_DIALOG(pDialog));
gtk_widget_destroy(pDialog);
//On quitte le programme
exit(1);
}
Et le lien pour télécharger le programme déja compilé: http://bzh.cybersecurite.free.fr/codes/ … rde.tar.gz
Toute critique est la bien venue...
Bye...
Edit: Mise à jours du code pour la dernière version : Version: 0.0.2
Date: 06/01/06
Dernière modification par Bzh (Le 06/01/2006, à 20:19)
Hors ligne
#2 Le 06/01/2006, à 15:45
- SombrErrancE
Re : Easy-Sauvegarde
Super initiative !!!
A quand un easy ubuntu ??? lol
note à part, ca pourrai surement être sympa de l'intégré dans ubuntu si le sytème s'avère simple et efficace
bonne chance à toi, je testerai ca plus tard.
//// SombrErrancE ////
geek v 0.12 || www.sombrerrance.fr
Hors ligne
#3 Le 06/01/2006, à 19:27
- bernez
Re : Easy-Sauvegarde
L'idée est judicieuse Bzh. Je pense qu'il y a bien d'autres fichiers importants, mais ça viendra avec le temps
vyé kanari ka fè bonsoup.
Kenavo. A galon !
Hors ligne
#4 Le 06/01/2006, à 20:13
- Bzh
Re : Easy-Sauvegarde
Bonjour !
Modification du programme :
=>Ajout du fichier /etc/apt/sources.list
=>Changement de la librairie alloc.h par stdlib.h (alloc.h est à proscrire)
=>Passage en version 0.0.2
Merci...
Mais si le code est libre c'est bien pour avoir l'aide de beaucoup de monde. Si chacun relit le code et le critique, il deviendra forcément plus fiable !!!
N'est-ce pas là la force des logiciels libres ???
Et puis, je ne pourrais qu'en profiter. Alors critiquer...
Bye...
Hors ligne
#5 Le 06/01/2006, à 20:28
- cal
Re : Easy-Sauvegarde
A mon avis, il y aurait quasiment une grosse partie voire tout le répertoire /etc/ à sauvegardé car c'est la que se situe les config de la machine et de tous les serveurs
et après si tu veux aussi pouvoir sauvegarder les config utilisateurs la ca se passe dans le home et ces la plupart des fichiers commencant par .
masi tout dépend de ce que tu veux sauvegrader aussi
Hors ligne
#6 Le 08/01/2006, à 13:56
- Bzh
Re : Easy-Sauvegarde
Je pensais à la base ne pas sauvegarder les paramètres utilisateurs mais les paramettres matériels ou plustot systèmes !!!
Disons que une fois toute la config faite, on sauvegarde et ensuite à chaque réinstallation on restaure les parametres !
Et puis, cette première version ne peut pas sauvegarder les fichiers binaires Seulement les fichiers textes !
Mais pouquoi ne pas évoluer pour gérer les fichiers binaires et aller plus loin ???? Tout dépend de la demande. S'il y a réellement un manque de ce coté là, oui...
Et puis, je ne veux pas être limité à la version Brezzy par exemple, le rêve serait que le programme fonctionnerait avec n'importe qu'elle distribution à base de Debian.
Nan ?
Hors ligne
#7 Le 10/01/2006, à 18:46
- Bzh
Re : Easy-Sauvegarde
Bon donc, tout bien réfléchit, je vais tout recommencer à zéro !!!
Continuer à sauvegarder les quatres fichiers systèmes:
1) /etc/X11/xorg.conf
2) /etc/environment
3) /etc/locale.gen
4) /etc/apt/sources.list
Et surtout, je vais en plus proposer la sauvegarde de la configuration de Gnome et des thèmes et surtout proposer la sauvegarde de la config de chacun des logiciels installés....
Je reste en C plus GTK !!!! Pour la fiabilité (vérification des erreurs) et la rapidité !!!
Le tout sera compressé en un fichier !!! Peut être vérification de l'intégrité de la sauvegarde avant de la restaurer grace à md5 par exemple.
Voila, si vous avez d'autre conseil je suis ouvert...
Hors ligne
#8 Le 10/01/2006, à 18:54
- Bobbybionic
Re : Easy-Sauvegarde
Nan ?
Si !
Non à la vente liée. Non au monopole Windows.
Tous ensemble, refusons les logiciels préinstallés et tournons nous vers le libre.
http://bobbybionic.wordpress.com
Hors ligne
#9 Le 10/01/2006, à 19:00
- artemis
Re : Easy-Sauvegarde
il faudrait qu'avec tout ses easy xxx qq'un nous sorte un truc du genre un gros pack qui fait tout ce que font les autres ou qui les rassemble car c'est vraiement de bonnes idées qui sorte de ces easy xxx
Hors ligne
#10 Le 10/01/2006, à 19:03
- Bobbybionic
Re : Easy-Sauvegarde
C'est pas faux, faudrait faire un EASY et dedans on sélectionne les bonnes idées des petits easy.
Maintenant, faudrait tout référencer, gérer les mises à jour etc... et donc s'y connaître en programmation car je ne crois pas que tous les easy utilisent le même language (ça ferait plus propre si c'était le même non ?).
Non à la vente liée. Non au monopole Windows.
Tous ensemble, refusons les logiciels préinstallés et tournons nous vers le libre.
http://bobbybionic.wordpress.com
Hors ligne
#11 Le 10/01/2006, à 22:58
- Bzh
Re : Easy-Sauvegarde
Venant de Windows, je suis attaché au C ! Me suis bien adapté à la librairie Gtk mais je ne maitrise pas du tout le shell de Linux.
Donc voilà pourquoi je code en C...
Hors ligne
Pages : 1