Skip to content

Commit

Permalink
new release (2.24.1)
Browse files Browse the repository at this point in the history
Signed-off-by: Jaroslav Škarvada <jskarvad@redhat.com>
  • Loading branch information
yarda committed Nov 26, 2024
1 parent 35eed3c commit 90c24ee
Show file tree
Hide file tree
Showing 8 changed files with 68 additions and 17 deletions.
14 changes: 7 additions & 7 deletions com.redhat.tuned.policy
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand Down Expand Up @@ -103,7 +103,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand All @@ -113,7 +113,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand All @@ -123,7 +123,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand Down Expand Up @@ -223,7 +223,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand Down Expand Up @@ -253,7 +253,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand All @@ -263,7 +263,7 @@
<defaults>
<allow_any>auth_admin</allow_any>
<allow_inactive>auth_admin</allow_inactive>
<allow_active>yes</allow_active>
<allow_active>auth_admin</allow_active>
</defaults>
</action>

Expand Down
10 changes: 9 additions & 1 deletion tuned.spec
Original file line number Diff line number Diff line change
Expand Up @@ -59,7 +59,7 @@

Summary: A dynamic adaptive system tuning daemon
Name: tuned
Version: 2.24.0
Version: 2.24.1
Release: 1%{?prerel1}%{?with_snapshot:.%{git_suffix}}%{?dist}
License: GPL-2.0-or-later AND CC-BY-SA-3.0
Source0: https://github.com/redhat-performance/%{name}/archive/v%{version}%{?prerel2}/%{name}-%{version}%{?prerel2}.tar.gz
Expand Down Expand Up @@ -631,6 +631,14 @@ fi
%config(noreplace) %{_sysconfdir}/tuned/ppd.conf

%changelog
* Tue Nov 26 2024 Jaroslav Škarvada <jskarvad@redhat.com> - 2.24.1-1
- new release
- fixed privileged execution of arbitrary scripts by active local user
resolves: CVE-2024-52336
- added sanity checks for API methods parameters
resolves: CVE-2024-52337
- tuned-ppd: fixed controller init to correctly set _on_battery

* Wed Aug 7 2024 Jaroslav Škarvada <jskarvad@redhat.com> - 2.24.0-1
- new release
- clear plugin repository when stopping tuning
Expand Down
4 changes: 4 additions & 0 deletions tuned/consts.py
Original file line number Diff line number Diff line change
@@ -1,4 +1,8 @@
import logging
import string

NAMES_ALLOWED_CHARS = string.ascii_letters + string.digits + " !@'+-.,/:;_$&*()%<=>?#[]{|}^~" + '"'
NAMES_MAX_LENGTH = 4096

GLOBAL_CONFIG_FILE = "/etc/tuned/tuned-main.conf"
ACTIVE_PROFILE_FILE = "/etc/tuned/active_profile"
Expand Down
35 changes: 27 additions & 8 deletions tuned/daemon/controller.py
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,8 @@ def _switch_profile(self, profile_name, manual):
def switch_profile(self, profile_name, caller = None):
if caller == "":
return (False, "Unauthorized")
if not self._cmd.is_valid_name(profile_name):
return (False, "Invalid profile_name")
return self._switch_profile(profile_name, True)

@exports.export("", "(bs)")
Expand Down Expand Up @@ -262,8 +264,8 @@ def profiles2(self, caller = None):

@exports.export("s", "(bsss)")
def profile_info(self, profile_name, caller = None):
if caller == "":
return tuple(False, "", "", "")
if caller == "" or not self._cmd.is_valid_name(profile_name):
return (False, "", "", "")
if profile_name is None or profile_name == "":
profile_name = self.active_profile()
return tuple(self._daemon.profile_loader.profile_locator.get_profile_attrs(profile_name, [consts.PROFILE_ATTR_SUMMARY, consts.PROFILE_ATTR_DESCRIPTION], [""]))
Expand Down Expand Up @@ -294,7 +296,7 @@ def get_all_plugins(self, caller = None):
dictionary -- {plugin_name: {parameter_name: default_value}}
"""
if caller == "":
return False
return {}
plugins = {}
for plugin_class in self._daemon.get_all_plugins():
plugin_name = plugin_class.__module__.split(".")[-1].split("_", 1)[1]
Expand All @@ -307,8 +309,8 @@ def get_all_plugins(self, caller = None):
@exports.export("s","s")
def get_plugin_documentation(self, plugin_name, caller = None):
"""Return docstring of plugin's class"""
if caller == "":
return False
if caller == "" or not self._cmd.is_valid_name(plugin_name):
return ""
return self._daemon.get_plugin_documentation(str(plugin_name))

@exports.export("s","a{ss}")
Expand All @@ -321,8 +323,8 @@ def get_plugin_hints(self, plugin_name, caller = None):
Return:
dictionary -- {parameter_name: hint}
"""
if caller == "":
return False
if caller == "" or not self._cmd.is_valid_name(plugin_name):
return {}
return self._daemon.get_plugin_hints(str(plugin_name))

@exports.export("s", "b")
Expand All @@ -335,7 +337,7 @@ def register_socket_signal_path(self, path, caller = None):
Return:
bool -- True on success
"""
if caller == "":
if caller == "" or not self._cmd.is_valid_name(path):
return False
if self._daemon._application and self._daemon._application._unix_socket_exporter:
self._daemon._application._unix_socket_exporter.register_signal_path(path)
Expand All @@ -349,6 +351,10 @@ def register_socket_signal_path(self, path, caller = None):
def instance_acquire_devices(self, devices, instance_name, caller = None):
if caller == "":
return (False, "Unauthorized")
if not self._cmd.is_valid_name(devices):
return (False, "Invalid devices")
if not self._cmd.is_valid_name(instance_name):
return (False, "Invalid instance_name")
found = False
for instance_target in self._daemon._unit_manager.instances:
if instance_target.name == instance_name:
Expand Down Expand Up @@ -399,6 +405,8 @@ def get_instances(self, plugin_name, caller = None):
"""
if caller == "":
return (False, "Unauthorized", [])
if not self._cmd.is_valid_name(plugin_name):
return (False, "Invalid plugin_name", [])
if plugin_name != "" and plugin_name not in self.get_all_plugins().keys():
rets = "Plugin '%s' does not exist" % plugin_name
log.error(rets)
Expand All @@ -422,6 +430,8 @@ def instance_get_devices(self, instance_name, caller = None):
"""
if caller == "":
return (False, "Unauthorized", [])
if not self._cmd.is_valid_name(instance_name):
return (False, "Invalid instance_name", [])
for instance in self._daemon._unit_manager.instances:
if instance.name == instance_name:
return (True, "OK", sorted(list(instance.processed_devices)))
Expand All @@ -444,6 +454,13 @@ def instance_create(self, plugin_name, instance_name, options, caller = None):
"""
if caller == "":
return (False, "Unauthorized")
if not self._cmd.is_valid_name(plugin_name):
return (False, "Invalid plugin_name")
if not self._cmd.is_valid_name(instance_name):
return (False, "Invalid instance_name")
for (key, value) in options.items():
if not self._cmd.is_valid_name(key) or not self._cmd.is_valid_name(value):
return (False, "Invalid options")
plugins = {p.name: p for p in self._daemon._unit_manager.plugins}
if not plugin_name in plugins.keys():
rets = "Plugin '%s' not found" % plugin_name
Expand Down Expand Up @@ -499,6 +516,8 @@ def instance_destroy(self, instance_name, caller = None):
"""
if caller == "":
return (False, "Unauthorized")
if not self._cmd.is_valid_name(instance_name):
return (False, "Invalid instance_name")
try:
instance = [i for i in self._daemon._unit_manager.instances if i.name == instance_name][0]
except IndexError:
Expand Down
12 changes: 12 additions & 0 deletions tuned/plugins/base.py
Original file line number Diff line number Diff line change
Expand Up @@ -213,6 +213,14 @@ def _instance_pre_static(self, instance, enabling):
def _instance_post_static(self, instance, enabling):
pass

def _safe_script_path(self, path):
path = os.path.realpath(path)
profile_paths = self._global_cfg.get_list(consts.CFG_PROFILE_DIRS, consts.CFG_DEF_PROFILE_DIRS)
for p in profile_paths:
if path.startswith(p):
return True
return False

def _call_device_script(self, instance, script, op, devices, rollback = consts.ROLLBACK_SOFT):
if script is None:
return None
Expand All @@ -223,6 +231,10 @@ def _call_device_script(self, instance, script, op, devices, rollback = consts.R
log.error("Relative paths cannot be used in script_pre or script_post. " \
+ "Use ${i:PROFILE_DIR}.")
return False
if not self._safe_script_path(script):
log.error("Paths outside of the profile directories cannot be used in the " \
+ "script_pre or script_post, ignoring script: '%s'" % script)
return False
dir_name = os.path.dirname(script)
ret = True
for dev in devices:
Expand Down
4 changes: 4 additions & 0 deletions tuned/plugins/plugin_script.py
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,10 @@ def _call_scripts(self, scripts, arguments):
for script in scripts:
environ = os.environ
environ.update(self._variables.get_env())
if not self._safe_script_path(script):
log.error("Paths outside of the profile directories cannot be used in the script, " \
+ "ignoring script: '%s'." % script)
continue
log.info("calling script '%s' with arguments '%s'" % (script, str(arguments)))
log.debug("using environment '%s'" % str(list(environ.items())))
try:
Expand Down
4 changes: 4 additions & 0 deletions tuned/utils/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -548,3 +548,7 @@ def tr(self, text, source_chars, dest_chars):
import string
trans = string.maketrans(source_chars, dest_chars)
return text.translate(trans)

# Checks if name contains only valid characters and has valid length or is empty string or None
def is_valid_name(self, name):
return not name or (all(c in consts.NAMES_ALLOWED_CHARS for c in name) and len(name) <= consts.NAMES_MAX_LENGTH)
2 changes: 1 addition & 1 deletion tuned/version.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
TUNED_VERSION_MAJOR = 2
TUNED_VERSION_MINOR = 24
TUNED_VERSION_PATCH = 0
TUNED_VERSION_PATCH = 1

TUNED_VERSION_STR = "%d.%d.%d" % (TUNED_VERSION_MAJOR, TUNED_VERSION_MINOR, TUNED_VERSION_PATCH)

0 comments on commit 90c24ee

Please sign in to comment.