2020-09-26 17:30:59 +02:00
|
|
|
import json
|
|
|
|
|
2020-09-27 23:06:16 +02:00
|
|
|
from xion.xfconf import Xfconf
|
2020-09-26 17:30:59 +02:00
|
|
|
|
|
|
|
|
|
|
|
class Xion:
|
2020-09-27 17:28:59 +02:00
|
|
|
"""Manipulate Xfconf settings trees."""
|
2020-09-26 17:30:59 +02:00
|
|
|
|
2020-11-02 23:53:36 +01:00
|
|
|
def __init__(self, xq=None, verbose=False):
|
|
|
|
self.xfconf = Xfconf(xq=xq, verbose=verbose)
|
|
|
|
self.verbose = verbose
|
2020-09-26 17:30:59 +02:00
|
|
|
|
|
|
|
def build_tree(self, channel, root="/"):
|
2020-09-27 17:28:59 +02:00
|
|
|
"""Return a dict of properties in this channel, filtering on root.
|
2020-09-26 17:30:59 +02:00
|
|
|
|
2020-09-27 17:28:59 +02:00
|
|
|
Return None on error. Root has to start with "/" to be valid. Arrays are
|
|
|
|
added to the tree as a list of properties.
|
2020-09-26 17:30:59 +02:00
|
|
|
"""
|
2020-09-27 16:31:13 +02:00
|
|
|
if not root.startswith("/"):
|
|
|
|
print("Invalid root, must start with /")
|
|
|
|
return None
|
2020-09-26 17:30:59 +02:00
|
|
|
props = self.xfconf.get_property_list(channel, root=root)
|
|
|
|
if props is None:
|
|
|
|
print(f"Failed to get property list for channel {channel}.")
|
|
|
|
return None
|
|
|
|
tree = {}
|
|
|
|
for prop_name in props:
|
|
|
|
prop = self.xfconf.get_property(channel, prop_name)
|
|
|
|
if prop is None:
|
|
|
|
print(f"Failed to get property {prop_name}.")
|
|
|
|
return None
|
|
|
|
if isinstance(prop, list):
|
|
|
|
leaf = [Xion._build_prop_leaf(p) for p in prop]
|
|
|
|
else:
|
|
|
|
leaf = Xion._build_prop_leaf(prop)
|
|
|
|
tree[prop_name] = leaf
|
|
|
|
return tree
|
|
|
|
|
2020-09-27 18:13:03 +02:00
|
|
|
@staticmethod
|
|
|
|
def _build_prop_leaf(prop):
|
|
|
|
return {"type": prop.gtype, "value": str(prop.value)}
|
|
|
|
|
2020-09-27 16:31:13 +02:00
|
|
|
def export_tree(self, channel, root, tree, output_path):
|
2020-09-27 17:28:59 +02:00
|
|
|
"""Export a property tree as a sorted JSON file."""
|
2020-09-27 16:31:13 +02:00
|
|
|
tree["channel"] = channel
|
|
|
|
tree["root"] = root
|
2020-09-26 17:30:59 +02:00
|
|
|
with open(output_path, "wt") as output_file:
|
|
|
|
json.dump(tree, output_file, indent=2, sort_keys=True)
|
|
|
|
|
2020-09-27 16:31:13 +02:00
|
|
|
def import_tree(self, file_path):
|
2020-09-27 17:28:59 +02:00
|
|
|
"""Load a property tree."""
|
2020-09-27 16:31:13 +02:00
|
|
|
with open(file_path, "rt") as input_file:
|
|
|
|
tree = json.load(input_file)
|
|
|
|
try:
|
|
|
|
channel = tree.pop("channel")
|
|
|
|
root = tree.pop("root")
|
|
|
|
except KeyError:
|
|
|
|
print("Missing channel or root in JSON.")
|
|
|
|
return None, None, tree
|
|
|
|
return channel, root, tree
|
|
|
|
|
|
|
|
def apply_tree(self, channel, root, tree, confirm=True, replace=False):
|
2020-09-27 17:28:59 +02:00
|
|
|
"""Apply tree settings under root to channel, return True on success."""
|
2020-09-27 16:31:13 +02:00
|
|
|
num_changes = len(tree)
|
2020-09-27 17:28:59 +02:00
|
|
|
print(f"{num_changes} changes to do in {channel} for {root}.")
|
2020-09-27 16:31:13 +02:00
|
|
|
if replace:
|
2020-09-27 17:28:59 +02:00
|
|
|
print("This will erase all properties in the channel.")
|
2020-09-27 16:31:13 +02:00
|
|
|
if confirm and input("Confirm? [y/N]") != "y":
|
|
|
|
print("Operation cancelled.")
|
2020-09-27 17:28:59 +02:00
|
|
|
return False
|
2020-09-28 23:04:16 +02:00
|
|
|
if replace:
|
|
|
|
if not self.clear_tree(channel, root):
|
|
|
|
print("Failed to clear properties.")
|
|
|
|
return False
|
2020-09-27 16:31:13 +02:00
|
|
|
for prop, content in tree.items():
|
2020-09-27 17:28:59 +02:00
|
|
|
if not self.apply_property(channel, prop, content):
|
|
|
|
print(f"Failed to apply property {prop}.")
|
|
|
|
return False
|
|
|
|
print("Done.")
|
|
|
|
return True
|
|
|
|
|
|
|
|
def clear_tree(self, channel, root):
|
|
|
|
"""Remove all channel configs under root, return True on success."""
|
|
|
|
return self.xfconf.reset_root(channel, root)
|
2020-09-27 16:31:13 +02:00
|
|
|
|
|
|
|
def apply_property(self, channel, name, content):
|
|
|
|
"""Update one property in Xfconf, return True on success."""
|
2020-09-27 18:13:03 +02:00
|
|
|
if isinstance(content, list):
|
|
|
|
return self.xfconf.set_property_array(channel, name, content)
|
2020-09-27 16:31:13 +02:00
|
|
|
prop_type = content["type"]
|
|
|
|
value = content["value"]
|
2020-09-27 18:13:03 +02:00
|
|
|
return self.xfconf.set_property(channel, name, prop_type, value)
|