com.whaleal.icefrog.extra.compress.extractor.SenvenZExtractor Maven / Gradle / Ivy
package com.whaleal.icefrog.extra.compress.extractor;
import com.whaleal.icefrog.core.io.FileUtil;
import com.whaleal.icefrog.core.io.IORuntimeException;
import com.whaleal.icefrog.core.io.IoUtil;
import com.whaleal.icefrog.core.lang.Preconditions;
import com.whaleal.icefrog.core.lang.Filter;
import org.apache.commons.compress.archivers.ArchiveEntry;
import org.apache.commons.compress.archivers.sevenz.SevenZArchiveEntry;
import org.apache.commons.compress.archivers.sevenz.SevenZFile;
import org.apache.commons.compress.utils.SeekableInMemoryByteChannel;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.channels.SeekableByteChannel;
/**
* 7z格式数据解压器,即将归档打包的数据释放
*
* @author Looly
* @author wh
* @since 1.0.0
*/
public class SenvenZExtractor implements Extractor {
private final SevenZFile sevenZFile;
/**
* 构造
*
* @param file 包文件
*/
public SenvenZExtractor(File file) {
this(file, null);
}
/**
* 构造
*
* @param file 包文件
* @param password 密码,null表示无密码
*/
public SenvenZExtractor(File file, char[] password) {
try {
this.sevenZFile = new SevenZFile(file, password);
} catch (IOException e) {
throw new IORuntimeException(e);
}
}
/**
* 构造
*
* @param in 包流
*/
public SenvenZExtractor(InputStream in) {
this(in, null);
}
/**
* 构造
*
* @param in 包流
* @param password 密码,null表示无密码
*/
public SenvenZExtractor(InputStream in, char[] password) {
this(new SeekableInMemoryByteChannel(IoUtil.readBytes(in)), password);
}
/**
* 构造
*
* @param channel {@link SeekableByteChannel}
*/
public SenvenZExtractor(SeekableByteChannel channel) {
this(channel, null);
}
/**
* 构造
*
* @param channel {@link SeekableByteChannel}
* @param password 密码,null表示无密码
*/
public SenvenZExtractor(SeekableByteChannel channel, char[] password) {
try {
this.sevenZFile = new SevenZFile(channel, password);
} catch (IOException e) {
throw new IORuntimeException(e);
}
}
/**
* 释放(解压)到指定目录,结束后自动关闭流,此方法只能调用一次
*
* @param targetDir 目标目录
* @param filter 解压文件过滤器,用于指定需要释放的文件,null表示不过滤。当{@link Filter#accept(Object)}为true时释放。
*/
@Override
public void extract(File targetDir, Filter filter) {
try {
extractInternal(targetDir, filter);
} catch (IOException e) {
throw new IORuntimeException(e);
} finally {
close();
}
}
/**
* 释放(解压)到指定目录
*
* @param targetDir 目标目录
* @param filter 解压文件过滤器,用于指定需要释放的文件,null表示不过滤。当{@link Filter#accept(Object)}为true时释放。
* @throws IOException IO异常
*/
private void extractInternal(File targetDir, Filter filter) throws IOException {
Preconditions.isTrue(null != targetDir && ((false == targetDir.exists()) || targetDir.isDirectory()), "target must be dir.");
final SevenZFile sevenZFile = this.sevenZFile;
SevenZArchiveEntry entry;
File outItemFile;
while (null != (entry = this.sevenZFile.getNextEntry())) {
outItemFile = FileUtil.file(targetDir, entry.getName());
if (entry.isDirectory()) {
// 创建对应目录
//noinspection ResultOfMethodCallIgnored
outItemFile.mkdirs();
} else if(entry.hasStream()){
// 读取entry对应数据流
FileUtil.writeFromStream(new Seven7EntryInputStream(sevenZFile, entry), outItemFile);
} else {
// 无数据流的文件创建为空文件
FileUtil.touch(outItemFile);
}
}
}
@Override
public void close() {
IoUtil.close(this.sevenZFile);
}
}