2019-06-21 06:55:11 +00:00
|
|
|
#!/usr/bin/env python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
"""
|
|
|
|
Pulling together User IP information and Email.
|
|
|
|
Enable the API under Settings > Access Control and remember your API key.
|
2018-03-19 16:33:44 +00:00
|
|
|
Shutdown Tautulli and open your config.ini file in a text editor.
|
2017-07-12 18:18:50 +00:00
|
|
|
Set api_sql = 1 in the config file.
|
2018-03-19 16:33:44 +00:00
|
|
|
Restart Tautulli.
|
2017-07-12 18:18:50 +00:00
|
|
|
Place in Playback Start
|
|
|
|
"""
|
2020-07-04 20:08:59 +00:00
|
|
|
from __future__ import print_function
|
2020-07-04 20:31:02 +00:00
|
|
|
from __future__ import unicode_literals
|
2020-07-04 20:23:47 +00:00
|
|
|
from builtins import object
|
2017-07-12 18:18:50 +00:00
|
|
|
import argparse
|
|
|
|
import requests
|
|
|
|
import sys
|
|
|
|
from email.mime.text import MIMEText
|
|
|
|
import email.utils
|
|
|
|
import smtplib
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
# ## -sn {show_name} -ena {episode_name} -ssn {season_num00} -enu {episode_num00} -srv {server_name} -med {media_type} -pos {poster_url} -tt {title} -sum {summary} -lbn {library_name} -ip {ip_address} -us {user} -uid {user_id} -pf {platform} -pl {player} -da {datestamp} -ti {timestamp}
|
2017-07-12 18:18:50 +00:00
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
# ## EDIT THESE SETTINGS ##
|
2018-03-19 16:33:44 +00:00
|
|
|
TAUTULLI_APIKEY = 'xxxxxxxxxxx' # Your Tautulli API key
|
|
|
|
TAUTULLI_URL = 'http://localhost:8181/' # Your Tautulli URL
|
2017-07-12 18:18:50 +00:00
|
|
|
|
|
|
|
# Replace LAN IP addresses that start with the LAN_SUBNET with a WAN IP address
|
|
|
|
# to retrieve geolocation data. Leave REPLACEMENT_WAN_IP blank for no replacement.
|
|
|
|
LAN_SUBNET = ('10.201', '127.0.0')
|
|
|
|
REPLACEMENT_WAN_IP = ''
|
|
|
|
|
|
|
|
# The notification subject and body
|
|
|
|
# - Use "{p.argument}" for script arguments
|
|
|
|
# - Use "{g.value}" for geolocation data
|
|
|
|
# - Use "{u.value}" for user data
|
|
|
|
SUBJECT_TEXT = "New IP has been detected using Plex."
|
|
|
|
BODY_TEXT = """\
|
|
|
|
<html>
|
|
|
|
<head></head>
|
|
|
|
<body>
|
|
|
|
<p>Hi!<br>
|
|
|
|
<br>{p.user}, someone has accessed your account.<br>
|
|
|
|
<br>They watched {p.media_type}:{p.title} from a new IP address: {p.ip_address}<br>
|
|
|
|
<br>On {p.platform}[{p.player}] in <a href="http://maps.google.com/?q={g.city},{g.country},{g.postal_code}">{g.city}, {g.country} {g.postal_code}</a> at {p.timestamp} on {p.datestamp}<br>
|
|
|
|
<br><br>
|
|
|
|
</p>
|
|
|
|
</body>
|
|
|
|
</html>
|
|
|
|
"""
|
|
|
|
|
|
|
|
# Email settings
|
2019-06-21 06:55:11 +00:00
|
|
|
name = '' # Your name
|
|
|
|
sender = '' # From email address
|
|
|
|
email_server = 'smtp.gmail.com' # Email server (Gmail: smtp.gmail.com)
|
2017-07-12 18:18:50 +00:00
|
|
|
email_port = 587 # Email port (Gmail: 587)
|
2019-06-21 06:55:11 +00:00
|
|
|
email_username = '' # Your email username
|
|
|
|
email_password = '' # Your email password
|
2017-07-12 18:18:50 +00:00
|
|
|
email_subject = "New IP has been detected using Plex."
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
IGNORE_LST = ['123456', '123456'] # User_id
|
|
|
|
|
|
|
|
|
|
|
|
# ##Geo Space##
|
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
|
|
|
|
class GeoData(object):
|
|
|
|
def __init__(self, data=None):
|
|
|
|
data = data or {}
|
|
|
|
self.country = data.get('country', 'N/A')
|
|
|
|
self.city = data.get('city', 'N/A')
|
|
|
|
self.postal_code = data.get('postal_code', 'N/A')
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
|
|
|
|
# ##USER Space##
|
|
|
|
|
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
class UserEmail(object):
|
|
|
|
def __init__(self, data=None):
|
|
|
|
data = data or {}
|
|
|
|
self.email = data.get('email', 'N/A')
|
|
|
|
self.user_id = data.get('user_id', 'N/A')
|
|
|
|
self.user_thumb = data.get('user_thumb', 'N/A')
|
2019-06-21 06:55:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
# ##API Space##
|
|
|
|
|
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
def get_user_ip_addresses(user_id='', ip_address=''):
|
2018-03-19 16:33:44 +00:00
|
|
|
# Get the user IP list from Tautulli
|
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-07-12 18:18:50 +00:00
|
|
|
'cmd': 'get_user_ips',
|
|
|
|
'user_id': user_id,
|
|
|
|
'search': ip_address}
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
r = requests.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-07-12 18:18:50 +00:00
|
|
|
response = r.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
|
|
|
data = response['response']['data']
|
|
|
|
if data.get('error'):
|
|
|
|
raise Exception(data['error'])
|
|
|
|
else:
|
|
|
|
sys.stdout.write("Successfully retrieved UserIPs data.")
|
|
|
|
if response['response']['data']['recordsFiltered'] == 0:
|
|
|
|
sys.stdout.write("IP has no history.")
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
sys.stdout.write("IP has history, killing script.")
|
|
|
|
exit()
|
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'get_user_ip_addresses' request failed: {0}.".format(e))
|
2017-07-12 18:18:50 +00:00
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
def get_geoip_info(ip_address=''):
|
2018-03-19 16:33:44 +00:00
|
|
|
# Get the geo IP lookup from Tautulli
|
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-07-12 18:18:50 +00:00
|
|
|
'cmd': 'get_geoip_lookup',
|
|
|
|
'ip_address': ip_address}
|
|
|
|
|
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
r = requests.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-07-12 18:18:50 +00:00
|
|
|
response = r.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
|
|
|
data = response['response']['data']
|
|
|
|
if data.get('error'):
|
|
|
|
raise Exception(data['error'])
|
|
|
|
else:
|
|
|
|
sys.stdout.write("Successfully retrieved geolocation data.")
|
|
|
|
return GeoData(data=data)
|
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'get_geoip_lookup' request failed: {0}.".format(e))
|
2017-07-12 18:18:50 +00:00
|
|
|
return GeoData()
|
|
|
|
|
|
|
|
|
|
|
|
def get_user_email(user_id=''):
|
2018-03-19 16:33:44 +00:00
|
|
|
# Get the user email from Tautulli
|
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-07-12 18:18:50 +00:00
|
|
|
'cmd': 'get_user',
|
|
|
|
'user_id': user_id}
|
|
|
|
|
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
r = requests.get(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-07-12 18:18:50 +00:00
|
|
|
response = r.json()
|
|
|
|
|
|
|
|
if response['response']['result'] == 'success':
|
|
|
|
data = response['response']['data']
|
|
|
|
if data.get('error'):
|
|
|
|
raise Exception(data['error'])
|
|
|
|
else:
|
|
|
|
sys.stdout.write("Successfully retrieved user email data.")
|
|
|
|
return UserEmail(data=data)
|
|
|
|
else:
|
|
|
|
raise Exception(response['response']['message'])
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'get_user' request failed: {0}.".format(e))
|
2017-07-12 18:18:50 +00:00
|
|
|
return UserEmail()
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
def send_notification(arguments=None, geodata=None, useremail=None):
|
|
|
|
# Format notification text
|
|
|
|
try:
|
|
|
|
email_subject = SUBJECT_TEXT.format(p=arguments, g=geodata, u=useremail)
|
|
|
|
body = BODY_TEXT.format(p=arguments, g=geodata, u=useremail)
|
|
|
|
|
|
|
|
message = MIMEText(body, 'html')
|
|
|
|
message['Subject'] = email_subject
|
|
|
|
message['From'] = email.utils.formataddr((name, sender))
|
|
|
|
|
|
|
|
mailserver = smtplib.SMTP(email_server, email_port)
|
|
|
|
mailserver.ehlo()
|
|
|
|
mailserver.starttls()
|
|
|
|
mailserver.ehlo()
|
|
|
|
mailserver.login(email_username, email_password)
|
|
|
|
mailserver.sendmail(sender, u.email, message.as_string())
|
|
|
|
mailserver.quit()
|
2019-06-21 06:55:11 +00:00
|
|
|
print('Email sent')
|
2017-07-12 18:18:50 +00:00
|
|
|
except Exception as e:
|
|
|
|
sys.stderr.write("Email Failure: {0}.".format(e))
|
|
|
|
|
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
def clr_sql(ip):
|
2017-07-12 18:18:50 +00:00
|
|
|
try:
|
2018-03-19 16:33:44 +00:00
|
|
|
payload = {'apikey': TAUTULLI_APIKEY,
|
2017-07-12 18:18:50 +00:00
|
|
|
'cmd': 'sql',
|
|
|
|
'query': 'DELETE FROM session_history WHERE ip_address = "' + ip + '";'}
|
|
|
|
|
2018-03-19 16:33:44 +00:00
|
|
|
requests.post(TAUTULLI_URL.rstrip('/') + '/api/v2', params=payload)
|
2017-07-12 18:18:50 +00:00
|
|
|
|
|
|
|
except Exception as e:
|
2018-03-19 16:33:44 +00:00
|
|
|
sys.stderr.write("Tautulli API 'get_sql' request failed: {0}.".format(e))
|
2017-07-12 18:18:50 +00:00
|
|
|
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
if __name__ == '__main__':
|
2018-03-19 16:33:44 +00:00
|
|
|
# Parse arguments from Tautulli
|
2017-07-12 18:18:50 +00:00
|
|
|
parser = argparse.ArgumentParser()
|
|
|
|
|
|
|
|
parser.add_argument('-ip', '--ip_address', action='store', default='',
|
|
|
|
help='The IP address of the stream')
|
|
|
|
parser.add_argument('-us', '--user', action='store', default='',
|
|
|
|
help='Username of the person watching the stream')
|
|
|
|
parser.add_argument('-uid', '--user_id', action='store', default='',
|
2019-06-21 06:55:11 +00:00
|
|
|
help='User_ID of the person watching the stream')
|
2017-07-12 18:18:50 +00:00
|
|
|
parser.add_argument('-med', '--media_type', action='store', default='',
|
|
|
|
help='The media type of the stream')
|
|
|
|
parser.add_argument('-tt', '--title', action='store', default='',
|
|
|
|
help='The title of the media')
|
|
|
|
parser.add_argument('-pf', '--platform', action='store', default='',
|
|
|
|
help='The platform of the stream')
|
|
|
|
parser.add_argument('-pl', '--player', action='store', default='',
|
|
|
|
help='The player of the stream')
|
|
|
|
parser.add_argument('-da', '--datestamp', action='store', default='',
|
|
|
|
help='The date of the stream')
|
|
|
|
parser.add_argument('-ti', '--timestamp', action='store', default='',
|
|
|
|
help='The time of the stream')
|
|
|
|
parser.add_argument('-sn', '--show_name', action='store', default='',
|
|
|
|
help='The name of the TV show')
|
|
|
|
parser.add_argument('-ena', '--episode_name', action='store', default='',
|
|
|
|
help='The name of the episode')
|
|
|
|
parser.add_argument('-ssn', '--season_num', action='store', default='',
|
|
|
|
help='The season number of the TV show')
|
|
|
|
parser.add_argument('-enu', '--episode_num', action='store', default='',
|
|
|
|
help='The episode number of the TV show')
|
|
|
|
parser.add_argument('-srv', '--plex_server', action='store', default='',
|
|
|
|
help='The name of the Plex server')
|
|
|
|
parser.add_argument('-pos', '--poster', action='store', default='',
|
|
|
|
help='The poster url')
|
|
|
|
parser.add_argument('-sum', '--summary', action='store', default='',
|
|
|
|
help='The summary of the TV show')
|
|
|
|
parser.add_argument('-lbn', '--library_name', action='store', default='',
|
|
|
|
help='The name of the TV show')
|
2019-06-21 06:55:11 +00:00
|
|
|
|
2017-07-12 18:18:50 +00:00
|
|
|
p = parser.parse_args()
|
|
|
|
|
|
|
|
if p.user_id not in IGNORE_LST:
|
|
|
|
|
|
|
|
get_user_ip_addresses(user_id=p.user_id, ip_address=p.ip_address)
|
|
|
|
g = get_geoip_info(ip_address=p.ip_address)
|
|
|
|
u = get_user_email(user_id=p.user_id)
|
|
|
|
|
|
|
|
send_notification(arguments=p, geodata=g, useremail=u)
|
|
|
|
|
|
|
|
clr_sql(p.ip_address)
|
|
|
|
|
|
|
|
else:
|
|
|
|
sys.stdout.write("User_id is ignored.")
|