Python wrapped implementation of the Snobal model applied at a point.
The code in pointsnobal/c_snobal
is the same underlying algorithms described
in A spatially distributed energy balance snowmelt model for application in mountain basins (Marks 1999),
which details Snobal. This code was originally available in IPW.
This software takes in a csv of HOURLY input data and writes a csv of daily snowpack data.
🚀 Calling Snow Researchers & Students! 🚀
At M3 Works, we’re passionate about advancing scientific research and education! 🌍📊
That’s why we offer free access to our PointSnobal API for qualifying research and educational projects. Whether you’re modeling snow processes or exploring hydrology, we’re here to support your work!
🔑 Request an API key on our website -> https://m3works.io/contact
Let’s collaborate and push the boundaries of environmental modeling together! ❄️📡
If you use the pointsnobal API we ask that you credit M3Works. To cite, please use the following DOI:
https://zenodo.org/records/14814813
This API is provided under the same license listed in this directory. The API and code are provided “as is”, and M3 Works LLC makes no guarantees of functionality, performance, or fitness for any particular purpose.
M3 Works LLC shall not be held liable for any direct, indirect, incidental, or consequential damages arising from the use or inability to use this API or its associated code.
Use of this API constitutes acceptance of the terms and conditions outlined in the accompanying license.
Example of api usage in python
from pathlib import Path
import requests
import pandas as pd
api_key = "<your key>"
api_id = "bktiz24e19"
file_path = Path("<path to your input csv file>")
elevation = 1000 # your point elevation in meters
url = f"https://{api_id}.execute-api.us-west-2.amazonaws.com/m3works/snobal"
params = {"elevation": elevation}
output_file_name = file_path.name.replace('inputs', 'snobal')
output_file = file_path.parent.joinpath(output_file_name)
# Headers
headers = {
"x-api-key": api_key,
"Content-Type": "text/csv"
}
print("Reading file and calling API")
# Load the CSV file as binary data
with open(str(file_path), "rb") as file:
response = requests.post(
url, headers=headers, params=params, data=file)
print("API request finished")
# error if we failed
response.raise_for_status()
result = response.json()
print("Parsing results")
# Get result into pandas
df = pd.DataFrame.from_dict(result['results']["data"])
Use scripts/use_api.py
to call the api from the command line
python3 scripts/use_api.py <path to your file> \
<your point elevation> --api_key <your api key>
This will output a csv file of the results.
Run python3 scripts/use_api.py --help
for a full list of options.
These variables are directly used within Snobal
air_temp
- modeled air temp at 2m above groundpercent_snow
- % mix of snow vs rain (1 == all snow) [decimal percent]precip
- precipitation mass [mm]precip_temp
- wet bulb temperaturesnow_density
- density of the NEW snow that hour [kg/m^3]vapor_pressure
- modeled vapor pressurewind_speed
- Wind speed at 5m above ground [m/s]soil_temp
- Average temperature of the soil column down to 30cmnet_solar
- net solar into the snowpack [w/m^2]thermal
- net longwave radiation into the snowpack [w/m^2]
See ./tests/data/inputs_csl_2023.csv
for an example of data format
- wind height: 5m
- air temp height: 2m
- soil temp depth: 0.3m
Important
Watch out for...
- Snobal expects temperatures to be in Kelvin. This code expects Celcius.
We do the conversion in
get_timestep_force
- Precip mass (
precip
) is a big driver here. Without accurate conditions, model results will be poor
Tip
Creating a local virtual environment with your tool of choice is recommend to isolate your code prior to installation.
Navigate to a directory where you would like to download the, for example a projects
directory in your home, and
clone the repository.
cd ~/projects
git clone git@github.com:M3Works/pointsnobal.git
cd pointsnobal
Requirements can be found in requirements.txt
.
Note
A C-compiler with OpenMP support is required, on linux this is generally available. On macOS using Homebrew is a simple option.
brew install gcc libomp
For local build:
pip install -r requirements.txt
python3 setup.py build_ext --inplace
python3 install .
The entrypoint is make_snow
once installed:
make_snow <path to input file> <elevation in meters>
For example, the installation can be quickly verified by running the test problem.
make_snow ./tests/data/inputs_csl_2023.csv 1000 --output_file test.csv
Using metloom for station data that
can be used for validation. get_daily_data
returns a GeoPandas DataFrame
of the variables and units on a daily timestep. Validation is crucial in
snowpack modeling!
# Imports
import pandas as pd
from metloom.pointdata import CDECPointData
from metloom.variables import CdecStationVariables
# Specify start and end date
start_date = pd.to_datetime("2019-10-01")
end_date = pd.to_datetime("2020-06-01")
# List of variables to request
desired_variables = [
CdecStationVariables.SWE, CdecStationVariables.SNOWDEPTH
]
# Define the point
point = CDECPointData("GRV", "Graveyard Meadow")
# Request the data
df = point.get_daily_data(
start_date, end_date, desired_variables
)
# Data comes back indexed on `datetime` and `site`, reset to just datetime
df = df.reset_index().set_index("datetime")
# Show the results
print(df)
# store in csv if you want
df.to_csv(f"{point.id}_station_data.csv", index_label="datetime")
If you are getting 'omp.h' file not found
or ld: library not found for -lomp
during the
setup.py build_ext
step you need to be sure that the correct compiler is being utilized and the OpenMP libraries are available.
First, set the CC
environment variable to your C-compiler. For example, the following
is are the paths used on macOS when using homebrew.
export CC=/usr/local/bin/gcc-14 # Intel
export CC=/opt/homebrew/bin/gcc-14 # Apple silicon
Second, be sure the OpenMP libraries are available to the compiler. This can
be accomplished by setting the LDFLAGS
envornment variable. For example, for
libomp
install with homebrew.
export LDFLAGS="-L/opt/homebrew/opt/libomp/lib" # Intel
export LDFLAGS="-L/opt/hombrew/Cellar/libomp/lib" # Apple silicon
For additional help on these path, when using homebrew, utilize the brew info gcc
and brew info libomp
commands.