SVN - Guide rapide

Qu'est-ce que le système de contrôle de version?

Version Control System (VCS) est un logiciel qui aide les développeurs de logiciels à travailler ensemble et à conserver un historique complet de leur travail.

Following are the goals of a Version Control System.

  • Permettez aux développeurs de travailler simultanément.
  • N'écrasez pas les modifications les unes des autres.
  • Conservez l'historique de chaque version de tout.

A VCS is divided into two categories.

  • Système de contrôle de version centralisé (CVCS), et
  • Système de contrôle de version distribué / décentralisé (DVCS).

Dans ce tutoriel, nous nous concentrerons uniquement sur le système de contrôle de version centralisé et en particulier Subversion. Subversion relève d'un système de contrôle de version centralisé, ce qui signifie qu'il utilise un serveur central pour stocker tous les fichiers et permet la collaboration en équipe.

Terminologies de contrôle de version

Commençons par discuter de certains des termes que nous utiliserons dans ce didacticiel.

  • Repository:Un référentiel est le cœur de tout système de contrôle de version. C'est le lieu central où les développeurs stockent tout leur travail. Le référentiel stocke non seulement les fichiers mais également l'historique. Le référentiel est accessible via un réseau, agissant comme un serveur et un outil de contrôle de version agissant en tant que client. Les clients peuvent se connecter au référentiel, puis ils peuvent stocker / récupérer leurs modifications vers / depuis le référentiel. En stockant les modifications, un client met ces modifications à la disposition d'autres personnes et en récupérant les modifications, un client prend les modifications d'autres personnes comme une copie de travail.

  • Trunk: Le tronc est un répertoire où se déroule tout le développement principal et est généralement extrait par les développeurs pour travailler sur le projet.

  • Tags: Le répertoire des balises est utilisé pour stocker des instantanés nommés du projet. L'opération de balise permet de donner des noms descriptifs et mémorables à une version spécifique dans le référentiel.

    Par exemple, LAST_STABLE_CODE_BEFORE_EMAIL_SUPPORT est plus mémorable que

    UUID du référentiel: 7ceef8cb-3799-40dd-a067-c216ec2e5247 et

    Révision: 13

  • Branches:L'opération de branche est utilisée pour créer une autre ligne de développement. Il est utile lorsque vous souhaitez que votre processus de développement se déroule dans deux directions différentes. Par exemple, lorsque vous publiez la version 5.0, vous souhaiterez peut-être créer une branche afin que le développement des fonctionnalités 6.0 puisse être séparé des correctifs de bogues 5.0.

  • Working copy:La copie de travail est un instantané du référentiel. Le référentiel est partagé par toutes les équipes, mais les gens ne le modifient pas directement. Au lieu de cela, chaque développeur extrait la copie de travail. La copie de travail est un lieu de travail privé où les développeurs peuvent faire leur travail en restant isolés du reste de l'équipe.

  • Commit changes:Commit est un processus de stockage des modifications du lieu de travail privé au serveur central. Après validation, les modifications sont mises à la disposition de toute l'équipe. D'autres développeurs peuvent récupérer ces modifications en mettant à jour leur copie de travail. Commit est une opération atomique. Soit la validation complète réussit, soit elle est annulée. Les utilisateurs ne voient jamais le commit à moitié terminé.

Installation SVN

Subversion est un outil de contrôle de version open-source populaire. Il est open-source et disponible gratuitement sur Internet. Il est livré par défaut avec la plupart des distributions GNU / Linux, il est donc peut-être déjà installé sur votre système. Pour vérifier s'il est installé ou non, utilisez la commande suivante.

[jerry@CentOS ~]$ svn --version

Si le client Subversion n'est pas installé, la commande signalera une erreur, sinon elle affichera la version du logiciel installé.

[jerry@CentOS ~]$ svn --version
-bash: svn: command not found

Si vous utilisez GNU / Linux basé sur RPM, utilisez yumcommande pour l'installation. Une fois l'installation réussie, exécutez lesvn --version commander.

[jerry@CentOS ~]$ su - Password: [root@CentOS ~]# yum install subversion [jerry@CentOS ~]$ svn --version
svn, version 1.6.11 (r934486)
compiled Jun 23 2012, 00:44:03

Et si vous utilisez GNU / Linux basé sur Debian, utilisez apt commande pour l'installation.

[jerry@Ubuntu]$ sudo apt-get update [sudo] password for jerry: [jerry@Ubuntu]$ sudo apt-get install subversion

[jerry@Ubuntu]$ svn --version
svn, version 1.7.5 (r1336830)
compiled Jun 21 2013, 22:11:49

Configuration d'Apache

Nous avons vu comment installer le client Subversion sur GNU / Linux. Voyons comment créer un nouveau référentiel et permettre l'accès aux utilisateurs.

Sur le serveur, nous devons installer Apache httpd module et svnadmin outil.

[jerry@CentOS ~]$ su -
Password: 
[root@CentOS ~]# yum install mod_dav_svn subversion

le mod_dav_svn package permet d'accéder à un référentiel via HTTP, via le serveur Apache httpd et subversion package installe l'outil svnadmin.

La subversion lit sa configuration depuis /etc/httpd/conf.d/subversion.conffichier. Après avoir ajouté la configuration,subversion.conf le fichier ressemble à ceci:

LoadModule dav_svn_module     modules/mod_dav_svn.so
LoadModule authz_svn_module   modules/mod_authz_svn.so

<Location /svn>
   DAV svn
   SVNParentPath /var/www/svn
   AuthType Basic
   AuthName "Authorization Realm"
   AuthUserFile /etc/svn-users
   Require valid-user
</Location>

Créons des utilisateurs Subversion et accordons-leur l'accès au référentiel. La commande htpasswd est utilisée pour créer et mettre à jour les fichiers de texte brut qui sont utilisés pour stocker les noms d'utilisateur et les mots de passe pour l'authentification de base des utilisateurs HTTP. Les options '-c' créent un fichier de mots de passe , si le fichier de mots de passe existe déjà, il est écrasé. C'est pourquoi n'utilisez l'option '-c' que la première fois. L'option '-m' active le cryptage MD5 pour les mots de passe.

Configuration utilisateur

Laissez-nous créer un utilisateur tom.

[root@CentOS ~]# htpasswd -cm /etc/svn-users tom
New password: 
Re-type new password: 
Adding password for user tom

Laissez-nous créer un utilisateur jerry

[root@CentOS ~]# htpasswd -m /etc/svn-users jerry
New password: 
Re-type new password: 
Adding password for user jerry
[root@CentOS ~]#

Créez un répertoire parent Subversion pour stocker tout le travail (voir /etc/httpd/conf.d/subversion.conf ).

[root@CentOS ~]# mkdir /var/www/svn
[root@CentOS ~]# cd /var/www/svn/

Configuration du référentiel

Créez un référentiel de projet nommé project_repo . La commande svnadmin créera un nouveau référentiel et quelques autres répertoires à l'intérieur pour stocker les métadonnées.

[root@CentOS svn]# svnadmin create project_repo

[root@CentOS svn]# ls -l project_repo
total 24
drwxr-xr-x. 2 root root 4096 Aug  4 22:30 conf
drwxr-sr-x. 6 root root 4096 Aug  4 22:30 db
-r--r--r--. 1 root root    2 Aug  4 22:30 format
drwxr-xr-x. 2 root root 4096 Aug  4 22:30 hooks
drwxr-xr-x. 2 root root 4096 Aug  4 22:30 locks
-rw-r--r--. 1 root root  229 Aug  4 22:30 README.txt

Modifions la propriété de l'utilisateur et du groupe du référentiel.

[root@CentOS svn]# chown -R apache.apache project_repo/

Vérifiez si SELinux est activé ou non à l'aide de l'outil d'état SELinux.

[root@CentOS svn]# sestatus
SELinux status:                 enabled
SELinuxfs mount:                /selinux
Current mode:                   enforcing
Mode from config file:          enforcing
Policy version:                 24
Policy from config file:        targeted

Pour notre serveur, SELinux est activé, nous devons donc changer le contexte de sécurité SELinux.

[root@CentOS svn]# chcon -R -t httpd_sys_content_t /var/www/svn/project_repo/

Pour autoriser les validations via HTTP, exécutez la commande suivante.

[root@CentOS svn]# chcon -R -t httpd_sys_rw_content_t /var/www/svn/project_repo/

Redémarrez le serveur Apache et nous en avons terminé avec la configuration du serveur Apache.

[root@CentOS svn]# service httpd restart
Stopping httpd:                                            [FAILED]
Starting httpd: httpd: apr_sockaddr_info_get() failed for CentOS
httpd: Could not reliably determine the server's fully qualified domain name, using 127.0.0.1 for ServerName
                                                           [  OK  ]
[root@CentOS svn]# service httpd status
httpd (pid  1372) is running...
[root@CentOS svn]#

Nous avons configuré le serveur Apache avec succès, nous allons maintenant configurer le référentiel. Fournir un accès au référentiel uniquement aux utilisateurs authentiques et utiliser le fichier d'autorisation par défaut; ajoutez les lignes suivantes au fichier project_repo / conf / svnserve.conf .

anon-access = none
authz-db = authz

Conventionnellement, chaque projet Subversion a trunk, tags, et branches répertoires directement sous le répertoire racine du projet.

Le tronc est un répertoire où se déroule tout le développement principal et est généralement extrait par les développeurs pour travailler sur le projet.

Le répertoire tags est utilisé pour stocker des instantanés nommés du projet. Lors de la création d'une version de production, l'équipe marquera le code qui entre dans la version.

Le répertoire des branches est utilisé lorsque vous souhaitez poursuivre différentes lignes de développement.

Créons la structure des répertoires trunk, tags et branches sous le référentiel du projet.

[root@CentOS svn]# mkdir /tmp/svn-template
[root@CentOS svn]# mkdir /tmp/svn-template/trunk
[root@CentOS svn]# mkdir /tmp/svn-template/branches
[root@CentOS svn]# mkdir /tmp/svn-template/tags

Importez maintenant les répertoires de /tmp/svn-template au référentiel.

[root@CentOS svn]# svn import -m 'Create trunk, branches, tags directory structure' /tmp/svn-template/ 
Adding         /tmp/svn-template/trunk
Adding         /tmp/svn-template/branches
Adding         /tmp/svn-template/tags
Committed revision 1.
[root@CentOS svn]#

C'est fait! Nous avons créé avec succès le référentiel et autorisé l'accès àTom et Jerry. À partir de maintenant, ils peuvent effectuer toutes les opérations prises en charge sur le référentiel.

Le cycle de vie d'un système de contrôle de version est traité dans ce chapitre. Dans les chapitres suivants, nous verrons la commande Subversion pour chaque opération.

Créer un référentiel:

Le référentiel est un endroit central où les développeurs stockent tout leur travail. Le référentiel stocke non seulement les fichiers, mais également l'historique des modifications. Ce qui signifie qu'il conserve un historique des modifications apportées aux fichiers.

L'opération 'create' est utilisée pour créer un nouveau référentiel. La plupart du temps, cette opération n'est effectuée qu'une seule fois. Lorsque vous créez un nouveau référentiel, votre VCS s'attendra à ce que vous disiez quelque chose pour l'identifier, par exemple où vous voulez qu'il soit créé, ou quel nom doit être donné au référentiel.

Check-out

L'opération 'Checkout' est utilisée pour créer une copie de travail à partir du référentiel. La copie de travail est un lieu de travail privé où les développeurs apportent leurs modifications, puis soumettent ces modifications au référentiel.

Mettre à jour

Comme son nom l'indique, l'opération de «mise à jour» est utilisée pour mettre à jour la copie de travail. Cette opération synchronise la copie de travail avec le référentiel. Comme le référentiel est partagé par toutes les équipes, les autres développeurs peuvent valider leurs modifications et votre copie de travail vieillit.

Supposons que Tom et Jerry soient les deux développeurs travaillant sur un projet. Les deux récupèrent la dernière version du référentiel et commencent à travailler. À ce stade, leurs copies de travail sont complètement synchronisées avec le référentiel. Jerry termine son travail de manière très efficace et valide ses modifications dans le référentiel.

Maintenant , Tom la copie de travail est obsolète. L'opération de mise à jour extraira les dernières modifications de Jerry du référentiel et mettra à jour la copie de travail de Tom .

Effectuer des changements

Après la vérification, on peut effectuer diverses opérations pour effectuer des modifications. L'édition est l'opération la plus courante. On peut modifier le fichier existant pour ajouter / supprimer le contenu du fichier.

On peut ajouter des fichiers / répertoires. Mais immédiatement, ces fichiers / répertoires ne font pas partie du référentiel, à la place ils sont ajoutés à la liste de modifications en attente et deviennent une partie du référentiel après l'opération de validation.

De même, on peut supprimer des fichiers / répertoires. L'opération de suppression supprime immédiatement le fichier de la copie de travail, mais la suppression réelle du fichier est ajoutée à la liste de modifications en attente et des modifications sont apportées au référentiel après l'opération de validation.

L'opération 'Renommer' change le nom du fichier / répertoire. L'opération «Déplacer» est utilisée pour déplacer des fichiers / répertoires d'un endroit à un autre dans une arborescence de référentiels.

Examiner les modifications

Lorsque vous extrayez la copie de travail ou mettez à jour la copie de travail, votre copie de travail est complètement synchronisée avec le référentiel. Mais à mesure que vous modifiez votre copie de travail, elle devient plus récente que le référentiel. Et c'est une bonne pratique de revoir vos modifications avant l'opération «commit».

L'opération «Statut» répertorie les modifications qui ont été apportées à la copie de travail. Comme nous l'avons mentionné précédemment, chaque fois que vous effectuez des modifications dans la copie de travail, toutes ces modifications font partie de la liste de modifications en attente. Et l'opération 'status' est utilisée pour voir la liste des modifications en attente.

L'opération «Statut» ne fournit qu'une liste des modifications mais pas les détails à leur sujet. On peut utiliser l' opération diff pour afficher les détails des modifications qui ont été apportées à la copie de travail.

Corriger les erreurs

Supposons que l'on ait apporté des modifications à sa copie de travail, mais maintenant, il veut rejeter ces modifications. Dans cette situation, l'opération «retour» aidera.

L'opération Revert rétablit les modifications qui ont été apportées à la copie de travail. Il est possible de restaurer un ou plusieurs fichiers / répertoires. Il est également possible de restaurer l'intégralité de la copie de travail. Dans ce cas, l'opération 'revert' détruira la liste de modifications en attente et ramènera la copie de travail à son état d'origine.

Résoudre des conflits:

Des conflits peuvent survenir au moment de la fusion. L'opération de «fusion» gère automatiquement tout ce qui peut être fait en toute sécurité. Tout le reste est considéré comme un conflit. Par exemple, le fichier "hello.c" a été modifié dans la branche et supprimé dans une autre branche. Une telle situation oblige une personne à prendre la décision. L'opération «résoudre» est utilisée pour aider l'utilisateur à comprendre les choses et pour informer le VCS des moyens de gérer les conflits.

Valider les modifications

L'opération «Valider» est utilisée pour appliquer les modifications de la copie de travail au référentiel. Cette opération modifie le référentiel et les autres développeurs peuvent voir ces modifications en mettant à jour leur copie de travail.

Avant de commettre, il faut ajouter des fichiers / répertoires à la liste de modifications en attente. C'est ici que les modifications attendent d'être validées. Avec commit, nous fournissons généralement un message de journal pour expliquer pourquoi quelqu'un a apporté des modifications. Ce message de journal fait partie de l'historique du référentiel. La validation est une opération atomique, ce qui signifie que la validation entière réussit ou qu'elle est annulée. Les utilisateurs ne voient jamais le commit à moitié terminé.

Subversion fournit la commande d' extraction pour extraire une copie de travail d'un référentiel. La commande ci-dessous créera un nouveau répertoire dans le répertoire de travail actuel avec le nom project_repo . Ne vous inquiétez pas de l'URL du dépôt, car la plupart du temps, elle est déjà fournie par l'administrateur de subversion avec un accès approprié.

[tom@CentOS ~]$ svn checkout http://svn.server.com/svn/project_repo --username=tom

La commande ci-dessus produira le résultat suivant.

A    project_repo/trunk
A    project_repo/branches
A    project_repo/tags
Checked out revision 1.

Après chaque opération de vérification réussie, le numéro de révision sera imprimé. Si vous souhaitez afficher plus d'informations sur le référentiel, exécutez la commande info .

[tom@CentOS trunk]$ pwd
/home/tom/project_repo/trunk

[tom@CentOS trunk]$ svn info

La commande ci-dessus produira le résultat suivant.

Path: .
URL: http://svn.server.com/svn/project_repo/trunk
Repository Root: http://svn.server.com/svn/project_repo
Repository UUID: 7ceef8cb-3799-40dd-a067-c216ec2e5247
Revision: 1
Node Kind: directory
Schedule: normal
Last Changed Author: jerry
Last Changed Rev: 0
Last Changed Date: 2013-08-24 18:15:52 +0530 (Sat, 24 Aug 2013)

[tom@CentOS trunk]$

Jerry extrait la dernière version du référentiel et commence à travailler sur un projet. Il crée le fichier array.c dans le répertoire trunk.

[jerry@CentOS ~]$ cd project_repo/trunk/ [jerry@CentOS trunk]$ cat array.c

La commande ci-dessus produira le résultat suivant.

#include <stdio.h>
#define MAX 16

int main(void) {
   int i, n, arr[MAX];
   printf("Enter the total number of elements: ");
   scanf("%d", &n);

   printf("Enter the elements\n");

   for (i = 0; i < n; ++i) scanf("%d", &arr[i]);
   printf("Array has following elements\n");
   for (i = 0; i < n; ++i) printf("|%d| ", arr[i]);
   
   printf("\n");
   return 0;
}

Il veut tester son code avant de s'engager.

[jerry@CentOS trunk]$ make array cc array.c -o array [jerry@CentOS trunk]$ ./array 
Enter the total number of elements: 5
Enter the elements
1
2
3
4
5
Array has following elements
|1| |2| |3| |4| |5|

Il a compilé et testé son code et tout fonctionne comme prévu, il est maintenant temps de valider les changements.

[jerry@CentOS trunk]$ svn status
?       array.c
?       array

Subversion affiche '?' devant les noms de fichiers car il ne sait pas quoi faire avec ces fichiers.

Avant de valider, Jerry doit ajouter ce fichier à la liste de modifications en attente.

[jerry@CentOS trunk]$ svn add array.c 
A         array.c

Vérifions-le avec l'opération 'status'. Spectacles de SubversionAavant array.c , cela signifie que le fichier est ajouté avec succès à la liste de modifications en attente.

[jerry@CentOS trunk]$ svn status
?       array
A       array.c

Pour stocker le fichier array.c dans le référentiel, utilisez la commande commit avec l'option -m suivie du message commit. Si vous omettez l'option -m, Subversion affichera l'éditeur de texte où vous pourrez taper un message sur plusieurs lignes.

[jerry@CentOS trunk]$ svn commit -m "Initial commit"
Adding         trunk/array.c
Transmitting file data .
Committed revision 2.

Le fichier array.c est maintenant correctement ajouté au référentiel et le numéro de révision est incrémenté de un.

Jerry a déjà ajouté le fichier array.c au référentiel. Tom vérifie également le dernier code et commence à travailler.

[tom@CentOS ~]$ svn co http://svn.server.com/svn/project_repo --username=tom

La commande ci-dessus produira le résultat suivant.

A    project_repo/trunk
A    project_repo/trunk/array.c
A    project_repo/branches
A    project_repo/tags
Checked out revision 2.

Mais, il a constaté que quelqu'un avait déjà ajouté le code. Il est donc curieux de savoir qui a fait cela et il vérifie le message du journal pour voir plus de détails à l'aide de la commande suivante:

[tom@CentOS trunk]$ svn log

La commande ci-dessus produira le résultat suivant.

------------------------------------------------------------------------
r2 | jerry | 2013-08-17 20:40:43 +0530 (Sat, 17 Aug 2013) | 1 line

Initial commit
------------------------------------------------------------------------
r1 | jerry | 2013-08-04 23:43:08 +0530 (Sun, 04 Aug 2013) | 1 line

Create trunk, branches, tags directory structure
------------------------------------------------------------------------

Lorsque Tom observe le code de Jerry , il remarque immédiatement un bug là-dedans. Jerry ne recherchait pas de débordement de tableau, ce qui pouvait causer de sérieux problèmes. Alors Tom décide de résoudre ce problème. Après modification, array.c ressemblera à ceci.

#include <stdio.h>

#define MAX 16

int main(void)
{
   int i, n, arr[MAX];

   printf("Enter the total number of elements: ");
   scanf("%d", &n);

   /* handle array overflow condition */
   if (n > MAX) {
      fprintf(stderr, "Number of elements must be less than %d\n", MAX);
      return 1;
   }

   printf("Enter the elements\n");

   for (i = 0; i < n; ++i)
      scanf("%d", &arr[i]);

   printf("Array has following elements\n");
   for (i = 0; i < n; ++i)
      printf("|%d| ", arr[i]);
      printf("\n");

   return 0;
}

Tom veut utiliser l'opération d'état pour voir la liste des modifications en attente.

[tom@CentOS trunk]$ svn status
M       array.c

array.c est modifié, c'est pourquoi Subversion afficheMlettre avant le nom du fichier. Ensuite, Tom compile et teste son code et cela fonctionne très bien. Avant de valider les modifications, il souhaite les revérifier en examinant les modifications qu'il a apportées.

[tom@CentOS trunk]$ svn diff
Index: array.c
===================================================================
--- array.c   (revision 2)
+++ array.c   (working copy)
@@ -9,6 +9,11 @@
    printf("Enter the total number of elements: ");
    scanf("%d", &n);
 
+   if (n > MAX) {
+      fprintf(stderr, "Number of elements must be less than %d\n", MAX);
+      return 1;
+   }
+
    printf("Enter the elements\n");
 
    for (i = 0; i < n; ++i)

Tom a ajouté quelques lignes dans le fichier array.c , c'est pourquoi Subversion affiche + signe avant de nouvelles lignes. Il est maintenant prêt à valider ses changements.

[tom@CentOS trunk]$ svn commit -m "Fix array overflow problem"

La commande ci-dessus produira le résultat suivant.

Sending        trunk/array.c
Transmitting file data .
Committed revision 3.

Les modifications de Tom sont correctement validées dans le référentiel.

Jerry avait commis la première version du code. Mais il pense qu'il devrait écrire deux fonctions pour accepter l'entrée et afficher le contenu du tableau. Après modification, array.c se présente comme suit.

#include <stdio.h>
#define MAX 16

void accept_input(int *arr, int n) {
   int i;
   for (i = 0; i < n; ++i) 
   scanf("%d", &arr[i]);
}

void display(int *arr, int n) {
   int i;
   for (i = 0; i < n; ++i) 
   printf("|%d| ", arr[i]);
   
   printf("\n");
}

int main(void) {
   int i, n, arr[MAX];

   printf("Enter the total number of elements: ");
   scanf("%d", &n);

   printf("Enter the elements\n");
   accept_input(arr, n);

   printf("Array has following elements\n");
   display(arr, n);

   return 0;
}

Jerry compile et teste son code et est prêt à valider les modifications. Avant cela, il souhaite revoir les modifications à l'aide de la commande suivante.

[jerry@CentOS trunk]$ svn diff

La commande ci-dessus produira le résultat suivant.

Index: array.c
===================================================================
--- array.c   (revision 2)
+++ array.c   (working copy)
@@ -2,6 +2,24 @@
 
 #define MAX 16
 
+void accept_input(int *arr, int n)
+{
+   int i;
+
+   for (i = 0; i & n; ++i)
+      scanf("%d", &arr[i]);
+}
+
+void display(int *arr, int n)
+{
+   int i;
+
+   for (i = 0; i < n; ++i)
+      printf("|%d| ", arr[i]);
+   
+   printf("\n");
+}
+
 int main(void)
 {
    int i, n, arr[MAX];
@@ -10,15 +28,10 @@
    scanf("%d", &n);
 
    printf("Enter the elements\n");
+   accept_input(arr, n);
 
-   for (i = 0; i < n; ++i)
-      scanf("%d", &arr[i]);
-
    printf("Array has following elements\n");
-   for (i = 0; i < n; ++i)
-      printf("|%d| ", arr[i]);
-   
-   printf("\n");
+   display(arr, n);
 
    return 0;
 }

Pour les nouvelles lignes ajoutées, Subversion montre + signe avant la ligne et pour la ligne supprimée, il montre -signe. Maintenant, Jerry essaie de valider les modifications à l'aide de la commande suivante:

[jerry@CentOS trunk]$ svn commit -m "Add function to accept input and to display array contents"

La commande ci-dessus produira le résultat suivant.

Sending        trunk/array.c
svn: Commit failed (details follow):
svn: File or directory 'array.c' is out of date; try updating
svn: resource out of date; try updating

Subversion ne permet pas de valider les modifications de Jerry , car Tom a déjà modifié le référentiel et la copie de travail de Jerry est obsolète. Pour éviter d'écraser les modifications de l'autre, Subversion échoue cette opération. Jerry doit mettre à jour la copie de travail avant de valider ses modifications. Il utilise donc la commande de mise à jour comme indiqué ci-dessous.

[jerry@CentOS trunk]$ svn update
G    array.c
Updated to revision 3.

Subversion montre la lettre G avant le nom de fichier, ce qui signifie que ce fichier a été fusionné.

[jerry@CentOS trunk]$ svn diff

La commande ci-dessus produira le résultat suivant.

Index: array.c
===================================================================
--- array.c   (revision 3)
+++ array.c   (working copy)
@@ -2,6 +2,24 @@
 
 #define MAX 16
 
+void accept_input(int *arr, int n)
+{
+   int i;
+
+   for (i = 0; i < n; ++i)
+      scanf("%d", &arr[i]);
+}
+
+void display(int *arr, int n)
+{
+   int i;
+
+   for (i = 0; i < n; ++i)
+      printf("|%d| ", arr[i]);
+   
+   printf("\n");
+}

+
 int main(void)
 {
    int i, n, arr[MAX];
@@ -15,15 +33,10 @@
    }
 
    printf("Enter the elements\n");
+   accept_input(arr, n);
 
-   for (i = 0; i < n; ++i)
-      scanf("%d", &arr[i]);
-
    printf("Array has following elements\n");
-   for (i = 0; i < n; ++i)
-      printf("|%d| ", arr[i]);
-   
-   printf("\n");
+   display(arr, n);
 
    return 0;
 }

Subversion n'affiche que les modifications de Jerry , mais le fichier array.c est fusionné. Si vous observez attentivement, Subversion affiche maintenant le numéro de révision 3. Dans la sortie précédente, il montrait le numéro de révision 2. Vérifiez simplement qui a fait des changements dans le fichier et dans quel but.

jerry@CentOS trunk]$ svn log
------------------------------------------------------------------------
r3 | tom   | 2013-08-18 20:21:50 +0530 (Sun, 18 Aug 2013)   | 1 line

Fix array overflow problem
------------------------------------------------------------------------
r2 | jerry | 2013-08-17 20:40:43 +0530 (Sat, 17 Aug 2013) | 1 line

Initial commit
------------------------------------------------------------------------
r1 | jerry | 2013-08-04 23:43:08 +0530 (Sun, 04 Aug 2013) | 1 line

Create trunk, branches, tags directory structure
------------------------------------------------------------------------

La copie de travail de Jerry est maintenant synchronisée avec le référentiel et il peut valider ses modifications en toute sécurité.

[jerry@CentOS trunk]$ svn commit -m "Add function to accept input and to display array contents"
Sending        trunk/array.c
Transmitting file data .
Committed revision 4.

Supposons que Jerry modifie accidentellement le fichier array.c et qu'il obtienne des erreurs de compilation. Maintenant, il veut abandonner les changements. Dans cette situation, l'opération «retour» aidera. L'opération de restauration annulera toutes les modifications locales d'un fichier ou d'un répertoire et résoudra tout état de conflit.

[jerry@CentOS trunk]$ svn status

La commande ci-dessus produira le résultat suivant.

M       array.c

Essayons de créer un tableau comme suit:

[jerry@CentOS trunk]$ make array

La commande ci-dessus produira le résultat suivant.

cc     array.c   -o array
array.c: In function ‘main’:
array.c:26: error: ‘n’ undeclared (first use in this function)
array.c:26: error: (Each undeclared identifier is reported only once
array.c:26: error: for each function it appears in.)
array.c:34: error: ‘arr’ undeclared (first use in this function)
make: *** [array] Error 1

Jerry effectue une opération de «retour» sur le fichier array.c .

[jerry@CentOS trunk]$ svn revert array.c 
Reverted 'array.c'

[jerry@CentOS trunk]$ svn status [jerry@CentOS trunk]$

Maintenant compilez le code.

[jerry@CentOS trunk]$ make array
cc     array.c   -o array

Après l'opération de restauration, sa copie de travail revient à son état d'origine. L'opération de restauration peut restaurer un seul fichier ainsi qu'un répertoire complet. Pour restaurer un répertoire, utilisez l'option -R comme indiqué ci-dessous.

[jerry@CentOS project_repo]$ pwd
/home/jerry/project_repo

[jerry@CentOS project_repo]$ svn revert -R trunk

Jusqu'à présent, nous avons vu comment annuler les modifications apportées à la copie de travail. Mais que faire si vous souhaitez annuler une révision validée! L'outil du système de contrôle de version ne permet pas de supprimer l'historique du référentiel. Nous ne pouvons ajouter que l'histoire. Cela se produira même si vous supprimez des fichiers du référentiel. Pour annuler une ancienne révision, nous devons annuler les modifications apportées à l'ancienne révision, puis valider une nouvelle révision. C'est ce qu'on appelle une fusion inversée.

Supposons que Jerry ajoute un code pour l'opération de recherche linéaire. Après vérification, il valide ses modifications.

[jerry@CentOS trunk]$ svn diff
Index: array.c
===================================================================
--- array.c   (revision 21)
+++ array.c   (working copy)
@@ -2,6 +2,16 @@
 
 #define MAX 16
 
+int linear_search(int *arr, int n, int key)
+{
+   int i;
+
+   for (i = 0; i < n; ++i)
+      if (arr[i] == key)
+         return i;
+   return -1;
+}
+
 void bubble_sort(int *arr, int n)
 {
    int i, j, temp, flag = 1;

[jerry@CentOS trunk]$ svn status ? array M array.c [jerry@CentOS trunk]$ svn commit -m "Added code for linear search"
Sending        trunk/array.c
Transmitting file data .
Committed revision 22.

Jerry est curieux de savoir ce que fait Tom. Il vérifie donc les messages du journal Subversion.

[jerry@CentOS trunk]$ svn log

La commande ci-dessus produira le résultat suivant.

------------------------------------------------------------------------
r5 | tom   | 2013-08-24 17:15:28 +0530 (Sat, 24 Aug 2013) | 1 line

Add binary search operation
------------------------------------------------------------------------
r4 | jerry | 2013-08-18 20:43:25 +0530 (Sun, 18 Aug 2013) | 1 line

Add function to accept input and to display array contents

Après avoir consulté les messages du journal, Jerry se rend compte qu'il a commis une grave erreur. Parce que Tom a déjà implémenté une opération de recherche binaire, qui est meilleure que la recherche linéaire; son code est redondant, et maintenant Jerry doit revenir ses modifications à la révision précédente. Alors, recherchez d'abord la révision actuelle du référentiel. Actuellement, le référentiel est à la révision 22 et nous devons le retourner à la révision précédente, à savoir la révision 21.

[jerry@CentOS trunk]$ svn up 
At revision 22.

[jerry@CentOS trunk]$ svn merge -r 22:21 array.c --- Reverse-merging r22 into 'array.c': U array.c [jerry@CentOS trunk]$ svn commit -m "Reverted to revision 21"
Sending        trunk/array.c
Transmitting file data .
Committed revision 23.

Tom décide d'ajouter un fichier README pour leur projet. Il crée donc le fichier README et y ajoute la liste TODO. Après avoir ajouté cela, le référentiel de fichiers est à la révision 6.

[tom@CentOS trunk]$ cat README /* TODO: Add contents in README file */ [tom@CentOS trunk]$ svn status
?       README

[tom@CentOS trunk]$ svn add README A README [tom@CentOS trunk]$ svn commit -m "Added README file. Will update it's content in future."
Adding         trunk/README
Transmitting file data .
Committed revision 6.

Jerry vérifie le dernier code qui est à la révision 6. Et immédiatement il commence à travailler. Après quelques heures, Tom met à jour le fichier README et valide ses modifications. Le README modifié ressemblera à ceci.

[tom@CentOS trunk]$ cat README * Supported operations: 1) Accept input 2) Display array elements [tom@CentOS trunk]$ svn status
M       README

[tom@CentOS trunk]$ svn commit -m "Added supported operation in README"
Sending        trunk/README
Transmitting file data .
Committed revision 7.

Maintenant, le référentiel est à la révision 7 et la copie de travail de Jerry est obsolète. Jerry met également à jour le fichier README et essaie de valider ses modifications.

Le fichier README de Jerry ressemble à ceci.

[jerry@CentOS trunk]$ cat README 
* File list

1) array.c	Implementation of array operation.
2) README	Instructions for user.

[jerry@CentOS trunk]$ svn status M README [jerry@CentOS trunk]$ svn commit -m "Updated README"
Sending        trunk/README
svn: Commit failed (details follow):
svn: File or directory 'README' is out of date; try updating
svn: resource out of date; try updating

Étape 1: Afficher les conflits

Subversion a détecté que le fichier README a changé depuis la dernière mise à jour. Donc, Jerry doit mettre à jour sa copie de travail.

[jerry@CentOS trunk]$ svn up
Conflict discovered in 'README'.
Select: (p) postpone, (df) diff-full, (e) edit,
        (mc) mine-conflict, (tc) theirs-conflict,
        (s) show all options:

Subversion se plaint qu'il y a un conflit avec le fichier README, et Subversion ne sait pas comment résoudre ce problème. Alors Jerry choisit la df possibilité d'examiner le conflit.

[jerry@CentOS trunk]$ svn up
Conflict discovered in 'README'.
Select: (p) postpone, (df) diff-full, (e) edit,
        (mc) mine-conflict, (tc) theirs-conflict,
        (s) show all options: df
--- .svn/text-base/README.svn-base	Sat Aug 24 18:07:13 2013
+++ .svn/tmp/README.tmp	Sat Aug 24 18:13:03 2013
@@ -1 +1,11 @@
-/* TODO: Add contents in README file */
+<<<<<<< .mine
+* File list
+
+1) array.c	Implementation of array operation.
+2) README	Instructions for user.
+=======
+* Supported operations:
+
+1) Accept input
+2) Display array elements
+>>>>>>> .r7
Select: (p) postpone, (df) diff-full, (e) edit, (r) resolved,
        (mc) mine-conflict, (tc) theirs-conflict,
        (s) show all options:

Étape 2: reporter les conflits

Ensuite, Jerry choisit les options de report (p) , afin de pouvoir résoudre le conflit.

Select: (p) postpone, (df) diff-full, (e) edit, (r) resolved,
        (mc) mine-conflict, (tc) theirs-conflict,
        (s) show all options: p
C    README
Updated to revision 7.
Summary of conflicts:
  Text conflicts: 1

Après avoir ouvert le README dans l'éditeur de texte, il se rend compte que Subversion a inclus à la fois le code de Tom et son code avec des marqueurs de conflit.

[jerry@CentOS trunk]$ cat README
<<<<<<< .min
* File list

1) array.c	Implementation of array operation.
2) README	Instructions for user.
=======
* Supported operations:

1) Accept input
2) Display array elements
>>>>>>> .r7

Jerry veut les changements de Tom aussi bien que les siens, alors il supprime simplement les lignes contenant les marqueurs de conflit.

Ainsi, le README modifié ressemblera à ceci.

[jerry@CentOS trunk]$ cat README
* File list

1) array.c	Implementation of array operation.
2) README	Instructions for user.

* Supported operations:

1) Accept input
2) Display array elements

Jerry a résolu le conflit et il réessaye de commettre.

[jerry@CentOS trunk]$ svn commit -m "Updated README" svn: Commit failed (details follow): svn: Aborting commit: '/home/jerry/project_repo/trunk/README' remains in conflict [jerry@CentOS trunk]$ svn status
?       README.r6
?       README.r7
?       README.mine
C       README

Étape 3: résoudre les conflits

Dans le commit ci-dessus, la lettre Cindique qu'il y a un conflit dans le fichier README. Jerry a résolu le conflit mais n'a pas dit à Subversion qu'il avait résolu le conflit. Il utilise la commande de résolution pour informer Subversion de la résolution du conflit.

[jerry@CentOS trunk]$ svn resolve --accept=working README Resolved conflicted state of 'README' [jerry@CentOS trunk]$ svn status
M       README

[jerry@CentOS trunk]$ svn commit -m "Updated README"
Sending        trunk/README
Transmitting file data .
Committed revision 8.

Le système de contrôle de version prend en charge l' opération de balise en utilisant ce concept selon lequel on peut donner un nom significatif à une version spécifique du code. La balise permet de donner des noms descriptifs et mémorables à une version spécifique du code. Par exempleBASIC_ARRAY_OPERATIONS est plus mémorable que revision 4.

Voyons le fonctionnement des balises avec un exemple. Tom décide de créer une balise afin de pouvoir accéder plus facilement au code.

[tom@CentOS project_repo]$ svn copy --revision=4 trunk/ tags/basic_array_operations

La commande ci-dessus produira le résultat suivant.

A    tags/basic_array_operations/array.c
Updated to revision 4.
A         tags/basic_array_operations

Une fois terminé, le nouveau répertoire sera créé dans le répertoire des balises .

[tom@CentOS project_repo]$ ls -l tags/
total 4
drwxrwxr-x. 3 tom tom 4096 Aug 24 18:18 basic_array_operations

Tom veut le vérifier avant de s'engager. L'opération d'état montre que l'opération d'étiquette est réussie, afin qu'il puisse valider ses modifications en toute sécurité.

[tom@CentOS project_repo]$ svn status
A  +    tags/basic_array_operations

[tom@CentOS project_repo]$ svn commit -m "Created tag for basic array operations"
Adding         tags/basic_array_operations

Committed revision 5.

L'opération de branche crée une autre ligne de développement. C'est utile lorsque quelqu'un souhaite que le processus de développement se déroule dans deux directions différentes. Supposons que vous ayez publié un produit de la version 1.0, vous voudrez peut-être créer une nouvelle branche afin que le développement de 2.0 puisse être séparé des corrections de bogues 1.0.

Dans cette section, nous verrons comment créer, traverser et fusionner une branche. Jerry n'est pas content à cause du conflit, alors il décide de créer une nouvelle branche privée.

[jerry@CentOS project_repo]$ ls
branches  tags  trunk

[jerry@CentOS project_repo]$ svn copy trunk branches/jerry_branch A branches/jerry_branch [jerry@CentOS project_repo]$ svn status
A  +    branches/jerry_branch

[jerry@CentOS project_repo]$ svn commit -m "Jerry's private branch" Adding branches/jerry_branch Adding branches/jerry_branch/README Committed revision 9. [jerry@CentOS project_repo]$

Maintenant, Jerry travaille dans sa branche privée. Il ajoute une opération de tri pour le tableau. Le code modifié de Jerry ressemble à ceci.

[jerry@CentOS project_repo]$ cd branches/jerry_branch/ [jerry@CentOS jerry_branch]$ cat array.c

La commande ci-dessus produira le résultat suivant.

#include <stdio.h>
#define MAX 16

void bubble_sort(int *arr, int n)
{
   int i, j, temp, flag = 1;
   for (i = 1; i < n && flag == 1; ++i) {
      flag = 0;
      for (j = 0; j < n - i; ++j) {
         if (arr[j] > arr[j + 1]) {
            flag = 1;
            temp = arr[j];
            arr[j] = arr[j + 1];
            arr[j + 1] = temp;
         }
      }
   }
}

void accept_input(int *arr, int n)
{
   int i;

   for (i = 0; i < n; ++i) 
   scanf("%d", &arr[i]);
}

void display(int *arr, int n)
{
   int i;

   for (i = 0; i < n; ++i)
   printf("|%d| ", arr[i]);

   printf("\n");
}

int main(void)
{
   int i, n, key, ret, arr[MAX];

   printf("Enter the total number of elements: ");
   scanf("%d", &n);

   /* Error handling for array overflow */
   if (n >MAX) {
      fprintf(stderr, "Number of elements must be less than %d\n", MAX);
      return 1;
   }

   printf("Enter the elements\n");
   accept_input(arr, n);

   printf("Array has following elements\n");
   display(arr, n);

   printf("Sorted data is\n");
   bubble_sort(arr, n);
   display(arr, n);

   return 0;
}

Jerry compile et teste son code et est prêt à valider ses modifications.

[jerry@CentOS jerry_branch]$ make array cc array.c -o array [jerry@CentOS jerry_branch]$ ./array

La commande ci-dessus produira le résultat suivant.

Enter the total number of elements: 5
Enter the elements
10
-4
2
7 
9
Array has following elements
|10| |-4| |2| |7| |9| 
Sorted data is
|-4| |2| |7| |9| |10| 

[jerry@CentOS jerry_branch]$ svn status ? array M array.c [jerry@CentOS jerry_branch]$ svn commit -m "Added sort operation"
Sending        jerry_branch/array.c
Transmitting file data .
Committed revision 10.

Pendant ce temps, dans le coffre, Tom décide de mettre en œuvre une opération de recherche. Tom ajoute du code pour l'opération de recherche et son code ressemble à ceci.

[tom@CentOS trunk]$ svn diff

La commande ci-dessus produira le résultat suivant.

Index: array.c
===================================================================
--- array.c   (revision 10)
+++ array.c   (working copy)
@@ -2,6 +2,27 @@
 
 #define MAX 16
 
+int bin_search(int *arr, int n, int key)
+{
+   int low, high, mid;
+
+   low   = 0;
+   high   = n - 1;
+   mid   = low + (high - low) / 2;
+
+   while (low <= high) {
+      if (arr[mid] == key)
+         return mid;
+      if (arr[mid] > key)
+         high = mid - 1;
+      else
+         low = mid + 1;
+      mid = low + (high - low) / 2;
+   }
+
+   return -1;
+}
+
 void accept_input(int *arr, int n)
 {
    int i;
@@ -22,7 +43,7 @@
 
 int main(void)
 {
-   int i, n, arr[MAX];
+   int i, n, ret, key, arr[MAX];
 
    printf("Enter the total number of elements: ");
    scanf("%d", &n);
@@ -39,5 +60,16 @@
    printf("Array has following elements\n");
    display(arr, n);
 
+   printf("Enter the element to be searched: ");
+   scanf("%d", &key);
+
+   ret = bin_search(arr, n, key);
+   if (ret < 0) {
+      fprintf(stderr, "%d element not present in array\n", key);
+      return 1;
+   }
+
+   printf("%d element found at location %d\n", key, ret + 1);
+
    return 0;
 }

Après examen, il valide ses modifications.

[tom@CentOS trunk]$ svn status
?       array
M       array.c

[tom@CentOS trunk]$ svn commit -m "Added search operation"
Sending        trunk/array.c
Transmitting file data .
Committed revision 11.

Mais Tom est curieux de savoir ce que Jerry fait dans sa branche privée.

[tom@CentOS trunk]$ cd ../branches/
[tom@CentOS branches]$ svn up A jerry_branch A jerry_branch/array.c A jerry_branch/README [tom@CentOS branches]$ svn log
------------------------------------------------------------------------
r9 | jerry | 2013-08-27 21:56:51 +0530 (Tue, 27 Aug 2013) | 1 line

Added sort operation
------------------------------------------------------------------------

En visualisant le message du journal de Subversion, Tom a découvert que Jerry avait implémenté l'opération de «tri». Tom a implémenté une opération de recherche à l'aide d'un algorithme de recherche binaire, il attend toujours les données dans l'ordre trié. Mais que se passe-t-il si l'utilisateur fournit des données dans un ordre non trié? Dans cette situation, l'opération de recherche binaire échouera. Il décide donc de prendre le code de Jerry pour trier les données avant l'opération de recherche. Il demande donc à Subversion de fusionner le code de la branche de Jerry dans le tronc.

[tom@CentOS trunk]$ pwd /home/tom/project_repo/trunk [tom@CentOS trunk]$ svn merge ../branches/jerry_branch/
--- Merging r9 through r11 into '.':
U    array.c

Après la fusion, array.c ressemblera à ceci.

[tom@CentOS trunk]$ cat array.c

La commande ci-dessus produira le résultat suivant.

#include <stdio.h>
#define MAX 16

void bubble_sort(int *arr, int n)
{
   int i, j, temp, flag = 1;

   for (i = 1; i < n && flag == 1; ++i) {
      flag = 0;
      for (j = 0; j < n - i; ++j) {
         if (arr[j] > arr[j + 1]) {
            flag      	= 1;
            temp      	= arr[j];
            arr[j]      = arr[j + 1];
            arr[j + 1]	= temp;
         }
      }
   }
}

int bin_search(int *arr, int n, int key)
{
   int low, high, mid;

   low   = 0;
   high  = n - 1;
   mid   = low + (high - low) / 2;

   while (low <= high) {
      if (arr[mid] == key)
         return mid;
      if (arr[mid] > key)
         high = mid - 1;
      else
         low = mid + 1;
      mid = low + (high - low) / 2;
   }
   return -1;
}

void accept_input(int *arr, int n)
{
   int i;

   for (i = 0; i < n; ++i)
      scanf("%d", &arr[i]);
}

void display(int *arr, int n)
{
   int i;
   for (i = 0; i < n; ++i)
      printf("|%d| ", arr[i]);
   printf("\n");
}

int main(void)
{
   int i, n, ret, key, arr[MAX];

   printf("Enter the total number of elements: ");
   scanf("%d", &n);

   /* Error handling for array overflow */
   if (n > MAX) {
      fprintf(stderr, "Number of elements must be less than %d\n", MAX);
      return 1;
   }

   printf("Enter the elements\n");
   accept_input(arr, n);

   printf("Array has following elements\n");
   display(arr, n);

   printf("Sorted data is\n");
   bubble_sort(arr, n);
   display(arr, n);

   printf("Enter the element to be searched: ");
   scanf("%d", &key);

   ret = bin_search(arr, n, key);
   if (ret < 0) {
      fprintf(stderr, "%d element not present in array\n", key);
      return 1;
   }

   printf("%d element found at location %d\n", key, ret + 1);

   return 0;
}

Après compilation et test, Tom valide ses modifications dans le référentiel.

[tom@CentOS trunk]$ make array
cc     array.c   -o array

[tom@CentOS trunk]$ ./array Enter the total number of elements: 5 Enter the elements 10 -2 8 15 3 Array has following elements |10| |-2| |8| |15| |3| Sorted data is |-2| |3| |8| |10| |15| Enter the element to be searched: -2 -2 element found at location 1 [tom@CentOS trunk]$ svn commit -m "Merge changes from Jerry's code"
Sending        trunk
Sending        trunk/array.c
Transmitting file data .
Committed revision 12.

[tom@CentOS trunk]$