beets-config/plugins/beetslabels.py

189 lines
5.1 KiB
Python

from beets import ui
from beets.plugins import BeetsPlugin
from beets.ui import Subcommand, print_
from beets.ui.commands import print_and_modify
from beets.dbcore import types
from beets.dbcore import FieldQuery
import optparse
import json
def do_modify_labels(lib, objs, label, action):
changed = []
for obj in objs:
labels = {}
if action != 2:
if "mylabels" in obj:
labels = json.loads(obj["mylabels"])
split = label.split(":")
if action == 0:
if len(split) == 1:
labels[label] = 0
else:
labels[split[0]] = int(split[1])
elif label in labels:
del labels[label]
obj_mods = {
"mylabels": json.dumps(labels)
}
if print_and_modify(obj, obj_mods, []) and obj not in changed:
changed.append(obj)
# Still something to do?
if not changed:
print_("No changes to make.")
return
# Confirm action.
changed = ui.input_select_objects(
"Really modify",
changed,
lambda o: print_and_modify(o, mods, dels),
)
# Apply changes to database and files
with lib.transaction():
for obj in changed:
obj.try_sync(True, False, False)
action_map = {
"add": 0,
"remove": 1,
"removeall": 2,
"show": 3,
"transfer": 4
}
def modify_labels(lib, opts, args):
action = args[0]
if action not in action_map:
print_("%s is not a valid action. " % action)
print_("Valid actions are: add, remove, removeall, show, transfer")
return
actnum = action_map[action]
if actnum == 4: # transfer
transfer_labels(lib, opts, args[1:])
return
if actnum >= 2:
label = None
query = args[1:]
else:
label = args[1]
query = args[2:]
items = lib.items(query=query)
if actnum == 3:
for obj in items:
if "mylabels" in obj:
labels = json.loads(obj["mylabels"])
labelstr = ";".join([f"{key}:{labels[key]}" for key in labels.keys()])
print_(f"{obj.title}: {labelstr}")
else:
do_modify_labels(lib, items, label, actnum)
def transfer_labels(lib, opts, args):
source_query = args[0]
dest_query = args[1]
source_items = lib.items(query=source_query)
dest_items = lib.items(query=dest_query)
source_list = list(source_items)
dest_list = list(dest_items)
if len(source_list) == 0:
print_("No source items found.")
return
if len(dest_list) == 0:
print_("No destination items found.")
return
if len(source_list) > 1:
print_("Multiple source items found. Please refine your query to select one source item.")
for item in source_list:
print_(f" - {item.artist} - {item.title} ({item.path})")
return
if len(dest_list) > 1:
print_("Multiple destination items found. Please refine your query to select one destination item.")
for item in dest_list:
print_(f" - {item.artist} - {item.title} ({item.path})")
return
source = source_list[0]
dest = dest_list[0]
print_(f"Source: {source.artist} - {source.title}")
print_(f"Dest: {dest.artist} - {dest.title}")
print_("")
if "mylabels" not in source or not source["mylabels"]:
print_("Source has no labels to transfer.")
return
labels = json.loads(source["mylabels"])
labelstr = ";".join([f"{key}:{labels[key]}" for key in labels.keys()])
print_(f"Labels to transfer: {labelstr}")
if "mylabels" in dest and dest["mylabels"]:
dest_labels = json.loads(dest["mylabels"])
dest_labelstr = ";".join([f"{key}:{dest_labels[key]}" for key in dest_labels.keys()])
print_(f"WARNING: Destination already has labels: {dest_labelstr}")
print_("These will be overwritten!")
print_("")
if not ui.input_yn("Transfer labels? (y/n)", True):
return
with lib.transaction():
dest["mylabels"] = source["mylabels"]
dest.try_sync(True, False, False)
print_("Transfer complete.")
labels_command = Subcommand('labels', help='Add or remove labels')
labels_command.func = modify_labels
class HasLabelQuery(FieldQuery):
def __init__(self, _, pattern: str, __):
super().__init__("mylabels", pattern, False)
@classmethod
def value_match(self, pattern, jsonstr):
if jsonstr is not None:
label = pattern
value = None
if "." in pattern:
label,value = pattern.split(".")
labels = json.loads(jsonstr)
if value == None:
return label in labels
else:
return label in labels and str(labels[label]) == value
return False
class BeetsLabelsPlugin(BeetsPlugin):
# item_queries = { 'has_label': DelimitedHasExact }
# item_types = {'mylabels': types.SEMICOLON_SPACE_DSV}
def __init__(self):
super().__init__()
self.item_queries = {"label": HasLabelQuery}
def commands(self):
return [labels_command]