/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package ro.nextreports.engine.exporter;
import java.io.PrintStream;
import java.io.UnsupportedEncodingException;
import java.util.HashSet;
import java.util.Set;
import ro.nextreports.engine.Report;
import ro.nextreports.engine.band.Band;
import ro.nextreports.engine.band.BandElement;
import ro.nextreports.engine.band.Hyperlink;
import ro.nextreports.engine.band.HyperlinkBandElement;
import ro.nextreports.engine.band.ReportBandElement;
import ro.nextreports.engine.exporter.CellElement;
import ro.nextreports.engine.exporter.ExporterBean;
import ro.nextreports.engine.exporter.ResultExporter;
import ro.nextreports.engine.queryexec.QueryException;
import ro.nextreports.engine.util.StringUtil;
/**
* @author daniel.avieritei
*/
public class JSONSimpleExporter extends ResultExporter {
private PrintStream stream;
private char separator = ',';
/**
* quote character, usually '\"' '\'' for SOL used to enclose fields
* containing a separator character.
*/
private char quote = '\"';
/**
* line separator to use. We use Windows style for all platforms since csv
* is a Windows format file.
*/
private static final String lineSeparator = "]} \r\n";
/**
* true if there has was a field previously written to this line, meaning
* there is a comma pending to be written.
*/
private boolean wasPreviousField = false;
/**
* how much extra quoting you want
*/
private int quoteLevel = 1;
/**
* true if write should trim lead/trail whitespace from fields before
* writing them.
*/
private final boolean trim = true;
private boolean isFirstRow;
public JSONSimpleExporter(ExporterBean bean) {
super(bean);
}
public JSONSimpleExporter(ExporterBean bean, char delimiter) {
super(bean);
separator = delimiter;
}
protected void initExport() throws QueryException {
stream = createPrintStream();
stream.print("{" + quote + "rows" + quote + " : [");
isFirstRow = true;
}
protected void finishExport() {
stream.print("]}");
stream.flush();
if (!bean.isSubreport()) {
stream.close();
}
}
protected void close() {
if (!bean.isSubreport()) {
stream.close();
}
}
protected void flush() {
if (resultSetRow % FLUSH_ROWS == 0) {
flushNow();
}
}
protected void flushNow() {
stream.flush();
}
protected Set<CellElement> getIgnoredCells(Band band) {
return new HashSet<CellElement>();
}
protected void exportCell(String bandName, BandElement bandElement, Object value, int gridRow, int row, int column,
int cols, int rowSpan, int colSpan, boolean isImage) {
if (bandElement instanceof HyperlinkBandElement) {
Hyperlink hyperlink = ((HyperlinkBandElement) bandElement).getHyperlink();
value = hyperlink.getText();
} else if (bandElement instanceof ReportBandElement) {
Report report = ((ReportBandElement) bandElement).getReport();
ExporterBean eb = null;
try {
eb = getSubreportExporterBean(report);
JSONSimpleExporter subExporter = new JSONSimpleExporter(eb);
subExporter.export();
value = subExporter.getSubreportData();
} catch (Exception e) {
e.printStackTrace();
} finally {
if ((eb != null) && (eb.getResult() != null)) {
eb.getResult().close();
}
}
}
put(stream, StringUtil.getValueAsString(value, getPattern(bandElement), getReportLanguage()));
if (column == cols - 1) {
nl(stream);
}
}
protected void afterRowExport() {
}
protected String getNullElement() {
return null;
}
/**
* Write one csv field to the file, followed by a separator unless it is the
* last field on the line. Lead and trailing blanks will be removed.
*
* @param p
* print stream
* @param s
* The string to write. Any additional quotes or embedded quotes
* will be provided by put. Null means start a new line.
*/
private void put(PrintStream p, String s) {
if (s == null) {
// nl();
put(p, " ");
return;
}
if (wasPreviousField) {
p.print(separator);
} else {
// first character
//do not put at first row?
if (isFirstRow){
p.print("{\"row\": [");
isFirstRow = false;
} else {
p.print(",{\"row\": [");
}
}
if (trim) {
s = s.trim();
}
if (s.indexOf(quote) >= 0) {
/*
* worst case, needs surrounding quotes and internal quotes doubled
*/
p.print(quote);
for (int i = 0; i < s.length(); i++) {
char c = s.charAt(i);
if (c == quote) {
p.print(quote);
p.print(quote);
} else {
p.print(c);
}
}
p.print(quote);
} else if (quoteLevel == 2 || quoteLevel == 1 && s.indexOf(' ') >= 0 || s.indexOf(separator) >= 0) {
/* need surrounding quotes */
p.print(quote);
p.print(s);
p.print(quote);
} else {
/* ordinary case, no surrounding quotes needed */
p.print(quote);
p.print(s);
p.print(quote);
}
/* make a note to print trailing comma later */
wasPreviousField = true;
}
/**
* Write a new line in the CVS output file to demark the end of record.
*
* @param p
* print stream
*/
public void nl(PrintStream p) {
/* don't bother to write last pending comma on the line */
p.print(lineSeparator);
wasPreviousField = false;
}
private String getSubreportData() {
try {
return subreportStream.toString("UTF-8");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
return "";
} finally {
stream.close();
}
}
public PrintStream getStream() {
return stream;
}
}