openuds/client-py3/full/src/UDSClient.py

404 lines
12 KiB
Python
Raw Normal View History

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
#
# Copyright (c) 2014-2021 Virtual Cable S.L.U.
# All rights reserved.
#
# Redistribution and use in source and binary forms, with or without modification,
# are permitted provided that the following conditions are met:
#
# * Redistributions of source code must retain the above copyright notice,
# this list of conditions and the following disclaimer.
# * 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.
# * Neither the name of Virtual Cable S.L.U. nor the names of its contributors
# may be used to endorse or promote products derived from this software
# without specific prior written permission.
#
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
2021-05-04 14:05:53 +03:00
'''
@author: Adolfo Gómez, dkmaster at dkmon dot com
2021-05-04 14:05:53 +03:00
'''
import sys
import os
2021-06-19 16:16:49 +03:00
import platform
import time
import webbrowser
import threading
2021-06-19 16:16:49 +03:00
import typing
from PyQt5 import QtCore, QtWidgets
from PyQt5.QtCore import QSettings
from uds.rest import RestApi, RetryException, InvalidVersion, UDSException
# Just to ensure there are available on runtime
from uds.forward import forward # type: ignore
from uds.tunnel import forward as f2 # type: ignore
from uds.log import logger
from uds import tools
from uds import VERSION
from UDSWindow import Ui_MainWindow
class UDSClient(QtWidgets.QMainWindow):
ticket: str = ''
scrambler: str = ''
withError = False
animTimer: typing.Optional[QtCore.QTimer] = None
anim: int = 0
animInverted: bool = False
api: RestApi
2021-06-19 16:16:49 +03:00
def __init__(self, api: RestApi, ticket: str, scrambler: str):
QtWidgets.QMainWindow.__init__(self)
self.api = api
2021-06-19 16:16:49 +03:00
self.ticket = ticket
self.scrambler = scrambler
self.setWindowFlags(QtCore.Qt.FramelessWindowHint | QtCore.Qt.WindowStaysOnTopHint) # type: ignore
self.ui = Ui_MainWindow()
self.ui.setupUi(self)
self.ui.progressBar.setValue(0)
self.ui.cancelButton.clicked.connect(self.cancelPushed)
2021-05-04 14:05:53 +03:00
self.ui.info.setText('Initializing...')
screen = QtWidgets.QDesktopWidget().screenGeometry()
mysize = self.geometry()
hpos = (screen.width() - mysize.width()) // 2
vpos = (screen.height() - mysize.height() - mysize.height()) // 2
self.move(hpos, vpos)
self.animTimer = QtCore.QTimer()
2021-06-19 16:16:49 +03:00
self.animTimer.timeout.connect(self.updateAnim) # type: ignore
# QtCore.QObject.connect(self.animTimer, QtCore.SIGNAL('timeout()'), self.updateAnim)
self.activateWindow()
self.startAnim()
def closeWindow(self):
self.close()
def showError(self, error):
2021-05-04 14:05:53 +03:00
logger.error('got error: %s', error)
self.stopAnim()
self.ui.info.setText(
'UDS Plugin Error'
) # In fact, main window is hidden, so this is not visible... :)
self.closeWindow()
2021-06-19 16:16:49 +03:00
QtWidgets.QMessageBox.critical(
None, # type: ignore
'UDS Plugin Error',
'{}'.format(error),
2021-06-19 16:26:05 +03:00
QtWidgets.QMessageBox.Ok,
2021-06-19 16:16:49 +03:00
)
self.withError = True
def cancelPushed(self):
self.close()
def updateAnim(self):
self.anim += 2
if self.anim > 99:
self.animInverted = not self.animInverted
self.ui.progressBar.setInvertedAppearance(self.animInverted)
self.anim = 0
self.ui.progressBar.setValue(self.anim)
def startAnim(self):
self.ui.progressBar.invertedAppearance = False # type: ignore
self.anim = 0
self.animInverted = False
self.ui.progressBar.setInvertedAppearance(self.animInverted)
self.animTimer.start(40)
def stopAnim(self):
self.ui.progressBar.invertedAppearance = False # type: ignore
self.animTimer.stop()
def getVersion(self):
try:
self.api.getVersion()
except InvalidVersion as e:
2021-06-19 16:16:49 +03:00
QtWidgets.QMessageBox.critical(
self,
'Upgrade required',
'A newer connector version is required.\nA browser will be opened to download it.',
2021-06-19 16:26:05 +03:00
QtWidgets.QMessageBox.Ok,
2021-06-19 16:16:49 +03:00
)
webbrowser.open(e.downloadUrl)
self.closeWindow()
return
except Exception as e:
self.showError(e)
self.getTransportData()
def getTransportData(self):
try:
script, params = self.api.getScriptAndParams(self.ticket, self.scrambler)
self.stopAnim()
2021-05-04 14:05:53 +03:00
if 'darwin' in sys.platform:
self.showMinimized()
# QtCore.QTimer.singleShot(3000, self.endScript)
# self.hide()
self.closeWindow()
exec(script, globals(), {'parent': self, 'sp': params})
2021-06-19 16:16:49 +03:00
# Execute the waiting tasks...
threading.Thread(target=endScript).start()
except RetryException as e:
self.ui.info.setText(str(e) + ', retrying access...')
# Retry operation in ten seconds
QtCore.QTimer.singleShot(10000, self.getTransportData)
except Exception as e:
2021-06-19 16:26:05 +03:00
# logger.exception('Got exception on getTransportData')
self.showError(e)
def start(self):
"""
Starts proccess by requesting version info
"""
2021-05-04 14:05:53 +03:00
self.ui.info.setText('Initializing...')
QtCore.QTimer.singleShot(100, self.getVersion)
2021-06-19 16:26:05 +03:00
def endScript():
2021-06-19 16:16:49 +03:00
# Wait a bit before start processing ending sequence
time.sleep(3)
2021-06-21 18:50:12 +03:00
try:
# Remove early stage files...
tools.unlinkFiles(early=True)
except Exception as e:
logger.debug('Unlinking files on early stage: %s', e)
# After running script, wait for stuff
try:
2021-06-19 16:16:49 +03:00
logger.debug('Wating for tasks to finish...')
tools.waitForTasks()
2021-06-21 18:11:32 +03:00
except Exception as e:
logger.debug('Watiting for tasks to finish: %s', e)
try:
2021-06-19 16:16:49 +03:00
logger.debug('Unlinking files')
2021-06-21 18:50:12 +03:00
tools.unlinkFiles(early=False)
2021-06-21 18:11:32 +03:00
except Exception as e:
2021-06-21 18:50:12 +03:00
logger.debug('Unlinking files on later stage: %s', e)
2021-06-19 16:26:05 +03:00
# Removing
try:
2021-06-19 16:16:49 +03:00
logger.debug('Executing threads before exit')
tools.execBeforeExit()
2021-06-21 18:11:32 +03:00
except Exception as e:
logger.debug('execBeforeExit: %s', e)
2021-06-19 16:16:49 +03:00
logger.debug('endScript done')
# Ask user to approve endpoint
def approveHost(hostName: str):
settings = QtCore.QSettings()
2021-05-04 14:05:53 +03:00
settings.beginGroup('endpoints')
# approved = settings.value(hostName, False).toBool()
approved = bool(settings.value(hostName, False))
2021-05-04 14:05:53 +03:00
errorString = '<p>The server <b>{}</b> must be approved:</p>'.format(hostName)
errorString += (
'<p>Only approve UDS servers that you trust to avoid security issues.</p>'
)
2021-05-04 14:05:53 +03:00
if not approved:
2021-06-19 16:26:05 +03:00
if (
QtWidgets.QMessageBox.warning(
None, # type: ignore
'ACCESS Warning',
errorString,
QtWidgets.QMessageBox.Yes | QtWidgets.QMessageBox.No, # type: ignore
)
== QtWidgets.QMessageBox.Yes
):
settings.setValue(hostName, True)
approved = True
settings.endGroup()
return approved
2021-06-19 16:26:05 +03:00
def sslError(hostname: str, serial):
settings = QSettings()
settings.beginGroup('ssl')
approved = settings.value(serial, False)
if (
approved
or QtWidgets.QMessageBox.warning(
2021-06-19 16:26:05 +03:00
None, # type: ignore
'SSL Warning',
f'Could not check sll certificate for {hostname}',
QtWidgets.QMessageBox.Yes | QtWidgets.QMessageBox.No, # type: ignore
)
== QtWidgets.QMessageBox.Yes
):
approved = True
settings.setValue(serial, True)
settings.endGroup()
return approved
2021-06-21 12:05:38 +03:00
# Used only if command line says so
2021-06-19 16:16:49 +03:00
def minimal(api: RestApi, ticket: str, scrambler: str):
try:
logger.info('M1 Execution')
logger.debug('Getting version')
try:
api.getVersion()
except InvalidVersion as e:
QtWidgets.QMessageBox.critical(
None, # type: ignore
'Upgrade required',
'A newer connector version is required.\nA browser will be opened to download it.',
2021-06-19 16:26:05 +03:00
QtWidgets.QMessageBox.Ok,
2021-06-19 16:16:49 +03:00
)
webbrowser.open(e.downloadUrl)
return 0
logger.debug('Transport data')
script, params = api.getScriptAndParams(ticket, scrambler)
# Execute UDS transport script
exec(script, globals(), {'parent': None, 'sp': params})
# Execute the waiting task...
threading.Thread(target=endScript).start()
except RetryException as e:
QtWidgets.QMessageBox.warning(
2021-06-19 16:26:05 +03:00
None, # type: ignore
'Service not ready',
'{}'.format('.\n'.join(str(e).split('.')))
+ '\n\nPlease, retry again in a while.',
QtWidgets.QMessageBox.Ok,
)
2021-06-19 16:16:49 +03:00
except Exception as e:
2021-06-19 16:26:05 +03:00
# logger.exception('Got exception on getTransportData')
QtWidgets.QMessageBox.critical(
None, # type: ignore
'Error',
2021-06-21 12:05:38 +03:00
'{}'.format(str(e)) + '\n\nPlease, retry again in a while.',
2021-06-19 16:26:05 +03:00
QtWidgets.QMessageBox.Ok,
)
2021-06-19 16:38:19 +03:00
return 0
2021-06-19 16:16:49 +03:00
if __name__ == "__main__":
2021-06-19 16:57:53 +03:00
logger.debug('Initializing connector for %s(%s)', sys.platform, platform.machine())
# Initialize app
app = QtWidgets.QApplication(sys.argv)
# Set several info for settings
2021-05-04 14:05:53 +03:00
QtCore.QCoreApplication.setOrganizationName('Virtual Cable S.L.U.')
QtCore.QCoreApplication.setApplicationName('UDS Connector')
2021-05-04 14:05:53 +03:00
if 'darwin' not in sys.platform:
logger.debug('Mac OS *NOT* Detected')
app.setStyle('plastique') # type: ignore
else:
logger.debug('Platform is Mac OS, adding homebrew possible paths')
2021-06-21 18:50:12 +03:00
os.environ['PATH'] += ''.join(
os.pathsep + i
for i in (
'/usr/local/bin',
'/opt/homebrew/bin',
)
)
logger.debug('Now path is %s', os.environ['PATH'])
# First parameter must be url
useMinimal = False
try:
uri = sys.argv[1]
if uri == '--minimal':
useMinimal = True
uri = sys.argv[2] # And get URI
2021-05-04 14:05:53 +03:00
if uri == '--test':
sys.exit(0)
2021-05-04 14:05:53 +03:00
logger.debug('URI: %s', uri)
if uri[:6] != 'uds://' and uri[:7] != 'udss://':
raise Exception()
2021-05-04 14:05:53 +03:00
ssl = uri[3] == 's'
2021-06-19 16:16:49 +03:00
host, ticket, scrambler = uri.split('//')[1].split('/') # type: ignore
logger.debug(
'ssl:%s, host:%s, ticket:%s, scrambler:%s',
ssl,
host,
UDSClient.ticket,
UDSClient.scrambler,
)
except Exception:
2021-05-04 14:05:53 +03:00
logger.debug('Detected execution without valid URI, exiting')
QtWidgets.QMessageBox.critical(
None, # type: ignore
'Notice',
'UDS Client Version {}'.format(VERSION),
QtWidgets.QMessageBox.Ok,
)
sys.exit(1)
# Setup REST api endpoint
2021-06-19 16:26:05 +03:00
api = RestApi(
'{}://{}/uds/rest/client'.format(['http', 'https'][ssl], host), sslError
)
try:
2021-05-04 14:05:53 +03:00
logger.debug('Starting execution')
# Approbe before going on
if approveHost(host) is False:
2021-05-04 14:05:53 +03:00
raise Exception('Host {} was not approved'.format(host))
2021-06-19 16:16:49 +03:00
win = UDSClient(api, ticket, scrambler)
win.show()
win.start()
exitVal = app.exec_()
2021-05-04 14:05:53 +03:00
logger.debug('Execution finished correctly')
except Exception as e:
2021-05-04 14:05:53 +03:00
logger.exception('Got an exception executing client:')
exitVal = 128
QtWidgets.QMessageBox.critical(
None, 'Error', str(e), QtWidgets.QMessageBox.Ok # type: ignore
)
2021-05-04 14:05:53 +03:00
logger.debug('Exiting')
sys.exit(exitVal)