mirror of
https://github.com/Dasharo/systemd.git
synced 2026-03-06 15:02:31 -08:00
388 lines
13 KiB
C
388 lines
13 KiB
C
/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/
|
|
|
|
/***
|
|
This file is part of systemd.
|
|
|
|
Copyright 2013 Zbigniew Jędrzejewski-Szmek <zbyszek@in.waw.pl>
|
|
|
|
systemd is free software; you can redistribute it and/or modify it
|
|
under the terms of the GNU Lesser General Public License as published by
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
|
(at your option) any later version.
|
|
|
|
systemd is distributed in the hope that it will be useful, but
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public License
|
|
along with systemd; If not, see <http://www.gnu.org/licenses/>.
|
|
***/
|
|
|
|
#define PY_SSIZE_T_CLEAN
|
|
#pragma GCC diagnostic push
|
|
#pragma GCC diagnostic ignored "-Wredundant-decls"
|
|
#include <Python.h>
|
|
#pragma GCC diagnostic pop
|
|
|
|
#include "systemd/sd-login.h"
|
|
#include "pyutil.h"
|
|
#include "util.h"
|
|
#include "strv.h"
|
|
|
|
PyDoc_STRVAR(module__doc__,
|
|
"Python interface to the libsystemd-login library."
|
|
);
|
|
|
|
#define helper(name) \
|
|
static PyObject* name(PyObject *self, PyObject *args) { \
|
|
_cleanup_strv_free_ char **list = NULL; \
|
|
int r; \
|
|
PyObject *ans; \
|
|
\
|
|
assert(args == NULL); \
|
|
\
|
|
r = sd_get_##name(&list); \
|
|
if (r < 0) { \
|
|
errno = -r; \
|
|
return PyErr_SetFromErrno(PyExc_IOError); \
|
|
} \
|
|
\
|
|
ans = PyList_New(r); \
|
|
if (!ans) \
|
|
return NULL; \
|
|
\
|
|
for (r--; r >= 0; r--) { \
|
|
PyObject *s = unicode_FromString(list[r]); \
|
|
if (!s) { \
|
|
Py_DECREF(ans); \
|
|
return NULL; \
|
|
} \
|
|
\
|
|
PyList_SetItem(ans, r, s); \
|
|
} \
|
|
\
|
|
return ans; \
|
|
}
|
|
|
|
helper(seats)
|
|
helper(sessions)
|
|
helper(machine_names)
|
|
#undef helper
|
|
|
|
static PyObject* uids(PyObject *self, PyObject *args) {
|
|
_cleanup_free_ uid_t *list = NULL;
|
|
int r;
|
|
PyObject *ans;
|
|
|
|
assert(args == NULL);
|
|
|
|
r = sd_get_uids(&list);
|
|
if (r < 0) {
|
|
errno = -r;
|
|
return PyErr_SetFromErrno(PyExc_IOError);
|
|
}
|
|
|
|
ans = PyList_New(r);
|
|
if (!ans)
|
|
return NULL;
|
|
|
|
for (r--; r >= 0; r--) {
|
|
PyObject *s = long_FromLong(list[r]);
|
|
if (!s) {
|
|
Py_DECREF(ans);
|
|
return NULL;
|
|
}
|
|
|
|
PyList_SetItem(ans, r, s);
|
|
}
|
|
|
|
return ans;
|
|
}
|
|
|
|
PyDoc_STRVAR(seats__doc__,
|
|
"seats() -> list\n\n"
|
|
"Returns a list of currently available local seats.\n"
|
|
"Wraps sd_get_seats(3)."
|
|
);
|
|
|
|
PyDoc_STRVAR(sessions__doc__,
|
|
"sessions() -> list\n\n"
|
|
"Returns a list of current login sessions.\n"
|
|
"Wraps sd_get_sessions(3)."
|
|
);
|
|
|
|
PyDoc_STRVAR(machine_names__doc__,
|
|
"machine_names() -> list\n\n"
|
|
"Returns a list of currently running virtual machines\n"
|
|
"and containers on the system.\n"
|
|
"Wraps sd_get_machine_names(3)."
|
|
);
|
|
|
|
PyDoc_STRVAR(uids__doc__,
|
|
"uids() -> list\n\n"
|
|
"Returns a list of uids of users who currently have login sessions.\n"
|
|
"Wraps sd_get_uids(3)."
|
|
);
|
|
|
|
static PyMethodDef methods[] = {
|
|
{ "seats", seats, METH_NOARGS, seats__doc__},
|
|
{ "sessions", sessions, METH_NOARGS, sessions__doc__},
|
|
{ "machine_names", machine_names, METH_NOARGS, machine_names__doc__},
|
|
{ "uids", uids, METH_NOARGS, uids__doc__},
|
|
{} /* Sentinel */
|
|
};
|
|
|
|
|
|
typedef struct {
|
|
PyObject_HEAD
|
|
sd_login_monitor *monitor;
|
|
} Monitor;
|
|
static PyTypeObject MonitorType;
|
|
|
|
static void Monitor_dealloc(Monitor* self)
|
|
{
|
|
sd_login_monitor_unref(self->monitor);
|
|
Py_TYPE(self)->tp_free((PyObject*)self);
|
|
}
|
|
|
|
PyDoc_STRVAR(Monitor__doc__,
|
|
"Monitor([category]) -> ...\n\n"
|
|
"Monitor may be used to monitor login sessions, users, seats,\n"
|
|
"and virtual machines/containers. Monitor provides a file\n"
|
|
"descriptor which can be integrated in an external event loop.\n"
|
|
"See man:sd_login_monitor_new(3) for the details about what\n"
|
|
"can be monitored.");
|
|
static int Monitor_init(Monitor *self, PyObject *args, PyObject *keywds)
|
|
{
|
|
const char *category = NULL;
|
|
int r;
|
|
|
|
static const char* const kwlist[] = {"category", NULL};
|
|
if (!PyArg_ParseTupleAndKeywords(args, keywds, "|z:__init__", (char**) kwlist,
|
|
&category))
|
|
return -1;
|
|
|
|
Py_BEGIN_ALLOW_THREADS
|
|
r = sd_login_monitor_new(category, &self->monitor);
|
|
Py_END_ALLOW_THREADS
|
|
|
|
return set_error(r, NULL, "Invalid category");
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_fileno__doc__,
|
|
"fileno() -> int\n\n"
|
|
"Get a file descriptor to poll for events.\n"
|
|
"This method wraps sd_login_monitor_get_fd(3).");
|
|
static PyObject* Monitor_fileno(Monitor *self, PyObject *args)
|
|
{
|
|
int fd = sd_login_monitor_get_fd(self->monitor);
|
|
set_error(fd, NULL, NULL);
|
|
if (fd < 0)
|
|
return NULL;
|
|
return long_FromLong(fd);
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_get_events__doc__,
|
|
"get_events() -> int\n\n"
|
|
"Returns a mask of poll() events to wait for on the file\n"
|
|
"descriptor returned by .fileno().\n\n"
|
|
"See man:sd_login_monitor_get_events(3) for further discussion.");
|
|
static PyObject* Monitor_get_events(Monitor *self, PyObject *args)
|
|
{
|
|
int r = sd_login_monitor_get_events(self->monitor);
|
|
set_error(r, NULL, NULL);
|
|
if (r < 0)
|
|
return NULL;
|
|
return long_FromLong(r);
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_get_timeout__doc__,
|
|
"get_timeout() -> int or None\n\n"
|
|
"Returns a timeout value for usage in poll(), the time since the\n"
|
|
"epoch of clock_gettime(2) in microseconds, or None if no timeout\n"
|
|
"is necessary.\n\n"
|
|
"The return value must be converted to a relative timeout in\n"
|
|
"milliseconds if it is to be used as an argument for poll().\n"
|
|
"See man:sd_login_monitor_get_timeout(3) for further discussion.");
|
|
static PyObject* Monitor_get_timeout(Monitor *self, PyObject *args)
|
|
{
|
|
int r;
|
|
uint64_t t;
|
|
|
|
r = sd_login_monitor_get_timeout(self->monitor, &t);
|
|
set_error(r, NULL, NULL);
|
|
if (r < 0)
|
|
return NULL;
|
|
|
|
if (t == (uint64_t) -1)
|
|
Py_RETURN_NONE;
|
|
|
|
assert_cc(sizeof(unsigned long long) == sizeof(t));
|
|
return PyLong_FromUnsignedLongLong(t);
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_get_timeout_ms__doc__,
|
|
"get_timeout_ms() -> int\n\n"
|
|
"Returns a timeout value suitable for usage in poll(), the value\n"
|
|
"returned by .get_timeout() converted to relative ms, or -1 if\n"
|
|
"no timeout is necessary.");
|
|
static PyObject* Monitor_get_timeout_ms(Monitor *self, PyObject *args)
|
|
{
|
|
int r;
|
|
uint64_t t;
|
|
|
|
r = sd_login_monitor_get_timeout(self->monitor, &t);
|
|
set_error(r, NULL, NULL);
|
|
if (r < 0)
|
|
return NULL;
|
|
|
|
return absolute_timeout(t);
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_close__doc__,
|
|
"close() -> None\n\n"
|
|
"Free resources allocated by this Monitor object.\n"
|
|
"This method invokes sd_login_monitor_unref().\n"
|
|
"See man:sd_login_monitor_unref(3).");
|
|
static PyObject* Monitor_close(Monitor *self, PyObject *args)
|
|
{
|
|
assert(self);
|
|
assert(!args);
|
|
|
|
sd_login_monitor_unref(self->monitor);
|
|
self->monitor = NULL;
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor_flush__doc__,
|
|
"flush() -> None\n\n"
|
|
"Reset the wakeup state of the monitor object.\n"
|
|
"This method invokes sd_login_monitor_flush().\n"
|
|
"See man:sd_login_monitor_flush(3).");
|
|
static PyObject* Monitor_flush(Monitor *self, PyObject *args)
|
|
{
|
|
assert(self);
|
|
assert(!args);
|
|
|
|
Py_BEGIN_ALLOW_THREADS
|
|
sd_login_monitor_flush(self->monitor);
|
|
Py_END_ALLOW_THREADS
|
|
Py_RETURN_NONE;
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor___enter____doc__,
|
|
"__enter__() -> self\n\n"
|
|
"Part of the context manager protocol.\n"
|
|
"Returns self.\n");
|
|
static PyObject* Monitor___enter__(PyObject *self, PyObject *args)
|
|
{
|
|
assert(self);
|
|
assert(!args);
|
|
|
|
Py_INCREF(self);
|
|
return self;
|
|
}
|
|
|
|
|
|
PyDoc_STRVAR(Monitor___exit____doc__,
|
|
"__exit__(type, value, traceback) -> None\n\n"
|
|
"Part of the context manager protocol.\n"
|
|
"Closes the monitor..\n");
|
|
static PyObject* Monitor___exit__(Monitor *self, PyObject *args)
|
|
{
|
|
return Monitor_close(self, args);
|
|
}
|
|
|
|
|
|
static PyMethodDef Monitor_methods[] = {
|
|
{"fileno", (PyCFunction) Monitor_fileno, METH_NOARGS, Monitor_fileno__doc__},
|
|
{"get_events", (PyCFunction) Monitor_get_events, METH_NOARGS, Monitor_get_events__doc__},
|
|
{"get_timeout", (PyCFunction) Monitor_get_timeout, METH_NOARGS, Monitor_get_timeout__doc__},
|
|
{"get_timeout_ms", (PyCFunction) Monitor_get_timeout_ms, METH_NOARGS, Monitor_get_timeout_ms__doc__},
|
|
{"close", (PyCFunction) Monitor_close, METH_NOARGS, Monitor_close__doc__},
|
|
{"flush", (PyCFunction) Monitor_flush, METH_NOARGS, Monitor_flush__doc__},
|
|
{"__enter__", (PyCFunction) Monitor___enter__, METH_NOARGS, Monitor___enter____doc__},
|
|
{"__exit__", (PyCFunction) Monitor___exit__, METH_VARARGS, Monitor___exit____doc__},
|
|
{} /* Sentinel */
|
|
};
|
|
|
|
static PyTypeObject MonitorType = {
|
|
PyVarObject_HEAD_INIT(NULL, 0)
|
|
.tp_name = "login.Monitor",
|
|
.tp_basicsize = sizeof(Monitor),
|
|
.tp_dealloc = (destructor) Monitor_dealloc,
|
|
.tp_flags = Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE,
|
|
.tp_doc = Monitor__doc__,
|
|
.tp_methods = Monitor_methods,
|
|
.tp_init = (initproc) Monitor_init,
|
|
.tp_new = PyType_GenericNew,
|
|
};
|
|
|
|
|
|
#pragma GCC diagnostic push
|
|
#pragma GCC diagnostic ignored "-Wmissing-prototypes"
|
|
|
|
#if PY_MAJOR_VERSION < 3
|
|
|
|
PyMODINIT_FUNC initlogin(void) {
|
|
PyObject *m;
|
|
|
|
if (PyType_Ready(&MonitorType) < 0)
|
|
return;
|
|
|
|
m = Py_InitModule3("login", methods, module__doc__);
|
|
if (m == NULL)
|
|
return;
|
|
|
|
PyModule_AddStringConstant(m, "__version__", PACKAGE_VERSION);
|
|
|
|
Py_INCREF(&MonitorType);
|
|
PyModule_AddObject(m, "Monitor", (PyObject *) &MonitorType);
|
|
}
|
|
#else
|
|
|
|
static struct PyModuleDef module = {
|
|
PyModuleDef_HEAD_INIT,
|
|
"login", /* name of module */
|
|
module__doc__, /* module documentation, may be NULL */
|
|
-1, /* size of per-interpreter state of the module */
|
|
methods
|
|
};
|
|
|
|
PyMODINIT_FUNC PyInit_login(void) {
|
|
PyObject *m;
|
|
|
|
if (PyType_Ready(&MonitorType) < 0)
|
|
return NULL;
|
|
|
|
m = PyModule_Create(&module);
|
|
if (m == NULL)
|
|
return NULL;
|
|
|
|
if (PyModule_AddStringConstant(m, "__version__", PACKAGE_VERSION)) {
|
|
Py_DECREF(m);
|
|
return NULL;
|
|
}
|
|
|
|
Py_INCREF(&MonitorType);
|
|
if (PyModule_AddObject(m, "Monitor", (PyObject *) &MonitorType)) {
|
|
Py_DECREF(&MonitorType);
|
|
Py_DECREF(m);
|
|
return NULL;
|
|
}
|
|
|
|
return m;
|
|
}
|
|
|
|
#endif
|
|
|
|
#pragma GCC diagnostic pop
|