2021-05-20 12:05:33 -04:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
# qrzlogger
|
|
|
|
# =========
|
|
|
|
#
|
|
|
|
# This script is a QRZ.com command line QSO logger.
|
|
|
|
# It does the following:
|
|
|
|
# 1) asks the user for a call sign
|
|
|
|
# 2) displays available call sign info pulled from QRZ.com
|
|
|
|
# 3) displays all previous QSOs with this call (pulled from QRZ.com logbook)
|
|
|
|
# 4) alles the user to enter QSO specific data (date, time, report, band etc.)
|
|
|
|
# 5) uploads the QSO to QRZ.com's logbook
|
|
|
|
# 6) lists the last 5 logged QSOs ((pulled from QRZ.com logbook)
|
|
|
|
# 7) starts again from 1)
|
|
|
|
#
|
|
|
|
#
|
|
|
|
# MIT License
|
|
|
|
#
|
|
|
|
# Copyright (c) 2021 Michael Clemens, DL6MHC
|
|
|
|
#
|
|
|
|
# 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
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS 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.
|
|
|
|
|
|
|
|
|
|
|
|
# WARNING: This software is beta and is really not working properly yet!
|
|
|
|
# I'll remove this warning when it's done
|
|
|
|
|
|
|
|
|
|
|
|
import requests
|
|
|
|
import urllib
|
|
|
|
import re
|
|
|
|
import datetime
|
|
|
|
import os
|
|
|
|
import xmltodict
|
|
|
|
from prettytable import PrettyTable
|
|
|
|
from requests.structures import CaseInsensitiveDict
|
|
|
|
from datetime import date
|
|
|
|
from datetime import timezone
|
|
|
|
import configparser
|
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
# read the config file
|
2021-05-20 12:05:33 -04:00
|
|
|
config = configparser.ConfigParser()
|
|
|
|
config.read('config.ini')
|
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
# headers for all POST requests
|
2021-05-20 12:05:33 -04:00
|
|
|
headers = CaseInsensitiveDict()
|
|
|
|
headers["Content-Type"] = "application/x-www-form-urlencoded"
|
|
|
|
|
|
|
|
session = None
|
|
|
|
session_key = None
|
|
|
|
|
|
|
|
|
|
|
|
# Generate a session for QRZ.com's xml service with
|
|
|
|
# the help of the QRZ.com username and password
|
|
|
|
def get_session():
|
|
|
|
global session
|
|
|
|
global session_key
|
|
|
|
xml_auth_url = '''https://xmldata.QRZ.com/xml/current/?username={0}&password={1}'''.format(
|
|
|
|
config['qrzlogger']['qrz_user'],config['qrzlogger']['qrz_pass'])
|
|
|
|
session = requests.Session()
|
|
|
|
session.verify = bool(os.getenv('SSL_VERIFY', True))
|
|
|
|
r = session.get(xml_auth_url)
|
|
|
|
if r.status_code == 200:
|
|
|
|
raw_session = xmltodict.parse(r.content)
|
|
|
|
session_key = raw_session.get('QRZDatabase').get('Session').get('Key')
|
|
|
|
if session_key:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
# Query QRZ.com's xml api to gather information
|
|
|
|
# about a specific call sign
|
|
|
|
def getCallData(call):
|
|
|
|
global session
|
|
|
|
global session_key
|
|
|
|
|
|
|
|
xml_url = """https://xmldata.QRZ.com/xml/current/?s={0}&callsign={1}""" .format(session_key, call)
|
|
|
|
r = session.get(xml_url)
|
|
|
|
if r.status_code != 200:
|
|
|
|
print("nope")
|
|
|
|
raw = xmltodict.parse(r.content).get('QRZDatabase')
|
|
|
|
if not raw:
|
|
|
|
print("nope")
|
|
|
|
if raw['Session'].get('Error'):
|
|
|
|
errormsg = raw['Session'].get('Error')
|
|
|
|
else:
|
|
|
|
calldata = raw.get('Callsign')
|
|
|
|
if calldata:
|
|
|
|
return calldata
|
|
|
|
return "nope"
|
|
|
|
|
|
|
|
|
|
|
|
# Query QRZ.com's logbook for all previous QSOs
|
|
|
|
# with a specific call sign
|
|
|
|
def getQSOsForCallsign(callsign):
|
2021-05-20 17:36:19 -04:00
|
|
|
post_data = {
|
|
|
|
'KEY' : config['qrzlogger']['api_key'],
|
|
|
|
'ACTION' : 'FETCH',
|
|
|
|
'OPTION' : "TYPE:ADIF,CALL:" + callsign
|
|
|
|
}
|
|
|
|
post_data_enc = urllib.parse.urlencode(post_data)
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
resp = requests.post(config['qrzlogger']['api_url'], headers=headers, data=post_data_enc)
|
2021-05-20 12:05:33 -04:00
|
|
|
|
|
|
|
str_resp = resp.content.decode("utf-8")
|
|
|
|
response = urllib.parse.unquote(str_resp)
|
|
|
|
|
|
|
|
resp_list = response.splitlines()
|
|
|
|
result = [{}]
|
|
|
|
for i in resp_list:
|
|
|
|
if not i:
|
|
|
|
result.append({})
|
|
|
|
else:
|
|
|
|
if any(s+":" in i for s in config['qrzlogger']['xml_fields']):
|
|
|
|
i = re.sub('<','',i, flags=re.DOTALL)
|
|
|
|
i = re.sub(':.*>',":",i, flags=re.DOTALL)
|
|
|
|
v = re.sub('^.*:',"",i, flags=re.DOTALL)
|
|
|
|
k = re.sub(':.*$',"",i, flags=re.DOTALL)
|
|
|
|
result[-1][k] = v
|
|
|
|
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
|
|
|
# Generate a pretty ascii table containing all
|
|
|
|
# previous QSOs with a specific call sign
|
|
|
|
def getQSOTable(result):
|
|
|
|
t = PrettyTable(['Date', 'Time', 'Band', 'Mode', 'RST-S', 'RST-R', 'Comment'])
|
|
|
|
for d in result:
|
|
|
|
if "qso_date" in d:
|
|
|
|
date = datetime.datetime.strptime(d["qso_date"], '%Y%m%d').strftime('%Y/%m/%d')
|
|
|
|
time = datetime.datetime.strptime(d["time_on"], '%H%M').strftime('%H:%M')
|
|
|
|
comment = ""
|
|
|
|
try:
|
|
|
|
comment = d["comment"]
|
|
|
|
except:
|
|
|
|
comment = ""
|
|
|
|
t.add_row([date, time, d["band"], d["mode"], d["rst_sent"], d["rst_rcvd"], comment])
|
|
|
|
t.align = "r"
|
|
|
|
return t
|
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
|
2021-05-20 12:05:33 -04:00
|
|
|
# Print a pretty ascii table containing all interesting
|
|
|
|
# data found for a specific call sign
|
|
|
|
def getXMLQueryTable(result):
|
|
|
|
t = PrettyTable(['key', 'value'])
|
|
|
|
if "fname" in result:
|
|
|
|
t.add_row(["First Name", result["fname"]])
|
|
|
|
if "name" in result:
|
|
|
|
t.add_row(["Last Name", result["name"]])
|
|
|
|
if "addr1" in result:
|
|
|
|
t.add_row(["Street", result["addr1"]])
|
|
|
|
if "addr2" in result:
|
|
|
|
t.add_row(["City", result["addr2"]])
|
|
|
|
if "state" in result:
|
|
|
|
t.add_row(["State", result["state"]])
|
|
|
|
if "country" in result:
|
|
|
|
t.add_row(["Country", result["country"]])
|
|
|
|
if "grid" in result:
|
|
|
|
t.add_row(["Locator", result["grid"]])
|
|
|
|
if "email" in result:
|
|
|
|
t.add_row(["Email", result["email"]])
|
|
|
|
if "qslmgr" in result:
|
|
|
|
t.add_row(["QSL via:", result["qslmgr"]])
|
|
|
|
t.align = "l"
|
|
|
|
t.header = False
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
|
|
# Print a pretty ascii table containing all
|
|
|
|
# previously entered user data
|
|
|
|
def getQSODetailTable(qso):
|
|
|
|
t = PrettyTable(['key', 'value'])
|
|
|
|
for q in qso:
|
|
|
|
t.add_row([qso[q][0], qso[q][1]])
|
|
|
|
t.align = "l"
|
|
|
|
t.header = False
|
|
|
|
return t
|
|
|
|
|
|
|
|
|
|
|
|
# Queries QSO specific data from the user via
|
|
|
|
# the command line
|
|
|
|
def queryQSOData(qso):
|
|
|
|
dt = datetime.datetime.now(timezone.utc)
|
|
|
|
dt_now = dt.replace(tzinfo=timezone.utc)
|
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
# pre-fill the fields with date, time and
|
|
|
|
# default values from the config file
|
2021-05-20 12:05:33 -04:00
|
|
|
qso_date = dt_now.strftime("%Y%m%d")
|
2021-05-20 15:44:59 -04:00
|
|
|
time_on = dt_now.strftime("%H%M")
|
2021-05-20 17:36:19 -04:00
|
|
|
band = config['qrzlogger']['band']
|
|
|
|
mode = config['qrzlogger']['mode']
|
|
|
|
rst_rcvd = config['qrzlogger']['rst_rcvd']
|
|
|
|
rst_sent = config['qrzlogger']['rst_sent']
|
|
|
|
tx_pwr = config['qrzlogger']['tx_pwr']
|
2021-05-20 12:05:33 -04:00
|
|
|
comment = ""
|
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
# If this is the first try filling out the QSO fields
|
|
|
|
# then we use defaults
|
2021-05-20 12:05:33 -04:00
|
|
|
if qso is None:
|
|
|
|
questions = {
|
|
|
|
"qso_date" : ["QSO Date: ",qso_date],
|
2021-05-20 15:44:59 -04:00
|
|
|
"time_on": ["QSO Time: ", time_on],
|
2021-05-20 12:05:33 -04:00
|
|
|
"band": ["Band: ", band],
|
|
|
|
"mode": ["Mode: ", mode],
|
2021-05-20 15:37:43 -04:00
|
|
|
"rst_rcvd": ["RST Received: ", rst_rcvd],
|
|
|
|
"rst_sent": ["RST Sent: ", rst_sent],
|
2021-05-20 15:44:59 -04:00
|
|
|
"tx_pwr": ["Power (in W): ", tx_pwr],
|
2021-05-20 12:05:33 -04:00
|
|
|
"comment": ["Comment: ", comment]
|
|
|
|
}
|
2021-05-20 17:36:19 -04:00
|
|
|
# if this is not the first try, we pre-fill the
|
|
|
|
# vaulues we got from the last try
|
2021-05-20 12:05:33 -04:00
|
|
|
else:
|
|
|
|
questions = qso
|
|
|
|
|
2021-05-20 17:36:19 -04:00
|
|
|
# We now loop through all defined fields and ask
|
|
|
|
# the user for input
|
2021-05-20 12:05:33 -04:00
|
|
|
for q in questions:
|
|
|
|
inp = input(questions[q][0]+" ["+questions[q][1]+"]: " )
|
2021-05-20 17:36:19 -04:00
|
|
|
# If the user just hits enter, we keep the default value.
|
|
|
|
# If not, we keep the data provided by the user
|
2021-05-20 12:05:33 -04:00
|
|
|
if inp != "":
|
|
|
|
questions[q][1] = inp
|
|
|
|
|
|
|
|
return questions
|
|
|
|
|
|
|
|
|
|
|
|
# Sends the previously collected QSO information as a new
|
|
|
|
# QRZ.com logbook entry via the API
|
|
|
|
def sendQSO(qso):
|
2021-05-21 04:47:02 -04:00
|
|
|
is_ok = False
|
2021-05-20 15:37:43 -04:00
|
|
|
|
2021-05-20 16:55:40 -04:00
|
|
|
# construct ADIF QSO entry
|
|
|
|
adif = '<station_callsign:' + str(len(config['qrzlogger']['station_call'])) + '>' + config['qrzlogger']['station_call']
|
|
|
|
adif += '<call:' + str(len(call)) + '>' + call
|
|
|
|
for field in qso:
|
|
|
|
adif += '<' + field + ':' + str(len(qso[field][1])) + '>' + qso[field][1]
|
|
|
|
adif += '<eor>'
|
|
|
|
|
2021-05-21 06:19:02 -04:00
|
|
|
print(adif)
|
|
|
|
|
2021-05-20 16:55:40 -04:00
|
|
|
# construct POST data
|
|
|
|
post_data = { 'KEY' : config['qrzlogger']['api_key'], 'ACTION' : 'INSERT', 'ADIF' : adif }
|
|
|
|
|
|
|
|
# URL encode the payload
|
|
|
|
data = urllib.parse.urlencode(post_data)
|
|
|
|
# send the POST request to QRZ.com
|
2021-05-20 12:05:33 -04:00
|
|
|
resp = requests.post(config['qrzlogger']['api_url'], headers=headers, data=data)
|
2021-05-21 04:47:02 -04:00
|
|
|
str_resp = resp.content.decode("utf-8")
|
|
|
|
response = urllib.parse.unquote(str_resp)
|
|
|
|
# Check if the upload failed and print out
|
|
|
|
# the reason plus some additional info
|
|
|
|
if "STATUS=FAIL" in response:
|
|
|
|
print("\nQSO upload failed. QRZ.com has send the following reason:\n")
|
|
|
|
resp_list = response.split("&")
|
|
|
|
for item in resp_list:
|
|
|
|
print(item)
|
|
|
|
print("\nPlease review the following request that led to this error:\n")
|
|
|
|
print(post_data)
|
|
|
|
else:
|
|
|
|
print("QSO successfully uploaded to QRZ.com")
|
|
|
|
is_ok = True
|
|
|
|
return is_ok
|
2021-05-20 12:05:33 -04:00
|
|
|
|
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
# ask a user a simple y/n question
|
|
|
|
# returns True if "y"
|
|
|
|
# returns False in "n"
|
|
|
|
def askUser(question):
|
|
|
|
while True:
|
|
|
|
inp = input("\n" + question + " [y/n]: ")
|
|
|
|
if inp == "y":
|
|
|
|
return True
|
|
|
|
elif inp == "n":
|
|
|
|
return False
|
2021-05-20 12:05:33 -04:00
|
|
|
|
|
|
|
|
|
|
|
# Main routine
|
|
|
|
if __name__ == '__main__':
|
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
keeponlogging = True
|
2021-05-20 12:05:33 -04:00
|
|
|
get_session()
|
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
print(" _ ")
|
|
|
|
print(" __ _ _ _ __| |___ __ _ __ _ ___ _ _ ")
|
|
|
|
print(" / _` | '_|_ / / _ \/ _` / _` / -_) '_|")
|
|
|
|
print(" \__, |_| /__|_\___/\__, \__, \___|_| ")
|
|
|
|
print(" |_| |___/|___/ ")
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
while keeponlogging:
|
|
|
|
call = input("\n\nEnter Callsign: ")
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
print('\nQRZ.com results for {0}:\n'.format(call))
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
result = getCallData(call)
|
|
|
|
tab = getXMLQueryTable(result)
|
|
|
|
print(tab)
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
print('\n\nPrevious QSOs with {0}:\n'.format(call))
|
2021-05-20 12:05:33 -04:00
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
result = getQSOsForCallsign(call)
|
|
|
|
tab = getQSOTable(result)
|
2021-05-20 12:05:33 -04:00
|
|
|
print(tab)
|
|
|
|
|
2021-05-21 04:47:02 -04:00
|
|
|
print('\nEnter new QSO details below:\n')
|
|
|
|
|
|
|
|
qso_ok = False
|
|
|
|
qso = None
|
|
|
|
ask_try_again = False
|
|
|
|
|
|
|
|
while not qso_ok:
|
|
|
|
# query QSO details from thbe user
|
|
|
|
qso = queryQSOData(qso)
|
2021-05-21 06:19:02 -04:00
|
|
|
print(qso)
|
2021-05-21 04:47:02 -04:00
|
|
|
# generate a pretty table
|
|
|
|
tab = getQSODetailTable(qso)
|
|
|
|
print(tab)
|
|
|
|
# ask user if everything is ok. If not, start over.
|
|
|
|
if askUser("Is this correct?"):
|
|
|
|
qso_ok = sendQSO(qso)
|
|
|
|
# QSO successfully sent.
|
|
|
|
if qso_ok:
|
|
|
|
qso = None
|
|
|
|
keeponlogging = askUser("Log another QSO?")
|
|
|
|
# QSO upload failed
|
|
|
|
else:
|
|
|
|
ask_try_again = True
|
|
|
|
else:
|
|
|
|
ask_try_again = True
|
|
|
|
# We ask the user if he/she wants to try again
|
|
|
|
# and - if not - another QSO should be logged
|
|
|
|
if ask_try_again:
|
|
|
|
if not askUser("Try again?"):
|
|
|
|
# user answered with "n"
|
|
|
|
# we quit the loop and reset the QSO fields
|
|
|
|
qso_ok = True
|
|
|
|
qso = None
|
|
|
|
if not askUser("Log another QSO?"):
|
|
|
|
# quit the application
|
|
|
|
keeponlogging = False
|
|
|
|
|
|
|
|
print("\nBye, bye!")
|