Scare Crowe d2ebfd0519 QortalOS Titan 5.60.12
Screw the description like that inbred T3Q
2022-03-05 21:17:59 +05:00

230 lines
6.4 KiB
C++

/*
A helper that's run using KAuth and does the system modifications.
SPDX-FileCopyrightText: 1998 Luca Montecchiani <m.luca@usa.net>
SPDX-License-Identifier: GPL-2.0-or-later
*/
#include "helper.h"
#include <config-workspace.h>
#include <sys/time.h>
#include <time.h>
#include <unistd.h>
#include <KComponentData>
#include <KConfig>
#include <KConfigGroup>
#include <KProcess>
#include <KStandardDirs>
#include <QDebug>
#include <QFile>
#if defined(USE_SOLARIS)
#include <KTemporaryFile>
#include <sys/param.h>
#include <sys/stat.h>
#include <sys/types.h>
#endif
// We cannot rely on the $PATH environment variable, because D-Bus activation
// clears it. So we have to use a reasonable default.
static const QString exePath = QStringLiteral("/usr/sbin:/usr/bin:/sbin:/bin");
int ClockHelper::ntp(const QStringList &ntpServers, bool ntpEnabled)
{
int ret = 0;
// write to the system config file
QFile config_file(KDE_CONFDIR "/kcmclockrc");
if (!config_file.exists()) {
config_file.open(QIODevice::WriteOnly);
config_file.close();
config_file.setPermissions(QFile::ReadOwner | QFile::WriteOwner | QFile::ReadGroup | QFile::ReadOther);
}
KConfig _config(config_file.fileName(), KConfig::SimpleConfig);
KConfigGroup config(&_config, "NTP");
config.writeEntry("servers", ntpServers);
config.writeEntry("enabled", ntpEnabled);
QString ntpUtility = QStandardPaths::findExecutable(QStringLiteral("ntpdate"));
if (ntpUtility.isEmpty()) {
ntpUtility = QStandardPaths::findExecutable(QStringLiteral("rdate"));
}
if (ntpEnabled && !ntpUtility.isEmpty()) {
// NTP Time setting
QString timeServer = ntpServers.first();
if (timeServer.indexOf(QRegExp(QStringLiteral(".*\\(.*\\)$"))) != -1) {
timeServer.remove(QRegExp(QStringLiteral(".*\\(")));
timeServer.remove(QRegExp(QStringLiteral("\\).*")));
// Would this be better?: s/^.*\(([^)]*)\).*$/\1/
}
KProcess proc;
proc << ntpUtility << timeServer;
if (proc.execute() != 0) {
ret |= NTPError;
} else {
toHwclock();
}
} else if (ntpEnabled) {
ret |= NTPError;
}
return ret;
}
int ClockHelper::date(const QString &newdate, const QString &olddate)
{
struct timeval tv;
tv.tv_sec = newdate.toULong() - olddate.toULong() + time(nullptr);
tv.tv_usec = 0;
if (settimeofday(&tv, nullptr)) {
return DateError;
}
toHwclock();
return 0;
}
// on non-Solaris systems which do not use /etc/timezone?
int ClockHelper::tz(const QString &selectedzone)
{
int ret = 0;
// only allow letters, numbers hyphen underscore plus and forward slash
// allowed pattern taken from time-util.c in systemd
if (!QRegExp(QStringLiteral("[a-zA-Z0-9-_+/]*")).exactMatch(selectedzone)) {
return ret;
}
QString val;
#if defined(USE_SOLARIS) // MARCO
KTemporaryFile tf;
tf.setPrefix("kde-tzone");
tf.open();
QTextStream ts(&tf);
QFile fTimezoneFile(INITFILE);
bool updatedFile = false;
if (fTimezoneFile.open(QIODevice::ReadOnly)) {
bool found = false;
QTextStream is(&fTimezoneFile);
for (QString line = is.readLine(); !line.isNull(); line = is.readLine()) {
if (line.find("TZ=") == 0) {
ts << "TZ=" << selectedzone << endl;
found = true;
} else {
ts << line << endl;
}
}
if (!found) {
ts << "TZ=" << selectedzone << endl;
}
updatedFile = true;
fTimezoneFile.close();
}
if (updatedFile) {
ts.device()->reset();
fTimezoneFile.remove();
if (fTimezoneFile.open(QIODevice::WriteOnly | QIODevice::Truncate)) {
QTextStream os(&fTimezoneFile);
for (QString line = ts->readLine(); !line.isNull(); line = ts->readLine()) {
os << line << endl;
}
fchmod(fTimezoneFile.handle(), S_IXUSR | S_IRUSR | S_IRGRP | S_IXGRP | S_IROTH | S_IXOTH);
fTimezoneFile.close();
}
}
val = selectedzone;
#else
QString tz = "/usr/share/zoneinfo/" + selectedzone;
if (QFile::exists(tz)) { // make sure the new TZ really exists
QFile::remove(QStringLiteral("/etc/localtime"));
} else {
return TimezoneError;
}
if (!QFile::link(tz, QStringLiteral("/etc/localtime"))) { // fail if we can't setup the new timezone
return TimezoneError;
}
QFile fTimezoneFile(QStringLiteral("/etc/timezone"));
if (fTimezoneFile.exists() && fTimezoneFile.open(QIODevice::WriteOnly | QIODevice::Truncate)) {
QTextStream t(&fTimezoneFile);
t << selectedzone;
fTimezoneFile.close();
}
#endif // !USE_SOLARIS
val = ':' + selectedzone;
setenv("TZ", val.toLocal8Bit().constData(), 1);
tzset();
return ret;
}
int ClockHelper::tzreset()
{
#if !defined(USE_SOLARIS) // Do not update the System!
unlink("/etc/timezone");
unlink("/etc/localtime");
setenv("TZ", "", 1);
tzset();
#endif // !USE SOLARIS
return 0;
}
void ClockHelper::toHwclock()
{
QString hwclock = QStandardPaths::findExecutable(QStringLiteral("hwclock"), exePath.split(QLatin1Char(':')));
if (!hwclock.isEmpty()) {
KProcess::execute(hwclock, QStringList() << QStringLiteral("--systohc"));
}
}
ActionReply ClockHelper::save(const QVariantMap &args)
{
bool _ntp = args.value(QStringLiteral("ntp")).toBool();
bool _date = args.value(QStringLiteral("date")).toBool();
bool _tz = args.value(QStringLiteral("tz")).toBool();
bool _tzreset = args.value(QStringLiteral("tzreset")).toBool();
int ret = 0; // error code
// The order here is important
if (_ntp)
ret |= ntp(args.value(QStringLiteral("ntpServers")).toStringList(), args.value(QStringLiteral("ntpEnabled")).toBool());
if (_date)
ret |= date(args.value(QStringLiteral("newdate")).toString(), args.value(QStringLiteral("olddate")).toString());
if (_tz)
ret |= tz(args.value(QStringLiteral("tzone")).toString());
if (_tzreset)
ret |= tzreset();
if (ret == 0) {
return ActionReply::SuccessReply();
} else {
ActionReply reply(ActionReply::HelperErrorReply());
reply.setErrorCode(static_cast<ActionReply::Error>(ret));
return reply;
}
}
KAUTH_HELPER_MAIN("org.kde.kcontrol.kcmclock", ClockHelper)