Как я могу получить IEnumMoniker.Next для возврата моникеров при работе от имени администратора?

Этот код отлично работал в течение многих лет внутри служебной программы. Недавно мы обновили программу для обеспечения контроля учетных записей, но обнаружили, что этот код работает, только когда НЕ работает от имени администратора; код внутри цикла while никогда не выполняется при запуске от имени администратора, но тот же код возвращает список имен моникеров при выполнении без выравнивания.

using System;
using System.Runtime.InteropServices;
using System.Runtime.InteropServices.ComTypes;

namespace ROTExplorer
{
    class Program
    {
    [DllImport("ole32.dll")]
    static extern int GetRunningObjectTable(uint reserved, out IRunningObjectTable rot);

    [DllImport("Ole32.Dll")]
    static extern int CreateBindCtx(int reserved, out IBindCtx bindCtx);

    static void Main(string[] args)
    {
        FindEntryInROT();
        Console.WriteLine("Press any key to continue.");
        Console.ReadKey();
    }

    private static string FindEntryInROT()
    {
        IRunningObjectTable rot = null;
        IBindCtx bindCtx = null;
        IEnumMoniker enumMoniker = null;
        IMoniker[] monikers = new IMoniker[1];
        string displayName = null;
        try
        {
            GetRunningObjectTable(0, out rot);
            CreateBindCtx(0, out bindCtx);
            rot.EnumRunning(out enumMoniker);
            IntPtr fetched = IntPtr.Zero;
            while (enumMoniker.Next(1, monikers, fetched) == 0)
            {
                string tempName;
                monikers[0].GetDisplayName(bindCtx, null, out tempName);
                Marshal.ReleaseComObject(monikers[0]);
                monikers[0] = null;
                try
                {
                    Console.WriteLine(tempName);
                }
                catch
                {
                    Console.WriteLine("Bad string");
                }
            }
        }
        catch (Exception ex)
        {
            Console.WriteLine("Failure while examining ROT: " + ex.Message);
        }
        finally
        {
            ReleaseCOMObject(monikers[0]);
            ReleaseCOMObject(enumMoniker);
            ReleaseCOMObject(bindCtx);
            ReleaseCOMObject(rot);
        }
        Console.WriteLine(displayName);
        return displayName;
    }

    private static void ReleaseCOMObject(object comObject)
    {
        if (comObject != null)
        {
            Marshal.ReleaseComObject(comObject);
            comObject = null;
        }
    }

}

Я пробовал это на 2 машинах. Может кто-то еще, пожалуйста, попробуйте это и подтвердите, что этот код возвращает список имен только тогда, когда НЕ работает от имени администратора.

Кто-нибудь имеет какие-либо мысли о том, почему IEnumMoniker не возвращает никаких имен, когда работает в процессе с повышенными правами, но возвращает список, когда он не запущен от имени администратора?

1 ответ

Я открыл билет с Microsoft. Он обострился, и я наконец получил ответ: он работает как задумано. Вот соответствующий разговор:

Поддержка Microsoft:

Служба SCM/RPCSS - это то место, где находится таблица запущенных объектов. Когда таблица перечисляется, служба выполняет несколько проверок. Одна из этих проверок специально предназначена для сопоставления уровня повышения клиентского токена с уровнем повышения записи токена. Если он не совпадает, то запись не будет возвращена. Поведение, которое вы видите, является дизайном.

Мне:

Можете ли вы отправить мне ссылку, где это задокументировано? Наличие "повышенных" привилегий должно дать пользователю доступ к большему количеству объектов, а не к меньшим. То, что вы описываете, похоже на то, чтобы просто войти как другой пользователь.

Поддержка Microsoft:

Это не задокументировано напрямую.
В некотором смысле, ваше последнее утверждение верно. Администратор имеет два токена безопасности: один для нормальной работы и один для повышенных. Они никогда не используются одновременно. https://docs.microsoft.com/en-us/windows/security/identity-protection/user-account-control/how-user-account-control-works Когда администратор входит в систему, создаются два отдельных токена доступа для пользователь: стандартный токен доступа пользователя и токен доступа администратора. Стандартный токен доступа пользователя содержит ту же информацию, специфичную для пользователя, что и токен доступа администратора, но административные привилегии Windows и SID удалены. Я полагаю, что причина всего этого связана с безопасностью, но я не могу объяснить это очень хорошо.

Мне:

Токен администратора имеет доступ к файлам стандартного пользователя; почему COM-объекты пользователя обрабатываются иначе, чем файловые объекты пользователя?

Поддержка Microsoft:

Потому что это дизайнерское решение, принятое группой продуктов. У меня нет дополнительной информации о том, почему они так сделали.

Мне:

Это действительно звучит как ошибка или неправильный дизайн. Есть ли способ поставить это на радаре группы продуктов?

Поддержка Microsoft:

К сожалению, нет никаких рычагов для получения каких-либо изменений в этой области.

Другие вопросы по тегам