Skip to content
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

Add JRC spectral factor correction #2088

Merged
merged 22 commits into from
Jun 20, 2024
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -14,5 +14,6 @@ Spectrum
spectrum.spectral_factor_firstsolar
spectrum.spectral_factor_sapm
spectrum.spectral_factor_pvspec
spectrum.spectral_factor_jrc
spectrum.sr_to_qe
spectrum.qe_to_sr
1 change: 1 addition & 0 deletions pvlib/spectrum/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@
spectral_factor_firstsolar,
spectral_factor_sapm,
spectral_factor_pvspec,
spectral_factor_jrc,
sr_to_qe,
qe_to_sr
)
101 changes: 101 additions & 0 deletions pvlib/spectrum/mismatch.py
Original file line number Diff line number Diff line change
Expand Up @@ -689,6 +689,107 @@ def spectral_factor_pvspec(airmass_absolute, clearsky_index,
return mismatch


def spectral_factor_jrc(airmass, clearsky_index, module_type=None,
coefficients=None):
r"""
Estimate a technology-specific spectral mismatch modifier from absolute
airmass and clear sky index using the JRC model.

The JRC spectral mismatch model includes the effects of cloud cover on
the irradiance spectrum. Model coefficients are derived using measurements
of irradiance and module performance at the Joint Research Centre (JRC) in
Ispra, Italy (45°48'N, 8°37'E). Coefficients for two module types are
RDaxini marked this conversation as resolved.
Show resolved Hide resolved
available via the ``module_type`` parameter. More details on the model can
be found in [1]_.

Parameters
----------
airmass_absolute : numeric
absolute (pressure-adjusted) airmass. [unitless]
RDaxini marked this conversation as resolved.
Show resolved Hide resolved

clearsky_index: numeric
clear sky index. [unitless]

module_type : str, optional
One of the following PV technology strings from [1]_:

* ``'cdte'``, - anonymous CdTe module.
RDaxini marked this conversation as resolved.
Show resolved Hide resolved
* ``'multisi'`` - anonymous multicrystalline Si module.

coefficients : array-like, optional
user-defined coefficients, if not using one of the default coefficient
sets via the ``module_type`` parameter.

Returns
-------
mismatch: numeric
spectral mismatch factor (unitless) which is multiplied
with broadband irradiance reaching a module's cells to estimate
effective irradiance, i.e., the irradiance that is converted to
electrical current.

Notes
-----
The JRC model parameterises the spectral mismatch factor as a function
of absolute air mass and the clear sky index as follows:

.. math::

M = 1 + a_1(e^{-k_c}-e^{-1}) + a_2(k_c-1)+a_3(AM-1.5),
RDaxini marked this conversation as resolved.
Show resolved Hide resolved

where :math:`M` is the spectral mismatch factor, :math:`k_c` is the clear
sky index, :math:`AM` is the air mass, :math:`e` is Euler's number, and
:math:`a_1, a_2, a_3` are module-specific coefficients. In the JRC model
publication, the model used to estimate the air mass (denoted as :math:`AM`
) is not stated. The clear sky index, which is the ratio of GHI to clear
sky GHI, uses the ESRA model [2]_ to estimate the clear sky GHI. Prior to
the calculation of :math:`k_c`, the irradiance measurements are corrected
for angle of incidence using the Martin and Ruiz model [3]_.

References
----------
.. [1] Huld, T., Tony, S., and Ewan, D., 2009. A simple model
RDaxini marked this conversation as resolved.
Show resolved Hide resolved
for estimating the influence of spectrum variations on PV performance.
In Proceedings of the 24th European Photovoltaic Solar Energy
Conference, Hamburg, Germany pp. 3385-3389. 2009. Accessed at:
https://www.researchgate.net/publication/256080247
.. [2] Rigollier, C., Bauer, O., and Wald, L., 2000. On the clear sky model
of the ESRA—European Solar Radiation Atlas—with respect to the Heliosat
method. Solar energy, 68(1), pp.33-48.
:doi:`10.1016/S0038-092X(99)00055-9`
.. [3] Martin, N. and Ruiz, J. M., 2001. Calculation of the PV modules
angular losses under field conditions by means of an analytical model.
Solar Energy Materials and Solar Cells, 70(1), 25-38.
:doi:`10.1016/S0927-0248(00)00408-6`
"""

_coefficients = {}
_coefficients['multisi'] = (0.00172, 0.000508, 0.00000357)
_coefficients['cdte'] = (0.000643, 0.000130, 0.0000108)

if module_type is not None and coefficients is None:
coefficients = _coefficients[module_type.lower()]
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm a bit in favour of handling KeyError's and having custom error messages with the available keys, but I remember some comment somewhere that deemed that unnecessary. Just in case someone else wants to weight in, I'm not requesting changes.

elif module_type is None and coefficients is not None:
pass
elif module_type is None and coefficients is None:
raise ValueError('No valid input provided, both module_type and ' +
'coefficients are None. module_type can be one of ' +
", ".join(_coefficients.keys()))
else:
raise ValueError('Cannot resolve input, must supply only one of ' +
'module_type and coefficients. module_type can be ' +
'one of' ", ".join(_coefficients.keys()))

coeff = coefficients
mismatch = (
1
+ coeff[0] * (np.exp(-clearsky_index) - np.exp(-1))
+ coeff[1] * (clearsky_index - 1)
+ coeff[2] * (airmass - 1.5)
)
return mismatch


def sr_to_qe(sr, wavelength=None, normalize=False):
"""
Convert spectral responsivities to quantum efficiencies.
Expand Down
48 changes: 48 additions & 0 deletions pvlib/tests/test_spectrum.py
Original file line number Diff line number Diff line change
Expand Up @@ -374,6 +374,54 @@ def test_spectral_factor_pvspec_supplied_ambiguous():
coefficients=None)


@pytest.mark.parametrize("module_type,expected", [
('multisi', np.array([1.00021361, 1.00010800, 1.00004028, 0.99999459])),
('cdte', np.array([1.00011106, 1.00006434, 1.00003177, 0.99997402])),
])
def test_spectral_factor_jrc(module_type, expected):
ams = np.array([1.0, 1.5, 2.0, 1.5])
kcs = np.array([0.4, 0.6, 0.8, 1.4])
out = spectrum.spectral_factor_jrc(ams, kcs,
module_type=module_type)
assert np.allclose(expected, out, atol=1e-8)


@pytest.mark.parametrize("module_type,expected", [
('multisi', np.array([1.00021361, 1.00010800, 1.00004028, 0.99999459])),
('cdte', np.array([1.00011106, 1.00006434, 1.00003177, 0.99997402])),
])
def test_spectral_factor_jrc_series(module_type, expected):
ams = pd.Series([1.0, 1.5, 2.0, 1.5])
kcs = pd.Series([0.4, 0.6, 0.8, 1.4])
out = spectrum.spectral_factor_jrc(ams, kcs,
module_type=module_type)
assert isinstance(out, pd.Series)
assert np.allclose(expected, out, atol=1e-8)


def test_spectral_factor_jrc_supplied():
# use the multisi coeffs
coeffs = (0.00172, 0.000508, 0.00000357)
out = spectrum.spectral_factor_jrc(1.0, 0.8, coefficients=coeffs)
expected = 1.00003671
assert_allclose(out, expected, atol=1e-5)


def test_spectral_factor_jrc_supplied_redundant():
# Error when specifying both module_type and coefficients
coeffs = (0.00172, 0.000508, 0.00000357)
with pytest.raises(ValueError, match='supply only one of'):
spectrum.spectral_factor_jrc(1.0, 0.8, module_type='multisi',
coefficients=coeffs)


def test_spectral_factor_jrc_supplied_ambiguous():
# Error when specifying neither module_type nor coefficients
with pytest.raises(ValueError, match='No valid input provided'):
spectrum.spectral_factor_jrc(1.0, 0.8, module_type=None,
coefficients=None)


@pytest.fixture
def sr_and_eqe_fixture():
# Just some arbitrary data for testing the conversion functions
Expand Down
Loading