2020-05-29 07:50:55 +00:00
|
|
|
# SPDX-FileCopyrightText: : 2017-2020 The PyPSA-Eur Authors
|
|
|
|
#
|
2021-09-14 14:37:41 +00:00
|
|
|
# SPDX-License-Identifier: MIT
|
2020-05-29 07:50:55 +00:00
|
|
|
|
2018-02-10 16:16:20 +00:00
|
|
|
# coding: utf-8
|
2019-08-08 13:02:28 +00:00
|
|
|
"""
|
2019-11-09 08:08:25 +00:00
|
|
|
Retrieves conventional powerplant capacities and locations from `powerplantmatching <https://github.com/FRESNA/powerplantmatching>`_, assigns these to buses and creates a ``.csv`` file. It is possible to amend the powerplant database with custom entries provided in ``data/custom_powerplants.csv``.
|
2019-08-11 09:40:47 +00:00
|
|
|
|
|
|
|
Relevant Settings
|
|
|
|
-----------------
|
|
|
|
|
2019-08-11 20:34:18 +00:00
|
|
|
.. code:: yaml
|
|
|
|
|
2019-10-31 17:01:33 +00:00
|
|
|
electricity:
|
2019-11-01 12:27:42 +00:00
|
|
|
powerplants_filter:
|
|
|
|
custom_powerplants:
|
2019-08-11 11:17:36 +00:00
|
|
|
|
2019-10-30 22:09:41 +00:00
|
|
|
.. seealso::
|
2019-08-13 08:03:46 +00:00
|
|
|
Documentation of the configuration file ``config.yaml`` at
|
2019-11-06 14:59:34 +00:00
|
|
|
:ref:`electricity`
|
2019-08-13 08:03:46 +00:00
|
|
|
|
2019-08-11 09:40:47 +00:00
|
|
|
Inputs
|
|
|
|
------
|
|
|
|
|
2019-08-11 20:34:18 +00:00
|
|
|
- ``networks/base.nc``: confer :ref:`base`.
|
2019-11-06 14:59:34 +00:00
|
|
|
- ``data/custom_powerplants.csv``: custom powerplants in the same format as `powerplantmatching <https://github.com/FRESNA/powerplantmatching>`_ provides
|
2019-08-11 20:34:18 +00:00
|
|
|
|
2019-08-11 09:40:47 +00:00
|
|
|
Outputs
|
|
|
|
-------
|
|
|
|
|
2019-08-12 17:01:53 +00:00
|
|
|
- ``resource/powerplants.csv``: A list of conventional power plants (i.e. neither wind nor solar) with fields for name, fuel type, technology, country, capacity in MW, duration, commissioning year, retrofit year, latitude, longitude, and dam information as documented in the `powerplantmatching README <https://github.com/FRESNA/powerplantmatching/blob/master/README.md>`_; additionally it includes information on the closest substation/bus in ``networks/base.nc``.
|
|
|
|
|
2019-08-14 13:36:46 +00:00
|
|
|
.. image:: ../img/powerplantmatching.png
|
2019-08-12 17:01:53 +00:00
|
|
|
:scale: 30 %
|
|
|
|
|
|
|
|
**Source:** `powerplantmatching on GitHub <https://github.com/FRESNA/powerplantmatching>`_
|
2019-08-11 20:34:18 +00:00
|
|
|
|
2019-08-11 09:40:47 +00:00
|
|
|
Description
|
|
|
|
-----------
|
|
|
|
|
2019-11-06 14:59:34 +00:00
|
|
|
The configuration options ``electricity: powerplants_filter`` and ``electricity: custom_powerplants`` can be used to control whether data should be retrieved from the original powerplants database or from custom amendmends. These specify `pandas.query <https://pandas.pydata.org/pandas-docs/stable/reference/api/pandas.DataFrame.query.html>`_ commands.
|
|
|
|
|
|
|
|
1. Adding all powerplants from custom:
|
|
|
|
|
|
|
|
.. code:: yaml
|
|
|
|
|
|
|
|
powerplants_filter: false
|
|
|
|
custom_powerplants: true
|
|
|
|
|
|
|
|
2. Replacing powerplants in e.g. Germany by custom data:
|
|
|
|
|
|
|
|
.. code:: yaml
|
|
|
|
|
|
|
|
powerplants_filter: Country not in ['Germany']
|
|
|
|
custom_powerplants: true
|
|
|
|
|
2019-11-09 08:08:25 +00:00
|
|
|
or
|
2019-11-06 14:59:34 +00:00
|
|
|
|
|
|
|
.. code:: yaml
|
|
|
|
|
|
|
|
powerplants_filter: Country not in ['Germany']
|
|
|
|
custom_powerplants: Country in ['Germany']
|
|
|
|
|
|
|
|
|
|
|
|
3. Adding additional built year constraints:
|
|
|
|
|
|
|
|
.. code:: yaml
|
|
|
|
|
|
|
|
powerplants_filter: Country not in ['Germany'] and YearCommissioned <= 2015
|
|
|
|
custom_powerplants: YearCommissioned <= 2015
|
|
|
|
|
2019-08-08 13:02:28 +00:00
|
|
|
"""
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-10-31 13:48:10 +00:00
|
|
|
import logging
|
2019-11-28 07:22:52 +00:00
|
|
|
from _helpers import configure_logging
|
|
|
|
|
2018-02-10 16:16:20 +00:00
|
|
|
import pypsa
|
2019-10-30 22:09:41 +00:00
|
|
|
import powerplantmatching as pm
|
2019-10-31 13:30:05 +00:00
|
|
|
import pandas as pd
|
2020-02-03 14:01:24 +00:00
|
|
|
import numpy as np
|
2019-10-31 13:30:05 +00:00
|
|
|
|
2020-12-03 18:50:53 +00:00
|
|
|
from scipy.spatial import cKDTree as KDTree
|
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2021-09-14 14:34:02 +00:00
|
|
|
def add_custom_powerplants(ppl, custom_powerplants, custom_ppl_query=False):
|
2019-10-31 13:48:10 +00:00
|
|
|
if not custom_ppl_query:
|
2019-08-19 16:04:51 +00:00
|
|
|
return ppl
|
2021-09-14 14:34:02 +00:00
|
|
|
add_ppls = pd.read_csv(custom_powerplants, index_col=0,
|
2019-12-02 23:52:52 +00:00
|
|
|
dtype={'bus': 'str'})
|
2019-10-31 13:48:10 +00:00
|
|
|
if isinstance(custom_ppl_query, str):
|
2019-12-02 23:52:52 +00:00
|
|
|
add_ppls.query(custom_ppl_query, inplace=True)
|
2020-03-17 08:50:38 +00:00
|
|
|
return ppl.append(add_ppls, sort=False, ignore_index=True, verify_integrity=True)
|
2019-10-31 13:48:10 +00:00
|
|
|
|
2018-11-12 20:40:05 +00:00
|
|
|
|
2019-02-05 22:00:35 +00:00
|
|
|
if __name__ == "__main__":
|
|
|
|
if 'snakemake' not in globals():
|
2019-12-09 20:29:15 +00:00
|
|
|
from _helpers import mock_snakemake
|
|
|
|
snakemake = mock_snakemake('build_powerplants')
|
2019-11-28 07:22:52 +00:00
|
|
|
configure_logging(snakemake)
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-02-05 22:00:35 +00:00
|
|
|
n = pypsa.Network(snakemake.input.base_network)
|
2019-10-30 22:09:41 +00:00
|
|
|
countries = n.buses.country.unique()
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-10-30 22:09:41 +00:00
|
|
|
ppl = (pm.powerplants(from_url=True)
|
2019-11-09 08:08:25 +00:00
|
|
|
.powerplant.fill_missing_decommyears()
|
2019-10-30 22:09:41 +00:00
|
|
|
.powerplant.convert_country_to_alpha2()
|
|
|
|
.query('Fueltype not in ["Solar", "Wind"] and Country in @countries')
|
|
|
|
.replace({'Technology': {'Steam Turbine': 'OCGT'}})
|
|
|
|
.assign(Fueltype=lambda df: (
|
|
|
|
df.Fueltype
|
|
|
|
.where(df.Fueltype != 'Natural Gas',
|
|
|
|
df.Technology.replace('Steam Turbine',
|
|
|
|
'OCGT').fillna('OCGT')))))
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-10-31 13:59:51 +00:00
|
|
|
ppl_query = snakemake.config['electricity']['powerplants_filter']
|
2019-10-31 13:48:10 +00:00
|
|
|
if isinstance(ppl_query, str):
|
|
|
|
ppl.query(ppl_query, inplace=True)
|
2018-07-10 14:29:11 +00:00
|
|
|
|
2021-09-14 14:34:02 +00:00
|
|
|
# add carriers from own powerplant files:
|
|
|
|
ppl = add_custom_powerplants(ppl, custom_powerplants = snakemake.input.custom_powerplants,
|
|
|
|
custom_ppl_query = snakemake.config['electricity']['custom_powerplants'])
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-10-30 22:09:41 +00:00
|
|
|
cntries_without_ppl = [c for c in countries if c not in ppl.Country.unique()]
|
2018-11-12 20:40:05 +00:00
|
|
|
|
2019-11-01 12:27:42 +00:00
|
|
|
for c in countries:
|
|
|
|
substation_i = n.buses.query('substation_lv and country == @c').index
|
|
|
|
kdtree = KDTree(n.buses.loc[substation_i, ['x','y']].values)
|
|
|
|
ppl_i = ppl.query('Country == @c').index
|
2018-11-12 20:40:05 +00:00
|
|
|
|
2020-02-03 14:01:24 +00:00
|
|
|
tree_i = kdtree.query(ppl.loc[ppl_i, ['lon','lat']].values)[1]
|
|
|
|
ppl.loc[ppl_i, 'bus'] = substation_i.append(pd.Index([np.nan]))[tree_i]
|
2018-11-12 20:40:05 +00:00
|
|
|
|
2019-02-05 22:00:35 +00:00
|
|
|
if cntries_without_ppl:
|
2019-10-30 22:09:41 +00:00
|
|
|
logging.warning(f"No powerplants known in: {', '.join(cntries_without_ppl)}")
|
2018-11-12 20:40:05 +00:00
|
|
|
|
2019-02-05 22:00:35 +00:00
|
|
|
bus_null_b = ppl["bus"].isnull()
|
|
|
|
if bus_null_b.any():
|
2019-10-30 22:09:41 +00:00
|
|
|
logging.warning(f"Couldn't find close bus for {bus_null_b.sum()} powerplants")
|
2018-02-10 16:16:20 +00:00
|
|
|
|
2019-02-05 22:00:35 +00:00
|
|
|
ppl.to_csv(snakemake.output[0])
|