You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

116 lines
4.3 KiB
Python

#!/usr/bin/env python2
import argparse, pythonwhois, json, datetime, sys
try:
from collections import OrderedDict
except ImportError as e:
from ordereddict import OrderedDict
parser = argparse.ArgumentParser(description="Retrieves and parses WHOIS data for a domain name.")
parser.add_argument("-r", "--raw", action="store_true", help="Outputs raw WHOIS data and doesn't attempt to parse it. Segments are separated by a double-dash (--).")
parser.add_argument("-j", "--json", action="store_true", help="Outputs structured WHOIS data in JSON format, according to the pythonwhois API.")
parser.add_argument("-f", "--file", action="store", help="Loads and parses raw double-dash-delimited WHOIS data from a specified file, instead of contacting a WHOIS server.", default=None)
parser.add_argument("domain", nargs=1)
args = parser.parse_args()
def json_fallback(obj):
if isinstance(obj, datetime.datetime):
return obj.isoformat()
else:
return obj
if args.file is None:
data, server_list = pythonwhois.net.get_whois_raw(args.domain[0], with_server_list=True)
else:
server_list = []
with open(args.file, "r") as f:
data = f.read().split("\n--\n")
if args.raw == True:
print("\n--\n".join([x.encode("utf-8") for x in data]))
else:
if len(server_list) > 0:
parsed = pythonwhois.parse.parse_raw_whois(data, normalized=True, never_query_handles=False, handle_server=server_list[-1])
else:
parsed = pythonwhois.parse.parse_raw_whois(data, normalized=True)
if args.json == True:
print(json.dumps(parsed, default=json_fallback))
else:
data_map = OrderedDict({})
# This defines the fields shown in the output
data_map["id"] = ("Domain ID", 1)
data_map["status"] = ("Status", 1)
data_map["registrar"] = ("Registrar", 1)
data_map["creation_date"] = ("Registration date", 1)
data_map["expiration_date"] = ("Expiration date", 1)
data_map["updated_date"] = ("Last update", 1)
data_map["nameservers"] = ("Name server", "+")
data_map["emails"] = ("E-mail address", "+")
widest_label = 0
for key, value in data_map.items():
if len(value[0]) > widest_label:
widest_label = len(value[0])
for key, value in data_map.items():
if key in parsed and parsed[key] is not None:
label = value[0] + (" " * (widest_label - len(value[0]))) + " :"
if value[1] == 1:
print("%s %s" % (label, parsed[key][0]))
elif value[1] == "+":
for item in parsed[key]:
print("%s %s" % (label, item))
if parsed["contacts"] is not None:
# This defines the contacts shown in the output
contacts_map = OrderedDict({})
contacts_map["registrant"] ="Registrant"
contacts_map["tech"] = "Technical Contact"
contacts_map["admin"] = "Administrative Contact"
contacts_map["billing"] = "Billing Contact"
# This defines the contact data shown in the output
data_map = OrderedDict({})
data_map["handle"] ="NIC handle"
data_map["name"] ="Name"
data_map["organization"] = "Organization"
data_map["street"] = "Street address"
data_map["postalcode"] = "Postal code"
data_map["city"] = "City"
data_map["state"] = "State / Province"
data_map["country"] = "Country"
data_map["email"] = "E-mail address"
data_map["phone"] = "Phone number"
data_map["fax"] = "Fax number"
data_map["creationdate"] = "Created"
data_map["changedate"] = "Last changed"
for contact in contacts_map:
if parsed["contacts"][contact] is not None:
contact_data = parsed["contacts"][contact]
print("\n" + contacts_map[contact])
for key, value in data_map.items():
if len(value) > widest_label:
widest_label = len(value)
for key, value in data_map.items():
if key in contact_data and contact_data[key] is not None:
label = " " + value + (" " * (widest_label - len(value))) + " :"
if sys.version_info < (3, 0):
if type(contact_data[key]) == str:
actual_data = contact_data[key].decode("utf-8")
elif type(contact_data[key]) == datetime.datetime:
actual_data = unicode(contact_data[key])
else:
actual_data = contact_data[key]
else:
actual_data = str(contact_data[key])
if "\n" in actual_data: # Indent multi-line values properly
lines = actual_data.split("\n")
actual_data = "\n".join([lines[0]] + [(" " * (widest_label + 7)) + line for line in lines[1:]])
print("%s %s" % (label, actual_data))