mirror of
https://github.com/Ryujinx/ryuko-ng.git
synced 2025-01-03 14:25:29 +00:00
Log reading improvements (#11)
* Added warnings for several settings: - Expand DRAM hack - Memory Manager Mode - Ignore Missing Services - Anisotropic Filtering set to not Auto - Debug logs enabled - New severity level for PPTC and Shader cache warnings * Various fixes: - Warn for outdated keys/firmware, - Error snippet fix when no game boots - Embed improvements - Improve duplicate log upload tracking to link to last uploaded file * Larger download header range, handle larger files. * Move notes visibility to show on startup crash * Added vsync disabled warning and dump hash error * Clean up controller warning to declutter empty log message * Add .NET 6 shader warning and genericise shader init error
This commit is contained in:
parent
32a8b6b431
commit
22f81b449b
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -102,3 +102,4 @@ config.py
|
||||||
|
|
||||||
# Prevent data files from being committed
|
# Prevent data files from being committed
|
||||||
data/
|
data/
|
||||||
|
robocop_ng/config_template.py
|
||||||
|
|
|
@ -3,7 +3,7 @@ import re
|
||||||
|
|
||||||
import aiohttp
|
import aiohttp
|
||||||
import config
|
import config
|
||||||
import discord
|
from discord import Colour, Embed
|
||||||
from discord.ext.commands import Cog
|
from discord.ext.commands import Cog
|
||||||
|
|
||||||
logging.basicConfig(
|
logging.basicConfig(
|
||||||
|
@ -15,14 +15,14 @@ logging.basicConfig(
|
||||||
class LogFileReader(Cog):
|
class LogFileReader(Cog):
|
||||||
def __init__(self, bot):
|
def __init__(self, bot):
|
||||||
self.bot = bot
|
self.bot = bot
|
||||||
# Allows log analysis in #support and #patreon-support channels respectively
|
|
||||||
self.bot_log_allowed_channels = config.bot_log_allowed_channels
|
self.bot_log_allowed_channels = config.bot_log_allowed_channels
|
||||||
self.uploaded_log_filenames = []
|
self.ryujinx_blue = Colour(0x4A90E2)
|
||||||
|
self.uploaded_log_info = []
|
||||||
|
|
||||||
async def download_file(self, log_url):
|
async def download_file(self, log_url):
|
||||||
async with aiohttp.ClientSession() as session:
|
async with aiohttp.ClientSession() as session:
|
||||||
# Grabs first and last few bytes of log file to prevent abuse from large files
|
# Grabs first and last few bytes of log file to prevent abuse from large files
|
||||||
headers = {"Range": "bytes=0-25000, -6000"}
|
headers = {"Range": "bytes=0-35000, -6000"}
|
||||||
async with session.get(log_url, headers=headers) as response:
|
async with session.get(log_url, headers=headers) as response:
|
||||||
return await response.text("UTF-8")
|
return await response.text("UTF-8")
|
||||||
|
|
||||||
|
@ -48,7 +48,9 @@ class LogFileReader(Cog):
|
||||||
"settings": {
|
"settings": {
|
||||||
"audio_backend": "Unknown",
|
"audio_backend": "Unknown",
|
||||||
"docked": "Unknown",
|
"docked": "Unknown",
|
||||||
|
"expand_ram": "Unknown",
|
||||||
"ignore_missing_services": "Unknown",
|
"ignore_missing_services": "Unknown",
|
||||||
|
"memory_manager": "Unknown",
|
||||||
"pptc": "Unknown",
|
"pptc": "Unknown",
|
||||||
"shader_cache": "Unknown",
|
"shader_cache": "Unknown",
|
||||||
"vsync": "Unknown",
|
"vsync": "Unknown",
|
||||||
|
@ -172,9 +174,7 @@ class LogFileReader(Cog):
|
||||||
)
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
log_embed = discord.Embed(
|
log_embed = Embed(title=f"{cleaned_game_name}", colour=self.ryujinx_blue)
|
||||||
title=f"{cleaned_game_name}", colour=discord.Colour(0x4A90E2)
|
|
||||||
)
|
|
||||||
log_embed.set_footer(text=f"Log uploaded by {author_name}")
|
log_embed.set_footer(text=f"Log uploaded by {author_name}")
|
||||||
log_embed.add_field(
|
log_embed.add_field(
|
||||||
name="General Info",
|
name="General Info",
|
||||||
|
@ -191,10 +191,13 @@ class LogFileReader(Cog):
|
||||||
value=graphics_settings_info,
|
value=graphics_settings_info,
|
||||||
inline=True,
|
inline=True,
|
||||||
)
|
)
|
||||||
if cleaned_game_name == "Unknown":
|
if (
|
||||||
|
cleaned_game_name == "Unknown"
|
||||||
|
and self.embed["game_info"]["errors"] == "No errors found in log"
|
||||||
|
):
|
||||||
log_embed.add_field(
|
log_embed.add_field(
|
||||||
name="Empty Log",
|
name="Empty Log",
|
||||||
value=f"""This log file appears to be empty. To get a proper log, follow these steps:
|
value=f"""The log file appears to be empty. To get a proper log, follow these steps:
|
||||||
1) In Logging settings, ensure `Enable Logging to File` is checked.
|
1) In Logging settings, ensure `Enable Logging to File` is checked.
|
||||||
2) Ensure the following default logs are enabled: `Info`, `Warning`, `Error`, `Guest` and `Stub`.
|
2) Ensure the following default logs are enabled: `Info`, `Warning`, `Error`, `Guest` and `Stub`.
|
||||||
3) Start a game up.
|
3) Start a game up.
|
||||||
|
@ -202,11 +205,25 @@ class LogFileReader(Cog):
|
||||||
5) Upload the latest log file.""",
|
5) Upload the latest log file.""",
|
||||||
inline=False,
|
inline=False,
|
||||||
)
|
)
|
||||||
|
if (
|
||||||
|
cleaned_game_name == "Unknown"
|
||||||
|
and self.embed["game_info"]["errors"] != "No errors found in log"
|
||||||
|
):
|
||||||
log_embed.add_field(
|
log_embed.add_field(
|
||||||
name="Latest Error Snippet",
|
name="Latest Error Snippet",
|
||||||
value=self.embed["game_info"]["errors"],
|
value=self.embed["game_info"]["errors"],
|
||||||
inline=False,
|
inline=False,
|
||||||
)
|
)
|
||||||
|
log_embed.add_field(
|
||||||
|
name="No Game Boot Detected",
|
||||||
|
value=f"""No game boot has been detected in log file. To get a proper log, follow these steps:
|
||||||
|
1) In Logging settings, ensure `Enable Logging to File` is checked.
|
||||||
|
2) Ensure the following default logs are enabled: `Info`, `Warning`, `Error`, `Guest` and `Stub`.
|
||||||
|
3) Start a game up.
|
||||||
|
4) Play until your issue occurs.
|
||||||
|
5) Upload the latest log file.""",
|
||||||
|
inline=False,
|
||||||
|
)
|
||||||
else:
|
else:
|
||||||
log_embed.add_field(
|
log_embed.add_field(
|
||||||
name="Latest Error Snippet",
|
name="Latest Error Snippet",
|
||||||
|
@ -276,7 +293,6 @@ class LogFileReader(Cog):
|
||||||
}
|
}
|
||||||
setting[name] = aspect_map[setting_value]
|
setting[name] = aspect_map[setting_value]
|
||||||
if name in [
|
if name in [
|
||||||
"ignore_missing_services",
|
|
||||||
"pptc",
|
"pptc",
|
||||||
"shader_cache",
|
"shader_cache",
|
||||||
"vsync",
|
"vsync",
|
||||||
|
@ -291,7 +307,9 @@ class LogFileReader(Cog):
|
||||||
"aspect_ratio": "AspectRatio",
|
"aspect_ratio": "AspectRatio",
|
||||||
"audio_backend": "AudioBackend",
|
"audio_backend": "AudioBackend",
|
||||||
"docked": "EnableDockedMode",
|
"docked": "EnableDockedMode",
|
||||||
|
"expand_ram": "ExpandRam",
|
||||||
"ignore_missing_services": "IgnoreMissingServices",
|
"ignore_missing_services": "IgnoreMissingServices",
|
||||||
|
"memory_manager": "MemoryManagerMode",
|
||||||
"pptc": "EnablePtc",
|
"pptc": "EnablePtc",
|
||||||
"resolution_scale": "ResScale",
|
"resolution_scale": "ResScale",
|
||||||
"shader_cache": "EnableShaderCache",
|
"shader_cache": "EnableShaderCache",
|
||||||
|
@ -306,16 +324,6 @@ class LogFileReader(Cog):
|
||||||
f"Settings exception: {setting_name}: {type(error).__name__}"
|
f"Settings exception: {setting_name}: {type(error).__name__}"
|
||||||
)
|
)
|
||||||
continue
|
continue
|
||||||
# Game name parsed last so that user settings are visible with empty log
|
|
||||||
self.embed["game_info"]["game_name"] = (
|
|
||||||
re.search(
|
|
||||||
r"Loader LoadNca: Application Loaded:\s([^;\n\r]*)",
|
|
||||||
log_file,
|
|
||||||
re.MULTILINE,
|
|
||||||
)
|
|
||||||
.group(1)
|
|
||||||
.rstrip()
|
|
||||||
)
|
|
||||||
|
|
||||||
def analyse_error_message(log_file=log_file):
|
def analyse_error_message(log_file=log_file):
|
||||||
try:
|
try:
|
||||||
|
@ -340,14 +348,20 @@ class LogFileReader(Cog):
|
||||||
return False
|
return False
|
||||||
|
|
||||||
shader_cache_collision = error_search(["Cache collision found"])
|
shader_cache_collision = error_search(["Cache collision found"])
|
||||||
dump_hash_warning = error_search(["ResultFsInvalidIvfcHash"])
|
dump_hash_warning = error_search(
|
||||||
shader_cache_corruption = error_search(
|
|
||||||
[
|
[
|
||||||
"""Object reference not set to an instance of an object.
|
"ResultFsInvalidIvfcHash",
|
||||||
at Ryujinx.Graphics.Gpu.Shader.ShaderCache.Initialize()""",
|
"ResultFsNonRealDataVerificationFailed",
|
||||||
"System.IO.InvalidDataException: End of Central Directory record could not be found",
|
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
shader_cache_corruption = error_search(
|
||||||
|
[
|
||||||
|
"Ryujinx.Graphics.Gpu.Shader.ShaderCache.Initialize()",
|
||||||
|
"System.IO.InvalidDataException: End of Central Directory record could not be found",
|
||||||
|
"ICSharpCode.SharpZipLib.Zip.ZipException: Cannot find central directory",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
update_keys_error = error_search(["LibHac.MissingKeyException"])
|
||||||
last_errors = "\n".join(
|
last_errors = "\n".join(
|
||||||
errors[-1][:2] if "|E|" in errors[-1][0] else ""
|
errors[-1][:2] if "|E|" in errors[-1][0] else ""
|
||||||
)
|
)
|
||||||
|
@ -358,6 +372,7 @@ class LogFileReader(Cog):
|
||||||
shader_cache_collision,
|
shader_cache_collision,
|
||||||
dump_hash_warning,
|
dump_hash_warning,
|
||||||
shader_cache_corruption,
|
shader_cache_corruption,
|
||||||
|
update_keys_error,
|
||||||
)
|
)
|
||||||
|
|
||||||
# Finds the lastest error denoted by |E| in the log and its first line
|
# Finds the lastest error denoted by |E| in the log and its first line
|
||||||
|
@ -367,6 +382,7 @@ class LogFileReader(Cog):
|
||||||
shader_cache_warn,
|
shader_cache_warn,
|
||||||
dump_hash_warning,
|
dump_hash_warning,
|
||||||
shader_cache_corruption_warn,
|
shader_cache_corruption_warn,
|
||||||
|
update_keys_error,
|
||||||
) = analyse_error_message()
|
) = analyse_error_message()
|
||||||
if last_error_snippet:
|
if last_error_snippet:
|
||||||
self.embed["game_info"]["errors"] = f"```{last_error_snippet}```"
|
self.embed["game_info"]["errors"] = f"```{last_error_snippet}```"
|
||||||
|
@ -400,6 +416,12 @@ class LogFileReader(Cog):
|
||||||
dump_hash_warning = f"⚠️ Dump error detected. Investigate possible bad game/firmware dump issues"
|
dump_hash_warning = f"⚠️ Dump error detected. Investigate possible bad game/firmware dump issues"
|
||||||
self.embed["game_info"]["notes"].append(dump_hash_warning)
|
self.embed["game_info"]["notes"].append(dump_hash_warning)
|
||||||
|
|
||||||
|
if update_keys_error:
|
||||||
|
update_keys_error = (
|
||||||
|
f"⚠️ Keys or firmware out of date, consider updating them"
|
||||||
|
)
|
||||||
|
self.embed["game_info"]["notes"].append(update_keys_error)
|
||||||
|
|
||||||
timestamp_regex = re.compile(r"\d{2}:\d{2}:\d{2}\.\d{3}")
|
timestamp_regex = re.compile(r"\d{2}:\d{2}:\d{2}\.\d{3}")
|
||||||
latest_timestamp = re.findall(timestamp_regex, log_file)[-1]
|
latest_timestamp = re.findall(timestamp_regex, log_file)[-1]
|
||||||
if latest_timestamp:
|
if latest_timestamp:
|
||||||
|
@ -419,7 +441,6 @@ class LogFileReader(Cog):
|
||||||
]
|
]
|
||||||
return mods_status
|
return mods_status
|
||||||
|
|
||||||
# Find information on installed mods
|
|
||||||
game_mods = mods_information()
|
game_mods = mods_information()
|
||||||
if game_mods:
|
if game_mods:
|
||||||
self.embed["game_info"]["mods"] = "\n".join(game_mods)
|
self.embed["game_info"]["mods"] = "\n".join(game_mods)
|
||||||
|
@ -434,9 +455,14 @@ class LogFileReader(Cog):
|
||||||
# also maintains the list order
|
# also maintains the list order
|
||||||
input_status = list(dict.fromkeys(input_status))
|
input_status = list(dict.fromkeys(input_status))
|
||||||
input_string = "\n".join(input_status)
|
input_string = "\n".join(input_status)
|
||||||
else:
|
self.embed["game_info"]["notes"].append(input_string)
|
||||||
|
# If emulator crashes on startup without game load, there is no need to show controller notification at all
|
||||||
|
if (
|
||||||
|
not controllers
|
||||||
|
and self.embed["game_info"]["game_name"] != "Unknown"
|
||||||
|
):
|
||||||
input_string = "⚠️ No controller information found"
|
input_string = "⚠️ No controller information found"
|
||||||
self.embed["game_info"]["notes"].append(input_string)
|
self.embed["game_info"]["notes"].append(input_string)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
ram_available_regex = re.compile(r"Available\s(\d+)(?=\sMB)")
|
ram_available_regex = re.compile(r"Available\s(\d+)(?=\sMB)")
|
||||||
|
@ -461,7 +487,6 @@ class LogFileReader(Cog):
|
||||||
intel_gpu_warning = "**⚠️ Intel iGPUs are known to have driver issues, consider using a discrete GPU**"
|
intel_gpu_warning = "**⚠️ Intel iGPUs are known to have driver issues, consider using a discrete GPU**"
|
||||||
self.embed["game_info"]["notes"].append(intel_gpu_warning)
|
self.embed["game_info"]["notes"].append(intel_gpu_warning)
|
||||||
try:
|
try:
|
||||||
# Find information on logs, whether defaults are enabled or not
|
|
||||||
default_logs = ["Info", "Warning", "Error", "Guest", "Stub"]
|
default_logs = ["Info", "Warning", "Error", "Guest", "Stub"]
|
||||||
user_logs = (
|
user_logs = (
|
||||||
self.embed["emu_info"]["logs_enabled"]
|
self.embed["emu_info"]["logs_enabled"]
|
||||||
|
@ -469,6 +494,9 @@ class LogFileReader(Cog):
|
||||||
.replace(" ", "")
|
.replace(" ", "")
|
||||||
.split(",")
|
.split(",")
|
||||||
)
|
)
|
||||||
|
if "Debug" in user_logs:
|
||||||
|
debug_warning = f"⚠️ **Debug logs enabled will have a negative impact on performance**"
|
||||||
|
self.embed["game_info"]["notes"].append(debug_warning)
|
||||||
disabled_logs = set(default_logs).difference(set(user_logs))
|
disabled_logs = set(default_logs).difference(set(user_logs))
|
||||||
if disabled_logs:
|
if disabled_logs:
|
||||||
logs_status = [
|
logs_status = [
|
||||||
|
@ -485,6 +513,12 @@ class LogFileReader(Cog):
|
||||||
firmware_warning = f"**❌ Nintendo Switch firmware not found**"
|
firmware_warning = f"**❌ Nintendo Switch firmware not found**"
|
||||||
self.embed["game_info"]["notes"].append(firmware_warning)
|
self.embed["game_info"]["notes"].append(firmware_warning)
|
||||||
|
|
||||||
|
if self.embed["settings"]["anisotropic_filtering"] != "Auto":
|
||||||
|
anisotropic_filtering_warning = "⚠️ Anisotropic filtering not set to `Auto` can cause graphical issues"
|
||||||
|
self.embed["game_info"]["notes"].append(
|
||||||
|
anisotropic_filtering_warning
|
||||||
|
)
|
||||||
|
|
||||||
if self.embed["settings"]["audio_backend"] == "Dummy":
|
if self.embed["settings"]["audio_backend"] == "Dummy":
|
||||||
dummy_warning = (
|
dummy_warning = (
|
||||||
f"⚠️ Dummy audio backend, consider changing to SDL2 or OpenAL"
|
f"⚠️ Dummy audio backend, consider changing to SDL2 or OpenAL"
|
||||||
|
@ -492,13 +526,33 @@ class LogFileReader(Cog):
|
||||||
self.embed["game_info"]["notes"].append(dummy_warning)
|
self.embed["game_info"]["notes"].append(dummy_warning)
|
||||||
|
|
||||||
if self.embed["settings"]["pptc"] == "Disabled":
|
if self.embed["settings"]["pptc"] == "Disabled":
|
||||||
pptc_warning = f"⚠️ PPTC cache should be enabled"
|
pptc_warning = f"🔴 **PPTC cache should be enabled**"
|
||||||
self.embed["game_info"]["notes"].append(pptc_warning)
|
self.embed["game_info"]["notes"].append(pptc_warning)
|
||||||
|
|
||||||
if self.embed["settings"]["shader_cache"] == "Disabled":
|
if self.embed["settings"]["shader_cache"] == "Disabled":
|
||||||
shader_warning = f"⚠️ Shader cache should be enabled"
|
shader_warning = f"🔴 **Shader cache should be enabled**"
|
||||||
self.embed["game_info"]["notes"].append(shader_warning)
|
self.embed["game_info"]["notes"].append(shader_warning)
|
||||||
|
|
||||||
|
if self.embed["settings"]["expand_ram"] == "True":
|
||||||
|
expand_ram_warning = f"⚠️ `Expand DRAM size to 6GB` should only be enabled for 4K mods"
|
||||||
|
self.embed["game_info"]["notes"].append(expand_ram_warning)
|
||||||
|
|
||||||
|
if self.embed["settings"]["memory_manager"] == "SoftwarePageTable":
|
||||||
|
software_memory_manager_warning = "⚠️ `Software` setting in Memory Manager Mode will give slower performance than the default setting of `Host unchecked`"
|
||||||
|
self.embed["game_info"]["notes"].append(
|
||||||
|
software_memory_manager_warning
|
||||||
|
)
|
||||||
|
|
||||||
|
if self.embed["settings"]["ignore_missing_services"] == "True":
|
||||||
|
ignore_missing_services_warning = "⚠️ `Ignore Missing Services` being enabled can cause instability"
|
||||||
|
self.embed["game_info"]["notes"].append(
|
||||||
|
ignore_missing_services_warning
|
||||||
|
)
|
||||||
|
|
||||||
|
if self.embed["settings"]["vsync"] == "Disabled":
|
||||||
|
vsync_warning = f"⚠️ V-Sync disabled can cause instability like games running faster than intended or longer load times"
|
||||||
|
self.embed["game_info"]["notes"].append(vsync_warning)
|
||||||
|
|
||||||
mainline_version = re.compile(r"^\d\.\d\.(\d){4}$")
|
mainline_version = re.compile(r"^\d\.\d\.(\d){4}$")
|
||||||
pr_version = re.compile(r"^\d\.\d\.\d\+([a-f]|\d){7}$")
|
pr_version = re.compile(r"^\d\.\d\.\d\+([a-f]|\d){7}$")
|
||||||
ldn_version = re.compile(r"^\d\.\d\.\d\-ldn\d\.\d$")
|
ldn_version = re.compile(r"^\d\.\d\.\d\-ldn\d\.\d$")
|
||||||
|
@ -528,7 +582,7 @@ class LogFileReader(Cog):
|
||||||
self.embed["game_info"]["notes"].append(custom_firmware_warning)
|
self.embed["game_info"]["notes"].append(custom_firmware_warning)
|
||||||
|
|
||||||
def severity(log_note_string):
|
def severity(log_note_string):
|
||||||
symbols = ["❌", "⚠️", "ℹ", "✅"]
|
symbols = ["❌", "🔴", "⚠️", "ℹ", "✅"]
|
||||||
return next(
|
return next(
|
||||||
i
|
i
|
||||||
for i, symbol in enumerate(symbols)
|
for i, symbol in enumerate(symbols)
|
||||||
|
@ -557,11 +611,13 @@ class LogFileReader(Cog):
|
||||||
if message.author.bot:
|
if message.author.bot:
|
||||||
return
|
return
|
||||||
try:
|
try:
|
||||||
|
author_id = message.author.id
|
||||||
author_mention = message.author.mention
|
author_mention = message.author.mention
|
||||||
filename = message.attachments[0].filename
|
filename = message.attachments[0].filename
|
||||||
# Any message over 2000 chars is uploaded as message.txt, so this is accounted for
|
# Any message over 2000 chars is uploaded as message.txt, so this is accounted for
|
||||||
ryujinx_log_file_regex = re.compile(r"^Ryujinx_.*\.log|message\.txt$")
|
ryujinx_log_file_regex = re.compile(r"^Ryujinx_.*\.log|message\.txt$")
|
||||||
log_file = re.compile(r"^.*\.log|.*\.txt$")
|
log_file = re.compile(r"^.*\.log|.*\.txt$")
|
||||||
|
log_file_link = message.jump_url
|
||||||
is_ryujinx_log_file = re.match(ryujinx_log_file_regex, filename)
|
is_ryujinx_log_file = re.match(ryujinx_log_file_regex, filename)
|
||||||
is_log_file = re.match(log_file, filename)
|
is_log_file = re.match(log_file, filename)
|
||||||
|
|
||||||
|
@ -569,32 +625,58 @@ class LogFileReader(Cog):
|
||||||
message.channel.id in self.bot_log_allowed_channels.values()
|
message.channel.id in self.bot_log_allowed_channels.values()
|
||||||
and is_ryujinx_log_file
|
and is_ryujinx_log_file
|
||||||
):
|
):
|
||||||
if filename not in self.uploaded_log_filenames:
|
uploaded_logs_exist = [
|
||||||
|
True for elem in self.uploaded_log_info if filename in elem.values()
|
||||||
|
]
|
||||||
|
if not any(uploaded_logs_exist):
|
||||||
reply_message = await message.channel.send(
|
reply_message = await message.channel.send(
|
||||||
"Log detected, parsing..."
|
"Log detected, parsing..."
|
||||||
)
|
)
|
||||||
try:
|
try:
|
||||||
embed = await self.log_file_read(message)
|
embed = await self.log_file_read(message)
|
||||||
if "Ryujinx_" in filename:
|
if "Ryujinx_" in filename:
|
||||||
self.uploaded_log_filenames.append(filename)
|
self.uploaded_log_info.append(
|
||||||
|
{
|
||||||
|
"filename": filename,
|
||||||
|
"link": log_file_link,
|
||||||
|
"author": author_id,
|
||||||
|
}
|
||||||
|
)
|
||||||
# Avoid duplicate log file analysis, at least temporarily; keep track of the last few filenames of uploaded logs
|
# Avoid duplicate log file analysis, at least temporarily; keep track of the last few filenames of uploaded logs
|
||||||
# this should help support channels not be flooded with too many log files
|
# this should help support channels not be flooded with too many log files
|
||||||
# fmt: off
|
# fmt: off
|
||||||
self.uploaded_log_filenames = self.uploaded_log_filenames[-5:]
|
self.uploaded_log_info = self.uploaded_log_info[-5:]
|
||||||
# fmt: on
|
# fmt: on
|
||||||
return await reply_message.edit(content=None, embed=embed)
|
return await reply_message.edit(content=None, embed=embed)
|
||||||
except UnicodeDecodeError:
|
except UnicodeDecodeError:
|
||||||
return await message.channel.send(
|
return await message.channel.send(
|
||||||
f"This log file appears to be invalid {author_mention}. Please re-check and re-upload your log file."
|
content=author_mention,
|
||||||
|
embed=Embed(
|
||||||
|
description=f"This log file appears to be invalid. Please re-check and re-upload your log file.",
|
||||||
|
colour=self.ryujinx_blue,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
except Exception as error:
|
except Exception as error:
|
||||||
await reply_message.edit(
|
await reply_message.edit(
|
||||||
content=f"Error: Couldn't parse log; parser threw {type(error).__name__} exception."
|
content=f"Error: Couldn't parse log; parser threw `{type(error).__name__}` exception."
|
||||||
)
|
)
|
||||||
print(logging.warn(error))
|
print(logging.warn(error))
|
||||||
else:
|
else:
|
||||||
|
duplicate_log_file = next(
|
||||||
|
(
|
||||||
|
elem
|
||||||
|
for elem in self.uploaded_log_info
|
||||||
|
if elem["filename"] == filename
|
||||||
|
and elem["author"] == author_id
|
||||||
|
),
|
||||||
|
None,
|
||||||
|
)
|
||||||
await message.channel.send(
|
await message.channel.send(
|
||||||
f"The log file `{filename}` appears to be a duplicate {author_mention}. Please upload a more recent file."
|
content=author_mention,
|
||||||
|
embed=Embed(
|
||||||
|
description=f"The log file `{filename}` appears to be a duplicate [already uploaded here]({duplicate_log_file['link']}). Please upload a more recent file.",
|
||||||
|
colour=self.ryujinx_blue,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
elif (
|
elif (
|
||||||
is_log_file
|
is_log_file
|
||||||
|
@ -602,23 +684,31 @@ class LogFileReader(Cog):
|
||||||
and message.channel.id in self.bot_log_allowed_channels.values()
|
and message.channel.id in self.bot_log_allowed_channels.values()
|
||||||
):
|
):
|
||||||
return await message.channel.send(
|
return await message.channel.send(
|
||||||
f"{author_mention} Your file does not match the Ryujinx log format. Please check your file."
|
content=author_mention,
|
||||||
|
embed=Embed(
|
||||||
|
description=f"Your file does not match the Ryujinx log format. Please check your file.",
|
||||||
|
colour=self.ryujinx_blue,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
elif (
|
elif (
|
||||||
is_log_file
|
is_log_file
|
||||||
and not message.channel.id in self.bot_log_allowed_channels.values()
|
and not message.channel.id in self.bot_log_allowed_channels.values()
|
||||||
):
|
):
|
||||||
return await message.author.send(
|
return await message.author.send(
|
||||||
"\n".join(
|
content=author_mention,
|
||||||
(
|
embed=Embed(
|
||||||
f"{author_mention} Please upload Ryujinx log files to the correct location:\n",
|
description="\n".join(
|
||||||
f'<#{config.bot_log_allowed_channels["support"]}>: General help and troubleshooting',
|
(
|
||||||
f'<#{config.bot_log_allowed_channels["patreon-support"]}>: Help and troubleshooting for Patreon subscribers',
|
f"Please upload Ryujinx log files to the correct location:\n",
|
||||||
f'<#{config.bot_log_allowed_channels["development"]}>: Ryujinx development discussion',
|
f'<#{config.bot_log_allowed_channels["support"]}>: General help and troubleshooting',
|
||||||
f'<#{config.bot_log_allowed_channels["pr-testing"]}>: Discussion of in-progress pull request builds',
|
f'<#{config.bot_log_allowed_channels["patreon-support"]}>: Help and troubleshooting for Patreon subscribers',
|
||||||
f'<#{config.bot_log_allowed_channels["linux-master-race"]}>: Linux support and discussion',
|
f'<#{config.bot_log_allowed_channels["development"]}>: Ryujinx development discussion',
|
||||||
)
|
f'<#{config.bot_log_allowed_channels["pr-testing"]}>: Discussion of in-progress pull request builds',
|
||||||
)
|
f'<#{config.bot_log_allowed_channels["linux-master-race"]}>: Linux support and discussion',
|
||||||
|
)
|
||||||
|
),
|
||||||
|
colour=self.ryujinx_blue,
|
||||||
|
),
|
||||||
)
|
)
|
||||||
except IndexError:
|
except IndexError:
|
||||||
pass
|
pass
|
||||||
|
|
Loading…
Reference in a new issue