mirror of
				https://github.com/JHDev2006/Super-Mario-Bros.-Remastered-Public.git
				synced 2025-10-25 00:40:54 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			600 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			GDScript
		
	
	
	
	
	
			
		
		
	
	
			600 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			GDScript
		
	
	
	
	
	
| @tool
 | |
| class_name ModLoaderLog
 | |
| extends Object
 | |
| ##
 | |
| ## This Class provides methods for logging, retrieving logged data, and internal methods for working with log files.
 | |
| 
 | |
| 
 | |
| # Path to the latest log file.
 | |
| const MOD_LOG_PATH := "user://logs/modloader.log"
 | |
| 
 | |
| const _LOG_NAME := "ModLoader:Log"
 | |
| 
 | |
| ## Denotes the severity of a log entry
 | |
| enum VERBOSITY_LEVEL {
 | |
| 	ERROR, 		## For errors and fatal errors
 | |
| 	WARNING, 	## For warnings
 | |
| 	INFO, 		## For everything informational and successes
 | |
| 	DEBUG, 		## For debugging, can get quite verbose
 | |
| }
 | |
| 
 | |
| ## Keeps track of logged messages, to avoid flooding the log with duplicate notices
 | |
| ## Can also be used by mods, eg. to create an in-game developer console that
 | |
| ## shows messages
 | |
| static var logged_messages := {
 | |
| 	"all": {},
 | |
| 	"by_mod": {},
 | |
| 	"by_type": {
 | |
| 		"fatal-error": {},
 | |
| 		"error": {},
 | |
| 		"warning": {},
 | |
| 		"info": {},
 | |
| 		"success": {},
 | |
| 		"debug": {},
 | |
| 		"hint": {},
 | |
| 	}
 | |
| }
 | |
| 
 | |
| ## Verbosity/Logging level.
 | |
| ## Used to filter out messages below the set level
 | |
| ## (if the [enum VERBOSITY_LEVEL] int of a new entry is larger than the [member verbosity] it is ignored)
 | |
| static var verbosity: VERBOSITY_LEVEL = VERBOSITY_LEVEL.DEBUG
 | |
| 
 | |
| ## Array of mods that should be ignored when logging messages (contains mod IDs as strings)
 | |
| static var ignored_mods: Array[String] = []
 | |
| 
 | |
| ## Highlighting color for hint type log messages
 | |
| static var hint_color := Color("#70bafa")
 | |
| 
 | |
| ## This Sub-Class represents a log entry in ModLoader.
 | |
| class ModLoaderLogEntry:
 | |
| 	extends Resource
 | |
| 
 | |
| 	## Name of the mod or ModLoader class this entry refers to.
 | |
| 	var mod_name: String
 | |
| 
 | |
| 	## The message of the log entry.
 | |
| 	var message: String
 | |
| 
 | |
| 	## The log type, which indicates the verbosity level of this entry.
 | |
| 	var type: String
 | |
| 
 | |
| 	## The readable format of the time when this log entry was created.
 | |
| 	## Used for printing in the log file and output.
 | |
| 	var time: String
 | |
| 
 | |
| 	## The timestamp when this log entry was created.
 | |
| 	## Used for comparing and sorting log entries by time.
 | |
| 	var time_stamp: int
 | |
| 
 | |
| 	## An array of ModLoaderLogEntry objects.
 | |
| 	## If the message has been logged before, it is added to the stack.
 | |
| 	var stack := []
 | |
| 
 | |
| 
 | |
| 	## Initialize a ModLoaderLogEntry object with provided values.[br]
 | |
| 	##[br]
 | |
| 	## [b]Parameters:[/b][br]
 | |
| 	## [param _mod_name] ([String]): Name of the mod or ModLoader class this entry refers to.[br]
 | |
| 	## [param _message] ([String]): The message of the log entry.[br]
 | |
| 	## [param _type] ([String]): The log type, which indicates the verbosity level of this entry.[br]
 | |
| 	## [param _time] ([String]): The readable format of the time when this log entry was created.[br]
 | |
| 	##[br]
 | |
| 	## [b]Returns:[/b][br]
 | |
| 	## - No return value[br]
 | |
| 	func _init(_mod_name: String, _message: String, _type: String, _time: String) -> void:
 | |
| 		mod_name = _mod_name
 | |
| 		message = _message
 | |
| 		type = _type
 | |
| 		time = _time
 | |
| 		time_stamp = Time.get_ticks_msec()
 | |
| 
 | |
| 
 | |
| 	## Get the log entry as a formatted string.[br]
 | |
| 	## [br]
 | |
| 	## [b]Returns:[/b] [String]
 | |
| 	func get_entry() -> String:
 | |
| 		return str(time, get_prefix(), message)
 | |
| 
 | |
| 
 | |
| 	## Get the prefix string for the log entry, including the log type and mod name.[br]
 | |
| 	## [br]
 | |
| 	## [b]Returns:[/b] [String]
 | |
| 	func get_prefix() -> String:
 | |
| 		return "%s %s: " % [type.to_upper(), mod_name]
 | |
| 
 | |
| 
 | |
| 	## Generate an MD5 hash of the log entry (prefix + message).[br]
 | |
| 	## [br]
 | |
| 	## [b]Returns:[/b] [String]
 | |
| 	func get_md5() -> String:
 | |
| 		return str(get_prefix(), message).md5_text()
 | |
| 
 | |
| 
 | |
| 	## Get all log entries, including the current entry and entries in the stack.[br]
 | |
| 	## [br]
 | |
| 	## [b]Returns:[/b] [Array]
 | |
| 	func get_all_entries() -> Array:
 | |
| 		var entries := [self]
 | |
| 		entries.append_array(stack)
 | |
| 
 | |
| 		return entries
 | |
| 
 | |
| 
 | |
| # API log functions - logging
 | |
| # =============================================================================
 | |
| 
 | |
| 
 | |
| ## Logs the error in red and a stack trace. Prefixed FATAL-ERROR.[br]
 | |
| ## Always logged.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as an error.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| ## [br]
 | |
| ## ===[br]
 | |
| ## [b]Note:[color=bug "Breakpoint"][/color][/b][br]
 | |
| ## Stops execution in the editor, use this when something really needs to be fixed.[br]
 | |
| ## ===[br]
 | |
| static func fatal(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "fatal-error", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message and pushes an error. Prefixed ERROR.[br]
 | |
| ## Always logged.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as an error.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func error(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "error", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message and pushes a warning. Prefixed WARNING.[br]
 | |
| ## Logged with verbosity level at or above warning ([code]-v[/code] or [code]--log-warning[/code]).[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as a warning.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func warning(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "warning", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message. Prefixed INFO.[br]
 | |
| ## Logged with verbosity level at or above info ([code]-vv[/code] or [code]--log-info[/code]).[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as an information.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func info(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "info", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message. Prefixed SUCCESS.[br]
 | |
| ## Logged with verbosity level at or above info ([code]-vv[/code] or [code]--log-info[/code]).[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as a success.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func success(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "success", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message. Prefixed DEBUG.[br]
 | |
| ## Logged with verbosity level at or above debug ([code]-vvv[/code] or [code]--log-debug[/code]).[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as a debug.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func debug(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "debug", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message. Prefixed HINT and highligted.[br]
 | |
| ## Logged with verbosity level at or above debug ([code]-vvv[/code] or [code]--log-debug[/code]) and in the editor only. Not written to mod loader log.[br]
 | |
| ## ===[br]
 | |
| ## [b]Note:[/b][br]
 | |
| ## Use this to help other developers debug issues by giving them error-specific hints.[br]
 | |
| ## ===[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as a debug.[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func hint(message: String, mod_name: String, only_once := false) -> void:
 | |
| 	_log(message, mod_name, "hint", only_once)
 | |
| 
 | |
| 
 | |
| ## Logs the message formatted with [method JSON.print]. Prefixed DEBUG.[br]
 | |
| ## Logged with verbosity level at or above debug ([code]-vvv[/code] or [code]--log-debug[/code]).[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param message] ([String]): The message to be logged as a debug.[br]
 | |
| ## [param json_printable] (Variant): The variable to be formatted and printed using [method JSON.print].[br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with this log entry.[br]
 | |
| ## [param only_once] ([bool]): (Optional) If true, the log entry will only be logged once, even if called multiple times. Default is false.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - No return value[br]
 | |
| static func debug_json_print(message: String, json_printable, mod_name: String, only_once := false) -> void:
 | |
| 	message = "%s\n%s" % [message, JSON.stringify(json_printable, "  ")]
 | |
| 	_log(message, mod_name, "debug", only_once)
 | |
| 
 | |
| 
 | |
| # API log functions - stored logs
 | |
| # =============================================================================
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a resource.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as resource.
 | |
| static func get_all_as_resource() -> Array:
 | |
| 	return get_all()
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a string.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as strings.
 | |
| static func get_all_as_string() -> Array:
 | |
| 	var log_entries := get_all()
 | |
| 	return get_all_entries_as_string(log_entries)
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a resource for a specific mod_name.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as resource for the specified [code]mod_name[/code].
 | |
| static func get_by_mod_as_resource(mod_name: String) -> Array:
 | |
| 	return get_by_mod(mod_name)
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a string for a specific mod_name.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as strings for the specified [code]mod_name[/code].
 | |
| static func get_by_mod_as_string(mod_name: String) -> Array:
 | |
| 	var log_entries := get_by_mod(mod_name)
 | |
| 	return get_all_entries_as_string(log_entries)
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a resource for a specific type.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param type] ([String]): The log type associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as resource for the specified [code]type[/code].
 | |
| static func get_by_type_as_resource(type: String) -> Array:
 | |
| 	return get_by_type(type)
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries as a string for a specific type.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param type] ([String]): The log type associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as strings for the specified [code]type[/code].
 | |
| static func get_by_type_as_string(type: String) -> Array:
 | |
| 	var log_entries := get_by_type(type)
 | |
| 	return get_all_entries_as_string(log_entries)
 | |
| 
 | |
| 
 | |
| ## Returns an array of all log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of all log entries.
 | |
| static func get_all() -> Array:
 | |
| 	var log_entries := []
 | |
| 
 | |
| 	# Get all log entries
 | |
| 	for entry_key in logged_messages.all.keys():
 | |
| 		var entry: ModLoaderLogEntry = logged_messages.all[entry_key]
 | |
| 		log_entries.append_array(entry.get_all_entries())
 | |
| 
 | |
| 	# Sort them by time
 | |
| 	log_entries.sort_custom(Callable(ModLoaderLogCompare, "time"))
 | |
| 
 | |
| 	return log_entries
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries for a specific mod_name.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param mod_name] ([String]): The name of the mod or ModLoader class associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries for the specified [code]mod_name[/code].
 | |
| static func get_by_mod(mod_name: String) -> Array:
 | |
| 	var log_entries := []
 | |
| 
 | |
| 	if not logged_messages.by_mod.has(mod_name):
 | |
| 		error("\"%s\" not found in logged messages." % mod_name, _LOG_NAME)
 | |
| 		return []
 | |
| 
 | |
| 	for entry_key in logged_messages.by_mod[mod_name].keys():
 | |
| 		var entry: ModLoaderLogEntry = logged_messages.by_mod[mod_name][entry_key]
 | |
| 		log_entries.append_array(entry.get_all_entries())
 | |
| 
 | |
| 	return log_entries
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries for a specific type.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param type] ([String]): The log type associated with the log entries.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries for the specified [code]type[/code].
 | |
| static func get_by_type(type: String) -> Array:
 | |
| 	var log_entries := []
 | |
| 
 | |
| 	for entry_key in logged_messages.by_type[type].keys():
 | |
| 		var entry: ModLoaderLogEntry = logged_messages.by_type[type][entry_key]
 | |
| 		log_entries.append_array(entry.get_all_entries())
 | |
| 
 | |
| 	return log_entries
 | |
| 
 | |
| 
 | |
| ## Returns an array of log entries represented as strings.[br]
 | |
| ## [br]
 | |
| ## [b]Parameters:[/b][br]
 | |
| ## [param log_entries] ([Array]): An array of ModLoaderLogEntry Objects.[br]
 | |
| ## [br]
 | |
| ## [b]Returns:[/b][br]
 | |
| ## - [Array]: An array of log entries represented as strings.
 | |
| static func get_all_entries_as_string(log_entries: Array) -> Array:
 | |
| 	var log_entry_strings := []
 | |
| 
 | |
| 	# Get all the strings
 | |
| 	for entry in log_entries:
 | |
| 		log_entry_strings.push_back(entry.get_entry())
 | |
| 
 | |
| 	return log_entry_strings
 | |
| 
 | |
| 
 | |
| # Internal log functions
 | |
| # =============================================================================
 | |
| 
 | |
| static func _log(message: String, mod_name: String, log_type: String = "info", only_once := false) -> void:
 | |
| 	if _is_mod_name_ignored(mod_name):
 | |
| 		return
 | |
| 
 | |
| 	var time := "%s   " % _get_time_string()
 | |
| 	var log_entry := ModLoaderLogEntry.new(mod_name, message, log_type, time)
 | |
| 
 | |
| 	if only_once and _is_logged_before(log_entry):
 | |
| 		return
 | |
| 
 | |
| 	_store_log(log_entry)
 | |
| 
 | |
| 	# Check if the scene_tree is available
 | |
| 	if Engine.get_main_loop() and ModLoader:
 | |
| 		ModLoader.emit_signal("logged", log_entry)
 | |
| 
 | |
| 	_code_note(str(
 | |
| 		"If you are seeing this after trying to run the game, there is an error in your mod somewhere.",
 | |
| 		"Check the Debugger tab (below) to see the error.",
 | |
| 		"Click through the files listed in Stack Frames to trace where the error originated.",
 | |
| 		"View Godot's documentation for more info:",
 | |
| 		"https://docs.godotengine.org/en/stable/tutorials/scripting/debug/debugger_panel.html#doc-debugger-panel"
 | |
| 	))
 | |
| 
 | |
| 	match log_type.to_lower():
 | |
| 		"fatal-error":
 | |
| 			push_error(message)
 | |
| 			_write_to_log_file(log_entry.get_entry())
 | |
| 			_write_to_log_file(JSON.stringify(get_stack(), "  "))
 | |
| 			assert(false, message)
 | |
| 		"error":
 | |
| 			printerr(log_entry.get_prefix() + message)
 | |
| 			push_error(message)
 | |
| 			_write_to_log_file(log_entry.get_entry())
 | |
| 		"warning":
 | |
| 			if verbosity >= VERBOSITY_LEVEL.WARNING:
 | |
| 				print(log_entry.get_prefix() + message)
 | |
| 				push_warning(message)
 | |
| 				_write_to_log_file(log_entry.get_entry())
 | |
| 		"info", "success":
 | |
| 			if verbosity >= VERBOSITY_LEVEL.INFO:
 | |
| 				print(log_entry.get_prefix() + message)
 | |
| 				_write_to_log_file(log_entry.get_entry())
 | |
| 		"debug":
 | |
| 			if verbosity >= VERBOSITY_LEVEL.DEBUG:
 | |
| 				print(log_entry.get_prefix() + message)
 | |
| 				_write_to_log_file(log_entry.get_entry())
 | |
| 		"hint":
 | |
| 			if OS.has_feature("editor") and verbosity >= VERBOSITY_LEVEL.DEBUG:
 | |
| 				print_rich("[color=%s]%s[/color]" % [hint_color.to_html(false), log_entry.get_prefix() + message])
 | |
| 
 | |
| 
 | |
| static func _is_mod_name_ignored(mod_name: String) -> bool:
 | |
| 	if ignored_mods.is_empty():
 | |
| 		return false
 | |
| 
 | |
| 	if mod_name in ignored_mods:
 | |
| 		return true
 | |
| 
 | |
| 	return false
 | |
| 
 | |
| 
 | |
| static func _store_log(log_entry: ModLoaderLogEntry) -> void:
 | |
| 	var existing_entry: ModLoaderLogEntry
 | |
| 
 | |
| 	# Store in all
 | |
| 	# If it's a new entry
 | |
| 	if not logged_messages.all.has(log_entry.get_md5()):
 | |
| 		logged_messages.all[log_entry.get_md5()] = log_entry
 | |
| 	# If it's a existing entry
 | |
| 	else:
 | |
| 		existing_entry = logged_messages.all[log_entry.get_md5()]
 | |
| 		existing_entry.time = log_entry.time
 | |
| 		existing_entry.stack.push_back(log_entry)
 | |
| 
 | |
| 	# Store in by_mod
 | |
| 	# If the mod is not yet in "by_mod" init the entry
 | |
| 	if not logged_messages.by_mod.has(log_entry.mod_name):
 | |
| 		logged_messages.by_mod[log_entry.mod_name] = {}
 | |
| 
 | |
| 	logged_messages.by_mod[log_entry.mod_name][log_entry.get_md5()] = log_entry if not existing_entry else existing_entry
 | |
| 
 | |
| 	# Store in by_type
 | |
| 	logged_messages.by_type[log_entry.type.to_lower()][log_entry.get_md5()] = log_entry if not existing_entry else existing_entry
 | |
| 
 | |
| 
 | |
| static func _is_logged_before(entry: ModLoaderLogEntry) -> bool:
 | |
| 	if not logged_messages.all.has(entry.get_md5()):
 | |
| 		return false
 | |
| 
 | |
| 	return true
 | |
| 
 | |
| 
 | |
| class ModLoaderLogCompare:
 | |
| 	# Custom sorter that orders logs by time
 | |
| 	static func time(a: ModLoaderLogEntry, b: ModLoaderLogEntry) -> bool:
 | |
| 		if a.time_stamp > b.time_stamp:
 | |
| 			return true # a -> b
 | |
| 		else:
 | |
| 			return false # b -> a
 | |
| 
 | |
| 
 | |
| # Internal Date Time
 | |
| # =============================================================================
 | |
| 
 | |
| # Returns the current time as a string in the format hh:mm:ss
 | |
| static func _get_time_string() -> String:
 | |
| 	var date_time := Time.get_datetime_dict_from_system()
 | |
| 	return "%02d:%02d:%02d" % [ date_time.hour, date_time.minute, date_time.second ]
 | |
| 
 | |
| 
 | |
| # Returns the current date as a string in the format yyyy-mm-dd
 | |
| static func _get_date_string() -> String:
 | |
| 	var date_time := Time.get_datetime_dict_from_system()
 | |
| 	return "%s-%02d-%02d" % [ date_time.year, date_time.month, date_time.day ]
 | |
| 
 | |
| 
 | |
| # Returns the current date and time as a string in the format yyyy-mm-dd_hh:mm:ss
 | |
| static func _get_date_time_string() -> String:
 | |
| 	return "%s_%s" % [ _get_date_string(), _get_time_string() ]
 | |
| 
 | |
| 
 | |
| 
 | |
| # Internal File
 | |
| # =============================================================================
 | |
| 
 | |
| static func _write_to_log_file(string_to_write: String) -> void:
 | |
| 	if not FileAccess.file_exists(MOD_LOG_PATH):
 | |
| 		_rotate_log_file()
 | |
| 
 | |
| 	var log_file := FileAccess.open(MOD_LOG_PATH, FileAccess.READ_WRITE)
 | |
| 
 | |
| 	if log_file == null:
 | |
| 		assert(false, "Could not open log file, error code: %s" % error)
 | |
| 		return
 | |
| 
 | |
| 	log_file.seek_end()
 | |
| 	log_file.store_string("\n" + string_to_write)
 | |
| 	log_file.close()
 | |
| 
 | |
| 
 | |
| # Keeps log backups for every run, just like the Godot gdscript implementation of
 | |
| # https://github.com/godotengine/godot/blob/1d14c054a12dacdc193b589e4afb0ef319ee2aae/core/io/logger.cpp#L151
 | |
| static func _rotate_log_file() -> void:
 | |
| 	var MAX_LOGS: int = ProjectSettings.get_setting("debug/file_logging/max_log_files")
 | |
| 
 | |
| 	if FileAccess.file_exists(MOD_LOG_PATH):
 | |
| 		if MAX_LOGS > 1:
 | |
| 			var datetime := _get_date_time_string().replace(":", ".")
 | |
| 			var backup_name: String = MOD_LOG_PATH.get_basename() + "_" + datetime
 | |
| 			if MOD_LOG_PATH.get_extension().length() > 0:
 | |
| 				backup_name += "." + MOD_LOG_PATH.get_extension()
 | |
| 
 | |
| 			var dir := DirAccess.open(MOD_LOG_PATH.get_base_dir())
 | |
| 			if not dir == null:
 | |
| 				dir.copy(MOD_LOG_PATH, backup_name)
 | |
| 			_clear_old_log_backups()
 | |
| 
 | |
| 	# only File.WRITE creates a new file, File.READ_WRITE throws an error
 | |
| 	var log_file := FileAccess.open(MOD_LOG_PATH, FileAccess.WRITE)
 | |
| 	if log_file == null:
 | |
| 		assert(false, "Could not open log file, error code: %s" % error)
 | |
| 	log_file.store_string('%s Created log' % _get_date_string())
 | |
| 	log_file.close()
 | |
| 
 | |
| 
 | |
| static func _clear_old_log_backups() -> void:
 | |
| 	var MAX_LOGS := int(ProjectSettings.get_setting("debug/file_logging/max_log_files"))
 | |
| 	var MAX_BACKUPS := MAX_LOGS - 1 # -1 for the current new log (not a backup)
 | |
| 	var basename := MOD_LOG_PATH.get_file().get_basename() as String
 | |
| 	var extension := MOD_LOG_PATH.get_extension() as String
 | |
| 
 | |
| 	var dir := DirAccess.open(MOD_LOG_PATH.get_base_dir())
 | |
| 	if dir == null:
 | |
| 		return
 | |
| 
 | |
| 	dir.list_dir_begin() # TODOGODOT4 fill missing arguments https://github.com/godotengine/godot/pull/40547
 | |
| 	var file := dir.get_next()
 | |
| 	var backups := []
 | |
| 	while file.length() > 0:
 | |
| 		if (not dir.current_is_dir() and
 | |
| 				file.begins_with(basename) and
 | |
| 				file.get_extension() == extension and
 | |
| 				not file == MOD_LOG_PATH.get_file()):
 | |
| 			backups.append(file)
 | |
| 		file = dir.get_next()
 | |
| 	dir.list_dir_end()
 | |
| 
 | |
| 	if backups.size() > MAX_BACKUPS:
 | |
| 		backups.sort()
 | |
| 		backups.resize(backups.size() - MAX_BACKUPS)
 | |
| 		for file_to_delete in backups:
 | |
| 			dir.remove(file_to_delete)
 | |
| 
 | |
| 
 | |
| # Internal util funcs
 | |
| # =============================================================================
 | |
| # This are duplicates of the functions in mod_loader_utils.gd to prevent
 | |
| # a cyclic reference error between ModLoaderLog and ModLoaderUtils.
 | |
| 
 | |
| 
 | |
| # This is a dummy func. It is exclusively used to show notes in the code that
 | |
| # stay visible after decompiling a PCK, as is primarily intended to assist new
 | |
| # modders in understanding and troubleshooting issues.
 | |
| static func _code_note(_msg:String):
 | |
| 	pass
 | 
