mirror of
https://git.huckle.dev/Huckles-Minecraft-Archive/jpexs-decompiler.git
synced 2026-05-26 21:16:48 +00:00
261 lines
7.2 KiB
Java
261 lines
7.2 KiB
Java
/*
|
|
* Copyright (C) 2010-2024 JPEXS, All rights reserved.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 3.0 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library.
|
|
*/
|
|
package com.jpexs.decompiler.flash;
|
|
|
|
import com.jpexs.helpers.Helper;
|
|
import com.jpexs.helpers.MemoryInputStream;
|
|
import com.jpexs.helpers.ReReadableInputStream;
|
|
import com.jpexs.helpers.streams.SeekableInputStream;
|
|
import java.io.ByteArrayInputStream;
|
|
import java.io.File;
|
|
import java.io.FileInputStream;
|
|
import java.io.FileOutputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStream;
|
|
import java.util.HashMap;
|
|
import java.util.HashSet;
|
|
import java.util.Map;
|
|
import java.util.Set;
|
|
import java.util.zip.ZipEntry;
|
|
import java.util.zip.ZipInputStream;
|
|
import java.util.zip.ZipOutputStream;
|
|
|
|
/**
|
|
* Bundle implementation for ZIP files.
|
|
* @author JPEXS
|
|
*/
|
|
public class ZippedBundle implements Bundle {
|
|
|
|
/**
|
|
*
|
|
*/
|
|
protected Set<String> keySet = new HashSet<>();
|
|
|
|
/**
|
|
* File input stream for the ZIP file.
|
|
*/
|
|
protected FileInputStream fis;
|
|
|
|
/**
|
|
* Re-readable input stream for the ZIP file.
|
|
*/
|
|
protected ReReadableInputStream is;
|
|
|
|
/**
|
|
* File name of the ZIP file.
|
|
*/
|
|
protected File filename;
|
|
|
|
/**
|
|
* Constructs a new ZippedBundle from an input stream.
|
|
* @param is Input stream
|
|
* @throws IOException
|
|
*/
|
|
public ZippedBundle(InputStream is) throws IOException {
|
|
this(is, null);
|
|
}
|
|
|
|
/**
|
|
* Constructs a new ZippedBundle from a file.
|
|
* @param filename File
|
|
* @throws IOException
|
|
*/
|
|
public ZippedBundle(File filename) throws IOException {
|
|
this(null, filename);
|
|
}
|
|
|
|
/**
|
|
* Constructs a new ZippedBundle from an input stream and a file.
|
|
* @param is Input stream
|
|
* @param filename File
|
|
* @throws IOException
|
|
*/
|
|
protected ZippedBundle(InputStream is, File filename) throws IOException {
|
|
initBundle(is, filename);
|
|
}
|
|
|
|
/**
|
|
* Initializes the bundle.
|
|
* @param is Input stream
|
|
* @param filename File
|
|
* @throws IOException
|
|
*/
|
|
protected void initBundle(InputStream is, File filename) throws IOException {
|
|
if (filename != null) {
|
|
fis = new FileInputStream(filename);
|
|
is = fis;
|
|
}
|
|
this.filename = filename;
|
|
this.is = new ReReadableInputStream(is);
|
|
ZipInputStream zip = new ZipInputStream(this.is);
|
|
ZipEntry entry;
|
|
keySet.clear();
|
|
|
|
while ((entry = zip.getNextEntry()) != null) {
|
|
if (entry.getName().toLowerCase().endsWith(".swf")
|
|
|| entry.getName().toLowerCase().endsWith(".gfx")
|
|
|| entry.getName().toLowerCase().endsWith(".abc")) {
|
|
keySet.add(entry.getName());
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Gets the number of entries in the bundle.
|
|
* @return
|
|
*/
|
|
@Override
|
|
public int length() {
|
|
return keySet.size();
|
|
}
|
|
|
|
/**
|
|
* Gets the keys in the bundle.
|
|
* @return
|
|
*/
|
|
@Override
|
|
public Set<String> getKeys() {
|
|
return keySet;
|
|
}
|
|
|
|
/**
|
|
* Gets the input stream for a key.
|
|
* @param key Key
|
|
* @return Input stream
|
|
* @throws IOException
|
|
*/
|
|
@Override
|
|
public SeekableInputStream getOpenable(String key) throws IOException {
|
|
if (!keySet.contains(key)) {
|
|
return null;
|
|
}
|
|
//if (!cachedSWFs.containsKey(key)) {
|
|
SeekableInputStream ret = null;
|
|
this.is.reset();
|
|
ZipInputStream zip = new ZipInputStream(this.is);
|
|
ZipEntry entry;
|
|
|
|
while ((entry = zip.getNextEntry()) != null) {
|
|
if (entry.getName().equals(key)) {
|
|
MemoryInputStream mis = new MemoryInputStream(Helper.readStream(zip));
|
|
ret = mis;
|
|
//cachedSWFs.put(key, mis);
|
|
break;
|
|
}
|
|
zip.closeEntry();
|
|
}
|
|
|
|
return ret;
|
|
//return cachedSWFs.get(key);
|
|
}
|
|
|
|
/**
|
|
* Gets all input streams in the bundle.
|
|
* @return Map of key to input stream
|
|
* @throws IOException
|
|
*/
|
|
@Override
|
|
public Map<String, SeekableInputStream> getAll() throws IOException {
|
|
Map<String, SeekableInputStream> ret = new HashMap<>();
|
|
for (String key : getKeys()) { // cache everything first
|
|
ret.put(key, getOpenable(key));
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* Gets the extension of the bundle.
|
|
* @return Extension
|
|
*/
|
|
@Override
|
|
public String getExtension() {
|
|
return "zip";
|
|
}
|
|
|
|
/**
|
|
* Gets whether the bundle is read-only.
|
|
* @return Whether the bundle is read-only
|
|
*/
|
|
@Override
|
|
public boolean isReadOnly() {
|
|
return this.filename == null || !this.filename.canWrite();
|
|
}
|
|
|
|
|
|
/**
|
|
* Replaces the input stream for a key.
|
|
* @param key Key
|
|
* @param swfIs New input stream
|
|
* @return Whether the operation was successful
|
|
* @throws IOException
|
|
*/
|
|
@Override
|
|
public boolean putOpenable(String key, InputStream swfIs) throws IOException {
|
|
if (this.isReadOnly()) {
|
|
return false;
|
|
}
|
|
if (key == null) {
|
|
return false;
|
|
}
|
|
if (!getKeys().contains(key)) { //replace only existing keys
|
|
return false;
|
|
}
|
|
//Write to temp file first
|
|
File tempFile = new File((filename.getAbsolutePath()) + ".tmp");
|
|
|
|
try {
|
|
ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(tempFile));
|
|
this.is.reset();
|
|
ZipInputStream zis = new ZipInputStream(this.is);
|
|
ZipEntry entryIn;
|
|
ZipEntry entryOut;
|
|
|
|
byte[] swfData = Helper.readStream(swfIs);
|
|
|
|
try {
|
|
while ((entryIn = zis.getNextEntry()) != null) {
|
|
InputStream src;
|
|
if (entryIn.getName().equals(key)) {
|
|
entryOut = new ZipEntry(key);
|
|
src = new ByteArrayInputStream(swfData);
|
|
} else {
|
|
src = zis;
|
|
entryOut = entryIn;
|
|
}
|
|
zos.putNextEntry(entryOut);
|
|
Helper.copyStream(src, zos, entryOut.getSize() == -1 ? Long.MAX_VALUE : entryOut.getSize());
|
|
zos.closeEntry();
|
|
zis.closeEntry();
|
|
}
|
|
} finally {
|
|
zis.close();
|
|
zos.close();
|
|
}
|
|
this.is.close();
|
|
this.fis.close();
|
|
} catch (IOException ex) {
|
|
tempFile.delete();
|
|
throw ex;
|
|
}
|
|
filename.delete();
|
|
tempFile.renameTo(filename);
|
|
initBundle(null, filename);
|
|
|
|
return true;
|
|
}
|
|
}
|