Skip to content

Commit 255be72

Browse files
authored
Merge pull request #140 from wind-python/dev
Merge into master for release 0.2.2
2 parents 70f09b4 + 8f11ea4 commit 255be72

35 files changed

+709
-206
lines changed

.github/workflows/tests-coverage.yml

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
# Tests with pytest the package and monitors the coverage and sends it to coveralls.io
2+
# Coverage is only send to coveralls.io when no pytest tests fail
3+
name: "Tests & Coverage"
4+
5+
on: [push]
6+
7+
# Cancel jobs on new push
8+
concurrency:
9+
group: ${{ github.workflow }}-${{ github.ref || github.run_id }}
10+
cancel-in-progress: true
11+
12+
jobs:
13+
build:
14+
name: "${{ matrix.name-suffix }} at py${{ matrix.python-version }} on ${{ matrix.os }}"
15+
runs-on: ${{ matrix.os }}
16+
17+
strategy:
18+
fail-fast: false
19+
matrix:
20+
include:
21+
- name-suffix: "coverage"
22+
os: ubuntu-latest
23+
python-version: "3.11"
24+
- name-suffix: "basic"
25+
os: ubuntu-latest
26+
python-version: "3.10"
27+
- name-suffix: "basic"
28+
os: ubuntu-latest
29+
python-version: "3.12"
30+
- name-suffix: "basic"
31+
os: windows-latest
32+
python-version: "3.11"
33+
34+
steps:
35+
- name: Checkout repo
36+
uses: actions/checkout@v3
37+
38+
- name: Set up Python
39+
uses: actions/setup-python@v4
40+
with:
41+
python-version: ${{ matrix.python-version }}
42+
43+
- name: Set up Conda
44+
if: runner.os == 'Windows'
45+
uses: conda-incubator/setup-miniconda@v2
46+
with:
47+
miniconda-version: "latest"
48+
python-version: ${{ matrix.python-version }}
49+
activate-environment: testenv
50+
51+
- name: Install dependencies
52+
run: |
53+
python -m pip install --upgrade pip wheel setuptools
54+
python -m pip install .[dev]
55+
56+
- name: Run tests
57+
if: ${{ !(runner.os == 'Linux' && matrix.python-version == 3.9 && matrix.name-suffix == 'coverage') }}
58+
run: |
59+
python -m pytest --disable-warnings --color=yes -v
60+
61+
- name: Run tests, coverage and send to coveralls
62+
if: runner.os == 'Linux' && matrix.python-version == 3.9 && matrix.name-suffix == 'coverage'
63+
run: |
64+
coverage run --source=windpowerlib -m pytest --disable-warnings --color=yes -v
65+
coveralls
66+
env:
67+
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
68+
COVERALLS_SERVICE_NAME: github

.readthedocs.yml

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
# .readthedocs.yml
2+
# Read the Docs configuration file
3+
# See https://docs.readthedocs.io/en/stable/config-file/v2.html for details
4+
5+
# Required
6+
version: 2
7+
8+
# Build documentation in the docs/ directory with Sphinx
9+
sphinx:
10+
configuration: doc/conf.py
11+
12+
# Build documentation with MkDocs
13+
#mkdocs:
14+
# configuration: mkdocs.yml
15+
16+
# Optionally build your docs in additional formats such as PDF and ePub
17+
formats: all
18+
19+
# Optionally set the version of Python and requirements required to build your docs
20+
python:
21+
install:
22+
- requirements: doc/requirements.txt
23+
24+
# Set the version of Python
25+
build:
26+
os: ubuntu-22.04
27+
tools:
28+
python: "3.11"

.travis.yml

+3-3
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,9 @@ language: python
22

33
matrix:
44
include:
5-
- python: 3.6
6-
- python: 3.7
7-
- python: 3.8
5+
- python: 3.10
6+
- python: 3.11
7+
- python: 3.12
88

99
# command to install dependencies
1010
#before_install:

README.rst

+4-4
Original file line numberDiff line numberDiff line change
@@ -33,13 +33,13 @@ Go to the `download page <http://readthedocs.org/projects/windpowerlib/downloads
3333
Installation
3434
============
3535

36-
If you have a working Python 3 (>= 3.6) environment, use pypi to install the latest windpowerlib version:
36+
If you have a working Python 3 environment, use pypi to install the latest windpowerlib version:
3737

3838
::
3939

4040
pip install windpowerlib
4141

42-
The windpowerlib is designed for Python 3 and tested on Python >= 3.5. We highly recommend to use virtual environments.
42+
The windpowerlib is designed for Python 3 and tested on Python >= 3.10. We highly recommend to use virtual environments.
4343
Please see the `installation page <http://oemof.readthedocs.io/en/stable/installation_and_setup.html>`_ of the oemof documentation for complete instructions on how to install python and a virtual environment on your operating system.
4444

4545
Optional Packages
@@ -97,8 +97,8 @@ To update your local files with the latest version of the `oedb turbine library
9797

9898
.. code:: python
9999
100-
from windpowerlib.wind_turbine import load_turbine_data_from_oedb
101-
load_turbine_data_from_oedb()
100+
from windpowerlib.data import store_turbine_data_from_oedb
101+
store_turbine_data_from_oedb()
102102
103103
If you find your turbine in the database it is very easy to use it in the
104104
windpowerlib

doc/conf.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@
6969

7070
# General information about the project.
7171
project = u'windpowerlib'
72-
copyright = u'2016-2021, oemof developer group'
72+
copyright = u'2016-2023, oemof developer group'
7373
author = u'oemof developer group'
7474

7575
import windpowerlib

doc/getting_started.rst

+7-7
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ If you have a working Python 3 environment, use pypi to install the latest windp
4343

4444
pip install windpowerlib
4545

46-
The windpowerlib is designed for Python 3 and tested on Python >= 3.5. We highly recommend to use virtual environments.
46+
The windpowerlib is designed for Python 3 and tested on Python >= 3.10. We highly recommend to use virtual environments.
4747
Please see the `installation page <http://oemof.readthedocs.io/en/stable/installation_and_setup.html>`_ of the oemof documentation for complete instructions on how to install python and a virtual environment on your operating system.
4848

4949
Optional Packages
@@ -69,9 +69,9 @@ The basic usage of the windpowerlib is shown in the ModelChain example that is a
6969
To run the example you need the example weather and turbine data used:
7070

7171
* :download:`Example weather data file <../example/weather.csv>`
72-
* :download:`Example power curve data file <../example/data/power_curves.csv>`
73-
* :download:`Example power coefficient curve data file <../example/data/power_coefficient_curves.csv>`
74-
* :download:`Example nominal power data file <../example/data/turbine_data.csv>`
72+
* :download:`Example power curve data file <../windpowerlib/data/default_turbine_data/power_curves.csv>`
73+
* :download:`Example power coefficient curve data file <../windpowerlib/data/default_turbine_data/power_coefficient_curves.csv>`
74+
* :download:`Example nominal power data file <../windpowerlib/data/default_turbine_data/turbine_data.csv>`
7575

7676
Furthermore, you have to install the windpowerlib and to run the notebook you also need to install `notebook` using pip3. To launch jupyter notebook type ``jupyter notebook`` in the terminal.
7777
This will open a browser window. Navigate to the directory containing the notebook to open it. See the jupyter notebook quick start guide for more information on `how to install <http://jupyter-notebook-beginner-guide.readthedocs.io/en/latest/install.html>`_ and
@@ -102,8 +102,8 @@ To update your local files with the latest version of the `oedb turbine library
102102

103103
.. code:: python
104104
105-
from windpowerlib.wind_turbine import load_turbine_data_from_oedb
106-
load_turbine_data_from_oedb()
105+
from windpowerlib.data import store_turbine_data_from_oedb
106+
store_turbine_data_from_oedb()
107107
108108
If you find your turbine in the database it is very easy to use it in the
109109
windpowerlib
@@ -223,4 +223,4 @@ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
223223
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
224224
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
225225
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
226-
SOFTWARE.
226+
SOFTWARE.

doc/requirements.txt

+1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
sphinx>=1.4
2+
sphinx_rtd_theme
23
ipykernel
34
nbsphinx
45
pandas

doc/whats_new.rst

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ These are new features and improvements of note in each release
88
:local:
99
:backlinks: top
1010

11+
.. include:: whatsnew/v0-2-2.rst
1112
.. include:: whatsnew/v0-2-1.rst
1213
.. include:: whatsnew/v0-2-0.rst
1314
.. include:: whatsnew/v0-1-3.rst

doc/whatsnew/v0-2-2.rst

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
v0.2.2 (February 20, 2024)
2+
++++++++++++++++++++++++++++++
3+
4+
* Updated the code basis to work for newer versions of python (support for python 3.6 to
5+
python 3.9 is discontinued, supported python versions are now >= python 3.9) and added
6+
github actions to run tests automatically when changes are pushed to github
7+
(`PR 136 <https://github.com/wind-python/windpowerlib/pull/136>`_).
8+
9+
Contributors
10+
############
11+
* Birgit Schachler
12+
* Florian Maurer

example/modelchain_example.ipynb

+3-2
Original file line numberDiff line numberDiff line change
@@ -144,7 +144,8 @@
144144
" file,\n",
145145
" index_col=0,\n",
146146
" header=[0, 1],\n",
147-
" date_parser=lambda idx: pd.to_datetime(idx, utc=True))\n",
147+
" )\n",
148+
" weather_df.index = pd.to_datetime(weather_df.index, utc=True)\n",
148149
" \n",
149150
" # change time zone\n",
150151
" weather_df.index = weather_df.index.tz_convert(\n",
@@ -517,7 +518,7 @@
517518
" x='wind_speed', y='value', style='*',\n",
518519
" title='Enercon E126 power coefficient curve')\n",
519520
" plt.xlabel('Wind speed in m/s')\n",
520-
" plt.ylabel('Power in W')\n",
521+
" plt.ylabel('Power coefficient $\\mathrm{C}_\\mathrm{P}$')\n",
521522
" plt.show()\n",
522523
" if e126.power_curve is not None:\n",
523524
" e126.power_curve.plot(x='wind_speed', y='value', style='*',\n",

example/modelchain_example.py

+2-2
Original file line numberDiff line numberDiff line change
@@ -85,8 +85,8 @@ def get_weather_data(filename="weather.csv", **kwargs):
8585
file,
8686
index_col=0,
8787
header=[0, 1],
88-
date_parser=lambda idx: pd.to_datetime(idx, utc=True),
8988
)
89+
weather_df.index = pd.to_datetime(weather_df.index, utc=True)
9090

9191
# change time zone
9292
weather_df.index = weather_df.index.tz_convert("Europe/Berlin")
@@ -274,7 +274,7 @@ def plot_or_print(my_turbine, e126, my_turbine2):
274274
title="Enercon E126 power curve",
275275
)
276276
plt.xlabel("Wind speed in m/s")
277-
plt.ylabel("Power in W")
277+
plt.ylabel("Power coefficient $\mathrm{C}_\mathrm{P}$")
278278
plt.show()
279279
if my_turbine.power_curve is not False:
280280
my_turbine.power_curve.plot(

example/simple_example.py

+145
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,145 @@
1+
"""
2+
The ``turbine_cluster_modelchain_example`` module shows how to calculate the
3+
power output of wind farms and wind turbine clusters with the windpowerlib.
4+
A cluster can be useful if you want to calculate the feed-in of a region for
5+
which you want to use one single weather data point.
6+
7+
Functions that are used in the ``modelchain_example``, like the initialization
8+
of wind turbines, are imported and used without further explanations.
9+
10+
SPDX-FileCopyrightText: 2019 oemof developer group <contact@oemof.org>
11+
SPDX-License-Identifier: MIT
12+
"""
13+
import pandas as pd
14+
import requests
15+
import os
16+
17+
from windpowerlib import WindFarm
18+
from windpowerlib import WindTurbine
19+
from windpowerlib import TurbineClusterModelChain
20+
from windpowerlib import WindTurbineCluster
21+
22+
# You can use the logging package to get logging messages from the windpowerlib
23+
# Change the logging level if you want more or less messages
24+
import logging
25+
26+
logging.getLogger().setLevel(logging.INFO)
27+
28+
29+
def get_weather_data(filename="weather.csv", **kwargs):
30+
r"""
31+
Imports weather data from a file.
32+
33+
The data include wind speed at two different heights in m/s, air
34+
temperature in two different heights in K, surface roughness length in m
35+
and air pressure in Pa. The height in m for which the data applies is
36+
specified in the second row.
37+
In case no weather data file exists, an example weather data file is
38+
automatically downloaded and stored in the same directory as this example.
39+
40+
Parameters
41+
----------
42+
filename : str
43+
Filename of the weather data file. Default: 'weather.csv'.
44+
45+
Other Parameters
46+
----------------
47+
datapath : str, optional
48+
Path where the weather data file is stored.
49+
Default is the same directory this example is stored in.
50+
51+
Returns
52+
-------
53+
:pandas:`pandas.DataFrame<frame>`
54+
DataFrame with time series for wind speed `wind_speed` in m/s,
55+
temperature `temperature` in K, roughness length `roughness_length`
56+
in m, and pressure `pressure` in Pa.
57+
The columns of the DataFrame are a MultiIndex where the first level
58+
contains the variable name as string (e.g. 'wind_speed') and the
59+
second level contains the height as integer at which it applies
60+
(e.g. 10, if it was measured at a height of 10 m). The index is a
61+
DateTimeIndex.
62+
63+
"""
64+
65+
if "datapath" not in kwargs:
66+
kwargs["datapath"] = os.path.dirname(__file__)
67+
68+
file = os.path.join(kwargs["datapath"], filename)
69+
70+
# download example weather data file in case it does not yet exist
71+
if not os.path.isfile(file):
72+
logging.debug("Download weather data for example.")
73+
req = requests.get("https://osf.io/59bqn/download")
74+
with open(file, "wb") as fout:
75+
fout.write(req.content)
76+
77+
# read csv file
78+
weather_df = pd.read_csv(
79+
file,
80+
index_col=0,
81+
header=[0, 1],
82+
)
83+
weather_df.index = pd.to_datetime(weather_df.index, utc=True)
84+
85+
# change time zone
86+
weather_df.index = weather_df.index.tz_convert("Europe/Berlin")
87+
88+
return weather_df
89+
90+
91+
def run_example():
92+
r"""
93+
Runs the example.
94+
95+
"""
96+
weather = get_weather_data("weather.csv")
97+
e126 = WindTurbine(
98+
**{
99+
"turbine_type": "E-126/4200", # turbine type as in register
100+
"hub_height": 135, # in m
101+
}
102+
)
103+
v90 = WindTurbine(
104+
**{
105+
"turbine_type": "V90/2000", # turbine name as in register
106+
"hub_height": 120, # in m
107+
}
108+
)
109+
110+
# specification of wind farm data (2) containing a wind farm efficiency
111+
# wind turbine fleet is provided using the to_group function
112+
example_farm = WindFarm(
113+
**{
114+
"name": "example_farm_2",
115+
"wind_turbine_fleet": [
116+
v90.to_group(number_turbines=6),
117+
e126.to_group(total_capacity=12.6e6),
118+
],
119+
"efficiency": 0.5
120+
}
121+
)
122+
123+
example_cluster = WindTurbineCluster(**{
124+
"name": "Offshore_cluster",
125+
"wind_farms": [example_farm, example_farm],
126+
})
127+
128+
# ModelChain with wind farm
129+
mc_farm = TurbineClusterModelChain(
130+
example_farm, wake_losses_model=None,
131+
).run_model(weather)
132+
flh_farm = mc_farm.power_output.sum() / example_farm.nominal_power
133+
134+
# ModelChain with wind cluster
135+
mc_cluster = TurbineClusterModelChain(
136+
example_cluster, wake_losses_model=None,
137+
).run_model(weather)
138+
flh_cluster = mc_cluster.power_output.sum() / example_cluster.nominal_power
139+
140+
print("Full Load Hours of cluster:", flh_cluster)
141+
print("Full Load Hours of farm:", flh_farm)
142+
143+
144+
if __name__ == "__main__":
145+
run_example()

0 commit comments

Comments
 (0)