——————————————————package
Astro::FITS::HdrTrans::Base;
=head1 NAME
Astro::FITS::HdrTrans::Base - Base class for header translation
=head1 SYNOPSIS
use base qw/ Astro::FITS::HdrTrans::Base /;
%generic = Astro::FITS::HdrTrans::Base->translate_from_FITS( \%fits );
%fits = Astro::FITS::HdrTrans::Base->translate_to_FITS( \%gen );
=head1 DESCRIPTION
This is the header translation base class. Not to be confused with
C<Astro::FITS::HdrTrans> itself, which is a high level abstraction
class. In general users should use C<Astro::FITS::HdrTrans>
for initiating header translations unless they know what they are
doing. Also C<Astro::FITS::HdrTrans> is the only public interface
to the header translation.
=cut
use
5.006;
use
strict;
use
warnings;
use
Carp;
$VERSION
=
"1.50"
;
=head1 PUBLIC METHODS
All methods in this class are CLASS METHODS. No state is retained
outside of the hash argument.
=over 4
=item B<translate_from_FITS>
Do the header translation from FITS for the specified class.
%generic = $class->translate_to_FITS( \%fitshdr,
prefix => $prefix,
frameset => $wcs,
);
Prefix is attached to the keys in the returned hash if it
is defined. The frameset is an optional Starlink::AST object.
If a translation results in an undefined value (for example, if the
headers can represent both imaging and spectroscopy there may be no
requirement for a DISPERSION header), the result is not stored in the
translated hash.
A list of failed translations is available in the _UNDEFINED_TRANSLATIONS
key in the generic hash. This points to a reference to an array of all
the failed generic translations.
The class used for the translation is stored in the key _TRANSLATION_CLASS.
This can then be used to reverse the translation without having to
re-scan the headers.
=cut
sub
translate_from_FITS {
my
$class
=
shift
;
my
$FITS
=
shift
;
my
%opts
=
@_
;
my
$prefix
=
''
;
if
(
exists
(
$opts
{prefix} ) &&
defined
(
$opts
{prefix} ) ) {
$prefix
=
$opts
{prefix};
}
my
$frameset
;
if
(
exists
(
$opts
{frameset} ) &&
defined
(
$opts
{frameset} ) ) {
$frameset
=
$opts
{frameset};
}
croak
"translate_from_FITS: Not a hash reference!"
unless
(
ref
(
$FITS
) &&
ref
(
$FITS
) eq
'HASH'
);
# Now we need to loop over the known generic headers
# which we obtain from Astro::FITS::HdrTrans
my
@GEN
= Astro::FITS::HdrTrans->generic_headers;
my
%generic
;
my
@failed
;
for
my
$g
(
@GEN
) {
my
$method
=
"to_$g"
;
if
(
$class
->can(
$method
)) {
my
$result
=
$class
->
$method
(
$FITS
,
$frameset
);
if
(
defined
$result
) {
$generic
{
"$prefix$g"
} =
$result
;
}
else
{
push
(
@failed
,
$g
);
}
}
}
# store the failed translations (if we had any)
$generic
{_UNDEFINED_TRANSLATIONS} = \
@failed
if
@failed
;
# store the translation class
$generic
{_TRANSLATION_CLASS} =
$class
;
return
%generic
;
}
=item B<translate_to_FITS>
Do the header translation from generic headers to FITS
for the specified class.
%fits = $class->translate_to_FITS( \%generic );
=cut
sub
translate_to_FITS {
my
$class
=
shift
;
my
$generic
=
shift
;
croak
"translate_to_FITS: Not a hash reference!"
unless
(
ref
(
$generic
) &&
ref
(
$generic
) eq
'HASH'
);
# Now we need to loop over the known generic headers
# which we obtain from Astro::FITS::HdrTrans
my
@GEN
= Astro::FITS::HdrTrans->generic_headers;
my
%FITS
;
for
my
$g
(
@GEN
) {
my
$method
=
"from_$g"
;
if
(
$class
->can(
$method
)) {
%FITS
= (
%FITS
,
$class
->
$method
(
$generic
));
}
}
return
%FITS
;
}
=back
=head1 PROTECTED METHODS
These methods are available to translation subclasses and should
not be used by external classes.
=over 4
=item B<can_translate>
Returns true if the supplied headers can be handled by this class.
$cando = $class->can_translate( \%hdrs );
The base class version of this method returns true if either the C<INSTRUME>
or C<INSTRUMENT> key exist and match the value returned by the
C<this_instrument> method. Comparisons are case-insensitive and can use
regular expressions on instrument name if provided by the base class.
=cut
sub
can_translate {
my
$class
=
shift
;
my
$headers
=
shift
;
# get the reference instrument string
my
$ref
=
$class
->this_instrument();
return
0
unless
defined
$ref
;
# For consistency in subsequent algorithm convert
# a string to a pattern match object
if
(not
ref
(
$ref
)) {
$ref
=
quotemeta
(
$ref
);
$ref
=
qr/^$ref$/
i;
}
# check against the FITS and Generic versions.
my
$inst
;
for
my
$k
(
qw/ INSTRUME INSTRUMENT /
) {
if
(
exists
$headers
->{
$k
} &&
defined
$headers
->{
$k
}) {
$inst
=
$headers
->{
$k
};
last
;
}
}
# no recognizable instrument
return
0
unless
defined
$inst
;
# Now do the test
return
(
$inst
=~
$ref
);
}
=item B<this_instrument>
Name of the instrument that can be translated by this class.
Defaults to an empty string. The method must be subclassed.
$inst = $class->this_instrument();
Can return a regular expresion object (C<qr>).
=cut
sub
this_instrument {
return
""
;
}
=item B<valid_class>
Historically this method was used to determine whether this class can
handle the supplied FITS headers. The headers can be either in
generic form or in FITS form.
$isvalid = $class->valid_class( \%fits );
The base class always returns false. This is a backwards compatibility
method to prevent mixing of translation modules from earlier release
of C<Astro::FITS::HdrTrans> with the current object-oriented version.
See the C<can_translate> method for the new interface.
=cut
sub
valid_class {
return
0;
}
=item B<_generate_lookup_methods>
We generate the unit and constant mapping methods automatically from a
lookup table.
Astro::FITS::HdrTrans::UKIRT->_generate_lookup_methods( \%const, \%unit);
This method generates all the simple internal methods. Expects two arguments,
both references to hashes. The first is a reference to a hash with
constant mapping from FITS to generic (and no reverse mapping), the
second is a reference to a hash with unit mappings (both from and to
methods are created). The methods are placed into the package given
by the class supplied to the method.
Astro::FITS::HdrTrans::UKIRT->_generate_lookup_methods( \%const, \%unit, \%null);
Additionally, an optional third argument can be used to indicate
methods that should be null translations. This is a reference to an array
of generic keywords and should be used in the rare cases when a base
class implementation should be nullified. This will result in undefined
values in the generic hash but no value in the generic to FITS mapping.
A fourth optional argument can specify those unit mappings that should
use the final entry in a subheader (if a subheader is present). Mainly
associated with END events such as AIRMASS_END or ELEVATION_END.
Astro::FITS::HdrTrans::UKIRT->_generate_lookup_methods( \%const, \%unit,
\%null, \%endobs);
These methods will have the standard interface of
$generic = $class->_to_GENERIC_NAME( \%fits );
%fits = $class->_from_GENERIC_NAME( \%generic );
Generic unit map translations use the via_subheader() method in scalar
context and so will retrieve the first sub header value if the keyword
is not present in the primary header.
=cut
sub
_generate_lookup_methods {
my
$class
=
shift
;
my
$const
=
shift
;
my
$unit
=
shift
;
my
$null
=
shift
;
my
$endobs
=
shift
;
# Have to go into a different package
my
$p
=
"{\n package $class;\n"
;
my
$ep
=
"\n}"
;
# close the scope
# Loop over the keys to the unit mapping hash
# The keys are the GENERIC name
for
my
$key
(
keys
%$unit
) {
# Get the original FITS header name
my
$fhdr
=
$unit
->{
$key
};
# print "Processing $key and $ohdr and $fhdr\n";
# First generate the code to generate Generic headers
my
$subname
=
"to_$key"
;
my
$sub
=
qq/ $p sub $subname { scalar \$_[0]->via_subheader_undef_check(\$_[1],\"$fhdr\"); } $ep /
;
eval
"$sub"
;
#print "Sub: $sub\n";
# Now the from
$subname
=
"from_$key"
;
$sub
=
qq/ $p sub $subname { (\"$fhdr\", \$_[1]->{\"$key\"}); } $ep/
;
eval
"$sub"
;
#print "Sub: $sub\n";
}
# and the CONSTANT mappings (only to_GENERIC_NAME)
for
my
$key
(
keys
%$const
) {
my
$subname
=
"to_$key"
;
my
$val
=
$const
->{
$key
};
# A method so no gain in using a null prototype
my
$sub
=
qq/ $p sub $subname { \"$val\" } $ep /
;
eval
"$sub"
;
}
# the null mappings
if
(
defined
$null
) {
for
my
$key
(
@$null
) {
# to generic
my
$subname
=
"to_$key"
;
my
$sub
=
qq/ $p sub $subname { } $ep /
;
eval
"$sub"
;
# to generic
$subname
=
"from_$key"
;
$sub
=
qq/ $p sub $subname { return (); } $ep /
;
eval
"$sub"
;
}
}
# the mappings that are unit mappings but from the end of a subheader
# group (eg ELEVATION_END)
if
(
defined
$endobs
) {
for
my
$key
(
keys
%$endobs
) {
# Get the original FITS header name
my
$fhdr
=
$endobs
->{
$key
};
# print "Processing $key and $ohdr and $fhdr\n";
# First generate the code to generate Generic headers
my
$subname
=
"to_$key"
;
my
$sub
=
qq/ $p sub $subname {
my \@allresults = \$_[0]->via_subheader_undef_check(\$_[1],\"$fhdr\");
return \$allresults[-1];
} $ep /
;
eval
"$sub"
;
#print "Sub: $sub\n";
# Now the from
$subname
=
"from_$key"
;
$sub
=
qq/ $p sub $subname { (\"$fhdr\", \$_[1]->{\"$key\"}); } $ep/
;
eval
"$sub"
;
#print "Sub: $sub\n";
}
}
}
=item B<nint>
Return the nearest integer to a supplied floating point
value. 0.5 is rounded up.
$int = Astro::FITS::HdrTrans->nint( $value );
=cut
sub
nint {
my
$class
=
shift
;
my
$value
=
shift
;
if
(
$value
>= 0) {
return
(
int
(
$value
+ 0.5));
}
else
{
return
(
int
(
$value
- 0.5));
}
}
=item B<_parse_iso_date>
Converts a UT date in form YYYY-MM-DDTHH:MM:SS.sss into a date
object (Time::Piece).
$object = $trans->_parse_iso_date( $date );
=cut
sub
_parse_iso_date {
my
$self
=
shift
;
my
$datestr
=
shift
;
my
$return
;
if
(
defined
$datestr
) {
# Not part of standard but we can deal with it
$datestr
=~ s/Z//g;
# Time::Piece can not do fractional seconds. Should switch to DateTime
$datestr
=~ s/\.\d+$//;
# parse
$return
= Time::Piece->strptime(
$datestr
,
"%Y-%m-%dT%T"
);
}
return
$return
;
}
=item B<_parse_yyyymmdd_date>
Converts a UT date in format YYYYMMDD into a date object.
$ojbect = $trans->_parse_yyyymmdd_date( $date, $sep );
Where $sep is the separator string and can be an empty string.
This allows 20090215, 2009-02-15 and 2009:02:15 to be parsed
by the same routine by using '', '-' and ':' respectively.
=cut
sub
_parse_yyyymmdd_date {
my
$self
=
shift
;
my
$datestr
=
shift
;
my
$sep
=
shift
;
$sep
=
''
unless
defined
$sep
;
# OSX Leopard has a completely broken strptime that can not
# handle %Y%m%d. We need to change the string to make it
# into a parseable form (or switch to DateTime).
if
(!
$sep
) {
$sep
=
"-"
;
$datestr
=
join
(
$sep
,
substr
(
$datestr
,0,4),
substr
(
$datestr
,4,2),
substr
(
$datestr
,6));
}
return
Time::Piece->strptime(
$datestr
,
join
(
$sep
,
'%Y'
,
'%m'
,
'%d'
) );
}
=item B<_add_seconds>
Add the supplied number of seconds to the supplied time object
and return a new object.
$new = $trans->_add_seconds( $base, $delta );
=cut
sub
_add_seconds {
my
$self
=
shift
;
my
$base
=
shift
;
my
$delta
=
shift
;
return
(
$base
+ Time::Seconds->new(
$delta
) );
}
=item B<_utdate_to_object>
Converts a UT date in YYYYMMDD format to a date object at midnight.
$obj = $trans->_utdate_to_object( $YYYYMMDD );
=cut
sub
_utdate_to_object {
my
$self
=
shift
;
my
$utdate
=
shift
;
my
$year
=
substr
(
$utdate
, 0, 4);
my
$month
=
substr
(
$utdate
, 4, 2);
my
$day
=
substr
(
$utdate
, 6, 2);
my
$basedate
=
$self
->_parse_iso_date(
$year
.
"-"
.
$month
.
"-"
.
$day
.
"T00:00:00"
);
return
$basedate
;
}
=item B<cosdeg>
Return the cosine of the angle. The angle must be in degrees.
=cut
sub
cosdeg {
my
$self
=
shift
;
my
$deg
=
shift
;
cos
( deg2rad(
$deg
) );
}
=item B<sindeg>
Return the sine of the angle. The angle must be in degrees.
=cut
sub
sindeg {
my
$self
=
shift
;
my
$deg
=
shift
;
sin
( deg2rad(
$deg
) );
}
=item B<via_subheader>
For the supplied FITS header item, first check the primary header
for existence, then check SUBHEADERS, then check "In" named subheaders.
In scalar context returns the first value that matches.
$value = $trans->via_subheader( $FITS_headers, $keyword );
In list context returns all the available values in order.
@values = $trans->via_subheader( $FITS_headers, $keyword );
=cut
sub
via_subheader {
my
$self
=
shift
;
my
$FITS_headers
=
shift
;
my
$keyword
=
shift
;
my
@values
;
if
(
exists
$FITS_headers
->{
$keyword
}
&&
defined
$FITS_headers
->{
$keyword
}) {
if
(
ref
(
$FITS_headers
->{
$keyword
} ) eq
'ARRAY'
) {
@values
= @{
$FITS_headers
->{
$keyword
}};
}
else
{
push
(
@values
,
$FITS_headers
->{
$keyword
});
}
}
elsif
(
$FITS_headers
->{SUBHEADERS}
&&
exists
$FITS_headers
->{SUBHEADERS}->[0]->{
$keyword
}) {
my
@subs
= @{
$FITS_headers
->{SUBHEADERS}};
for
my
$s
(
@subs
) {
if
(
exists
$s
->{
$keyword
} &&
defined
$s
->{
$keyword
}) {
push
(
@values
,
$s
->{
$keyword
});
}
}
}
elsif
(
exists
$FITS_headers
->{I1}
&&
exists
$FITS_headers
->{I1}->{
$keyword
}) {
# need to find out how many In we have
my
$i
= 1;
while
(
exists
$FITS_headers
->{
"I$i"
}) {
push
(
@values
,
$FITS_headers
->{
"I$i"
}->{
$keyword
});
$i
++;
}
}
return
(
wantarray
?
@values
:
$values
[0] );
}
=item B<via_subheader_undef_check>
Version of via_subheader that removes undefined values from the list before
returning the answer. Useful for SCUBA-2 where the first dark may not include
the TCS information.
Same interface as via_subheader.
=cut
sub
via_subheader_undef_check {
my
$self
=
shift
;
my
@values
=
$self
->via_subheader(
@_
);
# completely filter out undefs
@values
=
grep
{
defined
$_
}
@values
;
return
(
wantarray
?
@values
:
$values
[0] );
}
=back
=head1 PROTECTED IMPORTS
Not all translation methods warrant a full blown inheritance. For
cases where one or two translation routines should be imported
(e.g. reading DATE-OBS FITS standard headers without importing the
additional FITS methods) a special import routine can be used when
using the class.
use Astro::FITS::HdrTrans::FITS qw/ ROTATION /;
This will load the from_ROTATION and to_ROTATION methods into
the namespace.
=cut
sub
import
{
my
$class
=
shift
;
# this is where we are going to install the methods
my
$callpkg
=
caller
();
# Prepend the from_ and to_ prefixes
for
my
$key
(
@_
) {
# The key can be fully specified with from_ and to_ already
# In that case we do not want to loop over from_ and to_
my
@directions
=
qw/ from_ to_ /
;
if
(
$key
=~ /^from_/ ||
$key
=~ /^to_/) {
@directions
= (
''
);
# empty prefix
}
for
my
$dir
(
@directions
) {
my
$method
=
$dir
.
$key
;
#print "Importing method $method\n";
no
strict
'refs'
;
if
(!
defined
*{
"$class\::$method"
}) {
croak
"Method $method is not available for export from class $class"
;
}
# assign it
*{
"$callpkg\::$method"
} = \&{
"$class\::$method"
};
}
}
}
=head1 WRITING A TRANSLATION CLASS
In order to create a translation class for a new instrument it is
first necessary to work out the different types of translations that
are required; whether they are unit mappings (a simple change of
keyword but no change in value), constant mappings (a constant is
returned independently of the FITS header), mappings that already
exist in another class or complex mappings that have to be explicitly
coded.
All translation classes must ultimately inherit from
C<Astro::FITS::HdrTrans::Base>.
The first step in creation of a class is to handle the "can this class
translate the supplied headers" query that will be requested from
the C<Astro::FITS::HdrTrans> package. If the instrument name is present
in the standard "INSTRUME" FITS header then this can be achieved simply
by writing a C<this_instrument> method in the subclass that will return
the name of the instrument that can be translated. If a more complex
decision is required it will be necessary to subclass the C<can_translate>
method. This takes the headers that are to be translated (either in FITS
or generic form since the method is queried for either direction) and
returns a boolean indicating whether the class can be used.
Once the class can declare it's translation instrument the next
step is to write the actual translation methods themselves. If any
unit- or constant-mappings are required they can be setup by defining
the %UNIT_MAP and %CONST_MAP (the names are unimportant) hashes
and calling the base class automated method constructor:
__PACKAGE__->_generate_lookup_methods( \%CONST_MAP, \%UNIT_MAP );
If your translations are very similar to an existing set of translations
you can inherit from that class instead of C<Astro::FITS::HdrTrans::Base>.
Multiple inheritance is supported if, for example, you need to
inherit from both the standard FITS translations (eg for DATE-OBS
processing) and from a more telescope-specific set of translations.
If inheritance causes some erroneous mappings to leak through it is
possible to disable a specific mapping by specifying a @NULL_MAP
array to the method generation. This is an array of generic keywords.
__PACKAGE__->_generate_lookup_methods( \%CONST_MAP, \%UNIT_MAP,
\@NULL_MAP );
If a subset of translation methods are required from another class
but there is no desire to inherit the full set of methods then it
is possible to import specific translation methods from other classes.
use Astro::FITS::HdrTrans::FITS qw/ UTSTART UTEND /;
would import just the DATE-OBS and DATE-END handling functions from
the FITS class. Note that both the from- and to- translations will
be imported.
At some point you may want to write your own more complex translations.
To do this you must write to- and from- methods. The API for all
the from_FITS translations is identical:
$translation = $class->to_GENERIC_KEYWORD( \%fits_headers );
ie given a reference to a hash of FITS headers (which can be
a tied C<Astro::FITS::Header> object), return a scalar value which
is the translated value.
To convert from generic to FITS the interface is:
%fits_subset = $class->from_GENERIC_KEYWORD( \%generic_header );
ie multiple FITS keywords and values can be returned since in some
cases a single generic keyword is obtained by combining information
from multiple FITS headers.
Finally, if this translation module is to be part of the
C<Astro::FITS::HdrTrans> distribution the default list of translation
classes must be updated in C<Astro::FITS::HdrTrans>. If this is to be
a runtime plugin, then the list of classes can be expanded at
runtime. For example, it should be possible for
C<Astro::FITS::HdrTrans::MyNewInst> to automatically append itself to
the list of known classes if the module is explicitly loaded by the
user (rather than dynamically loaded to test the headers).
Some generic keywords actually return scalar objects. Any new instruments
must consistently return compatible objects. For example, UTDATE,
UTSTART and UTEND return (currently) Time::Piece objects.
=head1 REVISION
$Id$
=head1 SEE ALSO
C<Astro::FITS::HdrTrans>
=head1 AUTHOR
Tim Jenness E<lt>t.jenness@jach.hawaii.eduE<gt>
=head1 COPYRIGHT
Copyright (C) 2003-2005 Particle Physics and Astronomy Research Council.
All Rights Reserved.
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 2 of the License, 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., 59 Temple
Place,Suite 330, Boston, MA 02111-1307, USA
=cut
1;