/**
* JsonRandomAccessFile
* Copyright 2015 by Michael Peter Christen
* First released 04.10.2015
*
* 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 2.1 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 program in the file lgpl21.txt
* If not, see <http://www.gnu.org/licenses/>.
*/
package org.loklak.tools.storage;
import java.io.File;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.concurrent.ArrayBlockingQueue;
import org.eclipse.jetty.util.log.Log;
import org.json.JSONObject;
import org.loklak.tools.BufferedRandomAccessFile;
import org.loklak.tools.UTF8;
public class JsonRandomAccessFile extends BufferedRandomAccessFile implements JsonReader {
private File file;
private int concurrency;
private ArrayBlockingQueue<JsonFactory> jsonline;
/**
* if a JsonRandomAccessFile object in initiated, it must be wrapped with a Thread object and started.
* @param dumpFile
* @param concurrency
* @throws IOException
*/
public JsonRandomAccessFile(final File dumpFile, final int concurrency) throws IOException {
super(dumpFile, "rw");
this.file = dumpFile;
this.concurrency = concurrency;
this.jsonline = new ArrayBlockingQueue<>(1000);
}
public String getName() {
return this.file.getAbsolutePath();
}
public int getConcurrency() {
return this.concurrency;
}
public JsonFactory take() throws InterruptedException {
return this.jsonline.take();
}
public void run() {
try {
BufferedRandomAccessFile.IndexedLine line;
while ((line = this.readIndexedLine()) != null) {
try {
byte[] textb = line.getText();
if (textb == null || textb.length == 0) continue;
JSONObject json = new JSONObject(new String(textb, StandardCharsets.UTF_8));
this.jsonline.put(new JsonHandle(json, line.getPos(), textb.length));
} catch (Throwable e) {
Log.getLog().warn("cannot parse line in file " + JsonRandomAccessFile.this.file + ": \"" + line + "\"", e);
}
}
} catch (IOException e) {
Log.getLog().warn(e);
} finally {
for (int i = 0; i < this.concurrency; i++) {
try {this.jsonline.put(JsonReader.POISON_JSON_MAP);} catch (InterruptedException e) {}
}
}
}
/**
* The JsonHandle class is a bundle of a json with the information about the
* seek location in the file and the length of bytes of the original json string
*/
public static class JsonHandle implements JsonFactory {
private JSONObject json;
private long index;
private int length;
public JsonHandle(JSONObject json, long index, int length) {
this.json = json;
this.index = index;
this.length = length;
}
public JSONObject getJSON() {
return json;
}
public long getIndex() {
return index;
}
public int getLength() {
return length;
}
public String toString() {
return new JSONObject(this.json).toString();
}
}
public JsonFactory getJsonFactory(long index, int length) {
return new ReaderJsonFactory(index, length);
}
public class ReaderJsonFactory implements JsonFactory {
private long index;
private int length;
public ReaderJsonFactory(long index, int length) {
this.index = index;
this.length = length;
}
@Override
public JSONObject getJSON() throws IOException {
byte[] b = new byte[this.length];
JsonRandomAccessFile.this.read(b, this.index);
return new JSONObject(UTF8.String(b));
}
public long getIndex() {
return this.index;
}
public int getLength() {
return this.length;
}
public File getFile() {
return JsonRandomAccessFile.this.file;
}
public String toString() {
try {
return this.getJSON().toString();
} catch (IOException e) {
Log.getLog().warn(e);
return "";
}
}
}
public void close() throws IOException {
super.close();
}
}