mirror of
https://github.com/zlib-ng/minizip-ng
synced 2025-03-28 21:13:18 +00:00
Fix for Names truncated at 256 bytes #690
Co-authored-by: Nathan Moinvaziri <nathan@nathanm.com>
This commit is contained in:
parent
706737f980
commit
693edec9fa
96
mz_zip_rw.c
96
mz_zip_rw.c
@ -650,38 +650,45 @@ int32_t mz_zip_reader_entry_save_file(void *handle, const char *path) {
|
||||
int32_t err_attrib = 0;
|
||||
int32_t err = MZ_OK;
|
||||
int32_t err_cb = MZ_OK;
|
||||
char pathwfs[512];
|
||||
char directory[512];
|
||||
size_t path_length = 0;
|
||||
char *pathwfs = NULL;
|
||||
char *directory = NULL;
|
||||
|
||||
if (mz_zip_reader_is_open(reader) != MZ_OK)
|
||||
return MZ_PARAM_ERROR;
|
||||
if (!reader->file_info || !path)
|
||||
return MZ_PARAM_ERROR;
|
||||
|
||||
path_length = strlen(path);
|
||||
|
||||
/* Convert to forward slashes for unix which doesn't like backslashes */
|
||||
strncpy(pathwfs, path, sizeof(pathwfs) - 1);
|
||||
pathwfs[sizeof(pathwfs) - 1] = 0;
|
||||
pathwfs = (char *)calloc(path_length + 1, sizeof(char));
|
||||
if (!pathwfs)
|
||||
return MZ_MEM_ERROR;
|
||||
strncat(pathwfs, path, path_length);
|
||||
mz_path_convert_slashes(pathwfs, MZ_PATH_SLASH_UNIX);
|
||||
|
||||
if (reader->entry_cb)
|
||||
reader->entry_cb(handle, reader->entry_userdata, reader->file_info, pathwfs);
|
||||
|
||||
strncpy(directory, pathwfs, sizeof(directory) - 1);
|
||||
directory[sizeof(directory) - 1] = 0;
|
||||
directory = (char *)calloc(path_length + 1, sizeof(char));
|
||||
if (!directory)
|
||||
return MZ_MEM_ERROR;
|
||||
strncat(directory, pathwfs, path_length);
|
||||
mz_path_remove_filename(directory);
|
||||
|
||||
/* If it is a directory entry then create a directory instead of writing file */
|
||||
if ((mz_zip_entry_is_dir(reader->zip_handle) == MZ_OK) &&
|
||||
(mz_zip_entry_is_symlink(reader->zip_handle) != MZ_OK)) {
|
||||
err = mz_dir_make(directory);
|
||||
return err;
|
||||
goto save_cleanup;
|
||||
}
|
||||
|
||||
/* Check if file exists and ask if we want to overwrite */
|
||||
if (reader->overwrite_cb && mz_os_file_exists(pathwfs) == MZ_OK) {
|
||||
err_cb = reader->overwrite_cb(handle, reader->overwrite_userdata, reader->file_info, pathwfs);
|
||||
if (err_cb != MZ_OK)
|
||||
return err;
|
||||
goto save_cleanup;
|
||||
/* We want to overwrite the file so we delete the existing one */
|
||||
mz_os_unlink(pathwfs);
|
||||
}
|
||||
@ -697,7 +704,7 @@ int32_t mz_zip_reader_entry_save_file(void *handle, const char *path) {
|
||||
if (mz_os_is_dir(directory) != MZ_OK) {
|
||||
err = mz_dir_make(directory);
|
||||
if (err != MZ_OK)
|
||||
return err;
|
||||
goto save_cleanup;
|
||||
}
|
||||
|
||||
/* If it is a symbolic link then create symbolic link instead of writing file */
|
||||
@ -708,8 +715,10 @@ int32_t mz_zip_reader_entry_save_file(void *handle, const char *path) {
|
||||
} else if (reader->file_info->uncompressed_size < UINT16_MAX) {
|
||||
/* Create symbolic link from zip entry contents */
|
||||
stream = mz_stream_mem_create();
|
||||
if (!stream)
|
||||
return MZ_MEM_ERROR;
|
||||
if (!stream) {
|
||||
err = MZ_MEM_ERROR;
|
||||
goto save_cleanup;
|
||||
}
|
||||
|
||||
err = mz_stream_mem_open(stream, NULL, MZ_OPEN_MODE_CREATE);
|
||||
|
||||
@ -730,13 +739,15 @@ int32_t mz_zip_reader_entry_save_file(void *handle, const char *path) {
|
||||
}
|
||||
|
||||
/* Don't check return value because we aren't validating symbolic link target */
|
||||
return err;
|
||||
goto save_cleanup;
|
||||
}
|
||||
|
||||
/* Create the file on disk so we can save to it */
|
||||
stream = mz_stream_os_create();
|
||||
if (!stream)
|
||||
return MZ_MEM_ERROR;
|
||||
if (!stream) {
|
||||
err = MZ_MEM_ERROR;
|
||||
goto save_cleanup;
|
||||
}
|
||||
|
||||
err = mz_stream_os_open(stream, pathwfs, MZ_OPEN_MODE_CREATE);
|
||||
|
||||
@ -761,6 +772,10 @@ int32_t mz_zip_reader_entry_save_file(void *handle, const char *path) {
|
||||
mz_os_set_file_attribs(pathwfs, target_attrib);
|
||||
}
|
||||
|
||||
save_cleanup:
|
||||
free(pathwfs);
|
||||
free(directory);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
@ -808,10 +823,12 @@ int32_t mz_zip_reader_entry_save_buffer_length(void *handle) {
|
||||
int32_t mz_zip_reader_save_all(void *handle, const char *destination_dir) {
|
||||
mz_zip_reader *reader = (mz_zip_reader *)handle;
|
||||
int32_t err = MZ_OK;
|
||||
int32_t utf8_name_size = 0;
|
||||
int32_t resolved_name_size = 0;
|
||||
char *utf8_string = NULL;
|
||||
char path[512];
|
||||
char utf8_name[256];
|
||||
char resolved_name[256];
|
||||
char *path = NULL;
|
||||
char *utf8_name = NULL;
|
||||
char *resolved_name = NULL;
|
||||
|
||||
err = mz_zip_reader_goto_first_entry(handle);
|
||||
|
||||
@ -819,29 +836,53 @@ int32_t mz_zip_reader_save_all(void *handle, const char *destination_dir) {
|
||||
return err;
|
||||
|
||||
while (err == MZ_OK) {
|
||||
/* Assume 4 bytes per character needed + 1 for terminating null */
|
||||
utf8_name_size = reader->file_info->filename_size * 4 + 1;
|
||||
resolved_name_size = utf8_name_size;
|
||||
|
||||
if (destination_dir) {
|
||||
/* +1 is for the "/" separator */
|
||||
resolved_name_size += (int)strlen(destination_dir) + 1;
|
||||
}
|
||||
|
||||
if (!path) {
|
||||
path = (char *)malloc(resolved_name_size);
|
||||
utf8_name = (char *)malloc(utf8_name_size);
|
||||
resolved_name = (char *)malloc(resolved_name_size);
|
||||
} else {
|
||||
path = (char *)realloc(path, resolved_name_size);
|
||||
utf8_name = (char *)realloc(utf8_name, utf8_name_size);
|
||||
resolved_name = (char *)realloc(resolved_name, resolved_name_size);
|
||||
}
|
||||
|
||||
if (!path || !utf8_name || !resolved_name) {
|
||||
err = MZ_MEM_ERROR;
|
||||
goto save_all_cleanup;
|
||||
}
|
||||
|
||||
/* Construct output path */
|
||||
path[0] = 0;
|
||||
|
||||
strncpy(utf8_name, reader->file_info->filename, sizeof(utf8_name) - 1);
|
||||
utf8_name[sizeof(utf8_name) - 1] = 0;
|
||||
strncpy(utf8_name, reader->file_info->filename, utf8_name_size - 1);
|
||||
utf8_name[utf8_name_size - 1] = 0;
|
||||
|
||||
if ((reader->encoding > 0) && (reader->file_info->flag & MZ_ZIP_FLAG_UTF8) == 0) {
|
||||
utf8_string = mz_os_utf8_string_create(reader->file_info->filename, reader->encoding);
|
||||
if (utf8_string) {
|
||||
strncpy(utf8_name, utf8_string, sizeof(utf8_name) - 1);
|
||||
utf8_name[sizeof(utf8_name) - 1] = 0;
|
||||
strncpy(utf8_name, (char *)utf8_string, utf8_name_size - 1);
|
||||
utf8_name[utf8_name_size - 1] = 0;
|
||||
mz_os_utf8_string_delete(&utf8_string);
|
||||
}
|
||||
}
|
||||
|
||||
err = mz_path_resolve(utf8_name, resolved_name, sizeof(resolved_name));
|
||||
err = mz_path_resolve(utf8_name, resolved_name, resolved_name_size);
|
||||
if (err != MZ_OK)
|
||||
break;
|
||||
|
||||
if (destination_dir)
|
||||
mz_path_combine(path, destination_dir, sizeof(path));
|
||||
mz_path_combine(path, destination_dir, resolved_name_size);
|
||||
|
||||
mz_path_combine(path, resolved_name, sizeof(path));
|
||||
mz_path_combine(path, resolved_name, resolved_name_size);
|
||||
|
||||
/* Save file to disk */
|
||||
err = mz_zip_reader_entry_save_file(handle, path);
|
||||
@ -851,7 +892,12 @@ int32_t mz_zip_reader_save_all(void *handle, const char *destination_dir) {
|
||||
}
|
||||
|
||||
if (err == MZ_END_OF_LIST)
|
||||
return MZ_OK;
|
||||
err = MZ_OK;
|
||||
|
||||
save_all_cleanup:
|
||||
free(path);
|
||||
free(utf8_name);
|
||||
free(resolved_name);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user