Arhivator tar

  • Introducere

Mihai și Ilinca s-au distrat foarte mult cu ocazia sărbătorilor. Pentru că au fost copii cuminți, au primit o vizită de la Moș Crăciun cu care au făcut foooarte multe poze.

Moșul i-a rugat pe cei doi să îi trimită o **arhivă tar** cu pozele făcute, deoarece clientul său de mail nu suportă decât un singur fișier ca atașament.

Cei doi sunt foarte încântanți să îi trimită mail-ul, însă nu au găsit un utilitar care să îi ajute să realizeze arhiva. Tot ce au găsit este ( https://www.mkssoftware.com/docs/man4/tar.4.asp ) o specificatie a formatului. Așa că au nevoie de ajutorul vostru pentru a îi face pe plac Moșului. 🙂

  • Funcționarea programului

Implementarea voastră trebuie să genereze un executabil numit my_tar. Acesta primește comenzi de la **standard input** și se folosește de 2 fișiere adiționale, al căror nume este predefinit, și care trebuie să se afle în același director cu executabilul:

1) usermap.txt

2) file_ls

Programul vostru trebuie să fie capabil de a verifica existența celor două fișiere. Acesta poate presupune că informația găsită în interior, este una corectă.

  • usermap.txt

Conține output-ul comenzii.

cat /etc/passwd

Exemplu:

[...]
sshd:x:114:65534::/var/run/sshd:/usr/sbin/nologin
[...]
admin:x:1003:118::/home/admin:/bin/sh
[...]
mysql:x:116:127:MySQL Server,,,:/nonexistent:/bin/false
[...]
ntp:x:121:135::/home/ntp:/bin/false
[...]
lvasilescu:x:1008:1008:Laura Vasilescu,,,:/home/lvasilescu:/bin/bash

*) Am folosit ”[…]” pentru a simboliza faptul că mai existau și alte linii; Fișierul ”/etc/passwd” este, de regulă, destul de lung, dar nu avea sens postarea lui integrală aici.

  • file_ls

Se obține din output-ul generat de comanda

ls -la --time-style=full-iso

în directorul în care se află fișierele pe care dorim să le arhivăm.

Se păstrează doar linile care fac referire la fișierele pe care dorim să le includem în arhivă. Exemplu:

-rw-rw-r-- 1 lvasilescu lvasilescu 8 2013-12-16 17:55:15.882089310 +0200 a

-rw-rw-r-- 1 lvasilescu lvasilescu 214200 2013-12-16 17:55:10.190055687 +0200 test
  •  Comenzi de intrare

Programul acceptă comenzi de la **standard input**.

Fiecare comandă reprezintă o linie. Puteți presupune că o linie are maximum

    \[511\]

caractere.

Mai jos sunt descrise comenzile pe care trebuie să le recunoască programul vostru:

  1. load archivename
    • Această comandă creează arhiva, folosindu-se de informații extrase din usermap.txt și file_ls.
    • ”archivename” reprezintă numele arhivei ce va fi creată.
    • În urma acestei comenzi trebuie să se obțină o arhivă cu numele cerut, care să conțină toate fişierele (fără directoare) enumerate în file_ls.
  2. list archivename
    • Comanda va lista în consolă numele fișierelor din arhivă. Câte unul pe linie.
  3. get archivename filename
    • Comanda va extrage din arhivă conținutul fișierului ”filename” și îl va afișa în consolă.
    • În urma acestei comenzi, conținutul arhivei rămâne identic.
  4. quit
    • Închide programul.
  • Formatul arhivei tar

Mai multe detalii găsiți aici http://www.mksxserver.com/docs/man4/tar.4.asp.

Practic, structura header-ului ar trebui să fie ceva de genul:

union record {
        char charptr[512];
        struct header {
                char name[100];
                char mode[8];
                char uid[8];
                char gid[8];
                char size[12];
                char mtime[12];
                char chksum[8];
                char typeflag;
                char linkname[100];
                char magic[8];
                char uname[32];
                char gname[32];
                char devmajor[8];
                char devminor[8];
        } header;
};

Câmpurile **name**, **mode**, **uid**, **gid**, **size**, **mtime**, **uname**, **gname** vor fi determinate din fișierele usermap.txt și/sau file_ls.

**chksum** se compltează făcând suma între toți octeții ce alcătuiesc headerul.

**typeflag** va fi 0.

**linkname** reprezintă numele fișierului.

**magic** o să fie string-ul “GNUtar ” (atenție, e un spațiu la final!)

**devmajor** și **devminor** o să fie 0.

Toate valorile trebuie să fie în **octal**. Vedeți în documentație.

Autor:  Laura Vasilescu

Temă 2013