inst/actions/geosapi_publish_ogc_services.R

function(action, entity, config){
  
  if(!requireNamespace("geosapi", quietly = TRUE)){
    stop("The 'geosapi-publish-ogc-services' action requires the 'geosapi' package")
  }
  
  #options
  options <- action$options
  createWorkspace <- action$getOption("createWorkspace")
  createStore <- action$getOption("createStore")
  store_basepath <- action$getOption("store_basepath")
  store_basepath_match = attr(regexpr("^/+", store_basepath), "match.length")
  if(store_basepath_match > 0){
    if(store_basepath_match<3){
      store_basepath = paste0(paste0(rep("/",3-store_basepath_match),collapse=""), store_basepath)
    }else{
      store_basepath = paste0("///", substr(store_basepath, start = max(store_basepath_match)+1, nchar(store_basepath)))
    }
  }
  store_description <- action$getOption("store_description")
  overwrite <- action$getOption("overwrite")
  overwrite_upload <- action$getOption("overwrite_upload")
  overwrite_layer <- action$getOption("overwrite_layer")
  if(!overwrite){
    overwrite_upload <- FALSE
    overwrite_layer <- FALSE
  }
  
  #check presence of data
  if(is.null(entity$data)){
    warnMsg <- sprintf("No data object associated to entity '%s'. Skipping data publication!", 
                       entity$identifiers[["id"]])
    config$logger$WARN(warnMsg)
    return(NULL)
  }
  
  data_objects <- list()
  if(length(entity$data$getData())==0){
    data_objects <- list(entity$data)
  }else{
    data_objects <- entity$data$getData()
  }
  
  if(length(data_objects)>0) for(data_object in data_objects){
  
    #datasource
    datasource <- data_object$uploadSource[[1]]
    datasource_name <- NULL
    datasource_file <- NULL
    if(!is.null(datasource)){
      datasource_name <- unlist(strsplit(datasource, "\\."))[1]
      datasource_file <- attr(datasource, "uri")
      attributes(datasource) <- NULL
    }else{
      if(data_object$upload){
        errMsg <- sprintf("Upload source is missing!")
        stop(errMsg)
      }
    }
    
    #shortcut for gs config
    GS_CONFIG <- config$software$output$geoserver_config
    GS <- config$software$output$geoserver
    if(is.null(GS)){
      errMsg <- "This action requires a GeoServer software to be declared in the configuration"
      config$logger$ERROR(errMsg)
      stop(errMsg)
    }
    
    workspace <- GS_CONFIG$properties$workspace
    if(!is.null(data_object$workspaces$geoserver)) workspace <- data_object$workspaces$geoserver
    if(is.null(workspace)){
      errMsg <- "The geoserver configuration requires a workspace for publishing action"
      config$logger$ERROR(errMsg)
      stop(errMsg)
    }
    
    store <- GS_CONFIG$properties$store
    if(!is.null(data_object$store)) store <- data_object$store
    if(is.null(store)){
      errMsg <- "The geoserver configuration requires a data/coverage store for publishing action"
      config$logger$ERROR(errMsg)
      stop(errMsg)
    }
    
    if(data_object$uploadType == "other"){
      warnMsg <- "No 'geosapi' action possible for type 'other'. Action skipped"
      config$logger$WARN(warnMsg)
      return(NULL)
    }
    
    #layername/sourcename
    layername <- if(!is.null(data_object$layername)) data_object$layername else entity$identifiers$id
    layer <- GS$getLayer(layername)
    
    # Check existence of data/coverage store
    the_store <- switch(data_object$spatialRepresentationType,
                        "vector" = GS$getDataStore(workspace, store),
                        "grid" = GS$getCoverageStore(workspace, store)
    )
    # If store does not exist
    # Check if createStore is TRUE
    if(length(the_store)==0){
      if(createStore){
        switch(data_object$uploadType,
               #vector/features upload types
               #===========================================================================================
               #vector/GeoPackage
               #-------------------------------------------------------------------------------------------
               "gpkg"= {
                 the_store<-GSGeoPackageDataStore$new(
                   name = store, 
                   description = store_description , 
                   enabled = TRUE, 
                   database = paste0("file:",store_basepath,"/",workspace,"/",datasource_name,".gpkg")
                 )
               },
               #vector/dbtable
               #-------------------------------------------------------------------------------------------
               "dbtable"= {
                 dbi<-config$software$output$dbi_config
                 if(is.null(dbi)) dbi<-config$software$output$dbi_config
                 if(is.null(dbi)) {
                   errMsg <- sprintf("Error during Geoserver '%s' datastore creation, this datastore type requires a DBI type software declaration in the configuration", store)
                   config$logger$ERROR(errMsg)
                   stop(errMsg)   
                 }
                 Postgres<-dbi$parameters$drv %in% c("Postgres","PostreSQL")
                 if(!Postgres){
                   errMsg <- sprintf("Error during Geoserver '%s' datastore creation, the DBI software declared in the configuration is not a PostGis database", store)
                   config$logger$ERROR(errMsg)
                   stop(errMsg)   
                 }
                 the_store<-GSPostGISDataStore$new(name=store, description = store_description, enabled = TRUE)
                 the_store$setHost(dbi$parameters$host)
                 the_store$setPort(dbi$parameters$port)
                 the_store$setDatabase(dbi$parameters$dbname)
                 #the_store$setSchema()#Not yet implemented in dbi software arguments
                 the_store$setUser(dbi$parameters$user)
                 the_store$setPassword(dbi$parameters$password)
               },
               #vector/dbquery
               #-------------------------------------------------------------------------------------------
               "dbquery"= {
                 dbi<-config$software$output$dbi_config
                 if(is.null(dbi)) dbi<-config$software$output$dbi_config
                 if(is.null(dbi)) {
                   errMsg <- sprintf("Error during Geoserver '%s' datastore creation, this datastore type requires a DBI type software declaration in the configuration", store)
                   config$logger$ERROR(errMsg)
                   stop(errMsg)   
                 }
                 Postgres<-dbi$parameters$drv %in% c("Postgres","PostreSQL")
                 if(!Postgres){
                   errMsg <- sprintf("Error during Geoserver '%s' datastore creation, the DBI software declared in the configuration is not a PostGis database", store)
                   config$logger$ERROR(errMsg)
                   stop(errMsg)   
                 }
                 the_store<-GSPostGISDataStore$new(name=store, description = store_description, enabled = TRUE)
                 the_store$setHost(dbi$parameters$host)
                 the_store$setPort(dbi$parameters$port)
                 the_store$setDatabase(dbi$parameters$dbname)
                 #the_store$setSchema()#Not yet implemented in dbi software arguments
                 the_store$setUser(dbi$parameters$user)
                 the_store$setPassword(dbi$parameters$password)
               },
               #vector/shapefile (ESRI)
               #-------------------------------------------------------------------------------------------
               "shp"= {
                 the_store <- GSShapefileDirectoryDataStore$new(
                   name=store, 
                   description = store_description,
                   enabled = TRUE,
                   url = paste0("file:",store_basepath,"/",workspace)
                 )
               },
               #grid/coverages upload types
               #-----------------------------------------------
               "geotiff" = {
                 the_store <- GSGeoTIFFCoverageStore$new(name = store, description = store_description, enabled = TRUE)
               }
        )
        if(is.null(the_store)){
          errMsg <- sprintf("Error during Geoserver data/coverage store creation, format '%s' not supported. Aborting 'geosapi' action!",data_object$uploadType)
          config$logger$ERROR(errMsg)
          stop(errMsg)      
        }else{
          created <- switch(data_object$spatialRepresentationType,
                            "vector" = GS$createDataStore(workspace, the_store),
                            "grid" = GS$createCoverageStore(workspace, the_store)
          )
          if(created){
            infoMsg <- sprintf("Successful Geoserver '%s' data/coverage store creaction", store)
            config$logger$INFO(infoMsg)
          }else{
            errMsg <- "Error during Geoserver data/coverage store creation. Aborting 'geosapi' action!"
            config$logger$ERROR(errMsg)
            stop(errMsg)
          }
        }
      }else{
        # If createStore is FALSE edit ERROR Message
        errMsg <- sprintf("Data/Coverage store '%s' does not exist and 'createStore' option = FALSE, please verify config if data/coverage store already exists or change createStore = TRUE to create it",store)
        config$logger$ERROR(errMsg)
        stop(errMsg)
      }    
    }
    
    #upload
    #-------------------------------------------------------------------------------------------------
    proceed_with_upload = data_object$upload
    if(!overwrite_upload) proceed_with_upload = FALSE
    if(proceed_with_upload){
      
      config$logger$INFO("Upload mode is set to true")
      if(startsWith(data_object$uploadType,"db") || data_object$uploadType == "other"){
        warnMsg <- "Skipping upload: Upload mode is only valid for types 'shp', 'spatialite' or 'h2'"
        config$logger$WARN(warnMsg)
      }else{
        uploaded <- FALSE
        config$logger$INFO("Upload from local file(s)")
        filepath <- file.path(getwd(), "data", datasource)
        config$logger$INFO("File to upload to Geoserver: %s", filepath)
        if(file.exists(filepath)){
          config$logger$INFO("Upload file '%s' [%s] to GeoServer...", filepath, data_object$uploadType)
          uploaded <- switch(data_object$spatialRepresentationType,
                             #vector/features upload
                             "vector" = switch(data_object$uploadType,
                                "shp" = GS$uploadShapefile(
                                  workspace, store, endpoint = "file", configure = "none", update = "overwrite",
                                  filename = filepath, charset = "UTF-8"
                                ),
                                "gpkg" = GS$uploadGeoPackage(
                                  workspace, store, endpoint = "file", configure = "none", update = "overwrite",
                                  filename = filepath, charset = "UTF-8"
                                )
                             ),
                             #grid/coverages upload
                             "grid" = GS$uploadCoverage(
                               workspace, store, endpoint = "file", configure = "none", update = "overwrite",
                               filename = filepath, extension = data_object$uploadType,
                               contentType = switch(data_object$uploadType,
                                                    "geotiff" = "text/plain",
                                                    "arcgrid" = "text/plain",
                                                    "worldimage" = "application/zip",
                                                    "imagemosaic" = "application/zip"
                               )
                             )
          )
        }else{
          errMsg <- sprintf("Upload from local file(s): no file found for source '%s' (%s)", filepath, datasource)
          config$logger$ERROR(errMsg)
          stop(errMsg)
        }
        
        if(uploaded){
          infoMsg <- sprintf("Successful Geoserver upload for file '%s' (%s)", datasource_file, data_object$uploadType)
          config$logger$INFO(infoMsg)
        }else{
          errMsg <- "Error during Geoserver file upload. Aborting 'geosapi' action!"
          config$logger$ERROR(errMsg)
          stop(errMsg)
        }
      }
    }else{
      config$logger$INFO("Data upload is skipped for layer '%s' (overwrite and/or overwrite_upload is set to FALSE)", layername)
    }
    
    #featuretype/coverage  +layer publication
    #--------------------------------------------------------------------------------------------------
    if(!is.null(data_object$sql)) data_object$uploadType <- "dbquery"
    
    proceed_with_layer <- TRUE
    if(!is.null(layer) & !overwrite_layer) proceed_with_layer <- FALSE
    
    if(proceed_with_layer){
    
      #variables
      epsgCode <- sprintf("EPSG:%s", entity$srid)
      
      #build resource (either featuretype or coverage)
      resource <- switch(data_object$spatialRepresentationType,
                         "vector" = GSFeatureType$new(),
                         "grid" = GSCoverage$new()
      )
      resource$setName(layername)
      nativename <- datasource_name
      if(data_object$uploadType == "dbquery") nativename <- layername
      if(data_object$spatialRepresentationType == "grid") nativename <- store
      resource$setNativeName(nativename)
      abstract <- entity$descriptions$abstract
      if(!is.null(data_object$layerdesc)) abstract = data_object$layerdesc
      resource$setAbstract(abstract)
      title <- entity$titles[["title"]]
      if(length(data_objects)>1) title <- paste0(title, " - ", layername)
      if(!is.null(data_object$layertitle)) title = data_object$layertitle
      resource$setTitle(title)
      resource$setSrs(epsgCode)
      resource$setNativeCRS(epsgCode)
      resource$setEnabled(TRUE)
      resource$setProjectionPolicy("FORCE_DECLARED")
      bbox <- entity$spatial_bbox
      resource$setNativeBoundingBox(bbox$xmin, bbox$ymin, bbox$xmax, bbox$ymax, crs = epsgCode)
      geo_bbox <- entity$geo_bbox
      resource$setLatLonBoundingBox(geo_bbox$xmin, geo_bbox$ymin, geo_bbox$xmax, geo_bbox$ymax, crs = 4326)
      for(subject in entity$subjects){
        kwds <- subject$keywords
        for(kwd in kwds) resource$addKeyword(kwd$name)
      }
      
      #add metadata links
      #in case (only if) geoflow defines either CSW or Geonetwork software, we can add metadata links
      md_link_xml <- NULL
      md_link_html <- NULL
      if(!is.null(config$software$output$csw)|!is.null(config$software$output$geonetwork)){
        meta_id <- entity$identifiers[["id"]]
        if(!is.null(entity$identifiers[["uuid"]])) meta_id <- entity$identifiers[["uuid"]]
        if(!is.null(config$software$output$csw)){
          md_link_xml <- paste0(config$software$output$csw_config$parameters$url, "?service=CSW&request=GetRecordById&Version=", config$software$output$csw_config$parameters$version,
                                "&elementSetName=full&outputSchema=http%3A//www.isotc211.org/2005/gmd&id=", meta_id)
        }
        if(!is.null(config$software$output$geonetwork)){
          md_link_xml <- paste0(config$software$output$geonetwork_config$parameters$url, "/srv/eng/csw?service=CSW&request=GetRecordById&Version=2.0.2",
                                "&elementSetName=full&outputSchema=http%3A//www.isotc211.org/2005/gmd&id=", meta_id)
          if(startsWith(config$software$output$geonetwork_config$parameters$version, "2")){
            md_link_html <- paste0(config$software$output$geonetwork_config$parameters$url, "/srv/en/main.home?uuid=", meta_id)
          }else if(startsWith(config$software$output$geonetwork_config$parameters$version, "3")){
            md_link_html <- paste0(config$software$output$geonetwork_config$parameters$url, "/srv/eng/catalog.search#/metadata/", meta_id)
          }
        }
      }
      if(!is.null(md_link_xml)){
        md_xml <- GSMetadataLink$new(type = "text/xml", metadataType = "ISO19115:2003", content = md_link_xml)
        resource$addMetadataLink(md_xml)
      }
      if(!is.null(md_link_html)){
        md_html <- GSMetadataLink$new(type = "text/html", metadataType = "ISO19115:2003", content = md_link_html)
        resource$addMetadataLink(md_html)
      }
      
      #resource type specific properties
      switch(data_object$spatialRepresentationType,
             "vector" = {
               #cql filter?
               if(!is.null(data_object$cqlfilter)){
                 resource$setCqlFilter(data_object$cqlfilter)
               }
               
               #virtual table?
               if(data_object$uploadType == "dbquery"){
                 vt <- GSVirtualTable$new()
                 vt$setName(layername)
                 vt$setSql(data_object$sql)
                 #if the virtual table is spatialized
                 if(!is.null(data_object$geometryField) & !is.null(data_object$geometryType)){
                   vtg <- GSVirtualTableGeometry$new(
                     name = data_object$geometryField, 
                     type = data_object$geometryType, 
                     srid = entity$srid
                   )
                   vt$setGeometry(vtg)
                 }
                 #if the virtual table has service parameters
                 if(length(data_object$parameters)>0){
                   for(param in data_object$parameters){
                     vtp <- GSVirtualTableParameter$new(
                       name = param$name, 
                       defaultValue = param$defaultvalue, 
                       regexpValidator = param$regexp
                     )
                     vt$addParameter(vtp)
                   }
                 }
                 resource$setVirtualTable(vt)
               }
             },
             "grid" = {
               
               #coverage view?
               if(length(data_object$bands)>0){
                 coview <- GSCoverageView$new()
                 coview$setName(layername)
                 coview$setEnvelopeCompositionType(data_object$envelopeCompositionType)
                 coview$setSelectedResolution(data_object$selectedResolution)
                 coview$setSelectedResolutionIndex(data_object$selectedResolutionIndex)
                 for(band in data_object$bands){
                   cvb <- GSCoverageBand$new()
                   covname <- if(!is.null(band$name)) band$name else layername
                   cvb$setDefinition(paste0(covname,"@", band$index))
                   cvb$setIndex(band$index)
                   cvb$addInputBand(GSInputCoverageBand$new( coverageName = covname, band = band$index))
                   coview$addBand(cvb)
                 }
                 resource$setView(coview)
               }else{
                 #check nb of bands, if > 3 we configure a coverage view
                 bands <- names(data_object$coverages)
                 if(length(bands)>3){
                   coview <- GSCoverageView$new()
                   coview$setName(layername)
                   ect <- data_object$envelopeCompositionType
                   if(is.null(ect)) ect <- "INTERSECTION"
                   coview$setEnvelopeCompositionType(ect)
                   sr <- data_object$selectedResolution
                   if(is.null(sr)) sr <- "BEST"
                   coview$setSelectedResolution(sr)
                   sri <- data_object$selectedResolutionIndex
                   if(is.null(sri)) sri <- -1
                   coview$setSelectedResolutionIndex(sri)
                   for(i in 1:length(bands)){
                     band <- bands[i]
                     cvb <- GSCoverageBand$new()
                     covname <- layername
                     cvb$setDefinition(paste0(covname,"@", i-1))
                     cvb$setIndex(i-1)
                     cvb$addInputBand(GSInputCoverageBand$new( coverageName = covname, band = i-1))
                     coview$addBand(cvb)
                   }
                   resource$setView(coview)
                 }
               }
               
             }
      )
      
      #styles publication if needed
      gs_styles <- c(GS$getStyleNames(), GS$getStyleNames(ws = workspace))
      if(data_object$styleUpload) if(length(data_object$styles)>0){
        reload_styles = FALSE
        for(i in 1:length(data_object$styles)){
          style <- data_object$styles[i]
          #check if any style SLD file is available in source
          style_sldfile <- paste0(style,".sld")
          style_sldfilepath = file.path(getwd(), "data", style_sldfile)
          if(file.exists(style_sldfilepath)){
            if(!style %in% gs_styles){
              config$logger$WARN("No style '%s' in Geoserver", style)
              config$logger$INFO("Creating GeoServer style '%s' from SLD style file '%s' available as data", style, style_sldfile)
              created <- GS$createStyle(file = style_sldfilepath, name = style, ws = workspace)
              if(created) reload_styles = TRUE
            }else{
              config$logger$WARN("Existing style '%s' in Geoserver", style)
              config$logger$INFO("Updating GeoServer style '%s' from SLD style file '%s' available as data", style, style_sldfile)
              updated <- GS$updateStyle(file = style_sldfilepath, name = style, ws = workspace)
              if(updated) reload_styles = TRUE
            }
          }
        }
        if(reload_styles){
          GS$reload()
          gs_styles <- c(GS$getStyleNames(), GS$getStyleNames(ws = workspace)) 
        }
      }
      
      #layer build and publication
      switch(data_object$spatialRepresentationType,
             "vector" = {
               layer <- GSLayer$new()
               layer$setName(layername)
               if(length(data_object$styles)>0){
                 for(i in 1:length(data_object$styles)){
                   style <- data_object$styles[[i]]
                   if(style %in% gs_styles){
                    if(i==1) layer$setDefaultStyle(style) else layer$addStyle(style)
                   }
                 }
               }else{
                 if(entity$identifiers[["id"]] %in% gs_styles){
                   layer$setDefaultStyle(entity$identifiers[["id"]])
                 }
                 layer$setDefaultStyle("generic")
               }
               
               #publish
               try(GS$unpublishLayer(workspace, store, layername))
               out <- GS$publishLayer(workspace, store, resource, layer)
               if(!out){
                 errMsg <- sprintf("Error during layer '%s' publication for entity '%s'!",layername, entity$identifiers[["id"]])
                 config$logger$ERROR(errMsg)
               }else{
                 infoMsg <- sprintf("Successful layer'%s' publication in Geoserver for entity '%s'!", layername, entity$identifiers[["id"]])
               }
             },
             "grid" = {
               out <- FALSE
               cov <- GS$getCoverage(ws = workspace, cs = store, cv = layername)
               if(is.null(cov)){
                 out <- GS$createCoverage(ws = workspace, cs = store, coverage = resource)
               }else{
                 out <- GS$updateCoverage(ws = workspace, cs = store, coverage = resource)
               }
               #manage coverage styles by updating associated layer object
               layer <- GS$getLayer(layername)
               if(is(layer, "GSLayer")){
                 layer$setName(layername)
                 if(length(data_object$styles)>0){
                   layer$styles <- list()
                   for(i in 1:length(data_object$styles)){
                     style <- data_object$styles[[i]]
                     if(style %in% gs_styles){
                      if(i==1) layer$setDefaultStyle(style) else layer$addStyle(style)
                     }
                   }
                 }else{
                   if(entity$identifiers[["id"]] %in% gs_styles){
                     layer$setDefaultStyle(entity$identifiers[["id"]])
                   }
                   layer$setDefaultStyle("generic")
                 }
                 GS$updateLayer(layer)  
               }
               
               if(!out){
                 errMsg <- sprintf("Error during layer '%s' publication for entity '%s'!",layername, entity$identifiers[["id"]])
                 config$logger$ERROR(errMsg)
               }else{
                 infoMsg <- sprintf("Successful layer'%s' publication in Geoserver for entity '%s'!", layername, entity$identifiers[["id"]])
               }
             }
      )
    }else{
      config$logger$INFO("Layer update is skipped for layer '%s' (overwrite and/or overwrite_layer is set to FALSE)", layername)
    }
  }
}

Try the geoflow package in your browser

Any scripts or data that you put into this service are public.

geoflow documentation built on Dec. 12, 2025, 5:08 p.m.