-
Notifications
You must be signed in to change notification settings - Fork 5
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Neuman sloution Moench et al. 2001 #6
base: main
Are you sure you want to change the base?
Changes from 3 commits
e95b3f5
a7c56cd
f26a9cc
b618afd
6201b0a
6ec28c5
728c5ba
1a3f275
6db00fc
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,89 @@ | ||
name: Continuous Integration | ||
|
||
on: | ||
push: | ||
branches: | ||
- "master" | ||
- "develop" | ||
tags: | ||
- "*" | ||
pull_request: | ||
branches: | ||
- "develop" | ||
# Allows you to run this workflow manually from the Actions tab | ||
workflow_dispatch: | ||
|
||
env: | ||
# needed by coveralls | ||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||
|
||
jobs: | ||
build_sdist: | ||
name: sdist on ${{ matrix.os }} with py ${{ matrix.python-version }} | ||
runs-on: ${{ matrix.os }} | ||
strategy: | ||
fail-fast: false | ||
matrix: | ||
os: [ubuntu-latest, windows-latest, macos-latest] | ||
python-version: [3.5, 3.6, 3.7, 3.8, 3.9] | ||
|
||
steps: | ||
- uses: actions/checkout@v2 | ||
with: | ||
fetch-depth: '0' | ||
|
||
- name: Set up Python ${{ matrix.python-version }} | ||
uses: actions\setup-python@v2 | ||
with: | ||
python-version: ${{ matrix.python-version }} | ||
|
||
- name: Install dependencies | ||
run: | | ||
python -m pip install --upgrade pip | ||
pip install -U wheel | ||
pip install -r requirements_setup.txt | ||
pip install -r requirements.txt | ||
pip install -r requirements_test.txt | ||
pip install coveralls>=3.0.0 | ||
|
||
- name: Build sdist | ||
run: | | ||
python setup.py sdist --formats=gztar bdist_wheel -d dist | ||
|
||
- name: Run tests | ||
run: | | ||
python -m pytest --cov anaflow --cov-report term-missing -v tests/ | ||
python -m coveralls --service=github | ||
|
||
- uses: actions/upload-artifact@v2 | ||
if: matrix.os == 'ubuntu-latest' && matrix.python-version == '3.9' | ||
with: | ||
path: dist | ||
|
||
upload_to_pypi: | ||
needs: [build_sdist] | ||
runs-on: ubuntu-latest | ||
|
||
steps: | ||
- uses: actions/download-artifact@v2 | ||
with: | ||
name: artifact | ||
path: dist | ||
|
||
- name: Publish to Test PyPI | ||
# only if working on develop | ||
if: github.ref == 'refs/heads/master' || github.ref == 'refs/heads/develop' | ||
uses: pypa/gh-action-pypi-publish@master | ||
with: | ||
user: __token__ | ||
password: ${{ secrets.test_pypi_password }} | ||
repository_url: https://test.pypi.org/legacy/ | ||
skip_existing: true | ||
|
||
- name: Publish to PyPI | ||
# only if tagged | ||
if: startsWith(github.ref, 'refs/tags') | ||
uses: pypa/gh-action-pypi-publish@master | ||
with: | ||
user: __token__ | ||
password: ${{ secrets.pypi_password }} |
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,162 @@ | ||
# -*- coding: utf-8 -*- | ||
""" | ||
Anaflow subpackage providing the Neuman equation for homogeneous aquifer. | ||
|
||
.. currentmodule:: anaflow.flow.Neuman | ||
|
||
The following functions are provided | ||
|
||
.. autosummary:: | ||
|
||
""" | ||
# pylint: disable=C0103 | ||
import numpy as np | ||
from scipy.special import k0, k1 | ||
from scipy.integrate import quad as integ | ||
|
||
import matplotlib.pyplot as plt | ||
|
||
from anaflow.tools.laplace import get_lap_inv | ||
from anaflow.flow.laplace import grf_laplace | ||
from anaflow.tools.special import Shaper, sph_surf | ||
|
||
__all__ = [] | ||
|
||
|
||
def neuman( | ||
s, | ||
rad, | ||
storage, | ||
transmissivity, | ||
sat_thickness=52.0, | ||
rate=-1e-4, | ||
d=0.6, | ||
l=12.6, | ||
rw=0.1, | ||
kz=0.14, | ||
kr=0.23, | ||
ss=4.27e-5, | ||
n_numbers=52, | ||
): | ||
""" | ||
The Neuman solution. | ||
|
||
|
||
Parameters | ||
---------- | ||
s : :class:`numpy.ndarray` | ||
Array with all time-points where the function should be evaluated in the Laplace space. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. "Laplace-space-points" not "time-points" |
||
rad : :class:`numpy.ndarray` | ||
Array with all radii where the function should be evaluated. | ||
storage : :class:`float` | ||
Storage of the aquifer. | ||
transmissivity : :class:`float` | ||
Geometric-mean transmissivity. | ||
sat_thickness : :class:`float`, optional | ||
Saturated thickness of the aquifer. | ||
rate : :class:`float`, optional | ||
Pumpingrate at the well. Default: -1e-4 | ||
d : :class:`float`, optional | ||
Vertical distance from initial water table to top of | ||
pumped well screen | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Would use |
||
l : :class:`float`, optional | ||
Vertical distance from initial water table to bottom | ||
of pumped well screen | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. would rename to |
||
rw : :class:`float`, optional | ||
Outside radius of the pumped well screen | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I would use the solution from Moench 1993, where infinitely small wells are assumed. I think we will run into numerical problems, that we don't have the time to solve. So I would vote for leaving |
||
kz : :class:`float`, optional | ||
Hydraulic conductivity in the vertical direction | ||
kr : :class:`float`, optional | ||
Hydraulic conductivity in the horizontal direction | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. actually we don't need these two values but rather a single |
||
Ss : :class:`float`, optional | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. would rename to |
||
specific storage | ||
""" | ||
z = float(sat_thickness - l) | ||
kd = kz / kr | ||
s = np.squeeze(s).reshape(-1) | ||
rad = np.squeeze(rad).reshape(-1) | ||
res = np.zeros(s.shape + rad.shape) | ||
|
||
for si, se in enumerate(s): | ||
for ri, re in enumerate(rad): | ||
if re < np.inf: | ||
for n in range(n_numbers): | ||
epsilon_n = np.pi / 2 + n * np.pi | ||
betaw = kd * rw ** 2 | ||
qn = ((epsilon_n ** 2) * betaw + se) ** 0.5 | ||
rd = re / rw | ||
E = k0(qn * rd) * np.cos(epsilon_n * (z / sat_thickness)) / ( | ||
( | ||
np.sin(epsilon_n * (1 - (d / sat_thickness))) | ||
- np.sin(epsilon_n * (1 - (l / sat_thickness))) | ||
) | ||
/ (qn * k1(qn) * (epsilon_n + 0.5 * np.sin(2 * epsilon_n))) | ||
) | ||
A = k0(qn) * ( | ||
( | ||
np.sin(epsilon_n * (1 - (d / sat_thickness))) | ||
- np.sin(epsilon_n * (1 - (l / sat_thickness))) | ||
) | ||
** 2 | ||
/ ( | ||
epsilon_n | ||
* qn | ||
* k1(qn) | ||
* (epsilon_n + 0.5 * np.sin(2 * epsilon_n)) | ||
) | ||
) | ||
E = 2 * E | ||
A = (2 / ((l - d) / sat_thickness)) * A | ||
sw = storage | ||
wd = np.pi * rw ** 2 / (2 * np.pi * rw ** 2 * ss * (l - d)) | ||
res[si, ri] = (2 * E) / (se * ((l - d) / sat_thickness) * (1 + wd * se * (A + sw))) | ||
res *= rate / (2 * np.pi * transmissivity) | ||
return res | ||
|
||
|
||
def neuman_from_laplace( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. rename to |
||
time, | ||
rad, | ||
storage, | ||
transmissivity, | ||
rate=-1e-4, | ||
h_bound=0.0, | ||
struc_grid=True, | ||
lap_kwargs=None, | ||
): | ||
"""Neuman solution form laplace solution.""" | ||
Input = Shaper(time, rad, struc_grid) | ||
lap_kwargs = {} if lap_kwargs is None else lap_kwargs | ||
|
||
if not transmissivity > 0.0: | ||
raise ValueError("The Transmissivity needs to be positive.") | ||
if not storage > 0.0: | ||
raise ValueError("The Storage needs to be positive.") | ||
kwargs = { | ||
"rad": rad, | ||
"storage": storage, | ||
"transmissivity": transmissivity, | ||
"rate": rate, | ||
} | ||
kwargs.update(lap_kwargs) | ||
res = np.zeros((Input.time_no, Input.rad_no)) | ||
lap_inv = get_lap_inv(neuman, **kwargs) | ||
# call the laplace inverse function (only at time points > 0) | ||
res[Input.time_gz, :] = lap_inv(Input.time[Input.time_gz]) | ||
# reshaping results | ||
res = Input.reshape(res) | ||
# add the reference head | ||
res += h_bound | ||
return res | ||
|
||
|
||
time = [10, 600, 36000] # 10s, 10min, 10h | ||
rad = np.geomspace(0.1, 10) | ||
# default parameters | ||
T = 1e-4 | ||
S = 1e-4 | ||
Q = -1e-4 | ||
|
||
neu = neuman_from_laplace(time, rad, storage=S, transmissivity=T) | ||
for i in range(len(time)): | ||
plt.plot(rad, neu[i, :], color="C0", linestyle=":", linewidth=4) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Don't forget to remove this after testing. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
rename to
neuman_unconfined_laplace