2020-10-26 22:45:22 +00:00
|
|
|
#!/usr/bin/python3
|
|
|
|
# -*- coding: utf-8 -*-
|
2022-01-01 12:00:49 +00:00
|
|
|
# input-remapper - GUI for device specific keyboard mappings
|
2022-01-01 12:52:33 +00:00
|
|
|
# Copyright (C) 2022 sezanzeb <proxima@sezanzeb.de>
|
2020-10-26 22:45:22 +00:00
|
|
|
#
|
2022-01-01 12:00:49 +00:00
|
|
|
# This file is part of input-remapper.
|
2020-10-26 22:45:22 +00:00
|
|
|
#
|
2022-01-01 12:00:49 +00:00
|
|
|
# input-remapper is free software: you can redistribute it and/or modify
|
2020-10-26 22:45:22 +00:00
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
2022-01-01 12:00:49 +00:00
|
|
|
# input-remapper is distributed in the hope that it will be useful,
|
2020-10-26 22:45:22 +00:00
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
2022-01-01 12:00:49 +00:00
|
|
|
# along with input-remapper. If not, see <https://www.gnu.org/licenses/>.
|
2020-10-26 22:45:22 +00:00
|
|
|
|
2022-04-17 10:19:23 +00:00
|
|
|
# TODO: convert everything to use pathlib.Path
|
2020-10-26 22:45:22 +00:00
|
|
|
|
2020-11-09 22:16:30 +00:00
|
|
|
"""Path constants to be used."""
|
2020-10-26 22:45:22 +00:00
|
|
|
|
|
|
|
|
2020-11-02 00:25:43 +00:00
|
|
|
import os
|
2020-11-29 17:24:50 +00:00
|
|
|
import shutil
|
2022-07-23 08:53:41 +00:00
|
|
|
from typing import List, Union
|
2020-11-29 17:24:50 +00:00
|
|
|
|
2022-04-17 10:19:23 +00:00
|
|
|
from inputremapper.logger import logger, VERSION, IS_BETA
|
|
|
|
from inputremapper.user import USER, HOME
|
|
|
|
|
|
|
|
rel_path = ".config/input-remapper"
|
|
|
|
if IS_BETA:
|
|
|
|
rel_path = os.path.join(rel_path, f"beta_{VERSION}")
|
|
|
|
CONFIG_PATH = os.path.join(HOME, rel_path)
|
2020-11-18 19:03:37 +00:00
|
|
|
|
2020-11-02 21:06:05 +00:00
|
|
|
|
2021-02-15 17:27:25 +00:00
|
|
|
def chown(path):
|
2021-02-15 17:33:09 +00:00
|
|
|
"""Set the owner of a path to the user."""
|
2021-02-15 17:27:25 +00:00
|
|
|
try:
|
|
|
|
shutil.chown(path, user=USER, group=USER)
|
|
|
|
except LookupError:
|
2021-02-15 17:33:09 +00:00
|
|
|
# the users group was unknown in one case for whatever reason
|
2021-02-15 17:27:25 +00:00
|
|
|
shutil.chown(path, user=USER)
|
|
|
|
|
|
|
|
|
2022-07-23 08:53:41 +00:00
|
|
|
def touch(path: os.PathLike, log=True):
|
2020-11-29 17:24:50 +00:00
|
|
|
"""Create an empty file and all its parent dirs, give it to the user."""
|
2022-07-23 08:53:41 +00:00
|
|
|
if str(path).endswith("/"):
|
2021-09-26 10:44:56 +00:00
|
|
|
raise ValueError(f"Expected path to not end with a slash: {path}")
|
2021-01-07 16:15:12 +00:00
|
|
|
|
2020-11-29 17:24:50 +00:00
|
|
|
if os.path.exists(path):
|
|
|
|
return
|
|
|
|
|
|
|
|
if log:
|
|
|
|
logger.info('Creating file "%s"', path)
|
|
|
|
|
|
|
|
mkdir(os.path.dirname(path), log=False)
|
|
|
|
|
|
|
|
os.mknod(path)
|
2021-02-15 17:27:25 +00:00
|
|
|
chown(path)
|
2020-11-29 17:24:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
def mkdir(path, log=True):
|
|
|
|
"""Create a folder, give it to the user."""
|
2021-09-26 10:44:56 +00:00
|
|
|
if path == "" or path is None:
|
2021-03-21 18:15:20 +00:00
|
|
|
return
|
|
|
|
|
2020-11-29 17:24:50 +00:00
|
|
|
if os.path.exists(path):
|
|
|
|
return
|
|
|
|
|
|
|
|
if log:
|
|
|
|
logger.info('Creating dir "%s"', path)
|
|
|
|
|
|
|
|
# give all newly created folders to the user.
|
2022-01-01 12:00:49 +00:00
|
|
|
# e.g. if .config/input-remapper/mouse/ is created the latter two
|
2020-11-29 17:24:50 +00:00
|
|
|
base = os.path.split(path)[0]
|
|
|
|
mkdir(base, log=False)
|
|
|
|
|
|
|
|
os.makedirs(path)
|
2021-02-15 17:27:25 +00:00
|
|
|
chown(path)
|
2020-11-29 17:24:50 +00:00
|
|
|
|
|
|
|
|
2022-07-23 08:53:41 +00:00
|
|
|
def split_all(path: Union[os.PathLike, str]) -> List[str]:
|
|
|
|
parts = []
|
|
|
|
while True:
|
|
|
|
path, tail = os.path.split(path)
|
|
|
|
parts.append(tail)
|
|
|
|
if path == os.path.sep:
|
|
|
|
# we arrived at the root '/'
|
|
|
|
parts.append(path)
|
|
|
|
break
|
|
|
|
if not path:
|
|
|
|
# arrived at start of relative path
|
|
|
|
break
|
|
|
|
|
|
|
|
parts.reverse()
|
|
|
|
return parts
|
|
|
|
|
|
|
|
|
2021-03-21 18:15:20 +00:00
|
|
|
def remove(path):
|
2022-04-18 11:52:59 +00:00
|
|
|
"""Remove whatever is at the path."""
|
2021-03-21 18:15:20 +00:00
|
|
|
if not os.path.exists(path):
|
|
|
|
return
|
|
|
|
|
|
|
|
if os.path.isdir(path):
|
|
|
|
shutil.rmtree(path)
|
|
|
|
else:
|
|
|
|
os.remove(path)
|
|
|
|
|
|
|
|
|
2022-10-29 11:48:16 +00:00
|
|
|
def sanitize_path_component(group_name: str) -> str:
|
|
|
|
"""replace characters listed in
|
|
|
|
https://en.wikipedia.org/wiki/Filename#Reserved_characters_and_words
|
|
|
|
with an underscore
|
|
|
|
"""
|
|
|
|
for c in '/\\?%*:|"<>':
|
|
|
|
if c in group_name:
|
|
|
|
group_name = group_name.replace(c, "_")
|
|
|
|
return group_name
|
|
|
|
|
|
|
|
|
2021-04-23 09:51:21 +00:00
|
|
|
def get_preset_path(group_name=None, preset=None):
|
2020-11-18 09:33:59 +00:00
|
|
|
"""Get a path to the stored preset, or to store a preset to."""
|
2021-09-26 10:44:56 +00:00
|
|
|
presets_base = os.path.join(CONFIG_PATH, "presets")
|
2020-12-24 00:26:34 +00:00
|
|
|
|
2021-04-23 09:51:21 +00:00
|
|
|
if group_name is None:
|
2020-12-24 00:26:34 +00:00
|
|
|
return presets_base
|
2020-11-24 23:23:34 +00:00
|
|
|
|
2022-10-29 11:48:16 +00:00
|
|
|
group_name = sanitize_path_component(group_name)
|
|
|
|
|
2020-11-24 23:23:34 +00:00
|
|
|
if preset is not None:
|
|
|
|
# the extension of the preset should not be shown in the ui.
|
2020-11-25 22:36:03 +00:00
|
|
|
# if a .json extension arrives this place, it has not been
|
|
|
|
# stripped away properly prior to this.
|
2022-07-23 08:53:41 +00:00
|
|
|
if not preset.endswith(".json"):
|
|
|
|
preset = f"{preset}.json"
|
2020-11-24 23:23:34 +00:00
|
|
|
|
2020-11-18 09:33:59 +00:00
|
|
|
if preset is None:
|
2021-04-23 09:51:21 +00:00
|
|
|
return os.path.join(presets_base, group_name)
|
2020-12-24 00:26:34 +00:00
|
|
|
|
2021-04-23 09:51:21 +00:00
|
|
|
return os.path.join(presets_base, group_name, preset)
|
2020-12-24 00:26:34 +00:00
|
|
|
|
2020-11-24 23:23:34 +00:00
|
|
|
|
2020-12-24 00:26:34 +00:00
|
|
|
def get_config_path(*paths):
|
2022-04-18 11:52:59 +00:00
|
|
|
"""Get a path in ~/.config/input-remapper/."""
|
2020-12-24 00:26:34 +00:00
|
|
|
return os.path.join(CONFIG_PATH, *paths)
|