Reference¶
pyhamtools.locator¶
- pyhamtools.locator.calculate_distance(locator1, locator2)¶
calculates the (shortpath) distance between two Maidenhead locators
- Parameters:
locator1 (string) – Locator, either 4 or 6 characters
locator2 (string) – Locator, either 4 or 6 characters
- Returns:
Distance in km
- Return type:
float
- Raises:
ValueError – When called with wrong or invalid input arg
AttributeError – When args are not a string
Example
The following calculates the distance between two Maidenhead locators in km
>>> from pyhamtools.locator import calculate_distance >>> calculate_distance("JN48QM", "QF67bf") 16466.413
- pyhamtools.locator.calculate_distance_longpath(locator1, locator2)¶
calculates the (longpath) distance between two Maidenhead locators
- Parameters:
locator1 (string) – Locator, either 4 or 6 characters
locator2 (string) – Locator, either 4 or 6 characters
- Returns:
Distance in km
- Return type:
float
- Raises:
ValueError – When called with wrong or invalid input arg
AttributeError – When args are not a string
Example
The following calculates the longpath distance between two Maidenhead locators in km
>>> from pyhamtools.locator import calculate_distance_longpath >>> calculate_distance_longpath("JN48QM", "QF67bf") 23541.5867
- pyhamtools.locator.calculate_heading(locator1, locator2)¶
calculates the heading from the first to the second locator
- Parameters:
locator1 (string) – Locator, either 4 or 6 characters
locator2 (string) – Locator, either 4 or 6 characters
- Returns:
Heading in deg
- Return type:
float
- Raises:
ValueError – When called with wrong or invalid input arg
AttributeError – When args are not a string
Example
The following calculates the heading from locator1 to locator2
>>> from pyhamtools.locator import calculate_heading >>> calculate_heading("JN48QM", "QF67bf") 74.3136
- pyhamtools.locator.calculate_heading_longpath(locator1, locator2)¶
calculates the heading from the first to the second locator (long path)
- Parameters:
locator1 (string) – Locator, either 4 or 6 characters
locator2 (string) – Locator, either 4 or 6 characters
- Returns:
Long path heading in deg
- Return type:
float
- Raises:
ValueError – When called with wrong or invalid input arg
AttributeError – When args are not a string
Example
The following calculates the long path heading from locator1 to locator2
>>> from pyhamtools.locator import calculate_heading_longpath >>> calculate_heading_longpath("JN48QM", "QF67bf") 254.3136
- pyhamtools.locator.calculate_sunrise_sunset(locator, calc_date=None)¶
calculates the next sunset and sunrise for a Maidenhead locator at a give date & time
- Parameters:
locator1 (string) – Maidenhead Locator, either 4 or 6 characters
calc_date (datetime, optional) – Starting datetime for the calculations (UTC)
- Returns:
Containing datetimes for morning_dawn, sunrise, evening_dawn, sunset
- Return type:
dict
- Raises:
ValueError – When called with wrong or invalid input arg
AttributeError – When args are not a string
Example
The following calculates the next sunrise & sunset for JN48QM on the 1./Jan/2014
>>> from pyhamtools.locator import calculate_sunrise_sunset >>> from datetime import datetime, timezone >>> myDate = datetime(year=2014, month=1, day=1, tzinfo=timezone.utc) >>> calculate_sunrise_sunset("JN48QM", myDate) { 'morning_dawn': datetime.datetime(2014, 1, 1, 6, 36, 51, 710524, tzinfo=datetime.timezone.utc), 'sunset': datetime.datetime(2014, 1, 1, 16, 15, 23, 31016, tzinfo=datetime.timezone.utc), 'evening_dawn': datetime.datetime(2014, 1, 1, 15, 38, 8, 355315, tzinfo=datetime.timezone.utc), 'sunrise': datetime.datetime(2014, 1, 1, 7, 14, 6, 162063, tzinfo=datetime.timezone.utc) }
- pyhamtools.locator.latlong_to_locator(latitude, longitude)¶
converts WGS84 coordinates into the corresponding Maidenhead Locator
- Parameters:
latitude (float) – Latitude
longitude (float) – Longitude
- Returns:
Maidenhead locator
- Return type:
string
- Raises:
ValueError – When called with wrong or invalid input args
TypeError – When args are non float values
Example
The following example converts latitude and longitude into the Maidenhead locator
>>> from pyhamtools.locator import latlong_to_locator >>> latitude = 48.5208333 >>> longitude = 9.375 >>> latlong_to_locator(latitude, longitude) 'JN48QM'
Note
Latitude (negative = West, positive = East) Longitude (negative = South, positive = North)
- pyhamtools.locator.locator_to_latlong(locator)¶
converts Maidenhead locator in the corresponding WGS84 coordinates
- Parameters:
locator (string) – Locator, either 4 or 6 characters
- Returns:
Latitude, Longitude
- Return type:
tuple (float, float)
- Raises:
ValueError – When called with wrong or invalid input arg
TypeError – When arg is not a string
Example
The following example converts a Maidenhead locator into Latitude and Longitude
>>> from pyhamtools.locator import locator_to_latlong >>> latitude, longitude = locator_to_latlong("JN48QM") >>> print latitude, longitude 48.5208333333 9.375
Note
Latitude (negative = West, positive = East) Longitude (negative = South, positive = North)
pyhamtools.qsl¶
- pyhamtools.qsl.get_clublog_users(**kwargs)¶
Download the latest official list of `Clublog`__ users.
- Args:
url (str, optional): Download URL
- Returns:
- dict: Dictionary containing (if data available) the fields:
firstqso, lastqso, last-lotw, lastupload (datetime), locator (string) and oqrs (boolean)
- Raises:
IOError: When network is unavailable, file can’t be downloaded or processed
- Example:
The following example downloads the Clublog user list and returns a dictionary with the data of HC2/AL1O:
>>> from pyhamtools.qsl import get_clublog_users >>> clublog = get_lotw_users() >>> clublog['HC2/AL1O'] {'firstqso': datetime.datetime(2012, 1, 1, 19, 59, 27), 'last-lotw': datetime.datetime(2013, 5, 9, 1, 56, 23), 'lastqso': datetime.datetime(2013, 5, 5, 6, 39, 3), 'lastupload': datetime.datetime(2013, 5, 8, 15, 0, 6), 'oqrs': True}
- pyhamtools.qsl.get_eqsl_users(**kwargs)¶
Download the latest official list of `EQSL.cc`__ users. The list of users can be found here.
- Args:
url (str, optional): Download URL
- Returns:
list: List containing the callsigns of EQSL users (unicode)
- Raises:
IOError: When network is unavailable, file can’t be downloaded or processed
- Example:
The following example downloads the EQSL user list and checks if DH1TW is a user:
>>> from pyhamtools.qsl import get_eqsl_users >>> mylist = get_eqsl_users() >>> try: >>> mylist.index('DH1TW') >>> except ValueError as e: >>> print e 'DH1TW' is not in list
- pyhamtools.qsl.get_lotw_users(**kwargs)¶
Download the latest official list of `ARRL Logbook of the World (LOTW)`__ users.
- Args:
url (str, optional): Download URL
- Returns:
dict: Dictionary containing the callsign (unicode) date of the last LOTW upload (datetime)
- Raises:
IOError: When network is unavailable, file can’t be downloaded or processed
ValueError: Raised when data from file can’t be read
- Example:
The following example downloads the LOTW user list and check when DH1TW has made his last LOTW upload:
>>> from pyhamtools.qsl import get_lotw_users >>> mydict = get_lotw_users() >>> mydict['DH1TW'] datetime.datetime(2014, 9, 7, 0, 0)
pyhamtools.frequency¶
- pyhamtools.frequency.freq_to_band(freq)¶
converts a Frequency [kHz] into the band and mode according to the IARU bandplan
- Parameters:
frequency (float) – Frequency in kHz
- Returns:
Dictionary containing the band (int) and mode (str)
- Return type:
dict
- Raises:
KeyError – Wrong frequency or out of band
Example
The following example converts the frequency 14005.3 kHz into band and mode.
>>> from pyhamtools.utils import freq_to_band >>> print freq_to_band(14005.3) { 'band': 20, 'mode': CW }
Note
Modes are:
CW
USB
LSB
DIGITAL
pyhamtools.callinfo¶
- class pyhamtools.callinfo.Callinfo(lookuplib, logger=None)¶
The purpose of this class is to return data (country, latitude, longitude, CQ Zone…etc) for an Amateur Radio callsign. The class can be used with any lookup database, provided through an Instance of
LookupLib
. An instance ofLookuplib
has to be injected on object construction.- Parameters:
lookuplib (
LookupLib
) – instance ofLookupLib
logger (logging.getLogger(__name__), optional) – Python logger
- get_adif_id(callsign, timestamp=None)¶
Returns ADIF id of a callsign’s country
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
containing the country ADIF id
- Return type:
int
- Raises:
KeyError – No Country found for callsign
- get_all(callsign, timestamp=None)¶
Lookup a callsign and return all data available from the underlying database
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
Dictionary containing the callsign specific data
- Return type:
dict
- Raises:
KeyError – Callsign could not be identified
Example
The following code returns all available information from the country-files.com database for the callsign “DH1TW”
>>> from pyhamtools import LookupLib, Callinfo >>> my_lookuplib = LookupLib(lookuptype="countryfile") >>> cic = Callinfo(my_lookuplib) >>> cic.get_all("DH1TW") { 'country': 'Fed. Rep. of Germany', 'adif': 230, 'continent': 'EU', 'latitude': 51.0, 'longitude': -10.0, 'cqz': 14, 'ituz': 28 }
Note
The content of the returned data depends entirely on the injected
LookupLib
(and the used database). While the country-files.com provides for example the ITU Zone, Clublog doesn’t. Consequently, the item “ituz” would be missing with Clublog (API or XML)LookupLib
.
- get_continent(callsign, timestamp=None)¶
Returns the continent Identifier of a callsign
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
continent identified
- Return type:
str
- Raises:
KeyError – No Continent found for callsign
Note
The following continent identifiers are used:
EU: Europe
NA: North America
SA: South America
AS: Asia
AF: Africa
OC: Oceania
AN: Antarctica
- get_country_name(callsign, timestamp=None)¶
Returns the country name where the callsign is located
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
name of the Country
- Return type:
str
- Raises:
KeyError – No Country found for callsign
Note
Don’t rely on the country name when working with several instances of py:class:Callinfo. Clublog and Country-files.org use slightly different names for countries. Example:
Country-files.com: “Fed. Rep. of Germany”
Clublog: “FEDERAL REPUBLIC OF GERMANY”
- get_cqz(callsign, timestamp=None)¶
Returns CQ Zone of a callsign
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
containing the callsign’s CQ Zone
- Return type:
int
- Raises:
KeyError – no CQ Zone found for callsign
- static get_homecall(callsign)¶
Strips off country prefixes (HC2/DH1TW) and activity suffixes (DH1TW/P).
- Parameters:
callsign (str) – Amateur Radio callsign
- Returns:
callsign without country/activity pre/suffixes
- Return type:
str
- Raises:
ValueError – No callsign found in string
Example
The following code retrieves the home call for “HC2/DH1TW/P”
>>> from pyhamtools import LookupLib, Callinfo >>> my_lookuplib = LookupLib(lookuptype="countryfile") >>> cic = Callinfo(my_lookuplib) >>> cic.get_homecall("HC2/DH1TW/P") DH1TW
- get_ituz(callsign, timestamp=None)¶
Returns ITU Zone of a callsign
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
containing the callsign’s CQ Zone
- Return type:
int
- Raises:
KeyError – No ITU Zone found for callsign
Note
Currently, only Country-files.com lookup database contains ITU Zones
- get_lat_long(callsign, timestamp=None)¶
Returns Latitude and Longitude for a callsign
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
Containing Latitude and Longitude
- Return type:
dict
- Raises:
KeyError – No data found for callsign
Example
The following code returns Latitude & Longitude for “DH1TW”
>>> from pyhamtools import LookupLib, Callinfo >>> my_lookuplib = LookupLib(lookuptype="countryfile") >>> cic = Callinfo(my_lookuplib) >>> cic.get_lat_long("DH1TW") { 'latitude': 51.0, 'longitude': -10.0 }
Note
Unfortunately, in most cases the returned Latitude and Longitude are not very precise. Clublog and Country-files.com use the country’s capital coordinates in most cases, if no dedicated entry in the database exists. Best results will be retrieved with QRZ.com Lookup.
- is_valid_callsign(callsign, timestamp=None)¶
Checks if a callsign is valid
- Parameters:
callsign (str) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
True / False
- Return type:
bool
Example
The following checks if “DH1TW” is a valid callsign
>>> from pyhamtools import LookupLib, Callinfo >>> my_lookuplib = LookupLib(lookuptype="countryfile") >>> cic = Callinfo(my_lookuplib) >>> cic.is_valid_callsign("DH1TW") True
pyhamtools.lookuplib¶
- class pyhamtools.lookuplib.LookupLib(lookuptype='countryfile', apikey=None, apiv='1.3.3', filename=None, logger=None, username=None, pwd=None, redis_instance=None, redis_prefix=None)¶
This class is a wrapper for the following three Amateur Radio databases:
Clublog.org (daily updated XML File)
Clublog.org (HTTPS lookup)
Country-files.com (infrequently updated PLIST File)
QRZ.com (HTTP / XML Lookup)
It’s aim is to provide a homogeneous interface to different databases.
Typically an instance of this class is injected as a dependency in the
Callinfo
class, but it can also be used directly.Even the interface is the same for all lookup sources, the returning data can be different. The documentation of the various methods provide more detail.
By default, LookupLib requires an Internet connection to download the libraries or perform the lookup against the Clublog API or QRZ.com.
The entire lookup data (where database files are downloaded) can also be copied into Redis, which an extremely fast in-memory Key/Value store. A LookupLib object can be instantiated to perform then all lookups in Redis, instead processing and loading the data from Internet / File. This saves some time and allows several instances of
LookupLib
to query the same data concurrently.- Parameters:
lookuptype (str) – “clublogxml” or “clublogapi” or “countryfile” or “redis” or “qrz”
apikey (str) – Clublog API Key
username (str) – QRZ.com username
pwd (str) – QRZ.com password
apiv (str, optional) – QRZ.com API Version
filename (str, optional) – Filename for Clublog XML or Country-files.com cty.plist file. When a local file is
used –
necessary. (no Internet connection not API Key is) –
logger (logging.getLogger(__name__), optional) – Python logger
redis_instance (redis.Redis(), optional) – Instance of Redis
redis_prefix (str, optional) – Prefix to identify the lookup data set in Redis
- copy_data_in_redis(redis_prefix, redis_instance)¶
Copy the complete lookup data into redis. Old data will be overwritten.
- Parameters:
redis_prefix (str) – Prefix to distinguish the data in redis for the different looktypes
redis_instance (str) – an Instance of Redis
- Returns:
returns True when the data has been copied successfully into Redis
- Return type:
bool
Example
Copy the entire lookup data from the Country-files.com PLIST File into Redis. This example requires a running instance of Redis, as well the python Redis connector (pip install redis-py).
>>> from pyhamtools import LookupLib >>> import redis >>> r = redis.Redis() >>> my_lookuplib = LookupLib(lookuptype="countryfile") >>> print my_lookuplib.copy_data_in_redis(redis_prefix="CF", redis_instance=r) True
Now let’s create an instance of LookupLib, using Redis to query the data
>>> from pyhamtools import LookupLib >>> import redis >>> r = redis.Redis() >>> my_lookuplib = LookupLib(lookuptype="redis", redis_instance=r, redis_prefix="CF") >>> my_lookuplib.lookup_callsign("3D2RI") { u'adif': 460, u'continent': u'OC', u'country': u'Rotuma Island', u'cqz': 32, u'ituz': 56, u'latitude': -12.48, u'longitude': 177.08 }
Note
This method is available for the following lookup type
clublogxml
countryfile
- is_invalid_operation(callsign, timestamp=None)¶
Returns True if an operations is known as invalid
- Parameters:
callsign (string) – Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
True if a record exists for this callsign (at the given time)
- Return type:
bool
- Raises:
KeyError – No matching callsign found
APIKeyMissingError – API Key for Clublog missing or incorrect
Example
The following code checks the Clublog XML database if the operation is valid for two dates.
>>> from pyhamtools import LookupLib >>> from datetime import datetime, timezone >>> my_lookuplib = LookupLib(lookuptype="clublogxml", apikey="myapikey") >>> print my_lookuplib.is_invalid_operation("5W1CFN") True >>> try: >>> timestamp = datetime(year=2012, month=1, day=31, tzinfo=timezone.utc) >>> my_lookuplib.is_invalid_operation("5W1CFN", timestamp) >>> except KeyError: >>> print "Seems to be invalid operation before 31.1.2012" Seems to be an invalid operation before 31.1.2012
Note
This method is available for
clublogxml
redis
- lookup_callsign(callsign=None, timestamp=None)¶
Returns lookup data if an exception exists for a callsign
- Parameters:
callsign (string) – Amateur radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
Dictionary containing the country specific data of the callsign
- Return type:
dict
- Raises:
KeyError – No matching callsign found
APIKeyMissingError – API Key for Clublog missing or incorrect
Example
The following code queries the the online Clublog API for the callsign “VK9XO” on a specific date.
>>> from pyhamtools import LookupLib >>> from datetime import datetime, timezone >>> my_lookuplib = LookupLib(lookuptype="clublogapi", apikey="myapikey") >>> timestamp = datetime(year=1962, month=7, day=7, tzinfo=timezone.utc) >>> print my_lookuplib.lookup_callsign("VK9XO", timestamp) { 'country': u'CHRISTMAS ISLAND', 'longitude': 105.7, 'cqz': 29, 'adif': 35, 'latitude': -10.5, 'continent': u'OC' }
Note
This method is available for
clublogxml
clublogapi
countryfile
qrz.com
redis
- lookup_entity(entity=None)¶
Returns lookup data of an ADIF Entity
- Parameters:
entity (int) – ADIF identifier of country
- Returns:
Dictionary containing the country specific data
- Return type:
dict
- Raises:
KeyError – No matching entity found
Example
The following code queries the the Clublog XML database for the ADIF entity Turkmenistan, which has the id 273.
>>> from pyhamtools import LookupLib >>> my_lookuplib = LookupLib(lookuptype="clublogapi", apikey="myapikey") >>> print my_lookuplib.lookup_entity(273) { 'deleted': False, 'country': u'TURKMENISTAN', 'longitude': 58.4, 'cqz': 17, 'prefix': u'EZ', 'latitude': 38.0, 'continent': u'AS' }
Note
This method is available for the following lookup type
clublogxml
redis
qrz.com
- lookup_prefix(prefix, timestamp=None)¶
Returns lookup data of a Prefix
- Parameters:
prefix (string) – Prefix of a Amateur Radio callsign
timestamp (datetime, optional) – datetime in UTC (tzinfo=timezone.utc)
- Returns:
Dictionary containing the country specific data of the Prefix
- Return type:
dict
- Raises:
KeyError – No matching Prefix found
APIKeyMissingError – API Key for Clublog missing or incorrect
Example
The following code shows how to obtain the information for the prefix “DH” from the countryfile.com database (default database).
>>> from pyhamtools import LookupLib >>> myLookupLib = LookupLib() >>> print myLookupLib.lookup_prefix("DH") { 'adif': 230, 'country': u'Fed. Rep. of Germany', 'longitude': 10.0, 'cqz': 14, 'ituz': 28, 'latitude': 51.0, 'continent': u'EU' }
Note
This method is available for
clublogxml
countryfile
redis
- lookup_zone_exception(callsign, timestamp=None)¶
Returns a CQ Zone if an exception exists for the given callsign
Args: callsign (string): Amateur radio callsign timestamp (datetime, optional): datetime in UTC (tzinfo=timezone.utc)
- Returns:
Value of the the CQ Zone exception which exists for this callsign (at the given time)
- Return type:
int
- Raises:
KeyError – No matching callsign found
APIKeyMissingError – API Key for Clublog missing or incorrect
Example
The following code checks the Clublog XML database if a CQ Zone exception exists for the callsign DP0GVN.
>>> from pyhamtools import LookupLib >>> my_lookuplib = LookupLib(lookuptype="clublogxml", apikey="myapikey") >>> print my_lookuplib.lookup_zone_exception("DP0GVN") 38
The prefix “DP” It is assigned to Germany, but the station is located in Antarctica, and therefore in CQ Zone 38
Note
This method is available for
clublogxml
redis
pyhamtools.utils (deprecated)¶
- pyhamtools.utils.freq_to_band(freq)¶
converts a Frequency [kHz] into the band and mode according to the IARU bandplan
Note
DEPRECATION NOTICE This function has been moved to pyhamtools.frequency with PyHamTools 0.4.1 Please don’t use this module/function anymore. It will be removed soon.