first working code
This commit is contained in:
parent
63fb8982d1
commit
8bb5798385
7 changed files with 445 additions and 3 deletions
80
SMSubProcess.cpp
Normal file
80
SMSubProcess.cpp
Normal file
|
@ -0,0 +1,80 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* smsub Server Manager Subprocess
|
||||||
|
* Copyright (C) 2020 Syping
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#include <QCoreApplication>
|
||||||
|
#include <QTextStream>
|
||||||
|
#include "SMSubProcess.h"
|
||||||
|
#include "smsub.h"
|
||||||
|
|
||||||
|
SMSubProcess::SMSubProcess(const QString &executable, const QStringList &arguments, const QString &workingDirectory)
|
||||||
|
{
|
||||||
|
process.setProgram(executable);
|
||||||
|
process.setArguments(arguments);
|
||||||
|
process.setWorkingDirectory(workingDirectory);
|
||||||
|
process.setProcessChannelMode(QProcess::MergedChannels);
|
||||||
|
|
||||||
|
QObject::connect(&process, SIGNAL(readyRead()), this, SLOT(readyRead()));
|
||||||
|
QObject::connect(&process, SIGNAL(finished(int)), this, SLOT(processExit(int)));
|
||||||
|
QObject::connect(&process, SIGNAL(errorOccurred(QProcess::ProcessError)), this, SLOT(processError(QProcess::ProcessError)));
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::start()
|
||||||
|
{
|
||||||
|
process.start(QIODevice::ReadWrite);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::readyRead()
|
||||||
|
{
|
||||||
|
while (process.canReadLine()) {
|
||||||
|
const QByteArray readData = process.readLine().trimmed();
|
||||||
|
emit outputWritten(readData + '\n');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::processExit(int exitCode)
|
||||||
|
{
|
||||||
|
emit processStopped();
|
||||||
|
QCoreApplication::exit(exitCode);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::processError(QProcess::ProcessError error)
|
||||||
|
{
|
||||||
|
if (likely(error == QProcess::FailedToStart)) {
|
||||||
|
QTextStream(stderr) << "Process failed to start!" << endl;
|
||||||
|
QCoreApplication::exit(1);
|
||||||
|
}
|
||||||
|
else if (error == QProcess::UnknownError) {
|
||||||
|
QTextStream(stderr) << "Unknown error occurred!" << endl;
|
||||||
|
QCoreApplication::exit(1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::killProcess()
|
||||||
|
{
|
||||||
|
process.kill();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::stopProcess()
|
||||||
|
{
|
||||||
|
process.terminate();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubProcess::writeInput(const QByteArray &input)
|
||||||
|
{
|
||||||
|
process.write(input);
|
||||||
|
}
|
51
SMSubProcess.h
Normal file
51
SMSubProcess.h
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* smsub Server Manager Subprocess
|
||||||
|
* Copyright (C) 2020 Syping
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef SMSUBPROCESS_H
|
||||||
|
#define SMSUBPROCESS_H
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QProcess>
|
||||||
|
|
||||||
|
class SMSubProcess : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
SMSubProcess(const QString &executable, const QStringList &arguments, const QString& workingDirectory);
|
||||||
|
void start();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QProcess process;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void killProcess();
|
||||||
|
void stopProcess();
|
||||||
|
void writeInput(const QByteArray &input);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void readyRead();
|
||||||
|
void processExit(int exitCode);
|
||||||
|
void processError(QProcess::ProcessError error);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void outputWritten(const QByteArray &output);
|
||||||
|
void processStopped();
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // SMSUBPROCESS_H
|
83
SMSubServer.cpp
Normal file
83
SMSubServer.cpp
Normal file
|
@ -0,0 +1,83 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* smsub Server Manager Subprocess
|
||||||
|
* Copyright (C) 2020 Syping
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#include "SMSubServer.h"
|
||||||
|
#include "smsub.h"
|
||||||
|
|
||||||
|
SMSubServer::SMSubServer(const QString &socket)
|
||||||
|
{
|
||||||
|
listen(socket);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubServer::incomingConnection(quintptr socketDescriptor)
|
||||||
|
{
|
||||||
|
QLocalSocket *localSocket = new QLocalSocket();
|
||||||
|
localSocket->setSocketDescriptor(socketDescriptor);
|
||||||
|
QObject::connect(localSocket, SIGNAL(readyRead()), this, SLOT(readyRead()));
|
||||||
|
QObject::connect(localSocket, SIGNAL(disconnected()), this, SLOT(deleteSocket()));
|
||||||
|
sockets << localSocket;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubServer::readyRead()
|
||||||
|
{
|
||||||
|
QLocalSocket *socket = (QLocalSocket*)sender();
|
||||||
|
while (socket->canReadLine()) {
|
||||||
|
const QByteArray readData = socket->readLine().trimmed();
|
||||||
|
if (readData.startsWith("+log")) {
|
||||||
|
socket->setProperty("ReceiveLog", true);
|
||||||
|
}
|
||||||
|
else if (readData.startsWith("-log")) {
|
||||||
|
socket->setProperty("ReceiveLog", false);
|
||||||
|
}
|
||||||
|
else if (readData.startsWith("kill")) {
|
||||||
|
emit killRequested();
|
||||||
|
}
|
||||||
|
else if (readData.startsWith("stop")) {
|
||||||
|
emit stopRequested();
|
||||||
|
}
|
||||||
|
else if (readData.startsWith("wl")) {
|
||||||
|
emit inputWritten(readData.mid(3) + '\n');
|
||||||
|
}
|
||||||
|
else if (readData.startsWith("w")) {
|
||||||
|
emit inputWritten(readData.mid(2));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubServer::deleteSocket()
|
||||||
|
{
|
||||||
|
QLocalSocket *socket = (QLocalSocket*)sender();
|
||||||
|
sockets.removeAll(socket);
|
||||||
|
socket->deleteLater();
|
||||||
|
}
|
||||||
|
|
||||||
|
void SMSubServer::writeOutput(const QByteArray &output)
|
||||||
|
{
|
||||||
|
QVector<QLocalSocket*>::const_iterator it = sockets.constBegin();
|
||||||
|
QVector<QLocalSocket*>::const_iterator end = sockets.constEnd();
|
||||||
|
while (it != end) {
|
||||||
|
const QVariant variant = (*it)->property("ReceiveLog");
|
||||||
|
if (unlikely(variant.type() == QVariant::Bool)) {
|
||||||
|
bool receiveLog = variant.toBool();
|
||||||
|
if (likely(receiveLog)) {
|
||||||
|
(*it)->write(output);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
it++;
|
||||||
|
}
|
||||||
|
}
|
49
SMSubServer.h
Normal file
49
SMSubServer.h
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* smsub Server Manager Subprocess
|
||||||
|
* Copyright (C) 2020 Syping
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef SMSUBSERVER_H
|
||||||
|
#define SMSUBSERVER_H
|
||||||
|
|
||||||
|
#include <QLocalServer>
|
||||||
|
#include <QLocalSocket>
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
class SMSubServer : public QLocalServer
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
public:
|
||||||
|
SMSubServer(const QString &socket);
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void writeOutput(const QByteArray &output);
|
||||||
|
|
||||||
|
private slots:
|
||||||
|
void incomingConnection(quintptr socketDescriptor);
|
||||||
|
void deleteSocket();
|
||||||
|
void readyRead();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QVector<QLocalSocket*> sockets;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void inputWritten(const QByteArray &input);
|
||||||
|
void killRequested();
|
||||||
|
void stopRequested();
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif // SMSUBSERVER_H
|
119
main.cpp
119
main.cpp
|
@ -12,13 +12,23 @@
|
||||||
* this list of conditions and the following disclaimer in the documentation
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
* and/or other materials provided with the distribution.
|
* and/or other materials provided with the distribution.
|
||||||
*
|
*
|
||||||
* 3. This software is provided as-is, no warranties are given, we are not
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
* responsible for anything with use of the software, you are self responsible.
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
|
|
||||||
#include <QCommandLineParser>
|
#include <QCommandLineParser>
|
||||||
#include <QCommandLineOption>
|
#include <QCommandLineOption>
|
||||||
#include <QCoreApplication>
|
#include <QCoreApplication>
|
||||||
|
#include <QJsonDocument>
|
||||||
|
#include <QJsonObject>
|
||||||
|
#include <QTextStream>
|
||||||
|
#include <QJsonValue>
|
||||||
|
#include <QJsonArray>
|
||||||
|
#include <QFileInfo>
|
||||||
|
#include <QFile>
|
||||||
|
#include "SMSubProcess.h"
|
||||||
|
#include "SMSubServer.h"
|
||||||
|
#include "smsub.h"
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
int main(int argc, char *argv[])
|
||||||
{
|
{
|
||||||
|
@ -33,5 +43,112 @@ int main(int argc, char *argv[])
|
||||||
QCommandLineOption processManifest("json", "JSON process manifest.", "json");
|
QCommandLineOption processManifest("json", "JSON process manifest.", "json");
|
||||||
commandLineParser.addOption(processManifest);
|
commandLineParser.addOption(processManifest);
|
||||||
|
|
||||||
|
QCommandLineOption processExecutable(QStringList() << "exec" << "executable", "Process executable to run.", "exec");
|
||||||
|
commandLineParser.addOption(processExecutable);
|
||||||
|
|
||||||
|
QCommandLineOption processArguments(QStringList() << "args" << "arguments", "Arguments given to process.", "args");
|
||||||
|
commandLineParser.addOption(processArguments);
|
||||||
|
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
QCommandLineOption subprocessSocket(QStringList() << "sock" << "socket", "IPC socket used for communication.", "sock");
|
||||||
|
#else
|
||||||
|
QCommandLineOption subprocessSocket(QStringList() << "sock" << "socket", "Unix socket used for communication.", "sock");
|
||||||
|
#endif
|
||||||
|
commandLineParser.addOption(subprocessSocket);
|
||||||
|
|
||||||
|
commandLineParser.process(a);
|
||||||
|
|
||||||
|
if (unlikely(commandLineParser.isSet(processManifest) && commandLineParser.isSet(processExecutable))) {
|
||||||
|
QTextStream(stderr) << "You can't define a Process executable and a JSON process manifest at the same time!" << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString executable;
|
||||||
|
QString workingDirectory;
|
||||||
|
QStringList argumentList;
|
||||||
|
if (likely(commandLineParser.isSet(processManifest))) {
|
||||||
|
QFile manifestFile(commandLineParser.value(processManifest));
|
||||||
|
if (likely(manifestFile.open(QIODevice::ReadOnly))) {
|
||||||
|
const QByteArray jsonData = manifestFile.readAll();
|
||||||
|
QJsonDocument jsonDocument = QJsonDocument::fromJson(jsonData);
|
||||||
|
QJsonObject jsonObject = jsonDocument.object();
|
||||||
|
|
||||||
|
if (likely(jsonObject.contains("Executable"))) {
|
||||||
|
const QJsonValue jsonExecutable = jsonObject.value("Executable");
|
||||||
|
if (unlikely(!jsonExecutable.isString())) {
|
||||||
|
QTextStream(stderr) << "Executable is not a string in manifest, aborting!" << endl;
|
||||||
|
manifestFile.close();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
executable = jsonExecutable.toString();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
QTextStream(stderr) << "Executable is not defined in manifest, aborting!" << endl;
|
||||||
|
manifestFile.close();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (likely(jsonObject.contains("WorkingDirectory"))) {
|
||||||
|
const QJsonValue jsonWorkingDirectory = jsonObject.value("WorkingDirectory");
|
||||||
|
if (unlikely(!jsonWorkingDirectory.isString())) {
|
||||||
|
QTextStream(stderr) << "Working Directory is not a string in manifest, aborting!" << endl;
|
||||||
|
manifestFile.close();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
workingDirectory = jsonWorkingDirectory.toString();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
workingDirectory = QFileInfo(executable).absolutePath();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (likely(jsonObject.contains("Arguments"))) {
|
||||||
|
const QJsonValue jsonArguments = jsonObject.value("Arguments");
|
||||||
|
if (likely(jsonArguments.isArray())) {
|
||||||
|
const QJsonArray jsonArray = jsonArguments.toArray();
|
||||||
|
QJsonArray::const_iterator it = jsonArray.constBegin();
|
||||||
|
QJsonArray::const_iterator end = jsonArray.constEnd();
|
||||||
|
while (it != end) {
|
||||||
|
argumentList << it->toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
QTextStream(stderr) << "Arguments is not a array in manifest, aborting!" << endl;
|
||||||
|
manifestFile.close();
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
manifestFile.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (unlikely(commandLineParser.isSet(processArguments))) {
|
||||||
|
QTextStream(stderr) << "Arguments over command line are not supported yet!" << endl;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
QString socket;
|
||||||
|
if (likely(commandLineParser.isSet(subprocessSocket))) {
|
||||||
|
socket = commandLineParser.value(subprocessSocket);
|
||||||
|
}
|
||||||
|
|
||||||
|
SMSubServer subServer(socket);
|
||||||
|
if (unlikely(!subServer.isListening())) {
|
||||||
|
#ifdef Q_OS_WIN
|
||||||
|
QTextStream(stderr) << "Failed to start IPC socket!" << endl;
|
||||||
|
#else
|
||||||
|
QTextStream(stderr) << "Failed to start Unix socket!" << endl;
|
||||||
|
#endif
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
SMSubProcess subProcess(executable, argumentList, workingDirectory);
|
||||||
|
|
||||||
|
QObject::connect(&subProcess, SIGNAL(outputWritten(QByteArray)), &subServer, SLOT(writeOutput(QByteArray)));
|
||||||
|
QObject::connect(&subServer, SIGNAL(inputWritten(QByteArray)), &subProcess, SLOT(writeInput(QByteArray)));
|
||||||
|
QObject::connect(&subServer, SIGNAL(killRequested()), &subProcess, SLOT(killProcess()));
|
||||||
|
QObject::connect(&subServer, SIGNAL(stopRequested()), &subProcess, SLOT(stopProcess()));
|
||||||
|
|
||||||
|
subProcess.start();
|
||||||
|
|
||||||
return a.exec();
|
return a.exec();
|
||||||
}
|
}
|
||||||
|
|
38
smsub.h
Normal file
38
smsub.h
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
/*****************************************************************************
|
||||||
|
* smsub Server Manager Subprocess
|
||||||
|
* Copyright (C) 2020 Syping
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted provided that the following conditions are met:
|
||||||
|
*
|
||||||
|
* 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
* this list of conditions and the following disclaimer in the documentation
|
||||||
|
* and/or other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
* responsible for anything with use of the software, you are self responsible.
|
||||||
|
*****************************************************************************/
|
||||||
|
|
||||||
|
#ifndef SMSUB_H
|
||||||
|
#define SMSUB_H
|
||||||
|
|
||||||
|
#ifndef SMSUB_WITHOUT_EXPECT
|
||||||
|
#ifndef likely
|
||||||
|
#define likely(x) __builtin_expect((x),1)
|
||||||
|
#endif
|
||||||
|
#ifndef unlikely
|
||||||
|
#define unlikely(x) __builtin_expect((x),0)
|
||||||
|
#endif
|
||||||
|
#else
|
||||||
|
#ifndef likely
|
||||||
|
#define likely(x) (x)
|
||||||
|
#endif
|
||||||
|
#ifndef unlikely
|
||||||
|
#define unlikely(x) (x)
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#endif // SMSUB_H
|
28
smsub.pro
28
smsub.pro
|
@ -1,10 +1,34 @@
|
||||||
|
###############################################################################
|
||||||
|
# smsub Server Manager Subprocess
|
||||||
|
# Copyright (C) 2020 Syping
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
# are permitted provided that the following conditions are met:
|
||||||
|
#
|
||||||
|
# 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
# this list of conditions and the following disclaimer.
|
||||||
|
#
|
||||||
|
# 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
# this list of conditions and the following disclaimer in the documentation
|
||||||
|
# and/or other materials provided with the distribution.
|
||||||
|
#
|
||||||
|
# This software is provided as-is, no warranties are given to you, we are not
|
||||||
|
# responsible for anything with use of the software, you are self responsible.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
QT -= gui
|
QT -= gui
|
||||||
|
QT += network
|
||||||
|
|
||||||
CONFIG += c++11 console
|
CONFIG += c++11 console
|
||||||
CONFIG -= app_bundle
|
CONFIG -= app_bundle
|
||||||
|
|
||||||
SOURCES += \
|
SOURCES += main.cpp \
|
||||||
main.cpp
|
SMSubProcess.cpp \
|
||||||
|
SMSubServer.cpp
|
||||||
|
|
||||||
|
HEADERS += smsub.h \
|
||||||
|
SMSubProcess.h \
|
||||||
|
SMSubServer.h
|
||||||
|
|
||||||
qnx: target.path = /tmp/$${TARGET}/bin
|
qnx: target.path = /tmp/$${TARGET}/bin
|
||||||
else: unix:!android: target.path = /opt/$${TARGET}/bin
|
else: unix:!android: target.path = /opt/$${TARGET}/bin
|
||||||
|
|
Loading…
Reference in a new issue