2019-04-03 15:40:03 +00:00
|
|
|
"""Sensor for monitoring the contents of a folder."""
|
2018-02-22 07:21:07 +00:00
|
|
|
from datetime import timedelta
|
|
|
|
import glob
|
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
|
|
|
|
import voluptuous as vol
|
|
|
|
|
|
|
|
from homeassistant.helpers.entity import Entity
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
|
|
|
from homeassistant.components.sensor import PLATFORM_SCHEMA
|
|
|
|
|
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
CONF_FOLDER_PATHS = 'folder'
|
|
|
|
CONF_FILTER = 'filter'
|
|
|
|
DEFAULT_FILTER = '*'
|
|
|
|
|
2018-03-03 22:03:06 +00:00
|
|
|
SCAN_INTERVAL = timedelta(minutes=1)
|
2018-02-22 07:21:07 +00:00
|
|
|
|
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({
|
|
|
|
vol.Required(CONF_FOLDER_PATHS): cv.isdir,
|
|
|
|
vol.Optional(CONF_FILTER, default=DEFAULT_FILTER): cv.string,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
def get_files_list(folder_path, filter_term):
|
|
|
|
"""Return the list of files, applying filter."""
|
|
|
|
query = folder_path + filter_term
|
|
|
|
files_list = glob.glob(query)
|
|
|
|
return files_list
|
|
|
|
|
|
|
|
|
|
|
|
def get_size(files_list):
|
|
|
|
"""Return the sum of the size in bytes of files in the list."""
|
2018-04-08 09:32:49 +00:00
|
|
|
size_list = [os.stat(f).st_size for f in files_list if os.path.isfile(f)]
|
2018-02-22 07:21:07 +00:00
|
|
|
return sum(size_list)
|
|
|
|
|
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2018-02-22 07:21:07 +00:00
|
|
|
"""Set up the folder sensor."""
|
|
|
|
path = config.get(CONF_FOLDER_PATHS)
|
|
|
|
|
|
|
|
if not hass.config.is_allowed_path(path):
|
|
|
|
_LOGGER.error("folder %s is not valid or allowed", path)
|
|
|
|
else:
|
|
|
|
folder = Folder(path, config.get(CONF_FILTER))
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities([folder], True)
|
2018-02-22 07:21:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
class Folder(Entity):
|
|
|
|
"""Representation of a folder."""
|
|
|
|
|
|
|
|
ICON = 'mdi:folder'
|
|
|
|
|
|
|
|
def __init__(self, folder_path, filter_term):
|
|
|
|
"""Initialize the data object."""
|
|
|
|
folder_path = os.path.join(folder_path, '') # If no trailing / add it
|
|
|
|
self._folder_path = folder_path # Need to check its a valid path
|
|
|
|
self._filter_term = filter_term
|
|
|
|
self._number_of_files = None
|
|
|
|
self._size = None
|
|
|
|
self._name = os.path.split(os.path.split(folder_path)[0])[1]
|
|
|
|
self._unit_of_measurement = 'MB'
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Update the sensor."""
|
|
|
|
files_list = get_files_list(self._folder_path, self._filter_term)
|
|
|
|
self._number_of_files = len(files_list)
|
|
|
|
self._size = get_size(files_list)
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def state(self):
|
|
|
|
"""Return the state of the sensor."""
|
|
|
|
decimals = 2
|
|
|
|
size_mb = round(self._size/1e6, decimals)
|
|
|
|
return size_mb
|
|
|
|
|
|
|
|
@property
|
|
|
|
def icon(self):
|
|
|
|
"""Icon to use in the frontend, if any."""
|
|
|
|
return self.ICON
|
|
|
|
|
|
|
|
@property
|
|
|
|
def device_state_attributes(self):
|
|
|
|
"""Return other details about the sensor state."""
|
|
|
|
attr = {
|
|
|
|
'path': self._folder_path,
|
|
|
|
'filter': self._filter_term,
|
|
|
|
'number_of_files': self._number_of_files,
|
|
|
|
'bytes': self._size,
|
|
|
|
}
|
|
|
|
return attr
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unit_of_measurement(self):
|
|
|
|
"""Return the unit of measurement of this entity, if any."""
|
|
|
|
return self._unit_of_measurement
|