diff --git a/.gitmodules b/.gitmodules new file mode 100644 index 0000000..4a73e8f --- /dev/null +++ b/.gitmodules @@ -0,0 +1,3 @@ +[submodule "hydrofabric"] + path = hydrofabric + url = https://github.com/NOAA-OWP/hydrofabric.git diff --git a/hydrofabric b/hydrofabric new file mode 160000 index 0000000..a1a2c9f --- /dev/null +++ b/hydrofabric @@ -0,0 +1 @@ +Subproject commit a1a2c9f1b1997201c0fecb67e830092e994d5d73 diff --git a/subset_network.R b/subset_network.R deleted file mode 100644 index 2c491d9..0000000 --- a/subset_network.R +++ /dev/null @@ -1,339 +0,0 @@ -# MIT License -# -# Copyright (c) 2021 Mike Johnson -# -# Permission is hereby granted, free of charge, to any person obtaining a copy -# of this software and associated documentation files (the "Software"), to deal -# in the Software without restriction, including without limitation the rights -# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -# copies of the Software, and to permit persons to whom the Software is -# furnished to do so, subject to the following conditions: -# -# The above copyright notice and this permission notice shall be included in all -# copies or substantial portions of the Software. -# -# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE -# SOFTWARE. - -read_qml = function (qml_file) { - paste(readLines(qml_file), collapse = "\n") -} - -create_style_row = function (gpkg_path, - layer_name, - style_name, - style_qml) { - geom_col <- - sf::st_read( - gpkg_path, - query = paste0( - "SELECT column_name from gpkg_geometry_columns ", - "WHERE table_name = '", - layer_name, - "'" - ), - quiet = TRUE - )[1, - 1] - data.frame( - f_table_catalog = "", - f_table_schema = "", - f_table_name = layer_name, - f_geometry_column = geom_col, - styleName = style_name, - styleQML = style_qml, - styleSLD = "", - useAsDefault = TRUE, - description = "Generated for hydrofabric", - owner = "", - ui = NA, - update_time = Sys.time() - ) -} - -append_style = function (gpkg_path, - qml_dir = system.file("qml", package = "hydrofabric"), - layer_names) { - f = list.files(qml_dir, full.names = TRUE) - - good_layers = gsub(".qml", "", basename(f)) - - layer_names = layer_names[layer_names %in% good_layers] - - files = grep(paste(layer_names, collapse = "|"), f, value = TRUE) - - styles <- sapply(files, read_qml) - style_names <- - sapply(layer_names, paste0, "__hydrofabric_style") - style_rows <- - do.call( - rbind, - mapply( - create_style_row, - layer_names, - style_names, - styles, - MoreArgs = list(gpkg_path = gpkg_path), - SIMPLIFY = FALSE - ) - ) - if ("layer_styles" %in% sf::st_layers(gpkg_path)$name) { - try(sf::st_delete(gpkg_path, "layer_styles"), silent = TRUE) - } - sf::st_write( - style_rows, - gpkg_path, - layer = "layer_styles", - append = FALSE, - quiet = TRUE - ) - return(gpkg_path) -} - -#' Access Hydrofabric Network -#' @param VPU Vector Processing Unit -#' @param base_s3 the base hydrofabric directory to access in Lynker's s3 -#' @param cache_dir should data be cached to a local directory? Will speed up multiple subsets in the same region -#' @param cache_overwrite description. Should a cached file be overwritten -#' @return file path -#' @export - -get_fabric = function(VPU, - base_s3 = 's3://lynker-spatial/pre-release/', - cache_dir = NULL, - cache_overwrite = FALSE) { - Key <- NULL - - xx = aws.s3::get_bucket_df( - bucket = dirname(base_s3), - prefix = basename(base_s3), - region = 'us-west-2' - ) |> - dplyr::filter(grepl(basename(base_s3), Key) & - grepl(paste0(VPU, ".gpkg$"), Key)) |> - dplyr::filter(!grepl("[.]_", Key)) |> - dplyr::filter(!grepl("/", dirname(Key))) - - if (!is.null(cache_dir)) { - dir.create(cache_dir, - recursive = TRUE, - showWarnings = FALSE) - gpkg = glue::glue("{cache_dir}/{basename(xx$Key)}") - if (cache_overwrite) { - unlink(gpkg) - } - temp = FALSE - } else { - gpkg = tempfile(fileext = ".gpkg") - temp = TRUE - } - - if (!file.exists(gpkg)) { - aws.s3::save_object( - bucket = xx$Bucket, - object = xx$Key, - file = gpkg, - region = 'us-west-2' - ) - } - - return(gpkg) - -} - -#' Subset Hydrofabric Network -#' @param id hydrofabric id (relevant only to nextgen fabrics) -#' @param comid NHDPlusV2 COMID -#' @param hl_uri hydrolocation URI (relevant only to nextgen fabrics) -#' @param nldi_feature list with names 'featureSource' and 'featureID' where 'featureSource' is derived from the "source" column of the response of dataRetrieval::get_nldi_sources() and the 'featureID' is a known identifier from the specified 'featureSource'. -#' @param loc Location given as vector of XY in CRS 4326 (long, lat) -#' @param base_s3 the base hydrofabric directory to access in Lynker's s3 -#' @param lyrs layers to extract. Default is all possible in the hydrofabric GPKG data model -#' @param outfile file path to write to. Must have ".gpkg" extension -#' @param cache_dir should data be cached to a local directory? Will speed up multiple subsets in the same region -#' @param cache_overwrite description. Should a cached file be overwritten -#' @return file path (outfile) or list of features -#' @export - -subset_network = function(id = NULL, - comid = NULL, - hl_uri = NULL, - nldi_feature = NULL, - loc = NULL, - base_s3 = 's3://lynker-spatial/pre-release/', - lyrs = c( - "divides", - "nexus", - "flowpaths", - "network", - "hydrolocations", - "reference_flowline", - "reference_catchment", - "refactored_flowpaths", - "refactored_divides" - ), - outfile = NULL, - cache_dir = NULL, - qml_dir = system.file("qml", package = "hydrofabric"), - cache_overwrite = FALSE) { - Key <- - hf_hydroseq <- - hf_id <- hydroseq <- member_COMID <- toid <- vpu <- NULL - - net = arrow::open_dataset(glue::glue(base_s3, "conus_net.parquet")) |> - dplyr::select(id, toid, hf_id, hl_uri, hf_hydroseq, hydroseq, vpu) |> - dplyr::collect() |> - dplyr::distinct() - - if (!is.null(id) & !is.null(net)) { - comid = dplyr::filter(net, id == !!id | toid == !!id) |> - dplyr::slice_max(hf_hydroseq) |> - dplyr::pull(hf_id) - } - - if (!is.null(nldi_feature)) { - comid = nhdplusTools::get_nldi_feature(nldi_feature)$comid - } - - if (!is.null(loc)) { - comid = nhdplusTools::discover_nhdplus_id(point = sf::st_sfc(sf::st_point(c(loc[1], loc[2])), crs = 4326)) - } - - if (!is.null(hl_uri) & !is.null(net)) { - origin = dplyr::filter(net, hl_uri == !!hl_uri) |> - dplyr::slice_max(hf_hydroseq) |> - dplyr::pull(toid) |> - unique() - } - - if (!is.null(comid) & !is.null(net)) { - origin = dplyr::filter(net, hf_id == comid) |> - dplyr::slice_max(hf_hydroseq) |> - dplyr::pull(id) |> - unique() - } else if (is.null(net)) { - origin = comid - } - - if (is.null(origin)) { - stop("origin not found") - } - - - if (is.null(net)) { - xx = suppressMessages({ - nhdplusTools::get_nhdplus(comid = comid) - }) - - v = nhdplusTools::vpu_boundaries - - vpuid = v$VPUID[which(lengths(sf::st_intersects(sf::st_transform( - v, sf::st_crs(xx) - ), xx)) > 0)] - } else { - vpuid = unique(dplyr::pull(dplyr::filter(net, id == origin | - toid == origin), vpu)) - } - - - gpkg = get_fabric( - VPU = vpuid, - base_s3 = base_s3, - cache_dir = cache_dir, - cache_overwrite = cache_overwrite - ) - lyrs = lyrs[lyrs %in% sf::st_layers(gpkg)$name] - - db <- DBI::dbConnect(RSQLite::SQLite(), gpkg) - on.exit(DBI::dbDisconnect(db)) - - if (!is.null(net)) { - sub_net = dplyr::distinct(dplyr::select(dplyr::filter(net, vpu == vpuid), id, toid)) - } else { - lookup <- - c( - id = "ID", - id = "COMID", - toid = "toID", - toid = "toCOMID" - ) - dplyr::sub_net = dplyr::tbl(db, lyrs[grepl("flowline|flowpath", lyrs)]) |> - dplyr::select(dplyr::any_of( - c( - "id", - "toid", - 'COMID', - 'toCOMID', - "ID", - "toID", - "member_COMID" - ) - )) |> - dplyr::collect() |> - dplyr::rename(dplyr::any_of(lookup)) - - if ("member_COMID" %in% names(sub_net)) { - origin = dplyr::filter(sub_net, grepl(origin, member_COMID)) |> - dplyr::pull(id) - - sub_net = dplyr::select(sub_net, -member_COMID) - } - } - - message("Starting from: `", origin, "`") - - tmap = suppressWarnings({ - nhdplusTools::get_sorted(dplyr::distinct(sub_net), outlets = origin) - }) - - if (grepl("nex", utils::tail(tmap$id, 1))) { - tmap = utils::head(tmap, -1) - } - - ids = unique(c(unlist(tmap))) - - hydrofabric = list() - - for (j in 1:length(lyrs)) { - message(glue::glue("Subsetting: {lyrs[j]} ({j}/{length(lyrs)})")) - - crs = sf::st_layers(gpkg)$crs - - t = dplyr::tbl(db, lyrs[j]) |> - dplyr::filter(dplyr::if_any(dplyr::any_of( - c('COMID', 'FEATUREID', 'divide_id', 'id', 'ds_id', "ID") - ), ~ . %in% !!ids)) |> - dplyr::collect() - - if (all(!any(is.na(as.character(crs[[j]]))), nrow(t) > 0)) { - if (any(c("geometry", "geom") %in% names(t))) { - t = sf::st_as_sf(t, crs = crs[[j]]) - } else { - t = t - } - } - - if (!is.null(outfile)) { - sf::write_sf(t, outfile, lyrs[j]) - } else { - hydrofabric[[lyrs[j]]] = t - } - } - - if (is.null(cache_dir)) { - unlink(gpkg) - } - - if (!is.null(outfile)) { - outfile = append_style(outfile, qml_dir = qml_dir, layer_names = lyrs) - return(outfile) - } else { - hydrofabric - } - -} diff --git a/subset_network.R b/subset_network.R new file mode 120000 index 0000000..92865cd --- /dev/null +++ b/subset_network.R @@ -0,0 +1 @@ +hydrofabric/R/subset_network.R \ No newline at end of file