Browse Source

Fix user dirs, add docs

Ely Alon 3 months ago
parent
commit
17746db555
4 changed files with 223 additions and 57 deletions
  1. 3 0
      core/os/os2/errors.odin
  2. 90 7
      core/os/os2/user.odin
  3. 117 43
      core/os/os2/user_posix.odin
  4. 13 7
      core/os/os2/user_windows.odin

+ 3 - 0
core/os/os2/errors.odin

@@ -27,6 +27,8 @@ General_Error :: enum u32 {
 
 	Pattern_Has_Separator,
 
+	No_HOME_Variable,
+
 	Unsupported,
 }
 
@@ -73,6 +75,7 @@ error_string :: proc(ferr: Error) -> string {
 		case .Invalid_Command:   return "invalid command"
 		case .Unsupported:       return "unsupported"
 		case .Pattern_Has_Separator: return "pattern has separator"
+		case .No_HOME_Variable:  return "no $HOME variable"
 		}
 	case io.Error:
 		switch e {

+ 90 - 7
core/os/os2/user.odin

@@ -2,64 +2,147 @@ package os2
 
 import "base:runtime"
 
+// ```
+// Windows:  C:\Users\Alice
+// macOS:    /Users/Alice
+// Linux:    /home/alice
+// ```
 @(require_results)
 user_home_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_home_dir(allocator)
 }
 
-// application caches, logs, temporary files
+// Files that applications can regenerate/refetch at a loss of speed, e.g. shader caches
+//
+// Sometimes deleted for system maintenance
+//
+// ```
+// Windows:  C:\Users\Alice\AppData\Local
+// macOS:    /Users/Alice/Library/Caches
+// Linux:    /home/alice/.cache
+// ```
 @(require_results)
 user_cache_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_cache_dir(allocator)
 }
 
-// application assets
+// User-hidden application data
+//
+// ```
+// Windows:  C:\Users\Alice\AppData\Local ("C:\Users\Alice\AppData\Roaming" if `roaming`)
+// macOS:    /Users/Alice/Library/Application Support
+// Linux:    /home/alice/.local/share
+// ```
+//
+// NOTE: (Windows only) `roaming` is for syncing across multiple devices within a *domain network*
 @(require_results)
-user_data_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
-	return _user_data_dir(allocator)
+user_data_dir :: proc(allocator: runtime.Allocator, roaming := false) -> (dir: string, err: Error) {
+	return _user_data_dir(allocator, roaming)
 }
 
-// application history, ui layout state, logs
+// Non-essential application data, e.g. history, ui layout state
+//
+// ```
+// Windows:  C:\Users\Alice\AppData\Local
+// macOS:    /Users/Alice/Library/Application Support
+// Linux:    /home/alice/.local/state
+// ```
 @(require_results)
 user_state_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_state_dir(allocator)
 }
 
+// Application log files
+//
+// ```
+// Windows:  C:\Users\Alice\AppData\Local
+// macOS:    /Users/Alice/Library/Logs
+// Linux:    /home/alice/.local/state
+// ```
 @(require_results)
-user_config_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
-	return _user_config_dir(allocator)
+user_log_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
+	return _user_log_dir(allocator)
 }
 
+// Application settings/preferences
+//
+// ```
+// Windows:  C:\Users\Alice\AppData\Local ("C:\Users\Alice\AppData\Roaming" if `roaming`)
+// macOS:    /Users/Alice/Library/Application Support
+// Linux:    /home/alice/.config
+// ```
+//
+// NOTE: (Windows only) `roaming` is for syncing across multiple devices within a *domain network*
+@(require_results)
+user_config_dir :: proc(allocator: runtime.Allocator, roaming := false) -> (dir: string, err: Error) {
+	return _user_config_dir(allocator, roaming)
+}
+
+// ```
+// Windows:  C:\Users\Alice\Music
+// macOS:    /Users/Alice/Music
+// Linux:    /home/alice/Music
+// ```
 @(require_results)
 user_music_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_music_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Desktop
+// macOS:    /Users/Alice/Desktop
+// Linux:    /home/alice/Desktop
+// ```
 @(require_results)
 user_desktop_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_desktop_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Documents
+// macOS:    /Users/Alice/Documents
+// Linux:    /home/alice/Documents
+// ```
 @(require_results)
 user_documents_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_documents_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Downloads
+// macOS:    /Users/Alice/Downloads
+// Linux:    /home/alice/Downloads
+// ```
 @(require_results)
 user_downloads_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_downloads_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Pictures
+// macOS:    /Users/Alice/Pictures
+// Linux:    /home/alice/Pictures
+// ```
 @(require_results)
 user_pictures_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_pictures_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Public
+// macOS:    /Users/Alice/Public
+// Linux:    /home/alice/Public
+// ```
 @(require_results)
 user_public_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_public_dir(allocator)
 }
 
+// ```
+// Windows:  C:\Users\Alice\Videos
+// macOS:    /Users/Alice/Movies
+// Linux:    /home/alice/Videos
+// ```
 @(require_results)
 user_videos_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	return _user_videos_dir(allocator)

+ 117 - 43
core/os/os2/user_posix.odin

@@ -2,103 +2,115 @@
 package os2
 
 import "base:runtime"
+import "core:strings"
+import "core:sys/posix"
+import "core:fmt" //remove
 
 _user_cache_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Library/Caches", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_CACHE_HOME", "HOME", "/.cache", allocator)
+		return _xdg_lookup("", "/Library/Caches", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_CACHE_HOME", "/.cache", allocator)
 	}
 }
 
-_user_config_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
+_user_config_dir :: proc(allocator: runtime.Allocator, _roaming: bool) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/.config", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_CONFIG_HOME", "HOME", "/.config", allocator)
+		return _xdg_lookup("", "/Library/Application Support", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_CONFIG_HOME", "/.config", allocator)
 	}
 }
 
 _user_state_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/.local/state", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_STATE_HOME", "HOME", "/.local/state", allocator)
+		return _xdg_lookup("", "/Library/Application Support", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_STATE_HOME", "/.local/state", allocator)
 	}
 }
 
-_user_data_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
+_user_log_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/.local/share", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_DATA_HOME", "HOME", "/.local/share", allocator)
+		return _xdg_lookup("", "/Library/Logs", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_STATE_HOME", "/.local/state", allocator)
+	}
+}
+
+_user_data_dir :: proc(allocator: runtime.Allocator, _roaming: bool) -> (dir: string, err: Error) {
+	#partial switch ODIN_OS {
+	case .Darwin:
+		return _xdg_lookup("", "/Library/Application Support", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_DATA_HOME", "/.local/share", allocator)
 	}
 }
 
 _user_music_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Music", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_MUSIC_DIR", "HOME", "/Music", allocator)
+		return _xdg_lookup("", "/Music", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_MUSIC_DIR", "/Music", allocator)
 	}
 }
 
 _user_desktop_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Desktop", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_DESKTOP_DIR", "HOME", "/Desktop", allocator)
+		return _xdg_lookup("", "/Desktop", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_DESKTOP_DIR", "/Desktop", allocator)
 	}
 }
 
 _user_documents_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Documents", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_DOCUMENTS_DIR", "HOME", "/Documents", allocator)
+		return _xdg_lookup("", "/Documents", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_DOCUMENTS_DIR", "/Documents", allocator)
 	}
 }
 
 _user_downloads_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Downloads", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_DOWNLOAD_DIR", "HOME", "/Downloads", allocator)
+		return _xdg_lookup("", "/Downloads", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_DOWNLOAD_DIR", "/Downloads", allocator)
 	}
 }
 
 _user_pictures_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Pictures", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_PICTURES_DIR", "HOME", "/Pictures", allocator)
+		return _xdg_lookup("", "/Pictures", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_PICTURES_DIR", "/Pictures", allocator)
 	}
 }
 
 _user_public_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Public", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_PUBLIC_DIR", "HOME", "/Public", allocator)
+		return _xdg_lookup("", "/Public", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_PUBLICSHARE_DIR", "/Public", allocator)
 	}
 }
 
 _user_videos_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	#partial switch ODIN_OS {
 	case .Darwin:
-		return _xdg_lookup("", "HOME", "/Movies", allocator)
-	case: // All other UNIX systems
-		return _xdg_lookup("XDG_VIDEOS_DIR", "HOME", "/Videos", allocator)
+		return _xdg_lookup("", "/Movies", allocator)
+	case: // Unix
+		return _xdg_lookup("XDG_VIDEOS_DIR", "/Videos", allocator)
 	}
 }
 
@@ -106,27 +118,89 @@ _user_home_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error
 	if v := get_env("HOME", allocator); v != "" {
 		return v, nil
 	}
-	return "", .Invalid_Path
+	err = .No_HOME_Variable
+	return
 }
 
-_xdg_lookup :: proc(xdg_env, fallback_env: string, fallback_suffix: string, allocator: runtime.Allocator) -> (dir: string, err: Error) {
+_xdg_lookup :: proc(xdg_key: string, fallback_suffix: string, allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	temp_allocator  := TEMP_ALLOCATOR_GUARD({ allocator })
 
-	if xdg_env == "" { // Darwin doesn't have XDG paths.
-		dir = get_env(fallback_env, temp_allocator)
+	if xdg_key == "" { // Darwin doesn't have XDG paths.
+		dir = get_env("HOME", temp_allocator)
 		if dir == "" {
-			return "", .Invalid_Path
+			err = .No_HOME_Variable
+			return
 		}
 		return concatenate({dir, fallback_suffix}, allocator)
 	} else {
-		dir = get_env(xdg_env, allocator)
+		if strings.ends_with(xdg_key, "_DIR") {
+			dir = _xdg_user_dirs_lookup(xdg_key, allocator) or_return
+		} else {
+			dir = get_env(xdg_key, allocator)
+		}
+
 		if dir == "" {
-			dir = get_env(fallback_env, temp_allocator)
+			dir = get_env("HOME", temp_allocator)
 			if dir == "" {
-				return "", .Invalid_Path
+				err = .No_HOME_Variable
+				return
 			}
 			dir = concatenate({dir, fallback_suffix}, allocator) or_return
 		}
 		return
 	}
+}
+
+// If `<config-dir>/user-dirs.dirs` doesn't exist, or `xdg_key` can't be found there: returns `""`
+_xdg_user_dirs_lookup :: proc(xdg_key: string, allocator: runtime.Allocator) -> (dir: string, err: Error) {
+	temp_allocator  := TEMP_ALLOCATOR_GUARD({ allocator })
+
+	config_dir := user_config_dir(temp_allocator) or_return
+
+	user_dirs_path := concatenate({config_dir, "/user-dirs.dirs"}, temp_allocator) or_return
+	user_dirs_content_bytes, read_err := read_entire_file(user_dirs_path, temp_allocator)
+	if read_err == .Not_Exist {
+		return
+	} else if read_err != nil {
+		err = read_err
+		return
+	}
+	user_dirs_content := string(user_dirs_content_bytes)
+
+	lines := strings.split_lines(user_dirs_content, temp_allocator) or_return
+
+	home_env := get_env("HOME", temp_allocator)
+	if home_env == "" {
+		err = .No_HOME_Variable
+		return
+	}
+
+	for line in lines {
+		ss := strings.split_n(line, "=", 2, temp_allocator) or_return
+		(len(ss) == 2) or_continue
+		sl := strings.trim_space(ss[0])
+		sr := ss[1]
+
+		(sl == xdg_key) or_continue
+
+		(len(sr) > 2) or_continue
+
+		lq := strings.index_byte(sr, '"')
+		(lq != -1) or_continue
+		rq := strings.index_byte(sr[lq+1:], '"') + lq+1
+		(rq != -1) or_continue
+
+		sr = sr[lq+1:rq]
+
+		we: posix.wordexp_t
+		we_err := posix.wordexp(strings.clone_to_cstring(sr, temp_allocator), &we, nil)
+		(we_err == nil) or_continue
+		defer posix.wordfree(&we)
+
+		(we.we_wordc == 1) or_continue
+
+		dir = strings.clone_from_cstring(we.we_wordv[0], allocator) or_return
+		return
+	}
+	return
 }

+ 13 - 7
core/os/os2/user_windows.odin

@@ -3,25 +3,31 @@ package os2
 import "base:runtime"
 @(require) import win32 "core:sys/windows"
 
-_user_cache_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
+_local_appdata :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	guid := win32.FOLDERID_LocalAppData
 	return _get_known_folder_path(&guid, allocator)
 }
 
-_user_config_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
-	guid := win32.FOLDERID_RoamingAppData
+_local_appdata_or_roaming :: proc(allocator: runtime.Allocator, roaming: bool) -> (dir: string, err: Error) {
+	guid := win32.FOLDERID_LocalAppData
+	if roaming {
+		guid = win32.FOLDERID_RoamingAppData
+	}
 	return _get_known_folder_path(&guid, allocator)
 }
 
+_user_config_dir :: _local_appdata_or_roaming
+_user_data_dir :: _local_appdata_or_roaming
+
+_user_state_dir :: _local_appdata
+_user_log_dir :: _local_appdata
+_user_cache_dir :: _local_appdata
+
 _user_home_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	guid := win32.FOLDERID_Profile
 	return _get_known_folder_path(&guid, allocator)
 }
 
-_user_data_dir :: _user_config_dir
-
-_user_state_dir :: _user_cache_dir
-
 _user_music_dir :: proc(allocator: runtime.Allocator) -> (dir: string, err: Error) {
 	guid := win32.FOLDERID_Music
 	return _get_known_folder_path(&guid, allocator)