/**
* Copyright (c) 2011-2014, OpenIoT
*
* This file is part of OpenIoT.
*
* OpenIoT 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, version 3 of the License.
*
* OpenIoT 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 OpenIoT. If not, see <http://www.gnu.org/licenses/>.
*
* Contact: OpenIoT mailto: info@openiot.eu
* @author Ali Salehi
* @author Mehdi Riahi
* @author Timotee Maret
* @author Milos Stojanovic
*/
package org.openiot.gsn.http.datarequest;
import org.openiot.gsn.Main;
import org.openiot.gsn.beans.StreamElement;
import org.openiot.gsn.http.MultiDataDownload;
import org.openiot.gsn.storage.DataEnumerator;
import java.io.OutputStream;
import java.io.PrintWriter;
import java.sql.Connection;
import java.sql.SQLException;
import java.util.Calendar;
import java.util.Date;
import java.util.Iterator;
import java.util.Map;
import java.util.Map.Entry;
import org.apache.log4j.Logger;
public class DownloadData extends AbstractDataRequest {
private static transient Logger logger = Logger.getLogger(MultiDataDownload.class);
private static final String PARAM_OUTPUT_TYPE = "outputtype";
public enum AllowedOutputType {
csv,
xml
}
private AllowedOutputType ot;
private String csvDelimiter = ",";
public DownloadData(Map<String, String[]> requestParameters) throws DataRequestException {
super(requestParameters);
}
@Override
public void process() throws DataRequestException {
String outputType = QueriesBuilder.getParameter(requestParameters, PARAM_OUTPUT_TYPE);
try {
if (outputType == null) {
throw new DataRequestException("The following >" + PARAM_OUTPUT_TYPE + "< parameter is missing in your query.");
}
ot = AllowedOutputType.valueOf(outputType);
if (ot == AllowedOutputType.csv) {
//
if (QueriesBuilder.getParameter(requestParameters, "delimiter") != null && !QueriesBuilder.getParameter(requestParameters, "delimiter").equals("")) {
String reqdelimiter = QueriesBuilder.getParameter(requestParameters, "delimiter");
if (reqdelimiter.equals("tab")) {
csvDelimiter = "\t";
} else if (reqdelimiter.equals("space")) {
csvDelimiter = " ";
} else if (reqdelimiter.equals("semicolon")) {
csvDelimiter = ";";
} else if (reqdelimiter.equals("other") && QueriesBuilder.getParameter(requestParameters, "otherdelimiter") != null && !QueriesBuilder.getParameter(requestParameters, "otherdelimiter").equals("")) {
csvDelimiter = QueriesBuilder.getParameter(requestParameters, "otherdelimiter");
}
}
}
}
catch (IllegalArgumentException e) {
throw new DataRequestException("The >" + outputType + "< output type is not supported.");
}
}
// public String outputResult() {
// ByteArrayOutputStream baos = new ByteArrayOutputStream();
// outputResult(baos);
// return baos.toString();
// }
@Override
public void outputResult(OutputStream os) {
PrintWriter respond = new PrintWriter(os);
Iterator<Entry<String, AbstractQuery>> iter = qbuilder.getSqlQueries().entrySet().iterator();
Entry<String, AbstractQuery> nextSqlQuery;
DataEnumerator de = null;
try {
if (ot == AllowedOutputType.xml) {
respond.println("<result>");
}
while (iter.hasNext()) {
nextSqlQuery = iter.next();
Connection connection = null;
connection = Main.getStorage(nextSqlQuery.getKey()).getConnection();
de = Main.getStorage(nextSqlQuery.getKey()).streamedExecuteQuery(nextSqlQuery.getValue(), true, connection);
logger.debug("Data Enumerator: " + de);
if (ot == AllowedOutputType.csv) {
respond.println("##vsname:" + nextSqlQuery.getKey());
respond.println("##query:" + nextSqlQuery.getValue().getStandardQuery() + (nextSqlQuery.getValue().getLimitCriterion() == null ? "" : "(" + nextSqlQuery.getValue().getLimitCriterion() + ")"));
} else if (ot == AllowedOutputType.xml) {
respond.println("\t<!-- " + nextSqlQuery.getValue().getStandardQuery() + " -->");
respond.println("\t<data vsname=\"" + nextSqlQuery.getKey() + "\">");
}
FieldsCollection fc = qbuilder.getVsnamesAndStreams().get(nextSqlQuery.getKey());
boolean wantTimed = true;
boolean firstLine = true;
while (de.hasMoreElements()) {
if (ot == AllowedOutputType.csv) {
formatCSVElement(respond, de.nextElement(), wantTimed, csvDelimiter, firstLine);
} else if (ot == AllowedOutputType.xml) {
formatXMLElement(respond, de.nextElement(), wantTimed, firstLine);
}
firstLine = false;
}
if (ot == AllowedOutputType.xml)
respond.println("\t</data>");
}
if (ot == AllowedOutputType.xml) {
respond.println("</result>");
}
} catch (SQLException e) {
logger.debug(e.getMessage());
} finally {
respond.flush();
if (de != null)
de.close();
}
}
private void formatCSVElement(PrintWriter respond, StreamElement se, boolean wantTimed, String cvsDelimiter, boolean firstLine) {
if (firstLine) {
respond.print("#");
for (int i = 0; i < se.getData().length; i++) {
respond.print(se.getFieldNames()[i]);
if (i != se.getData().length - 1)
respond.print(cvsDelimiter);
}
if (wantTimed && se.getData().length != 0)
respond.print(cvsDelimiter);
if (wantTimed)
respond.print("timed");
respond.println();
}
for (int i = 0; i < se.getData().length; i++) {
respond.print(se.getData()[i]);
if (i != se.getData().length - 1)
respond.print(cvsDelimiter);
}
if (wantTimed) {
if (se.getData().length != 0)
respond.print(cvsDelimiter);
respond.print(qbuilder.getSdf() == null ? timestampInUTC(se.getTimeStamp()) : qbuilder.getSdf().format(new Date(se.getTimeStamp())));
}
respond.println();
}
private void formatXMLElement(PrintWriter respond, StreamElement se, boolean wantTimed, boolean firstLine) {
if (firstLine) {
respond.println("\t\t<header>");
for (int i = 0; i < se.getData().length; i++) {
respond.println("\t\t\t<field>" + se.getFieldNames()[i] + "</field>");
}
if (wantTimed)
respond.println("\t\t\t<field>timed</field>");
respond.println("\t\t</header>");
}
respond.println("\t\t<tuple>");
for (int i = 0; i < se.getData().length; i++) {
respond.println("\t\t\t<field>" + se.getData()[i] + "</field>");
}
if (wantTimed)
respond.println("\t\t\t<field>" + (qbuilder.getSdf() == null ? timestampInUTC(se.getTimeStamp()) : qbuilder.getSdf().format(new Date(se.getTimeStamp()))) + "</field>");
respond.println("\t\t</tuple>");
}
private long timestampInUTC(long timestamp) {
Calendar cal = Calendar.getInstance();
cal.setTimeInMillis(timestamp);
return cal.getTimeInMillis() + cal.getTimeZone().getOffset(cal.getTimeInMillis());
}
public AllowedOutputType getOt() {
return ot;
}
}