从Visual Studio扩展访问当前的Microsoft帐户

本文关键字:Microsoft 帐户 访问 Visual Studio 扩展 | 更新日期: 2023-09-27 18:02:22

我正在写一个VS扩展,它需要与服务器通信并识别用户,我想如果可能的话,因为这个扩展将是连接到服务器的唯一客户端,使用Visual Studio对微软帐户的内置支持将比实现我自己的帐户管理基础设施更有意义。

起初,由于Visual Studio开发人员可以使用各种有用的api,人们可能会认为获取当前用户的信息很容易。然而,实际上似乎没有任何明显的api可以用来访问帐户;我在这里检查了一下,没有任何相关的服务列出("配置文件"服务只允许您读取/写入为当前用户存储的设置)。

有没有人知道一个(相对)简单的方法来访问微软的帐户从Visual Studio扩展?

编辑

我尝试了Hadi Brais的建议,起初它似乎有效(我成功地检索了信息);但是每次,Visual Studio都会在30秒后崩溃。我注释掉了与注册表交互的行,并用变量的静态值替换它们,崩溃停止了。显然,访问Visual Studio的注册表项会导致它崩溃。我甚至尝试了using语句和其他保护措施,但是似乎没有一种方法可以安全地从扩展访问Visual Studio注册表项。那么,有没有人知道任何官方api 可以用来检索这些信息而不会使Visual Studio崩溃?

从Visual Studio扩展访问当前的Microsoft帐户

对于Visual Studio 2015 (Version 14.0),这是如何获取当前在Visual Studio中登录的用户的信息。您需要添加using Microsoft.Win32;

private static string GetUserEmailAddressVS14()
{
    // It's a good practice to request explicit permission from
    // the user that you want to use his email address and any
    // other information. This enables the user to be in control
    // of his/her privacy.
    // Assuming permission is granted, we obtain the email address.
    const string SubKey = "Software''Microsoft''VSCommon''ConnectedUser''IdeUser''Cache";
    const string EmailAddressKeyName = "EmailAddress";
    const string UserNameKeyName = "DisplayName";
    RegistryKey root = Registry.CurrentUser;
    RegistryKey sk = root.OpenSubKey(SubKey);
    if (sk == null)
    {
        // The user is currently not signed in.
        return null;
    }
    else
    {
        // Get user email address.
        return (string)sk.GetValue(EmailAddressKeyName);
        // You can also get user name like this.
        // return (string)sk.GetValue(UserNameKeyName);
    }
}

现在有多个版本的IdeUser键。我从另一个答案中重新实现了这个算法:

public static string GetUserEmailAddressFromVisualStudioRegistry()
{
    try
    {
        const string ConnectedUserSubKey = @"Software'Microsoft'VSCommon'ConnectedUser";
        const string EmailAddressKeyName = "EmailAddress";
        RegistryKey connectedUserSubKey = Registry.CurrentUser.OpenSubKey( ConnectedUserSubKey );
        string[] subKeyNames = connectedUserSubKey?.GetSubKeyNames();
        if ( subKeyNames == null || subKeyNames.Length == 0 )
        {
            return null;
        }
        int[] subKeysOrder = new int[subKeyNames.Length];
        for ( int i = 0; i < subKeyNames.Length; i++ )
        {
            Match match = Regex.Match( subKeyNames[i], @"^IdeUser(?:V(?<version>'d+))?$" );
            if ( !match.Success )
            {
                subKeysOrder[i] = -1;
                continue;
            }
            string versionString = match.Groups["version"]?.Value;
            if ( string.IsNullOrEmpty( versionString ) )
            {
                subKeysOrder[i] = 0;
            }
            else if ( !int.TryParse( versionString, out subKeysOrder[i] ) )
            {
                subKeysOrder[i] = -1;
            }
        }
        Array.Sort( subKeysOrder, subKeyNames );
        for ( int i = subKeyNames.Length - 1; i >= 0; i++ )
        {
            string cacheSubKeyName = $@"{subKeyNames[i]}'Cache";
            RegistryKey cacheKey = connectedUserSubKey.OpenSubKey( cacheSubKeyName );
            string emailAddress = cacheKey?.GetValue( EmailAddressKeyName ) as string;
            if ( !string.IsNullOrWhiteSpace( emailAddress ) )
            {
                return emailAddress;
            }
        }
    }
    catch
    {
        // Handle exceptions here if it's wanted.
    }
    return null;
}

这个算法尝试从最新版本开始的所有版本,然后是所有其他的兄弟版本。如果失败,它返回null。