src.org.python.modules.bz2.PyBZ2File Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jython-standalone Show documentation
Show all versions of jython-standalone Show documentation
Jython is an implementation of the high-level, dynamic, object-oriented
language Python written in 100% Pure Java, and seamlessly integrated with
the Java platform. It thus allows you to run Python on any Java platform.
package org.python.modules.bz2;
import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import java.util.Iterator;
import org.apache.commons.compress.compressors.bzip2.BZip2CompressorInputStream;
import org.apache.commons.compress.compressors.bzip2.BZip2CompressorOutputStream;
import org.python.core.ArgParser;
import org.python.core.Py;
import org.python.core.PyException;
import org.python.core.PyIterator;
import org.python.core.PyList;
import org.python.core.PyLong;
import org.python.core.PyNone;
import org.python.core.PyObject;
import org.python.core.PySequence;
import org.python.core.PyString;
import org.python.core.PyType;
import org.python.core.io.BinaryIOWrapper;
import org.python.core.io.BufferedReader;
import org.python.core.io.BufferedWriter;
import org.python.core.io.IOBase;
import org.python.core.io.StreamIO;
import org.python.core.io.TextIOBase;
import org.python.core.io.UniversalIOWrapper;
import org.python.expose.ExposedGet;
import org.python.expose.ExposedMethod;
import org.python.expose.ExposedNew;
import org.python.expose.ExposedType;
@ExposedType(name = "bz2.BZ2File")
public class PyBZ2File extends PyObject {
public static final PyType TYPE = PyType.fromClass(PyBZ2File.class);
private int buffering;
@ExposedGet(name = "newlines")
public PyObject PyBZ2File_newlines() {
if (buffer != null) {
return buffer.getNewlines();
} else {
return Py.None;
}
}
private TextIOBase buffer;
private String fileName = null;
private boolean inIterMode = false;
private boolean inUniversalNewlineMode = false;
private boolean needReadBufferInit = false;
private boolean inReadMode = false;
private boolean inWriteMode = false;
public PyBZ2File() {
super(TYPE);
}
public PyBZ2File(PyType subType) {
super(subType);
}
@Override
protected void finalize() throws Throwable {
BZ2File_close();
super.finalize();
}
@ExposedNew
@ExposedMethod
final void BZ2File___init__(PyObject[] args, String[] kwds) {
ArgParser ap = new ArgParser("bz2file", args, kwds, new String[] {
"filename", "mode", "buffering", "compresslevel" }, 1);
PyObject filename = ap.getPyObject(0);
if (!(filename instanceof PyString)) {
throw Py.TypeError("coercing to Unicode: need string, '"
+ filename.getType().fastGetName() + "' type found");
}
String mode = ap.getString(1, "r");
int buffering = ap.getInt(2, 0);
int compresslevel = ap.getInt(3, 9);
BZ2File___init__((PyString) filename, mode, buffering, compresslevel);
}
private void BZ2File___init__(PyString inFileName, String mode,
int buffering, int compresslevel) {
try {
fileName = inFileName.asString();
this.buffering = buffering;
// check universal newline mode
if (mode.contains("U")) {
inUniversalNewlineMode = true;
}
if (mode.contains("w")) {
inWriteMode = true;
File f = new File(fileName);
if (!f.exists()) {
f.createNewFile();
}
BZip2CompressorOutputStream writeStream = new BZip2CompressorOutputStream(
new FileOutputStream(fileName), compresslevel);
buffer = new BinaryIOWrapper(
new BufferedWriter(
new SkippableStreamIO(writeStream, true),
buffering));
} else {
File f = new File(fileName);
if (!f.exists()) {
throw new FileNotFoundException();
}
inReadMode = true;
needReadBufferInit = true;
}
} catch (IOException e) {
throw Py.IOError("File " + fileName + " not found,");
}
}
private void makeReadBuffer() {
try {
FileInputStream fin = new FileInputStream(fileName);
BufferedInputStream bin = new BufferedInputStream(fin);
BZip2CompressorInputStream bZin = new BZip2CompressorInputStream(
bin, true);
BufferedReader bufferedReader = new BufferedReader(
new SkippableStreamIO(bZin, true), buffering);
if (inUniversalNewlineMode) {
buffer = new UniversalIOWrapper(bufferedReader);
} else {
buffer = new BinaryIOWrapper(bufferedReader);
}
} catch (FileNotFoundException fileNotFoundException) {
throw Py.IOError(fileNotFoundException);
} catch (IOException ioe) {
throw Py.EOFError(ioe.getMessage());
}
}
@ExposedMethod
public void __del__() {
BZ2File_close();
}
@ExposedMethod
public void BZ2File_close() {
needReadBufferInit = false;
BZ2File_flush();
if (buffer != null) {
buffer.close();
}
}
private void BZ2File_flush() {
if (buffer != null) {
buffer.flush();
}
}
@ExposedMethod
public PyObject BZ2File_read(PyObject[] args, String[] kwds) {
checkInIterMode();
checkReadBufferInit();
ArgParser ap = new ArgParser("read", args, kwds,
new String[] { "size" }, 0);
int size = ap.getInt(0, -1);
if (size == 0) { return Py.EmptyString; }
if (size < 0) { return new PyString(buffer.readall()); }
StringBuilder data = new StringBuilder(size);
while (data.length() < size) {
String chunk = buffer.read(size - data.length());
if (chunk.length() == 0) {
break;
}
data.append(chunk);
}
return new PyString(data.toString());
}
@ExposedMethod
public PyObject BZ2File_next(PyObject[] args, String[] kwds) {
checkReadBufferInit();
if (buffer == null || buffer.closed()) {
throw Py.ValueError("Cannot call next() on closed file");
}
inIterMode = true;
return null;
}
@ExposedMethod
public PyString BZ2File_readline(PyObject[] args, String[] kwds) {
checkInIterMode();
checkReadBufferInit();
ArgParser ap = new ArgParser("read", args, kwds,
new String[] { "size" }, 0);
int size = ap.getInt(0, -1);
return new PyString(buffer.readline(size));
}
@ExposedMethod
public PyList BZ2File_readlines(PyObject[] args, String[] kwds) {
checkInIterMode();
checkReadBufferInit();
// make sure file data valid
if (buffer == null || buffer.closed()) {
throw Py.ValueError("Cannot call readlines() on a closed file");
}
PyList lineList = new PyList();
PyString line = null;
while (!(line = BZ2File_readline(args, kwds)).equals(Py.EmptyString)) {
lineList.add(line);
}
return lineList;
}
private void checkInIterMode() {
if (inReadMode && inIterMode) {
throw Py.ValueError("Cannot mix iteration and reads");
}
}
private void checkReadBufferInit() {
if (inReadMode && needReadBufferInit) {
makeReadBuffer();
needReadBufferInit = false;
}
}
@ExposedMethod
public PyList BZ2File_xreadlines() {
return BZ2File_readlines(new PyObject[0], new String[0]);
}
@ExposedMethod
public void BZ2File_seek(PyObject[] args, String[] kwds) {
if (!inReadMode) {
Py.IOError("seek works only while reading");
}
ArgParser ap = new ArgParser("seek", args, kwds, new String[] {
"offset", "whence" }, 1);
checkReadBufferInit();
int newOffset = ap.getInt(0);
int whence = ap.getInt(1, 0);
// normalise offset
long currentPos = buffer.tell();
long finalOffset = 0;
switch (whence) {
case 0: // offset from start of file
finalOffset = newOffset;
break;
case 1: // move relative to current position
finalOffset = currentPos + newOffset;
break;
case 2: // move relative to end of file
long fileSize = currentPos;
// in order to seek from the end of the stream we need to fully read
// the decompressed stream to get the size
for (;;) {
final String data = buffer.read(IOBase.DEFAULT_BUFFER_SIZE);
if (data.isEmpty()) {
break;
}
fileSize += data.length();
}
finalOffset = fileSize + newOffset;
// close and reset the buffer
buffer.close();
makeReadBuffer();
break;
}
if (finalOffset < 0) {
finalOffset = 0;
}
// can't seek backwards so close and reopen the stream at the start
if (whence != 2 && finalOffset < currentPos) {
buffer.close();
makeReadBuffer();
}
// seek operation
buffer.seek(finalOffset, 0);
}
@ExposedMethod
public PyLong BZ2File_tell() {
checkReadBufferInit();
if (buffer == null) {
return Py.newLong(0);
} else {
return Py.newLong(buffer.tell());
}
}
@ExposedMethod
public void BZ2File_write(PyObject[] args, String[] kwds) {
checkFileWritable();
ArgParser ap = new ArgParser("write", args, kwds,
new String[] { "data" }, 0);
PyObject data = ap.getPyObject(0);
if (data.getType() == PyNone.TYPE) {
throw Py.TypeError("Expecting str argument");
}
buffer.write(ap.getString(0));
}
@ExposedMethod
public void BZ2File_writelines(PyObject[] args, String[] kwds) {
checkFileWritable();
ArgParser ap = new ArgParser("writelines", args, kwds,
new String[] { "sequence_of_strings" }, 0);
PySequence seq = (PySequence) ap.getPyObject(0);
for (Iterator iterator = seq.asIterable().iterator(); iterator
.hasNext();) {
PyObject line = iterator.next();
BZ2File_write(new PyObject[] { line }, new String[] { "data" });
}
}
private void checkFileWritable() {
if (inReadMode) {
throw Py.IOError("File in read-only mode");
}
if (buffer == null || buffer.closed()) {
throw Py.ValueError("Stream closed");
}
}
@Override
@ExposedMethod
public PyObject __iter__() {
return new BZ2FileIterator();
}
private class BZ2FileIterator extends PyIterator {
@Override
public PyObject __iternext__() {
PyString s = BZ2File_readline(new PyObject[0], new String[0]);
if (s.equals(Py.EmptyString)) {
return null;
} else {
return s;
}
}
}
@ExposedMethod
public PyObject BZ2File___enter__() {
if (inWriteMode) {
if (buffer == null) {
throw Py.ValueError("Stream closed");
}
} else if (inReadMode && !needReadBufferInit) {
if (buffer == null || buffer.closed()) {
throw Py.ValueError("Stream closed");
}
}
return this;
}
@ExposedMethod
public boolean BZ2File___exit__(PyObject exc_type, PyObject exc_value,
PyObject traceback) {
BZ2File_close();
return false;
}
private static class SkippableStreamIO extends StreamIO {
private long position = 0;
public SkippableStreamIO(InputStream inputStream, boolean closefd) {
super(inputStream, closefd);
}
public SkippableStreamIO(OutputStream outputStream, boolean closefd) {
super(outputStream, closefd);
}
@Override
public int readinto(ByteBuffer buf) {
int bytesRead = 0;
try {
bytesRead = super.readinto(buf);
} catch (PyException pyex) {
// translate errors on read of decompressed stream to EOFError
throw Py.EOFError(pyex.value.asStringOrNull());
}
position += bytesRead;
return bytesRead;
}
@Override
public long tell() {
return position;
}
@Override
public long seek(long offset, int whence) {
long skipBytes = offset - position;
if (whence != 0 || skipBytes < 0) {
throw Py.IOError("can only seek forward");
}
if (skipBytes == 0) {
return position;
} else {
long skipped = 0;
try {
skipped = asInputStream().skip(skipBytes);
} catch (IOException ex) {
throw Py.IOError(ex);
}
long newPosition = position + skipped;
position = newPosition;
return newPosition;
}
}
}
}