contentmonster/classes/file.py

94 lines
2.9 KiB
Python
Raw Normal View History

2021-11-20 14:40:07 +00:00
from classes.chunk import Chunk
from classes.database import Database
from typing import Optional
2021-11-20 14:40:07 +00:00
import hashlib
2021-11-25 15:31:49 +00:00
import os.path
2021-11-20 14:40:07 +00:00
2021-11-20 14:40:07 +00:00
class File:
2021-11-25 15:31:49 +00:00
"""Object representing a file found in a local Directory
"""
def __init__(self, name: str, directory, uuid: Optional[str] = None) -> None:
"""Initialize new File object
2021-11-20 14:40:07 +00:00
Args:
name (str): Filename (basename without path) of the File to create
directory (classes.directory.Directory): Directory object the File
is located within
uuid (str, optional): Unique identifier of this File object. Will
be retrieved from database if None. Defaults to None.
2021-11-25 15:31:49 +00:00
Raises:
FileNotFoundError: Raised if the specified File does not exist
"""
2021-11-20 14:40:07 +00:00
self.name = name
self.directory = directory
2021-11-25 15:31:49 +00:00
if not self.exists():
raise FileNotFoundError(f"File {self.name} does not exist in {self.directory}!")
2021-11-20 14:40:07 +00:00
self.uuid = uuid or self.getUUID()
2021-11-25 15:31:49 +00:00
def exists(self) -> bool:
"""Check if the File exists on the local file system
Returns:
bool: True if the File exists, else False
"""
return os.path.isfile(self.directory.location / self.name)
def getUUID(self) -> str:
"""Return unique identifier for this File object
Returns:
str: File object's UUID retrieved from Database
"""
db = Database()
return db.getFileUUID(self)
def getFullPath(self) -> str:
"""Get the full path of the File
2021-11-20 14:40:07 +00:00
Returns:
str: Full path of the File on the local file system
"""
return self.directory.location / self.name
def getHash(self) -> str:
"""Get hash for this File
Returns:
str: SHA256 for the full content of this File
"""
2021-11-20 14:40:07 +00:00
return self.getChunk(-1).getHash()
def getChunk(self, count: int, size: Optional[int] = None) -> Chunk:
"""Get a Chunk of this File
Args:
count (int): Position of the Chunk in a list of equally large
Chunks (first index: 0). -1 represents a Chunk containing the
complete file.
size (int, optional): Size of the Chunk to create in Bytes. Must
be set if count is not -1. Defaults to None.
Returns:
classes.chunk.Chunk: Chunk object containing File content from
(count * size) bytes to ((count + 1) * size - 1) bytes
Raises:
ValueError: Raised if size is not set for a count that is not -1
"""
if count != -1 and not size:
raise ValueError(
"A Chunk size needs to be passed to getChunk() if not using the complete file (-1)!")
2021-11-20 14:40:07 +00:00
with open(self.getFullPath(), "rb") as binary:
binary.seek((count * size) if count > 0 else 0)
data = binary.read(size if count >= 0 else None)
return Chunk(self, count, data)