Coverage Report

Created: 2026-01-10 06:51

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/curl/lib/ftplistparser.c
Line
Count
Source
1
/***************************************************************************
2
 *                                  _   _ ____  _
3
 *  Project                     ___| | | |  _ \| |
4
 *                             / __| | | | |_) | |
5
 *                            | (__| |_| |  _ <| |___
6
 *                             \___|\___/|_| \_\_____|
7
 *
8
 * Copyright (C) Daniel Stenberg, <daniel@haxx.se>, et al.
9
 *
10
 * This software is licensed as described in the file COPYING, which
11
 * you should have received as part of this distribution. The terms
12
 * are also available at https://curl.se/docs/copyright.html.
13
 *
14
 * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15
 * copies of the Software, and permit persons to whom the Software is
16
 * furnished to do so, under the terms of the COPYING file.
17
 *
18
 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19
 * KIND, either express or implied.
20
 *
21
 * SPDX-License-Identifier: curl
22
 *
23
 ***************************************************************************/
24
#include "curl_setup.h"
25
26
#ifndef CURL_DISABLE_FTP
27
28
/**
29
 * Now implemented:
30
 *
31
 * 1) Unix version 1
32
 * drwxr-xr-x 1 user01 ftp  512 Jan 29 23:32 prog
33
 * 2) Unix version 2
34
 * drwxr-xr-x 1 user01 ftp  512 Jan 29 1997  prog
35
 * 3) Unix version 3
36
 * drwxr-xr-x 1      1   1  512 Jan 29 23:32 prog
37
 * 4) Unix symlink
38
 * lrwxr-xr-x 1 user01 ftp  512 Jan 29 23:32 prog -> prog2000
39
 * 5) DOS style
40
 * 01-29-97 11:32PM <DIR> prog
41
 */
42
43
#include "urldata.h"
44
#include "fileinfo.h"
45
#include "llist.h"
46
#include "ftp.h"
47
#include "ftplistparser.h"
48
#include "curl_fnmatch.h"
49
#include "multiif.h"
50
#include "curlx/strparse.h"
51
52
typedef enum {
53
  PL_UNIX_TOTALSIZE = 0,
54
  PL_UNIX_FILETYPE,
55
  PL_UNIX_PERMISSION,
56
  PL_UNIX_HLINKS,
57
  PL_UNIX_USER,
58
  PL_UNIX_GROUP,
59
  PL_UNIX_SIZE,
60
  PL_UNIX_TIME,
61
  PL_UNIX_FILENAME,
62
  PL_UNIX_SYMLINK
63
} pl_unix_mainstate;
64
65
typedef union {
66
  enum {
67
    PL_UNIX_TOTALSIZE_INIT = 0,
68
    PL_UNIX_TOTALSIZE_READING
69
  } total_dirsize;
70
71
  enum {
72
    PL_UNIX_HLINKS_PRESPACE = 0,
73
    PL_UNIX_HLINKS_NUMBER
74
  } hlinks;
75
76
  enum {
77
    PL_UNIX_USER_PRESPACE = 0,
78
    PL_UNIX_USER_PARSING
79
  } user;
80
81
  enum {
82
    PL_UNIX_GROUP_PRESPACE = 0,
83
    PL_UNIX_GROUP_NAME
84
  } group;
85
86
  enum {
87
    PL_UNIX_SIZE_PRESPACE = 0,
88
    PL_UNIX_SIZE_NUMBER
89
  } size;
90
91
  enum {
92
    PL_UNIX_TIME_PREPART1 = 0,
93
    PL_UNIX_TIME_PART1,
94
    PL_UNIX_TIME_PREPART2,
95
    PL_UNIX_TIME_PART2,
96
    PL_UNIX_TIME_PREPART3,
97
    PL_UNIX_TIME_PART3
98
  } time;
99
100
  enum {
101
    PL_UNIX_FILENAME_PRESPACE = 0,
102
    PL_UNIX_FILENAME_NAME,
103
    PL_UNIX_FILENAME_WINDOWSEOL
104
  } filename;
105
106
  enum {
107
    PL_UNIX_SYMLINK_PRESPACE = 0,
108
    PL_UNIX_SYMLINK_NAME,
109
    PL_UNIX_SYMLINK_PRETARGET1,
110
    PL_UNIX_SYMLINK_PRETARGET2,
111
    PL_UNIX_SYMLINK_PRETARGET3,
112
    PL_UNIX_SYMLINK_PRETARGET4,
113
    PL_UNIX_SYMLINK_TARGET,
114
    PL_UNIX_SYMLINK_WINDOWSEOL
115
  } symlink;
116
} pl_unix_substate;
117
118
typedef enum {
119
  PL_WINNT_DATE = 0,
120
  PL_WINNT_TIME,
121
  PL_WINNT_DIRORSIZE,
122
  PL_WINNT_FILENAME
123
} pl_winNT_mainstate;
124
125
typedef union {
126
  enum {
127
    PL_WINNT_TIME_PRESPACE = 0,
128
    PL_WINNT_TIME_TIME
129
  } time;
130
  enum {
131
    PL_WINNT_DIRORSIZE_PRESPACE = 0,
132
    PL_WINNT_DIRORSIZE_CONTENT
133
  } dirorsize;
134
  enum {
135
    PL_WINNT_FILENAME_PRESPACE = 0,
136
    PL_WINNT_FILENAME_CONTENT,
137
    PL_WINNT_FILENAME_WINEOL
138
  } filename;
139
} pl_winNT_substate;
140
141
/* This struct is used in wildcard downloading - for parsing LIST response */
142
struct ftp_parselist_data {
143
  enum {
144
    OS_TYPE_UNKNOWN = 0,
145
    OS_TYPE_UNIX,
146
    OS_TYPE_WIN_NT
147
  } os_type;
148
149
  union {
150
    struct {
151
      pl_unix_mainstate main;
152
      pl_unix_substate sub;
153
    } UNIX;
154
155
    struct {
156
      pl_winNT_mainstate main;
157
      pl_winNT_substate sub;
158
    } NT;
159
  } state;
160
161
  CURLcode error;
162
  struct fileinfo *file_data;
163
  unsigned int item_length;
164
  size_t item_offset;
165
  struct {
166
    size_t filename;
167
    size_t user;
168
    size_t group;
169
    size_t time;
170
    size_t perm;
171
    size_t symlink_target;
172
  } offsets;
173
};
174
175
static void fileinfo_dtor(void *user, void *element)
176
0
{
177
0
  (void)user;
178
0
  Curl_fileinfo_cleanup(element);
179
0
}
180
181
void Curl_wildcard_init(struct WildcardData *wc)
182
0
{
183
0
  Curl_llist_init(&wc->filelist, fileinfo_dtor);
184
0
  wc->state = CURLWC_INIT;
185
0
}
186
187
void Curl_wildcard_dtor(struct WildcardData **wcp)
188
0
{
189
0
  struct WildcardData *wc = *wcp;
190
0
  if(!wc)
191
0
    return;
192
193
0
  if(wc->dtor) {
194
0
    wc->dtor(wc->ftpwc);
195
0
    wc->dtor = ZERO_NULL;
196
0
    wc->ftpwc = NULL;
197
0
  }
198
0
  DEBUGASSERT(wc->ftpwc == NULL);
199
200
0
  Curl_llist_destroy(&wc->filelist, NULL);
201
0
  curlx_free(wc->path);
202
0
  wc->path = NULL;
203
0
  curlx_free(wc->pattern);
204
0
  wc->pattern = NULL;
205
0
  wc->state = CURLWC_INIT;
206
0
  curlx_free(wc);
207
0
  *wcp = NULL;
208
0
}
209
210
struct ftp_parselist_data *Curl_ftp_parselist_data_alloc(void)
211
0
{
212
0
  return curlx_calloc(1, sizeof(struct ftp_parselist_data));
213
0
}
214
215
void Curl_ftp_parselist_data_free(struct ftp_parselist_data **parserp)
216
0
{
217
0
  struct ftp_parselist_data *parser = *parserp;
218
0
  if(parser)
219
0
    Curl_fileinfo_cleanup(parser->file_data);
220
0
  curlx_free(parser);
221
0
  *parserp = NULL;
222
0
}
223
224
CURLcode Curl_ftp_parselist_geterror(struct ftp_parselist_data *pl_data)
225
0
{
226
0
  return pl_data->error;
227
0
}
228
229
0
#define FTP_LP_MALFORMATED_PERM 0x01000000
230
231
static unsigned int ftp_pl_get_permission(const char *str)
232
0
{
233
0
  unsigned int permissions = 0;
234
  /* USER */
235
0
  if(str[0] == 'r')
236
0
    permissions |= 1 << 8;
237
0
  else if(str[0] != '-')
238
0
    permissions |= FTP_LP_MALFORMATED_PERM;
239
0
  if(str[1] == 'w')
240
0
    permissions |= 1 << 7;
241
0
  else if(str[1] != '-')
242
0
    permissions |= FTP_LP_MALFORMATED_PERM;
243
244
0
  if(str[2] == 'x')
245
0
    permissions |= 1 << 6;
246
0
  else if(str[2] == 's') {
247
0
    permissions |= 1 << 6;
248
0
    permissions |= 1 << 11;
249
0
  }
250
0
  else if(str[2] == 'S')
251
0
    permissions |= 1 << 11;
252
0
  else if(str[2] != '-')
253
0
    permissions |= FTP_LP_MALFORMATED_PERM;
254
  /* GROUP */
255
0
  if(str[3] == 'r')
256
0
    permissions |= 1 << 5;
257
0
  else if(str[3] != '-')
258
0
    permissions |= FTP_LP_MALFORMATED_PERM;
259
0
  if(str[4] == 'w')
260
0
    permissions |= 1 << 4;
261
0
  else if(str[4] != '-')
262
0
    permissions |= FTP_LP_MALFORMATED_PERM;
263
0
  if(str[5] == 'x')
264
0
    permissions |= 1 << 3;
265
0
  else if(str[5] == 's') {
266
0
    permissions |= 1 << 3;
267
0
    permissions |= 1 << 10;
268
0
  }
269
0
  else if(str[5] == 'S')
270
0
    permissions |= 1 << 10;
271
0
  else if(str[5] != '-')
272
0
    permissions |= FTP_LP_MALFORMATED_PERM;
273
  /* others */
274
0
  if(str[6] == 'r')
275
0
    permissions |= 1 << 2;
276
0
  else if(str[6] != '-')
277
0
    permissions |= FTP_LP_MALFORMATED_PERM;
278
0
  if(str[7] == 'w')
279
0
    permissions |= 1 << 1;
280
0
  else if(str[7] != '-')
281
0
    permissions |= FTP_LP_MALFORMATED_PERM;
282
0
  if(str[8] == 'x')
283
0
    permissions |= 1;
284
0
  else if(str[8] == 't') {
285
0
    permissions |= 1;
286
0
    permissions |= 1 << 9;
287
0
  }
288
0
  else if(str[8] == 'T')
289
0
    permissions |= 1 << 9;
290
0
  else if(str[8] != '-')
291
0
    permissions |= FTP_LP_MALFORMATED_PERM;
292
293
0
  return permissions;
294
0
}
295
296
static CURLcode ftp_pl_insert_finfo(struct Curl_easy *data,
297
                                    struct fileinfo *infop)
298
0
{
299
0
  curl_fnmatch_callback compare;
300
0
  struct WildcardData *wc = data->wildcard;
301
0
  struct ftp_wc *ftpwc = wc->ftpwc;
302
0
  struct Curl_llist *llist = &wc->filelist;
303
0
  struct ftp_parselist_data *parser = ftpwc->parser;
304
0
  bool add = TRUE;
305
0
  struct curl_fileinfo *finfo = &infop->info;
306
307
  /* set the finfo pointers */
308
0
  char *str = curlx_dyn_ptr(&infop->buf);
309
0
  finfo->filename       = str + parser->offsets.filename;
310
0
  finfo->strings.group  = parser->offsets.group ?
311
0
                          str + parser->offsets.group : NULL;
312
0
  finfo->strings.perm   = parser->offsets.perm ?
313
0
                          str + parser->offsets.perm : NULL;
314
0
  finfo->strings.target = parser->offsets.symlink_target ?
315
0
                          str + parser->offsets.symlink_target : NULL;
316
0
  finfo->strings.time   = str + parser->offsets.time;
317
0
  finfo->strings.user   = parser->offsets.user ?
318
0
                          str + parser->offsets.user : NULL;
319
320
  /* get correct fnmatch callback */
321
0
  compare = data->set.fnmatch;
322
0
  if(!compare)
323
0
    compare = Curl_fnmatch;
324
325
  /* filter pattern-corresponding filenames */
326
0
  Curl_set_in_callback(data, TRUE);
327
0
  if(compare(data->set.fnmatch_data, wc->pattern,
328
0
             finfo->filename) == 0) {
329
    /* discard symlink which is containing multiple " -> " */
330
0
    if((finfo->filetype == CURLFILETYPE_SYMLINK) && finfo->strings.target &&
331
0
       (strstr(finfo->strings.target, " -> "))) {
332
0
      add = FALSE;
333
0
    }
334
0
  }
335
0
  else {
336
0
    add = FALSE;
337
0
  }
338
0
  Curl_set_in_callback(data, FALSE);
339
340
0
  if(add) {
341
0
    Curl_llist_append(llist, finfo, &infop->list);
342
0
  }
343
0
  else {
344
0
    Curl_fileinfo_cleanup(infop);
345
0
  }
346
347
0
  ftpwc->parser->file_data = NULL;
348
0
  return CURLE_OK;
349
0
}
350
351
0
#define MAX_FTPLIST_BUFFER 10000 /* arbitrarily set */
352
353
static CURLcode unix_filetype(const char c, curlfiletype *t)
354
0
{
355
0
  switch(c) {
356
0
  case '-':
357
0
    *t = CURLFILETYPE_FILE;
358
0
    break;
359
0
  case 'd':
360
0
    *t = CURLFILETYPE_DIRECTORY;
361
0
    break;
362
0
  case 'l':
363
0
    *t = CURLFILETYPE_SYMLINK;
364
0
    break;
365
0
  case 'p':
366
0
    *t = CURLFILETYPE_NAMEDPIPE;
367
0
    break;
368
0
  case 's':
369
0
    *t = CURLFILETYPE_SOCKET;
370
0
    break;
371
0
  case 'c':
372
0
    *t = CURLFILETYPE_DEVICE_CHAR;
373
0
    break;
374
0
  case 'b':
375
0
    *t = CURLFILETYPE_DEVICE_BLOCK;
376
0
    break;
377
0
  case 'D':
378
0
    *t = CURLFILETYPE_DOOR;
379
0
    break;
380
0
  default:
381
0
    return CURLE_FTP_BAD_FILE_LIST;
382
0
  }
383
0
  return CURLE_OK;
384
0
}
385
386
static CURLcode parse_unix_totalsize(struct ftp_parselist_data *parser,
387
                                     struct fileinfo *infop,
388
                                     const char c)
389
0
{
390
0
  size_t len = curlx_dyn_len(&infop->buf);
391
0
  char *mem = curlx_dyn_ptr(&infop->buf);
392
0
  switch(parser->state.UNIX.sub.total_dirsize) {
393
0
  case PL_UNIX_TOTALSIZE_INIT:
394
0
    if(c == 't') {
395
0
      parser->state.UNIX.sub.total_dirsize = PL_UNIX_TOTALSIZE_READING;
396
0
      parser->item_length++;
397
0
    }
398
0
    else {
399
0
      parser->state.UNIX.main = PL_UNIX_FILETYPE;
400
      /* continue to fall through */
401
0
    }
402
0
    break;
403
0
  case PL_UNIX_TOTALSIZE_READING:
404
0
    parser->item_length++;
405
0
    if(c == '\r') {
406
0
      parser->item_length--;
407
0
      if(len)
408
0
        curlx_dyn_setlen(&infop->buf, --len);
409
0
    }
410
0
    else if(c == '\n') {
411
0
      mem[parser->item_length - 1] = 0;
412
0
      if(!strncmp("total ", mem, 6)) {
413
0
        const char *endptr = mem + 6;
414
        /* here we can deal with directory size, pass the leading
415
           whitespace and then the digits */
416
0
        curlx_str_passblanks(&endptr);
417
0
        while(ISDIGIT(*endptr))
418
0
          endptr++;
419
0
        if(*endptr) {
420
0
          return CURLE_FTP_BAD_FILE_LIST;
421
0
        }
422
0
        parser->state.UNIX.main = PL_UNIX_FILETYPE;
423
0
        curlx_dyn_reset(&infop->buf);
424
0
      }
425
0
      else
426
0
        return CURLE_FTP_BAD_FILE_LIST;
427
0
    }
428
0
    break;
429
0
  }
430
0
  return CURLE_OK;
431
0
}
432
433
static CURLcode parse_unix_permission(struct ftp_parselist_data *parser,
434
                                      struct fileinfo *infop,
435
                                      const char c)
436
0
{
437
0
  char *mem = curlx_dyn_ptr(&infop->buf);
438
0
  parser->item_length++;
439
0
  if((parser->item_length <= 9) && !strchr("rwx-tTsS", c))
440
0
    return CURLE_FTP_BAD_FILE_LIST;
441
442
0
  else if(parser->item_length == 10) {
443
0
    unsigned int perm;
444
0
    if(c != ' ')
445
0
      return CURLE_FTP_BAD_FILE_LIST;
446
447
0
    mem[10] = 0; /* terminate permissions */
448
0
    perm = ftp_pl_get_permission(mem + parser->item_offset);
449
0
    if(perm & FTP_LP_MALFORMATED_PERM)
450
0
      return CURLE_FTP_BAD_FILE_LIST;
451
452
0
    parser->file_data->info.flags |= CURLFINFOFLAG_KNOWN_PERM;
453
0
    parser->file_data->info.perm = perm;
454
0
    parser->offsets.perm = parser->item_offset;
455
456
0
    parser->item_length = 0;
457
0
    parser->state.UNIX.main = PL_UNIX_HLINKS;
458
0
    parser->state.UNIX.sub.hlinks = PL_UNIX_HLINKS_PRESPACE;
459
0
  }
460
0
  return CURLE_OK;
461
0
}
462
463
static CURLcode parse_unix_hlinks(struct ftp_parselist_data *parser,
464
                                  struct fileinfo *infop,
465
                                  const char c)
466
0
{
467
0
  size_t len = curlx_dyn_len(&infop->buf);
468
0
  char *mem = curlx_dyn_ptr(&infop->buf);
469
470
0
  switch(parser->state.UNIX.sub.hlinks) {
471
0
  case PL_UNIX_HLINKS_PRESPACE:
472
0
    if(c != ' ') {
473
0
      if(ISDIGIT(c) && len) {
474
0
        parser->item_offset = len - 1;
475
0
        parser->item_length = 1;
476
0
        parser->state.UNIX.sub.hlinks = PL_UNIX_HLINKS_NUMBER;
477
0
      }
478
0
      else
479
0
        return CURLE_FTP_BAD_FILE_LIST;
480
0
    }
481
0
    break;
482
0
  case PL_UNIX_HLINKS_NUMBER:
483
0
    parser->item_length++;
484
0
    if(c == ' ') {
485
0
      const char *p = &mem[parser->item_offset];
486
0
      curl_off_t hlinks;
487
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
488
489
0
      if(!curlx_str_number(&p, &hlinks, LONG_MAX)) {
490
0
        parser->file_data->info.flags |= CURLFINFOFLAG_KNOWN_HLINKCOUNT;
491
0
        parser->file_data->info.hardlinks = (long)hlinks;
492
0
      }
493
0
      parser->item_length = 0;
494
0
      parser->item_offset = 0;
495
0
      parser->state.UNIX.main = PL_UNIX_USER;
496
0
      parser->state.UNIX.sub.user = PL_UNIX_USER_PRESPACE;
497
0
    }
498
0
    else if(!ISDIGIT(c))
499
0
      return CURLE_FTP_BAD_FILE_LIST;
500
501
0
    break;
502
0
  }
503
0
  return CURLE_OK;
504
0
}
505
506
static CURLcode parse_unix_user(struct ftp_parselist_data *parser,
507
                                struct fileinfo *infop,
508
                                const char c)
509
0
{
510
0
  size_t len = curlx_dyn_len(&infop->buf);
511
0
  char *mem = curlx_dyn_ptr(&infop->buf);
512
0
  switch(parser->state.UNIX.sub.user) {
513
0
  case PL_UNIX_USER_PRESPACE:
514
0
    if(c != ' ' && len) {
515
0
      parser->item_offset = len - 1;
516
0
      parser->item_length = 1;
517
0
      parser->state.UNIX.sub.user = PL_UNIX_USER_PARSING;
518
0
    }
519
0
    break;
520
0
  case PL_UNIX_USER_PARSING:
521
0
    parser->item_length++;
522
0
    if(c == ' ') {
523
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
524
0
      parser->offsets.user = parser->item_offset;
525
0
      parser->state.UNIX.main = PL_UNIX_GROUP;
526
0
      parser->state.UNIX.sub.group = PL_UNIX_GROUP_PRESPACE;
527
0
      parser->item_offset = 0;
528
0
      parser->item_length = 0;
529
0
    }
530
0
    break;
531
0
  }
532
0
  return CURLE_OK;
533
0
}
534
535
static CURLcode parse_unix_group(struct ftp_parselist_data *parser,
536
                                 struct fileinfo *infop,
537
                                 const char c)
538
0
{
539
0
  size_t len = curlx_dyn_len(&infop->buf);
540
0
  char *mem = curlx_dyn_ptr(&infop->buf);
541
0
  switch(parser->state.UNIX.sub.group) {
542
0
  case PL_UNIX_GROUP_PRESPACE:
543
0
    if(c != ' ' && len) {
544
0
      parser->item_offset = len - 1;
545
0
      parser->item_length = 1;
546
0
      parser->state.UNIX.sub.group = PL_UNIX_GROUP_NAME;
547
0
    }
548
0
    break;
549
0
  case PL_UNIX_GROUP_NAME:
550
0
    parser->item_length++;
551
0
    if(c == ' ') {
552
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
553
0
      parser->offsets.group = parser->item_offset;
554
0
      parser->state.UNIX.main = PL_UNIX_SIZE;
555
0
      parser->state.UNIX.sub.size = PL_UNIX_SIZE_PRESPACE;
556
0
      parser->item_offset = 0;
557
0
      parser->item_length = 0;
558
0
    }
559
0
    break;
560
0
  }
561
0
  return CURLE_OK;
562
0
}
563
564
static CURLcode parse_unix_size(struct ftp_parselist_data *parser,
565
                                struct fileinfo *infop,
566
                                const char c)
567
0
{
568
0
  size_t len = curlx_dyn_len(&infop->buf);
569
0
  char *mem = curlx_dyn_ptr(&infop->buf);
570
0
  switch(parser->state.UNIX.sub.size) {
571
0
  case PL_UNIX_SIZE_PRESPACE:
572
0
    if(c != ' ') {
573
0
      if(ISDIGIT(c) && len) {
574
0
        parser->item_offset = len - 1;
575
0
        parser->item_length = 1;
576
0
        parser->state.UNIX.sub.size = PL_UNIX_SIZE_NUMBER;
577
0
      }
578
0
      else
579
0
        return CURLE_FTP_BAD_FILE_LIST;
580
0
    }
581
0
    break;
582
0
  case PL_UNIX_SIZE_NUMBER:
583
0
    parser->item_length++;
584
0
    if(c == ' ') {
585
0
      const char *p = mem + parser->item_offset;
586
0
      curl_off_t fsize;
587
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
588
0
      if(!curlx_str_numblanks(&p, &fsize)) {
589
0
        if(p[0] == '\0' && fsize != CURL_OFF_T_MAX) {
590
0
          parser->file_data->info.flags |= CURLFINFOFLAG_KNOWN_SIZE;
591
0
          parser->file_data->info.size = fsize;
592
0
        }
593
0
        parser->item_length = 0;
594
0
        parser->item_offset = 0;
595
0
        parser->state.UNIX.main = PL_UNIX_TIME;
596
0
        parser->state.UNIX.sub.time = PL_UNIX_TIME_PREPART1;
597
0
      }
598
0
    }
599
0
    else if(!ISDIGIT(c))
600
0
      return CURLE_FTP_BAD_FILE_LIST;
601
602
0
    break;
603
0
  }
604
0
  return CURLE_OK;
605
0
}
606
607
static CURLcode parse_unix_time(struct ftp_parselist_data *parser,
608
                                struct fileinfo *infop,
609
                                const char c)
610
0
{
611
0
  size_t len = curlx_dyn_len(&infop->buf);
612
0
  char *mem = curlx_dyn_ptr(&infop->buf);
613
0
  struct curl_fileinfo *finfo = &infop->info;
614
615
0
  switch(parser->state.UNIX.sub.time) {
616
0
  case PL_UNIX_TIME_PREPART1:
617
0
    if(c != ' ') {
618
0
      if(ISALNUM(c) && len) {
619
0
        parser->item_offset = len - 1;
620
0
        parser->item_length = 1;
621
0
        parser->state.UNIX.sub.time = PL_UNIX_TIME_PART1;
622
0
      }
623
0
      else
624
0
        return CURLE_FTP_BAD_FILE_LIST;
625
0
    }
626
0
    break;
627
0
  case PL_UNIX_TIME_PART1:
628
0
    parser->item_length++;
629
0
    if(c == ' ')
630
0
      parser->state.UNIX.sub.time = PL_UNIX_TIME_PREPART2;
631
632
0
    else if(!ISALNUM(c) && c != '.')
633
0
      return CURLE_FTP_BAD_FILE_LIST;
634
635
0
    break;
636
0
  case PL_UNIX_TIME_PREPART2:
637
0
    parser->item_length++;
638
0
    if(c != ' ') {
639
0
      if(ISALNUM(c))
640
0
        parser->state.UNIX.sub.time = PL_UNIX_TIME_PART2;
641
0
      else
642
0
        return CURLE_FTP_BAD_FILE_LIST;
643
0
    }
644
0
    break;
645
0
  case PL_UNIX_TIME_PART2:
646
0
    parser->item_length++;
647
0
    if(c == ' ')
648
0
      parser->state.UNIX.sub.time = PL_UNIX_TIME_PREPART3;
649
0
    else if(!ISALNUM(c) && c != '.')
650
0
      return CURLE_FTP_BAD_FILE_LIST;
651
0
    break;
652
0
  case PL_UNIX_TIME_PREPART3:
653
0
    parser->item_length++;
654
0
    if(c != ' ') {
655
0
      if(ISALNUM(c))
656
0
        parser->state.UNIX.sub.time = PL_UNIX_TIME_PART3;
657
0
      else
658
0
        return CURLE_FTP_BAD_FILE_LIST;
659
0
    }
660
0
    break;
661
0
  case PL_UNIX_TIME_PART3:
662
0
    parser->item_length++;
663
0
    if(c == ' ') {
664
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
665
0
      parser->offsets.time = parser->item_offset;
666
0
      if(finfo->filetype == CURLFILETYPE_SYMLINK) {
667
0
        parser->state.UNIX.main = PL_UNIX_SYMLINK;
668
0
        parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_PRESPACE;
669
0
      }
670
0
      else {
671
0
        parser->state.UNIX.main = PL_UNIX_FILENAME;
672
0
        parser->state.UNIX.sub.filename = PL_UNIX_FILENAME_PRESPACE;
673
0
      }
674
0
    }
675
0
    else if(!ISALNUM(c) && c != '.' && c != ':')
676
0
      return CURLE_FTP_BAD_FILE_LIST;
677
0
    break;
678
0
  }
679
0
  return CURLE_OK;
680
0
}
681
682
static CURLcode parse_unix_filename(struct Curl_easy *data,
683
                                    struct ftp_parselist_data *parser,
684
                                    struct fileinfo *infop,
685
                                    const char c)
686
0
{
687
0
  size_t len = curlx_dyn_len(&infop->buf);
688
0
  char *mem = curlx_dyn_ptr(&infop->buf);
689
0
  CURLcode result = CURLE_OK;
690
691
0
  switch(parser->state.UNIX.sub.filename) {
692
0
  case PL_UNIX_FILENAME_PRESPACE:
693
0
    if(c != ' ' && len) {
694
0
      parser->item_offset = len - 1;
695
0
      parser->item_length = 1;
696
0
      parser->state.UNIX.sub.filename = PL_UNIX_FILENAME_NAME;
697
0
    }
698
0
    break;
699
0
  case PL_UNIX_FILENAME_NAME:
700
0
    parser->item_length++;
701
0
    if(c == '\r')
702
0
      parser->state.UNIX.sub.filename = PL_UNIX_FILENAME_WINDOWSEOL;
703
704
0
    else if(c == '\n') {
705
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
706
0
      parser->offsets.filename = parser->item_offset;
707
0
      parser->state.UNIX.main = PL_UNIX_FILETYPE;
708
0
      result = ftp_pl_insert_finfo(data, infop);
709
0
    }
710
0
    break;
711
0
  case PL_UNIX_FILENAME_WINDOWSEOL:
712
0
    if(c == '\n') {
713
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
714
0
      parser->offsets.filename = parser->item_offset;
715
0
      parser->state.UNIX.main = PL_UNIX_FILETYPE;
716
0
      result = ftp_pl_insert_finfo(data, infop);
717
0
    }
718
0
    else
719
0
      result = CURLE_FTP_BAD_FILE_LIST;
720
0
    break;
721
0
  }
722
0
  return result;
723
0
}
724
725
static CURLcode parse_unix_symlink(struct Curl_easy *data,
726
                                   struct ftp_parselist_data *parser,
727
                                   struct fileinfo *infop,
728
                                   const char c)
729
0
{
730
0
  size_t len = curlx_dyn_len(&infop->buf);
731
0
  char *mem = curlx_dyn_ptr(&infop->buf);
732
0
  CURLcode result = CURLE_OK;
733
734
0
  switch(parser->state.UNIX.sub.symlink) {
735
0
  case PL_UNIX_SYMLINK_PRESPACE:
736
0
    if(c != ' ' && len) {
737
0
      parser->item_offset = len - 1;
738
0
      parser->item_length = 1;
739
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_NAME;
740
0
    }
741
0
    break;
742
0
  case PL_UNIX_SYMLINK_NAME:
743
0
    parser->item_length++;
744
0
    if(c == ' ')
745
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_PRETARGET1;
746
747
0
    else if(c == '\r' || c == '\n')
748
0
      return CURLE_FTP_BAD_FILE_LIST;
749
750
0
    break;
751
0
  case PL_UNIX_SYMLINK_PRETARGET1:
752
0
    parser->item_length++;
753
0
    if(c == '-')
754
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_PRETARGET2;
755
756
0
    else if(c == '\r' || c == '\n')
757
0
      return CURLE_FTP_BAD_FILE_LIST;
758
0
    else
759
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_NAME;
760
0
    break;
761
0
  case PL_UNIX_SYMLINK_PRETARGET2:
762
0
    parser->item_length++;
763
0
    if(c == '>')
764
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_PRETARGET3;
765
0
    else if(c == '\r' || c == '\n')
766
0
      return CURLE_FTP_BAD_FILE_LIST;
767
0
    else
768
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_NAME;
769
770
0
    break;
771
0
  case PL_UNIX_SYMLINK_PRETARGET3:
772
0
    parser->item_length++;
773
0
    if(c == ' ') {
774
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_PRETARGET4;
775
      /* now place where is symlink following */
776
0
      mem[parser->item_offset + parser->item_length - 4] = 0;
777
0
      parser->offsets.filename = parser->item_offset;
778
0
      parser->item_length = 0;
779
0
      parser->item_offset = 0;
780
0
    }
781
0
    else if(c == '\r' || c == '\n')
782
0
      return CURLE_FTP_BAD_FILE_LIST;
783
0
    else
784
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_NAME;
785
0
    break;
786
0
  case PL_UNIX_SYMLINK_PRETARGET4:
787
0
    if(c != '\r' && c != '\n' && len) {
788
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_TARGET;
789
0
      parser->item_offset = len - 1;
790
0
      parser->item_length = 1;
791
0
    }
792
0
    else
793
0
      return CURLE_FTP_BAD_FILE_LIST;
794
795
0
    break;
796
0
  case PL_UNIX_SYMLINK_TARGET:
797
0
    parser->item_length++;
798
0
    if(c == '\r')
799
0
      parser->state.UNIX.sub.symlink = PL_UNIX_SYMLINK_WINDOWSEOL;
800
801
0
    else if(c == '\n') {
802
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
803
0
      parser->offsets.symlink_target = parser->item_offset;
804
0
      result = ftp_pl_insert_finfo(data, infop);
805
0
      if(result)
806
0
        break;
807
808
0
      parser->state.UNIX.main = PL_UNIX_FILETYPE;
809
0
    }
810
0
    break;
811
0
  case PL_UNIX_SYMLINK_WINDOWSEOL:
812
0
    if(c == '\n') {
813
0
      mem[parser->item_offset + parser->item_length - 1] = 0;
814
0
      parser->offsets.symlink_target = parser->item_offset;
815
0
      result = ftp_pl_insert_finfo(data, infop);
816
0
      if(result)
817
0
        break;
818
819
0
      parser->state.UNIX.main = PL_UNIX_FILETYPE;
820
0
    }
821
0
    else
822
0
      result = CURLE_FTP_BAD_FILE_LIST;
823
824
0
    break;
825
0
  }
826
0
  return result;
827
0
}
828
829
static CURLcode parse_unix(struct Curl_easy *data,
830
                           struct ftp_parselist_data *parser,
831
                           struct fileinfo *infop,
832
                           const char c)
833
0
{
834
0
  struct curl_fileinfo *finfo = &infop->info;
835
0
  CURLcode result = CURLE_OK;
836
837
0
  switch(parser->state.UNIX.main) {
838
0
  case PL_UNIX_TOTALSIZE:
839
0
    result = parse_unix_totalsize(parser, infop, c);
840
0
    if(result)
841
0
      break;
842
0
    if(parser->state.UNIX.main != PL_UNIX_FILETYPE)
843
0
      break;
844
0
    FALLTHROUGH();
845
0
  case PL_UNIX_FILETYPE:
846
0
    result = unix_filetype(c, &finfo->filetype);
847
0
    if(!result) {
848
0
      parser->state.UNIX.main = PL_UNIX_PERMISSION;
849
0
      parser->item_length = 0;
850
0
      parser->item_offset = 1;
851
0
    }
852
0
    break;
853
0
  case PL_UNIX_PERMISSION:
854
0
    result = parse_unix_permission(parser, infop, c);
855
0
    break;
856
0
  case PL_UNIX_HLINKS:
857
0
    result = parse_unix_hlinks(parser, infop, c);
858
0
    break;
859
0
  case PL_UNIX_USER:
860
0
    result = parse_unix_user(parser, infop, c);
861
0
    break;
862
0
  case PL_UNIX_GROUP:
863
0
    result = parse_unix_group(parser, infop, c);
864
0
    break;
865
0
  case PL_UNIX_SIZE:
866
0
    result = parse_unix_size(parser, infop, c);
867
0
    break;
868
0
  case PL_UNIX_TIME:
869
0
    result = parse_unix_time(parser, infop, c);
870
0
    break;
871
0
  case PL_UNIX_FILENAME:
872
0
    result = parse_unix_filename(data, parser, infop, c);
873
0
    break;
874
0
  case PL_UNIX_SYMLINK:
875
0
    result = parse_unix_symlink(data, parser, infop, c);
876
0
    break;
877
0
  }
878
0
  return result;
879
0
}
880
881
static CURLcode parse_winnt(struct Curl_easy *data,
882
                            struct ftp_parselist_data *parser,
883
                            struct fileinfo *infop,
884
                            const char c)
885
0
{
886
0
  struct curl_fileinfo *finfo = &infop->info;
887
0
  size_t len = curlx_dyn_len(&infop->buf);
888
0
  char *mem = curlx_dyn_ptr(&infop->buf);
889
0
  CURLcode result = CURLE_OK;
890
891
0
  switch(parser->state.NT.main) {
892
0
  case PL_WINNT_DATE:
893
0
    parser->item_length++;
894
0
    if(parser->item_length < 9) {
895
0
      if(!strchr("0123456789-", c)) { /* only simple control */
896
0
        return CURLE_FTP_BAD_FILE_LIST;
897
0
      }
898
0
    }
899
0
    else if(parser->item_length == 9) {
900
0
      if(c == ' ') {
901
0
        parser->state.NT.main = PL_WINNT_TIME;
902
0
        parser->state.NT.sub.time = PL_WINNT_TIME_PRESPACE;
903
0
      }
904
0
      else
905
0
        return CURLE_FTP_BAD_FILE_LIST;
906
0
    }
907
0
    else
908
0
      return CURLE_FTP_BAD_FILE_LIST;
909
0
    break;
910
0
  case PL_WINNT_TIME:
911
0
    parser->item_length++;
912
0
    switch(parser->state.NT.sub.time) {
913
0
    case PL_WINNT_TIME_PRESPACE:
914
0
      if(!ISBLANK(c))
915
0
        parser->state.NT.sub.time = PL_WINNT_TIME_TIME;
916
0
      break;
917
0
    case PL_WINNT_TIME_TIME:
918
0
      if(c == ' ') {
919
0
        parser->offsets.time = parser->item_offset;
920
0
        mem[parser->item_offset + parser->item_length - 1] = 0;
921
0
        parser->state.NT.main = PL_WINNT_DIRORSIZE;
922
0
        parser->state.NT.sub.dirorsize = PL_WINNT_DIRORSIZE_PRESPACE;
923
0
        parser->item_length = 0;
924
0
      }
925
0
      else if(!strchr("APM0123456789:", c))
926
0
        return CURLE_FTP_BAD_FILE_LIST;
927
0
      break;
928
0
    }
929
0
    break;
930
0
  case PL_WINNT_DIRORSIZE:
931
0
    switch(parser->state.NT.sub.dirorsize) {
932
0
    case PL_WINNT_DIRORSIZE_PRESPACE:
933
0
      if(c != ' ' && len) {
934
0
        parser->item_offset = len - 1;
935
0
        parser->item_length = 1;
936
0
        parser->state.NT.sub.dirorsize = PL_WINNT_DIRORSIZE_CONTENT;
937
0
      }
938
0
      break;
939
0
    case PL_WINNT_DIRORSIZE_CONTENT:
940
0
      parser->item_length++;
941
0
      if(c == ' ') {
942
0
        mem[parser->item_offset + parser->item_length - 1] = 0;
943
0
        if(strcmp("<DIR>", mem + parser->item_offset) == 0) {
944
0
          finfo->filetype = CURLFILETYPE_DIRECTORY;
945
0
          finfo->size = 0;
946
0
        }
947
0
        else {
948
0
          const char *p = mem + parser->item_offset;
949
0
          if(curlx_str_numblanks(&p, &finfo->size)) {
950
0
            return CURLE_FTP_BAD_FILE_LIST;
951
0
          }
952
          /* correct file type */
953
0
          parser->file_data->info.filetype = CURLFILETYPE_FILE;
954
0
        }
955
956
0
        parser->file_data->info.flags |= CURLFINFOFLAG_KNOWN_SIZE;
957
0
        parser->item_length = 0;
958
0
        parser->state.NT.main = PL_WINNT_FILENAME;
959
0
        parser->state.NT.sub.filename = PL_WINNT_FILENAME_PRESPACE;
960
0
      }
961
0
      break;
962
0
    }
963
0
    break;
964
0
  case PL_WINNT_FILENAME:
965
0
    switch(parser->state.NT.sub.filename) {
966
0
    case PL_WINNT_FILENAME_PRESPACE:
967
0
      if(c != ' ' && len) {
968
0
        parser->item_offset = len - 1;
969
0
        parser->item_length = 1;
970
0
        parser->state.NT.sub.filename = PL_WINNT_FILENAME_CONTENT;
971
0
      }
972
0
      break;
973
0
    case PL_WINNT_FILENAME_CONTENT:
974
0
      parser->item_length++;
975
0
      if(!len)
976
0
        return CURLE_FTP_BAD_FILE_LIST;
977
0
      if(c == '\r') {
978
0
        parser->state.NT.sub.filename = PL_WINNT_FILENAME_WINEOL;
979
0
        mem[len - 1] = 0;
980
0
      }
981
0
      else if(c == '\n') {
982
0
        parser->offsets.filename = parser->item_offset;
983
0
        mem[len - 1] = 0;
984
0
        result = ftp_pl_insert_finfo(data, infop);
985
0
        if(result)
986
0
          return result;
987
988
0
        parser->state.NT.main = PL_WINNT_DATE;
989
0
        parser->state.NT.sub.filename = PL_WINNT_FILENAME_PRESPACE;
990
0
      }
991
0
      break;
992
0
    case PL_WINNT_FILENAME_WINEOL:
993
0
      if(c == '\n') {
994
0
        parser->offsets.filename = parser->item_offset;
995
0
        result = ftp_pl_insert_finfo(data, infop);
996
0
        if(result)
997
0
          return result;
998
999
0
        parser->state.NT.main = PL_WINNT_DATE;
1000
0
        parser->state.NT.sub.filename = PL_WINNT_FILENAME_PRESPACE;
1001
0
      }
1002
0
      else
1003
0
        return CURLE_FTP_BAD_FILE_LIST;
1004
1005
0
      break;
1006
0
    }
1007
0
    break;
1008
0
  }
1009
1010
0
  return CURLE_OK;
1011
0
}
1012
1013
size_t Curl_ftp_parselist(char *buffer, size_t size, size_t nmemb,
1014
                          void *connptr)
1015
0
{
1016
0
  size_t bufflen = size * nmemb;
1017
0
  struct Curl_easy *data = (struct Curl_easy *)connptr;
1018
0
  struct ftp_wc *ftpwc = data->wildcard->ftpwc;
1019
0
  struct ftp_parselist_data *parser = ftpwc->parser;
1020
0
  size_t i = 0;
1021
0
  CURLcode result;
1022
0
  size_t retsize = bufflen;
1023
1024
0
  if(parser->error) { /* error in previous call */
1025
    /* scenario:
1026
     * 1. call => OK..
1027
     * 2. call => OUT_OF_MEMORY (or other error)
1028
     * 3. (last) call => is skipped RIGHT HERE and the error is handled later
1029
     *    in wc_statemach()
1030
     */
1031
0
    goto fail;
1032
0
  }
1033
1034
0
  if(parser->os_type == OS_TYPE_UNKNOWN && bufflen > 0) {
1035
    /* considering info about FILE response format */
1036
0
    parser->os_type = ISDIGIT(buffer[0]) ? OS_TYPE_WIN_NT : OS_TYPE_UNIX;
1037
0
  }
1038
1039
0
  while(i < bufflen) { /* FSM */
1040
0
    char c = buffer[i];
1041
0
    struct fileinfo *infop;
1042
0
    if(!parser->file_data) { /* tmp file data is not allocated yet */
1043
0
      parser->file_data = Curl_fileinfo_alloc();
1044
0
      if(!parser->file_data) {
1045
0
        parser->error = CURLE_OUT_OF_MEMORY;
1046
0
        goto fail;
1047
0
      }
1048
0
      parser->item_offset = 0;
1049
0
      parser->item_length = 0;
1050
0
      curlx_dyn_init(&parser->file_data->buf, MAX_FTPLIST_BUFFER);
1051
0
    }
1052
1053
0
    infop = parser->file_data;
1054
1055
0
    if(curlx_dyn_addn(&infop->buf, &c, 1)) {
1056
0
      parser->error = CURLE_OUT_OF_MEMORY;
1057
0
      goto fail;
1058
0
    }
1059
1060
0
    switch(parser->os_type) {
1061
0
    case OS_TYPE_UNIX:
1062
0
      result = parse_unix(data, parser, infop, c);
1063
0
      break;
1064
0
    case OS_TYPE_WIN_NT:
1065
0
      result = parse_winnt(data, parser, infop, c);
1066
0
      break;
1067
0
    default:
1068
0
      retsize = bufflen + 1;
1069
0
      goto fail;
1070
0
    }
1071
0
    if(result) {
1072
0
      parser->error = result;
1073
0
      goto fail;
1074
0
    }
1075
1076
0
    i++;
1077
0
  }
1078
0
  return retsize;
1079
1080
0
fail:
1081
1082
  /* Clean up any allocated memory. */
1083
0
  if(parser->file_data) {
1084
0
    Curl_fileinfo_cleanup(parser->file_data);
1085
0
    parser->file_data = NULL;
1086
0
  }
1087
1088
0
  return retsize;
1089
0
}
1090
1091
#endif /* !CURL_DISABLE_FTP */