Skip to content

Commit

Permalink
feat(maw-mfrcsv): add option to write MAW flow reduction to csv (#925)
Browse files Browse the repository at this point in the history
add option to write MAW extraction and injection flow reductions to csv.

Co-authored-by: kzeiler <kkzeiler@gsi-net.com>
  • Loading branch information
kzeiler and kzeiler authored Mar 29, 2022
1 parent 5be3f36 commit 99ad83b
Show file tree
Hide file tree
Showing 7 changed files with 437 additions and 3 deletions.
323 changes: 323 additions & 0 deletions autotest/test_gwf_maw10.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,323 @@
"""
MODFLOW 6 Autotest
Test to confirm that the sum of rate-actual and maw-reduction observations
is equal to the specified MAW extraction or injection pumping rate when
reported using the MAW_FLOW_REDUCE_CSV option. Injection and extraction
are tested for both MAW HEAD_LIMIT and RATE_SCALING options to reduce MAW
flow rates from the specified input (requested) values.
"""

import os
import pytest
import sys
import numpy as np

try:
import pymake
except:
msg = "Error. Pymake package is not available.\n"
msg += "Try installing using the following command:\n"
msg += " pip install https://github.com/modflowpy/pymake/zipball/master"
raise Exception(msg)

try:
import flopy
except:
msg = "Error. FloPy package is not available.\n"
msg += "Try installing using the following command:\n"
msg += " pip install flopy"
raise Exception(msg)

from framework import testing_framework
from simulation import Simulation

ex = ["maw10a", "maw10b", "maw10c", "maw10d"]
exdirs = []
for s in ex:
exdirs.append(os.path.join("temp", s))

# maw settings for runs a, b, c and d
mawsetting_a = {
0: [
[0, "rate", -2000.0],
[0, "head_limit", -0.4],
],
1: [
[0, "status", "inactive"],
],
}
mawsetting_b = {
0: [
[0, "rate", -2000.0],
[0, "rate_scaling", -1.4, 1.0],
],
1: [
[0, "status", "inactive"],
],
}
mawsetting_c = {
0: [
[0, "rate", 2000.0],
[0, "head_limit", 0.4],
],
1: [
[0, "status", "inactive"],
],
}
mawsetting_d = {
0: [
[0, "rate", 2000.0],
[0, "rate_scaling", 0.0, 1.0],
],
1: [
[0, "status", "inactive"],
],
}

#simple single stress period without going inactive in 2nd stress period
#mawsetting_a = [(0, "rate", -2000.0), (0, "head_limit", -0.4)]
#mawsetting_b = [[(0, "rate", -2000.0), (0, "rate_scaling", -1.4, 1.0)], ["status", "inactive"]]
#mawsetting_c = [[(0, "rate", 2000.0), (0, "head_limit", 0.4)], ["status", "inactive"]]
#mawsetting_d = [[(0, "rate", 2000.0), (0, "rate_scaling", 0.0, 1.0)], ["status", "inactive"]]
mawsettings = [mawsetting_a, mawsetting_b, mawsetting_c, mawsetting_d]


def build_model(idx, dir):

nlay, nrow, ncol = 1, 101, 101
nper = 2
perlen = [500.0,500.0]
nstp = [50,50]
tsmult = [1.2,1.2]
delr = delc = 142.0
top = 0.0
botm = [-1000.0]
strt = 0.0
hk = 10.0

nouter, ninner = 100, 100
hclose, rclose, relax = 1e-6, 1e-6, 1.0

tdis_rc = []
for i in range(nper):
tdis_rc.append((perlen[i], nstp[i], tsmult[i]))

name = ex[idx]

# build MODFLOW 6 files
ws = dir
sim = flopy.mf6.MFSimulation(sim_name=name, sim_ws=ws)

# create tdis package
tdis = flopy.mf6.ModflowTdis(
sim, time_units="DAYS", nper=nper, perioddata=tdis_rc
)

# create gwf model
gwf = flopy.mf6.MFModel(
sim,
model_type="gwf6",
modelname=name,
model_nam_file="{}.nam".format(name),
)

# create iterative model solution and register the gwf model with it
ims = flopy.mf6.ModflowIms(
sim,
print_option="SUMMARY",
outer_dvclose=hclose,
outer_maximum=nouter,
under_relaxation="NONE",
inner_maximum=ninner,
inner_dvclose=hclose,
rcloserecord=rclose,
linear_acceleration="CG",
scaling_method="NONE",
reordering_method="NONE",
relaxation_factor=relax,
)
sim.register_ims_package(ims, [gwf.name])

dis = flopy.mf6.ModflowGwfdis(
gwf,
nlay=nlay,
nrow=nrow,
ncol=ncol,
delr=delr,
delc=delc,
top=top,
botm=botm,
idomain=1,
filename="{}.dis".format(name),
)

# initial conditions
ic = flopy.mf6.ModflowGwfic(gwf, strt=strt, filename="{}.ic".format(name))

# node property flow
npf = flopy.mf6.ModflowGwfnpf(
gwf,
save_flows=True,
icelltype=1,
k=hk,
k33=hk,
filename="{}.npf".format(name),
)

# storage
sto = flopy.mf6.ModflowGwfsto(
gwf,
save_flows=True,
iconvert=0,
ss=1.0e-5,
sy=0.1,
steady_state={0: False},
transient={0: True},
filename="{}.sto".format(name),
)

# MAW
opth = "{}.maw.obs".format(name)
wellbottom = -1000
wellrecarray = [[0, 0.15, wellbottom, 0.0, "THIEM", 1]]
wellconnectionsrecarray = [[0, 0, (0, 50, 50), 0.0, wellbottom, 0.0, 0.0]]
wellperiodrecarray = mawsettings[idx]
mawo_dict = {}
mawo_dict["{}.maw.obs.csv".format(name)] = [
("m1head", "head", (0,)),
("m1rate", "rate", (0,)),
] # is this index one-based? Not if in a tuple
maw = flopy.mf6.ModflowGwfmaw(
gwf,
filename="{}.maw".format(name),
print_input=True,
print_head=True,
print_flows=True,
save_flows=True,
observations=mawo_dict,
packagedata=wellrecarray,
connectiondata=wellconnectionsrecarray,
perioddata=wellperiodrecarray,
mfrcsv_filerecord=f"{name}.maw-reduction.csv",
)

# output control
oc = flopy.mf6.ModflowGwfoc(
gwf,
budget_filerecord="{}.cbc".format(name),
head_filerecord="{}.hds".format(name),
headprintrecord=[
("COLUMNS", ncol, "WIDTH", 15, "DIGITS", 6, "GENERAL")
],
saverecord=[("HEAD", "ALL")],
printrecord=[("HEAD", "ALL"), ("BUDGET", "ALL")],
filename="{}.oc".format(name),
)

# head observations
obs_data0 = [("head_well_cell", "HEAD", (0, 0, 0))]
obs_recarray = {"{}.obs.csv".format(name): obs_data0}
obs = flopy.mf6.ModflowUtlobs(
gwf,
pname="head_obs",
filename="{}.obs".format(name),
digits=15,
print_input=True,
continuous=obs_recarray,
)

return sim, None

#2 tests to perform here:
#1. within the .maw-reduction.csv file, do values of actual + reduction = requested?
#2. do the values in .maw-reduction.csv file match with the .maw.obs.csv file at each time
# (and all are reduction times present in the obs file)?
def eval_mawred(sim):
print("evaluating MAW flow reduction outputs...")

# MODFLOW 6 maw results
idx = sim.idxsim
name = ex[idx]
fpthobs = os.path.join(sim.simpath, "{}.maw.obs.csv".format(name))
fpthmfr = os.path.join(sim.simpath, "{}.maw-reduction.csv".format(name))
try:
tcobs = np.genfromtxt(fpthobs, names=True, delimiter=",")
except:
assert False, 'could not load data from "{}"'.format(fpthobs)
try:
tcmfr = np.genfromtxt(fpthmfr, names=True, delimiter=",", deletechars="")
except:
assert False, 'could not load data from "{}"'.format(fpthmfr)

#test 1: does rate-requested = rate-actual + maw-reduction at each time in the .maw.reduced.csv?
for rowmfr in tcmfr:
v1 = rowmfr["rate-requested"]
v2 = rowmfr["rate-actual"] + rowmfr["maw-reduction"]
errmsg = f"MAW flow reduction output: requested rate must equal actual rate plus reduced rate.\n"
errmsg += f"{v1} /= {v2}"
assert np.allclose(v1, v2), errmsg

#test 2: do values of rate-actual in .maw.reduced.csv equal those flow values reported in .maw.obs.csv?
# (and are there matching times?)
for rowmfr in tcmfr:
timevalmfr = rowmfr["time"]
actvalmfr = rowmfr["rate-actual"]
msgtime = (
"There should be a matching time in the maw.obs.csv file for each "
"time in the maw.reduce.csv file, but no match was found for "
"time = {} in the maw.obs.csv file".format(timevalmfr)
)
blnFoundTimeMatch = False
for rowobs in tcobs:
timevalobs = rowobs["time"]
if np.isclose(timevalmfr,timevalobs):
blnFoundTimeMatch = True
actvalobs = rowobs["M1RATE"]
msgval = (
"The maw.obs.csv file rate-actual value of {} should have "
"matched the maw.reduce.csv file rate-actual value of {} "
"at time {}".format(actvalobs, actvalmfr, timevalobs)
)
break
assert blnFoundTimeMatch, msgtime
assert np.isclose(actvalmfr,actvalobs), msgval

return


# - No need to change any code below
@pytest.mark.parametrize(
"idx, dir",
list(enumerate(exdirs)),
)
def test_mf6model(idx, dir):
# initialize testing framework
test = testing_framework()

# build the models
test.build_mf6_models(build_model, idx, dir)

# run the test model
test.run_mf6(Simulation(dir, exfunc=eval_mawred, idxsim=idx))


def main():
# initialize testing framework
test = testing_framework()

# build the models
# run the test model
for idx, dir in enumerate(exdirs):
test.build_mf6_models(build_model, idx, dir)
sim = Simulation(dir, exfunc=eval_mawred, idxsim=idx)
test.run_mf6(sim)

return


if __name__ == "__main__":
# print message
print("standalone run of {}".format(os.path.basename(__file__)))

# run main routine
main()
2 changes: 1 addition & 1 deletion doc/ReleaseNotes/ReleaseNotes.tex
Original file line number Diff line number Diff line change
Expand Up @@ -228,7 +228,7 @@ \section{Changes Introduced in this Release}

%\underline{ADVANCED STRESS PACKAGES}
%\begin{itemize}
% \item xxx
% \item Add new MAW\_FLOW\_REDUCE\_CSV option for the Multi-Aquifer Well (MAW) Package. If activated, then this option will result in information being written to a comma-separated value file for each multi-aquifer well and for each time step in which the extraction or injection rate is reduced by the program. Information is not written for multi-aquifer wells in which the extraction or injection rate is equal to the user-specified extraction or injection rate.
% \item xxx
% \item xxx
%\end{itemize}
Expand Down
33 changes: 33 additions & 0 deletions doc/mf6io/mf6ivar/dfn/gwf-maw.dfn
Original file line number Diff line number Diff line change
Expand Up @@ -201,6 +201,39 @@ optional true
longname shutdown kappa
description value that defines the weight applied to discharge rate for wells that limit the water level in a discharging well (defined using the HEAD\_LIMIT keyword in the stress period data). SHUTDOWN\_KAPPA is used to control discharge rate oscillations when the flow rate from the aquifer is less than the specified flow rate from the aquifer to the well. Values range between 0.0 and 1.0, and larger values increase the weight applied to the well discharge rate. The HEAD\_LIMIT option has been included to facilitate backward compatibility with previous versions of MODFLOW but use of the RATE\_SCALING option instead of the HEAD\_LIMIT option is recommended. By default, SHUTDOWN\_KAPPA is 0.0001.

block options
name mfrcsv_filerecord
type record maw_flow_reduce_csv fileout mfrcsvfile
shape
reader urword
tagged true
optional true
longname
description

block options
name maw_flow_reduce_csv
type keyword
shape
in_record true
reader urword
tagged true
optional false
longname budget keyword
description keyword to specify that record corresponds to the output option in which a new record is written for each multi-aquifer well and for each time step in which the user-requested extraction or injection rate is reduced by the program.

block options
name mfrcsvfile
type string
preserve_case true
shape
in_record true
reader urword
tagged false
optional false
longname file keyword
description name of the comma-separated value (CSV) output file to write information about multi-aquifer well extraction or injection rates that have been reduced by the program. Entries are only written if the extraction or injection rates are reduced.

block options
name ts_filerecord
type record ts6 filein ts6_filename
Expand Down
3 changes: 3 additions & 0 deletions doc/mf6io/mf6ivar/md/mf6ivar.md
Original file line number Diff line number Diff line change
Expand Up @@ -496,6 +496,9 @@
| GWF | MAW | OPTIONS | FLOWING_WELLS | KEYWORD | keyword that activates the flowing wells option for the multi-aquifer well package. |
| GWF | MAW | OPTIONS | SHUTDOWN_THETA | DOUBLE PRECISION | value that defines the weight applied to discharge rate for wells that limit the water level in a discharging well (defined using the HEAD\_LIMIT keyword in the stress period data). SHUTDOWN\_THETA is used to control discharge rate oscillations when the flow rate from the aquifer is less than the specified flow rate from the aquifer to the well. Values range between 0.0 and 1.0, and larger values increase the weight (decrease under-relaxation) applied to the well discharge rate. The HEAD\_LIMIT option has been included to facilitate backward compatibility with previous versions of MODFLOW but use of the RATE\_SCALING option instead of the HEAD\_LIMIT option is recommended. By default, SHUTDOWN\_THETA is 0.7. |
| GWF | MAW | OPTIONS | SHUTDOWN_KAPPA | DOUBLE PRECISION | value that defines the weight applied to discharge rate for wells that limit the water level in a discharging well (defined using the HEAD\_LIMIT keyword in the stress period data). SHUTDOWN\_KAPPA is used to control discharge rate oscillations when the flow rate from the aquifer is less than the specified flow rate from the aquifer to the well. Values range between 0.0 and 1.0, and larger values increase the weight applied to the well discharge rate. The HEAD\_LIMIT option has been included to facilitate backward compatibility with previous versions of MODFLOW but use of the RATE\_SCALING option instead of the HEAD\_LIMIT option is recommended. By default, SHUTDOWN\_KAPPA is 0.0001. |
| GWF | MAW | OPTIONS | MAW_FLOW_REDUCE_CSV | KEYWORD | keyword to specify that record corresponds to the output option in which a new record is written for each multi-aquifer well and for each time step in which the user-requested extraction or injection rate is reduced by the program. |
| GWF | MAW | OPTIONS | FILEOUT | KEYWORD | keyword to specify that an output filename is expected next. |
| GWF | MAW | OPTIONS | MFRCSVFILE | STRING | name of the comma-separated value (CSV) output file to write information about multi-aquifer well extraction or injection rates that have been reduced by the program. Entries are only written if the extraction or injection rates are reduced. |
| GWF | MAW | OPTIONS | TS6 | KEYWORD | keyword to specify that record corresponds to a time-series file. |
| GWF | MAW | OPTIONS | FILEIN | KEYWORD | keyword to specify that an input filename is expected next. |
| GWF | MAW | OPTIONS | TS6_FILENAME | STRING | defines a time-series file defining time series that can be used to assign time-varying values. See the ``Time-Variable Input'' section for instructions on using the time-series capability. |
Expand Down
6 changes: 6 additions & 0 deletions doc/mf6io/mf6ivar/tex/gwf-maw-desc.tex
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,12 @@

\item \texttt{shutdown\_kappa}---value that defines the weight applied to discharge rate for wells that limit the water level in a discharging well (defined using the HEAD\_LIMIT keyword in the stress period data). SHUTDOWN\_KAPPA is used to control discharge rate oscillations when the flow rate from the aquifer is less than the specified flow rate from the aquifer to the well. Values range between 0.0 and 1.0, and larger values increase the weight applied to the well discharge rate. The HEAD\_LIMIT option has been included to facilitate backward compatibility with previous versions of MODFLOW but use of the RATE\_SCALING option instead of the HEAD\_LIMIT option is recommended. By default, SHUTDOWN\_KAPPA is 0.0001.

\item \texttt{MAW\_FLOW\_REDUCE\_CSV}---keyword to specify that record corresponds to the output option in which a new record is written for each multi-aquifer well and for each time step in which the user-requested extraction or injection rate is reduced by the program.

\item \texttt{FILEOUT}---keyword to specify that an output filename is expected next.

\item \texttt{mfrcsvfile}---name of the comma-separated value (CSV) output file to write information about multi-aquifer well extraction or injection rates that have been reduced by the program. Entries are only written if the extraction or injection rates are reduced.

\item \texttt{TS6}---keyword to specify that record corresponds to a time-series file.

\item \texttt{FILEIN}---keyword to specify that an input filename is expected next.
Expand Down
Loading

0 comments on commit 99ad83b

Please sign in to comment.