file_manager: implement metadata "move" functionality
If a file is moved from one location to another in the "gcodes" root it isn't necessary to rescan the file metadata as the underlying file information does not change. Add the ability to change the metadata's key and move associated thumbnails when a file is moved. Signed-off-by: Eric Callahan <arksine.code@gmail.com>
This commit is contained in:
parent
a9f1a98fa2
commit
21d290beb7
|
@ -635,8 +635,11 @@ class InotifyNode:
|
||||||
new_path = child_node.get_path()
|
new_path = child_node.get_path()
|
||||||
new_root = child_node.get_root()
|
new_root = child_node.get_root()
|
||||||
logging.debug(f"Moving node from '{prev_path}' to '{new_path}'")
|
logging.debug(f"Moving node from '{prev_path}' to '{new_path}'")
|
||||||
# TODO: It is possible to "move" metadata rather
|
# Attempt to move metadata
|
||||||
# than rescan.
|
move_success = await self.ihdlr.try_move_metadata(
|
||||||
|
prev_root, new_root, prev_path, new_path, is_dir=True)
|
||||||
|
if not move_success:
|
||||||
|
# Need rescan
|
||||||
mevts = child_node.scan_node()
|
mevts = child_node.scan_node()
|
||||||
if mevts:
|
if mevts:
|
||||||
mfuts = [e.wait() for e in mevts]
|
mfuts = [e.wait() for e in mevts]
|
||||||
|
@ -842,6 +845,30 @@ class INotifyHandler:
|
||||||
else:
|
else:
|
||||||
self.gcode_metadata.remove_file_metadata(rel_path)
|
self.gcode_metadata.remove_file_metadata(rel_path)
|
||||||
|
|
||||||
|
async def try_move_metadata(self, prev_root, new_root, prev_path,
|
||||||
|
new_path, is_dir=False):
|
||||||
|
if new_root == "gcodes":
|
||||||
|
if prev_root == "gcodes":
|
||||||
|
# moved within the gcodes root, move metadata
|
||||||
|
prev_rel_path = self.file_manager.get_relative_path(
|
||||||
|
"gcodes", prev_path)
|
||||||
|
new_rel_path = self.file_manager.get_relative_path(
|
||||||
|
"gcodes", new_path)
|
||||||
|
if is_dir:
|
||||||
|
self.gcode_metadata.move_directory_metadata(
|
||||||
|
prev_rel_path, new_rel_path)
|
||||||
|
else:
|
||||||
|
self.gcode_metadata.move_file_metadata(
|
||||||
|
prev_rel_path, new_rel_path)
|
||||||
|
else:
|
||||||
|
# move from a non-gcodes root to gcodes return true
|
||||||
|
self.clear_metadata(prev_root, prev_path, is_dir)
|
||||||
|
return False
|
||||||
|
elif prev_root == "gcodes":
|
||||||
|
# moved out of the gcodes root, remove metadata
|
||||||
|
self.clear_metadata(prev_root, prev_path, is_dir)
|
||||||
|
return True
|
||||||
|
|
||||||
def log_nodes(self):
|
def log_nodes(self):
|
||||||
if self.debug_enabled:
|
if self.debug_enabled:
|
||||||
debug_msg = f"Inotify Watches After Scan:"
|
debug_msg = f"Inotify Watches After Scan:"
|
||||||
|
@ -874,6 +901,7 @@ class INotifyHandler:
|
||||||
parent_node, name, hdl = self.pending_moves.pop(cookie)
|
parent_node, name, hdl = self.pending_moves.pop(cookie)
|
||||||
item_path = os.path.join(parent_node.get_path(), name)
|
item_path = os.path.join(parent_node.get_path(), name)
|
||||||
root = parent_node.get_root()
|
root = parent_node.get_root()
|
||||||
|
self.clear_metadata(root, item_path, is_dir)
|
||||||
action = "delete_file"
|
action = "delete_file"
|
||||||
if is_dir:
|
if is_dir:
|
||||||
# The supplied node is a child node
|
# The supplied node is a child node
|
||||||
|
@ -887,9 +915,6 @@ class INotifyHandler:
|
||||||
self.notify_filelist_changed(action, root, item_path)
|
self.notify_filelist_changed(action, root, item_path)
|
||||||
|
|
||||||
def _schedule_pending_move(self, evt, parent_node, is_dir):
|
def _schedule_pending_move(self, evt, parent_node, is_dir):
|
||||||
item_path = os.path.join(parent_node.get_path(), evt.name)
|
|
||||||
root = parent_node.get_root()
|
|
||||||
self.clear_metadata(root, item_path, is_dir)
|
|
||||||
hdl = IOLoop.current().call_later(
|
hdl = IOLoop.current().call_later(
|
||||||
INOTIFY_MOVE_TIME, self._handle_move_timeout,
|
INOTIFY_MOVE_TIME, self._handle_move_timeout,
|
||||||
evt.cookie, is_dir)
|
evt.cookie, is_dir)
|
||||||
|
@ -983,9 +1008,6 @@ class INotifyHandler:
|
||||||
logging.debug(f"Inotify file move to: {root}, "
|
logging.debug(f"Inotify file move to: {root}, "
|
||||||
f"{node_path}, {evt.name}")
|
f"{node_path}, {evt.name}")
|
||||||
file_path = os.path.join(node_path, evt.name)
|
file_path = os.path.join(node_path, evt.name)
|
||||||
if root == "gcodes":
|
|
||||||
mevt = self.parse_gcode_metadata(file_path)
|
|
||||||
await mevt.wait()
|
|
||||||
moved_evt = self.pending_moves.pop(evt.cookie, None)
|
moved_evt = self.pending_moves.pop(evt.cookie, None)
|
||||||
if moved_evt is not None:
|
if moved_evt is not None:
|
||||||
# Moved from a currently watched directory
|
# Moved from a currently watched directory
|
||||||
|
@ -993,10 +1015,19 @@ class INotifyHandler:
|
||||||
IOLoop.current().remove_timeout(hdl)
|
IOLoop.current().remove_timeout(hdl)
|
||||||
prev_root = prev_parent.get_root()
|
prev_root = prev_parent.get_root()
|
||||||
prev_path = os.path.join(prev_parent.get_path(), prev_name)
|
prev_path = os.path.join(prev_parent.get_path(), prev_name)
|
||||||
|
move_success = await self.try_move_metadata(
|
||||||
|
prev_root, root, prev_path, file_path)
|
||||||
|
if not move_success:
|
||||||
|
# Unable to move, metadata needs parsing
|
||||||
|
mevt = self.parse_gcode_metadata(file_path)
|
||||||
|
await mevt.wait()
|
||||||
self.notify_filelist_changed(
|
self.notify_filelist_changed(
|
||||||
"move_file", root, file_path,
|
"move_file", root, file_path,
|
||||||
prev_root, prev_path)
|
prev_root, prev_path)
|
||||||
else:
|
else:
|
||||||
|
if root == "gcodes":
|
||||||
|
mevt = self.parse_gcode_metadata(file_path)
|
||||||
|
await mevt.wait()
|
||||||
self.notify_filelist_changed(
|
self.notify_filelist_changed(
|
||||||
"create_file", root, file_path)
|
"create_file", root, file_path)
|
||||||
elif evt.mask & iFlags.MODIFY:
|
elif evt.mask & iFlags.MODIFY:
|
||||||
|
@ -1110,6 +1141,37 @@ class MetadataStorage:
|
||||||
except Exception:
|
except Exception:
|
||||||
logging.debug(f"Error removing thumb at {thumb_path}")
|
logging.debug(f"Error removing thumb at {thumb_path}")
|
||||||
|
|
||||||
|
def move_directory_metadata(self, prev_dir, new_dir):
|
||||||
|
if prev_dir[-1] != "/":
|
||||||
|
prev_dir += "/"
|
||||||
|
for prev_fname in list(self.mddb.keys()):
|
||||||
|
if prev_fname.startswith(prev_dir):
|
||||||
|
new_fname = os.path.join(new_dir, prev_fname[len(prev_dir):])
|
||||||
|
self.move_file_metadata(prev_fname, new_fname, False)
|
||||||
|
|
||||||
|
def move_file_metadata(self, prev_fname, new_fname, move_thumbs=True):
|
||||||
|
metadata = self.mddb.pop(prev_fname, None)
|
||||||
|
if metadata is None:
|
||||||
|
return
|
||||||
|
self.mddb[new_fname] = metadata
|
||||||
|
prev_dir = os.path.dirname(os.path.join(self.gc_path, prev_fname))
|
||||||
|
new_dir = os.path.dirname(os.path.join(self.gc_path, new_fname))
|
||||||
|
if "thumbnails" in metadata and move_thumbs:
|
||||||
|
for thumb in metadata["thumbnails"]:
|
||||||
|
path = thumb.get("relative_path", None)
|
||||||
|
if path is None:
|
||||||
|
continue
|
||||||
|
thumb_path = os.path.join(prev_dir, path)
|
||||||
|
if not os.path.isfile(thumb_path):
|
||||||
|
continue
|
||||||
|
new_path = os.path.join(new_dir, path)
|
||||||
|
try:
|
||||||
|
os.makedirs(os.path.dirname(new_path), exist_ok=True)
|
||||||
|
shutil.move(thumb_path, new_path)
|
||||||
|
except Exception:
|
||||||
|
logging.debug(f"Error moving thumb from {thumb_path}"
|
||||||
|
f" to {new_path}")
|
||||||
|
|
||||||
def parse_metadata(self, fname, path_info):
|
def parse_metadata(self, fname, path_info):
|
||||||
mevt = Event()
|
mevt = Event()
|
||||||
if fname in self.pending_requests or \
|
if fname in self.pending_requests or \
|
||||||
|
|
Loading…
Reference in New Issue