/* Copyright (c) 2011, 2012, Simon Howard Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. */ #include #include #include #include #include #include #include "lha_reader.h" #include "list.h" #include "safe.h" typedef struct { unsigned int num_files; unsigned int compressed_length; unsigned int length; unsigned int timestamp; } FileStatistics; typedef struct { char *name; unsigned int width; void (*handler)(LHAFileHeader *header); void (*footer)(FileStatistics *stats); } ListColumn; // Display OS type: static char *os_type_to_string(uint8_t os_type) { switch (os_type) { case LHA_OS_TYPE_MSDOS: return "[MS-DOS]"; case LHA_OS_TYPE_WIN95: return "[Win9x]"; case LHA_OS_TYPE_WINNT: return "[WinNT]"; case LHA_OS_TYPE_UNIX: return "[Unix]"; case LHA_OS_TYPE_OS2: return "[OS/2]"; case LHA_OS_TYPE_CPM: return "[CP/M]"; case LHA_OS_TYPE_MACOS: return "[Mac OS]"; case LHA_OS_TYPE_JAVA: return "[Java]"; case LHA_OS_TYPE_FLEX: return "[FLEX]"; case LHA_OS_TYPE_RUNSER: return "[Runser]"; case LHA_OS_TYPE_TOWNSOS: return "[TownsOS]"; case LHA_OS_TYPE_OS9: return "[OS-9]"; case LHA_OS_TYPE_OS9_68K: return "[OS-9/68K]"; case LHA_OS_TYPE_OS386: return "[OS-386]"; case LHA_OS_TYPE_HUMAN68K: return "[Human68K]"; case LHA_OS_TYPE_ATARI: return "[Atari]"; case LHA_OS_TYPE_AMIGA: return "[Amiga]"; case LHA_OS_TYPE_UNKNOWN: return "[generic]"; default: return "[unknown]"; } } // Print Unix file permissions. static void unix_permissions_print(LHAFileHeader *header) { const char *perms = "rwxrwxrwx"; unsigned int i; if (strcmp(header->compress_method, LHA_COMPRESS_TYPE_DIR) != 0) { printf("-"); } else if (header->symlink_target != NULL) { printf("l"); } else { printf("d"); } for (i = 0; i < 9; ++i) { if (header->unix_perms & (1U << (8 - i))) { printf("%c", perms[i]); } else { printf("-"); } } } // Print OS-9 file permissions. static void os9_permissions_print(LHAFileHeader *header) { const char *perms = "sewrewr"; unsigned int i; if (strcmp(header->compress_method, LHA_COMPRESS_TYPE_DIR) != 0) { printf("-"); } else { printf("d"); } for (i = 0; i < 7; ++i) { if (header->os9_perms & (1U << (6 - i))) { printf("%c", perms[i]); } else { printf("-"); } } printf(" "); } // File permissions static void permission_column_print(LHAFileHeader *header) { // Print permissions. If we do not have any permissions to // print, fall back to printing the OS type. if (LHA_FILE_HAVE_EXTRA(header, LHA_FILE_OS9_PERMS)) { os9_permissions_print(header); } else if (LHA_FILE_HAVE_EXTRA(header, LHA_FILE_UNIX_PERMS)) { unix_permissions_print(header); } else { printf("%-10s", os_type_to_string(header->os_type)); } } static void permission_column_footer(FileStatistics *stats) { printf(" Total "); } static ListColumn permission_column = { " PERMSSN", 10, permission_column_print, permission_column_footer }; // Unix UID/GID static void unix_uid_gid_column_print(LHAFileHeader *header) { if (LHA_FILE_HAVE_EXTRA(header, LHA_FILE_UNIX_UID_GID)) { printf("%5i/%-5i", header->unix_uid, header->unix_gid); } else { printf(" "); } } static void unix_uid_gid_column_footer(FileStatistics *stats) { // The UID/GID column has the total number of files // listed below it. if (stats->num_files == 1) { printf("%5i file ", stats->num_files); } else { printf("%5i files", stats->num_files); } } static ListColumn unix_uid_gid_column = { " UID GID", 11, unix_uid_gid_column_print, unix_uid_gid_column_footer }; // Compressed file size static void packed_column_print(LHAFileHeader *header) { printf("%7lu", (unsigned long) header->compressed_length); } static void packed_column_footer(FileStatistics *stats) { printf("%7lu", (unsigned long) stats->compressed_length); } static ListColumn packed_column = { " PACKED", 7, packed_column_print, packed_column_footer }; // Uncompressed file size static void size_column_print(LHAFileHeader *header) { printf("%7lu", (unsigned long) header->length); } static void size_column_footer(FileStatistics *stats) { printf("%7lu", (unsigned long) stats->length); } static ListColumn size_column = { " SIZE", 7, size_column_print, size_column_footer }; // Compression ratio static float compression_percent(size_t compressed, size_t uncompressed) { if (uncompressed > 0) { return ((float) compressed * 100.0f) / (float) uncompressed; } else { return 100.0f; } } static void ratio_column_print(LHAFileHeader *header) { if (!strcmp(header->compress_method, "-lhd-")) { printf("******"); } else { printf("%5.1f%%", compression_percent(header->compressed_length, header->length)); } } static void ratio_column_footer(FileStatistics *stats) { if (stats->length == 0) { printf("******"); } else { printf("%5.1f%%", compression_percent(stats->compressed_length, stats->length)); } } static ListColumn ratio_column = { " RATIO", 6, ratio_column_print, ratio_column_footer }; // Compression method and CRC checksum static void method_crc_column_print(LHAFileHeader *header) { printf("%-5s %04x", header->compress_method, header->crc); } static ListColumn method_crc_column = { "METHOD CRC", 10, method_crc_column_print }; // Get the current time. static time_t get_now_time(void) { // For test builds, allow the current time to be overridden using // an environment variable. This is because the list output can // change depending on the current date. #ifdef TEST_BUILD char *env_val; unsigned int result; env_val = getenv("TEST_NOW_TIME"); if (env_val != NULL && sscanf(env_val, "%u", &result) == 1) { return (time_t) result; } #endif return time(NULL); } // File timestamp static void output_timestamp(unsigned int timestamp) { const char *months[] = { "Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec" }; struct tm *ts; time_t tmp; if (timestamp == 0) { printf(" "); return; } tmp = (time_t) timestamp; ts = localtime(&tmp); // Print date: printf("%s %2d ", months[ts->tm_mon], ts->tm_mday); // If this is an old time (more than 6 months), print the year. // For recent timestamps, print the time. tmp = get_now_time(); if ((time_t) timestamp > tmp - 6 * 30 * 24 * 60 * 60) { printf("%02i:%02i", ts->tm_hour, ts->tm_min); } else { printf(" %04i", ts->tm_year + 1900); } } static void timestamp_column_print(LHAFileHeader *header) { output_timestamp(header->timestamp); }; static void timestamp_column_footer(FileStatistics *stats) { output_timestamp(stats->timestamp); }; static ListColumn timestamp_column = { " STAMP", 12, timestamp_column_print, timestamp_column_footer }; // Filename static void name_column_print(LHAFileHeader *header) { if (header->path != NULL) { safe_printf("%s", header->path); } if (header->filename != NULL) { safe_printf("%s", header->filename); } if (header->symlink_target != NULL) { safe_printf(" -> %s", header->symlink_target); } } static ListColumn name_column = { " NAME", 20, name_column_print }; static ListColumn short_name_column = { " NAME", 13, name_column_print }; // "Separate line" filename display. Used when the 'v' option // is added. static void whole_line_name_column_print(LHAFileHeader *header) { if (header->path != NULL) { safe_printf("%s", header->path); } if (header->filename != NULL) { safe_printf("%s", header->filename); } // For wide filename mode (-v), | is used as the symlink separator, // instead of ->. The reason is that this is how symlinks are // represented within the file headers - the Unix LHA tool only // does the parsing in normal list mode. if (header->symlink_target != NULL) { safe_printf("|%s", header->symlink_target); } printf("\n"); } static ListColumn whole_line_name_column = { "", 0, whole_line_name_column_print }; // Header level column. Used when the 'v' option is added. static void header_level_column_print(LHAFileHeader *header) { printf(" [%i]", header->header_level); } static ListColumn header_level_column = { "", 0, header_level_column_print }; // Get the last "real" column in a list of columns. This is the last // column with a width > 0. Beyond this last column it isn't necessary // to print any more whitespace. static ListColumn *last_column(ListColumn **columns) { ListColumn *last; unsigned int i; last = NULL; for (i = 0; columns[i] != NULL; ++i) { if (columns[i]->width != 0) { last = columns[i]; } } return last; } // Print the names of the column headings at the top of the file list. static void print_list_headings(ListColumn **columns) { ListColumn *last; unsigned int i, j; last = last_column(columns); for (i = 0; columns[i] != NULL; ++i) { j = (unsigned) printf("%s", columns[i]->name); if (columns[i]->width > 0 && columns[i] != last) { for (; j < columns[i]->width + 1; ++j) { printf(" "); } } } printf("\n"); } // Print separator lines shown at top and bottom of file list. static void print_list_separators(ListColumn **columns) { ListColumn *last; unsigned int i, j; last = last_column(columns); for (i = 0; columns[i] != NULL; ++i) { for (j = 0; j < columns[i]->width; ++j) { printf("-"); } if (columns[i]->width != 0 && columns[i] != last) { printf(" "); } } printf("\n"); } // Print a row in the list corresponding to a file. static void print_columns(ListColumn **columns, LHAFileHeader *header) { ListColumn *last; unsigned int i; last = last_column(columns); for (i = 0; columns[i] != NULL; ++i) { columns[i]->handler(header); if (columns[i]->width != 0 && columns[i] != last) { printf(" "); } } printf("\n"); } // Print footer information shown at end of list (overall file stats) static void print_footers(ListColumn **columns, FileStatistics *stats) { unsigned int i, j, len; unsigned int num_columns; // Work out how many columns there are to print, ignoring trailing // columns that have no footer: num_columns = 0; for (i = 0; columns[i] != NULL; ++i) { ++num_columns; } while (num_columns > 0 && columns[num_columns-1]->footer == NULL) { --num_columns; } // Print footers for each column. // Some columns do not have footers: fill in with spaces instead. for (i = 0; i < num_columns; ++i) { if (columns[i]->footer != NULL) { columns[i]->footer(stats); } else if (i + 1 < num_columns) { len = strlen(columns[i]->name); for (j = 0; j < len; ++j) { printf(" "); } } if (columns[i]->width != 0 && i + 1 < num_columns) { printf(" "); } } printf("\n"); } static unsigned int read_file_timestamp(FILE *fstream) { struct stat data; if (fstat(fileno(fstream), &data) != 0) { return (unsigned int) -1; } return (unsigned int) data.st_mtime; } // List contents of file, using the specified columns. // Different columns are provided for basic and verbose modes. static void list_file_contents(LHAFilter *filter, FILE *fstream, LHAOptions *options, ListColumn **columns) { FileStatistics stats; if (options->quiet < 2) { print_list_headings(columns); print_list_separators(columns); } stats.num_files = 0; stats.length = 0; stats.compressed_length = 0; stats.timestamp = read_file_timestamp(fstream); for (;;) { LHAFileHeader *header; header = lha_filter_next_file(filter); if (header == NULL) { break; } print_columns(columns, header); ++stats.num_files; stats.length += header->length; stats.compressed_length += header->compressed_length; } if (options->quiet < 2) { print_list_separators(columns); print_footers(columns, &stats); } } // Used for lha -l: static ListColumn *normal_column_headers[] = { &permission_column, &unix_uid_gid_column, &size_column, &ratio_column, ×tamp_column, &name_column, NULL }; // Used for lha -lv: static ListColumn *normal_column_headers_verbose[] = { &whole_line_name_column, &permission_column, &unix_uid_gid_column, &size_column, &ratio_column, ×tamp_column, &header_level_column, NULL }; // lha -l command. void list_file_basic(LHAFilter *filter, LHAOptions *options, FILE *fstream) { ListColumn **headers; if (options->verbose) { headers = normal_column_headers_verbose; } else { headers = normal_column_headers; } list_file_contents(filter, fstream, options, headers); } // Used for lha -v: static ListColumn *verbose_column_headers[] = { &permission_column, &unix_uid_gid_column, &packed_column, &size_column, &ratio_column, &method_crc_column, ×tamp_column, &short_name_column, NULL }; // Used for lha -vv: static ListColumn *verbose_column_headers_verbose[] = { &whole_line_name_column, &permission_column, &unix_uid_gid_column, &packed_column, &size_column, &ratio_column, &method_crc_column, ×tamp_column, &header_level_column, NULL }; // lha -v command. void list_file_verbose(LHAFilter *filter, LHAOptions *options, FILE *fstream) { ListColumn **headers; if (options->verbose) { headers = verbose_column_headers_verbose; } else { headers = verbose_column_headers; } list_file_contents(filter, fstream, options, headers); }