task_24859 #13

Merged
francesco.mangiacrapa merged 14 commits from task_24859 into master 2023-03-30 11:00:17 +02:00
4 changed files with 293 additions and 275 deletions
Showing only changes of commit 07730153ca - Show all commits

View File

@ -1,5 +1,9 @@
# Changelog for org.gcube.application.cms.sdi-plugins # Changelog for org.gcube.application.cms.sdi-plugins
## [v1.1.0-SNAPSHOT]
- Integrated the field 'geov_link' (Geoportal GisViewer link) in the centroid layer [#24859]
## [v1.0.4] - 2023-03-06 ## [v1.0.4] - 2023-03-06
- Fixed the import from joda-time to java.time - Fixed the import from joda-time to java.time
- [#24702] Fixed the default-lc-managers dependency - [#24702] Fixed the default-lc-managers dependency

View File

@ -5,7 +5,7 @@
<modelVersion>4.0.0</modelVersion> <modelVersion>4.0.0</modelVersion>
<artifactId>sdi-plugins</artifactId> <artifactId>sdi-plugins</artifactId>
<version>1.0.4</version> <version>1.1.0-SNAPSHOT</version>
<name>gCube CMS - SDI Plugins</name> <name>gCube CMS - SDI Plugins</name>
@ -85,12 +85,19 @@
<version>1.14</version> <version>1.14</version>
</dependency> </dependency>
<dependency>
<groupId>org.gcube.portlets.user</groupId>
<artifactId>uri-resolver-manager</artifactId>
<version>[1.0.0,2.0.0-SNAPSHOT)</version>
</dependency>
<dependency> <dependency>
<groupId>org.gcube.application.cms</groupId> <groupId>org.gcube.application.cms</groupId>
<artifactId>cms-test-commons</artifactId> <artifactId>cms-test-commons</artifactId>
<scope>test</scope> <scope>test</scope>
</dependency> </dependency>
</dependencies> </dependencies>
<build> <build>

View File

@ -19,6 +19,8 @@ public class DBConstants {
public static final String DISPLAYED="displayed_project"; public static final String DISPLAYED="displayed_project";
public static final String GEOVIEWER_LINK_FIELD="geov_link";
public static final String XCOORD_FIELD="xcoord"; public static final String XCOORD_FIELD="xcoord";
public static final String YCOORD_FIELD="ycoord"; public static final String YCOORD_FIELD="ycoord";

View File

@ -33,6 +33,9 @@ import org.gcube.application.geoportal.common.model.plugins.IndexerPluginDescrip
import org.gcube.application.geoportal.common.model.plugins.PluginDescriptor; import org.gcube.application.geoportal.common.model.plugins.PluginDescriptor;
import org.gcube.application.geoportal.common.model.rest.ConfigurationException; import org.gcube.application.geoportal.common.model.rest.ConfigurationException;
import org.gcube.application.geoportal.common.model.useCaseDescriptor.UseCaseDescriptor; import org.gcube.application.geoportal.common.model.useCaseDescriptor.UseCaseDescriptor;
import org.gcube.portlets.user.uriresolvermanager.UriResolverManager;
import org.gcube.portlets.user.uriresolvermanager.resolvers.query.GeoportalResolverQueryStringBuilder;
import org.gcube.portlets.user.uriresolvermanager.resolvers.query.GeoportalResolverQueryStringBuilder.RESOLVE_AS;
import org.geojson.Crs; import org.geojson.Crs;
import org.geojson.GeoJsonObject; import org.geojson.GeoJsonObject;
import org.geojson.LngLatAlt; import org.geojson.LngLatAlt;
@ -42,283 +45,285 @@ import java.sql.SQLException;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.Collection; import java.util.Collection;
import java.util.List; import java.util.List;
import java.util.Map;
@Slf4j @Slf4j
public class SDIIndexerPlugin extends SDIAbstractPlugin implements IndexerPluginInterface { public class SDIIndexerPlugin extends SDIAbstractPlugin implements IndexerPluginInterface {
static final PluginDescriptor DESCRIPTOR = new PluginDescriptor(Constants.INDEXER_PLUGIN_ID,
IndexerPluginDescriptor.INDEXER);
static final ArrayList<BBOXEvaluator> BBOX_EVALUATORS = new ArrayList<>();
static {
DESCRIPTOR.setDescription("SDI Indexer. " + "Manage Centroids layers.");
DESCRIPTOR.setVersion(new Semver("1.0.0"));
BBOX_EVALUATORS.add(new BBOXPathScanner());
BBOX_EVALUATORS.add(new BBOXByCoordinatePaths());
}
@Override
public PluginDescriptor getDescriptor() {
return DESCRIPTOR;
}
static final PluginDescriptor DESCRIPTOR=new PluginDescriptor(Constants.INDEXER_PLUGIN_ID, @Override
IndexerPluginDescriptor.INDEXER); public InitializationReport initInContext() throws InitializationException {
InitializationReport report = new InitializationReport();
report.setStatus(Report.Status.OK);
return report;
}
static final ArrayList<BBOXEvaluator> BBOX_EVALUATORS=new ArrayList<>();
/**
static { * Expected parameters : - indexName (unique) - workspace - flagInternalIndex:
DESCRIPTOR.setDescription("SDI Indexer. " + * boolean - centroidRecord (OPT)
"Manage Centroids layers."); *
DESCRIPTOR.setVersion(new Semver("1.0.0")); * @param request
* @return
BBOX_EVALUATORS.add(new BBOXPathScanner()); */
BBOX_EVALUATORS.add(new BBOXByCoordinatePaths());
@Override
public IndexDocumentReport index(IndexDocumentRequest request) throws InvalidPluginRequestException {
} log.info("Indexer {} : Serving Index Request {} ", this.getDescriptor().getId(), request);
@Override Project project = request.getDocument();
public PluginDescriptor getDescriptor() { UseCaseDescriptor useCaseDescriptor = request.getUseCaseDescriptor();
return DESCRIPTOR; Document requestArguments = request.getCallParameters();
}
IndexDocumentReport report = new IndexDocumentReport(request);
@Override try {
public InitializationReport initInContext() throws InitializationException { // ********* INIT INDEX
InitializationReport report = new InitializationReport(); // TODO CACHE
report.setStatus(Report.Status.OK); PostgisIndexer indexer = getIndexer(useCaseDescriptor, requestArguments);
return report;
} Document profileConfiguration = getConfigurationFromProfile(useCaseDescriptor).getConfiguration();
log.debug("UseCaseDescriptor Configuration is {} ", profileConfiguration);
/**
* Expected parameters : // ************* PREPARE RECORD
* - indexName (unique)
* - workspace JSONPathWrapper documentNavigator = new JSONPathWrapper(Serialization.write(project));
* - centroidRecord (OPT)
* Document centroidDoc = new Document();
* @param request if (requestArguments.containsKey("centroidRecord"))
* @return centroidDoc.putAll(requestArguments.get("centroidRecords", Document.class));
*/ // DEFAULT VALUES
centroidDoc.put(DBConstants.Defaults.PROJECT_ID, project.getId());
@Override centroidDoc.put(DBConstants.Defaults.DISPLAYED, true);
public IndexDocumentReport index(IndexDocumentRequest request) throws InvalidPluginRequestException {
boolean isInternalIndex = false;
log.info("Indexer {} : Serving Index Request {} ",this.getDescriptor().getId(),request); try {
isInternalIndex = requestArguments.getBoolean("flagInternalIndex");
Project project =request.getDocument(); log.debug("flagInternalIndex read as {} ", isInternalIndex);
UseCaseDescriptor useCaseDescriptor = request.getUseCaseDescriptor(); } catch (Exception e) {
Document requestArguments=request.getCallParameters(); // TODO: handle exception
}
IndexDocumentReport report= new IndexDocumentReport(request); log.info("flagInternalIndex is {} ", isInternalIndex);
try {
log.debug("Trying to generate Geoportal Gis Link...");
try{ //Contacting the Geoportal-Resolver via UriResolverManager
// ********* INIT INDEX UriResolverManager uriResolverManager = new UriResolverManager("GEO");
// TODO CACHE GeoportalResolverQueryStringBuilder builder = new GeoportalResolverQueryStringBuilder(project.getProfileID(),project.getId());
PostgisIndexer indexer = getIndexer(useCaseDescriptor,requestArguments); builder.scope(request.getContext().getId());
Document profileConfiguration =getConfigurationFromProfile(useCaseDescriptor).getConfiguration(); if(isInternalIndex) {
log.debug("UseCaseDescriptor Configuration is {} ",profileConfiguration); builder.resolverAs(RESOLVE_AS.PRIVATE);
}else {
builder.resolverAs(RESOLVE_AS.PUBLIC);
// ************* PREPARE RECORD }
Map<String, String> params = builder.buildQueryParameters();
String shortLink = uriResolverManager.getLink(params, true);
JSONPathWrapper documentNavigator=new JSONPathWrapper(Serialization.write(project)); log.info("Geoportal GisViewer link is {} ", shortLink);
centroidDoc.put(DBConstants.Defaults.GEOVIEWER_LINK_FIELD, shortLink);
Document centroidDoc = new Document(); }catch (Exception e) {
if(requestArguments.containsKey("centroidRecord")) // TODO: handle exception
centroidDoc.putAll(requestArguments.get("centroidRecords",Document.class)); }
// DEFAULT VALUES
centroidDoc.put(DBConstants.Defaults.PROJECT_ID, project.getId());
centroidDoc.put(DBConstants.Defaults.DISPLAYED,true); // ********************** EVALAUTE POSITION
log.debug("indexing UseCaseDescriptor {} : Evaluating Centroid... ", useCaseDescriptor.getId());
SpatialReference reference = null;
List<IdentificationReference> refs = project
// ********************** EVALAUTE POSITION .getIdentificationReferenceByType(SpatialReference.SPATIAL_REFERENCE_TYPE);
log.debug("indexing UseCaseDescriptor {} : Evaluating Centroid... ", useCaseDescriptor.getId()); if (!refs.isEmpty()) {
SpatialReference reference =null;
List<IdentificationReference> refs=project.getIdentificationReferenceByType(SpatialReference.SPATIAL_REFERENCE_TYPE); // Use existing Reference
if(!refs.isEmpty()){
reference = Serialization.convert(refs.get(0), SpatialReference.class);
// Use existing Reference
log.debug("Using already defined spatial reference " + reference);
reference = Serialization.convert(refs.get(0), SpatialReference.class);
GeoJsonObject object = Serialization.convert(reference.getGeoJson(), GeoJsonObject.class);
log.debug("Using already defined spatial reference " + reference);
GCubeSDILayer.BBOX bbox = GCubeSDILayer.BBOX.fromGeoJSON(object.getBbox());
GeoJsonObject object = Serialization.convert(reference.getGeoJson(), GeoJsonObject.class); log.info("Found declared BBOX {} ", bbox);
Double pointX = (bbox.getMaxX() + bbox.getMinX()) / 2;
GCubeSDILayer.BBOX bbox = GCubeSDILayer.BBOX.fromGeoJSON(object.getBbox()); Double pointY = (bbox.getMaxY() + bbox.getMinY()) / 2;
String wkt = String.format("POINT (%1$f %2$f) ", pointX, pointY);
log.info("Found declared BBOX {} ", bbox);
Double pointX = (bbox.getMaxX() + bbox.getMinX())/2; centroidDoc.put("geom", wkt);
Double pointY = (bbox.getMaxY() + bbox.getMinY())/2;
String wkt = String.format("POINT (%1$f %2$f) ", } else {
pointX, pointY); // unable to use current Spatial reference, try evaluating it
log.debug("UseCaseDescriptor {} : Getting evaluation paths from useCaseDescriptor.. ",
useCaseDescriptor.getId());
centroidDoc.put("geom", wkt);
// for each configuration option try until found
} else{ GCubeSDILayer.BBOX toSet = null;
// unable to use current Spatial reference, try evaluating it for (BBOXEvaluator evaluator : BBOX_EVALUATORS) {
log.debug("UseCaseDescriptor {} : Getting evaluation paths from useCaseDescriptor.. ", useCaseDescriptor.getId()); log.trace("UCD {}, Project {}. Evaluating BBOX with {}", useCaseDescriptor.getId(), project.getId(),
evaluator);
// for each configuration option try until found try {
GCubeSDILayer.BBOX toSet = null; if (evaluator.isConfigured(profileConfiguration)) {
for(BBOXEvaluator evaluator : BBOX_EVALUATORS){ toSet = evaluator.evaluate(profileConfiguration, useCaseDescriptor, documentNavigator);
log.trace("UCD {}, Project {}. Evaluating BBOX with {}",useCaseDescriptor.getId(),project.getId(),evaluator); if (toSet != null) {
try{ log.info("UCD {}, Project {}. Evaluated BBOX {} with method {}",
if(evaluator.isConfigured(profileConfiguration)){ useCaseDescriptor.getId(), project.getId(), toSet, evaluator);
toSet=evaluator.evaluate(profileConfiguration,useCaseDescriptor,documentNavigator); break;
if(toSet!=null) { }
log.info("UCD {}, Project {}. Evaluated BBOX {} with method {}", }
useCaseDescriptor.getId(),project.getId(),toSet,evaluator); } catch (Throwable t) {
break; log.warn("UCD {}, Project {}. Exception with {}", useCaseDescriptor.getId(), project.getId(),
} evaluator, t);
} }
}catch (Throwable t){ }
log.warn("UCD {}, Project {}. Exception with {}", if (toSet == null)
useCaseDescriptor.getId(),project.getId(),evaluator,t); throw new IndexingException("No BBOX has been evaluated from project");
}
} Double pointX = (toSet.getMaxX() + toSet.getMinX()) / 2;
if(toSet== null) Double pointY = (toSet.getMaxY() + toSet.getMinY()) / 2;
throw new IndexingException("No BBOX has been evaluated from project"); log.info("Evaluated BBOX {} ", toSet);
String wkt = String.format("POINT (%1$f %2$f) ", pointX, pointY);
Double pointX=(toSet.getMaxX()+toSet.getMinX())/2; // TODO support altitude
Double pointY = (toSet.getMaxY()+toSet.getMinY())/2; Double pointZ = 0d;
log.info("Evaluated BBOX {} ",toSet);
String wkt = String .format("POINT (%1$f %2$f) ", centroidDoc.put("geom", wkt);
pointX, pointY);
//TODO support altitude Point point = new Point();
Double pointZ= 0d; point.setCoordinates(new LngLatAlt(pointX, pointY, pointZ));
point.setBbox(toSet.asGeoJSONArray());
centroidDoc.put("geom",wkt); // TODO Manage CRS
point.setCrs(new Crs());
Point point = new Point(); reference = new SpatialReference(Serialization.asDocument(point));
point.setCoordinates(new LngLatAlt(pointX,pointY,pointZ)); log.info("UCD {} project {}, Setting Spatial Reference {} ", useCaseDescriptor.getId(), project.getId(),
point.setBbox(toSet.asGeoJSONArray()); Serialization.write(reference));
report.addIdentificationReference(reference);
//TODO Manage CRS }
point.setCrs(new Crs());
reference = new SpatialReference(Serialization.asDocument(point)); // *********** Additional Values from useCaseDescriptor
log.info("UCD {} project {}, Setting Spatial Reference {} ",useCaseDescriptor.getId(),project.getId(),Serialization.write(reference));
report.addIdentificationReference(reference); log.info("Setting additional values to centroid from mappings ..");
} for (MappingObject m : getMappings(useCaseDescriptor)) {
List<Object> foundValues = documentNavigator.getByPath(m.getPath());
Object toSetValue = null;
if (!foundValues.isEmpty()) {
//*********** Additional Values from useCaseDescriptor // NB CSV for multiple values
StringBuilder b = new StringBuilder();
log.info("Setting additional values to centroid from mappings .."); foundValues.forEach(o -> {
for(MappingObject m : getMappings(useCaseDescriptor)){ // Parser returns list of list
List<Object> foundValues = documentNavigator.getByPath(m.getPath()); if (o instanceof Collection)
Object toSetValue=null; ((Collection<?>) o).forEach(v -> b.append(v + ","));
if(!foundValues.isEmpty()) { else
// NB CSV for multiple values b.append(o + ",");
StringBuilder b=new StringBuilder(); });
foundValues.forEach(o-> { b.deleteCharAt(b.length() - 1);
// Parser returns list of list toSetValue = b.toString();
if (o instanceof Collection) ((Collection<?>) o).forEach(v ->b.append(v + ",")); }
else b.append(o+","); log.trace("Setting {} = {} in centroid doc ", m.getName(), toSetValue);
}); centroidDoc.put(m.getName(), toSetValue);
b.deleteCharAt(b.length()-1); }
toSetValue = b.toString();
} log.info("Inserting Centroid {} into {} ", Serialization.write(centroidDoc.toJson()), indexer);
log.trace("Setting {} = {} in centroid doc ",m.getName(),toSetValue); indexer.insert(centroidDoc);
centroidDoc.put(m.getName(),toSetValue);
} // Support to HIDE AND DISPLAY as requested by invoker
if (requestArguments.containsKey("_toHideIds")) {
log.info("Inserting Centroid {} into {} ",Serialization.write(centroidDoc.toJson()),indexer);
indexer.insert(centroidDoc); List<String> ids = Serialization.convert(requestArguments.get("_toHideIds"), List.class);
log.info("Requested to hide centroids {} ", ids);
// Support to HIDE AND DISPLAY as requested by invoker indexer.updateIsVisible(false, ids);
if(requestArguments.containsKey("_toHideIds")){ }
List<String> ids = Serialization.convert(requestArguments.get("_toHideIds"),List.class); if (requestArguments.containsKey("_toDisplayIds")) {
log.info("Requested to hide centroids {} ",ids); List<String> ids = Serialization.convert(requestArguments.get("_toDisplayIds"), List.class);
indexer.updateIsVisible(false,ids);
} log.info("Requested to display centroids {} ", ids);
indexer.updateIsVisible(true, ids);
if(requestArguments.containsKey("_toDisplayIds")){ }
List<String> ids = Serialization.convert(requestArguments.get("_toDisplayIds"),List.class);
report.setStatus(Report.Status.OK);
log.info("Requested to display centroids {} ",ids); } catch (SDIInteractionException e) {
indexer.updateIsVisible(true,ids); log.error("Unable to index " + request, e);
} report.setStatus(Report.Status.ERROR);
report.putMessage(e.getMessage());
} catch (Throwable t) {
log.error("Unable to index " + request, t);
report.setStatus(Report.Status.OK); report.setStatus(Report.Status.ERROR);
}catch (SDIInteractionException e){ report.putMessage(t.getMessage());
log.error("Unable to index "+request,e); } finally {
report.setStatus(Report.Status.ERROR); return report;
report.putMessage(e.getMessage()); }
}catch (Throwable t){ }
log.error("Unable to index "+request,t);
report.setStatus(Report.Status.ERROR); private List<MappingObject> getMappings(UseCaseDescriptor useCaseDescriptor) throws InvalidProfileException {
report.putMessage(t.getMessage()); return MappingObject.getMappingsFromUCD(useCaseDescriptor, getDescriptor().getId());
}finally{ }
return report;
} @Override
} public IndexDocumentReport deindex(IndexDocumentRequest request) throws InvalidPluginRequestException {
log.info("Indexer {} : Serving Index Request {} ", this.getDescriptor().getId(), request);
private List<MappingObject> getMappings(UseCaseDescriptor useCaseDescriptor) throws InvalidProfileException { IndexDocumentReport report = new IndexDocumentReport(request);
return MappingObject.getMappingsFromUCD(useCaseDescriptor, getDescriptor().getId()); try {
} PostgisIndexer indexer = getIndexer(request.getUseCaseDescriptor(), request.getCallParameters());
indexer.removeByFieldValue(PostgisIndexer.StandardFields.PROJECT_ID, request.getDocument().getId());
@Override } catch (SDIInteractionException e) {
public IndexDocumentReport deindex(IndexDocumentRequest request) throws InvalidPluginRequestException { log.error("Unable to index " + request, e);
log.info("Indexer {} : Serving Index Request {} ",this.getDescriptor().getId(),request); report.setStatus(Report.Status.ERROR);
IndexDocumentReport report= new IndexDocumentReport(request); report.putMessage(e.getMessage());
try{ } catch (Throwable t) {
PostgisIndexer indexer = getIndexer(request.getUseCaseDescriptor(),request.getCallParameters()); log.error("Unable to index " + request, t);
indexer.removeByFieldValue(PostgisIndexer.StandardFields.PROJECT_ID,request.getDocument().getId()); report.setStatus(Report.Status.ERROR);
}catch (SDIInteractionException e){ report.putMessage(t.getMessage());
log.error("Unable to index "+request,e); } finally {
report.setStatus(Report.Status.ERROR); return report;
report.putMessage(e.getMessage()); }
}catch (Throwable t){ }
log.error("Unable to index "+request,t);
report.setStatus(Report.Status.ERROR); /**
report.putMessage(t.getMessage()); * Expected parameters : workspace indexName
}finally{ *
return report; * @param request
} * @return
} * @throws ConfigurationException
*/
/** @Override
* Expected parameters : public Index getIndex(BaseRequest request) throws ConfigurationException {
* workspace try {
* indexName return getIndexer(request.getUseCaseDescriptor(), request.getCallParameters()).getIndexConfiguration();
* } catch (Throwable t) {
* @param request throw new ConfigurationException("Unable to get Postgis index for ucd "
* @return + request.getUseCaseDescriptor().getId() + " in " + request.getContext(), t);
* @throws ConfigurationException }
*/ }
@Override
public Index getIndex(BaseRequest request) throws ConfigurationException { // Inits index
try { // TODO CACHE
return getIndexer(request.getUseCaseDescriptor(), request.getCallParameters()).getIndexConfiguration(); private PostgisIndexer getIndexer(UseCaseDescriptor ucd, Document params)
}catch(Throwable t ){ throws ConfigurationException, SQLException, InvalidProfileException, SDIInteractionException {
throw new ConfigurationException("Unable to get Postgis index for ucd "+request.getUseCaseDescriptor().getId()+" in "+ request.getContext(),t); PostgisIndexer indexer = new PostgisIndexer(sdiCache.getObject(), ucd, postgisCache.getObject());
}
} List<MappingObject> mappingObjects = getMappings(ucd);
List<PostgisTable.Field> fields = PostgisTable.Field.fromMappings(mappingObjects);
// Inits index
// TODO CACHE indexer.initIndex(params.getString("indexName"), fields, params.getString("workspace"),
private PostgisIndexer getIndexer(UseCaseDescriptor ucd,Document params) throws ConfigurationException, SQLException, InvalidProfileException, SDIInteractionException { params.getString("indexName"));
PostgisIndexer indexer = new PostgisIndexer(sdiCache.getObject(), ucd, postgisCache.getObject()); return indexer;
}
List<MappingObject> mappingObjects = getMappings(ucd);
List<PostgisTable.Field> fields = PostgisTable.Field.fromMappings(mappingObjects);
indexer.initIndex(params.getString("indexName"),
fields,
params.getString("workspace"),
params.getString("indexName"));
return indexer;
}
} }