Skip to content

Commit

Permalink
Merge pull request #844 from AtlasOfLivingAustralia/spatial-object-query
Browse files Browse the repository at this point in the history
Spatial object query
  • Loading branch information
adam-collins authored Oct 26, 2023
2 parents 884b91d + 1041d25 commit 618359e
Show file tree
Hide file tree
Showing 8 changed files with 195 additions and 838 deletions.
85 changes: 85 additions & 0 deletions src/main/java/au/org/ala/biocache/dto/SpatialObjectDTO.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,85 @@
package au.org.ala.biocache.dto;

public class SpatialObjectDTO {
Double area_km;
String bbox;
String name;
String description;
String fieldname;
String pid;
String id;
String fid;
String wmsurl;

public Double getArea_km() {
return area_km;
}

public void setArea_km(Double area_km) {
this.area_km = area_km;
}

public String getBbox() {
return bbox;
}

public void setBbox(String bbox) {
this.bbox = bbox;
}

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

public String getDescription() {
return description;
}

public void setDescription(String description) {
this.description = description;
}

public String getFieldname() {
return fieldname;
}

public void setFieldname(String fieldname) {
this.fieldname = fieldname;
}

public String getPid() {
return pid;
}

public void setPid(String pid) {
this.pid = pid;
}

public String getId() {
return id;
}

public void setId(String id) {
this.id = id;
}

public String getFid() {
return fid;
}

public void setFid(String fid) {
this.fid = fid;
}

public String getWmsurl() {
return wmsurl;
}

public void setWmsurl(String wmsurl) {
this.wmsurl = wmsurl;
}
}
26 changes: 26 additions & 0 deletions src/main/java/au/org/ala/biocache/service/AlaLayersService.java
Original file line number Diff line number Diff line change
Expand Up @@ -14,19 +14,25 @@
***************************************************************************/
package au.org.ala.biocache.service;

import au.org.ala.biocache.dto.SpatialObjectDTO;
import com.fasterxml.jackson.core.type.TypeReference;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.cache.annotation.Cacheable;
import org.springframework.scheduling.annotation.Scheduled;
import org.springframework.stereotype.Component;
import org.springframework.util.StreamUtils;
import org.springframework.web.client.RestOperations;

import javax.annotation.PostConstruct;
import javax.inject.Inject;
import java.io.Reader;
import java.net.URL;
import java.net.URLConnection;
import java.net.URLEncoder;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
Expand Down Expand Up @@ -244,4 +250,24 @@ public Reader sample(String[] analysisLayers, double[][] points, Object o) {
// TODO: for on the fly intersection of layers not indexed
return null;
}

@Cacheable("spatialObject")
@Override
public SpatialObjectDTO getObject(String spatialObjectId) {
String url = layersServiceUrl + "/object/" + Integer.parseInt(spatialObjectId);
return restTemplate.getForObject(url, SpatialObjectDTO.class);
}

@Cacheable("wkt")
@Override
public String getObjectWkt(String spatialObjectId) {
String url = layersServiceUrl + "/shape/wkt/" + Integer.parseInt(spatialObjectId);

try {
URLConnection con = new URL(url).openConnection();
return StreamUtils.copyToString(con.getInputStream(), Charset.defaultCharset());
} catch (Exception e) {
return null;
}
}
}
8 changes: 7 additions & 1 deletion src/main/java/au/org/ala/biocache/service/LayersService.java
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
/**************************************************************************
* Copyright (C) 2013 Atlas of Living Australia
* All Rights Reserved.
*
*
* The contents of this file are subject to the Mozilla Public
* License Version 1.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
Expand All @@ -14,6 +14,8 @@
***************************************************************************/
package au.org.ala.biocache.service;

import au.org.ala.biocache.dto.SpatialObjectDTO;

import java.io.Reader;

/**
Expand Down Expand Up @@ -49,4 +51,8 @@ public interface LayersService {
String getLayersServiceUrl();

Reader sample(String[] analysisLayers, double[][] points, Object o);

SpatialObjectDTO getObject(String spatialObjectId);

String getObjectWkt(String spatialObjectId);
}
66 changes: 66 additions & 0 deletions src/main/java/au/org/ala/biocache/util/QueryFormatUtils.java
Original file line number Diff line number Diff line change
Expand Up @@ -71,6 +71,8 @@ public class QueryFormatUtils {
//Patterns that are used to prepare a SOLR query for execution
protected Pattern lsidPattern = Pattern.compile("(^|\\s|\"|\\(|\\[|'|-)taxonConceptID:\"?([a-zA-Z0-9/\\.:\\-_]*)\"?");
protected Pattern speciesListPattern = Pattern.compile("(^|\\s|\"|\\(|\\[|'|-)species_list:\"?(dr[0-9]*)\"?");

protected Pattern spatialObjectPattern = Pattern.compile("(^|\\s|\"|\\(|\\[|'|-)spatialObject:\"?([0-9]*)\"?");
protected Pattern urnPattern = Pattern.compile("\\burn:[a-zA-Z0-9\\.:-]*");
protected Pattern httpPattern = Pattern.compile("http:[a-zA-Z0-9/\\.:\\-_]*");
protected Pattern uidPattern = Pattern.compile("(?:[\"]*)?((?:[a-z_]*_uid:)|(?:[a-zA-Z]*Uid:))(\\w*)(?:[\"]*)?");
Expand Down Expand Up @@ -722,6 +724,68 @@ private boolean formatSpatial(String [] current) throws QidMissingException {
return false;
}

/**
* Insert spatialObject WKT.
*
* If the query string contains spatialObject: replace with the equivalent geohash:Intersects(WKT)
*
* @param current
* @return
* @throws QidMissingException
*/
private void formatSpatialObject(String [] current) {
if (current == null || current.length < 2 || current[1] == null) {
return;
}

//if the query string contains spatialObject: replace with the equivalent geohash:Intersects(WKT)
StringBuffer sb = new StringBuffer();
Matcher m = spatialObjectPattern.matcher(current[1]);
int max = getMaxBooleanClauses();
HashSet<String> failedObjects = new HashSet<>();
while (m.find()) {
String spatialObjectId = m.group(2);
String prefix = m.group(1);
try {
String wkt = layersService.getObjectWkt(spatialObjectId);

if (wkt == null) {
throw new Exception("invalid object id");
}
String q = prefix + spatialField + ":\"Intersects(" + wkt + ")\"";

m.appendReplacement(sb, q);
} catch (Exception e) {
logger.error("failed to get WKT for object: " + spatialObjectId);
m.appendReplacement(sb, prefix + "(NOT *:*)");
failedObjects.add(spatialObjectId);
}
}
m.appendTail(sb);
current[1] = sb.toString();

sb = new StringBuffer();
m = spatialObjectPattern.matcher(current[0]);
while (m.find()) {
String spatialObjectId = m.group(2);
String prefix = m.group(1);
if (failedObjects.contains(spatialObjectId)) {
m.appendReplacement(sb, prefix + "<span class=\"spatialObject failed\" id='" + HtmlEscapers.htmlEscaper().escape(spatialObjectId) + "'>" + HtmlEscapers.htmlEscaper().escape(spatialObjectId) + " (FAILED)</span>");
} else {
try {
SpatialObjectDTO obj = layersService.getObject(spatialObjectId);
String name = obj.getName();
m.appendReplacement(sb, prefix + "<span class='spatialObject' id='" + HtmlEscapers.htmlEscaper().escape(spatialObjectId) + "'>" + HtmlEscapers.htmlEscaper().escape(name) + "</span>");
} catch (Exception e) {
logger.error("Couldn't get spatial object name for " + spatialObjectId, e);
m.appendReplacement(sb, prefix + "<span class='spatialObject' id='" + HtmlEscapers.htmlEscaper().escape(spatialObjectId) + "'>Species list</span>");
}
}
}
m.appendTail(sb);
current[0] = sb.toString();
}

/**
* General formatting for formattedQuery and displayString.
*
Expand Down Expand Up @@ -818,6 +882,8 @@ public String[] formatQueryTerm(String query, SpatialSearchRequestDTO searchPara
formatHttp(formatted);
formatTitleMap(formatted);

formatSpatialObject(formatted);

if (!formatSpatial(formatted)) {
formatGeneral(formatted, searchParams);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -234,7 +234,7 @@ private DownloadStatusDTO download(DownloadRequestDTO requestParams,
//identify this download as too large
File file = new File(downloadService.biocacheDownloadDir + File.separator + UUID.nameUUIDFromBytes(dd.getRequestParams().getEmail().getBytes(StandardCharsets.UTF_8)) + File.separator + dd.getStartTime() + File.separator + "tooLarge");
FileUtils.forceMkdir(file.getParentFile());
FileUtils.writeStringToFile(file, "", "UTF-8");
FileUtils.writeStringToFile(file, requestParams.toString(), "UTF-8");
status.setDownloadUrl(downloadService.biocacheDownloadUrl);
status.setStatus(DownloadStatusDTO.DownloadStatus.TOO_LARGE);
status.setMessage(downloadService.downloadOfflineMsg);
Expand Down
4 changes: 2 additions & 2 deletions src/main/resources/spring.xml
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@
</list>
</property>
</bean>

<!-- MBean configuration for ehcache -->
<bean id="mbeanServer" class="org.springframework.jmx.support.MBeanServerFactoryBean">
<property name="locateExistingServerIfPossible" value="true" />
Expand All @@ -101,4 +101,4 @@
<property name="cacheManager" ref="ehCacheManager" />
</bean>

</beans>
</beans>
4 changes: 3 additions & 1 deletion src/main/webapp/WEB-INF/ehcache.xml
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,8 @@
<cache name="legendCache" maxElementsInMemory="2000" eternal="false" overflowToDisk="false"/>
<cache name="getColours" maxElementsInMemory="2000" eternal="false" overflowToDisk="false"/>
<cache name="fixWkt" maxElementsInMemory="100" eternal="false" overflowToDisk="false"/>
<cache name="spatialObject" maxElementsInMemory="2000" eternal="false" overflowToDisk="false"/>
<cache name="wkt" maxElementsInMemory="2000" eternal="false" overflowToDisk="false"/>
<!-- 30 minuite cache for WMS heatmap SOLR requests -->
<cache name="heatmapCache" maxElementsInMemory="2000" eternal="false" overflowToDisk="false" timeToLiveSeconds="1800"/>
<cache name="speciesListItems" maxElementsInMemory="1000" eternal="false" overflowToDisk="false" timeToLiveSeconds="600"/>
Expand All @@ -15,4 +17,4 @@
<cache name="apiKeys" maxElementsInMemory="100" eternal="false" overflowToDisk="false" timeToLiveSeconds="300" />
<cache name="rateLimit" maxElementsInMemory="1000" eternal="false" overflowToDisk="false" />
<cache name="user-profile" maxElementsInMemory="100" eternal="false" overflowToDisk="false" timeToLiveSeconds="60" />
</ehcache>
</ehcache>
Loading

0 comments on commit 618359e

Please sign in to comment.