hsclient HydroShare Python Client Resource Aggregation Operation Examples
The following code snippets show examples for how to use the hsclient HydroShare Python Client to manipulate aggregations of known content types in HydroShare. HydroShare's content type aggregations include individual file, fileset, time series, geographic feature, geographic raster, and multidimensional NetCDF.
Install the hsclient Python Client
The hsclient Python Client for HydroShare may not be installed by default in your Python environment, so it has to be installed first before you can work with it. Use the following command to install hsclient via the Python Package Index (PyPi).
!pip install hsclient
Authenticating with HydroShare
Before you start interacting with resources in HydroShare you will need to authenticate.
from hsclient import HydroShare
hs = HydroShare()
hs.sign_in()
Create a New Empty Resource
A "resource" is a container for your content in HydroShare. Think of it as a "working directory" into which you are going to organize the code and/or data you are using and want to share. The following code can be used to create a new, empty resource within which you can create content and metadata.
This code creates a new resource in HydroShare. It also creates an in-memory object representation of that resource in your local environment that you can then manipulate with further code.
# Create the new, empty resource
new_resource = hs.create()
# Get the HydroShare identifier for the new resource
res_identifier = new_resource.resource_id
print(f'The HydroShare Identifier for your new resource is: {res_identifier}')
# Construct a hyperlink for the new resource
print(f'Your new resource is available at: {new_resource.metadata.url}')
Resource Aggregation Handling
HydroShare allows you to create and manage aggregations of content files within resources that have specific types and associated metadata. These known content types include:
- Time series
- Geographic feature
- Geographic raster
- Multidimensional NetCDF
- Single file
- File set
- CSV
The general process for creating an aggregation within a resource requires adding files to the resource and then applying the appropriate aggregation type. For some of the aggregation types, some of the aggregation metadata fields will be automatically extracted from the files you upload. You can then set the values of the other aggregation-level metadata elements.
Create a Single File Aggregation
A single file aggregation in a HydroShare is any individual file to which you want to add extra metadata.
# Import the aggregation types
from hsmodels.schemas.enums import AggregationType
# Upload a single content file to the resource. This is a generic text file.
new_resource.file_upload('Example_Files/Data_File1.txt')
# Specify the file you want to add the aggregation to
file = new_resource.file(path='Data_File1.txt')
# Create a single file aggregation on the file and refresh the resource
agg = new_resource.file_aggregate(file, AggregationType.SingleFileAggregation)
# Print the title for the aggregation that was added to the resource
print(f'The following aggregation was added to the resource: {agg.metadata.title}')
print(f'Aggregation type: {agg.metadata.type}')
Add Metadata to the Aggregation
Once you have created an aggregation, you can edit and add metadata elements. For a single file aggregation, you can add a title, subject keywords, extended metadata as key-value pairs, and spatial and temporal coverage.
All of the metadata edits are stored locally until you call the save()
function on the aggregation to write the edits you have made to HydroShare.
Title and Keywords
The title of an aggregation is a string. Subject keywords are handled as a list of strings.
# Set the title and subject keywords for the aggregation
agg.metadata.title = 'A Single File Aggregation'
agg.metadata.subjects = ['Aggregation', 'Single File', 'Text']
# Print the title and keywords for the aggregation
print(f'Aggregation Title: {agg.metadata.title}')
print(f'Aggregation Keywords: {", ".join(agg.metadata.subjects)}')
# Save the aggregation to write the metadata to HydroShare
agg.save()
Extended Metadata Elements
Extended metadata elements for an aggregation are handled using a Python dictionary. You can add new elements using key-value pairs.
# Add an extended metadata element to the aggregation as a key-value pair
agg.metadata.additional_metadata['New Element Key'] = 'Text value of new element.'
# Remove an individual key-value pair from the aggregation using its key
del agg.metadata.additional_metadata['New Element Key']
# Or, you can clear out all extended metadata elements that might exist
agg.metadata.additional_metadata.clear()
# Add multiple key-value pairs to the aggregation at once using a Python dictionary
agg.metadata.additional_metadata = {
'Observed Variable': 'Water use',
'Site Location': 'Valley View Tower Dormitory on Utah State University\'s Campus in Logan, UT'
}
# Print the extended metadata elements
print('The extended metadata elements for the aggregation include:')
for key, value in agg.metadata.additional_metadata.items():
print(key + ':', value)
# Save the aggregation to write the metadata to HydroShare
agg.save()
Spatial and Temporal Coverage
Spatial and temporal coverage for an aggregation are handled in the same way they are handled for resource level metadata. Initially the spatial and temporal coverage for an aggregation are empty. To set them, you have to create a coverage object of the right type and set the spatial or temporal coverage to that object.
# Import the required metadata classes for coverage objects
from hsmodels.schemas.fields import BoxCoverage, PointCoverage, PeriodCoverage
from datetime import datetime
# Set the spatial coverage of the aggregation to a BoxCoverage object
agg.metadata.spatial_coverage = BoxCoverage(name='Logan, Utah',
northlimit=41.7910,
eastlimit=-111.7664,
southlimit=41.6732,
westlimit=-111.9079,
projection='WGS 84 EPSG:4326',
type='box',
units='Decimal degrees')
# You can remove the spatial coverage element by setting it to None
agg.metadata.spatial_coverage = None
# If you want to set the spatial coverage to a PointCoverage instead
agg.metadata.spatial_coverage = PointCoverage(name='Logan, Utah',
north=41.7371,
east=-111.8351,
projection='WGS 84 EPSG:4326',
type='point',
units='Decimal degrees')
# Create a beginning and ending date for a time period
beginDate = datetime.strptime('2020-12-01T00:00:00Z', '%Y-%m-%dT%H:%M:%S%fZ')
endDate = datetime.strptime('2020-12-31T00:00:00Z', '%Y-%m-%dT%H:%M:%S%fZ')
# Set the temporal coverage of the aggregation to a PeriodCoverage object
agg.metadata.period_coverage = PeriodCoverage(start=beginDate, end=endDate)
# Print the temporal coverage information
print('Temporal Coverage:')
print(agg.metadata.period_coverage)
# Print the spatial coverage information
print('\nSpatial Coverage:')
print(agg.metadata.spatial_coverage)
# Save the aggregation to write the metadata to HydroShare
agg.save()
Creating Other Aggregation Types
Geographic Feature Aggregation
Geographic feature aggregations are created in HydroShare from the set of files that make up an ESRI Shapefile. You need to upload the shapefile and then HydroShare will automatically set the aggregation on the set of files you upload. You can then retrieve the aggregation using its title or by searching for one of the files it contains.
# Create a list of the files that make up the shapefile to be uploaded
file_list = ['Example_Files/watersheds.cpg', 'Example_Files/watersheds.dbf',
'Example_Files/watersheds.prj', 'Example_Files/watersheds.sbn',
'Example_Files/watersheds.sbx', 'Example_Files/watersheds.shp',
'Example_Files/watersheds.shx', 'Example_Files/watersheds.shp.xml']
# Upload the files to the resource all at the same time
new_resource.file_upload(*file_list)
print('Files uploaded!')
If you upload all of the files of a shapefile together as shown above, HydroShare automatically recognizes the files as a shapefile and auto-aggregates the files into a geographic feature aggregation for you. So, you then just need to get the aggregation that was created if you want to further operate on it - e.g., to modify the aggregation-level metadata.
Metadata for a geographic feature aggregation includes a title, subject keywords, extended key-value pairs, temporal coverage, spatial coverage, geometry information, spatial reference, and attribute field information. When HydroShare creates the aggregation on the shapefile, the spatial coverage, geometry information, spatial reference, and attribute field information metadata will be automatically set for you. You can then set all of the other metadata elements as shown above for the single file aggregation if you need to.
# Get the aggregation that was just created
# You can get the aggregation by searching for a file that is inside of it
agg = new_resource.aggregation(file__name='watersheds.shp')
# Or, you can get the aggregation by searching for its title, which is initially
# set to the name of the shapefile
agg = new_resource.aggregation(title='watersheds')
# Print the title for the aggregation that was added to the resource
print(f'The following aggregation was added to the resource: {agg.metadata.title}')
print(f'Aggregation type: {agg.metadata.type}')
Geographic Raster Aggregation
Geographic raster aggregations are created in HydroShare from one or more raster data files that make up a raster dataset. HydroShare uses GeoTiff files for raster datasets. Like the geographic feature aggregation, when you upload all of the files for a geographic raster dataset (all .tif and a .vrt file) at once, HydroShare will automatically create the aggregation for you. You can then get the aggregation and set the other metadata elements as shown above for the single file aggregation.
HydroShare initially sets the title of the geographic raster aggregation to the first .tif file that appears in the .vrt file. The spatial coverage, spatial reference, and cell information are set automatically based on information extracted from the dataset.
# Upload the files making up the raster dataset to the resource
file_list = ['Example_Files/logan1.tif', 'Example_Files/logan2.tif', 'Example_Files/logan.vrt']
new_resource.file_upload(*file_list)
# Get the aggregation that was just created - initially the title will be "logan1"
# based on the name of the first .tif file that appears in the .vrt file
agg = new_resource.aggregation(title='logan1')
# Print the title for the aggregation that was added to the resource
print(f'The following aggregation was added to the resource: {agg.metadata.title}')
print(f'Aggregation type: {agg.metadata.type}')
Multidimensional NetCDF Aggregation
Multidimensional aggregations are created in HydroShare from a NetCDF file. Like the other aggregation types, you can upload the NetCDF file and HydroShare will automatically create the aggregation for you. HydroShare also automatically extracts metadata from the NetCDF file to populate the aggregation metadata. Some of this metadata may get propagated to the resource level if you haven't set things like the title and keywords. You can then get the aggregation and set the other metadata elements as shown above for the single file aggregation.
# Upload the NetCDF file to the resource
new_resource.file_upload('Example_Files/SWE_time.nc')
# Get the aggregation by searching for the NetCDF file that is inside of it
agg = new_resource.aggregation(file__name='SWE_time.nc')
# Print the title for the aggregation that was added to the resource
print(f'The following aggregation was added to the resource: {agg.metadata.title}')
print(f'Aggregation type: {agg.metadata.type}')
Time Series Aggregation
Time series aggregations are created in HydroShare from an ODM2 SQLite database file. The ODM2 SQLite database contain one or more time series
# Upload the SQLite file to the resource
new_resource.file_upload('Example_Files/ODM2.sqlite')
# Get the aggregation by searching for the SQLite file that is inside of it
agg = new_resource.aggregation(file__name='ODM2.sqlite')
# Print the title for the aggregation that was added to the resource
print(f'The following aggregation was added to the resource: {agg.metadata.title}')
print(f'Aggregation type: {agg.metadata.type}')
File Set Aggregation
A file set aggregation is any folder within a resource to which you want to add metadata. If you want to create a file set aggregation, you first have to create a folder, then upload files to it. After that, you can set the aggregation on the folder.
# Create a new folder for the file set aggregation
new_resource.folder_create('Fileset_Aggregation')
# Add some files to the folder
new_resource.file_upload('Example_Files/Data_File1.txt', 'Example_Files/Data_File2.txt',
destination_path='Fileset_Aggregation')
# set the folder to fileset aggregation
new_resource.file_aggregate(path='Fileset_Aggregation', agg_type=AggregationType.FileSetAggregation)
fs_agg = new_resource.aggregation(type='FileSet')
print(f"Aggregation Type: {fs_agg.metadata.type}")
CSV Aggregation
CSV aggregations are created in HydroShare from a CSV file. When you upload a csv file, HydroShare automatically creates a CSV aggregation and extracts metadata from the uploaded CSV file.
# Upload a CSV file to the resource
new_resource.file_upload('Example_Files/Data_File1.csv')
# Get the aggregation by searching for the CSV file
agg = new_resource.aggregation(file__name='Data_File1.csv')
# Print the title for the aggregation that was added to the resource
print(f"The following aggregation was added to the resource: {agg.metadata.title}")
print(f"Aggregation type: {agg.metadata.type}")
Get Aggregation Properties
Each aggregation in a resource has metadata properties associated with it. You can query/retrieve those properties for display. The following shows an example for the time series aggregation that was created above.
# Get the time series aggregation that was created above
agg = new_resource.aggregation(type='TimeSeries')
# Print the metadata associated with the aggregation
print(f'Aggregation Title: {agg.metadata.title}')
print(f'Aggregation Type: {agg.metadata.type}')
print(f'Aggregation Keywords: {", ".join(agg.metadata.subjects)}')
print(f'Aggregation Temporal Coverage: {agg.metadata.period_coverage}')
print(f'Aggregation Spatial Coverage: {agg.metadata.spatial_coverage}')
# Print the list of files in the aggregation
file_list = agg.files()
print('List of files contained within the aggregation:')
print(*file_list, sep='\n')
Example of editing extracted metadata for CSV aggregation
# Get the CSV aggregation you created above
agg = new_resource.aggregation(type="CSV")
# we can edit any extracted column properties (like 'title', 'description', 'datatype')
table = agg.metadata.tableSchema.table
# here is an example of editing the 'title' property of the 1st column
table.columns[0].title = "Date of Observation"
# here is an example of editing the 'description' of the 7th column
table.columns[6].description = "Total Volume over time"
# save the updated metadata
agg.save()
Searching for Aggregations within a Resource
If you need to find/get one or more aggregations within a resource so you can download or remove it from the resource, there are several filters available that allow you to return a list of aggregations that meet your search criteria.
# Get the list of all aggregations in the resource
aggregations = new_resource.aggregations()
# Get a list of all aggregations of a particular type
aggregations = new_resource.aggregations(type="TimeSeries")
# Get a list of aggregations with extended metadata searching by key
aggregations = new_resource.aggregations(additional_metadata__key="Observed Variable")
# Get a list of aggregations with extended metadata searching by value
aggregations = new_resource.aggregations(additional_metadata__value="Water Use")
# Get a list of aggregations with a subject keyword searching by value
aggregations = new_resource.aggregations(subjects__contains="Temperature")
# Get a list of aggregations searching by title (or any metadata attribute)
aggregations = new_resource.aggregations(title="watersheds")
# Get a list of aggregations searching by a nested metadata attribute (__)
aggregations = new_resource.aggregations(period_coverage__name="period_coverage name")
# Get a list of aggregations by combining field searching, filtered with "AND"
aggregations = new_resource.aggregations(period_coverage__name="period_coverage name", title="watersheds")
You can also search for individual aggregations within a resource.
# Search for an aggregation of type time series
aggregation = new_resource.aggregation(type="TimeSeries")
# Search for an aggregation with a specific title
aggregation = new_resource.aggregation(title="watersheds")
# Search for an aggregation that contains a particular file name
aggregation = new_resource.aggregation(file__name="ODM2.sqlite")
Downloading an Aggregation
When working with a resource, you may want to download one of the aggregations contained within the resource. If you want to download it to a particular location on your disk, you can pass a path to the location where you want the aggregation to be saved to the download()
function as a string. Aggregations are downloaded as a zipped file containing the aggregation content and metadata files.
# Get the geographic feature aggregation that was created above
agg = new_resource.aggregation(title="watersheds")
# Download the aggregation
new_resource.aggregation_download(agg)
Clean up the zippled aggregation file that was just downloaded.
!rm 'watersheds.shp.zip'
Remove and Delete an Aggregation
You may wish to remove an aggregation from a resource. There are two functions you can use to do this. The difference between the two is whether the aggregation's content files are preserved in the resource or deleted. To remove the aggregation-specific metadata and associations while maintaining the content files, call the remove()
function on the aggregation.
# Get the geographic raster aggregation that was created above
agg = new_resource.aggregation(title="logan1")
# Remove the aggregation and delete its metadata, but leave the file(s)
new_resource.aggregation_remove(agg)
If you want to delete the aggregation, including its metadata and the associated content files, you can call the delete()
function on the aggregation. Once you have called the remove()
function on an aggregation, the delete()
function will no longer work and you will have to delete the files individually.
# Get the multidimensional NetCDF aggregation that was created above
agg = new_resource.aggregation(type="NetCDF")
# Delete the aggregation and metadata along with files within aggregation
new_resource.aggregation_delete(agg)