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

Projected zenith convenience function #1904

Merged
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
60 commits
Select commit Hold shift + click to select a range
1e412ce
Function prototype
echedey-ls Nov 6, 2023
e4a95a3
Update shading.rst
echedey-ls Nov 6, 2023
54878db
Update shading.py
echedey-ls Nov 6, 2023
b79ebe6
Minimal test
echedey-ls Nov 6, 2023
c3c0e56
Implementation
echedey-ls Nov 8, 2023
5d4a2b4
Fix, fix, fix, fix & format
echedey-ls Nov 8, 2023
9ae5fa3
Format issues
echedey-ls Nov 8, 2023
f17379d
Extend tests (compare with singleaxis) & format with ruff
echedey-ls Nov 8, 2023
af3d27c
Format fixes
echedey-ls Nov 8, 2023
d36841a
Upgrade tests
echedey-ls Nov 9, 2023
19995e1
Array -> Axis
echedey-ls Nov 9, 2023
770e037
type
echedey-ls Nov 9, 2023
29bcef9
Whatsnew
echedey-ls Nov 9, 2023
61c2e3b
xd
echedey-ls Nov 9, 2023
935cfd4
Merge branch 'main' into projected-solar-zenith-angle-issue1734
echedey-ls Nov 9, 2023
965b0b4
bruh
echedey-ls Nov 9, 2023
346d060
Minor Python optimization a la tracking.singleaxis
echedey-ls Nov 9, 2023
085d017
Comment and minor optimizations
echedey-ls Nov 9, 2023
79b5f4f
Typo found by Mikofski
echedey-ls Nov 9, 2023
dc1035a
Surface -> Axis
echedey-ls Nov 9, 2023
88cbfc0
Elevation -> Zenith
echedey-ls Nov 9, 2023
1afec94
Elev -> Zenith
echedey-ls Nov 9, 2023
4b2c0e5
Update shading.py
echedey-ls Jan 24, 2024
4612442
Update docstring
echedey-ls Jan 24, 2024
6372cb7
Add comments from `tracking.singleaxis`
echedey-ls Jan 24, 2024
3c9392b
Singleaxis implementation port & test addition, based on old pvlib.tr…
echedey-ls Jan 25, 2024
93998d8
Merge branch 'main' into projected-solar-zenith-angle-issue1734
echedey-ls Jan 25, 2024
337f7f1
Update v0.10.4.rst
echedey-ls Jan 25, 2024
0923109
Linter
echedey-ls Jan 25, 2024
6955f71
Code review
echedey-ls Jan 25, 2024
939b241
Add Fig 5 [1] (still gotta check the built output)
echedey-ls Jan 27, 2024
b0d6f66
Add caption, change size and describe in alternate text
echedey-ls Jan 27, 2024
3df2e71
rST fixes ?
echedey-ls Jan 27, 2024
428852c
Figures have captions, images do not
echedey-ls Jan 27, 2024
64c97c7
Merge branch 'main' into projected-solar-zenith-angle-issue1734
echedey-ls Feb 8, 2024
aed85a3
Flip arguments order
echedey-ls Feb 8, 2024
5d37fb3
I forgot :skull:
echedey-ls Feb 8, 2024
38c2d4d
Linter are you happy now?
echedey-ls Feb 8, 2024
23aaa2a
Remove port test and add edge cases test
echedey-ls Feb 9, 2024
2c0fa51
Update test_shading.py
echedey-ls Feb 9, 2024
5b49706
Indentation xd
echedey-ls Feb 9, 2024
1a68390
Update test_shading.py
echedey-ls Feb 9, 2024
58d853f
I forgot how to code
echedey-ls Feb 9, 2024
2ba4cf7
Align data
echedey-ls Feb 9, 2024
8325c37
Merge branch 'main' into projected-solar-zenith-angle-issue1734
echedey-ls Feb 24, 2024
f2fcc88
Merge branch 'main' into projected-solar-zenith-angle-issue1734
echedey-ls Feb 27, 2024
069688e
Docstring suggestion from Kevin
echedey-ls Feb 27, 2024
c249224
Update link to example?
echedey-ls Feb 27, 2024
f6c245f
Link, please work
echedey-ls Feb 27, 2024
b52f51d
Update shading.py
echedey-ls Feb 27, 2024
96ce603
Update shading.py
echedey-ls Feb 28, 2024
ff42463
Update shading.py
echedey-ls Feb 28, 2024
1c28e7f
Update shading.py
echedey-ls Feb 28, 2024
cd346e9
Update shading.py
echedey-ls Feb 28, 2024
796c7a9
Update shading.py
echedey-ls Feb 28, 2024
761750c
Update shading.py
echedey-ls Feb 28, 2024
2be29f2
Update shading.py
echedey-ls Feb 28, 2024
8beef55
Lintaaaaaaarrrgh
echedey-ls Feb 28, 2024
5e2be60
Update pvlib/shading.py
echedey-ls Mar 4, 2024
3c5308b
Merge branch 'main' into projected-solar-zenith-angle-issue1734
kandersolar Mar 8, 2024
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 @@ -9,4 +9,5 @@ Shading
shading.ground_angle
shading.masking_angle
shading.masking_angle_passias
shading.sky_diffuse_passias
shading.sky_diffuse_passias
shading.projected_solar_zenith_angle
2 changes: 2 additions & 0 deletions docs/sphinx/source/whatsnew/v0.10.4.rst
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ v0.10.4 (Anticipated March, 2024)

Enhancements
~~~~~~~~~~~~
* Added function :py:func:`pvlib.shading.projected_solar_zenith_angle`,
a common calculation in shading and tracking. (:issue:`1734`, :pull:`1904`)


Bug fixes
Expand Down
81 changes: 81 additions & 0 deletions pvlib/shading.py
Original file line number Diff line number Diff line change
Expand Up @@ -232,3 +232,84 @@ def sky_diffuse_passias(masking_angle):
Available at https://www.nrel.gov/docs/fy18osti/67399.pdf
"""
return 1 - cosd(masking_angle/2)**2


def projected_solar_zenith_angle(axis_tilt, axis_azimuth,
solar_zenith, solar_azimuth):
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
r"""
Calculate projected solar zenith angle in degrees.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
This is the solar zenith angle projected onto the tracker rotation plane or
the plane defined by its normal vector and the azimuth.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved

Computing said value is common in track and shadow algorithms.
See [1]_ [2]_ [3]_.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved

Parameters
----------
axis_tilt : numeric
Copy link
Member

Choose a reason for hiding this comment

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

For a general purpose function, would plane_tilt be more appropriate? And similar for azimuth.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

There's a comment up there from @kandersolar asking for changes from surface_* to axis_* prefixes. I think surface is a close synonym of plane? So I'm unsure of applying that suggestion.

I'll request reviews for the current reviewers hoping they also give some insight regarding this. Anyway, if you are sure of it, I change it without no problems.

Copy link
Member

Choose a reason for hiding this comment

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

Is the issue that axis_tilt doesn't suggest that the equations are equally applicable to fixed tilt arrays?

I'm inclined (pun intended) to keep the term axis_tilt here. I think it's a useful, although perhaps not obvious at first, point of view to consider FT as a subset/special case of SAT anyway, in which case axis_tilt is the relevant quantity even for fixed tilt.

Perhaps a good compromise is to clarify that point in the parameter description/Notes?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I feel like these two terms refer to pure maths, and mapping those words to concise real-world systems is a bit counter-productive due to the variability. Is there some generic -maths- term we can use for a general description of this, a projection? Like basis, maybe? There's always a beautiful open paper backing the procedure and explaining what it does thoroughly. How the tracker works is up to the user or the other functions.

Just to weight in some more options outside of the current frame:
tracker_* (but this suggests everything except FT, right?), collector_*, reference_*, system_*, projection_system_*, coordinate_system_*, reference_system_*, basis_*, projection_basis_*... whatever other constructs you may think of.

Perhaps a good compromise is to clarify that point in the parameter description/Notes?

I don't dislike this possibility if we don't agree on anything else eventually, but we are overcomplicating the docs IMHO.

BTW, I'm +1 for the basis_*, projection_basis_* options. Anyway, remember I have 0 experience in PV outside of this repo.

Copy link
Member

Choose a reason for hiding this comment

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

I admit to suffering from tunnel vision for trackers, but in my view axis_ is still the best suggested prefix thus far. I think no matter what prefix we choose, we're going to have to include some kind of explanation/clarification. So IMHO we might as well choose axis_ for consistency with the reference and other pvlib functions.

How about:

    axis_tilt : numeric
        Tilt from horizontal of the coordinate axis pointing along the tracker
        torque tube (or along the length of a fixed-tilt row). [degrees]

Similar for axis_azimuth. And then a Notes section:

    Notes
    -----
    Although the cited reference derives the equations in the context of
    single-axis tracking arrays, the math is also applicable to fixed-tilt
    arrays.  For example, a south-facing fixed-tilt array on level
    terrain would have ``axis_tilt=0`` and ``axis_azimuth=90``.

Axis tilt angle in degrees. From horizontal plane to array plane.
axis_azimuth : numeric
Axis azimuth angle in degrees.
North = 0°; East = 90°; South = 180°; West = 270°
solar_zenith : numeric
Sun's apparent zenith in degrees.
solar_azimuth : numeric
Sun's azimuth in degrees.

Returns
-------
Projected_solar_zenith : numeric
In degrees.

References
----------
.. [1] K. Anderson and M. Mikofski, 'Slope-Aware Backtracking for
Single-Axis Trackers', National Renewable Energy Lab. (NREL), Golden,
CO (United States); Det Norske Veritas Group, Oslo (Norway),
NREL/TP-5K00-76626, Jul. 2020. :doi:`10.2172/1660126`.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
.. [2] W. F. Marion and A. P. Dobos, 'Rotation Angle for the Optimum
Tracking of One-Axis Trackers', National Renewable Energy Lab. (NREL),
Golden, CO (United States), NREL/TP-6A20-58891, Jul. 2013.
:doi:`10.2172/1089596`.
.. [3] E. Lorenzo, L. Narvarte, and J. Muñoz, 'Tracking and back-tracking',
Progress in Photovoltaics: Research and Applications, vol. 19, no. 6,
pp. 747-753, 2011, :doi:`10.1002/pip.1085`.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
"""
# Assume the tracker reference frame is right-handed. Positive y-axis is
# oriented along tracking axis; from north, the y-axis is rotated clockwise
# by the axis azimuth and tilted from horizontal by the axis tilt. The
# positive x-axis is 90 deg clockwise from the y-axis and parallel to
# horizontal (e.g., if the y-axis is south, the x-axis is west); the
# positive z-axis is normal to the x and y axes, pointed upward.

# Since elevation = 90 - zenith, sin(90-x) = cos(x) & cos(90-x) = sin(x):
# Notation from [1], modified to use zenith instead of elevation
# cos(elevation) = sin(zenith) and sin(elevation) = cos(zenith)
# Avoid recalculating these values
sind_solar_zenith = sind(solar_zenith)
cosd_axis_azimuth = cosd(axis_azimuth)
sind_axis_azimuth = sind(axis_azimuth)
sind_axis_tilt = sind(axis_tilt)

# Sun's x, y, z coords
sx = sind_solar_zenith * sind(solar_azimuth)
sy = sind_solar_zenith * cosd(solar_azimuth)
sz = cosd(solar_zenith)
# Eq. (4); sx', sz' values from sun coordinates projected onto surface
sx_prime = sx * cosd_axis_azimuth - sy * sind_axis_azimuth
sz_prime = (
sx * sind_axis_azimuth * sind_axis_tilt
+ sy * sind_axis_tilt * cosd_axis_azimuth
+ sz * cosd(axis_tilt)
)
# The ideal tracking angle wid is the rotation to place the sun position
# vector (xp, yp, zp) in the (x, z) plane, which is normal to the panel and
# contains the axis of rotation. wid = 0 indicates that the panel is
# horizontal. Here, our convention is that a clockwise rotation is
# positive, to view rotation angles in the same frame of reference as
# azimuth. For example, for a system with tracking axis oriented south, a
# rotation toward the east is negative, and a rotation to the west is
# positive. This is a right-handed rotation around the tracker y-axis.
# Eq. (5); angle between sun's beam and surface
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
theta_T = np.degrees(np.arctan2(sx_prime, sz_prime))
return theta_T
157 changes: 156 additions & 1 deletion pvlib/tests/test_shading.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,11 @@
import pandas as pd

from pandas.testing import assert_series_equal
from numpy.testing import assert_allclose
import pytest
from datetime import timezone, timedelta

import pvlib
from pvlib import shading


Expand Down Expand Up @@ -37,7 +40,7 @@ def test__ground_angle_zero_gcr():

@pytest.fixture
def surface_tilt():
idx = pd.date_range('2019-01-01', freq='h', periods=3)
idx = pd.date_range("2019-01-01", freq="h", periods=3)
return pd.Series([0, 20, 90], index=idx)


Expand Down Expand Up @@ -104,3 +107,155 @@ def test_sky_diffuse_passias_scalar(average_masking_angle, shading_loss):
for angle, loss in zip(average_masking_angle, shading_loss):
actual_loss = shading.sky_diffuse_passias(angle)
assert np.isclose(loss, actual_loss)


@pytest.fixture
def true_tracking_angle_and_inputs_NREL():
kandersolar marked this conversation as resolved.
Show resolved Hide resolved
# data from NREL 'Slope-Aware Backtracking for Single-Axis Trackers'
# doi.org/10.2172/1660126 ; Accessed on 2023-11-06.
tzinfo = timezone(timedelta(hours=-5))
axis_tilt_angle = 9.666 # deg
axis_azimuth_angle = 195.0 # deg
timedata = pd.DataFrame(
columns=("Apparent Elevation", "Solar Azimuth", "True-Tracking"),
data=(
(2.404287, 122.791770, -84.440),
(11.263058, 133.288729, -72.604),
(18.733558, 145.285552, -59.861),
(24.109076, 158.939435, -45.578),
(26.810735, 173.931802, -28.764),
(26.482495, 189.371536, -8.475),
(23.170447, 204.136810, 15.120),
(17.296785, 217.446538, 39.562),
(9.461862, 229.102218, 61.587),
(0.524817, 239.330401, 79.530),
),
)
timedata.index = pd.date_range(
"2019-01-01T08", "2019-01-01T17", freq="1H", tz=tzinfo
)
timedata["Apparent Zenith"] = 90.0 - timedata["Apparent Elevation"]
return (axis_tilt_angle, axis_azimuth_angle, timedata)


@pytest.fixture
def singleaxis_psz_implementation_port_data():
# data generated with the PSZ angle implementation in tracking.singleaxis
# See GitHub issue #1734 & PR #1904
axis_tilt_angle = 12.224
axis_azimuth_angle = 187.2

singleaxis_result = pd.DataFrame(
columns=[
"Apparent Zenith",
"Solar Azimuth",
"tracker_theta",
"surface_azimuth",
"surface_tilt",
],
data=[
[88.86131915, 116.14911543, -84.67346, 98.330924, 84.794565],
[85.67558254, 119.46577753, -80.544188, 99.219659, 80.760477],
[82.4784391, 122.90558458, -76.226064, 100.171259, 76.5443],
[79.37555806, 126.48822166, -71.79054, 101.184411, 72.217365],
[76.40491865, 130.23239671, -67.237442, 102.276947, 67.781439],
[73.59273783, 134.15525777, -62.55178, 103.476096, 63.224495],
[70.96318968, 138.2715258, -57.713941, 104.819827, 58.53107],
[68.54068323, 142.59233032, -52.702658, 106.361922, 53.685798],
[66.35031258, 147.12377575, -47.496592, 108.18131, 48.676053],
[64.41759166, 151.8653323, -42.07579, 110.39903, 43.495367],
[62.76775062, 156.80824414, -36.423404, 113.210504, 38.148938],
[61.42469841, 161.9342438, -30.527799, 116.950922, 32.663696],
[60.40974474, 167.21493901, -24.385012, 122.236817, 27.108957],
[59.74022062, 172.61222482, -18.001341, 130.288224, 21.645102],
[59.42818646, 178.07994717, -11.395651, 143.610698, 16.652493],
[59.47944177, 183.56677914, -4.600779, 166.390187, 13.048796],
[59.89302187, 189.01995634, 2.336615, 198.108, 12.441979],
[60.66128258, 194.38926277, 9.358232, 225.094855, 15.351466],
[61.77055542, 199.63057627, 16.398369, 241.465486, 20.352345],
[63.20224386, 204.70842576, 23.389598, 251.116742, 26.231294],
[64.93416116, 209.59729217, 30.268795, 257.259578, 32.425598],
[66.94189859, 214.28170196, 36.982274, 261.49605, 38.674352],
[69.20004673, 218.75538494, 43.489104, 264.617474, 44.841832],
[71.68314725, 223.01986867, 49.762279, 267.042188, 50.852813],
[74.36628597, 227.08285659, 55.787916, 269.007999, 56.666604],
[77.22520074, 230.95665462, 61.562937, 270.658956, 62.264111],
[80.23550305, 234.65680797, 67.091395, 272.086933, 67.639267],
[83.3693091, 238.20102038, 72.378024, 273.352342, 72.790188],
[86.57992299, 241.60837123, 77.408775, 274.492262, 77.698775],
[89.70940444, 244.89880789, 82.045935, 275.505443, 82.227402],
],
)
singleaxis_result.index = pd.date_range(
"2024-01-25 08:40",
"2024-01-25 18:20",
freq="20min",
tz=timezone(timedelta(hours=1)),
)
return (axis_tilt_angle, axis_azimuth_angle, singleaxis_result)


def test_projected_solar_zenith_angle_numeric(
true_tracking_angle_and_inputs_NREL,
singleaxis_psz_implementation_port_data
):
psz_func = shading.projected_solar_zenith_angle
axis_tilt, axis_azimuth, timedata = true_tracking_angle_and_inputs_NREL
# test against data provided by NREL
psz = psz_func(
axis_tilt,
axis_azimuth,
timedata["Apparent Zenith"],
timedata["Solar Azimuth"],
)
assert_allclose(psz, timedata["True-Tracking"], atol=1e-3)
# test by changing axis azimuth and tilt
psz = psz_func(
-axis_tilt,
axis_azimuth - 180,
timedata["Apparent Zenith"],
timedata["Solar Azimuth"],
)
assert_allclose(psz, -timedata["True-Tracking"], atol=1e-3)

# test implementation port from tracking.singleaxis
axis_tilt, axis_azimuth, singleaxis = \
singleaxis_psz_implementation_port_data
psz = pvlib.tracking.singleaxis(
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved
singleaxis["Apparent Zenith"],
singleaxis["Solar Azimuth"],
axis_tilt,
axis_azimuth,
backtrack=False,
)
assert_allclose(
psz["tracker_theta"],
singleaxis["tracker_theta"],
atol=1e-6
)


@pytest.mark.parametrize(
"cast_type, cast_func",
[
(float, lambda x: float(x)),
(np.ndarray, lambda x: np.array([x])),
(pd.Series, lambda x: pd.Series(data=[x])),
],
)
def test_projected_solar_zenith_angle_datatypes(
cast_type, cast_func, true_tracking_angle_and_inputs_NREL
):
psz_func = shading.projected_solar_zenith_angle
axis_tilt, axis_azimuth, timedata = true_tracking_angle_and_inputs_NREL
sun_apparent_zenith = timedata["Apparent Zenith"].iloc[0]
sun_azimuth = timedata["Solar Azimuth"].iloc[0]

axis_tilt, axis_azimuth, sun_apparent_zenith, sun_azimuth = (
cast_func(axis_tilt),
cast_func(axis_azimuth),
cast_func(sun_apparent_zenith),
cast_func(sun_azimuth),
)
psz = psz_func(axis_tilt, axis_azimuth, sun_apparent_zenith, axis_azimuth)
assert isinstance(psz, cast_type)
53 changes: 8 additions & 45 deletions pvlib/tracking.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@

from pvlib.tools import cosd, sind, tand, acosd, asind
from pvlib import irradiance
from pvlib import shading


def singleaxis(apparent_zenith, apparent_azimuth,
Expand Down Expand Up @@ -126,51 +127,13 @@ def singleaxis(apparent_zenith, apparent_azimuth,
if apparent_azimuth.ndim > 1 or apparent_zenith.ndim > 1:
raise ValueError('Input dimensions must not exceed 1')

# Calculate sun position x, y, z using coordinate system as in [1], Eq 1.

# NOTE: solar elevation = 90 - solar zenith, then use trig identities:
# sin(90-x) = cos(x) & cos(90-x) = sin(x)
sin_zenith = sind(apparent_zenith)
x = sin_zenith * sind(apparent_azimuth)
y = sin_zenith * cosd(apparent_azimuth)
z = cosd(apparent_zenith)

# Assume the tracker reference frame is right-handed. Positive y-axis is
# oriented along tracking axis; from north, the y-axis is rotated clockwise
# by the axis azimuth and tilted from horizontal by the axis tilt. The
# positive x-axis is 90 deg clockwise from the y-axis and parallel to
# horizontal (e.g., if the y-axis is south, the x-axis is west); the
# positive z-axis is normal to the x and y axes, pointed upward.

# Calculate sun position (xp, yp, zp) in tracker coordinate system using
# [1] Eq 4.

cos_axis_azimuth = cosd(axis_azimuth)
sin_axis_azimuth = sind(axis_azimuth)
cos_axis_tilt = cosd(axis_tilt)
sin_axis_tilt = sind(axis_tilt)
xp = x*cos_axis_azimuth - y*sin_axis_azimuth
# not necessary to calculate y'
# yp = (x*cos_axis_tilt*sin_axis_azimuth
# + y*cos_axis_tilt*cos_axis_azimuth
# - z*sin_axis_tilt)
zp = (x*sin_axis_tilt*sin_axis_azimuth
+ y*sin_axis_tilt*cos_axis_azimuth
+ z*cos_axis_tilt)

# The ideal tracking angle wid is the rotation to place the sun position
# vector (xp, yp, zp) in the (y, z) plane, which is normal to the panel and
# contains the axis of rotation. wid = 0 indicates that the panel is
# horizontal. Here, our convention is that a clockwise rotation is
# positive, to view rotation angles in the same frame of reference as
# azimuth. For example, for a system with tracking axis oriented south, a
# rotation toward the east is negative, and a rotation to the west is
# positive. This is a right-handed rotation around the tracker y-axis.
echedey-ls marked this conversation as resolved.
Show resolved Hide resolved

# Calculate angle from x-y plane to projection of sun vector onto x-z plane
# using [1] Eq. 5.

wid = np.degrees(np.arctan2(xp, zp))
# ideal tracking angle, does not account for row-to-row shading
wid = shading.projected_solar_zenith_angle(
axis_tilt=axis_tilt,
axis_azimuth=axis_azimuth,
solar_zenith=apparent_zenith,
solar_azimuth=apparent_azimuth,
)

# filter for sun above panel horizon
zen_gt_90 = apparent_zenith > 90
Expand Down
Loading