| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326 |
- using Ryujinx.Common.Logging;
- using Ryujinx.Common.Utilities;
- using System;
- using System.IO;
- using System.Runtime.Versioning;
- namespace Ryujinx.Common.Configuration
- {
- public static class AppDataManager
- {
- private const string DefaultBaseDir = "Ryujinx";
- private const string DefaultPortableDir = "portable";
- // The following 3 are always part of Base Directory
- private const string GamesDir = "games";
- private const string ProfilesDir = "profiles";
- private const string KeysDir = "system";
- public enum LaunchMode
- {
- UserProfile,
- Portable,
- Custom,
- }
- public static LaunchMode Mode { get; private set; }
- public static string BaseDirPath { get; private set; }
- public static string GamesDirPath { get; private set; }
- public static string ProfilesDirPath { get; private set; }
- public static string KeysDirPath { get; private set; }
- public static string KeysDirPathUser { get; }
- public static string LogsDirPath { get; private set; }
- public const string DefaultNandDir = "bis";
- public const string DefaultSdcardDir = "sdcard";
- private const string DefaultModsDir = "mods";
- public static string CustomModsPath { get; set; }
- public static string CustomSdModsPath { get; set; }
- public static string CustomNandPath { get; set; } // TODO: Actually implement this into VFS
- public static string CustomSdCardPath { get; set; } // TODO: Actually implement this into VFS
- static AppDataManager()
- {
- KeysDirPathUser = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".switch");
- }
- public static void Initialize(string baseDirPath)
- {
- string appDataPath = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData);
- if (appDataPath.Length == 0)
- {
- appDataPath = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
- }
- string userProfilePath = Path.Combine(appDataPath, DefaultBaseDir);
- string portablePath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, DefaultPortableDir);
- // On macOS, check for a portable directory next to the app bundle as well.
- if (OperatingSystem.IsMacOS() && !Directory.Exists(portablePath))
- {
- string bundlePath = Path.GetFullPath(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "..", ".."));
- // Make sure we're actually running within an app bundle.
- if (bundlePath.EndsWith(".app"))
- {
- portablePath = Path.GetFullPath(Path.Combine(bundlePath, "..", DefaultPortableDir));
- }
- }
- if (Directory.Exists(portablePath))
- {
- BaseDirPath = portablePath;
- Mode = LaunchMode.Portable;
- }
- else
- {
- BaseDirPath = userProfilePath;
- Mode = LaunchMode.UserProfile;
- }
- if (baseDirPath != null && baseDirPath != userProfilePath)
- {
- if (!Directory.Exists(baseDirPath))
- {
- Logger.Error?.Print(LogClass.Application, $"Custom Data Directory '{baseDirPath}' does not exist. Falling back to {Mode}...");
- }
- else
- {
- BaseDirPath = baseDirPath;
- Mode = LaunchMode.Custom;
- }
- }
- BaseDirPath = Path.GetFullPath(BaseDirPath); // convert relative paths
- if (IsPathSymlink(BaseDirPath))
- {
- Logger.Warning?.Print(LogClass.Application, $"Application data directory is a symlink. This may be unintended.");
- }
- SetupBasePaths();
- }
- public static string GetOrCreateLogsDir()
- {
- if (Directory.Exists(LogsDirPath))
- {
- return LogsDirPath;
- }
- Logger.Notice.Print(LogClass.Application, "Logging directory not found; attempting to create new logging directory.");
- LogsDirPath = SetUpLogsDir();
- return LogsDirPath;
- }
- private static string SetUpLogsDir()
- {
- string logDir = "";
- if (Mode == LaunchMode.Portable)
- {
- logDir = Path.Combine(BaseDirPath, "Logs");
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- return null;
- }
- }
- else
- {
- if (OperatingSystem.IsMacOS())
- {
- // NOTE: Should evaluate to "~/Library/Logs/Ryujinx/".
- logDir = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), "Library", "Logs", DefaultBaseDir);
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- logDir = "";
- }
- if (string.IsNullOrEmpty(logDir))
- {
- // NOTE: Should evaluate to "~/Library/Application Support/Ryujinx/Logs".
- logDir = Path.Combine(BaseDirPath, "Logs");
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- return null;
- }
- }
- }
- else if (OperatingSystem.IsWindows())
- {
- // NOTE: Should evaluate to a "Logs" directory in whatever directory Ryujinx was launched from.
- logDir = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "Logs");
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- logDir = "";
- }
- if (string.IsNullOrEmpty(logDir))
- {
- // NOTE: Should evaluate to "C:\Users\user\AppData\Roaming\Ryujinx\Logs".
- logDir = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), DefaultBaseDir, "Logs");
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- return null;
- }
- }
- }
- else if (OperatingSystem.IsLinux())
- {
- // NOTE: Should evaluate to "~/.config/Ryujinx/Logs".
- logDir = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), DefaultBaseDir, "Logs");
- try
- {
- Directory.CreateDirectory(logDir);
- }
- catch
- {
- Logger.Warning?.Print(LogClass.Application, $"Logging directory could not be created '{logDir}'");
- return null;
- }
- }
- }
- return logDir;
- }
- private static void SetupBasePaths()
- {
- Directory.CreateDirectory(BaseDirPath);
- LogsDirPath = SetUpLogsDir();
- Directory.CreateDirectory(GamesDirPath = Path.Combine(BaseDirPath, GamesDir));
- Directory.CreateDirectory(ProfilesDirPath = Path.Combine(BaseDirPath, ProfilesDir));
- Directory.CreateDirectory(KeysDirPath = Path.Combine(BaseDirPath, KeysDir));
- }
- // Check if existing old baseDirPath is a symlink, to prevent possible errors.
- // Should be removed, when the existence of the old directory isn't checked anymore.
- private static bool IsPathSymlink(string path)
- {
- try
- {
- FileAttributes attributes = File.GetAttributes(path);
- return (attributes & FileAttributes.ReparsePoint) == FileAttributes.ReparsePoint;
- }
- catch
- {
- return false;
- }
- }
- [SupportedOSPlatform("macos")]
- public static void FixMacOSConfigurationFolders()
- {
- string oldConfigPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile),
- ".config", DefaultBaseDir);
- if (Path.Exists(oldConfigPath) && !IsPathSymlink(oldConfigPath) && !Path.Exists(BaseDirPath))
- {
- FileSystemUtils.MoveDirectory(oldConfigPath, BaseDirPath);
- Directory.CreateSymbolicLink(oldConfigPath, BaseDirPath);
- }
- string correctApplicationDataDirectoryPath =
- Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData), DefaultBaseDir);
- if (IsPathSymlink(correctApplicationDataDirectoryPath))
- {
- //copy the files somewhere temporarily
- string tempPath = Path.Combine(Path.GetTempPath(), DefaultBaseDir);
- try
- {
- FileSystemUtils.CopyDirectory(correctApplicationDataDirectoryPath, tempPath, true);
- }
- catch (Exception exception)
- {
- Logger.Error?.Print(LogClass.Application,
- $"Critical error copying Ryujinx application data into the temp folder. {exception}");
- try
- {
- FileSystemInfo resolvedDirectoryInfo =
- Directory.ResolveLinkTarget(correctApplicationDataDirectoryPath, true);
- string resolvedPath = resolvedDirectoryInfo.FullName;
- Logger.Error?.Print(LogClass.Application, $"Please manually move your Ryujinx data from {resolvedPath} to {correctApplicationDataDirectoryPath}, and remove the symlink.");
- }
- catch (Exception symlinkException)
- {
- Logger.Error?.Print(LogClass.Application, $"Unable to resolve the symlink for Ryujinx application data: {symlinkException}. Follow the symlink at {correctApplicationDataDirectoryPath} and move your data back to the Application Support folder.");
- }
- return;
- }
- //delete the symlink
- try
- {
- //This will fail if this is an actual directory, so there is no way we can actually delete user data here.
- File.Delete(correctApplicationDataDirectoryPath);
- }
- catch (Exception exception)
- {
- Logger.Error?.Print(LogClass.Application,
- $"Critical error deleting the Ryujinx application data folder symlink at {correctApplicationDataDirectoryPath}. {exception}");
- try
- {
- FileSystemInfo resolvedDirectoryInfo =
- Directory.ResolveLinkTarget(correctApplicationDataDirectoryPath, true);
- string resolvedPath = resolvedDirectoryInfo.FullName;
- Logger.Error?.Print(LogClass.Application, $"Please manually move your Ryujinx data from {resolvedPath} to {correctApplicationDataDirectoryPath}, and remove the symlink.");
- }
- catch (Exception symlinkException)
- {
- Logger.Error?.Print(LogClass.Application, $"Unable to resolve the symlink for Ryujinx application data: {symlinkException}. Follow the symlink at {correctApplicationDataDirectoryPath} and move your data back to the Application Support folder.");
- }
- return;
- }
- //put the files back
- try
- {
- FileSystemUtils.CopyDirectory(tempPath, correctApplicationDataDirectoryPath, true);
- }
- catch (Exception exception)
- {
- Logger.Error?.Print(LogClass.Application,
- $"Critical error copying Ryujinx application data into the correct location. {exception}. Please manually move your application data from {tempPath} to {correctApplicationDataDirectoryPath}.");
- }
- }
- }
- public static string GetModsPath() => CustomModsPath ?? Directory.CreateDirectory(Path.Combine(BaseDirPath, DefaultModsDir)).FullName;
- public static string GetSdModsPath() => CustomSdModsPath ?? Directory.CreateDirectory(Path.Combine(BaseDirPath, DefaultSdcardDir, "atmosphere")).FullName;
- }
- }
|