那里的许多 Linux 发行版都需要超级用户权限才能执行 shutdown 或 halt,但是,如果您坐在计算机上,为什么不用 root 就可以关闭它?你打开一个菜单,点击Shutdown,它会在你没有变成root的情况下关闭,对吧?
嗯...这背后的基本原理是,如果您可以物理访问计算机,则几乎可以拉电源线并将其关闭,所以现在,许多发行版都允许电源-off 尽管可以通过dbus 访问本地系统总线。 dbus(或者通过它公开的服务)有问题吗?它在不断变化。我建议安装一个 dbus 查看器工具,例如 D-feet(请注意:它仍然很难可视化,但它可能会有所帮助)
看看these Dbus shutdown scripts。
如果您的分发版中仍有 HAL(即将被弃用),请尝试以下操作:
import dbus
sys_bus = dbus.SystemBus()
hal_srvc = sys_bus.get_object('org.freedesktop.Hal',
'/org/freedesktop/Hal/devices/computer')
pwr_mgmt = dbus.Interface(hal_srvc,
'org.freedesktop.Hal.Device.SystemPowerManagement')
shutdown_method = pwr_mgmt.get_dbus_method("Shutdown")
shutdown_method()
这适用于 Ubuntu 12.04(我刚刚关闭计算机以确保它正常工作)。如果你有更新的东西......好吧,它可能无法正常工作。这是这种方法的缺点:它是非常特定于分布的。
您可能需要安装 dbus-python 软件包才能使其正常工作 (http://dbus.freedesktop.org/doc/dbus-python/doc/tutorial.html)
更新 1:
我一直在做一些研究,看起来这是通过ConsoleKit 在较新的 Ubuntu 版本中完成的。我已经在我的 Ubuntu 12.04(它具有已弃用的 HAL 和更新的 ConsoleKit)中测试了以下代码,它确实关闭了我的计算机:
>>> import dbus
>>> sys_bus = dbus.SystemBus()
>>> ck_srv = sys_bus.get_object('org.freedesktop.ConsoleKit',
'/org/freedesktop/ConsoleKit/Manager')
>>> ck_iface = dbus.Interface(ck_srv, 'org.freedesktop.ConsoleKit.Manager')
>>> stop_method = ck_iface.get_dbus_method("Stop")
>>> stop_method()
更新 2:
可能为什么你可以做到这一点而不是root 值得更广泛的解释。让我们关注较新的ConsoleKit(HAL 更加复杂和混乱,恕我直言)。
ConsoleKit 是在您的系统中以root 运行的服务:
borrajax@borrajax:/tmp$ ps aux|grep console-kit
root 1590 0.0 0.0 1043056 3876 ? Sl Dec05 0:00 /usr/sbin/console-kit-daemon --no-daemon
现在,d-bus 只是一个消息传递系统。您有一个服务,例如 ConsoleKit,它向d-bus 公开了一个接口。公开的方法之一是Stop(如上所示)。 ConsoleKit 的权限由 PolKit 控制,它(尽管基于常规 Linux 权限)为 “谁可以做什么”。例如,PolKit 可以说 “如果用户登录到计算机,则允许他做某事。如果它是远程连接的,则不要。” .如果PolKit 确定您的用户可以调用ConsoleKit 的Stop 方法,则该请求将通过(或通过)@ 987654351@ 到 ConsoleKit(随后会关闭您的计算机,因为它可以...因为它值得...因为它是 root)
进一步阅读:
总结一下:如果没有root,您将无法关闭计算机。但是您可以告诉以root 运行的服务为您关闭系统。
更新 3:
2021 年 12 月,也就是原始答案写完七年后,我不得不再次这样做。这一次,在 Ubuntu 18.04 中。
不出所料,事情似乎发生了一些变化:
所以我们仍然可以使用非特权脚本关闭机器:
#! /usr/bin/python3
from pydbus import SystemBus
bus = SystemBus()
proxy = bus.get('org.freedesktop.login1', '/org/freedesktop/login1')
if proxy.CanPowerOff() == 'yes':
proxy.PowerOff(False) # False for 'NOT interactive'
更新 3.1:
它看起来不像我想的那样新 X-D
@Roeften 在同一线程中已经有an answer。
奖金:
我在您的一个 cmet 中读到,您想在完成一项耗时的任务后关闭计算机以防止其过热...您是否知道您可以可能在给定的时间打开它使用 RTC 的时间? (见this 和this)很酷,嗯? (当我发现我可以做到这一点时,我非常兴奋......):-D