-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathpatrol.py
executable file
·181 lines (159 loc) · 7.36 KB
/
patrol.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
#!/usr/bin/env python3
import time
from threading import Thread
from utils import config
from utils import logger
from utils.plex import Plex
log = logger.get_root_logger()
server = None
watchlist = []
ip_watchlist = {}
# ip watchlist stuff
def check_stream_count(user):
if user in ip_watchlist:
return len(ip_watchlist[user])
else:
return 0
def add_stream_ip(user, ip):
if user in ip_watchlist:
# does ip exist
if ip in ip_watchlist[user]:
log.debug("Skipping adding ip %s to watchlist for user %s because it is already in there!", ip, user)
return
else:
ip_watchlist[user].append(ip)
else:
ip_watchlist[user] = [ip]
return
# main
def kill_paused_stream(stream, check_again_mins, kick_reason):
log.info("%s will have their stream killed in %d mins, unless it is resumed", stream.user, check_again_mins)
#Tracker for total time spent paused
totalTimePaused = 0
while(1):
streamFound = False
current_streams = server.get_streams()
if current_streams is None:
log.error(
"Unable to check if %s stream is still paused because there was an error retrieving the active streams...",
stream.user)
watchlist.remove(stream.session_id)
return
for current_stream in current_streams:
if current_stream.session_id == stream.session_id:
streamFound = True
if current_stream.state == 'paused':
#Increment total pause time by check interval and proceed to see if u
totalTimePaused += config.CHECK_INTERVAL
if(totalTimePaused >= (check_again_mins * 60)):
if server.kill_stream(stream.session_id, kick_reason):
log.info("Kicked %s because their stream was still paused %d minutes later", stream.user,
check_again_mins)
watchlist.remove(stream.session_id)
return
else:
log.error("Unable to kick the stream of %s, not sure why...", stream.user)
watchlist.remove(stream.session_id)
return
else:
time.sleep(config.CHECK_INTERVAL)
break
else:
log.info("%s stream was resumed, so we wont kill their stream, they're in the clear!", stream.user)
watchlist.remove(stream.session_id)
return
if streamFound == False:
log.info("%s is no longer streaming...", stream.user)
watchlist.remove(stream.session_id)
return
def should_kick_stream(stream):
# is stream using a blacklisted client
for client in config.KICK_CLIENT_PLAYERS:
if client.lower() in stream.player.lower():
return True, 0, config.KICK_PLAYER_MESSAGE
# is this user already streaming from more than allowed ips?
if config.KICK_MULTIPLE_IP and config.KICK_MULTIPLE_IP_MAX:
add_stream_ip(stream.user, stream.ip)
if check_stream_count(stream.user) > config.KICK_MULTIPLE_IP_MAX:
return True, 0, config.KICK_MULTI_IP_MESSAGE
if stream.type == 'transcode':
# stream is transcode - check specifics
if stream.audio_decision == 'transcode' and config.KICK_AUDIO_TRANSCODES:
return True, 0, config.KICK_TRANSCODE_MESSAGE
elif stream.video_decision == 'transcode' and config.KICK_VIDEO_TRANSCODES:
return True, 0, config.KICK_TRANSCODE_MESSAGE
if stream.state == 'paused' and config.KICK_PAUSED_TRANSCODES:
return True, config.KICK_PAUSED_GRACE_MINS, config.KICK_PAUSED_MESSAGE
if stream.quality == '4K' and config.KICK_4K_TRANSCODE and stream.video_decision == 'transcode':
return True, 0, config.KICK_4K_TRANSCODE_MESSAGE
else:
# stream is directplay - check specifics
if stream.state == 'paused' and config.KICK_PAUSED_DIRECTPLAY:
return True, config.KICK_PAUSED_GRACE_MINS, config.KICK_PAUSED_MESSAGE
return False, 0, None
def check_streams():
log.debug("Retrieving active stream(s) for server: %s", server.name)
streams = server.get_streams()
if streams is None:
log.error("There was an error while retrieving the active streams...")
return
elif not streams:
log.debug("There's currently no streams to check")
return
else:
log.debug("Checking %d stream(s)", len(streams))
# clear ip watchlist before checking streams
ip_watchlist.clear()
# iterate streams and check / kick
for stream in streams:
if stream.user in config.WHITELISTED_USERS:
log.debug("Skipping whitelisted user: %s", stream.user)
continue
else:
log.info("Checking stream: %s", stream)
kick_stream, kick_mins, kick_msg = should_kick_stream(stream)
if kick_stream:
if not kick_mins:
# kick instantly....
if server.kill_stream(stream.session_id, kick_msg):
log.info("Kicked %s instantly, without a second thought", stream.user)
else:
log.error("Unable to kick the stream of %s, not sure why...", stream.user)
else:
# kick in X mins unless resumed
# is session already being watched?
if stream.session_id in watchlist:
log.info("%s stream is already on the watchlist, skipping..", stream.user)
else:
Thread(target=kill_paused_stream, args=(stream, kick_mins, kick_msg)).start()
watchlist.append(stream.session_id)
log.debug("Finished checking streams")
if __name__ == "__main__":
log.info("""
_ _ _
_ __ | | _____ __ _ __ __ _| |_ _ __ ___ | |
| '_ \| |/ _ \ \/ / | '_ \ / _` | __| '__/ _ \| |
| |_) | | __/> < | |_) | (_| | |_| | | (_) | |
| .__/|_|\___/_/\_\ | .__/ \__,_|\__|_| \___/|_|
|_| |_|
#########################################################################
# Author: l3uddz #
# URL: https://github.com/l3uddz/plex_patrol #
# -- #
# Part of the Cloudbox project: https://cloudbox.rocks #
#########################################################################
# GNU General Public License v3.0 #
#########################################################################
""")
log.info("Initializing")
log.info("Validating server %r with token %r", config.SERVER_URL, config.SERVER_TOKEN)
server = Plex(config.SERVER_NAME, config.SERVER_URL, config.SERVER_TOKEN)
if not server.validate():
log.error("Could not validate server token, are you sure its correct...")
exit(1)
else:
log.info("Server token was validated, proceeding to uphold the law!")
while True:
log.debug("Checking streams in %d seconds", config.CHECK_INTERVAL)
time.sleep(config.CHECK_INTERVAL)
check_streams()