Как получить путь к папке оболочки, такой как «Локальные настройки» или «Локальные данные приложения», для определенного пользователя, отличного от текущего пользователя?
Получение специального пути к папке для данного пользователя в Jscript
Ответы (1)
Хотя существуют методы получения специальных путей к папкам на узле сценариев Windows WshShell.SpecialFolders
и Shell.NameSpace
они возвращают пути только для текущего пользователя. Получить специальные пути к папкам других пользователей немного сложно.
Правильный способ сделать это — использовать Windows API SHGetKnownFolderPath
(или SHGetFolderPath
в версиях Windows до Vista ). Но проблема в том, что Windows Script Host не поддерживает вызов функций WinAPI, поэтому, чтобы использовать эти функции в своем сценарии, вам придется предоставлять их через специально написанный COM-компонент.
Другим возможным, но недокументированным решением является чтение путей к специальным папкам из куста реестра этого пользователя, в частности, ключа HKEY_USERS\<user_SID>\Software\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders
.
Пути в ключе User Shell Folders
обычно указываются с помощью переменной среды %USERPROFILE%
; поэтому, чтобы получить полные пути, вам придется заменить эту переменную значением ProfileImagePath
из ключа HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows NT\CurrentVersion\ProfileList\<user_SID>
.
Кроме того, клавиша HKEY_USERS\<user_SID>
доступна только тогда, когда соответствующий пользователь в данный момент находится в системе. В качестве общего решения вам придется загрузить куст пользователя (‹UserProfile›\ntuser.dat) во временный раздел реестра (скажем, HKEY_USERS\Temp
) и вместо этого прочитать значения из этого раздела.
Ниже приведен пример кода JScript, который демонстрирует, как можно выполнить вашу задачу. В Windows 7 и Vista вам может потребоваться запустить скрипт от имени администратора в зависимости от ваших настроек UAC.
ПРИМЕЧАНИЕ. Этот метод не рекомендуется, как объясняет Рэймонд Чен в своей статье Долгая и печальная история ключа Shell Folders. Нет никакой гарантии, что он продолжит работать в будущих версиях Windows.
var strUser = "foo";
var strDomain = "bar";
// If the account is local, domain name = computer name:
// var strDomain = getComputerName();
var strSID = getSID(strUser, strDomain);
var strProfilePath = getProfilePath(strSID);
// Load the user's registry hive into the HKEY_USERS\Temp key
var strTempKey = "Temp";
loadHKUHive(strTempKey, strProfilePath + "\\ntuser.dat");
// Get unexpanded path, e.g. %USERPROFILE%\AppData\Roaming
//var strAppData = getAppData(strSID);
var strAppData = getAppData(strTempKey);
WScript.Echo(strAppData);
// Expand the previous value to a fully-qualified path, e.g. C:\Users\foo\AppData\Roaming
strAppData = strAppData.replace(/%USERPROFILE%/i, strProfilePath);
WScript.Echo(strAppData);
// Unload the user's registry hive
unloadHKUHive(strTempKey);
function getComputerName() {
var oShell = new ActiveXObject("WScript.Shell");
return oShell.ExpandEnvironmentStrings("%COMPUTERNAME%");
}
function getSID(strUser, strDomain) {
var oAccount = GetObject("winmgmts:root/cimv2:Win32_UserAccount.Name='" + strUser + "',Domain='" + strDomain + "'");
return oAccount.SID;
}
function getProfilePath(strSID) {
var oShell = new ActiveXObject("WScript.Shell");
var strValue = oShell.RegRead("HKLM\\Software\\Microsoft\\Windows NT\\CurrentVersion\\ProfileList\\" + strSID + "\\ProfileImagePath");
return strValue;
}
function getAppData(strSID) {
var oShell = new ActiveXObject("WScript.Shell");
var strValue = oShell.RegRead("HKEY_USERS\\" + strSID + "\\Software\\Microsoft\\Windows\\CurrentVersion\\Explorer\\User Shell Folders\\AppData");
return strValue;
}
function loadHKUHive(strKeyName, strHiveFile) {
var oShell = new ActiveXObject("WScript.Shell");
oShell.Run("reg load HKU\\" + strKeyName + " " + strHiveFile, 0, true);
}
function unloadHKUHive(strKeyName) {
var oShell = new ActiveXObject("WScript.Shell");
oShell.Run("reg unload HKU\\" + strKeyName, 0, true);
}
FileSystemObject
было бы очень сложным и потенциально ненадежным.
- person Andy E; 10.08.2011