问题已修复。请参见下文。
嘿,大家好,我想实现一个zip下载。我有一个Web应用程序,用户可以点击一个按钮:
<template>
<button name="exportButton" id="exportButton" @click="convertConfigurationToZip" :class="['button']">{{
exportConfig
}}
</button>
</template>
methods: {
async convertConfigurationToZip() {
// Stores the current config in "store.config"
RestResource.convertConfigurationToZip();
const configData = store.config;
await new Promise((resolve) => setTimeout(resolve, 2000));
// A Blob represents raw binary data
const blob = new Blob([configData], {type: 'application/zip'});
// Use file-saver to trigger the download
saveAs(blob, 'config')
字符串
在RestResource中调用的函数如下:
convertConfigurationToZip: () => {
axios
.get("http://localhost:8080/api/conversion/convertConfigurationToZip/")
.then((response: any) => {
console.log(response.data);
store.config = response.data;
})
.catch((error: any) => {
console.log(error);
})
}
型
被调用的Controller看起来像这样:
package controller;
import io.swagger.annotations.*;
import jakarta.annotation.Resource;
import jakarta.ws.rs.GET;
import jakarta.ws.rs.Path;
import jakarta.ws.rs.container.AsyncResponse;
import jakarta.ws.rs.container.Suspended;
import jakarta.ws.rs.core.Response;
import services.ConversionService;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
// This Controller should be used for creating zip files that contains the ProCake-Configuration
@Api(tags = "Conversion Controller", authorizations = { @Authorization(value = "basicAuth") })
@Path("/conversion")
public class ConversionController
{
@Resource
ExecutorService executorService;
@ApiOperation(value = "Convert Config into Zip")
@ApiResponses({ @ApiResponse(code = 200, message = "Config as Zip"), @ApiResponse(code = 503, message = "ProCAKE not started/configured properly") })
@GET
@Path("/convertConfigurationToZip")
public void convertConfigurationToZip(@Suspended final AsyncResponse response)
{
executorService = Executors.newSingleThreadExecutor();
executorService.submit(() ->
{
try
{
response.resume(Response.status(200).entity(new ConversionService().convertConfigurationToZip()).build());
}
catch (Exception e)
{
response.resume(e);
}
executorService.shutdown();
});
}
}
型
正在处理Zip文件的服务看起来像这样:
package services;
import de.uni_trier.wi2.procake.data.model.Model;
import de.uni_trier.wi2.procake.data.model.ModelFactory;
import de.uni_trier.wi2.procake.similarity.SimilarityModel;
import de.uni_trier.wi2.procake.similarity.SimilarityModelFactory;
import de.uni_trier.wi2.procake.utils.io.IOUtil;
import org.apache.commons.compress.archivers.zip.ZipArchiveEntry;
import org.apache.commons.compress.archivers.zip.ZipArchiveOutputStream;
import org.apache.commons.io.IOUtils;
import org.apache.jena.sparql.exec.RowSet;
import java.io.*;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.zip.ZipInputStream;
public class ConversionService
{
public byte[] convertConfigurationToZip() throws IOException
{
try
{
// Create Zip on Harddrive
List<File> listOfFiles = new ArrayList<File>();
File hello = new File("pathToFile");
File world = new File("pathToFile");
listOfFiles.add(hello);
listOfFiles.add(world);
IOUtil.createZipFile("pathToZip", listOfFiles);
// ---------------------
// Create Zip File in Memory
ByteArrayOutputStream baos = IOUtil.createZipFileInMemory(listOfFiles);
;
// ---------------------
// Compare both Zip Files
byte[] zipContentHardDrive = Files.readAllBytes(Paths.get("pathToFile"));
byte[] zipContentMemory = baos.toByteArray();
int minLength = Math.min(zipContentMemory.length, zipContentHardDrive.length);
if (!Arrays.equals(
Arrays.copyOf(zipContentMemory, minLength),
Arrays.copyOf(zipContentHardDrive, minLength))) {
throw new IOException("The Zip Files are different.");
}
// ---------------------
return zipContentMemory;
}catch (Exception e) {
System.out.println("Could not create Zip File");
}
return null;
}
}
型
最后,在我给予更多细节之前,这里是IOUtil方法:
public static ByteArrayOutputStream createZipFileInMemory(List<File> files) throws IOException {
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
ZipArchiveOutputStream zipArchiveOutputStream = new ZipArchiveOutputStream(
byteArrayOutputStream);
for (File file : files) {
addZipEntry(zipArchiveOutputStream, file);
}
zipArchiveOutputStream.finish();
return byteArrayOutputStream;
}
private static void addZipEntry(ArchiveOutputStream archiveOutputStream, File file)
throws IOException {
String entryName = file.getName();
ArchiveEntry archiveEntry = new ZipArchiveEntry(file, entryName);
archiveOutputStream.putArchiveEntry(archiveEntry);
if (file.isFile()) {
Files.copy(file.toPath(), archiveOutputStream);
}
archiveOutputStream.closeArchiveEntry();
}
型
Vim ZipFile运行得很好。我的问题是,我下载的zip文件比我直接在硬盘上创建的文件大了大约200字节。我也无法打开更大的zip文件。用vim打开更大的zip文件后,我可以看到内容与较小的zip文件相比完全不同:
更小的zip:“zip.Vim version v33“浏览zipfile pathToZip“用光标选择一个文件并按ENTER键
hello. txt world.txt
这是更大的一个:PK^C^D^T^@^H ^H^@�W^@^
我不知道我在内存中创建zip文件的方法是错误的,还是在传输到store.exe时出错了。还有一件值得注意的事情是zipContentHardDrive和zipContentMemory是不同的:zipContentHarddrive:[80,75,3,4,20,0,0,8,8,0,-72,-114,-103,87,32,48,58,54,8,0,0,0,6,0,0,0,9,0,53,0,104,101,108,108,111,46,116,120,116、85、84、13、0、7、-99、-77、-119、101、-42、-76、-119、101、-99、-77、-119、101、10、0、32、0、0、0、0、1、0、24、0、-106、85、-35、-18、82、55,-38,1,4,18,-88,-87,83,55,-38,1,-128,76,-114,-18,82,55,-38,1,-53,72,-51,-55,-55,-25,2,0,+322更多]
zipContentMemory:[80,75,3,4,20,0,8,8,0,-72,-114,-103,87,0,0,0,0,0,0,0,0,0,0,0,9,0,53,0,104,101,108,108,111,46,116,120,116,85、84、13、0、7、-99、-77、-119、101、-42、-76、-119、101、-99、-77、-119、101、10、0、32、0、0、0、0、0、1、0、24、0、-106、85、-35、-18、82、55、-38,1,4,18,-88,-87,83,55,-38,1,-128,76,-114,-18,82,55,-38,1,-53,72,-51,-55,-55,-25,2,0,+354更多]
我不确定这是否是因为zip文件在内存中和不在内存中时通常是不同的。
如果有人能帮助我理解并解决我的问题,我将非常感激。非常感谢您抽出时间!
更新1在以下测试之后,我发现由于未知的原因,ZipFileInMemory方法不能像CoonversionService中那样工作。
@Test
public void testCreateZipFileInMemoryWithFiles() throws IOException {
IOUtil.writeFile(graph, PATH_GRAPH_TXT);
IOUtil.writeFile(graph, PATH_GRAPH_XML);
File fileText = new File(PATH_GRAPH_TXT);
File fileXML = new File(PATH_GRAPH_XML);
List<File> listOfFiles = new ArrayList<>();
listOfFiles.add(fileText);
listOfFiles.add(fileXML);
ByteArrayOutputStream zipFileInMemory = IOUtil.createZipFileInMemory(listOfFiles);
assertNotNull(zipFileInMemory);
// Now let's check if the data stored in the memory is actually the right Zip-file
// Therefore let's create an actual Zip-File on our hard drive
IOUtil.createZipFile(PATH_CREATED_ZIP, listOfFiles);
// Read the content of the created zip file
byte[] actualZipContent = Files.readAllBytes(Paths.get(PATH_CREATED_ZIP));
// Compare the contents of the files within the zip
ZipInputStream dataOfMemoryZip = new ZipInputStream(new ByteArrayInputStream(zipFileInMemory.toByteArray()));
ZipInputStream dataOfHardDriveZip = new ZipInputStream(new ByteArrayInputStream(actualZipContent));
ZipEntry memoryEntry;
ZipEntry hardDriveEntry;
while ((memoryEntry = dataOfMemoryZip.getNextEntry()) != null) {
hardDriveEntry = dataOfHardDriveZip.getNextEntry();
assertEquals(memoryEntry.getName(), hardDriveEntry.getName());
byte[] expectedFileContent = IOUtils.toByteArray(dataOfMemoryZip);
byte[] actualFileContent = IOUtils.toByteArray(dataOfHardDriveZip);
assertArrayEquals(expectedFileContent, actualFileContent);
}
}
型
这里的内容总是相同的。我不明白为什么ConversionService类中的内容不同,因为我使用相同的文件和相同的方法。
更新2这里是下载的zip文件的hexdump:这里是我从下载中得到的zip文件的hexdump:
504b0304140000808000efbfbdefbfbfbd5720303a36080000000600 00000900350068656c6c6f2e74787455540d0007efbfbdefbfbdefbfbd65 d6b4efbfbd65efbfbdefbfbdefbfbd650a002000000000001001800efbf bd55efbfbdefbfbd5237efbfbd010412efbfbdefbfbd5337efbfbd01efbf bd4cefbfbdefbfbd5237efbfbd01efbfbd48efbfbdefbfbdefbfbd0200504b0304140000080800efbfbdefbfbdefbfbd57efbfbd6138efbfbd 08000000060000009003500776f726c642e74787455540d000 7efbfbdef bfbdefbfbd65efbfbdefbfbdefbfbd65efbfbdefbfbdefbfbd650a002000 000@00000100180009efbfbd28efbfbd5237efbfbd01efbfbdefbfbdefbf bdefbfbd523 7efbfbd01efbfbd5aefbfbdefbfbd5237efbfbd012befbfbd 2fefbfbd49efbfbd0200504b0102140014000080800efbfbdefbfbdefbf bd5720303a36080000000060000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000686656c6fbdefbdefbfbdefbfbd650a002000 000000001001800efbfbd55efbfbdefbfbd5237efbfbd010412efbfbdef bfbd5337efbfbd01efbfbd4cefbfbdefbfbd5237efbfbd01504b01021400 140000080800efbfbdefbfbfbd57efbfbd6138efbfbd08000000600 000009002d000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 bd5aefbfbdefbfbd5237efbfbd01504b0506000000002000efbfbd00 00000efbfbd00000000
下面是工作zip文件的hexdump:
504b03041400000808000b88e995720303a3608000000060000009003500 68656c6c6f2e74787455540d00079db38965d6b489659db389650a002000 0000000010018009655ddee5237da010412a8a95337da01804c8eee5237 da01cb48cdc9c9e70200504b0304140000080800c08e9957a86138dd0800 00000600000009003500776f726c642e74787455540d0007a9b38965abb3 8965a9b389650a0020000000000000100180009ba28f65237da0183db6 5237da01805ab5f55237da012bcf2fca49e10200504b0102140014000008 0800b88e995720303a360800000000600000009002d000000000000000 0000000000068656c6c6f2e74787455540500079db389650a0020000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000 504b01021400140000080800c08e9957a86138dd08000000060000000900 2d0000000000000000000640000000000776f726c642e7478745554050007 a9b389650a00200000000000100180009ba28f65237da0183da01805ab5f55237da01504b05060000000002000200c80000000c80000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000
**问题修复:**以下是更改:
methods: {
async convertConfigurationToZip() {
if (this.conversionInProgress) {
return;
}
this.conversionInProgress = true;
// Stores the current config in "store.config"
const response = await RestResource.convertConfigurationToZip();
// A Blob represents raw binary data
const blob = new Blob([response.data], {type: 'application/zip'}); // #
saveAs(blob, 'config.zip');
this.conversionInProgress = false;
}
型
和
convertConfigurationToZip: async () => {
return axios.get("http://localhost:8080/api/conversion/convertConfigurationToZip/", {
responseType: "blob", // Ensure the response is treated as a blob
})
.then((response) => {
return response; // Return the entire Axios response object
})
.catch((error) => {
console.error(error);
throw error; // Rethrow the error to handle it appropriately in the calling context
});
}
型
我假设const configData = store.config
是一个字符串而不是二进制数据。
1条答案
按热度按时间mm5n2pyu1#
您创建的zip文件的部分十进制转储确实不同。
zipContentMemory
创建了一个 streamed zip文件,而zipContentHarddrive
没有。这足以解释为什么一个zip文件可以工作,而另一个不能。注意事项:如果没有看到完整的zip文件,就不可能100%确定。可能是zip文件中的一些其他差异导致了您看到的问题。如果有机会,请发布完整zip的转储。
部分
zipContentHarddrive
zip文件如下所示(下面的数据是用zipdetails创建的字符串
部分
zipContentMemory
zip文件如下所示型
更新
复制到2.zip的原始文件没有问题
型
下载的文件,复制到/tmp/1.zip,是非常腐败的zip文件
型
看看十六进制转储,有很多重复的
EF BF BD
序列。这三个字节恰好是Unicode替换字符的UTF8字节序列。这表明根本原因是您的服务器代码或客户端代码将zip文件作为UTF8字符串而不是字节序列处理。