forked from Jaffar-Hussein/PhyloExplora
-
Notifications
You must be signed in to change notification settings - Fork 0
/
sequences.c
256 lines (232 loc) · 6.17 KB
/
sequences.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
/*--------------------------------
Master BIBS
Universite Paris-Saclay
Projet MiniInfo 1 2023-2024
Sujet propose par George Marchment
george.marchment@lisn.fr
----------------------------------*/
#include "utils.h"
/*------------------------------------------------
Fonctions de manipulation de chaînes de caracteres
--------------------------------------------------*/
/*
Input : une chaîne de caracteres et un caractere
Output : None
Main : Ajoute le caractere à la fin de la chaîne de caracteres
*/
void appendString(char *str, char c)
{
int length = strlen(str);
str[length] = c;
str[length + 1] = '\0';
}
/*
Input : une chaîne de caracteres
Output : None
Main : Initialise la chaîne de caracteres comme une chaîne vide
*/
void set_empty_string(char *str)
{
str[0] = '\0';
}
/*------------------------------------
Fonctions de manipulation de sequences
--------------------------------------*/
/**
* Sets the ID of a Sequence object.
*
* @param sequence The Sequence object to set the ID for.
* @param ID The ID to set.
*/
void set_id(Sequence *sequence, char *ID)
{
int length = strlen(ID);
for (int i = 0; i < length; i++)
{
sequence->ID[i] = ID[i];
}
sequence->ID[length] = '\0';
}
/*
Input : une sequence et une chaîne de caracteres
Output : None
Main : Initialise la chaîne de caracteres en tant que sequence pour l'objet sequence
*/
void set_seq(Sequence *sequence, char *seq)
{
int length = strlen(seq);
for (int i = 0; i < length; i++)
{
sequence->seq[i] = seq[i];
}
sequence->seq[length] = '\0';
}
/*
Input : une sequence et un entier
Output : Un caractere
Main : Retourne le nucleotide correspondant à l'index
*/
char get_nucleotide(Sequence *sequence, int i)
{
return sequence->seq[i];
}
/*
Input : une sequence et deux chaînes de caracteres
Output : None
Main : Initialise l'ID et la sequence pour l'objet sequence
*/
void set_sequence(Sequence *sequence, char *ID, char *seq)
{
set_id(sequence, ID);
set_seq(sequence, seq);
}
/*
Input : une sequence
Output : None
Main : Effectue l'affichage de la sequence
*/
void affiche_sequence(Sequence *sequence)
{
printf("\t"COLOR_RED"*"COLOR_GREEN"ID : " COLOR_BLUE "%s" COLOR_RESET "\n\t"COLOR_RED"*"COLOR_GREEN"Sequence : " COLOR_PURPLE "%s" COLOR_RESET "\n", sequence->ID, sequence->seq);
}
/*-------------------------------
Fonctions de manipulation de fichiers
---------------------------------*/
/*
Input : Adresse d'un fichier
Output : Nombre de sequences dans le fichier
Main : Compte le nombre de sequences presentes dans le fichier
*/
// https://www.geeksforgeeks.org/c-program-to-read-contents-of-whole-file/
int get_number_entries(char *address)
{
int num = 0;
FILE *ptr;
char ch;
ptr = fopen(address, "r");
if (NULL == ptr)
{
printf("Le fichier ne peut pas être ouvert\n");
}
do
{
ch = fgetc(ptr);
if (ch == '>')
{
num += 1;
}
} while (ch != EOF);
fclose(ptr);
return num;
}
/*
Input : Adresse d'un fichier
Output : Une chaîne de caracteres
Main : Fonction qui lit un fichier et retourne la chaîne de caracteres
*/
// https://stackoverflow.com/a/15713419
char *readFile(char *fileName)
{
FILE *file = fopen(fileName, "r");
char *code;
size_t n = 0;
int c;
if (file == NULL)
return NULL; // Impossible d'ouvrir le fichier
fseek(file, 0, SEEK_END);
long f_size = ftell(file);
fseek(file, 0, SEEK_SET);
code = malloc(f_size);
while ((c = fgetc(file)) != EOF)
{
code[n++] = (char)c;
}
code[n] = '\0';
fclose(file);
return code;
}
/*
Input : - une chaîne de caracteres correspondant au code
- un entier correspondant à la position start, à partir de laquelle on va commencer à chercher
- Une sequence qu'on va set à partir de l'information trouvee dans le code
Output : - un entier correspondant à la position où on a fini de "lire"
Main : Fonction qui prend le code ainsi qu'une position start, elle va chercher la prochaine sequence
definie apres le start et set l'objet sequence avec cette information.
Elle retourne ensuite la derniere position qui est lu. Pour le cas de la derniere
sequence elle retourne -1
*/
int extract_next_sequence(char *code, int start_index, Sequence *sequence)
{
int is_reading_id = 1;
char id[ID_MAX_LENGTH];
set_empty_string(id);
char seq[seq_MAX_LENGTH];
set_empty_string(seq);
while (code[++start_index] != '>' && code[start_index] != '\0')
{
if (is_reading_id == 1)
{
if (code[start_index] == '\n')
{
is_reading_id = 0;
}
else
{
appendString(id, code[start_index]);
}
}
else
{
if (code[start_index] != '\n')
{
appendString(seq, code[start_index]);
}
}
}
set_sequence(sequence, id, seq);
if (code[start_index] == '\0')
{
return -1;
}
else
{
return start_index;
}
}
/*
Input : Adresse d'un fichier et une liste de sequences
Output : None
Main : Fonction qui lit un fichier, remplit la liste avec les sequences trouvees dans le fichier
*/
void parse_file(char *address, Sequence tab_sequences[])
{
char *code = readFile(address);
Sequence sequence;
int start = 0;
int counter = 0;
// Keep extracting sequences until extract_next_sequence returns -1
while (start != -1)
{
start = extract_next_sequence(code, start, &sequence);
tab_sequences[counter++] = sequence;
}
}
/*
Input : Adresse d'un fichier
Output : None
Main : Parse le fichier, sauvegarde les sequences dans une liste.
Puis affiche une sequence une par une.
*/
void show_sequences_file(char *file)
{
int nb_entries = get_number_entries(file);
printf("Nombre d'entrees : "COLOR_RED"%d"COLOR_RESET"\n", nb_entries);
printf("\n");
Sequence tab_sequences[nb_entries];
parse_file(file, tab_sequences);
for (int i = 0; i < nb_entries; i++)
{
affiche_sequence(&tab_sequences[i]);
printf("\n");
}
}