auth done even better!

This commit is contained in:
Nix 2025-03-31 19:06:12 -03:00 committed by GitHub
parent 18120360cb
commit 343788f38c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -1,43 +1,19 @@
import sys import sys
import subprocess
import re import re
import colorama
import requests
from PyQt5.QtWidgets import (QApplication, QDialog, QLabel, QVBoxLayout, from PyQt5.QtWidgets import (QApplication, QDialog, QLabel, QVBoxLayout,
QPushButton, QLineEdit, QMessageBox) QPushButton, QLineEdit, QMessageBox)
from PyQt5.QtCore import QThread, pyqtSignal, Qt, QUrl, QObject from PyQt5.QtCore import QThread, pyqtSignal, Qt, QUrl, QObject, QTimer
from PyQt5.QtGui import QDesktopServices from PyQt5.QtGui import QDesktopServices
from picomc.logging import logger
class AuthenticationParser: # Constants
@staticmethod URL_DEVICE_AUTH = "https://login.microsoftonline.com/consumers/oauth2/v2.0/devicecode"
def clean_ansi(text): URL_TOKEN = "https://login.microsoftonline.com/consumers/oauth2/v2.0/token"
ansi_clean = re.compile(r'\x1B(?:[@-Z\\-_]|\[[0-?]*[ -/]*[@-~])') CLIENT_ID = "c52aed44-3b4d-4215-99c5-824033d2bc0f"
printable_clean = re.compile(r'[^\x20-\x7E\n]') SCOPE = "XboxLive.signin offline_access"
text = ansi_clean.sub('', text) GRANT_TYPE = "urn:ietf:params:oauth:grant-type:device_code"
text = printable_clean.sub('', text)
return text.strip()
@staticmethod
def is_auth_error(output):
cleaned_output = AuthenticationParser.clean_ansi(output)
return "AADSTS70016" in cleaned_output and "not yet been authorized" in cleaned_output
@staticmethod
def parse_auth_output(output):
cleaned_output = AuthenticationParser.clean_ansi(output)
if AuthenticationParser.is_auth_error(cleaned_output):
return None
pattern = r"https://[^\s]+"
code_pattern = r"code\s+([A-Z0-9]+)"
url_match = re.search(pattern, cleaned_output)
code_match = re.search(code_pattern, cleaned_output, re.IGNORECASE)
if url_match and code_match:
return {
'url': url_match.group(0),
'code': code_match.group(1)
}
return None
class AuthDialog(QDialog): class AuthDialog(QDialog):
def __init__(self, url, code, parent=None, error_mode=False): def __init__(self, url, code, parent=None, error_mode=False):
@ -106,66 +82,81 @@ class AuthenticationThread(QThread):
error_occurred = pyqtSignal(str) error_occurred = pyqtSignal(str)
auth_error_detected = pyqtSignal(str) auth_error_detected = pyqtSignal(str)
finished = pyqtSignal() finished = pyqtSignal()
access_token_received = pyqtSignal(str, str)
def __init__(self, account): def __init__(self, account):
super().__init__() super().__init__()
self.account = account self.account = account
self.process = None self.device_code = None
self.is_running = True self.is_running = True
self.current_output = ""
self.waiting_for_auth = False
def run(self): def run(self):
try: try:
command = f'picomc account authenticate {self.account}' self.authenticate(self.account)
self.process = subprocess.Popen(
command,
shell=True,
stdout=subprocess.PIPE,
stderr=subprocess.STDOUT,
stdin=subprocess.PIPE,
text=True,
bufsize=1,
universal_newlines=True
)
self.current_output = ""
while self.is_running and self.process.poll() is None:
line = self.process.stdout.readline()
if line:
self.current_output += line
if not self.waiting_for_auth:
parsed_data = AuthenticationParser.parse_auth_output(self.current_output)
if parsed_data:
self.auth_data_received.emit(parsed_data)
self.waiting_for_auth = True
self.current_output = ""
elif AuthenticationParser.is_auth_error(self.current_output):
self.auth_error_detected.emit(self.current_output)
self.waiting_for_auth = False
self.current_output = ""
self.process.wait()
self.finished.emit()
except Exception as e: except Exception as e:
self.error_occurred.emit(str(e)) self.error_occurred.emit(str(e))
self.finished.emit() self.finished.emit()
def authenticate(self, account):
try:
data = {"client_id": CLIENT_ID, "scope": SCOPE}
# Request device code
resp = requests.post(URL_DEVICE_AUTH, data)
resp.raise_for_status()
j = resp.json()
self.device_code = j["device_code"]
user_code = j["user_code"]
link = j["verification_uri"]
# Format message with colorama
msg = j["message"]
msg = msg.replace(
user_code, colorama.Fore.RED + user_code + colorama.Fore.RESET
).replace(link, colorama.Style.BRIGHT + link + colorama.Style.NORMAL)
# Emit auth data received signal
self.auth_data_received.emit({'url': link, 'code': user_code})
except requests.exceptions.RequestException as e:
logger.error(f"Request failed: {e}")
self.error_occurred.emit(str(e))
self.finished.emit()
def poll_for_token(self):
try:
data = {"code": self.device_code, "grant_type": GRANT_TYPE, "client_id": CLIENT_ID}
resp = requests.post(URL_TOKEN, data)
if resp.status_code == 400:
j = resp.json()
logger.debug(j)
if j["error"] == "authorization_pending":
logger.warning(j["error_description"])
self.auth_error_detected.emit(j["error_description"])
return
else:
raise Exception(j["error_description"])
resp.raise_for_status()
j = resp.json()
access_token = j["access_token"]
refresh_token = j["refresh_token"]
logger.debug("OAuth device code flow successful")
self.access_token_received.emit(access_token, refresh_token)
self.finished.emit()
except requests.exceptions.RequestException as e:
logger.error(f"Request failed: {e}")
self.error_occurred.emit(str(e))
self.finished.emit()
def send_enter(self): def send_enter(self):
if self.process and self.process.poll() is None: self.poll_for_token()
self.process.stdin.write("\n")
self.process.stdin.flush()
def stop(self): def stop(self):
self.is_running = False self.is_running = False
if self.process:
self.process.terminate()
class MinecraftAuthenticator(QObject): # Changed to inherit from QObject class MinecraftAuthenticator(QObject):
auth_finished = pyqtSignal(bool) # Add signal for completion auth_finished = pyqtSignal(bool)
def __init__(self, parent=None): def __init__(self, parent=None):
super().__init__(parent) super().__init__(parent)
@ -175,15 +166,12 @@ class MinecraftAuthenticator(QObject): # Changed to inherit from QObject
self.success = False self.success = False
def authenticate(self, username): def authenticate(self, username):
"""
Start the authentication process for the given username
Returns immediately, authentication result will be emitted via auth_finished signal
"""
self.success = False self.success = False
self.auth_thread = AuthenticationThread(username) self.auth_thread = AuthenticationThread(username)
self.auth_thread.auth_data_received.connect(self.show_auth_dialog) self.auth_thread.auth_data_received.connect(self.show_auth_dialog)
self.auth_thread.auth_error_detected.connect(self.handle_auth_error) self.auth_thread.auth_error_detected.connect(self.handle_auth_error)
self.auth_thread.error_occurred.connect(self.show_error) self.auth_thread.error_occurred.connect(self.show_error)
self.auth_thread.access_token_received.connect(self.on_access_token_received)
self.auth_thread.finished.connect(self.on_authentication_finished) self.auth_thread.finished.connect(self.on_authentication_finished)
self.auth_thread.start() self.auth_thread.start()
@ -217,6 +205,11 @@ class MinecraftAuthenticator(QObject): # Changed to inherit from QObject
self.success = False self.success = False
self.auth_finished.emit(False) self.auth_finished.emit(False)
def on_access_token_received(self, access_token, refresh_token):
QMessageBox.information(None, "Success", "Authentication successful!")
self.success = True
self.auth_finished.emit(True)
def on_authentication_finished(self): def on_authentication_finished(self):
if self.auth_dialog is not None: if self.auth_dialog is not None:
self.auth_dialog.close() self.auth_dialog.close()
@ -226,8 +219,8 @@ class MinecraftAuthenticator(QObject): # Changed to inherit from QObject
self.auth_thread.stop() self.auth_thread.stop()
self.auth_thread = None self.auth_thread = None
self.success = True if not self.success:
self.auth_finished.emit(True) self.auth_finished.emit(False)
def cleanup(self): def cleanup(self):
if self.auth_dialog is not None: if self.auth_dialog is not None:
@ -240,5 +233,7 @@ class MinecraftAuthenticator(QObject): # Changed to inherit from QObject
# Example usage # Example usage
if __name__ == '__main__': if __name__ == '__main__':
app = QApplication(sys.argv)
authenticator = MinecraftAuthenticator() authenticator = MinecraftAuthenticator()
authenticator.authenticate("TestUser") authenticator.authenticate("TestUser")
sys.exit(app.exec_())