parser.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629
  1. /*
  2. * File: parser.c
  3. * Author: Arthur Brandao
  4. *
  5. * Created on 31 octobre 2018
  6. */
  7. #include <stdio.h>
  8. #include <stdlib.h>
  9. #include <unistd.h>
  10. #include <errno.h>
  11. #include <sys/types.h>
  12. #include <sys/stat.h>
  13. #include <fcntl.h>
  14. #include "error.h"
  15. #include "str.h"
  16. #include "wildcard.h"
  17. #include "parser.h"
  18. /* --- Extern --- */
  19. extern int serrno;
  20. /* --- Fonctions privées --- */
  21. /**
  22. * Indique le nombre de commande dans une ligne
  23. * @param char* La ligne à analyser
  24. * @return int Le nombre de commande
  25. */
  26. int nb_commands(char* line){
  27. //Initialisation variable
  28. int compteur = 0;
  29. boolean vide = true;
  30. //Parcours chaine pour chercher séparateur et compter le nombre de commande
  31. while(*line){
  32. //Si on croise un caractère
  33. if(vide && *line != ' ' && *line != '\0'){
  34. vide = false;
  35. }
  36. //Si un ;
  37. if(*line == ';'){
  38. compteur++;
  39. }
  40. //Si | ou ||
  41. else if(*line == '|'){
  42. //Verif que ce n'est pas le dernier carac
  43. if(*(line + 1) == '\0'){
  44. serrno = SEBADEND;
  45. return SHELL_ERR;
  46. }
  47. //Si un || on avance de 1 en plus
  48. else if(*(line + 1) == '|'){
  49. //Si à la fin de la chaine
  50. if(*(line + 2) == '\0'){
  51. serrno = SEBADEND;
  52. return SHELL_ERR;
  53. }
  54. line++;
  55. }
  56. compteur++;
  57. }
  58. //Si un &&
  59. if(*line == '&'){
  60. //Si celui d'apres est bien un &&
  61. if(*(line + 1) == '&'){
  62. line++;
  63. compteur++;
  64. }
  65. //Sinon il doit y avoir un vide pour etre le & du bck ou un > avant
  66. else if(*(line + 1) != '\0' && *(line - 1) != '>'){
  67. serrno = SEBADET;
  68. return SHELL_ERR;
  69. }
  70. }
  71. line++;
  72. }
  73. //Ajoute la dernière commande
  74. if(!vide){
  75. compteur++;
  76. }
  77. return compteur;
  78. }
  79. /**
  80. * Recup la 1er commande d'une chaine de caractère pour initialiser une
  81. * structure Command
  82. * @param Command* Structure commande à initialiser
  83. * @param char* Ligne avec la commande
  84. * @return int Le décallage à effectuer dans ligne
  85. */
  86. int get_command(Command* c, char* line){
  87. //Declaration variable
  88. char* deb = line;
  89. int length = 0, separator = 0, next = SHELL_NONE;
  90. //Parcours chaine pour chercher un séparateur
  91. while(*line){
  92. //Si un ;
  93. if(*line == ';'){
  94. separator = 1;
  95. break;
  96. }
  97. //Si | ou ||
  98. else if(*line == '|'){
  99. //Si 1 ou 2 |
  100. if(*(line + 1) == '|'){
  101. separator = 2;
  102. next = SHELL_ELSE;
  103. } else {
  104. separator = 1;
  105. next = SHELL_PIPE;
  106. }
  107. break;
  108. }
  109. //Si un &&
  110. if(*line == '&'){
  111. //Si celui d'apres est bien un &&
  112. if(*(line + 1) == '&'){
  113. separator = 2;
  114. next = SHELL_IF;
  115. break;
  116. }
  117. }
  118. length++;
  119. line++;
  120. }
  121. //Verif si c'est la dernière commande
  122. if(!*line){
  123. next = SHELL_END;
  124. }
  125. //Allocation memoire et copie chaine
  126. c->cmd = malloc(sizeof(char) * (length + 1));
  127. memset(c->cmd, 0, length + 1);
  128. strncpy(c->cmd, deb, length);
  129. c->next = next;
  130. c->argc = 0;
  131. c->input = STDIN;
  132. c->output = STDOUT;
  133. c->error = STDERR;
  134. c->erase[0] = false;
  135. c->erase[1] = false;
  136. //Retour
  137. return length + separator;
  138. }
  139. /**
  140. * Parametre les fichiers d'entrées et de sorties d'une commande
  141. * @param Command* La structure de la commande
  142. * @param char* Le nom du fichier
  143. * @param int Le type de redirection (constante SHELLR)
  144. * @return int SHELL_OK si réussite, SHELL_ERR sinon
  145. */
  146. int set_io(Command* c, char* filename, int redir){
  147. //Declaration variable
  148. int file;
  149. //Si fichier existe et on supprime
  150. if((redir == SHELLRE_OUT || redir == SHELLRE_ERR || redir == SHELLRE_ALL) && access(filename, F_OK) != ERR){
  151. if(unlink(filename) == ERR){
  152. addperror("Impossible de supprimer le fichier");
  153. serrno = SEOPENF;
  154. return SHELL_ERR;
  155. }
  156. }
  157. //Ouverture du fichier
  158. file = open(filename, O_CREAT | O_RDWR, S_IRWXU);
  159. if(file == ERR){
  160. addperror("Erreur lors de l'ouverture du fichier pour la redirection");
  161. serrno = SEOPENF;
  162. return SHELL_ERR;
  163. }
  164. //On se met à la fin du fichier
  165. if(lseek(file, 0L, SEEK_END) == ERR){
  166. addperror("Impossible de se deplacer dans le fichier");
  167. serrno = SEOPENF;
  168. return SHELL_ERR;
  169. }
  170. //Analyse dans quel descripteur il doit etre mis
  171. switch(redir){
  172. case SHELLR_IN:
  173. //Si un fichier deja ouvert
  174. if(c->input != STDIN){
  175. //On le ferme
  176. if(close(c->input) == -1){
  177. addperror("Erreur lors de la fermeture de l'ancien fichier de redirection");
  178. }
  179. }
  180. //Set nouveau fichier
  181. c->input = file;
  182. break;
  183. case SHELLR_OUT:
  184. case SHELLRE_OUT:
  185. //Si un fichier deja ouvert
  186. if(c->output != STDOUT){
  187. //On le ferme
  188. if(close(c->output) == -1){
  189. addperror("Erreur lors de la fermeture de l'ancien fichier de redirection");
  190. }
  191. }
  192. c->output = file;
  193. if(redir == SHELLRE_OUT){
  194. c->erase[STDOUT - 1] = true;
  195. }
  196. break;
  197. case SHELLR_ERR:
  198. case SHELLRE_ERR:
  199. //Si un fichier deja ouvert
  200. if(c->error != STDERR){
  201. //On le ferme
  202. if(close(c->error) == -1){
  203. addperror("Erreur lors de la fermeture de l'ancien fichier de redirection");
  204. }
  205. }
  206. c->error = file;
  207. if(redir == SHELLRE_ERR){
  208. c->erase[STDERR - 1] = true;
  209. }
  210. break;
  211. case SHELLR_ALL:
  212. case SHELLRE_ALL:
  213. //Si un fichier deja ouvert
  214. if(c->output != STDOUT){
  215. //On le ferme
  216. if(close(c->output) == -1){
  217. addperror("Erreur lors de la fermeture de l'ancien fichier de redirection");
  218. }
  219. }
  220. if(c->error != STDERR){
  221. //On le ferme
  222. if(close(c->error) == -1){
  223. addperror("Erreur lors de la fermeture de l'ancien fichier de redirection");
  224. }
  225. }
  226. c->output = file;
  227. c->error = file;
  228. if(redir == SHELLRE_ALL){
  229. c->erase[STDOUT - 1] = true;
  230. c->erase[STDERR - 1] = true;
  231. }
  232. break;
  233. default :
  234. serrno = SEREDIRTYPE;
  235. return SHELL_ERR;
  236. }
  237. //Si on arrive ici tous est ok
  238. return SHELL_OK;
  239. }
  240. /**
  241. * Parametre les redirection d'une commande
  242. * @param Command* La commande à analyser
  243. * @return int SHELL_OK si réussite, SHELL_ERR sinon
  244. */
  245. int set_redirection(Command* c){
  246. boolean first = true, guillemet = false;
  247. char* deb, * file, * buffer = c->cmd + 1;
  248. int redir = -1, compteur, finCmd = 0;
  249. //Parcours chaine
  250. while(*buffer){
  251. //Repere redirection
  252. while(*buffer){
  253. //Entrée
  254. if(*buffer == '<'){
  255. //Si il n'y a rien apres
  256. if(!(*(buffer + 1))){
  257. serrno = SEBADREDIR;
  258. return SHELL_ERR;
  259. }
  260. buffer++;
  261. redir = SHELLR_IN;
  262. break;
  263. }
  264. //Sortie
  265. else if (*buffer == '>'){
  266. //Si il n'y a rien apres
  267. if(!(*(buffer + 1))){
  268. serrno = SEBADREDIR;
  269. return SHELL_ERR;
  270. } else {
  271. buffer++;
  272. //Si >>
  273. if(*buffer == '>'){
  274. //Si il n'y a rien apres
  275. if(!(*(buffer + 1))){
  276. serrno = SEBADREDIR;
  277. return SHELL_ERR;
  278. } else {
  279. buffer++;
  280. //Si >>&
  281. if(*buffer == '&'){
  282. //Si il n'y a rien apres
  283. if(!(*(buffer + 1))){
  284. serrno = SEBADREDIR;
  285. return SHELL_ERR;
  286. }
  287. buffer++;
  288. redir = SHELLRE_ALL;
  289. }
  290. //Sinon toujours >>
  291. else {
  292. redir = SHELLR_OUT;
  293. }
  294. }
  295. }
  296. // Si >&
  297. else if(*buffer == '&'){
  298. //Si il n'y a rien apres
  299. if(!(*(buffer + 1))){
  300. serrno = SEBADREDIR;
  301. return SHELL_ERR;
  302. }
  303. redir = SHELLRE_ALL;
  304. buffer++;
  305. }
  306. //Sinon >
  307. else {
  308. redir = SHELLRE_OUT;
  309. }
  310. }
  311. break;
  312. }
  313. //Sortie erreur
  314. else if (*buffer == '2' && *(buffer - 1) == ' ' && *(buffer + 1) && *(buffer + 1) == '>'){
  315. buffer++;
  316. //Si il n'y a rien apres
  317. if(!(*(buffer + 1))){
  318. serrno = SEBADREDIR;
  319. return SHELL_ERR;
  320. } else {
  321. buffer++;
  322. //Si 2>>
  323. if(*buffer == '>'){
  324. buffer++;
  325. //Si il n'y a rien apres
  326. if(!(*(buffer + 1))){
  327. serrno = SEBADREDIR;
  328. return SHELL_ERR;
  329. }
  330. redir = SHELLR_ERR;
  331. }
  332. //Sinon 2>
  333. else {
  334. redir = SHELLRE_ERR;
  335. }
  336. }
  337. break;
  338. }
  339. buffer++;
  340. //On incremente pour trouver la fin de la commande
  341. if(first){
  342. finCmd++;
  343. }
  344. }
  345. //Apres ce n'est que des redirection
  346. first = false;
  347. //On passe les espaces
  348. while(*buffer == ' '){
  349. buffer++;
  350. }
  351. //Si on est à la fin de la chaine
  352. if(*buffer == '\0'){
  353. if(redir == -1){
  354. //Aucune redirection
  355. return finCmd + 1;
  356. }
  357. //Sinon on est dans un redirection non terminée
  358. serrno = SEBADCMD;
  359. return SHELL_ERR;
  360. }
  361. //Regarde si le nom du fichier est entre ""
  362. guillemet = *buffer == '"';
  363. //Sauvegarde debut nom fichier
  364. deb = buffer;
  365. compteur = 0;
  366. //Lecture nom du fichier
  367. while(*buffer){
  368. //Incremente
  369. compteur++;
  370. buffer++;
  371. //Test arret
  372. if(guillemet && *buffer == '"'){
  373. break;
  374. }
  375. else if(!guillemet && (*buffer == '<' || *buffer == '>' || (*buffer == ' ' && *(buffer - 1) != '\\'))){
  376. break;
  377. }
  378. }
  379. //Si fin de la commande et guillemet alors erreur
  380. if(!(*buffer) && guillemet){
  381. serrno = SEBADCMD;
  382. return SHELL_ERR;
  383. }
  384. //Retire guillement si besoin
  385. if(guillemet){
  386. deb++;
  387. compteur--;
  388. buffer++;
  389. }
  390. //Allocation file et copie nom fichier
  391. file = malloc(sizeof(char) * compteur);
  392. memset(file, 0, compteur);
  393. strncpy(file, deb, compteur);
  394. //Redirection
  395. if(set_io(c, file, redir) == SHELL_ERR){
  396. free(file);
  397. return SHELL_ERR;
  398. }
  399. free(file);
  400. //Passe les espaces
  401. while(*buffer == ' '){
  402. buffer++;
  403. }
  404. }
  405. //Si on arrive ici tous est ok
  406. return finCmd + 1;
  407. }
  408. /**
  409. * Sépare une commande en chaine de caractere en un tableau contenant chaque
  410. * argument
  411. * @param Command* La structure d'accueil du resultat
  412. * @param char* La commande à découper (sans les redirections)
  413. * @return int SHELL_OK si réussite, SHELL_ERR sinon
  414. */
  415. int split_command(Command* c, char* cmd){
  416. //Declaration variable
  417. char* deb = cmd;
  418. int nb = 0, length = 0, i = 0;
  419. char delim = ' ';
  420. boolean chercheFin = false;
  421. //Compte le nombre d'argument
  422. while(*cmd != '\0' && *cmd != '&'){
  423. //Cherche debut d'un mot
  424. if(*cmd != ' ' && !chercheFin){
  425. chercheFin = true;
  426. //Guillemet
  427. if(*cmd == '"'){
  428. //Verif que ce n'est pas la fin
  429. if(!(*(cmd + 1))){
  430. serrno = SEBADCMD;
  431. return SHELL_ERR;
  432. }
  433. cmd++;
  434. delim = '"';
  435. }
  436. //Mot sans guillemet autour
  437. else {
  438. length = 0;
  439. delim = ' ';
  440. }
  441. }
  442. //Fin d'un mot
  443. else if(*cmd == delim && chercheFin){
  444. //Adapte si il y avait des guillemet
  445. chercheFin = false;
  446. nb++;
  447. }
  448. //Incremente
  449. cmd++;
  450. }
  451. //Si on termine sur un mot sans "
  452. if(chercheFin){
  453. nb++;
  454. }
  455. //Allocation + retour au debut
  456. c->argv = malloc(sizeof(char*) * (nb + 1));
  457. c->argc = nb;
  458. cmd = deb;
  459. //Parcours chaine et decoupe
  460. while(*cmd != '\0' && *cmd != '&'){
  461. //Cherche debut d'un mot
  462. if(*cmd != ' ' && !chercheFin){
  463. chercheFin = true;
  464. //Guillemet
  465. if(*cmd == '"'){
  466. cmd++;
  467. deb = cmd;
  468. length = 0;
  469. delim = '"';
  470. }
  471. //Mot sans guillemet autour
  472. else {
  473. deb = cmd;
  474. length = 0;
  475. delim = ' ';
  476. }
  477. }
  478. //Fin d'un mot
  479. else if(*cmd == delim && chercheFin){
  480. chercheFin = false;
  481. //Recup le mot
  482. c->argv[i] = malloc(sizeof(char) * (length + 1));
  483. memset(c->argv[i], 0, length + 1);
  484. strncpy(c->argv[i++], deb, length);
  485. }
  486. //Incremente
  487. cmd++;
  488. length++;
  489. }
  490. //Recup le dernier mot si besoin
  491. if(chercheFin){
  492. c->argv[i] = malloc(sizeof(char) * (length + 1));
  493. memset(c->argv[i], 0, length + 1);
  494. strncpy(c->argv[i++], deb, length);
  495. }
  496. //Set la dernière case du tableau à null
  497. c->argv[i] = NULL;
  498. return SHELL_OK;
  499. }
  500. /* --- Fonctions publiques --- */
  501. int parse_line(CommandTab* ct, char* line){
  502. //Declaration variable
  503. int compteur, tmp;
  504. //Nettoyage ligne
  505. line = trim(mtrim(line, '\n'));
  506. //Compte le nombre de commande dans la ligne
  507. compteur = nb_commands(line);
  508. if(compteur == SHELL_ERR){
  509. return SHELL_ERR;
  510. }
  511. //Initialisation structure
  512. ct->cmd = malloc(sizeof(Command*) * compteur);
  513. ct->length = compteur;
  514. ct->bck = line[strlen(line) - 1] == '&';
  515. //Si il y a un un & on le retire (on à deja l'information)
  516. if(ct->bck){
  517. line[strlen(line) - 1] = '\0';
  518. }
  519. //Recupération de chaque commande
  520. for(int i = 0; i < compteur; i++){
  521. ct->cmd[i] = malloc(sizeof(Command));
  522. tmp = get_command(ct->cmd[i], line);
  523. line += tmp;
  524. //Si pas dernière commande on trim
  525. if(i + 1 < compteur){
  526. line = ltrim(line, ' ');
  527. }
  528. }
  529. //Retour
  530. return SHELL_OK;
  531. }
  532. int parse_command(Command* c){
  533. //Declaration variable
  534. int length, nbWildcard = 0, res;
  535. char* cmd, **wildcardTab;
  536. //Parse les redirections
  537. length = set_redirection(c);
  538. if(length == SHELL_ERR || length == 0){
  539. return SHELL_ERR;
  540. }
  541. //Recup la commande (sans redirection)
  542. cmd = malloc(sizeof(char) * (length + 1));
  543. memset(cmd, 0, length + 1);
  544. strncpy(cmd, c->cmd, length);
  545. //Split la commande
  546. split_command(c, cmd);
  547. //Analyse wildcard
  548. for(int i = 0; i < c->argc; i++){
  549. //Regarde si il faut remplacer l'argument par une suite de fichier
  550. nbWildcard = wildcard_result(c->argv[i]);
  551. if(nbWildcard > 0){
  552. //Si il y a des resultats on prepare un tableau pour les récupérer
  553. wildcardTab = malloc(sizeof(char*) * nbWildcard);
  554. nbWildcard = wildcard(c->argv[i], nbWildcard, wildcardTab);
  555. //Verif retour
  556. if(nbWildcard == ERR){
  557. serrno = SEWC;
  558. return SHELL_ERR;
  559. }
  560. //Ajoute les wildcard dans argv (le +1 est la pour garder le NULL à la fin)
  561. c->argv = insert_array(i, c->argv, c->argc + 1, wildcardTab, nbWildcard, &res);
  562. if(res == ERR){
  563. serrno = SEADDWC;
  564. return SHELL_ERR;
  565. }
  566. c->argc = res - 1; //On ne compte pas le NULL final
  567. }
  568. }
  569. //Ajout nom commande
  570. c->name = c->argv[0];
  571. //Ici tous est ok
  572. return SHELL_OK;
  573. }
  574. int parse_all_command(CommandTab* ct){
  575. int tmp;
  576. for(int i = 0; i < ct->length; i++){
  577. tmp = parse_command(ct->cmd[i]);
  578. if(tmp != SHELL_OK){
  579. return SHELL_FAIL;
  580. }
  581. }
  582. return SHELL_OK;
  583. }
  584. void clean_command(CommandTab* ct){
  585. extern int errno;
  586. //Vide le tableau
  587. for(int i = 0; i < ct->length; i++){
  588. //Si la commande a été parsée on vide les arguments
  589. if(ct->cmd[i]->argc > 0){
  590. ct->cmd[i]->name = NULL;
  591. for(int j = 0; j < ct->cmd[i]->argc; j++){
  592. free(ct->cmd[i]->argv[j]);
  593. }
  594. }
  595. //Ferme les fichiers ouverts si besoin
  596. if(ct->cmd[i]->input != STDIN){
  597. if(close(ct->cmd[i]->input)){
  598. fprintf(stderr, "Erreur lors de la fermeture du fichier d'input de %s : %s\n", ct->cmd[i]->cmd, strerror(errno));
  599. }
  600. }
  601. if(ct->cmd[i]->output != STDOUT){
  602. if(close(ct->cmd[i]->output)){
  603. fprintf(stderr, "Erreur lors de la fermeture du fichier d'output de %s : %s\n", ct->cmd[i]->cmd, strerror(errno));
  604. }
  605. }
  606. if(ct->cmd[i]->error != STDERR && ct->cmd[i]->error != ct->cmd[i]->output){ //Verif en plus qu'il est different du fichier de sortie standard (>& et >>&)
  607. if(close(ct->cmd[i]->error)){
  608. fprintf(stderr, "Erreur lors de la fermeture du fichier d'error de %s : %s\n", ct->cmd[i]->cmd, strerror(errno));
  609. }
  610. }
  611. //Supprime la ligne de commande
  612. free(ct->cmd[i]->cmd);
  613. //Supprime la structure
  614. free(ct->cmd[i]);
  615. }
  616. //Met à 0 la taille du tableau
  617. ct->length = 0;
  618. }