2021-08-16 14:15:12 -04:00
|
|
|
import asyncio
|
|
|
|
import logging
|
|
|
|
|
|
|
|
log = logging.getLogger(__name__)
|
2021-08-06 10:44:57 -04:00
|
|
|
|
|
|
|
|
|
|
|
class DiskSpaceManager:
|
|
|
|
|
2021-09-30 21:38:02 -04:00
|
|
|
def __init__(self, config, db, blob_manager, cleaning_interval=30 * 60, analytics=None):
|
2021-08-06 10:44:57 -04:00
|
|
|
self.config = config
|
2021-09-08 10:55:21 -04:00
|
|
|
self.db = db
|
|
|
|
self.blob_manager = blob_manager
|
2021-08-16 14:15:12 -04:00
|
|
|
self.cleaning_interval = cleaning_interval
|
|
|
|
self.running = False
|
|
|
|
self.task = None
|
2021-09-30 21:38:02 -04:00
|
|
|
self.analytics = analytics
|
2021-11-03 15:38:32 -03:00
|
|
|
self._used_space_bytes = None
|
2021-08-06 10:44:57 -04:00
|
|
|
|
2021-10-25 16:44:19 -03:00
|
|
|
async def get_free_space_mb(self, is_network_blob=False):
|
2021-10-24 18:19:06 -03:00
|
|
|
limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
2021-11-02 22:34:15 -03:00
|
|
|
space_used_mb = await self.get_space_used_mb()
|
|
|
|
space_used_mb = space_used_mb['network_storage'] if is_network_blob else space_used_mb['content_storage']
|
|
|
|
return max(0, limit_mb - space_used_mb)
|
2021-10-24 18:19:06 -03:00
|
|
|
|
2021-11-02 22:34:15 -03:00
|
|
|
async def get_space_used_bytes(self):
|
2021-11-03 15:38:32 -03:00
|
|
|
self._used_space_bytes = await self.db.get_stored_blob_disk_usage()
|
|
|
|
return self._used_space_bytes
|
2021-08-06 10:44:57 -04:00
|
|
|
|
2021-11-03 15:38:32 -03:00
|
|
|
async def get_space_used_mb(self, cached=True):
|
|
|
|
cached = cached and self._used_space_bytes is not None
|
|
|
|
space_used_bytes = self._used_space_bytes if cached else await self.get_space_used_bytes()
|
2021-11-02 22:34:15 -03:00
|
|
|
return {key: int(value/1024.0/1024.0) for key, value in space_used_bytes.items()}
|
2021-08-16 14:15:12 -04:00
|
|
|
|
2021-09-08 10:55:21 -04:00
|
|
|
async def clean(self):
|
2021-10-22 02:43:40 -03:00
|
|
|
await self._clean(False)
|
|
|
|
await self._clean(True)
|
|
|
|
|
2021-10-24 18:19:06 -03:00
|
|
|
async def _clean(self, is_network_blob=False):
|
2022-02-11 16:13:08 -03:00
|
|
|
space_used_mb = await self.get_space_used_mb(cached=False)
|
2021-11-02 22:34:15 -03:00
|
|
|
if is_network_blob:
|
2022-02-11 16:13:08 -03:00
|
|
|
space_used_mb = space_used_mb['network_storage']
|
2021-11-02 22:34:15 -03:00
|
|
|
else:
|
2022-02-11 16:13:08 -03:00
|
|
|
space_used_mb = space_used_mb['content_storage'] + space_used_mb['private_storage']
|
2021-10-24 18:19:06 -03:00
|
|
|
storage_limit_mb = self.config.network_storage_limit if is_network_blob else self.config.blob_storage_limit
|
2021-09-30 21:38:02 -04:00
|
|
|
if self.analytics:
|
2021-10-22 03:00:14 -03:00
|
|
|
asyncio.create_task(
|
2022-02-11 16:13:08 -03:00
|
|
|
self.analytics.send_disk_space_used(space_used_mb, storage_limit_mb, is_network_blob)
|
2021-10-22 03:00:14 -03:00
|
|
|
)
|
2021-09-08 10:55:21 -04:00
|
|
|
delete = []
|
2022-02-11 16:13:08 -03:00
|
|
|
available = storage_limit_mb - space_used_mb
|
|
|
|
if storage_limit_mb == 0 if not is_network_blob else available >= 0:
|
2021-09-10 10:53:52 -04:00
|
|
|
return 0
|
2021-10-24 18:19:06 -03:00
|
|
|
for blob_hash, file_size, _ in await self.db.get_stored_blobs(is_mine=False, is_network_blob=is_network_blob):
|
2021-09-10 10:53:52 -04:00
|
|
|
delete.append(blob_hash)
|
2022-02-11 16:13:08 -03:00
|
|
|
available += int(file_size/1024.0/1024.0)
|
|
|
|
if available >= 0:
|
2021-08-16 14:15:12 -04:00
|
|
|
break
|
2021-09-08 10:55:21 -04:00
|
|
|
if delete:
|
2021-09-15 10:37:08 -04:00
|
|
|
await self.db.stop_all_files()
|
2021-09-08 10:55:21 -04:00
|
|
|
await self.blob_manager.delete_blobs(delete, delete_from_db=True)
|
2022-02-11 16:13:08 -03:00
|
|
|
self._used_space_bytes = None
|
2021-09-08 10:55:21 -04:00
|
|
|
return len(delete)
|
2021-08-16 14:15:12 -04:00
|
|
|
|
|
|
|
async def cleaning_loop(self):
|
|
|
|
while self.running:
|
|
|
|
await asyncio.sleep(self.cleaning_interval)
|
2021-09-08 10:55:21 -04:00
|
|
|
await self.clean()
|
2021-08-16 14:15:12 -04:00
|
|
|
|
|
|
|
async def start(self):
|
|
|
|
self.running = True
|
|
|
|
self.task = asyncio.create_task(self.cleaning_loop())
|
|
|
|
self.task.add_done_callback(lambda _: log.info("Stopping blob cleanup service."))
|
|
|
|
|
|
|
|
async def stop(self):
|
|
|
|
if self.running:
|
|
|
|
self.running = False
|
|
|
|
self.task.cancel()
|