2005-02-02 06:29:31 +01:00
|
|
|
###
|
|
|
|
# Copyright (c) 2003-2005, Jeremiah Fincher
|
2012-09-01 16:16:48 +02:00
|
|
|
# Copyright (c) 2010-2011, James McCoy
|
2005-02-02 06:29:31 +01:00
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer.
|
|
|
|
# * Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
# this list of conditions, and the following disclaimer in the
|
|
|
|
# documentation and/or other materials provided with the distribution.
|
|
|
|
# * Neither the name of the author of this software nor the name of
|
|
|
|
# contributors to this software may be used to endorse or promote products
|
|
|
|
# derived from this software without specific prior written consent.
|
|
|
|
#
|
|
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
###
|
|
|
|
|
2013-01-05 18:46:56 +01:00
|
|
|
import time
|
2005-02-02 06:29:31 +01:00
|
|
|
import socket
|
|
|
|
import telnetlib
|
|
|
|
|
2015-08-30 17:33:39 +02:00
|
|
|
import supybot.conf as conf
|
2005-02-02 06:29:31 +01:00
|
|
|
import supybot.utils as utils
|
|
|
|
from supybot.commands import *
|
|
|
|
from supybot.utils.iter import any
|
|
|
|
import supybot.callbacks as callbacks
|
2010-10-17 15:36:26 +02:00
|
|
|
from supybot.i18n import PluginInternationalization, internationalizeDocstring
|
|
|
|
_ = PluginInternationalization('Internet')
|
2005-02-02 06:29:31 +01:00
|
|
|
|
2005-02-09 08:04:04 +01:00
|
|
|
class Internet(callbacks.Plugin):
|
2014-11-30 21:07:41 +01:00
|
|
|
"""Provides commands to query DNS, search WHOIS databases,
|
|
|
|
and convert IPs to hex."""
|
2005-02-02 06:29:31 +01:00
|
|
|
threaded = True
|
2010-10-17 15:36:26 +02:00
|
|
|
@internationalizeDocstring
|
2005-02-02 06:29:31 +01:00
|
|
|
def dns(self, irc, msg, args, host):
|
|
|
|
"""<host|ip>
|
|
|
|
|
|
|
|
Returns the ip of <host> or the reverse DNS hostname of <ip>.
|
|
|
|
"""
|
|
|
|
if utils.net.isIP(host):
|
|
|
|
hostname = socket.getfqdn(host)
|
|
|
|
if hostname == host:
|
2010-10-17 15:36:26 +02:00
|
|
|
irc.reply(_('Host not found.'))
|
2005-02-02 06:29:31 +01:00
|
|
|
else:
|
|
|
|
irc.reply(hostname)
|
|
|
|
else:
|
|
|
|
try:
|
2014-07-23 19:22:15 +02:00
|
|
|
ips = socket.getaddrinfo(host, None)
|
|
|
|
ips = map(lambda x:x[4][0], ips)
|
2014-07-23 19:46:15 +02:00
|
|
|
ordered_unique_ips = []
|
|
|
|
unique_ips = set()
|
|
|
|
for ip in ips:
|
|
|
|
if ip not in unique_ips:
|
|
|
|
ordered_unique_ips.append(ip)
|
|
|
|
unique_ips.add(ip)
|
2014-07-23 19:47:42 +02:00
|
|
|
irc.reply(format('%L', ordered_unique_ips))
|
2005-02-02 06:29:31 +01:00
|
|
|
except socket.error:
|
2010-10-17 15:36:26 +02:00
|
|
|
irc.reply(_('Host not found.'))
|
2005-02-02 06:29:31 +01:00
|
|
|
dns = wrap(dns, ['something'])
|
|
|
|
|
|
|
|
_domain = ['Domain Name', 'Server Name', 'domain']
|
2016-08-09 21:08:02 +02:00
|
|
|
_netrange = ['NetRange', 'inetnum']
|
2005-02-02 06:29:31 +01:00
|
|
|
_registrar = ['Sponsoring Registrar', 'Registrar', 'source']
|
2016-08-09 21:08:02 +02:00
|
|
|
_netname = ['NetName', 'inetname']
|
2005-02-02 06:29:31 +01:00
|
|
|
_updated = ['Last Updated On', 'Domain Last Updated Date', 'Updated Date',
|
2016-08-09 21:08:02 +02:00
|
|
|
'Last Modified', 'changed', 'last-modified']
|
|
|
|
_created = ['Created On', 'Domain Registration Date', 'Creation Date',
|
|
|
|
'created', 'RegDate']
|
2005-02-02 06:29:31 +01:00
|
|
|
_expires = ['Expiration Date', 'Domain Expiration Date']
|
|
|
|
_status = ['Status', 'Domain Status', 'status']
|
2010-10-17 15:36:26 +02:00
|
|
|
@internationalizeDocstring
|
2005-02-02 06:29:31 +01:00
|
|
|
def whois(self, irc, msg, args, domain):
|
|
|
|
"""<domain>
|
|
|
|
|
|
|
|
Returns WHOIS information on the registration of <domain>.
|
|
|
|
"""
|
2016-08-09 21:08:02 +02:00
|
|
|
if utils.net.isIP(domain):
|
|
|
|
whois_server = 'whois.arin.net'
|
|
|
|
usertld = 'ipaddress'
|
|
|
|
elif '.' in domain:
|
|
|
|
usertld = domain.split('.')[-1]
|
|
|
|
whois_server = '%s.whois-servers.net' % usertld
|
|
|
|
else:
|
|
|
|
usertld = None
|
|
|
|
whois_server = 'whois.iana.org'
|
2005-02-02 06:29:31 +01:00
|
|
|
try:
|
2016-08-09 21:08:02 +02:00
|
|
|
sock = utils.net.getSocket(whois_server,
|
2015-08-30 17:33:39 +02:00
|
|
|
vhost=conf.supybot.protocols.irc.vhost(),
|
|
|
|
vhostv6=conf.supybot.protocols.irc.vhostv6(),
|
|
|
|
)
|
2016-08-09 21:08:02 +02:00
|
|
|
sock.connect((whois_server, 43))
|
2014-01-20 15:49:15 +01:00
|
|
|
except socket.error as e:
|
2005-02-02 06:29:31 +01:00
|
|
|
irc.error(str(e))
|
|
|
|
return
|
2013-01-05 18:46:56 +01:00
|
|
|
sock.settimeout(5)
|
2013-01-05 18:20:52 +01:00
|
|
|
if usertld == 'com':
|
2013-01-06 16:58:02 +01:00
|
|
|
sock.send(b'=')
|
2016-08-09 21:08:02 +02:00
|
|
|
elif usertld == 'ipaddress':
|
|
|
|
sock.send(b'n + ')
|
2013-01-05 18:46:56 +01:00
|
|
|
sock.send(domain.encode('ascii'))
|
|
|
|
sock.send(b'\r\n')
|
|
|
|
|
2013-01-06 16:58:02 +01:00
|
|
|
s = b''
|
2013-01-05 18:46:56 +01:00
|
|
|
end_time = time.time() + 5
|
|
|
|
try:
|
|
|
|
while end_time>time.time():
|
2016-08-09 21:08:02 +02:00
|
|
|
time.sleep(0.1)
|
2013-01-05 18:46:56 +01:00
|
|
|
s += sock.recv(4096)
|
|
|
|
except socket.error:
|
|
|
|
pass
|
2016-08-09 21:08:02 +02:00
|
|
|
sock.close()
|
|
|
|
server = netrange = netname = registrar = updated = created = expires = status = ''
|
2005-02-02 06:29:31 +01:00
|
|
|
for line in s.splitlines():
|
2013-08-06 12:16:52 +02:00
|
|
|
line = line.decode('utf8').strip()
|
2005-02-02 06:29:31 +01:00
|
|
|
if not line or ':' not in line:
|
|
|
|
continue
|
|
|
|
if not server and any(line.startswith, self._domain):
|
|
|
|
server = ':'.join(line.split(':')[1:]).strip().lower()
|
|
|
|
# Let's add this check so that we don't respond with info for
|
|
|
|
# a different domain. E.g., doing a whois for microsoft.com
|
|
|
|
# and replying with the info for microsoft.com.wanadoodoo.com
|
|
|
|
if server != domain:
|
|
|
|
server = ''
|
|
|
|
continue
|
2016-08-09 21:08:02 +02:00
|
|
|
if not netrange and any(line.startswith, self._netrange):
|
|
|
|
netrange = ':'.join(line.split(':')[1:]).strip()
|
|
|
|
if not server and not netrange:
|
2005-02-02 06:29:31 +01:00
|
|
|
continue
|
|
|
|
if not registrar and any(line.startswith, self._registrar):
|
|
|
|
registrar = ':'.join(line.split(':')[1:]).strip()
|
2016-08-09 21:08:02 +02:00
|
|
|
elif not netname and any(line.startswith, self._netname):
|
|
|
|
netname = ':'.join(line.split(':')[1:]).strip()
|
2005-02-02 06:29:31 +01:00
|
|
|
elif not updated and any(line.startswith, self._updated):
|
|
|
|
s = ':'.join(line.split(':')[1:]).strip()
|
2010-10-17 15:36:26 +02:00
|
|
|
updated = _('updated %s') % s
|
2005-02-02 06:29:31 +01:00
|
|
|
elif not created and any(line.startswith, self._created):
|
|
|
|
s = ':'.join(line.split(':')[1:]).strip()
|
2010-10-17 15:36:26 +02:00
|
|
|
created = _('registered %s') % s
|
2005-02-02 06:29:31 +01:00
|
|
|
elif not expires and any(line.startswith, self._expires):
|
|
|
|
s = ':'.join(line.split(':')[1:]).strip()
|
2010-10-17 15:36:26 +02:00
|
|
|
expires = _('expires %s') % s
|
2005-02-02 06:29:31 +01:00
|
|
|
elif not status and any(line.startswith, self._status):
|
|
|
|
status = ':'.join(line.split(':')[1:]).strip().lower()
|
|
|
|
if not status:
|
|
|
|
status = 'unknown'
|
|
|
|
try:
|
|
|
|
t = telnetlib.Telnet('whois.pir.org', 43)
|
2014-01-20 15:49:15 +01:00
|
|
|
except socket.error as e:
|
2005-02-02 06:29:31 +01:00
|
|
|
irc.error(str(e))
|
|
|
|
return
|
2012-08-04 20:07:24 +02:00
|
|
|
t.write(b'registrar ')
|
|
|
|
t.write(registrar.split('(')[0].strip().encode('ascii'))
|
|
|
|
t.write(b'\n')
|
2005-02-02 06:29:31 +01:00
|
|
|
s = t.read_all()
|
|
|
|
url = ''
|
|
|
|
for line in s.splitlines():
|
2012-08-04 20:07:24 +02:00
|
|
|
line = line.decode('ascii').strip()
|
2005-02-02 06:29:31 +01:00
|
|
|
if not line:
|
|
|
|
continue
|
|
|
|
if line.startswith('Email'):
|
2010-10-17 15:36:26 +02:00
|
|
|
url = _(' <registered at %s>') % line.split('@')[-1]
|
2005-02-02 06:29:31 +01:00
|
|
|
elif line.startswith('Registrar Organization:'):
|
2010-10-17 15:36:26 +02:00
|
|
|
url = _(' <registered by %s>') % line.split(':')[1].strip()
|
2005-02-02 06:29:31 +01:00
|
|
|
elif line == 'Not a valid ID pattern':
|
|
|
|
url = ''
|
2016-08-09 21:08:02 +02:00
|
|
|
if (server or netrange) and status:
|
|
|
|
entity = server or 'Net range %s%s' % \
|
|
|
|
(netrange, ' (%s)' % netname if netname else '')
|
2005-02-02 06:29:31 +01:00
|
|
|
info = filter(None, [status, created, updated, expires])
|
2016-08-09 21:08:02 +02:00
|
|
|
s = format(_('%s%s is %L.'), entity, url, info)
|
2005-02-02 06:29:31 +01:00
|
|
|
irc.reply(s)
|
|
|
|
else:
|
2010-10-17 15:36:26 +02:00
|
|
|
irc.error(_('I couldn\'t find such a domain.'))
|
2005-02-02 06:29:31 +01:00
|
|
|
whois = wrap(whois, ['lowered'])
|
|
|
|
|
2010-10-17 15:36:26 +02:00
|
|
|
@internationalizeDocstring
|
2005-02-02 06:29:31 +01:00
|
|
|
def hexip(self, irc, msg, args, ip):
|
|
|
|
"""<ip>
|
|
|
|
|
|
|
|
Returns the hexadecimal IP for that IP.
|
|
|
|
"""
|
|
|
|
ret = ""
|
2011-06-07 04:29:21 +02:00
|
|
|
if utils.net.isIPV4(ip):
|
|
|
|
quads = ip.split('.')
|
|
|
|
for quad in quads:
|
|
|
|
i = int(quad)
|
|
|
|
ret += '%02X' % i
|
|
|
|
else:
|
|
|
|
octets = ip.split(':')
|
|
|
|
for octet in octets:
|
|
|
|
if octet:
|
|
|
|
i = int(octet, 16)
|
|
|
|
ret += '%04X' % i
|
|
|
|
else:
|
|
|
|
missing = (8 - len(octets)) * 4
|
|
|
|
ret += '0' * missing
|
|
|
|
irc.reply(ret)
|
2005-02-02 06:29:31 +01:00
|
|
|
hexip = wrap(hexip, ['ip'])
|
2010-10-26 09:32:12 +02:00
|
|
|
Internet = internationalizeDocstring(Internet)
|
2005-02-02 06:29:31 +01:00
|
|
|
|
|
|
|
Class = Internet
|
|
|
|
|
|
|
|
|
2006-02-11 16:52:51 +01:00
|
|
|
# vim:set shiftwidth=4 softtabstop=4 expandtab textwidth=79:
|