added watchdog file monitoring
This commit is contained in:
@ -0,0 +1,3 @@
|
||||
|
||||
from core.correctness.validation import *
|
||||
from core.correctness.vars import *
|
@ -1,7 +1,9 @@
|
||||
|
||||
from abc import ABCMeta
|
||||
from os.path import sep
|
||||
from typing import Any, _SpecialForm
|
||||
|
||||
from core.correctness.vars import VALID_PATH_CHARS
|
||||
|
||||
def check_input(variable:Any, expected_type:type, alt_types:list[type]=[],
|
||||
or_none:bool=False)->None:
|
||||
"""
|
||||
@ -73,8 +75,8 @@ def valid_dict(variable:dict[Any, Any], key_type:type, value_type:type,
|
||||
required_keys:list[Any]=[], optional_keys:list[Any]=[],
|
||||
strict:bool=True, min_length:int=1)->None:
|
||||
check_input(variable, dict)
|
||||
check_input(key_type, type, alt_types=[_SpecialForm, ABCMeta])
|
||||
check_input(value_type, type, alt_types=[_SpecialForm, ABCMeta])
|
||||
check_input(key_type, type, alt_types=[_SpecialForm])
|
||||
check_input(value_type, type, alt_types=[_SpecialForm])
|
||||
check_input(required_keys, list)
|
||||
check_input(optional_keys, list)
|
||||
check_input(strict, bool)
|
||||
@ -110,3 +112,12 @@ def valid_list(variable:list[Any], entry_type:type,
|
||||
f"of length {min_length}")
|
||||
for entry in variable:
|
||||
check_input(entry, entry_type, alt_types=alt_types)
|
||||
|
||||
def valid_path(variable:str, allow_base=False, extension:str="", min_length=1):
|
||||
valid_string(variable, VALID_PATH_CHARS, min_length=min_length)
|
||||
if not allow_base and variable.startswith(sep):
|
||||
raise ValueError(f"Cannot accept path '{variable}'. Must be relative.")
|
||||
if min_length > 0 and extension and not variable.endswith(extension):
|
||||
raise ValueError(f"Path '{variable}' does not have required "
|
||||
f"extension '{extension}'.")
|
||||
|
||||
|
@ -1,6 +1,8 @@
|
||||
|
||||
import os
|
||||
|
||||
from inspect import signature
|
||||
|
||||
CHAR_LOWERCASE = 'abcdefghijklmnopqrstuvwxyz'
|
||||
CHAR_UPPERCASE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ'
|
||||
CHAR_NUMERIC = '0123456789'
|
||||
@ -15,4 +17,48 @@ VALID_VARIABLE_NAME_CHARS = CHAR_UPPERCASE + CHAR_LOWERCASE + CHAR_NUMERIC + "_"
|
||||
VALID_JUPYTER_NOTEBOOK_FILENAME_CHARS = VALID_NAME_CHARS + "." + os.path.sep
|
||||
VALID_JUPYTER_NOTEBOOK_EXTENSIONS = [".ipynb"]
|
||||
|
||||
VALID_TRIGGERING_PATH_CHARS = VALID_NAME_CHARS + "." + os.path.sep
|
||||
VALID_PATH_CHARS = VALID_NAME_CHARS + "." + os.path.sep
|
||||
VALID_TRIGGERING_PATH_CHARS = VALID_NAME_CHARS + ".*" + os.path.sep
|
||||
|
||||
BAREBONES_NOTEBOOK = {
|
||||
"cells": [],
|
||||
"metadata": {},
|
||||
"nbformat": 4,
|
||||
"nbformat_minor": 4
|
||||
}
|
||||
|
||||
FILE_CREATE_EVENT = "file_created"
|
||||
FILE_MODIFY_EVENT = "file_modified"
|
||||
FILE_MOVED_EVENT = "file_moved"
|
||||
FILE_CLOSED_EVENT = "file_closed"
|
||||
FILE_DELETED_EVENT = "file_deleted"
|
||||
FILE_EVENTS = [
|
||||
FILE_CREATE_EVENT,
|
||||
FILE_MODIFY_EVENT,
|
||||
FILE_MOVED_EVENT,
|
||||
FILE_CLOSED_EVENT,
|
||||
FILE_DELETED_EVENT
|
||||
]
|
||||
|
||||
DIR_CREATE_EVENT = "dir_created"
|
||||
DIR_MODIFY_EVENT = "dir_modified"
|
||||
DIR_MOVED_EVENT = "dir_moved"
|
||||
DIR_DELETED_EVENT = "dir_deleted"
|
||||
DIR_EVENTS = [
|
||||
DIR_CREATE_EVENT,
|
||||
DIR_MODIFY_EVENT,
|
||||
DIR_MOVED_EVENT,
|
||||
DIR_DELETED_EVENT
|
||||
]
|
||||
|
||||
PIPE_READ = 0
|
||||
PIPE_WRITE = 1
|
||||
|
||||
def get_drt_imp_msg(base_class):
|
||||
return f"{base_class.__name__} may not be instantiated directly. " \
|
||||
f"Implement a child class."
|
||||
|
||||
def get_not_imp_msg(parent_class, class_function):
|
||||
return f"Children of the '{parent_class.__name__}' class must implement " \
|
||||
f"the '{class_function.__name__}({signature(class_function)})' " \
|
||||
"function"
|
||||
|
142
core/meow.py
142
core/meow.py
@ -1,11 +1,13 @@
|
||||
|
||||
import core.correctness.vars
|
||||
import core.correctness.validation
|
||||
|
||||
from abc import ABC, abstractmethod
|
||||
|
||||
from multiprocessing.connection import Connection
|
||||
from typing import Any
|
||||
|
||||
from core.correctness.vars import VALID_RECIPE_NAME_CHARS, \
|
||||
VALID_PATTERN_NAME_CHARS, VALID_RULE_NAME_CHARS, \
|
||||
get_not_imp_msg, get_drt_imp_msg
|
||||
from core.correctness.validation import valid_string
|
||||
|
||||
|
||||
class BaseRecipe:
|
||||
name:str
|
||||
recipe:Any
|
||||
@ -13,6 +15,15 @@ class BaseRecipe:
|
||||
requirements:dict[str, Any]
|
||||
def __init__(self, name:str, recipe:Any, parameters:dict[str,Any]={},
|
||||
requirements:dict[str,Any]={}):
|
||||
if (type(self)._is_valid_recipe == BaseRecipe._is_valid_recipe):
|
||||
msg = get_not_imp_msg(BaseRecipe, BaseRecipe._is_valid_recipe)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_parameters == BaseRecipe._is_valid_parameters):
|
||||
msg = get_not_imp_msg(BaseRecipe, BaseRecipe._is_valid_parameters)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_requirements == BaseRecipe._is_valid_requirements):
|
||||
msg = get_not_imp_msg(BaseRecipe, BaseRecipe._is_valid_requirements)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_name(name)
|
||||
self.name = name
|
||||
self._is_valid_recipe(recipe)
|
||||
@ -24,33 +35,39 @@ class BaseRecipe:
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
if cls is BaseRecipe:
|
||||
raise TypeError("BaseRecipe may not be instantiated directly")
|
||||
msg = get_drt_imp_msg(BaseRecipe)
|
||||
raise TypeError(msg)
|
||||
return object.__new__(cls)
|
||||
|
||||
def _is_valid_name(self, name:str)->None:
|
||||
core.correctness.validation.valid_string(
|
||||
name, core.correctness.vars.VALID_RECIPE_NAME_CHARS)
|
||||
valid_string(name, VALID_RECIPE_NAME_CHARS)
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_recipe(self, recipe:Any)->None:
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_parameters(self, parameters:Any)->None:
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_requirements(self, requirements:Any)->None:
|
||||
pass
|
||||
|
||||
|
||||
class BasePattern(ABC):
|
||||
class BasePattern:
|
||||
name:str
|
||||
recipe:str
|
||||
parameters:dict[str, Any]
|
||||
outputs:dict[str, Any]
|
||||
def __init__(self, name:str, recipe:str, parameters:dict[str,Any]={},
|
||||
outputs:dict[str,Any]={}):
|
||||
if (type(self)._is_valid_recipe == BasePattern._is_valid_recipe):
|
||||
msg = get_not_imp_msg(BasePattern, BasePattern._is_valid_recipe)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_parameters == BasePattern._is_valid_parameters):
|
||||
msg = get_not_imp_msg(BasePattern, BasePattern._is_valid_parameters)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_output == BasePattern._is_valid_output):
|
||||
msg = get_not_imp_msg(BasePattern, BasePattern._is_valid_output)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_name(name)
|
||||
self.name = name
|
||||
self._is_valid_recipe(recipe)
|
||||
@ -62,33 +79,37 @@ class BasePattern(ABC):
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
if cls is BasePattern:
|
||||
raise TypeError("BasePattern may not be instantiated directly")
|
||||
msg = get_drt_imp_msg(BasePattern)
|
||||
raise TypeError(msg)
|
||||
return object.__new__(cls)
|
||||
|
||||
def _is_valid_name(self, name:str)->None:
|
||||
core.correctness.validation.valid_string(
|
||||
name, core.correctness.vars.VALID_PATTERN_NAME_CHARS)
|
||||
valid_string(name, VALID_PATTERN_NAME_CHARS)
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_recipe(self, recipe:Any)->None:
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_parameters(self, parameters:Any)->None:
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_output(self, outputs:Any)->None:
|
||||
pass
|
||||
|
||||
|
||||
class BaseRule(ABC):
|
||||
class BaseRule:
|
||||
name:str
|
||||
pattern:BasePattern
|
||||
recipe:BaseRecipe
|
||||
pattern_type:str=""
|
||||
recipe_type:str=""
|
||||
def __init__(self, name:str, pattern:BasePattern, recipe:BaseRecipe):
|
||||
if (type(self)._is_valid_pattern == BaseRule._is_valid_pattern):
|
||||
msg = get_not_imp_msg(BaseRule, BaseRule._is_valid_pattern)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_recipe == BaseRule._is_valid_recipe):
|
||||
msg = get_not_imp_msg(BaseRule, BaseRule._is_valid_recipe)
|
||||
raise NotImplementedError(msg)
|
||||
|
||||
self._is_valid_name(name)
|
||||
self.name = name
|
||||
self._is_valid_pattern(pattern)
|
||||
@ -99,18 +120,16 @@ class BaseRule(ABC):
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
if cls is BaseRule:
|
||||
raise TypeError("BaseRule may not be instantiated directly")
|
||||
msg = get_drt_imp_msg(BaseRule)
|
||||
raise TypeError(msg)
|
||||
return object.__new__(cls)
|
||||
|
||||
def _is_valid_name(self, name:str)->None:
|
||||
core.correctness.validation.valid_string(
|
||||
name, core.correctness.vars.VALID_RULE_NAME_CHARS)
|
||||
valid_string(name, VALID_RULE_NAME_CHARS)
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_pattern(self, pattern:Any)->None:
|
||||
pass
|
||||
|
||||
@abstractmethod
|
||||
def _is_valid_recipe(self, recipe:Any)->None:
|
||||
pass
|
||||
|
||||
@ -121,3 +140,78 @@ class BaseRule(ABC):
|
||||
if self.recipe_type == "":
|
||||
raise AttributeError(f"Rule Class '{self.__class__.__name__}' "
|
||||
"does not set a recipe_type.")
|
||||
|
||||
|
||||
class BaseMonitor:
|
||||
rules: dict[str, BaseRule]
|
||||
report: Connection
|
||||
listen: Connection
|
||||
def __init__(self, rules:dict[str, BaseRule], report:Connection,
|
||||
listen:Connection) -> None:
|
||||
if (type(self).start == BaseMonitor.start):
|
||||
msg = get_not_imp_msg(BaseMonitor, BaseMonitor.start)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self).stop == BaseMonitor.stop):
|
||||
msg = get_not_imp_msg(BaseMonitor, BaseMonitor.stop)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_report == BaseMonitor._is_valid_report):
|
||||
msg = get_not_imp_msg(BaseMonitor, BaseMonitor._is_valid_report)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_report(report)
|
||||
self.report = report
|
||||
if (type(self)._is_valid_listen == BaseMonitor._is_valid_listen):
|
||||
msg = get_not_imp_msg(BaseMonitor, BaseMonitor._is_valid_listen)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_listen(listen)
|
||||
self.listen = listen
|
||||
if (type(self)._is_valid_rules == BaseMonitor._is_valid_rules):
|
||||
msg = get_not_imp_msg(BaseMonitor, BaseMonitor._is_valid_rules)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_rules(rules)
|
||||
self.rules = rules
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
if cls is BaseMonitor:
|
||||
msg = get_drt_imp_msg(BaseMonitor)
|
||||
raise TypeError(msg)
|
||||
return object.__new__(cls)
|
||||
|
||||
def _is_valid_report(self, report:Connection)->None:
|
||||
pass
|
||||
|
||||
def _is_valid_listen(self, listen:Connection)->None:
|
||||
pass
|
||||
|
||||
def _is_valid_rules(self, rules:dict[str, BaseRule])->None:
|
||||
pass
|
||||
|
||||
def start(self)->None:
|
||||
pass
|
||||
|
||||
def stop(self)->None:
|
||||
pass
|
||||
|
||||
|
||||
class BaseHandler:
|
||||
inputs:Any
|
||||
def __init__(self, inputs:Any) -> None:
|
||||
if (type(self).handle == BaseHandler.handle):
|
||||
msg = get_not_imp_msg(BaseHandler, BaseHandler.handle)
|
||||
raise NotImplementedError(msg)
|
||||
if (type(self)._is_valid_inputs == BaseHandler._is_valid_inputs):
|
||||
msg = get_not_imp_msg(BaseHandler, BaseHandler._is_valid_inputs)
|
||||
raise NotImplementedError(msg)
|
||||
self._is_valid_inputs(inputs)
|
||||
self.inputs = inputs
|
||||
|
||||
def __new__(cls, *args, **kwargs):
|
||||
if cls is BaseHandler:
|
||||
msg = get_drt_imp_msg(BaseHandler)
|
||||
raise TypeError(msg)
|
||||
return object.__new__(cls)
|
||||
|
||||
def _is_valid_inputs(self, inputs:Any)->None:
|
||||
pass
|
||||
|
||||
def handle()->None:
|
||||
pass
|
||||
|
Reference in New Issue
Block a user