Directed Acyclic Graph : graphe orienté acyclique.
git est un système de contrôle de version distribué (DVCS) :
Utilité :
D'autres références :
Un dépôt est une collection de fichiers se trouvant dans un répertoire et dont les modifications sont suivies par git
$ mkdir my-project
$ cd my-project
$ git init
Initialized empty Git repository in …/my-project/.git/
$ ls -a
. .. .git/
Tout les fichier que l'on veut suivre doivent être dans ce répertoire, ou un sous répertoire
La commande git status permet de connaître l'état du dépôt
Un commit est un instané de tous les fichiers du dépôt et de leur contenu à un instant donné, plus des méta-données (date, nom de la personne qui a crée le commit, …). Les commits sont crées en deux temps :
$ emacs README.md #ou n'importe quel autre éditeur de texte
$ git add README.md
$ git status
…
Changes to be committed:
(use "git rm --cached <file>..." to unstage)
new file: README.md
$ git commit #lance l'éditeur par défaut
[main (root-commit) 71dda77] Ajout du premier fichier.
1 file changed, 1 insertion(+)
create mode 100644 README.md
git ne possède pas de commande spéciale pour ajouter un fichier au dépôt. Ajouter un fichier, ou modifier un fichier existant se font de la même manière
⚠️ git ne peut stocker que des fichiers et leur chemin. En particulier, on peut pas ajouter un répertoire vide dans un dépôt git.
Un commit est identifié de manière unique par son hash (somme de contrôle).
git utilise l'algorithme SHA-1 : les hash font 160 bits (20 octets ou 40 chiffres hexadécimaux).
La commande git show abcde… permet de montrer l'objet git (commit mais aussi d'autres) dont le hash est donné. On n'est pas obligé de donner les 40 caractères, un prefixe suffit.
git garde un historique des modifications faites à un dépôt.
Chaque commit pointe vers son parent et le hash
du parent est pris en compte dans le calcul du hash du commit.
⇒ Impossible de falsifier un « commit » dans l'historique
$ emacs README.md
$ emacs test.ml
$ git add test.ml README.md
$ git commit
[main 30d4a5d] Ajout du fichier principal.
2 files changed, 2 insertions(+), 1 deletion(-)
create mode 100644 test.ml
$ git log #affiche les commits
Un commit git n'est pas un diff d'une
version du fichier à l'autre. C'est l'ensemble des
fichiers tels qu'ils étaient à un moment donné.
git stocke ces « images instantanées » de manière
compacte.
L'ensemble des commit d'un projet git forme un DAG :
$ emacs test.ml
$ git add test.ml
$ git commit
Sauf cas particulier, on est toujours sur une branche, i.e. sur un chemin nommé entre le commit courant et le commit initial
La branche principale, crée par défaut se nomme main
$ git branch
* main
Il est possible d'avoir plusieurs branches (pour faire des
essais, corriger des bugs, …) sans toucher à la branche
principale.
La commande git branch foo crée une
nouvelle branche appelée foo à partir du commit
courant.
$ git branch fibonacci
$ git branch
fibonacci
* main
Chaque branche se souvient de son dernier commit
On peut changer de branche avec git checkout
ma_branche
$ git checkout fibonacci
Switched to branch 'fibonacci'
$ emacs test.ml
$ git add test.ml
$ git commit
$ emacs README.md
$ git add README.md
$ git commit
Deux branches sont complètement indépendantes et peuvent évoluer en parallèle:
$ git checkout main
Switched to branch 'main'
$ emacs README.md
$ git add README.md
$ git commit
Une opération de merge (fusion) consiste à importer les modifications d'une branche dans une autre. Pour cela :
$ git checkout main
Switched to branch 'main'
$ git merge fibonacci
Auto-merging README.md
CONFLICT (content): Merge conflict in README.md
Automatic merge failed; fix conflicts and then commit the result.
git essaye de fusionner les deux branches automatiquement. La plupart du temps il y arrive. Si un même fichier est modifié de deux manière différentes, il y a un conflit.
Un conflit dans un fichier est simplement matérialisé par une section :
<<<<<<< HEAD
du texte
=======
une autre version
>>>>>>> branche_source
La portion du haut est le texte de la branche cible (sur
laquelle on est) et celui du bas celui de la branche
source.
Il faut retirer toutes les sections qui ont cette forme (en
choisissant l'une ou l'autre des versions, ou encore une
troisème), puis commiter.
$ emacs README.md #on règle le conflit
$ git add README.md
$ git commit
[main e01a960] Merge branch 'fibonacci'
Pour récupérer un projet git déjà existant, on utilise la commande git clone :
$ git clone https://gitlab.dsi.universite-paris-saclay.fr/
De telles URL sont appelées « remote »
L'URL source initiale est appelée origin.
Pour synchroniser le projet local avec un remote on utilise git pull et git push.
En réalité, un dépôt git (local) possède :
$ git branch -a
* main
remotes/origin/HEAD -> origin/main
remotes/origin/fibonacci
remotes/origin/main
La commande git pull est en fait un alias pour :
En fonction de l'état des dépôt distants, les commandes git pull ou git push peuvent créer des conflits :
Certains fichiers peuvent être présents mais ne doivent pas être versionnés :
On peut créer dans tous lse sous-répertoire du dépôt (et aussi à la racine) un fichier .gitignore qui contient les fichiers à ignorer. Git ne se plaindra pas de leur existance et interdira de les commiter par erreur.
Il existe de nombreux services autour de git. Ils ne sont pas nécessaire à l'utilisation de git pour un projet, mais offre des avantages :
On fera cepdant garde, les remarques usuelles sur la souveraineté des données s'appliquent.
dune est un outil de l'écosystème OCaml permettant de construire des projets complexes. Parmi ses fonctionalités :
Dans le cadre du projet :
OCaml possède des traits impératifs. Utilisés judicieusement, ces derniers permettent de gagner en performance ou parfois, d'exprimer certains problèmes plus simplements.
Pour le projet, deux types d'effets de bords sont utiles :
D'autres pourront être présentées en cas de besoin
OCaml propose deux types spécifiques :
On présente le type in_channel
Toutes les fonctions de lecture lèvent
l'exception End_of_file en fin de fichier.
D'autres fonctions existent, si besoin voir la documentation
de la bibliothèque standard.
Les tableaux sont semblables aux tableaux de C. Ils sont de taille fixe, et contiennent des valeurs du même type.
let tab = Array.create 100 0 (* tableau de taille 100
contenant 0 *)
let () = tab.(42) <- tab.(42) + 10 (* accès et affectation *)
Le module Array donne plein d'itérateurs sur les tableaux (Array.iter, Array.map, …)
(* Plutôt que d'utiliser une boucle for: *)
let () = Array.iteri
(fun i v -> Printf.printf "Case %d, valeur: %d\n" i v) tab
(* Si tab est un tableau d'entiers *)