Working
This commit is contained in:
parent
374d77a2d1
commit
51a0c28bcf
1
.github/workflows/ci.yml
vendored
1
.github/workflows/ci.yml
vendored
@ -6,6 +6,7 @@ on:
|
||||
- 'README.md'
|
||||
- 'LICENSE.md'
|
||||
- '.gitignore'
|
||||
- 'renovate.json'
|
||||
pull_request:
|
||||
|
||||
jobs:
|
||||
|
0
app/__init__.py
Normal file
0
app/__init__.py
Normal file
30
app/args.py
Normal file
30
app/args.py
Normal file
@ -0,0 +1,30 @@
|
||||
#!/usr/local/bin/python3
|
||||
|
||||
import argparse
|
||||
|
||||
from app.query_normalisation import get_public_ip
|
||||
|
||||
|
||||
def parse_args() -> argparse.Namespace:
|
||||
parser = argparse.ArgumentParser(
|
||||
description="Query information about an IP address or domain name."
|
||||
)
|
||||
parser.add_argument(
|
||||
"-q",
|
||||
"--query",
|
||||
help="IP/domain name to query (default: current public IP)",
|
||||
default=get_public_ip(),
|
||||
)
|
||||
parser.add_argument(
|
||||
"-p",
|
||||
"--prefixes",
|
||||
help="show advertised prefixes",
|
||||
action="store_true",
|
||||
)
|
||||
parser.add_argument(
|
||||
"-n",
|
||||
"--noheader",
|
||||
help="do not print header",
|
||||
action="store_true",
|
||||
)
|
||||
return parser.parse_args()
|
25
app/ip_info.py
Normal file
25
app/ip_info.py
Normal file
@ -0,0 +1,25 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
import ipaddress
|
||||
import requests
|
||||
|
||||
|
||||
def get_ip_information(ipv4_address: ipaddress.IPv4Address) -> dict:
|
||||
api_endpoint = "http://ip-api.com/json/{}".format(ipv4_address)
|
||||
resp = requests.get(api_endpoint).json()
|
||||
return resp
|
||||
|
||||
|
||||
def get_autonomous_system_number(as_info: str) -> str:
|
||||
as_number = as_info.split(" ")[0]
|
||||
return as_number
|
||||
|
||||
|
||||
def get_prefix_information(autonomous_system: int) -> list:
|
||||
api_endpoint = "https://api.hackertarget.com/aslookup/?q={}".format(
|
||||
str(autonomous_system)
|
||||
)
|
||||
resp = requests.get(api_endpoint).text
|
||||
prefixes = resp.split("\n")
|
||||
prefixes.pop(0)
|
||||
return prefixes
|
58
app/main.py
58
app/main.py
@ -1,8 +1,62 @@
|
||||
#!/usr/local/bin/python3
|
||||
|
||||
from app.args import parse_args
|
||||
from app.print_table import print_table, generate_prefix_string
|
||||
from app.query_normalisation import is_ip_address, resolve_domain_name
|
||||
from app.ip_info import (
|
||||
get_ip_information,
|
||||
get_autonomous_system_number,
|
||||
get_prefix_information,
|
||||
)
|
||||
|
||||
|
||||
HEADER = """-----------------------------------------------
|
||||
| IP Address Information Lookup Tool (iPilot) |
|
||||
| By Luke Tainton (@luketainton) |
|
||||
-----------------------------------------------\n"""
|
||||
|
||||
|
||||
def main():
|
||||
# Commands here
|
||||
|
||||
args = parse_args()
|
||||
if not args.noheader:
|
||||
print(HEADER)
|
||||
|
||||
# Set IP to passed IP address, or resolve passed domain name to IPv4
|
||||
ip = (
|
||||
resolve_domain_name(args.query) if not is_ip_address(args.query) else args.query
|
||||
)
|
||||
|
||||
# If not given an IPv4, and can't resolve to IPv4, then throw error and exit
|
||||
if not ip:
|
||||
print("ERROR: could not resolve query to IPv4 address.")
|
||||
exit(1)
|
||||
|
||||
# Get information from API
|
||||
ip_info = get_ip_information(ip)
|
||||
as_number = get_autonomous_system_number(ip_info.get("as"))
|
||||
|
||||
# Assemble list for table generation
|
||||
table_data = [
|
||||
["IP Address", ip_info.get("query")],
|
||||
["Organization", ip_info.get("org")],
|
||||
[
|
||||
"Location",
|
||||
"{}/{}/{}".format(
|
||||
ip_info.get("country"), ip_info.get("regionName"), ip_info.get("city")
|
||||
),
|
||||
],
|
||||
["Timezone", ip_info.get("timezone")],
|
||||
["Internet Service Provider", ip_info.get("isp")],
|
||||
["Autonomous System", as_number],
|
||||
]
|
||||
|
||||
# If wanted, get prefix information
|
||||
if args.prefixes:
|
||||
prefix_info = get_prefix_information(as_number)
|
||||
table_data.append(["Prefixes", generate_prefix_string(prefix_info)])
|
||||
|
||||
print_table(table_data)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
|
18
app/print_table.py
Normal file
18
app/print_table.py
Normal file
@ -0,0 +1,18 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from tabulate import tabulate
|
||||
|
||||
|
||||
def generate_prefix_string(prefixes: list) -> str:
|
||||
n = 4
|
||||
try:
|
||||
ret = ""
|
||||
for i in range(0, len(prefixes), n):
|
||||
ret += ", ".join(prefixes[i : i + n]) + "\n"
|
||||
return ret
|
||||
except AttributeError:
|
||||
return None
|
||||
|
||||
|
||||
def print_table(table_data) -> None:
|
||||
print(tabulate(table_data))
|
26
app/query_normalisation.py
Normal file
26
app/query_normalisation.py
Normal file
@ -0,0 +1,26 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
import ipaddress
|
||||
import requests
|
||||
import socket
|
||||
|
||||
|
||||
def is_ip_address(query: str) -> bool:
|
||||
try:
|
||||
ipaddress.ip_address(query)
|
||||
return True
|
||||
except ValueError:
|
||||
return False
|
||||
|
||||
|
||||
def resolve_domain_name(domain_name: str) -> ipaddress.IPv4Address:
|
||||
try:
|
||||
ip = socket.gethostbyname(domain_name)
|
||||
except socket.gaierror:
|
||||
ip = None
|
||||
return ip
|
||||
|
||||
|
||||
def get_public_ip() -> ipaddress.IPv4Address:
|
||||
ip = requests.get("https://api.ipify.org").text
|
||||
return ip
|
0
tests/__init__.py
Normal file
0
tests/__init__.py
Normal file
25
tests/test_ip_info.py
Normal file
25
tests/test_ip_info.py
Normal file
@ -0,0 +1,25 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from app.ip_info import (
|
||||
get_ip_information,
|
||||
get_autonomous_system_number,
|
||||
get_prefix_information,
|
||||
)
|
||||
|
||||
|
||||
def test_get_ip_information() -> None:
|
||||
test_query = "1.2.3.4"
|
||||
ip_info = get_ip_information(test_query)
|
||||
assert ip_info.get("status") == "success" and ip_info.get("query") == test_query
|
||||
|
||||
|
||||
def test_get_autonomous_system_number() -> None:
|
||||
as_info = "AS5089 Virgin Media Limited"
|
||||
as_number = get_autonomous_system_number(as_info)
|
||||
assert as_number == "AS5089"
|
||||
|
||||
|
||||
def test_get_prefix_information() -> None:
|
||||
autonomous_system = "AS109"
|
||||
prefixes = get_prefix_information(autonomous_system)
|
||||
assert "144.254.0.0/16" in prefixes
|
@ -0,0 +1,18 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
# from app.ip import is_ip_address
|
||||
|
||||
|
||||
# def test_is_ip_address_true() -> None:
|
||||
# test_query = "1.2.3.4"
|
||||
# assert is_ip_address(test_query)
|
||||
|
||||
|
||||
# def test_is_ip_address_false_ip() -> None:
|
||||
# test_query = "256.315.16.23"
|
||||
# assert not is_ip_address(test_query)
|
||||
|
||||
|
||||
# def test_is_ip_address_false_fqdn() -> None:
|
||||
# test_query = "google.com"
|
||||
# assert not is_ip_address(test_query)
|
34
tests/test_query_normalisation.py
Normal file
34
tests/test_query_normalisation.py
Normal file
@ -0,0 +1,34 @@
|
||||
#!/usr/bin/env python3
|
||||
|
||||
from app.query_normalisation import is_ip_address, resolve_domain_name, get_public_ip
|
||||
|
||||
|
||||
def test_is_ip_address_true() -> None:
|
||||
test_query = "1.2.3.4"
|
||||
assert is_ip_address(test_query)
|
||||
|
||||
|
||||
def test_is_ip_address_false_ip() -> None:
|
||||
test_query = "256.315.16.23"
|
||||
assert not is_ip_address(test_query)
|
||||
|
||||
|
||||
def test_is_ip_address_false_fqdn() -> None:
|
||||
test_query = "google.com"
|
||||
assert not is_ip_address(test_query)
|
||||
|
||||
|
||||
def test_resolve_domain_name_true() -> None:
|
||||
domain_name = "one.one.one.one"
|
||||
expected_results = ["1.1.1.1", "1.0.0.1"] # Could resolve to either IP
|
||||
assert resolve_domain_name(domain_name) in expected_results
|
||||
|
||||
|
||||
def test_resolve_domain_name_false() -> None:
|
||||
domain_name = "hrrijoresdo.com"
|
||||
assert not resolve_domain_name(domain_name)
|
||||
|
||||
|
||||
def test_get_public_ip() -> None:
|
||||
public_ip = get_public_ip()
|
||||
assert is_ip_address(public_ip)
|
Loading…
x
Reference in New Issue
Block a user