Large Overhaul - Jellyfin Temp Accounts
Temporary jellyfin accounts can now be made through messaging. Commands were moved out and into their own files and functions for organization.
This commit is contained in:
parent
bdc5d1fece
commit
237aec245e
@ -1,3 +1,5 @@
|
||||
__pycache__
|
||||
.DS_Store
|
||||
docker-compose.yaml
|
||||
docker-compose.yaml
|
||||
config.ini
|
||||
movies.db
|
4
.gitignore
vendored
4
.gitignore
vendored
@ -1,2 +1,4 @@
|
||||
__pycache__
|
||||
.DS_Store
|
||||
.DS_Store
|
||||
config.ini
|
||||
movies.db
|
@ -7,5 +7,7 @@ WORKDIR /
|
||||
COPY . .
|
||||
RUN pip install -r requirements.txt
|
||||
|
||||
ENV IN_DOCKER Yes
|
||||
|
||||
ENTRYPOINT [ "python" ]
|
||||
CMD [ "-u", "app/wsgi.py" ]
|
60
README.md
60
README.md
@ -1,62 +1,4 @@
|
||||
# Messagearr
|
||||
### Add movies to your Radarr library and check their download status through text messages
|
||||
|
||||
### 1. Docker Compose
|
||||
```
|
||||
version: '3.3'
|
||||
services:
|
||||
messagearr:
|
||||
ports:
|
||||
- '4545:4545'
|
||||
environment:
|
||||
- TZ=America/Chicago # OPTIONAL: Default is UTC
|
||||
- ENABLE_KUMA_NOTIFICATIONS=false # Whether or not to setup UptimeKuma SMS notifications
|
||||
- RADARR_HOST_URL=http://127.0.0.1:7878 # Change to your radarr host
|
||||
- RADARR_API_KEY=apikeyhere # Found by navigating to Settings > General
|
||||
- SMS_SERVICE=servicename # Currently only supporting Telnyx
|
||||
volumes:
|
||||
- /local/file/path:/data
|
||||
image: packetparker/messagearr:latest
|
||||
```
|
||||
### 2. Run the Container
|
||||
#### Run the container in non-daemon mode, you will get an error stating that there are variables that need to be set within the `config.yaml` file, this file is in the internal path of `/data/config.yaml`. Go into that file and being setting the variables that appear for you within that file. Note that every single entry must have a value, the only optional value is `home_domain` but requires `null` at the very least (this is also the default).
|
||||
|
||||
### 3. What Does Every Value in config.yaml Mean?
|
||||
- #### `IMPORTANT NOTE` ALL values that contain phone numbers should be placed within single quotation marks (e.g. '+18005269856, +18005247852, +18002365874').
|
||||
|
||||
- #### `quality_profile_id` There is a commented list of profiles that exist within your Radarr server. The list contains the profile id followed by the profile name. This value should be the single integer id of the quality profile that you would like movies to be added under.
|
||||
|
||||
- #### `home_domain` Defaults to null meaning 404/405 errors will just return the error to the browser. Replace this will a full URL if you would like those errors to be redirected to your website (e.g. https://pkrm.dev)
|
||||
|
||||
- #### `api_number` The number given to you by your SMS service, in international format, and in (e.g. '+18005282589')
|
||||
|
||||
- #### `valid_senders` Comma-space separated list of numbers that are allowed to send commands, this stops random numbers from being allowed to add movies to Radarr. This also must be in international format (e.g. '+18005269856, +18005247852, +18002365874').
|
||||
|
||||
- #### `root_folder_path` The folder path defined in your Radarr server. Find this value by logging into your Radarr server and navigate to Settings > Media Management, at the bottom of this page will be your root folder path (copy the full path!)
|
||||
|
||||
- #### `notif_receivers` and `authorization_header_token` Only appear if you have `ENABLE_KUMA_NOTIFICATIONS` set to true. `notif_receivers` is a list of phone numbers (e.g. '+18005269856, +18005247852, +18002365874') that will receive the notifications on service statuses. `authorization_header_token` other things that find the /kuma route cant send POST requests there - the value can be anything you choose, the default is `uptimekumaauthtoken`. For help on setting up UptimeKuma, see `step 6`.
|
||||
|
||||
- #### The last values will vary dependant on your SMS service, but they are just the authentication values for your service.
|
||||
|
||||
### 4. Setup your Domain
|
||||
#### It is recommended but not technically required to have a domain in order for this to work. This container runs a flask image in order to accept POST requests and must be open to the internet so that incoming messages can be accepted. The process for this is different for every reverse proxy and I recommend that you refer to your proxies documentation for help. Once you have correctly proxied the domain you can test it by navigating to the domain, you should recieve a 404 error or be redirect to `home_domain` if the value has been set.
|
||||
|
||||
#### If you do not have a domain you can use a DDNS service or you can open port 4545 (or whatever exposed port you used) on your router.
|
||||
|
||||
### 5. Add the Domain
|
||||
#### Once you have configured your domain or router go to your sms services console and buy a number for SMS messaging. Once you buy the number you need to configure how it handles incoming messages, there you should be able to have the service send a POST request to a webhook for incoming message. You should overwrite or set this value to `http(s)://yourdomain.com/incoming` or `http://yourip:port/incoming`
|
||||
|
||||
### 6. UptimeKuma Setup
|
||||
#### If you chose to enable UptimeKuma notifications and have already setup the container, please continue - otherwise, set the values in the config first and make the container is working. First go to your UptimeKuma dashboard, go to Settings > Notifications, then click `Setup Notification`. For notification type, choose `Webhook`, give it whatever name, `Post URL` should be set to `http(s)://ip:port/kuma` or `http(s)yourdomain.com/kuma`. Finally check the `Additional Headers` switch and paste in this -
|
||||
```
|
||||
{
|
||||
"Authorization": "YOURAUTHTOKENVALUEHERE"
|
||||
}
|
||||
```
|
||||
#### Finally, choose whether or not to have the notification be enabled by default or whether to apply this notification to all current monitors.
|
||||
|
||||
### 7. Further Help
|
||||
#### Please open an issue if you need help with any part of setting this up (I know the docs/instructions aren't great). You can also email me at [contact@pkrm.dev](mailto:contact@pkrm.dev)
|
||||
|
||||
|
||||
#### Happy coding!
|
||||
### Recently overhauled - new documentation coming soon!
|
72
app/commands/movie_show_response_newaccount.py
Normal file
72
app/commands/movie_show_response_newaccount.py
Normal file
@ -0,0 +1,72 @@
|
||||
import datetime
|
||||
import requests
|
||||
import random
|
||||
import string
|
||||
import sqlite3
|
||||
|
||||
import initialize_variables
|
||||
from create_message import create_message
|
||||
|
||||
|
||||
def movie_show_response_newaccount(from_number, message):
|
||||
if from_number not in initialize_variables.temp_new_account_requests.keys():
|
||||
create_message(from_number, "There is no current request that you can decide on. It might be that your /newaccount command timed out due since you took too long to response. Please try again. If this issue persists, please contact Parker.")
|
||||
return
|
||||
|
||||
# If its been 5 minutes since prompt was sent, alert user of timed out request
|
||||
if (datetime.datetime.now() - initialize_variables.temp_new_account_requests[from_number]).total_seconds() / 60 > 5:
|
||||
del initialize_variables.temp_new_account_requests[from_number]
|
||||
create_message(from_number, "You waited too long and therefore your request has timed out.\n\nPlease try again by re-running the /newaccount command. If this issue persists, please contact Parker.")
|
||||
return
|
||||
|
||||
if message.strip().lower() == "show":
|
||||
active_time = 24
|
||||
|
||||
elif message.strip().lower() == "movie":
|
||||
active_time = 4
|
||||
|
||||
else:
|
||||
create_message(from_number, "You did not enter a valid response. Please re-send the /newaccount command and try again. If you believe this is an error, please contact Parker.")
|
||||
return
|
||||
|
||||
# Otherwise, all checks have been completed
|
||||
username = ''.join(random.choices(string.ascii_lowercase + string.digits, k=5))
|
||||
password = ''.join(random.choices(string.ascii_lowercase + string.digits, k=15))
|
||||
|
||||
deletion_time = datetime.datetime.now() + datetime.timedelta(hours=active_time)
|
||||
# Create new Jellyfin account
|
||||
request_1 = requests.post(f'{initialize_variables.jellyfin_url}/Users/New', headers=initialize_variables.jellyfin_headers, json={'Name': username, 'Password': password})
|
||||
if request_1.status_code != 200:
|
||||
create_message(from_number, "Error creating Jellyfin account. Please try again. If the error persists, contact Parker.")
|
||||
return
|
||||
|
||||
user_id = request_1.json()['Id']
|
||||
# Get account policy and make edits
|
||||
request_2 = requests.get(f'{initialize_variables.jellyfin_url}/Users/{user_id}', headers=initialize_variables.jellyfin_headers)
|
||||
if request_2.status_code != 200:
|
||||
create_message(from_number, "Error creating Jellyfin account. Please try again. If the error persists, contact Parker.")
|
||||
return
|
||||
|
||||
policy = request_2.json()['Policy']
|
||||
policy['SyncPlayAccess'] = 'JoinGroups'
|
||||
policy['EnableContentDownloading'] = False
|
||||
policy['InvalidLoginAttemptCount'] = 3
|
||||
policy['MaxActiveSessions'] = 1
|
||||
# Update user with new policy
|
||||
request_3 = requests.post(f'{initialize_variables.jellyfin_url}/Users/{user_id}/Policy', headers=initialize_variables.jellyfin_headers, json=policy)
|
||||
if request_3.status_code != 204:
|
||||
create_message(from_number, "Error creating Jellyfin account. Please try again. If the error persists, contact Parker.")
|
||||
return
|
||||
|
||||
# Add information to the database
|
||||
db = sqlite3.connect(initialize_variables.db_path)
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
INSERT INTO jellyfin_accounts (user_id, deletion_time)
|
||||
VALUES(?, ?)
|
||||
''', (user_id, deletion_time))
|
||||
db.commit()
|
||||
db.close()
|
||||
|
||||
create_message(from_number, f"Username: {username}\nPassword: {password}\n\nYour account will expire in {active_time} hours.")
|
||||
return
|
59
app/commands/number_response_request.py
Normal file
59
app/commands/number_response_request.py
Normal file
@ -0,0 +1,59 @@
|
||||
import datetime
|
||||
import requests
|
||||
import sqlite3
|
||||
|
||||
import initialize_variables
|
||||
from create_message import create_message
|
||||
|
||||
|
||||
def number_response_request(from_number, message):
|
||||
if from_number not in initialize_variables.temp_movie_ids.keys():
|
||||
create_message(from_number, "There is no current request that you can decide on. It might be that your /request command timed out due since you took too long to response. Please try again. If this issue persists, please contact Parker.")
|
||||
return
|
||||
|
||||
# If its been 5 minutes since prompt was sent, alert user of timed out request
|
||||
if (datetime.datetime.now() - initialize_variables.temp_movie_ids[from_number]['time']).total_seconds() / 60 > 5:
|
||||
del initialize_variables.temp_movie_ids[from_number]
|
||||
create_message(from_number, "You waited too long and therefore your request has timed out.\n\nPlease try again by re-running the /request command. If this issue persists, please contact Parker.")
|
||||
return
|
||||
|
||||
# Otherwise, all checks have been completed
|
||||
create_message(from_number, "Just a moment while I add your movie to the library...")
|
||||
movie_number = initialize_variables.numbers_responses[message.strip()]
|
||||
try:
|
||||
tmdb_id = initialize_variables.temp_movie_ids[from_number]['ids'][movie_number - 1]
|
||||
except IndexError:
|
||||
create_message(from_number, "You did not enter a valid number. Please re-send the /request command and try again. If you believe this is an error, please contact Parker.")
|
||||
del initialize_variables.temp_movie_ids[from_number]
|
||||
return
|
||||
|
||||
data = requests.get(f'{initialize_variables.radarr_host_url}/api/v3/movie/lookup/tmdb?tmdbId={tmdb_id}', headers=initialize_variables.headers)
|
||||
|
||||
data = data.json()
|
||||
movie_title = data['title']
|
||||
# Change the qualityProfileId, monitored, and rootFolderPath values
|
||||
data['qualityProfileId'] = initialize_variables.quality_profile_id
|
||||
data['monitored'] = True
|
||||
data['rootFolderPath'] = initialize_variables.root_folder_path
|
||||
# Send data to Radarr API
|
||||
response = requests.post(f'{initialize_variables.radarr_host_url}/api/v3/movie', headers=initialize_variables.headers, json=data)
|
||||
data = response.json()
|
||||
movie_id = data['id']
|
||||
# Send message to user alerting them that the movie was added to the library
|
||||
create_message(from_number, f"🎉 {data['title']} has been added to the library!\n\nTo check up on the status of your movie(s) send /status - please wait at least 5 minutes before running this command in order to get an accurate time.")
|
||||
|
||||
# After everything is completed, send Radarr a request to search indexers for new movie
|
||||
requests.post(f'{initialize_variables.radarr_host_url}/api/v3/command', headers=initialize_variables.headers, json={'name': 'MoviesSearch', 'movieIds': [int(movie_id)]})
|
||||
|
||||
# Add the movie_id to the database so that users can check up on the status of their movie
|
||||
db = sqlite3.connect(initialize_variables.db_path)
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
INSERT INTO movies(from_number, movie_id, movie_title)
|
||||
VALUES(?, ?, ?)
|
||||
''', (from_number, movie_id, movie_title))
|
||||
db.commit()
|
||||
db.close()
|
||||
|
||||
del initialize_variables.temp_movie_ids[from_number]
|
||||
return
|
49
app/commands/request.py
Normal file
49
app/commands/request.py
Normal file
@ -0,0 +1,49 @@
|
||||
import requests
|
||||
import datetime
|
||||
|
||||
import initialize_variables
|
||||
from create_message import create_message
|
||||
|
||||
|
||||
def request(from_number, message):
|
||||
# If the user has already run the /request command, delete the entry
|
||||
# from the temp_movie_ids dict so that they can run the command again
|
||||
if from_number in initialize_variables.temp_movie_ids.keys():
|
||||
del initialize_variables.temp_movie_ids[from_number]
|
||||
|
||||
# If the user did not include a movie title, alert them to do so
|
||||
if len(message) <= 9:
|
||||
create_message(from_number, "Please include the movie title after the /request command.\nEX: /request The Dark Knight")
|
||||
return
|
||||
|
||||
incoming_message = message.split(' ', 1)[1]
|
||||
movie_request = incoming_message.replace(' ', '%20')
|
||||
if movie_request.endswith("%20"):
|
||||
movie_request = movie_request[:-3]
|
||||
|
||||
# Send a request to the radarr API to get the movie info
|
||||
response = requests.get(f'{initialize_variables.radarr_host_url}/api/v3/movie/lookup?term={movie_request}', headers=initialize_variables.headers)
|
||||
|
||||
if len(response.json()) == 0:
|
||||
create_message(from_number, "There were no results for that movie. Please make sure you typed the title correctly.")
|
||||
return
|
||||
# If the movie is already added to the library, return a message saying so.
|
||||
if response.json()[0]['added'] != '0001-01-01T05:51:00Z':
|
||||
create_message(from_number, "This movie is already added to the server.\n\nIf you believe this is an error, please contact Parker.")
|
||||
return
|
||||
|
||||
# Add top 3 results to a message
|
||||
message = ""
|
||||
for i in range(min(3, len(response.json()))):
|
||||
message += f"{i+1}. {response.json()[i]['folder']}\n\n"
|
||||
if from_number not in initialize_variables.temp_movie_ids.keys():
|
||||
initialize_variables.temp_movie_ids[from_number] = {
|
||||
'ids': [],
|
||||
'time': datetime.datetime.now()
|
||||
}
|
||||
initialize_variables.temp_movie_ids[from_number]['ids'].append(response.json()[i]['tmdbId'])
|
||||
|
||||
message += "Reply with the number associated with the movie you want to download. EX: 1\n\nIf the movie you want is not on the list, make sure you typed the title exactly as it is spelt, or ask Parker to manually add the movie."
|
||||
|
||||
create_message(from_number, message)
|
||||
return
|
76
app/commands/status.py
Normal file
76
app/commands/status.py
Normal file
@ -0,0 +1,76 @@
|
||||
import datetime
|
||||
import sqlite3
|
||||
import requests
|
||||
import humanize
|
||||
|
||||
import initialize_variables
|
||||
from create_message import create_message
|
||||
|
||||
|
||||
def status(from_number):
|
||||
# This returns a list of ALL movies being downloaded, but not all of them were
|
||||
# requested by the user, so we need to filter out the ones that were not requested
|
||||
response = requests.get(f'{initialize_variables.radarr_host_url}/api/v3/queue/', headers=initialize_variables.headers)
|
||||
# Get all the movie_ids that were requested by the user
|
||||
db = sqlite3.connect(initialize_variables.db_path)
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
SELECT movie_id, movie_title FROM movies WHERE from_number = ?
|
||||
''', (from_number,))
|
||||
movie_info = cursor.fetchall()
|
||||
db.close()
|
||||
|
||||
movies = {} # movie_id: movie_title
|
||||
for movie in movie_info:
|
||||
movies[movie[0]] = movie[1]
|
||||
|
||||
if len(movies) == 0:
|
||||
create_message(from_number, "You have no movies being downloaded at the moment.\n\nIf you previously added a movie, it is likely that it has finished downloading. If you believe this is an error, please contact Parker.")
|
||||
return
|
||||
|
||||
message = ""
|
||||
# Loop through the response from the radarr API and filter out the movies that were not requested by the user
|
||||
for movie in response.json()['records']:
|
||||
movie_id = str(movie['movieId'])
|
||||
if movie_id in movies.keys():
|
||||
if movie['status'] == 'downloading':
|
||||
# Humanize the time_left value
|
||||
try:
|
||||
time_left = humanize.precisedelta(datetime.datetime.strptime(movie['timeleft'], '%H:%M:%S') - datetime.datetime.strptime('00:00:00', '%H:%M:%S'), minimum_unit='seconds')
|
||||
except ValueError:
|
||||
# Sometimes movie downloads take a long time and include days in the time_left value
|
||||
# This is formated as 1.00:00:00
|
||||
time_left = humanize.precisedelta(datetime.datetime.strptime(movie['timeleft'], '%d.%H:%M:%S') - datetime.datetime.strptime('00:00:00', '%H:%M:%S'), minimum_unit='seconds')
|
||||
except KeyError:
|
||||
time_left = 'Unknown'
|
||||
|
||||
message += f"📥 {movies[movie_id]} - {time_left}\n"
|
||||
else:
|
||||
message += f"{movies[movie_id]} - {str(movie['status']).upper()}\n"
|
||||
|
||||
# If the message is empty, that means the user has no movies being downloaded
|
||||
# Or, no download was found for the movie they requested
|
||||
if message == "":
|
||||
# For all movie IDs within the database
|
||||
for movie_id in movies.keys():
|
||||
response = requests.get(f'{initialize_variables.radarr_host_url}/api/v3/movie/{movie_id}', headers=initialize_variables.headers)
|
||||
# This means that there is no current download, and no file has been found
|
||||
# MOST likely means a download just wasn't found, so alert the user
|
||||
data = response.json()
|
||||
if data['hasFile'] == False:
|
||||
message += f"{movies[movie_id]} - NOT FOUND\n\nThis means a download was not found for the movie(s), if this is a brand new movie that is likely the reason. If the movie has already been released on DVD/Blu-Ray, please contact Parker."
|
||||
|
||||
# Send message with info about download to user, otherwise, the user has
|
||||
# no movies being downloaded at the moment so alert them
|
||||
if message != "":
|
||||
create_message(from_number, message)
|
||||
return
|
||||
else:
|
||||
create_message(from_number, "You have no movies being downloaded at the moment.\n\nIf you previously added a movie, it is likely that it has finished downloading. If you believe this is an error, please contact Parker.")
|
||||
return
|
||||
|
||||
# Otherwise, add another part to the message containing movie data
|
||||
else:
|
||||
message += "\n\nIf movies consistently show as 'WARNING' or 'QUEUED' or any other error over multiple hours, please contact Parker."
|
||||
create_message(from_number, message)
|
||||
return
|
@ -1,21 +1,21 @@
|
||||
import telnyx
|
||||
from twilio.rest import Client
|
||||
|
||||
from initialize_variables import *
|
||||
import initialize_variables
|
||||
|
||||
def create_message(number, message):
|
||||
if sms_service == 'telnyx':
|
||||
telnyx.api_key = telnyx_api_key
|
||||
if initialize_variables.sms_service == 'telnyx':
|
||||
telnyx.api_key = initialize_variables.telnyx_api_key
|
||||
telnyx.Message.create(
|
||||
from_=api_number,
|
||||
from_=initialize_variables.api_number,
|
||||
to=number,
|
||||
text=message
|
||||
)
|
||||
|
||||
if sms_service == 'twilio':
|
||||
client = Client(twilio_account_sid, twilio_auth_token)
|
||||
if initialize_variables.sms_service == 'twilio':
|
||||
client = Client(initialize_variables.twilio_account_sid, initialize_variables.twilio_auth_token)
|
||||
client.messages.create(
|
||||
body=message,
|
||||
from_=api_number,
|
||||
from_=initialize_variables.api_number,
|
||||
to=number
|
||||
)
|
||||
|
@ -1,15 +1,16 @@
|
||||
from apscheduler.schedulers.background import BackgroundScheduler
|
||||
import sqlite3
|
||||
import requests
|
||||
import datetime
|
||||
|
||||
from initialize_variables import radarr_host_url, headers
|
||||
import initialize_variables
|
||||
|
||||
# Remove all entries from the database of movies that have already finished downloading
|
||||
# This helps to stop from entries building up in the database and slowing down everything
|
||||
sched = BackgroundScheduler(daemon=True)
|
||||
@sched.scheduled_job('cron', hour='0', minute='0')
|
||||
def clear_database():
|
||||
db = sqlite3.connect('/data/movies.db')
|
||||
db = sqlite3.connect(initialize_variables.db_path)
|
||||
cursor = db.cursor()
|
||||
# First get all of the movie ids in the database
|
||||
cursor.execute('''
|
||||
@ -17,7 +18,7 @@ def clear_database():
|
||||
''')
|
||||
movie_ids = cursor.fetchall()
|
||||
# Get all of the movie_ids that are currently downloading/queued and/or missing
|
||||
response = requests.get(f'{radarr_host_url}/api/v3/queue/', headers=headers)
|
||||
response = requests.get(f'{initialize_variables.radarr_host_url}/api/v3/queue/', headers=initialize_variables.headers)
|
||||
current_movie_ids = []
|
||||
for movie in response.json()['records']:
|
||||
current_movie_ids.append(str(movie['movieId']))
|
||||
@ -30,4 +31,21 @@ def clear_database():
|
||||
DELETE FROM movies WHERE movie_id = ?
|
||||
''', (movie_id[0],))
|
||||
db.commit()
|
||||
db.close()
|
||||
|
||||
@sched.scheduled_job('interval', seconds=10)
|
||||
def clear_jellyfin_accounts():
|
||||
db = sqlite3.connect(initialize_variables.db_path)
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
SELECT user_id, deletion_time FROM jellyfin_accounts
|
||||
''')
|
||||
data = cursor.fetchall()
|
||||
for user_id, deletion_time in data:
|
||||
if datetime.datetime.now() > datetime.datetime.strptime(deletion_time, '%Y-%m-%d %H:%M:%S.%f'):
|
||||
requests.delete(f'{initialize_variables.jellyfin_url}/Users/{user_id}', headers=initialize_variables.jellyfin_headers)
|
||||
cursor.execute('''
|
||||
DELETE FROM jellyfin_accounts WHERE user_id = ?
|
||||
''', (user_id,))
|
||||
db.commit()
|
||||
db.close()
|
32
app/db_setup.py
Normal file
32
app/db_setup.py
Normal file
@ -0,0 +1,32 @@
|
||||
import os
|
||||
import sqlite3
|
||||
import initialize_variables
|
||||
|
||||
"""
|
||||
This function is run before the application starts - creates database and sets connection string
|
||||
"""
|
||||
def setup_db():
|
||||
IN_DOCKER = os.environ.get('IN_DOCKER', False)
|
||||
if IN_DOCKER:
|
||||
db = sqlite3.connect('/data/movies.db')
|
||||
initialize_variables.db_path = '/data/movies.db'
|
||||
else:
|
||||
db = sqlite3.connect('movies.db')
|
||||
initialize_variables.db_path = 'movies.db'
|
||||
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
CREATE TABLE IF NOT EXISTS movies(
|
||||
from_number TEXT,
|
||||
movie_id TEXT,
|
||||
movie_title TEXT
|
||||
)
|
||||
''')
|
||||
cursor.execute('''
|
||||
CREATE TABLE IF NOT EXISTS jellyfin_accounts(
|
||||
user_id TEXT,
|
||||
deletion_time DATETIME
|
||||
)
|
||||
''')
|
||||
db.commit()
|
||||
db.close()
|
@ -1,93 +1,61 @@
|
||||
import os
|
||||
import yaml
|
||||
import requests
|
||||
|
||||
supported_sms_services = ['telnyx', 'twilio']
|
||||
|
||||
radarr_host_url = str(os.environ['RADARR_HOST_URL'])
|
||||
radarr_api_key = str(os.environ['RADARR_API_KEY'])
|
||||
enable_kuma_notifications = str(os.environ['ENABLE_KUMA_NOTIFICATIONS']).lower()
|
||||
sms_service = str(os.environ['SMS_SERVICE']).lower()
|
||||
|
||||
headers = {
|
||||
'Content-Type': 'application/json',
|
||||
'X-Api-Key': radarr_api_key
|
||||
}
|
||||
|
||||
# Open the config.yaml file and see if the config is set
|
||||
try:
|
||||
with open('/data/config.yaml', 'r') as f:
|
||||
file = yaml.load(f, Loader=yaml.FullLoader)
|
||||
try:
|
||||
quality_profile_id = int(file['quality_profile_id'])
|
||||
|
||||
if str(file['home_domain']) != 'null':
|
||||
home_domain = str(file['home_domain'])
|
||||
|
||||
api_number = str(file['api_number'])
|
||||
val_nums = str(file['valid_senders'])
|
||||
root_folder_path = str(file['root_folder_path'])
|
||||
|
||||
if enable_kuma_notifications == 'true':
|
||||
notif_receivers_nums = str(file['notif_receivers'])
|
||||
authorization_header_token = str(file['authorization_header_token'])
|
||||
|
||||
if sms_service not in supported_sms_services:
|
||||
print(f'{sms_service} is not a supported SMS service. Please choose from the supported list: {supported_sms_services}')
|
||||
exit()
|
||||
|
||||
if sms_service == 'telnyx':
|
||||
telnyx_api_key = str(file['telnyx_api_key'])
|
||||
|
||||
if sms_service == 'twilio':
|
||||
twilio_account_sid = str(file['twilio_account_sid'])
|
||||
twilio_auth_token = str(file['twilio_auth_token'])
|
||||
|
||||
value_not_set = False
|
||||
except:
|
||||
print('One or more values are not set or not set correctly within the config.yaml file. Please edit the file or refer to the docs for more information.')
|
||||
exit()
|
||||
|
||||
except FileNotFoundError:
|
||||
# Create the config.yaml file
|
||||
with open('/data/config.yaml', 'w') as f:
|
||||
value_not_set = True
|
||||
|
||||
if value_not_set:
|
||||
print('One or more values are not set or not set correctly within the config.yaml file. Please edit the file or refer to the docs for more information.')
|
||||
data = requests.get(f'{radarr_host_url}/api/v3/qualityprofile', headers=headers).json()
|
||||
# Open config.yaml and write each profile as a comment to the file
|
||||
with open('/data/config.yaml', 'w') as f:
|
||||
f.write('# Quality Profile ID\'s\n')
|
||||
for entry in data:
|
||||
f.write(f'# {entry["id"]} - {entry["name"]}\n')
|
||||
|
||||
f.write("quality_profile_id:\n")
|
||||
f.write("home_domain: null\n")
|
||||
f.write("api_number: ''\n")
|
||||
f.write("valid_senders: ''\n")
|
||||
f.write("root_folder_path:\n")
|
||||
|
||||
if enable_kuma_notifications == 'true':
|
||||
f.write("notif_receivers: ''\n")
|
||||
f.write("authorization_header_token: uptimekumaauthtoken\n")
|
||||
|
||||
if sms_service not in supported_sms_services:
|
||||
print(f'{sms_service} is not a supported SMS service. Please choose from the supported list: {supported_sms_services}')
|
||||
exit()
|
||||
|
||||
if sms_service == 'telnyx':
|
||||
f.write("telnyx_api_key:\n")
|
||||
|
||||
if sms_service == 'twilio':
|
||||
f.write("twilio_account_sid:\n")
|
||||
f.write("twilio_auth_token:\n")
|
||||
|
||||
f.write("\n\n# INFORMATION: There should be NO trailing spaced after you enter a value,\n# this will cause errors.\n# There should be one space after the colon though (e.g. quality_profile_id: 1)\n# Check docs for information on each value.")
|
||||
exit()
|
||||
|
||||
numbers_responses = {
|
||||
'1': 1, 'one': 1, '1.': 1,
|
||||
'2': 2, 'two': 2, '2.': 2,
|
||||
'3': 3, 'three': 3, '3.': 3
|
||||
}
|
||||
}
|
||||
|
||||
def init():
|
||||
global sms_service
|
||||
sms_service = ''
|
||||
global telnyx_api_key
|
||||
telnyx_api_key = ''
|
||||
global twilio_account_sid
|
||||
twilio_account_sid = ''
|
||||
global twilio_auth_token
|
||||
twilio_auth_token = ''
|
||||
global api_number
|
||||
api_number = ''
|
||||
global valid_senders
|
||||
valid_senders = []
|
||||
global radarr_host_url
|
||||
radarr_host_url = ''
|
||||
global headers
|
||||
headers = ''
|
||||
global root_folder_path
|
||||
root_folder_path = ''
|
||||
global quality_profile_id
|
||||
quality_profile_id = ''
|
||||
global authorization_header_tokens
|
||||
authorization_header_tokens= ''
|
||||
global notifs_recievers
|
||||
notifs_recievers = []
|
||||
global enable_jellyfin_temp_accounts
|
||||
enable_jellyfin_temp_accounts = ''
|
||||
global jellyfin_url
|
||||
jellyfin_url = ''
|
||||
global jellyfin_headers
|
||||
jellyfin_headers = ''
|
||||
global home_domain
|
||||
home_domain = ''
|
||||
global db_path
|
||||
db_path = ''
|
||||
|
||||
global temp_movie_ids
|
||||
temp_movie_ids = {}
|
||||
"""
|
||||
{
|
||||
'from_number': {
|
||||
'ids': ['tmdb_id_one', 'tmdb_id_two', 'tmdb_id_three'],
|
||||
'time': 'time of request'
|
||||
}
|
||||
}
|
||||
"""
|
||||
global temp_new_account_requests
|
||||
temp_new_account_requests = {}
|
||||
"""
|
||||
{
|
||||
'from_number': 'time'
|
||||
}
|
||||
"""
|
@ -1,50 +1,15 @@
|
||||
import flask
|
||||
import datetime
|
||||
import requests
|
||||
import sqlite3
|
||||
import humanize
|
||||
|
||||
from create_message import create_message
|
||||
from initialize_variables import *
|
||||
from commands.request import request
|
||||
from commands.status import status
|
||||
from commands.number_response_request import number_response_request
|
||||
from commands.movie_show_response_newaccount import movie_show_response_newaccount
|
||||
import initialize_variables
|
||||
|
||||
|
||||
"""
|
||||
Define variables to be used later on
|
||||
"""
|
||||
app = flask.Flask(__name__)
|
||||
|
||||
db = sqlite3.connect('/data/movies.db')
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
CREATE TABLE IF NOT EXISTS movies(
|
||||
from_number TEXT,
|
||||
movie_id TEXT,
|
||||
movie_title TEXT
|
||||
)
|
||||
''')
|
||||
db.commit()
|
||||
db.close()
|
||||
|
||||
temp_movie_ids = {}
|
||||
"""
|
||||
{
|
||||
'from_number': {
|
||||
'ids': ['tmdb_id_one', 'tmdb_id_two', 'tmdb_id_three'],
|
||||
'time': 'time of request'
|
||||
}
|
||||
}
|
||||
"""
|
||||
valid_senders = []
|
||||
|
||||
for number in val_nums.split(', '):
|
||||
valid_senders.append(number)
|
||||
|
||||
if notif_receivers_nums:
|
||||
notif_receivers = []
|
||||
|
||||
for number in notif_receivers_nums.split(', '):
|
||||
notif_receivers.append(number)
|
||||
|
||||
|
||||
"""
|
||||
POST request route to accept incoming notifications from UptimeKuma
|
||||
@ -52,9 +17,9 @@ regarding the status of certain services. Messages are sent to the
|
||||
'notif_receivers' list whenever a service goes down or comes back up.
|
||||
"""
|
||||
@app.route('/kuma', methods=['POST'])
|
||||
def api():
|
||||
def kuma():
|
||||
# Make sure the request is coming from UptimeKuma (Configured to use this authorization token)
|
||||
if flask.request.headers.get('Authorization') == authorization_header_token:
|
||||
if flask.request.headers.get('Authorization') == initialize_variables.authorization_header_tokens:
|
||||
data = flask.request.get_json()
|
||||
|
||||
if data['heartbeat']['status'] == 0:
|
||||
@ -63,7 +28,7 @@ def api():
|
||||
elif data['heartbeat']['status'] == 1:
|
||||
message = f"✅ {data['monitor']['name']} is up!"
|
||||
|
||||
for number in notif_receivers:
|
||||
for number in initialize_variables.notifs_recievers:
|
||||
create_message(number, message)
|
||||
|
||||
return 'OK'
|
||||
@ -80,183 +45,49 @@ and then run the command if it is valid.
|
||||
@app.route('/incoming', methods=['POST'])
|
||||
def incoming():
|
||||
# Get the data and define the from_number (number that sent the message)
|
||||
if sms_service == 'telnyx':
|
||||
if initialize_variables.sms_service == 'telnyx':
|
||||
from_number = flask.request.get_json()['data']['payload']['from']['phone_number']
|
||||
message = str(flask.request.get_json()['data']['payload']['text'])
|
||||
|
||||
if sms_service == 'twilio':
|
||||
if initialize_variables.sms_service == 'twilio':
|
||||
from_number = flask.request.form['From']
|
||||
message = str(flask.request.form['Body'])
|
||||
|
||||
# Make sure the number is a valid_sender, this stops random people from
|
||||
# adding movies to the library
|
||||
if from_number not in valid_senders:
|
||||
if from_number not in initialize_variables.valid_senders:
|
||||
return 'OK'
|
||||
# If the message starts with /request, that means the user is trying to add a movie
|
||||
|
||||
if message.startswith('/request'):
|
||||
# If the user has already run the /request command, delete the entry
|
||||
# from the temp_movie_ids dict so that they can run the command again
|
||||
if from_number in temp_movie_ids.keys():
|
||||
del temp_movie_ids[from_number]
|
||||
# If the user did not include a movie title, alert them to do so
|
||||
# Just check to make sure that the length of the message is greater than 9
|
||||
if len(message) <= 9:
|
||||
create_message(from_number, "Please include the movie title after the /request command.\nEX: /request The Dark Knight")
|
||||
return 'OK'
|
||||
|
||||
incoming_message = message.split(' ', 1)[1]
|
||||
movie_request = incoming_message.replace(' ', '%20')
|
||||
# Send a request to the radarr API to get the movie info
|
||||
response = requests.get(f'{radarr_host_url}/api/v3/movie/lookup?term={movie_request}', headers=headers)
|
||||
# If there are no results, alert the user
|
||||
if len(response.json()) == 0:
|
||||
create_message(from_number, "There were no results for that movie. Please make sure you typed the title correctly.")
|
||||
return 'OK'
|
||||
# If the movie is already added to the library, return a message saying so.
|
||||
if response.json()[0]['added'] != '0001-01-01T05:51:00Z':
|
||||
create_message(from_number, "This movie is already added to the server.\n\nIf you believe this is an error, please contact Parker.")
|
||||
return 'OK'
|
||||
# Define an empty message variable, we then loop through the first 3 results from the API
|
||||
# If there are less than 3 results, we loop through the amount of results there are
|
||||
message = ""
|
||||
for i in range(min(3, len(response.json()))):
|
||||
message += f"{i+1}. {response.json()[i]['folder']}\n\n"
|
||||
if from_number not in temp_movie_ids.keys():
|
||||
temp_movie_ids[from_number] = {
|
||||
'ids': [],
|
||||
'time': datetime.datetime.now()
|
||||
}
|
||||
temp_movie_ids[from_number]['ids'].append(response.json()[i]['tmdbId'])
|
||||
|
||||
message += "Reply with the number associated with the movie you want to download. EX: 1\n\nIf the movie you want is not on the list, make sure you typed the title exactly as it is spelt, or ask Parker to manually add the movie."
|
||||
|
||||
create_message(from_number, message)
|
||||
return 'OK'
|
||||
# Elif the user responded with a variation of 1, 2, or 3
|
||||
# This means they are replying to the previous prompt, so now we need to
|
||||
# add their movie choice to radarr for download
|
||||
elif message.strip() in numbers_responses.keys():
|
||||
# If there is no entry for the user in the temp_movie_ids dict, that means
|
||||
# they have not yet run the /request command, so alert them to do so.
|
||||
if from_number not in temp_movie_ids.keys():
|
||||
create_message(from_number, "There is no current request that you can decide on. It might be that your /request command timed out due since you took too long to response. Please try again. If this issue persists, please contact Parker.")
|
||||
return 'OK'
|
||||
# If the time is greater than 5 minutes, delete the entry from the dict, and alert
|
||||
# the user that their request timed out
|
||||
if (datetime.datetime.now() - temp_movie_ids[from_number]['time']).total_seconds() / 60 > 5:
|
||||
del temp_movie_ids[from_number]
|
||||
create_message(from_number, "You waited too long and therefore your request has timed out.\n\nPlease try again by re-running the /request command. If this issue persists, please contact Parker.")
|
||||
return 'OK'
|
||||
|
||||
# Otherwise, all checks have been completed, so alert the user of the
|
||||
# start of the process
|
||||
create_message(from_number, "Just a moment while I add your movie to the library...")
|
||||
movie_number = numbers_responses[message.strip()]
|
||||
try:
|
||||
tmdb_id = temp_movie_ids[from_number]['ids'][movie_number - 1]
|
||||
except IndexError:
|
||||
create_message(from_number, "You did not enter a valid number. Please re-send the /request command and try again. If you believe this is an error, please contact Parker.")
|
||||
del temp_movie_ids[from_number]
|
||||
return 'OK'
|
||||
|
||||
data = requests.get(f'{radarr_host_url}/api/v3/movie/lookup/tmdb?tmdbId={tmdb_id}', headers=headers)
|
||||
|
||||
data = data.json()
|
||||
movie_title = data['title']
|
||||
# Change the qualityProfileId, monitored, and rootFolderPath values
|
||||
data['qualityProfileId'] = quality_profile_id
|
||||
data['monitored'] = True
|
||||
data['rootFolderPath'] = root_folder_path
|
||||
# Pass this data into a pass request to the radarr API, this will add the movie to the library
|
||||
response = requests.post(f'{radarr_host_url}/api/v3/movie', headers=headers, json=data)
|
||||
data = response.json()
|
||||
movie_id = data['id']
|
||||
# Send message to user alerting them that the movie was added to the library
|
||||
create_message(from_number, f"🎉 {data['title']} has been added to the library!\n\nTo check up on the status of your movie(s) send /status - please wait at least 5 minutes before running this command in order to get an accurate time.")
|
||||
# Finally, as to not slow up the sending of the message, send this request
|
||||
# Send a POST request to the radarr API to search for the movie in the indexers
|
||||
requests.post(f'{radarr_host_url}/api/v3/command', headers=headers, json={'name': 'MoviesSearch', 'movieIds': [int(movie_id)]})
|
||||
|
||||
# Add the movie_id to the database so that users can check up on the status of their movie
|
||||
db = sqlite3.connect('/data/movies.db')
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
INSERT INTO movies(from_number, movie_id, movie_title)
|
||||
VALUES(?, ?, ?)
|
||||
''', (from_number, movie_id, movie_title))
|
||||
db.commit()
|
||||
db.close()
|
||||
|
||||
# Delete the entry from the temp_movie_ids dict
|
||||
del temp_movie_ids[from_number]
|
||||
request(from_number, message)
|
||||
return 'OK'
|
||||
|
||||
elif message.strip() == '/status':
|
||||
# This returns a list of ALL movies being downloaded, but not all of them were
|
||||
# requested by the user, so we need to filter out the ones that were not requested
|
||||
response = requests.get(f'{radarr_host_url}/api/v3/queue/', headers=headers)
|
||||
# Get all the movie_ids that were requested by the user
|
||||
db = sqlite3.connect('/data/movies.db')
|
||||
cursor = db.cursor()
|
||||
cursor.execute('''
|
||||
SELECT movie_id, movie_title FROM movies WHERE from_number = ?
|
||||
''', (from_number,))
|
||||
movie_info = cursor.fetchall()
|
||||
db.close()
|
||||
# Turn the movie_id, movie_title into key value pairs
|
||||
movies = {}
|
||||
for movie in movie_info:
|
||||
movies[movie[0]] = movie[1]
|
||||
# If a user responded with a number, they are responding to
|
||||
# the 'request' command prompt
|
||||
elif message.strip() in initialize_variables.numbers_responses.keys():
|
||||
number_response_request(from_number, message)
|
||||
return 'OK'
|
||||
|
||||
# If the user has no movies in the database, alert them to run the /request command
|
||||
if len(movies) == 0:
|
||||
create_message(from_number, "You have no movies being downloaded at the moment.\n\nIf you previously added a movie, it is likely that it has finished downloading. If you believe this is an error, please contact Parker.")
|
||||
return 'OK'
|
||||
elif message.startswith('/status'):
|
||||
status(from_number)
|
||||
return 'OK'
|
||||
|
||||
message = ""
|
||||
# Loop through the response from the radarr API and filter out the movies that were not requested by the user
|
||||
for movie in response.json()['records']:
|
||||
movie_id = str(movie['movieId'])
|
||||
if movie_id in movies.keys():
|
||||
if movie['status'] == 'downloading':
|
||||
# Humanize the time_left value
|
||||
try:
|
||||
time_left = humanize.precisedelta(datetime.datetime.strptime(movie['timeleft'], '%H:%M:%S') - datetime.datetime.strptime('00:00:00', '%H:%M:%S'), minimum_unit='seconds')
|
||||
except ValueError:
|
||||
# Sometimes movie downloads take a long time and include days in the time_left value
|
||||
# This is formated as 1.00:00:00
|
||||
time_left = humanize.precisedelta(datetime.datetime.strptime(movie['timeleft'], '%d.%H:%M:%S') - datetime.datetime.strptime('00:00:00', '%H:%M:%S'), minimum_unit='seconds')
|
||||
except KeyError:
|
||||
time_left = 'Unknown'
|
||||
elif message.startswith('/newaccount'):
|
||||
if initialize_variables.enable_jellyfin_temp_accounts.lower() == 'true':
|
||||
# If number is already in the temp dict, delete it so that they can redo the request
|
||||
if from_number in initialize_variables.temp_new_account_requests.keys():
|
||||
del initialize_variables.temp_new_account_requests[from_number]
|
||||
|
||||
message += f"📥 {movies[movie_id]} - {time_left}\n"
|
||||
else:
|
||||
message += f"{movies[movie_id]} - {str(movie['status']).upper()}\n"
|
||||
create_message(from_number, "Will you be watching a TV show or a movie?\n\nRespond with 'show' for TV show, 'movie' for movies")
|
||||
initialize_variables.temp_new_account_requests[from_number] = datetime.datetime.now()
|
||||
return 'OK'
|
||||
|
||||
# If the message is empty, that means the user has no movies being downloaded
|
||||
# Or, no download was found for the movie they requested
|
||||
if message == "":
|
||||
# For all movie IDs within the database
|
||||
for movie_id in movies.keys():
|
||||
response = requests.get(f'{radarr_host_url}/api/v3/movie/{movie_id}', headers=headers)
|
||||
# This means that there is no current download, and no file has been found
|
||||
# MOST likely means a download just wasn't found, so alert the user
|
||||
data = response.json()
|
||||
if data['hasFile'] == False:
|
||||
message += f"{movies[movie_id]} - NOT FOUND\n\nThis means a download was not found for the movie(s), if this is a brand new movie that is likely the reason. If the movie has already been released on DVD/Blu-Ray, please contact Parker."
|
||||
# User must be responding to above prompt
|
||||
elif message.strip().lower() in ['show', 'movie']:
|
||||
if initialize_variables.enable_jellyfin_temp_accounts.lower() == 'true':
|
||||
movie_show_response_newaccount(from_number, message)
|
||||
return 'OK'
|
||||
|
||||
# If the message is still empty, that means the user has no movies being downloaded
|
||||
if message != "":
|
||||
create_message(from_number, message)
|
||||
return 'OK'
|
||||
else:
|
||||
create_message(from_number, "You have no movies being downloaded at the moment.\n\nIf you previously added a movie, it is likely that it has finished downloading. If you believe this is an error, please contact Parker.")
|
||||
return 'OK'
|
||||
# Otherwise, add another part to the message containing movie data
|
||||
else:
|
||||
message += "\n\nIf movies consistently show as 'WARNING' or 'QUEUED' or any other error over multiple hours, please contact Parker."
|
||||
create_message(from_number, message)
|
||||
return 'OK'
|
||||
# No valid commands were found, so just return
|
||||
else:
|
||||
return 'OK'
|
||||
@ -265,12 +96,12 @@ def incoming():
|
||||
# Handle 405 errors - when a user attempts a GET request on a POST only route
|
||||
@app.errorhandler(405)
|
||||
def method_not_allowed(e):
|
||||
if home_domain != 'None':
|
||||
return flask.redirect(home_domain)
|
||||
if initialize_variables.home_domain != 'None':
|
||||
return flask.redirect(initialize_variables.home_domain)
|
||||
return 'Method Not Allowed'
|
||||
|
||||
@app.errorhandler(404)
|
||||
def page_not_found(e):
|
||||
if home_domain != 'None':
|
||||
return flask.redirect(home_domain)
|
||||
return 'Page Not Found'
|
||||
if initialize_variables.home_domain != 'None':
|
||||
return flask.redirect(initialize_variables.home_domain)
|
||||
return 'Page Not Found'
|
261
app/validate_config.py
Normal file
261
app/validate_config.py
Normal file
@ -0,0 +1,261 @@
|
||||
import configparser
|
||||
import os
|
||||
from simple_chalk import chalk
|
||||
import validators
|
||||
import requests
|
||||
import initialize_variables
|
||||
|
||||
def write_to_config(config):
|
||||
IN_DOCKER = os.environ.get('IN_DOCKER', False)
|
||||
if IN_DOCKER:
|
||||
with open('/data/config.ini', 'w') as configfile:
|
||||
config.write(configfile)
|
||||
else:
|
||||
with open('config.ini', 'w') as configfile:
|
||||
config.write(configfile)
|
||||
|
||||
|
||||
def validate_config(file_contents):
|
||||
config = configparser.ConfigParser()
|
||||
config.read_string(file_contents)
|
||||
|
||||
# Check SMS service
|
||||
if config['REQUIRED']['SMS_SERVICE'].lower() not in initialize_variables.supported_sms_services:
|
||||
print(chalk.red(f'Invalid or empty SMS_SERVICE option passed. Please choose from the supported list: {initialize_variables.supported_sms_services}'))
|
||||
exit()
|
||||
initialize_variables.sms_service = config['REQUIRED']['SMS_SERVICE'].lower()
|
||||
|
||||
# Check API key is Telnyx is selected
|
||||
if config['REQUIRED']['SMS_SERVICE'].lower() == 'telnyx':
|
||||
try:
|
||||
if not config['REQUIRED']['TELNYX_API_KEY']:
|
||||
print(chalk.red('Empty TELNYX_API_KEY option passed. Please enter the API key for your Telnyx account.'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['REQUIRED']['TELNYX_API_KEY'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty TELNYX_API_KEY option passed. Please enter the API key for your Telnyx account.'))
|
||||
initialize_variables.telnyx_api_key = config['REQUIRED']['TELNYX_API_KEY']
|
||||
|
||||
# Check account SID and auth token is Twilio is selected
|
||||
if config['REQUIRED']['SMS_SERVICE'].lower() == 'twilio':
|
||||
try:
|
||||
if not config['REQUIRED']['TWILIO_ACCOUNT_SID'] or not config['REQUIRED']['TWILIO_AUTH_TOKEN']:
|
||||
print(chalk.red('Empty TWILIO_ACCOUNT_SID or TWILIO_AUTH_TOKEN option passed. Please enter the account SID and auth token for your Twilio account.'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['REQUIRED']['TWILIO_ACCOUNT_SID'] = ''
|
||||
config['REQUIRED']['TWILIO_AUTH_TOKEN'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty TWILIO_ACCOUNT_SID or TWILIO_AUTH_TOKEN option passed. Please enter the account SID and auth token for your Twilio account.'))
|
||||
initialize_variables.twilio_account_sid = config['REQUIRED']['TWILIO_ACCOUNT_SID']
|
||||
initialize_variables.twilio_auth_token = config['REQUIRED']['TWILIO_AUTH_TOKEN']
|
||||
|
||||
# Check API_NUMBER
|
||||
if not config['REQUIRED']['API_NUMBER']:
|
||||
print(chalk.red('Empty API_NUMBER option passed. Please enter an internationally formatted phone number with no spaces.'))
|
||||
exit()
|
||||
if len(config['REQUIRED']['API_NUMBER']) < 12 or len(config['REQUIRED']['API_NUMBER']) > 13 or not config['REQUIRED']['API_NUMBER'].startswith('+'):
|
||||
print(chalk.red('API_NUMBER must be a valid international phone number with no spaces (e.g. +15459087689)'))
|
||||
exit()
|
||||
initialize_variables.api_number = config['REQUIRED']['API_NUMBER']
|
||||
|
||||
# Check VALID_SENDERS
|
||||
if not config['REQUIRED']['VALID_SENDERS']:
|
||||
print(chalk.red('Empty VALID_SENDERS option passed. Please enter a command separated list of internationally formatted phone numbers (e.g. +15359087689, +15256573847)'))
|
||||
exit()
|
||||
|
||||
for sender in config['REQUIRED']['VALID_SENDERS'].split(', '):
|
||||
if len(sender) < 12 or len(sender) > 13 or not sender.startswith('+'):
|
||||
print(chalk.red('At least one number within VALID_SENDER is malformed. Please enter a command separated list of internationally formatted phone numbers (e.g. +15359087689, +15256573847)'))
|
||||
exit()
|
||||
else:
|
||||
initialize_variables.valid_senders.append(sender)
|
||||
|
||||
# Check RADARR_HOST_URL
|
||||
if not validators.url(config['REQUIRED']['RADARR_HOST_URL']):
|
||||
print(chalk.red('Invalid or empty URL passed to RADARR_HOST_URL. Pass a valid URL (e.g. http://localhost:7878)'))
|
||||
exit()
|
||||
initialize_variables.radarr_host_url = config['REQUIRED']['RADARR_HOST_URL']
|
||||
|
||||
# Check RADARR_API_KEY
|
||||
if not config['REQUIRED']['RADARR_API_KEY']:
|
||||
print(chalk.red('Empty RADARR_API_KEY passed. Obtain an API key from your Radarr instance and paste it in this option.'))
|
||||
exit()
|
||||
|
||||
initialize_variables.headers = {
|
||||
'Content-Type': 'application/json',
|
||||
'X-Api-Key': config['REQUIRED']['RADARR_API_KEY']
|
||||
}
|
||||
|
||||
# Make sure connection to Radarr API can be established
|
||||
try:
|
||||
requests.get(config['REQUIRED']['RADARR_HOST_URL'], headers=initialize_variables.headers)
|
||||
except requests.exceptions.ConnectionError:
|
||||
print(chalk.red('Could not connect to Radarr API. Please check your RADARR_HOST_URL and RADARR_API_KEY'))
|
||||
exit()
|
||||
|
||||
# Check ROOT_FOLDER_PATH
|
||||
if not config['REQUIRED']['ROOT_FOLDER_PATH']:
|
||||
print(chalk.red('Empty ROOT_FOLDER_PATH option passed. Please enter a path to a folder within your Radarr instance.'))
|
||||
exit()
|
||||
initialize_variables.root_folder_path = config['REQUIRED']['ROOT_FOLDER_PATH']
|
||||
|
||||
# Check QUALITY_PROFILE_ID
|
||||
data = requests.get(f'{config["REQUIRED"]["RADARR_HOST_URL"]}/api/v3/qualityprofile', headers=initialize_variables.headers).json()
|
||||
all_ids = []
|
||||
for entry in data:
|
||||
all_ids.append(str(entry['id']))
|
||||
|
||||
if not config['REQUIRED']['QUALITY_PROFILE_ID'] or config['REQUIRED']['QUALITY_PROFILE_ID'] not in all_ids:
|
||||
config['AVAILABLE_QUALITY_IDS'] = {}
|
||||
for entry in data:
|
||||
config['AVAILABLE_QUALITY_IDS'][str(entry['id'])] = entry['name']
|
||||
|
||||
print(chalk.red('Empty or invalid QUALITY_PROFILE_ID passed. Pass one of the valid IDs which are now listed within the config.ini file.'))
|
||||
write_to_config(config)
|
||||
exit()
|
||||
initialize_variables.quality_profile_id = config['REQUIRED']['QUALITY_PROFILE_ID']
|
||||
|
||||
# Check ENABLE_KUMA_NOTIFICATIONS
|
||||
if not config['REQUIRED']['ENABLE_KUMA_NOTIFICATIONS'] or config['REQUIRED']['ENABLE_KUMA_NOTIFICATIONS'].lower() not in ['true', 'false']:
|
||||
print(chalk.red('ENABLE_KUMA_NOTIFICATIONS must be a boolean value (true/false)'))
|
||||
exit()
|
||||
|
||||
if config['REQUIRED']['ENABLE_KUMA_NOTIFICATIONS'].lower() == 'true':
|
||||
# Check existence
|
||||
try:
|
||||
if not config['KUMA_NOTIFICATIONS']['AUTHORIZATION_HEADER_TOKEN']:
|
||||
print(chalk.red('Empty AUTHORIZATION_HEADER_TOKEN passed. Make sure to set your authorization header in Uptime Kuma and copy the key here.'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['KUMA_NOTIFICATIONS']['AUTHORIZATION_HEADER_TOKEN'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty AUTHORIZATION_HEADER_TOKEN passed. Make sure to set your authorization header in Uptime Kuma and copy the key here.'))
|
||||
exit()
|
||||
initialize_variables.authorization_header_tokens = config['KUMA_NOTIFICATIONS']['AUTHORIZATION_HEADER_TOKEN']
|
||||
# Check existence
|
||||
try:
|
||||
if not config['KUMA_NOTIFICATIONS']['NOTIF_RECIEVERS']:
|
||||
print(chalk.red('Empty NOTIF_RECIEVERS passed. This should be a comma separated list of the numbers of people who should recieve uptime notifications - formatted the same way as VALID_SENDERS.'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['KUMA_NOTIFICATIONS']['NOTIF_RECIEVERS'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty NOTIF_RECIEVERS passed. This should be a comma separated list of the numbers of people who should recieve uptime notifications - formatted the same way as VALID_SENDERS.'))
|
||||
exit()
|
||||
|
||||
# Check validity of NOTIF_RECIEVERS
|
||||
for sender in config['KUMA_NOTIFICATIONS']['NOTIF_RECIEVERS'].split(', '):
|
||||
if len(sender) < 12 or len(sender) > 13 or not sender.startswith('+'):
|
||||
print(chalk.red('At least one number within NOTIF_RECIEVERS is malformed. Please enter a command separated list of internationally formatted phone numbers (e.g. +15359087689, +15256573847)'))
|
||||
exit()
|
||||
else:
|
||||
initialize_variables.notifs_recievers.append(sender)
|
||||
|
||||
# Check ENABLE_JELLYFIN_TEMP_ACCOUNTS
|
||||
if not config['REQUIRED']['ENABLE_JELLYFIN_TEMP_ACCOUNTS'] or config['REQUIRED']['ENABLE_JELLYFIN_TEMP_ACCOUNTS'].lower() not in ['true', 'false']:
|
||||
print(chalk.red('ENABLE_JELLYFIN_TEMP_ACCOUNTS must be a boolean value (true/false)'))
|
||||
exit()
|
||||
initialize_variables.enable_jellyfin_temp_accounts = config['REQUIRED']['ENABLE_JELLYFIN_TEMP_ACCOUNTS'].lower()
|
||||
|
||||
if config['REQUIRED']['ENABLE_JELLYFIN_TEMP_ACCOUNTS'].lower() == 'true':
|
||||
# Check existence
|
||||
try:
|
||||
if not config['JELLYFIN_ACCOUNTS']['JELLYFIN_URL']:
|
||||
print(chalk.red('Empty URL passed to JELLYFIN_URL. Pass a valid URL (e.g. http://localhost:8096)'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['JELLYFIN_ACCOUNTS']['JELLYFIN_URL'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty URL passed to JELLYFIN_URL. Pass a valid URL (e.g. http://localhost:8096)'))
|
||||
exit()
|
||||
# Check URL validity
|
||||
if not validators.url(config['JELLYFIN_ACCOUNTS']['JELLYFIN_URL']):
|
||||
print(chalk.red('Invalid URL passed to JELLYFIN_URL. Pass a valid URL (e.g. http://localhost:8096)'))
|
||||
exit()
|
||||
initialize_variables.jellyfin_url = config['JELLYFIN_ACCOUNTS']['JELLYFIN_URL']
|
||||
|
||||
# Check existence
|
||||
try:
|
||||
if not config['JELLYFIN_ACCOUNTS']['JELLYFIN_API_KEY']:
|
||||
print(chalk.red('Empty JELLYFIN_API_KEY passed. Create a Jellyfin API key in your Jellyfin dashboard and pass it here.'))
|
||||
exit()
|
||||
except KeyError:
|
||||
config['JELLYFIN_ACCOUNTS']['JELLYFIN_API_KEY'] = ''
|
||||
write_to_config(config)
|
||||
print(chalk.red('Empty JELLYFIN_API_KEY passed. Create a Jellyfin API key in your Jellyfin dashboard and pass it here.'))
|
||||
exit()
|
||||
|
||||
# Make sure connection to Jellyfin API can be established
|
||||
initialize_variables.jellyfin_headers = {
|
||||
'Content-Type': 'application/json',
|
||||
'Authorization': f"MediaBrowser Client=\"other\", device=\"Messagearr\", DeviceId=\"totally-unique-device-id\", Version=\"0.0.0\", Token=\"{config['JELLYFIN_ACCOUNTS']['JELLYFIN_API_KEY']}\""
|
||||
}
|
||||
|
||||
response = requests.get(f"{config['JELLYFIN_ACCOUNTS']['JELLYFIN_URL']}/Users", headers=initialize_variables.jellyfin_headers)
|
||||
if response.status_code != 200:
|
||||
print(chalk.red('Could not connect to Jellyfin API. Please check your JELLYFIN_URL and JELLYFIN_API_KEY'))
|
||||
exit()
|
||||
|
||||
# Validate home domain if it is set
|
||||
if config['OPTIONAL']['HOME_DOMAIN']:
|
||||
if not validators.url(config['OPTIONAL']['HOME_DOMAIN']):
|
||||
print(chalk.red('Invalid HOME_DOMAIN passed. Please enter a valid url (e.g. https://example.com)'))
|
||||
exit()
|
||||
else:
|
||||
initialize_variables.home_domain = config['OPTIONAL']['HOME_DOMAIN']
|
||||
|
||||
"""
|
||||
This method is called before starting the application - to make and validate the configuration
|
||||
"""
|
||||
def make_config():
|
||||
# Attempt to open and validate the configuration file
|
||||
try:
|
||||
with open('config.ini', 'r') as config:
|
||||
file_contents = config.read()
|
||||
validate_config(file_contents)
|
||||
|
||||
except FileNotFoundError:
|
||||
try:
|
||||
with open('/data/config.ini', 'r') as config:
|
||||
file_contents = config.read()
|
||||
validate_config(file_contents)
|
||||
|
||||
except FileNotFoundError:
|
||||
# Create the config.ini file
|
||||
config = configparser.ConfigParser()
|
||||
config['REQUIRED'] = {
|
||||
'SMS_SERVICE': '',
|
||||
'API_NUMBER': '',
|
||||
'VALID_SENDERS': '',
|
||||
'RADARR_HOST_URL': 'http://',
|
||||
'RADARR_API_KEY': '',
|
||||
'ROOT_FOLDER_PATH': '',
|
||||
'QUALITY_PROFILE_ID': '',
|
||||
'ENABLE_KUMA_NOTIFICATIONS': '',
|
||||
'ENABLE_JELLYFIN_TEMP_ACCOUNTS': ''
|
||||
}
|
||||
|
||||
config['OPTIONAL'] = {
|
||||
'HOME_DOMAIN': ''
|
||||
}
|
||||
|
||||
config['KUMA_NOTIFICATIONS'] = {
|
||||
'AUTHORIZATION_HEADER_TOKEN': '',
|
||||
'NOTIF_RECIEVERS': ''
|
||||
}
|
||||
|
||||
config['JELLYFIN_ACCOUNTS'] = {
|
||||
'JELLYFIN_URL': '',
|
||||
'JELLYFIN_API_KEY': ''
|
||||
}
|
||||
|
||||
IN_DOCKER = os.environ.get('IN_DOCKER', False)
|
||||
if IN_DOCKER:
|
||||
with open('/data/config.ini', 'w') as configfile:
|
||||
config.write(configfile)
|
||||
|
||||
else:
|
||||
with open('config.ini', 'w') as configfile:
|
||||
config.write(configfile)
|
@ -5,8 +5,14 @@ import multiprocessing
|
||||
import asyncio
|
||||
|
||||
from db_removal import sched
|
||||
import initialize_variables
|
||||
import validate_config
|
||||
import db_setup
|
||||
|
||||
if __name__ == '__main__':
|
||||
initialize_variables.init()
|
||||
db_setup.setup_db()
|
||||
validate_config.make_config()
|
||||
multiprocessing.Process(target=sched.start()).start()
|
||||
print('Starting server...')
|
||||
config = Config()
|
||||
|
@ -3,12 +3,6 @@ services:
|
||||
messagearr:
|
||||
ports:
|
||||
- '4545:4545'
|
||||
environment:
|
||||
- TZ=America/Chicago # OPTIONAL: Default is UTC
|
||||
- ENABLE_KUMA_NOTIFICATIONS=false # Whether or not to setup UptimeKuma SMS notifications
|
||||
- RADARR_HOST_URL=http://127.0.0.1:7878 # Change to your radarr host
|
||||
- RADARR_API_KEY=apikeyhere # Found by navigating to Settings > General
|
||||
- SMS_SERVICE= # Currently only supporting Telnyx and Twilio
|
||||
volumes:
|
||||
- /loca/file/path:/data
|
||||
image: packetparker/messagearr:latest
|
@ -5,6 +5,7 @@ PyYAML==6.0
|
||||
APScheduler==3.10.4
|
||||
hypercorn==0.14.4
|
||||
asyncio==3.4.3
|
||||
|
||||
simple-chalk==0.1.0
|
||||
telnyx==2.0.0
|
||||
twilio==8.8.0
|
||||
twilio==8.8.0
|
||||
validators==0.22.0
|
Reference in New Issue
Block a user