OLD | NEW |
1 // Copyright (c) 2009 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2009 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 // Version: MPL 1.1/GPL 2.0/LGPL 2.1 | 5 // Version: MPL 1.1/GPL 2.0/LGPL 2.1 |
6 // | 6 // |
7 // The contents of this file are subject to the Mozilla Public License Version | 7 // The contents of this file are subject to the Mozilla Public License Version |
8 // 1.1 (the "License"); you may not use this file except in compliance with | 8 // 1.1 (the "License"); you may not use this file except in compliance with |
9 // the License. You may obtain a copy of the License at | 9 // the License. You may obtain a copy of the License at |
10 // http://www.mozilla.org/MPL/ | 10 // http://www.mozilla.org/MPL/ |
(...skipping 23 matching lines...) Loading... |
34 // decision by deleting the provisions above and replace them with the notice | 34 // decision by deleting the provisions above and replace them with the notice |
35 // and other provisions required by the GPL or the LGPL. If you do not delete | 35 // and other provisions required by the GPL or the LGPL. If you do not delete |
36 // the provisions above, a recipient may use your version of this file under | 36 // the provisions above, a recipient may use your version of this file under |
37 // the terms of any one of the MPL, the GPL or the LGPL. | 37 // the terms of any one of the MPL, the GPL or the LGPL. |
38 | 38 |
39 // Derived from: | 39 // Derived from: |
40 // mozilla/netwerk/streamconv/converters/ParseFTPList.cpp revision 1.10 | 40 // mozilla/netwerk/streamconv/converters/ParseFTPList.cpp revision 1.10 |
41 | 41 |
42 #include "net/ftp/ftp_directory_parser.h" | 42 #include "net/ftp/ftp_directory_parser.h" |
43 | 43 |
44 #include <stdlib.h> | |
45 #include <string.h> | |
46 #include <ctype.h> | |
47 | |
48 #include "build/build_config.h" | |
49 #include "base/basictypes.h" | 44 #include "base/basictypes.h" |
50 #include "base/string_util.h" | 45 #include "base/string_util.h" |
51 | 46 |
| 47 using base::Time; |
| 48 |
52 // #undef anything you don't want to support | 49 // #undef anything you don't want to support |
53 #define SUPPORT_LSL // /bin/ls -l and dozens of variations therof | 50 #define SUPPORT_LSL // /bin/ls -l and dozens of variations therof |
54 #define SUPPORT_DLS // /bin/dls format (very, Very, VERY rare) | 51 #define SUPPORT_DLS // /bin/dls format (very, Very, VERY rare) |
55 #define SUPPORT_EPLF // Extraordinarily Pathetic List Format | 52 #define SUPPORT_EPLF // Extraordinarily Pathetic List Format |
56 #define SUPPORT_DOS // WinNT server in 'site dirstyle' dos | 53 #define SUPPORT_DOS // WinNT server in 'site dirstyle' dos |
57 #define SUPPORT_VMS // VMS (all: MultiNet, UCX, CMU-IP) | 54 #define SUPPORT_VMS // VMS (all: MultiNet, UCX, CMU-IP) |
58 #define SUPPORT_CMS // IBM VM/CMS,VM/ESA (z/VM and LISTING forms) | 55 #define SUPPORT_CMS // IBM VM/CMS,VM/ESA (z/VM and LISTING forms) |
59 #define SUPPORT_OS2 // IBM TCP/IP for OS/2 - FTP Server | 56 #define SUPPORT_OS2 // IBM TCP/IP for OS/2 - FTP Server |
60 #define SUPPORT_W16 // win16 hosts: SuperTCP or NetManage Chameleon | 57 #define SUPPORT_W16 // win16 hosts: SuperTCP or NetManage Chameleon |
61 | 58 |
62 // Implement the Unix gmtime_r() function for Windows. | |
63 #if defined(OS_WIN) | |
64 static struct tm *gmtime_r(const time_t *timer, struct tm *result) { | |
65 errno_t error = gmtime_s(result, timer); | |
66 if (error) { | |
67 errno = error; | |
68 return NULL; | |
69 } | |
70 return result; | |
71 } | |
72 #endif | |
73 | |
74 namespace net { | 59 namespace net { |
75 | 60 |
76 LineType ParseFTPLine(const char *line, | 61 LineType ParseFTPLine(const char *line, |
77 struct ListState *state, | 62 struct ListState *state, |
78 struct ListResult *result) { | 63 struct ListResult *result) { |
79 unsigned int carry_buf_len; | 64 unsigned int carry_buf_len; |
80 unsigned int linelen, pos; | 65 unsigned int linelen, pos; |
81 const char *p; | 66 const char *p; |
82 | 67 |
83 if (!line || !state || !result) | 68 if (!line || !state || !result) |
(...skipping 77 matching lines...) Loading... |
161 p = &line[pos++]; | 146 p = &line[pos++]; |
162 if (*p == '/') | 147 if (*p == '/') |
163 result->fe_type = FTP_TYPE_DIRECTORY; // Its a dir. | 148 result->fe_type = FTP_TYPE_DIRECTORY; // Its a dir. |
164 else if (*p == 'r') | 149 else if (*p == 'r') |
165 result->fe_type = FTP_TYPE_FILE; | 150 result->fe_type = FTP_TYPE_FILE; |
166 else if (*p == 'm') { | 151 else if (*p == 'm') { |
167 if (isdigit(line[pos])) { | 152 if (isdigit(line[pos])) { |
168 while (pos < linelen && isdigit(line[pos])) | 153 while (pos < linelen && isdigit(line[pos])) |
169 pos++; | 154 pos++; |
170 if (pos < linelen && line[pos] == ',') { | 155 if (pos < linelen && line[pos] == ',') { |
171 uint64 seconds = 0; | 156 uint64 seconds = StringToInt64(p + 1); |
172 seconds = StringToInt64(p + 1); | 157 Time t = Time::FromTimeT(seconds); |
173 time_t tim = static_cast<time_t>(seconds); | 158 t.LocalExplode(&(result->fe_time)); |
174 gmtime_r(&tim, &result->fe_time); | |
175 } | 159 } |
176 } | 160 } |
177 } else if (*p == 's') { | 161 } else if (*p == 's') { |
178 if (isdigit(line[pos])) { | 162 if (isdigit(line[pos])) { |
179 while (pos < linelen && isdigit(line[pos])) | 163 while (pos < linelen && isdigit(line[pos])) |
180 pos++; | 164 pos++; |
181 if (pos < linelen && line[pos] == ',' && | 165 if (pos < linelen && line[pos] == ',' && |
182 ((&line[pos]) - (p+1)) < | 166 ((&line[pos]) - (p+1)) < |
183 static_cast<int>(sizeof(result->fe_size) - 1)) { | 167 static_cast<int>(sizeof(result->fe_size) - 1)) { |
184 memcpy(result->fe_size, p + 1, | 168 memcpy(result->fe_size, p + 1, |
(...skipping 269 matching lines...) Loading... |
454 } // if (result->fe_type != FTP_TYPE_DIRECTORY) | 438 } // if (result->fe_type != FTP_TYPE_DIRECTORY) |
455 | 439 |
456 p = tokens[2] + 2; | 440 p = tokens[2] + 2; |
457 if (*p == '-') | 441 if (*p == '-') |
458 p++; | 442 p++; |
459 tbuf[0] = p[0]; | 443 tbuf[0] = p[0]; |
460 tbuf[1] = tolower(p[1]); | 444 tbuf[1] = tolower(p[1]); |
461 tbuf[2] = tolower(p[2]); | 445 tbuf[2] = tolower(p[2]); |
462 month_num = 0; | 446 month_num = 0; |
463 for (pos = 0; pos < (12*3); pos += 3) { | 447 for (pos = 0; pos < (12*3); pos += 3) { |
464 if (tbuf[0] == month_names[pos+0] && | 448 if (tbuf[0] == month_names[pos + 0] && |
465 tbuf[1] == month_names[pos + 1] && | 449 tbuf[1] == month_names[pos + 1] && |
466 tbuf[2] == month_names[pos + 2]) | 450 tbuf[2] == month_names[pos + 2]) |
467 break; | 451 break; |
468 month_num++; | 452 month_num++; |
469 } | 453 } |
470 if (month_num >= 12) | 454 if (month_num >= 12) |
471 month_num = 0; | 455 month_num = 0; |
472 result->fe_time.tm_mon = month_num; | 456 result->fe_time.month = month_num + 1; |
473 result->fe_time.tm_mday = StringToInt(tokens[2]); | 457 result->fe_time.day_of_month = StringToInt(tokens[2]); |
474 result->fe_time.tm_year = StringToInt(p + 4); | 458 result->fe_time.year = StringToInt(p + 4); |
475 // NSPR wants year as XXXX | |
476 | 459 |
477 p = tokens[3] + 2; | 460 p = tokens[3] + 2; |
478 if (*p == ':') | 461 if (*p == ':') |
479 p++; | 462 p++; |
480 if (p[2] == ':') | 463 if (p[2] == ':') |
481 result->fe_time.tm_sec = StringToInt(p + 3); | 464 result->fe_time.second = StringToInt(p + 3); |
482 result->fe_time.tm_hour = StringToInt(tokens[3]); | 465 result->fe_time.hour = StringToInt(tokens[3]); |
483 result->fe_time.tm_min = StringToInt(p); | 466 result->fe_time.minute = StringToInt(p); |
484 return result->fe_type; | 467 return result->fe_type; |
485 } // if (isdigit(*tokens[1])) | 468 } // if (isdigit(*tokens[1])) |
486 return FTP_TYPE_JUNK; // junk | 469 return FTP_TYPE_JUNK; // junk |
487 } // if (lstyle == 'V') | 470 } // if (lstyle == 'V') |
488 } // if (!lstyle && (!state->lstyle || state->lstyle == 'V')) | 471 } // if (!lstyle && (!state->lstyle || state->lstyle == 'V')) |
489 #endif // SUPPORT_VMS | 472 #endif // SUPPORT_VMS |
490 | 473 |
491 #if defined(SUPPORT_CMS) | 474 #if defined(SUPPORT_CMS) |
492 // Virtual Machine/Conversational Monitor System (IBM Mainframe) | 475 // Virtual Machine/Conversational Monitor System (IBM Mainframe) |
493 if (!lstyle && (!state->lstyle || state->lstyle == 'C')) { // VM/CMS | 476 if (!lstyle && (!state->lstyle || state->lstyle == 'C')) { // VM/CMS |
(...skipping 86 matching lines...) Loading... |
580 lstyle = 0; | 563 lstyle = 0; |
581 } | 564 } |
582 } // if (lstyle && !state->lstyle) | 565 } // if (lstyle && !state->lstyle) |
583 | 566 |
584 if (lstyle == 'C') { | 567 if (lstyle == 'C') { |
585 state->parsed_one = 1; | 568 state->parsed_one = 1; |
586 state->lstyle = lstyle; | 569 state->lstyle = lstyle; |
587 | 570 |
588 p = tokens[tokmarker+4]; | 571 p = tokens[tokmarker+4]; |
589 if (toklen[tokmarker+4] == 10) { // newstyle: YYYY-MM-DD format | 572 if (toklen[tokmarker+4] == 10) { // newstyle: YYYY-MM-DD format |
590 result->fe_time.tm_year = StringToInt(p + 0) - 1900; | 573 result->fe_time.year = StringToInt(p + 0); |
591 result->fe_time.tm_mon = StringToInt(p + 5) - 1; | 574 result->fe_time.month = StringToInt(p + 5); |
592 result->fe_time.tm_mday = StringToInt(p + 8); | 575 result->fe_time.day_of_month = StringToInt(p + 8); |
593 } else { // oldstyle: [M]M/DD/YY format | 576 } else { // oldstyle: [M]M/DD/YY format |
594 pos = toklen[tokmarker + 4]; | 577 pos = toklen[tokmarker + 4]; |
595 result->fe_time.tm_mon = StringToInt(p) - 1; | 578 result->fe_time.month = StringToInt(p); |
596 result->fe_time.tm_mday = StringToInt((p + pos)-5); | 579 result->fe_time.day_of_month = StringToInt((p + pos)-5); |
597 result->fe_time.tm_year = StringToInt((p + pos)-2); | 580 result->fe_time.year = StringToInt((p + pos)-2); |
598 if (result->fe_time.tm_year < 70) | 581 if (result->fe_time.year < 70) |
599 result->fe_time.tm_year += 100; | 582 result->fe_time.year += 100; |
| 583 result->fe_time.year += 1900; |
600 } | 584 } |
601 p = tokens[tokmarker + 5]; | 585 p = tokens[tokmarker + 5]; |
602 pos = toklen[tokmarker + 5]; | 586 pos = toklen[tokmarker + 5]; |
603 result->fe_time.tm_hour = StringToInt(p); | 587 result->fe_time.hour = StringToInt(p); |
604 result->fe_time.tm_min = StringToInt((p + pos) - 5); | 588 result->fe_time.minute = StringToInt((p + pos) - 5); |
605 result->fe_time.tm_sec = StringToInt((p + pos) - 2); | 589 result->fe_time.second = StringToInt((p + pos) - 2); |
606 | 590 |
607 result->fe_cinfs = 1; | 591 result->fe_cinfs = 1; |
608 result->fe_fname = tokens[0]; | 592 result->fe_fname = tokens[0]; |
609 result->fe_fnlen = toklen[0]; | 593 result->fe_fnlen = toklen[0]; |
610 result->fe_type = FTP_TYPE_FILE; | 594 result->fe_type = FTP_TYPE_FILE; |
611 | 595 |
612 p = tokens[tokmarker]; | 596 p = tokens[tokmarker]; |
613 if (toklen[tokmarker] == 3 && *p == 'D' && p[1] == 'I' && p[2] == 'R') | 597 if (toklen[tokmarker] == 3 && *p == 'D' && p[1] == 'I' && p[2] == 'R') |
614 result->fe_type = FTP_TYPE_DIRECTORY; | 598 result->fe_type = FTP_TYPE_DIRECTORY; |
615 | 599 |
(...skipping 82 matching lines...) Loading... |
698 result->fe_lname = p + 4; | 682 result->fe_lname = p + 4; |
699 result->fe_lnlen = &(line[linelen_sans_wsp]) | 683 result->fe_lnlen = &(line[linelen_sans_wsp]) |
700 - result->fe_lname; | 684 - result->fe_lname; |
701 break; | 685 break; |
702 } | 686 } |
703 p++; | 687 p++; |
704 } | 688 } |
705 } | 689 } |
706 } | 690 } |
707 | 691 |
708 result->fe_time.tm_mon = StringToInt(tokens[0]+0); | 692 result->fe_time.month = StringToInt(tokens[0] + 0); |
709 if (result->fe_time.tm_mon != 0) { | 693 if (result->fe_time.month != 0) { |
710 result->fe_time.tm_mon--; | 694 result->fe_time.day_of_month = StringToInt(tokens[0] + 3); |
711 result->fe_time.tm_mday = StringToInt(tokens[0]+3); | 695 result->fe_time.year = StringToInt(tokens[0] + 6); |
712 result->fe_time.tm_year = StringToInt(tokens[0]+6); | 696 // if year has only two digits then assume that |
713 if (result->fe_time.tm_year < 80) | 697 // 00-79 is 2000-2079 |
714 result->fe_time.tm_year += 100; | 698 // 80-99 is 1980-1999 |
| 699 if (result->fe_time.year < 80) |
| 700 result->fe_time.year += 2000; |
| 701 else if (result->fe_time.year < 100) |
| 702 result->fe_time.year += 1900; |
715 } | 703 } |
716 | 704 |
717 result->fe_time.tm_hour = StringToInt(tokens[1]+0); | 705 result->fe_time.hour = StringToInt(tokens[1]+0); |
718 result->fe_time.tm_min = StringToInt(tokens[1]+3); | 706 result->fe_time.minute = StringToInt(tokens[1]+3); |
719 if ((tokens[1][5]) == 'P' && result->fe_time.tm_hour < 12) | 707 if ((tokens[1][5]) == 'P' && result->fe_time.hour < 12) |
720 result->fe_time.tm_hour += 12; | 708 result->fe_time.hour += 12; |
721 | 709 |
722 // the caller should do this (if dropping "." and ".." is desired) | 710 // the caller should do this (if dropping "." and ".." is desired) |
723 // if (result->fe_type == FTP_TYPE_DIRECTORY && result->fe_fname[0] | 711 // if (result->fe_type == FTP_TYPE_DIRECTORY && result->fe_fname[0] |
724 // == '.' && | 712 // == '.' && |
725 // (result->fe_fnlen == 1 || (result->fe_fnlen == 2 && | 713 // (result->fe_fnlen == 1 || (result->fe_fnlen == 2 && |
726 // result->fe_fname[1] == '.'))) | 714 // result->fe_fname[1] == '.'))) |
727 // return FTP_TYPE_JUNK; | 715 // return FTP_TYPE_JUNK; |
728 | 716 |
729 return result->fe_type; | 717 return result->fe_type; |
730 } // if (lstyle == 'W' && (!state->lstyle || state->lstyle == lstyle)) | 718 } // if (lstyle == 'W' && (!state->lstyle || state->lstyle == lstyle)) |
(...skipping 67 matching lines...) Loading... |
798 } | 786 } |
799 | 787 |
800 if (result->fe_type != FTP_TYPE_DIRECTORY) { | 788 if (result->fe_type != FTP_TYPE_DIRECTORY) { |
801 pos = toklen[0]; | 789 pos = toklen[0]; |
802 if (pos > (sizeof(result->fe_size)-1)) | 790 if (pos > (sizeof(result->fe_size)-1)) |
803 pos = (sizeof(result->fe_size)-1); | 791 pos = (sizeof(result->fe_size)-1); |
804 memcpy(result->fe_size, tokens[0], pos); | 792 memcpy(result->fe_size, tokens[0], pos); |
805 result->fe_size[pos] = '\0'; | 793 result->fe_size[pos] = '\0'; |
806 } | 794 } |
807 | 795 |
808 result->fe_time.tm_mon = StringToInt(&p[35-18]) - 1; | 796 result->fe_time.month = StringToInt(&p[35-18]); |
809 result->fe_time.tm_mday = StringToInt(&p[38-18]); | 797 result->fe_time.day_of_month = StringToInt(&p[38 - 18]); |
810 result->fe_time.tm_year = StringToInt(&p[41-18]); | 798 result->fe_time.year = StringToInt(&p[41 - 18]); |
811 if (result->fe_time.tm_year < 80) | 799 if (result->fe_time.year < 80) |
812 result->fe_time.tm_year += 100; | 800 result->fe_time.year += 100; |
813 result->fe_time.tm_hour = StringToInt(&p[46-18]); | 801 result->fe_time.year += 1900; |
814 result->fe_time.tm_min = StringToInt(&p[49-18]); | 802 result->fe_time.hour = StringToInt(&p[46 - 18]); |
| 803 result->fe_time.minute = StringToInt(&p[49 - 18]); |
815 | 804 |
816 // The caller should do this (if dropping "." and ".." is desired) | 805 // The caller should do this (if dropping "." and ".." is desired) |
817 // if (result->fe_type == FTP_TYPE_DIRECTORY && | 806 // if (result->fe_type == FTP_TYPE_DIRECTORY && |
818 // result->fe_fname[0] == '.' && (result->fe_fnlen == 1 || | 807 // result->fe_fname[0] == '.' && (result->fe_fnlen == 1 || |
819 // (result->fe_fnlen == 2 && result->fe_fname[1] == '.'))) | 808 // (result->fe_fnlen == 2 && result->fe_fname[1] == '.'))) |
820 // return FTP_TYPE_JUNK; | 809 // return FTP_TYPE_JUNK; |
821 | 810 |
822 return result->fe_type; | 811 return result->fe_type; |
823 } // if (lstyle == 'O') | 812 } // if (lstyle == 'O') |
824 } // if (!lstyle && (!state->lstyle || state->lstyle == 'O')) | 813 } // if (!lstyle && (!state->lstyle || state->lstyle == 'O')) |
(...skipping 136 matching lines...) Loading... |
961 result->fe_type = FTP_TYPE_FILE; // (hopefully a regular file) | 950 result->fe_type = FTP_TYPE_FILE; // (hopefully a regular file) |
962 | 951 |
963 if (result->fe_type != FTP_TYPE_DIRECTORY) { | 952 if (result->fe_type != FTP_TYPE_DIRECTORY) { |
964 pos = toklen[tokmarker]; | 953 pos = toklen[tokmarker]; |
965 if (pos > (sizeof(result->fe_size)-1)) | 954 if (pos > (sizeof(result->fe_size)-1)) |
966 pos = (sizeof(result->fe_size)-1); | 955 pos = (sizeof(result->fe_size)-1); |
967 memcpy(result->fe_size, tokens[tokmarker], pos); | 956 memcpy(result->fe_size, tokens[tokmarker], pos); |
968 result->fe_size[pos] = '\0'; | 957 result->fe_size[pos] = '\0'; |
969 } | 958 } |
970 | 959 |
971 result->fe_time.tm_mon = month_num; | 960 result->fe_time.month = month_num + 1; |
972 result->fe_time.tm_mday = StringToInt(tokens[tokmarker+2]); | 961 result->fe_time.day_of_month = StringToInt(tokens[tokmarker+2]); |
973 if (result->fe_time.tm_mday == 0) | 962 if (result->fe_time.day_of_month == 0) |
974 result->fe_time.tm_mday++; | 963 result->fe_time.day_of_month++; |
975 | 964 |
976 p = tokens[tokmarker+3]; | 965 p = tokens[tokmarker+3]; |
977 pos = (unsigned int)StringToInt(p); | 966 pos = (unsigned int)StringToInt(p); |
978 if (p[1] == ':') // one digit hour | 967 if (p[1] == ':') // one digit hour |
979 p--; | 968 p--; |
980 if (p[2] != ':') { // year | 969 if (p[2] != ':') { // year |
981 result->fe_time.tm_year = pos; | 970 result->fe_time.year = pos; |
982 } else { | 971 } else { |
983 result->fe_time.tm_hour = pos; | 972 result->fe_time.hour = pos; |
984 result->fe_time.tm_min = StringToInt(p+3); | 973 result->fe_time.minute = StringToInt(p+3); |
985 if (p[5] == ':') | 974 if (p[5] == ':') |
986 result->fe_time.tm_sec = StringToInt(p+6); | 975 result->fe_time.second = StringToInt(p+6); |
987 | 976 if (!state->now_tm_valid) { |
988 if (!state->now_time) { | 977 Time t = Time::Now(); |
989 time_t now = time(NULL); | 978 t.LocalExplode(&(state->now_tm)); |
990 state->now_time = now * 1000000; | 979 state->now_tm_valid = 1; |
991 gmtime_r(&now, &state->now_tm); | |
992 } | 980 } |
993 | 981 result->fe_time.year = state->now_tm.year; |
994 result->fe_time.tm_year = state->now_tm.tm_year; | 982 if (((state->now_tm.month << 5) + state->now_tm.day_of_month) < |
995 if ( (( state->now_tm.tm_mon << 5) + state->now_tm.tm_mday) < | 983 ((result->fe_time.month << 5) + result->fe_time.day_of_month)) |
996 ((result->fe_time.tm_mon << 5) + result->fe_time.tm_mday) ) | 984 result->fe_time.year--; |
997 result->fe_time.tm_year--; | |
998 } // time/year | 985 } // time/year |
999 | 986 |
1000 result->fe_fname = tokens[tokmarker+4]; | 987 result->fe_fname = tokens[tokmarker+4]; |
1001 result->fe_fnlen = (&(line[linelen_sans_wsp])) | 988 result->fe_fnlen = (&(line[linelen_sans_wsp])) |
1002 - (result->fe_fname); | 989 - (result->fe_fname); |
1003 | 990 |
1004 if (result->fe_type == FTP_TYPE_SYMLINK && result->fe_fnlen > 4) { | 991 if (result->fe_type == FTP_TYPE_SYMLINK && result->fe_fnlen > 4) { |
1005 p = result->fe_fname + 1; | 992 p = result->fe_fname + 1; |
1006 for (pos = 1; pos < (result->fe_fnlen - 4); pos++) { | 993 for (pos = 1; pos < (result->fe_fnlen - 4); pos++) { |
1007 if (*p == ' ' && p[1] == '-' && p[2] == '>' && p[3] == ' ') { | 994 if (*p == ' ' && p[1] == '-' && p[2] == '>' && p[3] == ' ') { |
(...skipping 115 matching lines...) Loading... |
1123 pos = sizeof(result->fe_size) - 1; | 1110 pos = sizeof(result->fe_size) - 1; |
1124 memcpy(result->fe_size, p, pos); | 1111 memcpy(result->fe_size, p, pos); |
1125 result->fe_size[pos] = '\0'; | 1112 result->fe_size[pos] = '\0'; |
1126 } | 1113 } |
1127 | 1114 |
1128 p = tokens[2]; | 1115 p = tokens[2]; |
1129 if (toklen[2] == 3) { // Chameleon | 1116 if (toklen[2] == 3) { // Chameleon |
1130 tbuf[0] = toupper(p[0]); | 1117 tbuf[0] = toupper(p[0]); |
1131 tbuf[1] = tolower(p[1]); | 1118 tbuf[1] = tolower(p[1]); |
1132 tbuf[2] = tolower(p[2]); | 1119 tbuf[2] = tolower(p[2]); |
1133 for (pos = 0; pos < (12*3); pos+=3) { | 1120 for (pos = 0; pos < (12 * 3); pos += 3) { |
1134 if (tbuf[0] == month_names[pos+0] && | 1121 if (tbuf[0] == month_names[pos + 0] && |
1135 tbuf[1] == month_names[pos+1] && | 1122 tbuf[1] == month_names[pos + 1] && |
1136 tbuf[2] == month_names[pos+2]) { | 1123 tbuf[2] == month_names[pos + 2]) { |
1137 result->fe_time.tm_mon = pos/3; | 1124 result->fe_time.month = pos / 3 + 1; |
1138 result->fe_time.tm_mday = StringToInt(tokens[3]); | 1125 result->fe_time.day_of_month = StringToInt(tokens[3]); |
1139 result->fe_time.tm_year = StringToInt(tokens[4]) - 1900; | 1126 result->fe_time.year = StringToInt(tokens[4]); |
1140 break; | 1127 break; |
1141 } | 1128 } |
1142 } | 1129 } |
1143 pos = 5; // Chameleon toknum of date field | 1130 pos = 5; // Chameleon toknum of date field |
1144 } else { | 1131 } else { |
1145 result->fe_time.tm_mon = StringToInt(p+0)-1; | 1132 result->fe_time.month = StringToInt(p + 0); |
1146 result->fe_time.tm_mday = StringToInt(p+3); | 1133 result->fe_time.day_of_month = StringToInt(p + 3); |
1147 result->fe_time.tm_year = StringToInt(p+6); | 1134 result->fe_time.year = StringToInt(p+6); |
1148 if (result->fe_time.tm_year < 80) // SuperTCP | 1135 if (result->fe_time.year < 80) // SuperTCP |
1149 result->fe_time.tm_year += 100; | 1136 result->fe_time.year += 100; |
1150 | 1137 result->fe_time.year += 1900; |
1151 pos = 3; // SuperTCP toknum of date field | 1138 pos = 3; // SuperTCP toknum of date field |
1152 } | 1139 } |
1153 | 1140 |
1154 result->fe_time.tm_hour = StringToInt(tokens[pos]); | 1141 result->fe_time.hour = StringToInt(tokens[pos]); |
1155 result->fe_time.tm_min = StringToInt(&(tokens[pos][toklen[pos]-2])); | 1142 result->fe_time.minute = StringToInt(&(tokens[pos][toklen[pos]-2])); |
1156 | 1143 |
1157 // the caller should do this (if dropping "." and ".." is desired) | 1144 // the caller should do this (if dropping "." and ".." is desired) |
1158 // if (result->fe_type == FTP_TYPE_DIRECTORY && | 1145 // if (result->fe_type == FTP_TYPE_DIRECTORY && |
1159 // result->fe_fname[0] == '.' && (result->fe_fnlen == 1 || | 1146 // result->fe_fname[0] == '.' && (result->fe_fnlen == 1 || |
1160 // (result->fe_fnlen == 2 && result->fe_fname[1] == '.'))) | 1147 // (result->fe_fnlen == 2 && result->fe_fname[1] == '.'))) |
1161 // return FTP_TYPE_JUNK; | 1148 // return FTP_TYPE_JUNK; |
1162 | 1149 |
1163 return result->fe_type; | 1150 return result->fe_type; |
1164 } // (lstyle == 'w') | 1151 } // (lstyle == 'w') |
1165 } // if (!lstyle && (!state->lstyle || state->lstyle == 'w')) | 1152 } // if (!lstyle && (!state->lstyle || state->lstyle == 'w')) |
(...skipping 159 matching lines...) Loading... |
1325 pos--; | 1312 pos--; |
1326 } | 1313 } |
1327 p = tokens[month_num]; | 1314 p = tokens[month_num]; |
1328 if (isdigit(*tokens[pos]) | 1315 if (isdigit(*tokens[pos]) |
1329 && (toklen[pos] == 1 || | 1316 && (toklen[pos] == 1 || |
1330 (toklen[pos] == 2 && isdigit(tokens[pos][1]))) | 1317 (toklen[pos] == 2 && isdigit(tokens[pos][1]))) |
1331 && toklen[month_num] == 3 | 1318 && toklen[month_num] == 3 |
1332 && isalpha(*p) && isalpha(p[1]) && isalpha(p[2])) { | 1319 && isalpha(*p) && isalpha(p[1]) && isalpha(p[2])) { |
1333 pos = StringToInt(tokens[pos]); | 1320 pos = StringToInt(tokens[pos]); |
1334 if (pos > 0 && pos <= 31) { | 1321 if (pos > 0 && pos <= 31) { |
1335 result->fe_time.tm_mday = pos; | 1322 result->fe_time.day_of_month = pos; |
1336 month_num = 1; | 1323 month_num = 1; |
1337 for (pos = 0; pos < (12*3); pos += 3) { | 1324 for (pos = 0; pos < (12*3); pos += 3) { |
1338 if (p[0] == month_names[pos + 0] && | 1325 if (p[0] == month_names[pos + 0] && |
1339 p[1] == month_names[pos + 1] && | 1326 p[1] == month_names[pos + 1] && |
1340 p[2] == month_names[pos + 2]) | 1327 p[2] == month_names[pos + 2]) |
1341 break; | 1328 break; |
1342 month_num++; | 1329 month_num++; |
1343 } | 1330 } |
1344 if (month_num > 12) | 1331 if (month_num > 12) |
1345 result->fe_time.tm_mday = 0; | 1332 result->fe_time.day_of_month = 0; |
1346 else | 1333 else |
1347 result->fe_time.tm_mon = month_num - 1; | 1334 result->fe_time.month = month_num; |
1348 } | 1335 } |
1349 } | 1336 } |
1350 if (result->fe_time.tm_mday) { | 1337 if (result->fe_time.day_of_month) { |
1351 tokmarker += 3; // skip mday/mon/yrtime (to find " -> ") | 1338 tokmarker += 3; // skip mday/mon/yrtime (to find " -> ") |
1352 p = tokens[tokmarker]; | 1339 p = tokens[tokmarker]; |
1353 | 1340 |
1354 pos = StringToInt(p); | 1341 pos = StringToInt(p); |
1355 if (pos > 24) { | 1342 if (pos > 24) { |
1356 result->fe_time.tm_year = pos - 1900; | 1343 result->fe_time.year = pos; |
1357 } else { | 1344 } else { |
1358 if (p[1] == ':') | 1345 if (p[1] == ':') |
1359 p--; | 1346 p--; |
1360 result->fe_time.tm_hour = pos; | 1347 result->fe_time.hour = pos; |
1361 result->fe_time.tm_min = StringToInt(p + 3); | 1348 result->fe_time.minute = StringToInt(p + 3); |
1362 if (!state->now_time) { | 1349 if (!state->now_tm_valid) { |
1363 time_t now = time(NULL); | 1350 Time t = Time::Now(); |
1364 state->now_time = now * 1000000; | 1351 t.LocalExplode(&(state->now_tm)); |
1365 gmtime_r(&now, &state->now_tm); | 1352 state->now_tm_valid = 1; |
1366 } | 1353 } |
1367 result->fe_time.tm_year = state->now_tm.tm_year; | 1354 result->fe_time.year = state->now_tm.year; |
1368 if ((( state->now_tm.tm_mon << 4) + | 1355 if (((state->now_tm.month << 4) + |
1369 state->now_tm.tm_mday) < | 1356 state->now_tm.day_of_month) < |
1370 ((result->fe_time.tm_mon << 4) + | 1357 ((result->fe_time.month << 4) + |
1371 result->fe_time.tm_mday)) | 1358 result->fe_time.day_of_month)) |
1372 result->fe_time.tm_year--; | 1359 result->fe_time.year--; |
1373 } // got year or time | 1360 } // got year or time |
1374 } // got month/mday | 1361 } // got month/mday |
1375 } // may have year or time | 1362 } // may have year or time |
1376 } // enough remaining to possibly have date/time | 1363 } // enough remaining to possibly have date/time |
1377 | 1364 |
1378 if (numtoks > (tokmarker+2)) { | 1365 if (numtoks > (tokmarker+2)) { |
1379 pos = tokmarker+1; | 1366 pos = tokmarker+1; |
1380 p = tokens[pos]; | 1367 p = tokens[pos]; |
1381 if (toklen[pos] == 2 && *p == '-' && p[1] == '>') { | 1368 if (toklen[pos] == 2 && *p == '-' && p[1] == '>') { |
1382 p = &(tokens[numtoks - 1][toklen[numtoks - 1]]); | 1369 p = &(tokens[numtoks - 1][toklen[numtoks - 1]]); |
(...skipping 16 matching lines...) Loading... |
1399 } // if (!lstyle && (!state->lstyle || state->lstyle == 'D')) | 1386 } // if (!lstyle && (!state->lstyle || state->lstyle == 'D')) |
1400 #endif | 1387 #endif |
1401 } // if (linelen > 0) | 1388 } // if (linelen > 0) |
1402 | 1389 |
1403 if (state->parsed_one || state->lstyle) // junk if we fail to parse | 1390 if (state->parsed_one || state->lstyle) // junk if we fail to parse |
1404 return FTP_TYPE_JUNK; // this time but had previously parsed successfully | 1391 return FTP_TYPE_JUNK; // this time but had previously parsed successfully |
1405 return FTP_TYPE_COMMENT; // its part of a comment or error message | 1392 return FTP_TYPE_COMMENT; // its part of a comment or error message |
1406 } | 1393 } |
1407 | 1394 |
1408 } // namespace net | 1395 } // namespace net |
OLD | NEW |