NAME
Astro::Sunrise - Perl extension for computing the sunrise/sunset on a given day
VERSION
This documentation refers to Astro::Sunrise
version 0.99.
SYNOPSIS
# When did the sun rise on YAPC::Europe 2015?
use Astro::Sunrise;
my ($sunrise, $sunset) = sunrise( { year => 2015, month => 9, day => 2, # YAPC::EU starts on 2nd September 2015
lon => -3.6, lat => 37.17, # Granada is 37°10'N, 3°36'W
tz => 1, isdst => 1 } ); # This is still summer, therefore DST
# When does the sun rise today in Salt Lake City (home to YAPC::NA 2015)?
use Astro::Sunrise;
use DateTime;
$sunrise_today = sun_rise( { lon => -111.88, lat => 40.75 } ); # 40°45'N, 111°53'W
# And when does it set tomorrow at Salt Lake City?
use Astro::Sunrise;
use DateTime;
$sunset_tomorrow = sun_set( { lat => 40.75, # 40°45'N,
lon => -111.88, # 111°53'W
alt => -0.833, # standard value for the sun altitude at sunset
offset => 1 } ); # day offset up to tomorrow
DESCRIPTION
This module will return the sunrise and sunset for a given day.
Months are numbered 1 to 12, in the usual way, not 0 to 11 as in C and in Perl's localtime.
Eastern longitude is entered as a positive number
Western longitude is entered as a negative number
Northern latitude is entered as a positive number
Southern latitude is entered as a negative number
Please note that, when given as positional parameters, the longitude is specified before the latitude.
The time zone is given as the numeric value of the offset from UTC.
The precise
parameter is set to either 0 or 1. If set to 0 no Iteration will occur. If set to 1 Iteration will occur, which will give a more precise result. Default is 0.
There are a number of sun altitudes to chose from. The default is -0.833 because this is what most countries use. Feel free to specify it if you need to. Here is the list of values to specify altitude (ALT) with, including symbolic constants for each.
- 0 degrees
-
Center of Sun's disk touches a mathematical horizon
- -0.25 degrees
-
Sun's upper limb touches a mathematical horizon
- -0.583 degrees
-
Center of Sun's disk touches the horizon; atmospheric refraction accounted for
- -0.833 degrees, DEFAULT
-
Sun's upper limb touches the horizon; atmospheric refraction accounted for
- -6 degrees, CIVIL
-
Civil twilight (one can no longer read outside without artificial illumination)
- -12 degrees, NAUTICAL
-
Nautical twilight (navigation using a sea horizon no longer possible)
- -15 degrees, AMATEUR
-
Amateur astronomical twilight (the sky is dark enough for most astronomical observations)
- -18 degrees, ASTRONOMICAL
-
Astronomical twilight (the sky is completely dark)
USAGE
sunrise
($sunrise, $sunset) = sunrise( { year => $year, month => $month,
day => $day,
lon => $longitude, lat => $latitude,
tz => $tz_offset, isdst => $is_dst,
alt => $altitude, upper_limb => $upper_limb,
precise => $precise, polar => $action,
trace => $fh } );
($sunrise, $sunset) = sunrise(YYYY,MM,DD,longitude,latitude,Time Zone,DST);
($sunrise, $sunset) = sunrise(YYYY,MM,DD,longitude,latitude,Time Zone,DST,ALT);
($sunrise, $sunset) = sunrise(YYYY,MM,DD,longitude,latitude,Time Zone,DST,ALT,precise);
Returns the sunrise and sunset times, in HH:MM format.
The first form uses a hash reference to pass arguments by name. The other forms are kept for backward compatibility. The arguments are:
- year, month, day
-
The three elements of the date for which you want to compute the sunrise and sunset. Months are numbered 1 to 12, in the usual way, not 0 to 11 as in C and in Perl's localtime.
Mandatory, can be positional (#1, #2 and #3).
- lon, lat
-
The longitude and latitude of the place for which you want to compute the sunrise and sunset. They are given in decimal degrees. For example:
lon => -3.6, # 3° 36' W lat => 37.17, # 37° 10' N Eastern longitude is entered as a positive number Western longitude is entered as a negative number Northern latitude is entered as a positive number Southern latitude is entered as a negative number
Mandatory, can be positional (#4 and #5).
- tz
-
Time Zone is the offset from GMT
Mandatory, can be positional (#6).
- isdst
-
1 if daylight saving time is in effect, 0 if not.
Mandatory, can be positional (#7).
- alt
-
Altitude of the sun, in decimal degrees. Usually a negative number, because the sun should be under the mathematical horizon. But if there is a high mountain range sunward (that is, southward if you live in the Northern hemisphere), you may need to enter a positive altitude.
This parameter is optional. Its default value is -0.833. It can be positional (#8).
- upper_limb
-
If this parameter set to a true value (usually 1), the algorithm computes the sun apparent radius and takes it into account when computing the sun altitude. This parameter is useful only when the
alt
parameter is set to0
or-0.583
degrees. When using-0.25
or-0.833
degrees, the sun radius is already taken into account. When computing twilights (-6
to-18
), the sun radius is irrelevant.Since the default value for the
alt
parameter is -0.833, the default value forupper_limb
is 0.This parameter is optional and it can be specified only by keyword.
- polar
-
When dealing with a polar location, there may be dates where there is a polar night (sun never rises) or a polar day. The default behaviour of the module is to emit a warning in these cases ("Sun never rises!!" or "Sun never sets!!"). But some programmers may find this inconvenient. An alternate behaviour is to return special values reflecting the situation.
So, if the
polar
parameter is set to'warn'
, the module emits a warning. If thepolar
parameter is set to'retval'
, the module emits no warning, but it returns either'day'
or'night'
.Example:
# Loosely based on Alex Gough's activities: scientist and Perl programmer, who spent a year # in Halley Base in 2006. Let us suppose he arrived there on 15th January 2006. my ($sunrise, $sunset) = sunrise( { year => 2006, month => 1, day => 15, lon => -26.65, lat => -75.58, # Halley Base: 75°35'S 26°39'W tz => 3, polar => 'retval' } ); if ($sunrise eq 'day') { say "Alex Gough saw the midnight sun the first day he arrived at Halley Base"; } elsif ($sunrise eq 'night') { say "It would be days, maybe weeks, before the sun would rise."; } else { say "Alex saw his first antarctic sunset at $sunset"; }
This parameter is optional and it can be specified only by keyword.
- trace
-
This parameter should either be a false value or a filehandle opened for output. In the latter case, a few messages are printed to the filehandle, which allows the programmer to see step by step how the sunrise and the sunset are computed.
Used for analysis and debugging purposes. You need to read the text doc/astronomical-notes.pod to understand what the traced values represent.
This parameter is optional and it can be specified only by keyword.
- precise
-
Choice between a precise algorithm and a simpler algorithm. The default value is 0, that is, the simpler algorithm. Any true value switches to the precise algorithm.
The original method only gives an approximate value of the Sun's rise/set times. The error rarely exceeds one or two minutes, but at high latitudes, when the Midnight Sun soon will start or just has ended, the errors may be much larger. If you want higher accuracy, you must then use the precise algorithm. This feature is new as of version 0.7. Here is what I have tried to accomplish with this.
a) Compute sunrise or sunset as always, with one exception: to convert LHA from degrees to hours, divide by 15.04107 instead of 15.0 (this accounts for the difference between the solar day and the sidereal day).
b) Re-do the computation but compute the Sun's RA and Decl, and also GMST0, for the moment of sunrise or sunset last computed.
c) Iterate b) until the computed sunrise or sunset no longer changes significantly. Usually 2 iterations are enough, in rare cases 3 or 4 iterations may be needed.
This parameter is optional. It can be positional (#9).
For Example
($sunrise, $sunset) = sunrise( 2001, 3, 10, 17.384, 98.625, -5, 0 );
($sunrise, $sunset) = sunrise( 2002, 10, 14, -105.181, 41.324, -7, 1, -18);
($sunrise, $sunset) = sunrise( 2002, 10, 14, -105.181, 41.324, -7, 1, -18, 1);
sun_rise, sun_set
$sun_rise = sun_rise( { lon => $longitude, lat => $latitude,
alt => $altitude, upper_limb => $bool,
offset => $day_offset,
precise => $bool_precise, polar => $action } );
$sun_set = sun_set ( { lon => $longitude, lat => $latitude,
alt => $altitude, upper_limb => $bool,
offset => $day_offset,
precise => $bool_precise, polar => $action } );
$sun_rise = sun_rise( $longitude, $latitude );
$sun_rise = sun_rise( $longitude, $latitude, $alt );
$sun_rise = sun_rise( $longitude, $latitude, $alt, $day_offset );
Returns the sun rise time (resp. the sun set time) for the given location and for today's date (as given by DateTime), plus or minus some offset in days. The first form use all parameters and transmit them by name. The second form uses today's date (from DateTime) and the default altitude. The third form adds specifying a custom altitude. The fourth form allows for specifying an integer day offset from today, either positive or negative.
The parameters are the same as the parameters for sunrise
. There is an additional parameter, offset
, which allows using a date other than today: +1
for to-morrow, -7
for one week ago, etc.
The arguments are:
- lon, lat
-
The longitude and latitude of the place for which you want to compute the sunrise and sunset. They are given in decimal degrees. For example:
lon => -3.6, # 3° 36' W lat => 37.17, # 37° 10' N Eastern longitude is entered as a positive number Western longitude is entered as a negative number Northern latitude is entered as a positive number Southern latitude is entered as a negative number
Mandatory, can be positional (#1 and #2).
- alt
-
Altitude of the sun, in decimal degrees. Usually a negative number, because the sun should be under the mathematical horizon. But if there is a high mountain range sunward (that is, southward if you live in the Northern hemisphere), you may need to enter a positive altitude.
This parameter is optional. Its default value is -0.833. It can be positional (#3).
- offset
-
By default,
sun_rise
andsun_set
use the current day. If you need another day, you give an offset relative to the current day. For example,+7
means next week, while-365
means last year.This parameter has nothing to do with timezones.
Optional, 0 by default, can be positional (#4).
- time_zone
-
Time Zone is the Olson name for a timezone. By default, the functions
sun_rise
andsun_set
will try to use thelocal
timezone.This parameter is optional and it can be specified only by keyword.
- upper_limb
-
If this parameter set to a true value (usually 1), the algorithm computes the sun apparent radius and takes it into account when computing the sun altitude. This parameter is useful only when the
alt
parameter is set to0
or-0.583
degrees. When using-0.25
or-0.833
degrees, the sun radius is already taken into account. When computing twilights (-6
to-18
), the sun radius is irrelevant.Since the default value for the
alt
parameter is -0.833, the default value forupper_limb
is 0.This parameter is optional and it can be specified only by keyword.
- polar
-
When dealing with a polar location, there may be dates where there is a polar night (sun never rises) or a polar day. The default behaviour of the module is to emit a warning in these cases ("Sun never rises!!" or "Sun never sets!!"). But some programmers may find this inconvenient. An alternate behaviour is to return special values reflecting the situation.
So, if the
polar
parameter is set to'warn'
, the module emits a warning. If thepolar
parameter is set to'retval'
, the module emits no warning, but it returns either'day'
or'night'
.This parameter is optional and it can be specified only by keyword.
- trace
-
This parameter should either be a false value or a filehandle opened for output. In the latter case, a few messages are printed to the filehandle, which allows the programmer to see step by step how the sunrise and the sunset are computed.
Used for analysis and debugging purposes.
This parameter is optional and it can be specified only by keyword.
- precise
-
Choice between a precise algorithm and a simpler algorithm. The default value is 0, that is, the simpler algorithm. Any true value switches to the precise algorithm.
For more documentation, see the corresponding parameter for the
sunrise
function.This parameter is optional and it can be specified only by keyword.
For Example
$sunrise = sun_rise( -105.181, 41.324 );
$sunrise = sun_rise( -105.181, 41.324, -15 );
$sunrise = sun_rise( -105.181, 41.324, -12, +3 );
$sunrise = sun_rise( -105.181, 41.324, undef, -12);
Trigonometric functions using degrees
Since the module use trigonometry with degrees, the corresponding functions are available to the module user, free of charge. Just mention the tag :trig
in the use
statement. These functions are:
- sind, cosd, tand
-
The direct functions, that is, sine, cosine and tangent functions, respectively. Each one receives one parameter, in degrees, and returns the trigonometric value.
- asind, acosd, atand
-
The reverse functions, that is, arc-sine, arc-cosine, and arc-tangent. Each one receives one parameter, the trigonometric value, and returns the corresponding angle in degrees.
- atan2d
-
Arc-tangent. This function receives two parameters: the numerator and the denominator of a fraction equal to the tangent. Use this function instead of
atand
when you are not sure the denominator is not zero. E.g.:use Astro::Sunrise qw(:trig); say atan2d(1, 2) # prints 26,5 say atan2d(1, 0) # prints 90, without triggering a "division by zero" error
- equal
-
Not really a trigonometrical function, but still useful at some times. This function receives two floating values and an integer value. It compares the floating numbers, and returns "1" if their most significant digits are equal. The integer value specifies how many digits are kept. E.g.
say equal(22/7, 355/113, 3) # prints 1, because : 22/7 = 3.14285715286 rounded to 3.14 # 355/113 = 3.14159292035 rounded to 3.14 say equal(22/7, 355/113, 4) # prints 0, because : 22/7 = 3.14285715286 rounded to 3.143 # 355/113 = 3.14159292035 rounded to 3.142
EXPORTS
By default, the functions sunrise
, sun_rise
and sun_set
are exported.
The constants DEFAULT
, CIVIL
, NAUTICAL
, AMATEUR
and ASTRONOMICAL
are exported on request with the tag :constants
.
The functions sind
, cosd
, tand
, asind
, acosd
, atand
, atan2d
and equal
exported on request with the tag :trig
.
DEPENDENCIES
This module requires only core modules: POSIX, Math::Trig and Carp.
If you use the sun_rise
and sun_set
functions, you will need also DateTime.
BUGS AND ISSUES
Before reporting a bug, please read the text doc/astronomical-notes.pod because the strange behavior you observed may be a correct one, or it may be a corner case already known and already mentioned in the text.
Nevertheless, patches and (justified) bug reports are welcome.
See https://github.com/jforget/Astro-Sunrise/issues and https://github.com/jforget/Astro-Sunrise/pulls.
Precise Algorithm
The explanations for the precise algorithm give a pretty good reason for using an angular speed of 15.04107 instead of 15. Yet, computations with 15.04107 do not give results conforming to what the NOAA website and Stellarium give, while computations with 15 give conforming results. The implementation of the precise algorithm should be analysed and checked to find the reason why 15.04107 does not give the proper results.
Kwalitee
The CPANTS tools do not recognize the LICENSE POD paragraph. But any human reader will admit that this LICENSE paragraph exists and is valid.
Haiku-OS CPAN Tester
The built-in test t/06datetime.t fails on Haiku-OS because there is no way to extract the timezone name from the system parameters. This failure does not affect the core functions of Astro::Sunrise.
Also reported from a user working on a partially configured FreeBSD machine, see https://github.com/jforget/Astro-Sunrise/issues/16.
Hopefully, this will be fixed in the current version.
AUTHOR
Ron Hill rkhill@firstlight.net
Co-maintainer: Jean Forget (JFORGET at cpan dot org)
SPECIAL THANKS
Robert Creager [Astro-Sunrise@LogicalChaos.org] for providing help with converting Paul's C code to Perl, for providing code for sun_rise, sun_set subs. Also adding options for different altitudes.
Joshua Hoblitt [jhoblitt@ifa.hawaii.edu] for providing the patch to convert to DateTime.
Chris Phillips for providing patch for conversion to local time.
Brian D Foy for providing patch for constants :)
Gabor Szabo for pointing POD mistakes.
People at https://geocoder.opencagedata.com/ for noticing an endless loop condition and for fixing it.
CREDITS
- Paul Schlyter, Stockholm, Sweden
-
for his excellent web page on the subject.
- Rich Bowen (rbowen@rbowen.com)
-
for suggestions.
- Adrian Blockley [adrian.blockley@environ.wa.gov.au]
-
for finding a bug in the conversion to local time.
- Slaven Rezić
-
for finding and fixing a bug with DST.
Lightly verified against http://aa.usno.navy.mil/data/docs/RS_OneYear.html
In addition, checked to be compatible with a C implementation of Paul Schlyter's algorithm.
COPYRIGHT and LICENSE
Perl Module
Copyright (C) 1999-2003, 2013, 2015, 2017, 2019, 2021 Ron Hill and Jean Forget, all rights reserved. This program is distributed under the same terms as Perl 5.16.3: GNU Public License version 1 or later and Perl Artistic License
You can find the text of the licenses in the LICENSE file or at https://dev.perl.org/licenses/artistic.html and https://www.gnu.org/licenses/gpl-1.0.html.
Here is the summary of GPL:
This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 1, or (at your option) any later version.
This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., https://www.fsf.org/.
Original C program
Here is the copyright information provided by Paul Schlyter:
Written as DAYLEN.C, 1989-08-16
Modified to SUNRISET.C, 1992-12-01
(c) Paul Schlyter, 1989, 1992
Released to the public domain by Paul Schlyter, December 1992
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
SEE ALSO
perl(1).
DateTime::Event::Jewish::Sunrise
The text doc/astronomical-notes.pod (or its original French version doc/notes-astronomiques) in this distribution.