raimund@245: package de.intevation.flys.client.server; raimund@245: raimund@245: import java.util.ArrayList; teichmann@5567: import java.util.Arrays; teichmann@5567: import java.util.Comparator; raimund@245: import java.util.List; raimund@245: raimund@245: import javax.xml.xpath.XPathConstants; raimund@245: raimund@245: import org.w3c.dom.Document; raimund@245: import org.w3c.dom.Element; raimund@245: import org.w3c.dom.Node; raimund@245: import org.w3c.dom.NodeList; raimund@245: ingo@1367: import org.apache.log4j.Logger; ingo@1367: raimund@245: import com.google.gwt.user.server.rpc.RemoteServiceServlet; raimund@245: raimund@245: import de.intevation.artifacts.common.ArtifactNamespaceContext; raimund@245: import de.intevation.artifacts.common.utils.XMLUtils; raimund@245: raimund@245: import de.intevation.artifacts.httpclient.exceptions.ConnectionException; raimund@245: import de.intevation.artifacts.httpclient.http.HttpClient; raimund@245: import de.intevation.artifacts.httpclient.http.HttpClientImpl; raimund@245: raimund@245: import de.intevation.flys.client.shared.exceptions.ServerException; raimund@245: import de.intevation.flys.client.client.services.WQInfoService; raimund@245: import de.intevation.flys.client.shared.model.WQInfoObject; raimund@245: import de.intevation.flys.client.shared.model.WQInfoObjectImpl; raimund@245: raimund@245: raimund@245: /** raimund@245: * @author Ingo Weinzierl raimund@245: */ raimund@245: public class WQInfoServiceImpl raimund@245: extends RemoteServiceServlet raimund@245: implements WQInfoService raimund@245: { ingo@1367: private static final Logger logger = ingo@1367: Logger.getLogger(WQInfoServiceImpl.class); ingo@1367: raimund@245: public static final String ERROR_NO_WQINFO_FOUND = raimund@245: "error_no_wqinfo_found"; raimund@245: raimund@245: public static final String XPATH_WQS = "art:service/art:mainvalues/art:mainvalue"; raimund@245: raimund@245: raimund@245: public WQInfoObject[] getWQInfo( raimund@245: String locale, raimund@245: String river, raimund@245: double from, raimund@245: double to) raimund@245: throws ServerException raimund@245: { ingo@1367: logger.info("WQInfoServiceImpl.getWQInfo"); raimund@245: raimund@1425: String url = getServletContext().getInitParameter("server-url"); raimund@1425: raimund@245: Document doc = XMLUtils.newDocument(); raimund@245: raimund@245: XMLUtils.ElementCreator ec = new XMLUtils.ElementCreator( raimund@245: doc, raimund@245: ArtifactNamespaceContext.NAMESPACE_URI, raimund@245: ArtifactNamespaceContext.NAMESPACE_PREFIX); raimund@245: raimund@245: Element mainvalues = ec.create("mainvalues"); raimund@245: Element riverEl = ec.create("river"); raimund@245: Element startEl = ec.create("start"); raimund@245: Element endEl = ec.create("end"); raimund@245: raimund@245: riverEl.setTextContent(river); raimund@245: startEl.setTextContent(Double.valueOf(from).toString()); raimund@245: endEl.setTextContent(Double.valueOf(to).toString()); raimund@245: raimund@245: mainvalues.appendChild(riverEl); raimund@245: mainvalues.appendChild(startEl); raimund@245: mainvalues.appendChild(endEl); raimund@245: raimund@245: doc.appendChild(mainvalues); raimund@245: raimund@245: HttpClient client = new HttpClientImpl(url, locale); raimund@245: raimund@245: try { raimund@245: Document result = client.callService(url, "mainvalues", doc); raimund@245: ingo@1367: logger.debug("Extract wq info objects now."); raimund@245: WQInfoObject[] objects = extractWQInfoObjects(result); raimund@245: teichmann@5567: if (objects.length > 0) { raimund@245: return objects; raimund@245: } raimund@245: } raimund@245: catch (ConnectionException ce) { ingo@1367: logger.error(ce, ce); raimund@245: } raimund@245: raimund@245: throw new ServerException(ERROR_NO_WQINFO_FOUND); raimund@245: } raimund@245: raimund@245: raimund@245: /** raimund@245: * Extracts all wq info objects from result document. raimund@245: * raimund@245: * @param result The document retrieved by the server. raimund@245: * raimund@245: * @return a list of WQInfoObjects. raimund@245: */ raimund@245: protected WQInfoObject[] extractWQInfoObjects(Document result) raimund@245: throws ServerException raimund@245: { raimund@245: NodeList list = (NodeList) XMLUtils.xpath( raimund@245: result, raimund@245: XPATH_WQS, raimund@245: XPathConstants.NODESET, raimund@245: ArtifactNamespaceContext.INSTANCE); raimund@245: raimund@245: if (list == null || list.getLength() == 0) { ingo@1367: logger.warn("No wq info found."); raimund@245: raimund@245: throw new ServerException(ERROR_NO_WQINFO_FOUND); raimund@245: } raimund@245: teichmann@5567: boolean debug = logger.isDebugEnabled(); teichmann@5567: raimund@245: int num = list.getLength(); teichmann@5567: if (debug) { teichmann@5567: logger.debug("Response contains " + num + " objects."); teichmann@5567: } raimund@245: raimund@245: List objects = raimund@245: new ArrayList(num); raimund@245: raimund@245: for (int i = 0; i < num; i++) { raimund@245: WQInfoObject obj = buildWQInfoObject(list.item(i)); raimund@245: raimund@245: if (obj != null) { raimund@245: objects.add(obj); raimund@245: } raimund@245: } raimund@245: teichmann@5567: if (debug) { teichmann@5567: logger.debug("Retrieved " + objects.size() + " wq values"); teichmann@5567: } raimund@245: teichmann@5567: WQInfoObject [] array = (WQInfoObject[]) teichmann@5567: objects.toArray(new WQInfoObject[objects.size()]); teichmann@5567: teichmann@5567: Arrays.sort(array, WQ_INFO_OBJECT_CMP); teichmann@5567: teichmann@5567: return array; raimund@245: } raimund@245: teichmann@5567: public static final Comparator WQ_INFO_OBJECT_CMP = teichmann@5567: new Comparator() { teichmann@5567: @Override teichmann@5567: public int compare(WQInfoObject a, WQInfoObject b) { teichmann@5567: teichmann@5567: // Descending by type: Qs before Ds teichmann@5567: int cmp = a.getType().compareTo(b.getType()); teichmann@5567: if (cmp < 0) return +1; teichmann@5567: if (cmp > 0) return -1; teichmann@5567: teichmann@5567: // Ascending by value teichmann@5567: double diff = a.getValue() - b.getValue(); teichmann@5567: if (diff < 0d) return -1; teichmann@5567: if (diff > 0d) return +1; teichmann@5567: return 0; teichmann@5567: } teichmann@5567: }; raimund@245: raimund@245: /** raimund@245: * Extracts information for a single wq info object and intializes an raimund@245: * WQInfoObject with them. raimund@245: * raimund@245: * @param node The node that contains the information. raimund@245: * raimund@245: * @return a valid WQInfoObject. raimund@245: */ raimund@245: protected WQInfoObject buildWQInfoObject(Node node) { teichmann@5567: teichmann@5567: // TODO: Replace this expensive XPaths with simpler use of DOM. raimund@245: String name = XMLUtils.xpathString( raimund@245: node, "@name", ArtifactNamespaceContext.INSTANCE); raimund@245: raimund@245: String type = XMLUtils.xpathString( raimund@245: node, "@type", ArtifactNamespaceContext.INSTANCE); raimund@245: raimund@245: String value = XMLUtils.xpathString( raimund@245: node, "@value", ArtifactNamespaceContext.INSTANCE); raimund@245: raimund@245: raimund@245: if (name != null && type != null) { raimund@245: try { raimund@245: return new WQInfoObjectImpl( raimund@245: name, raimund@245: type, raimund@245: new Double(value)); raimund@245: } raimund@245: catch (NumberFormatException nfe) { ingo@1367: logger.warn(nfe.getLocalizedMessage()); raimund@245: } raimund@245: } raimund@245: ingo@1367: logger.warn("Invalid wq info object found."); raimund@245: raimund@245: return null; raimund@245: } raimund@245: } raimund@245: // vim:set ts=4 sw=4 si et sta sts=4 fenc=utf8 :