2019-10-27 04:53:40 +01:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
#
|
|
|
|
# Copyright (C) 2019 Chris Caron <lead2gold@gmail.com>
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# This code is licensed under the MIT License.
|
|
|
|
#
|
|
|
|
# Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
# of this software and associated documentation files(the "Software"), to deal
|
|
|
|
# in the Software without restriction, including without limitation the rights
|
|
|
|
# to use, copy, modify, merge, publish, distribute, sublicense, and / or sell
|
|
|
|
# copies of the Software, and to permit persons to whom the Software is
|
|
|
|
# furnished to do so, subject to the following conditions :
|
|
|
|
#
|
|
|
|
# The above copyright notice and this permission notice shall be included in
|
|
|
|
# all copies or substantial portions of the Software.
|
|
|
|
#
|
|
|
|
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE
|
|
|
|
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
# THE SOFTWARE.
|
2023-02-26 16:33:14 +01:00
|
|
|
import re
|
2019-10-27 04:53:40 +01:00
|
|
|
import os
|
|
|
|
import tempfile
|
|
|
|
import shutil
|
|
|
|
import gzip
|
|
|
|
import apprise
|
|
|
|
import hashlib
|
|
|
|
import errno
|
|
|
|
|
|
|
|
from django.conf import settings
|
|
|
|
|
2020-09-03 03:33:33 +02:00
|
|
|
# import the logging library
|
|
|
|
import logging
|
|
|
|
|
|
|
|
# Get an instance of a logger
|
2021-10-30 22:58:15 +02:00
|
|
|
logger = logging.getLogger('django')
|
2020-09-03 03:33:33 +02:00
|
|
|
|
|
|
|
|
|
|
|
class AppriseStoreMode(object):
|
|
|
|
"""
|
|
|
|
Defines the store modes of configuration
|
|
|
|
"""
|
|
|
|
# This is the default option. Content is cached and written by
|
|
|
|
# it's key
|
|
|
|
HASH = 'hash'
|
|
|
|
|
|
|
|
# Content is written straight to disk using it's key
|
|
|
|
# there is nothing further done
|
|
|
|
SIMPLE = 'simple'
|
|
|
|
|
|
|
|
# When set to disabled; stateful functionality is disabled
|
|
|
|
DISABLED = 'disabled'
|
|
|
|
|
|
|
|
|
|
|
|
STORE_MODES = (
|
|
|
|
AppriseStoreMode.HASH,
|
|
|
|
AppriseStoreMode.SIMPLE,
|
|
|
|
AppriseStoreMode.DISABLED,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
class SimpleFileExtension(object):
|
|
|
|
"""
|
|
|
|
Defines the simple file exension lookups
|
|
|
|
"""
|
|
|
|
# Simple Configuration file
|
|
|
|
TEXT = 'cfg'
|
|
|
|
|
|
|
|
# YAML Configuration file
|
|
|
|
YAML = 'yml'
|
|
|
|
|
|
|
|
|
|
|
|
SIMPLE_FILE_EXTENSION_MAPPING = {
|
|
|
|
apprise.ConfigFormat.TEXT: SimpleFileExtension.TEXT,
|
|
|
|
apprise.ConfigFormat.YAML: SimpleFileExtension.YAML,
|
|
|
|
SimpleFileExtension.TEXT: SimpleFileExtension.TEXT,
|
|
|
|
SimpleFileExtension.YAML: SimpleFileExtension.YAML,
|
|
|
|
}
|
|
|
|
|
|
|
|
SIMPLE_FILE_EXTENSIONS = (SimpleFileExtension.TEXT, SimpleFileExtension.YAML)
|
|
|
|
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
class AppriseConfigCache(object):
|
|
|
|
"""
|
|
|
|
Designed to make it easy to store/read contact back from disk in a cache
|
|
|
|
type structure that is fast.
|
|
|
|
"""
|
|
|
|
|
2020-09-03 03:33:33 +02:00
|
|
|
def __init__(self, cache_root, salt="apprise", mode=AppriseStoreMode.HASH):
|
2019-10-27 04:53:40 +01:00
|
|
|
"""
|
|
|
|
Works relative to the cache_root
|
|
|
|
"""
|
|
|
|
self.root = cache_root
|
|
|
|
self.salt = salt.encode()
|
2020-09-03 03:33:33 +02:00
|
|
|
self.mode = mode.strip().lower()
|
|
|
|
if self.mode not in STORE_MODES:
|
|
|
|
self.mode = AppriseStoreMode.DISABLED
|
|
|
|
logger.error(
|
|
|
|
'APPRISE_STATEFUL_MODE {} is not supported; '
|
|
|
|
'reverted to {}.'.format(mode, self.mode))
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
def put(self, key, content, fmt):
|
|
|
|
"""
|
|
|
|
Based on the key specified, content is written to disk (compressed)
|
|
|
|
|
|
|
|
key: is an alphanumeric string needed to write and read back this
|
|
|
|
file being written.
|
|
|
|
content: the content to be written to disk
|
|
|
|
fmt: the content config format (of type apprise.ConfigFormat)
|
|
|
|
|
|
|
|
"""
|
|
|
|
# There isn't a lot of error handling done here as it is presumed most
|
|
|
|
# of the checking has been done higher up.
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.DISABLED:
|
|
|
|
# Do nothing
|
|
|
|
return False
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
# First two characters are reserved for cache level directory writing.
|
|
|
|
path, filename = self.path(key)
|
2021-10-30 22:58:15 +02:00
|
|
|
try:
|
|
|
|
os.makedirs(path, exist_ok=True)
|
|
|
|
|
|
|
|
except OSError:
|
|
|
|
# Permission error
|
|
|
|
logger.error('Could not create directory {}'.format(path))
|
|
|
|
return False
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
# Write our file to a temporary file
|
|
|
|
_, tmp_path = tempfile.mkstemp(suffix='.tmp', dir=path)
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.HASH:
|
|
|
|
try:
|
|
|
|
with gzip.open(tmp_path, 'wb') as f:
|
|
|
|
# Write our content to disk
|
|
|
|
f.write(content.encode())
|
|
|
|
|
|
|
|
except OSError:
|
|
|
|
# Handle failure
|
|
|
|
os.remove(tmp_path)
|
|
|
|
return False
|
|
|
|
|
|
|
|
else: # AppriseStoreMode.SIMPLE
|
|
|
|
# Update our file extenion based on our fmt
|
|
|
|
fmt = SIMPLE_FILE_EXTENSION_MAPPING[fmt]
|
|
|
|
try:
|
|
|
|
with open(tmp_path, 'wb') as f:
|
|
|
|
# Write our content to disk
|
|
|
|
f.write(content.encode())
|
|
|
|
|
|
|
|
except OSError:
|
|
|
|
# Handle failure
|
|
|
|
os.remove(tmp_path)
|
|
|
|
return False
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
# If we reach here we successfully wrote the content. We now safely
|
|
|
|
# move our configuration into place. The following writes our content
|
2020-09-03 03:33:33 +02:00
|
|
|
# to disk
|
2019-10-27 04:53:40 +01:00
|
|
|
shutil.move(tmp_path, os.path.join(
|
|
|
|
path, '{}.{}'.format(filename, fmt)))
|
|
|
|
|
|
|
|
# perform tidy of any other lingering files of other type in case
|
|
|
|
# configuration changed from TEXT -> YAML or YAML -> TEXT
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.HASH:
|
|
|
|
if self.clear(key, set(apprise.CONFIG_FORMATS) - {fmt}) is False:
|
|
|
|
# We couldn't remove an existing entry; clear what we just
|
|
|
|
# created
|
|
|
|
self.clear(key, {fmt})
|
|
|
|
# fail
|
|
|
|
return False
|
|
|
|
|
|
|
|
elif self.clear(key, set(SIMPLE_FILE_EXTENSIONS) - {fmt}) is False:
|
|
|
|
# We couldn't remove an existing entry; clear what we just
|
|
|
|
# created
|
2019-10-27 04:53:40 +01:00
|
|
|
self.clear(key, {fmt})
|
|
|
|
# fail
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get(self, key):
|
|
|
|
"""
|
|
|
|
Based on the key specified, content is written to disk (compressed)
|
|
|
|
|
|
|
|
key: is an alphanumeric string needed to write and read back this
|
|
|
|
file being written.
|
|
|
|
|
|
|
|
The function returns a tuple of (content, fmt) where the content
|
|
|
|
is the uncompressed content found in the file and fmt is the
|
|
|
|
content representation (of type apprise.ConfigFormat).
|
|
|
|
|
|
|
|
If no data was found, then (None, None) is returned.
|
|
|
|
"""
|
|
|
|
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.DISABLED:
|
|
|
|
# Do nothing
|
|
|
|
return (None, '')
|
|
|
|
|
2019-10-27 04:53:40 +01:00
|
|
|
# There isn't a lot of error handling done here as it is presumed most
|
|
|
|
# of the checking has been done higher up.
|
|
|
|
|
|
|
|
# First two characters are reserved for cache level directory writing.
|
|
|
|
path, filename = self.path(key)
|
|
|
|
|
|
|
|
# prepare our format to return
|
|
|
|
fmt = None
|
|
|
|
|
|
|
|
# Test the only possible hashed files we expect to find
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.HASH:
|
|
|
|
text_file = os.path.join(
|
|
|
|
path, '{}.{}'.format(filename, apprise.ConfigFormat.TEXT))
|
|
|
|
yaml_file = os.path.join(
|
|
|
|
path, '{}.{}'.format(filename, apprise.ConfigFormat.YAML))
|
|
|
|
|
|
|
|
else: # AppriseStoreMode.SIMPLE
|
|
|
|
text_file = os.path.join(
|
|
|
|
path, '{}.{}'.format(filename, SimpleFileExtension.TEXT))
|
|
|
|
yaml_file = os.path.join(
|
|
|
|
path, '{}.{}'.format(filename, SimpleFileExtension.YAML))
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
if os.path.isfile(text_file):
|
|
|
|
fmt = apprise.ConfigFormat.TEXT
|
|
|
|
path = text_file
|
|
|
|
|
|
|
|
elif os.path.isfile(yaml_file):
|
|
|
|
fmt = apprise.ConfigFormat.YAML
|
|
|
|
path = yaml_file
|
|
|
|
|
|
|
|
else:
|
|
|
|
# Not found; we set the fmt to something other than none as
|
|
|
|
# an indication for the upstream handling to know that we didn't
|
|
|
|
# fail on error
|
|
|
|
return (None, '')
|
|
|
|
|
|
|
|
# Initialize our content
|
|
|
|
content = None
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.HASH:
|
|
|
|
try:
|
|
|
|
with gzip.open(path, 'rb') as f:
|
|
|
|
# Write our content to disk
|
|
|
|
content = f.read().decode()
|
|
|
|
|
|
|
|
except OSError:
|
|
|
|
# all none return means to let upstream know we had a hard
|
|
|
|
# failure
|
|
|
|
return (None, None)
|
2019-10-27 04:53:40 +01:00
|
|
|
|
2020-09-03 03:33:33 +02:00
|
|
|
else: # AppriseStoreMode.SIMPLE
|
|
|
|
try:
|
|
|
|
with open(path, 'rb') as f:
|
|
|
|
# Write our content to disk
|
|
|
|
content = f.read().decode()
|
|
|
|
|
|
|
|
except OSError:
|
|
|
|
# all none return means to let upstream know we had a hard
|
|
|
|
# failure
|
|
|
|
return (None, None)
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
# return our read content
|
|
|
|
return (content, fmt)
|
|
|
|
|
|
|
|
def clear(self, key, formats=None):
|
|
|
|
"""
|
|
|
|
Removes any content associated with the specified key should it
|
|
|
|
exist.
|
|
|
|
|
|
|
|
None is returned if there was nothing to clear
|
|
|
|
True is returned if content was cleared
|
|
|
|
False is returned if an internal error prevented data from being
|
|
|
|
cleared
|
|
|
|
"""
|
|
|
|
# Default our response None
|
|
|
|
response = None
|
|
|
|
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.DISABLED:
|
|
|
|
# Do nothing
|
|
|
|
return response
|
|
|
|
|
2019-10-27 04:53:40 +01:00
|
|
|
if formats is None:
|
|
|
|
formats = apprise.CONFIG_FORMATS
|
|
|
|
|
|
|
|
path, filename = self.path(key)
|
|
|
|
for fmt in formats:
|
|
|
|
# Eliminate any existing content if present
|
|
|
|
try:
|
|
|
|
# Handle failure
|
2020-09-03 03:33:33 +02:00
|
|
|
os.remove(os.path.join(path, '{}.{}'.format(
|
|
|
|
filename,
|
|
|
|
fmt if self.mode == AppriseStoreMode.HASH
|
|
|
|
else SIMPLE_FILE_EXTENSION_MAPPING[fmt])))
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
# If we reach here, an element was removed
|
|
|
|
response = True
|
|
|
|
|
|
|
|
except OSError as e:
|
|
|
|
if e.errno != errno.ENOENT:
|
|
|
|
# We were unable to remove the file
|
|
|
|
response = False
|
|
|
|
|
|
|
|
return response
|
|
|
|
|
|
|
|
def path(self, key):
|
|
|
|
"""
|
|
|
|
returns the path and filename content should be written to based on the
|
|
|
|
specified key
|
|
|
|
"""
|
2020-09-03 03:33:33 +02:00
|
|
|
if self.mode == AppriseStoreMode.HASH:
|
|
|
|
encoded_key = hashlib.sha224(self.salt + key.encode()).hexdigest()
|
|
|
|
path = os.path.join(self.root, encoded_key[0:2])
|
|
|
|
return (path, encoded_key[2:])
|
|
|
|
|
|
|
|
else: # AppriseStoreMode.SIMPLE
|
|
|
|
return (self.root, key)
|
2019-10-27 04:53:40 +01:00
|
|
|
|
|
|
|
|
|
|
|
# Initialize our singleton
|
|
|
|
ConfigCache = AppriseConfigCache(
|
2020-09-03 03:33:33 +02:00
|
|
|
settings.APPRISE_CONFIG_DIR, salt=settings.SECRET_KEY,
|
|
|
|
mode=settings.APPRISE_STATEFUL_MODE)
|
2023-02-26 16:33:14 +01:00
|
|
|
|
|
|
|
|
|
|
|
def apply_global_filters():
|
|
|
|
#
|
|
|
|
# Apply Any Global Filters (if identified)
|
|
|
|
#
|
|
|
|
if settings.APPRISE_ALLOW_SERVICES:
|
|
|
|
alphanum_re = re.compile(
|
|
|
|
r'^(?P<name>[a-z][a-z0-9]+)', re.IGNORECASE)
|
|
|
|
entries = \
|
|
|
|
[alphanum_re.match(x).group('name').lower()
|
|
|
|
for x in re.split(r'[ ,]+', settings.APPRISE_ALLOW_SERVICES)
|
|
|
|
if alphanum_re.match(x)]
|
|
|
|
|
|
|
|
for plugin in set(apprise.common.NOTIFY_SCHEMA_MAP.values()):
|
|
|
|
if entries:
|
|
|
|
# Get a list of the current schema's associated with
|
|
|
|
# a given plugin
|
|
|
|
schemas = set(apprise.plugins.details(plugin)
|
|
|
|
['tokens']['schema']['values'])
|
|
|
|
|
|
|
|
# Check what was defined and see if there is a hit
|
|
|
|
for entry in entries:
|
|
|
|
if entry in schemas:
|
|
|
|
# We had a hit; we're done
|
|
|
|
break
|
|
|
|
|
|
|
|
if entry in schemas:
|
|
|
|
entries.remove(entry)
|
|
|
|
# We can keep this plugin enabled and move along to the
|
|
|
|
# next one...
|
|
|
|
continue
|
|
|
|
|
|
|
|
# if we reach here, we have to block our plugin
|
|
|
|
plugin.enabled = False
|
|
|
|
|
|
|
|
for entry in entries:
|
|
|
|
# Generate some noise for those who have bad configurations
|
|
|
|
logger.warning(
|
|
|
|
'APPRISE_ALLOW_SERVICES plugin %s:// was not found - '
|
|
|
|
'ignoring.', entry)
|
|
|
|
|
|
|
|
elif settings.APPRISE_DENY_SERVICES:
|
|
|
|
alphanum_re = re.compile(
|
|
|
|
r'^(?P<name>[a-z][a-z0-9]+)', re.IGNORECASE)
|
|
|
|
entries = \
|
|
|
|
[alphanum_re.match(x).group('name').lower()
|
|
|
|
for x in re.split(r'[ ,]+', settings.APPRISE_DENY_SERVICES)
|
|
|
|
if alphanum_re.match(x)]
|
|
|
|
|
|
|
|
for name in entries:
|
|
|
|
try:
|
|
|
|
# Force plugin to be disabled
|
|
|
|
apprise.common.NOTIFY_SCHEMA_MAP[name].enabled = False
|
|
|
|
|
|
|
|
except KeyError:
|
|
|
|
logger.warning(
|
|
|
|
'APPRISE_DENY_SERVICES plugin %s:// was not found -'
|
|
|
|
' ignoring.', name)
|