Alexa技能和Azure AD身份验证

时间:2017-06-19 21:47:37

标签: asp.net oauth-2.0 azure-active-directory alexa alexa-skill

我正在尝试构建一个连接到使用Azure AD身份验证的企业应用程序的alexa技能。我们设置了所有内容,就像在本文https://blogs.msdn.microsoft.com/premier_developer/2016/12/12/amazon-alexa-skills-development-with-azure-active-directory-and-asp-net-core-1-0-web-api/中一样,但是我在将令牌从请求主体移动到标头时遇到了问题。 请求很好,我可以解析它,我将标记添加到标题但在亚马逊的测试页面上我收到此消息:“调用远程端点时出错,返回HTTP 302:发现”

这是将标记添加到标题的代码:

 public void ConfigureAuth(IAppBuilder app)
    {
        app.SetDefaultSignInAsAuthenticationType(CookieAuthenticationDefaults.AuthenticationType);
        app.Use(async (context, next) =>
        {
            string path = string.Format(ConfigurationManager.AppSettings["ExchangeTraceDrivePath"], "AlexaRequest", DateTime.Now.ToFileTime(), "");

            var stream = context.Request.Body;
            try
            {
                using (var buffer = new MemoryStream())
                {
                    await stream.CopyToAsync(buffer);
                    var bodyBuffer = new byte[buffer.Length];
                    buffer.Position = 0L;
                    buffer.Read(bodyBuffer, 0, bodyBuffer.Length);
                    var body = Encoding.UTF8.GetString(bodyBuffer);
                    using (var sw = new StreamWriter(path))
                    {
                        sw.WriteLine(DateTime.Now.ToString() + " body: " + body);
                        sw.WriteLine("---------------------------------------------------------------------------------------------");
                        foreach (var header in context.Request.Headers)
                        {
                            sw.WriteLine(DateTime.Now.ToString() + " header key: " + header.Key);
                            foreach (var val in header.Value)
                            {
                                sw.WriteLine(DateTime.Now.ToString() + " header value: " + val);
                            }
                        }
                        sw.WriteLine("---------------------------------------------------------------------------------------------");

                        dynamic json = JObject.Parse(body);
                        sw.WriteLine(DateTime.Now.ToString() + " parsed body: " + json);
                        sw.WriteLine("---------------------------------------------------------------------------------------------");

                        if (json?.session?.user?.accessToken != null)
                        {
                            sw.WriteLine(DateTime.Now.ToString() + " access accessToken found " +
                                         json?.session?.user?.accessToken);
                            sw.WriteLine("---------------------------------------------------------------------------------------------");

                            context.Request.Headers.Add("Authorization",
                                new string[] { string.Format("Bearer {0}", json?.session?.user?.accessToken) });
                            foreach (var header in context.Request.Headers)
                            {
                                sw.WriteLine(DateTime.Now.ToString() + " header key: " + header.Key);
                                foreach (var val in header.Value)
                                {
                                    sw.WriteLine(DateTime.Now.ToString() + " header value: " + val);
                                }
                            }
                            sw.WriteLine("---------------------------------------------------------------------------------------------");
                        }
                        buffer.Position = 0L;
                        context.Request.Body = buffer;
                    }
                }
            }
            catch
            {
            }
            finally
            {
                await next.Invoke();
                // Restore the original stream.
                context.Request.Body = stream;
            }

        });

        //ExpireTimeSpan and SlidinExpiration only work when  UseTokenLifetime = false
        app.UseCookieAuthentication(new CookieAuthenticationOptions
        {
            // This is NOT ASP.NET Session Timeout (that should be set to same value in web.config)
            // This is the expiration on the cookie that holds the Azure AD token
            ExpireTimeSpan = TimeSpan.FromMinutes(Convert.ToDouble(expirationTimeSpan)),

            // Set SlidingExpiration=true to instruct the middleware to re-issue a new cookie
            // with a new expiration time any time it processes a request which is more than
            // halfway through the expiration window.
            SlidingExpiration = true,

            Provider = new CookieAuthenticationProvider
            {
                // This method is called every time the cookie is authenticated, which
                // is every time a request is made to the web app
                OnValidateIdentity = CookieAuthNotification.OnValidateIdentity
            }
        });

        app.UseOpenIdConnectAuthentication(
            new OpenIdConnectAuthenticationOptions
            {
                ClientId = clientId,
                Authority = authority,
                UseTokenLifetime = false,
                /*
                * Skipping the Home Realm Discovery Page in Azure AD
                * http://www.cloudidentity.com/blog/2014/11/17/skipping-the-home-realm-discovery-page-in-azure-ad/
                */
                Notifications = new OpenIdConnectAuthenticationNotifications
                {
                    RedirectToIdentityProvider = OpenIdConnectNotification.RedirectToIdentityProvider,
                    MessageReceived = OpenIdConnectNotification.MessageReceived,
                    SecurityTokenReceived = OpenIdConnectNotification.SecurityTokenReceived,
                    SecurityTokenValidated = OpenIdConnectNotification.SecurityTokenValidated,
                    AuthorizationCodeReceived = OpenIdConnectNotification.AuthorizationCodeReceived,
                    AuthenticationFailed = OpenIdConnectNotification.AuthenticationFailed
                },

            });
    }

1 个答案:

答案 0 :(得分:0)

我最终创建了一个单独的中间件来将令牌从正文移动到标题

 public class AlexaJWTMiddleware : OwinMiddleware
{
    private readonly OwinMiddleware _next;

    public AlexaJWTMiddleware(OwinMiddleware next) : base(next)
    {
        _next = next;
    }


    public override Task Invoke(IOwinContext context)
    {
        var stream = context.Request.Body;
        if (context.Request.Headers.ContainsKey("SignatureCertChainUrl")
            && context.Request.Headers["SignatureCertChainUrl"]
                .Contains("https://s3.amazonaws.com/echo.api/echo-api-cert-4.pem")
            && !context.Request.Headers.ContainsKey("Authorization"))
        {

            try
            {
                using (var buffer = new MemoryStream())
                {
                    stream.CopyToAsync(buffer);
                    var bodyBuffer = new byte[buffer.Length];
                    buffer.Position = 0L;
                    buffer.Read(bodyBuffer, 0, bodyBuffer.Length);
                    var body = Encoding.UTF8.GetString(bodyBuffer);

                    dynamic json = JObject.Parse(body);
                    if (json?.session?.user?.accessToken != null)
                    {

                        context.Request.Headers.Add("Authorization",
                            new string[] { string.Format("Bearer {0}", json?.session?.user?.accessToken) });
                    }
                    buffer.Position = 0L;
                    context.Request.Body = buffer;
                }
            }
            catch
            {
            }
            finally
            {
                // Restore the original stream.
                context.Request.Body = stream;
            }
        }
        else
        {
            return _next.Invoke(context);

        }
        return _next.Invoke(context);

    }
}

然后在openId

之外添加jwt身份验证
  public void ConfigureAuth(IAppBuilder app)
    {
        app.SetDefaultSignInAsAuthenticationType(CookieAuthenticationDefaults.AuthenticationType);
        app.Use(typeof(AlexaJWTMiddleware));
        app.UseWindowsAzureActiveDirectoryBearerAuthentication(
            new WindowsAzureActiveDirectoryBearerAuthenticationOptions
            {
                Tenant = domain,
                TokenValidationParameters = new TokenValidationParameters
                {
                    ValidAudience = ConfigurationManager.AppSettings["ida:AppIdUri"]
                },
                AuthenticationType = "OAuth2Bearer",
            });

        //ExpireTimeSpan and SlidinExpiration only work when  UseTokenLifetime = false
        app.UseCookieAuthentication(new CookieAuthenticationOptions
        {
            // This is NOT ASP.NET Session Timeout (that should be set to same value in web.config)
            // This is the expiration on the cookie that holds the Azure AD token
            ExpireTimeSpan = TimeSpan.FromMinutes(Convert.ToDouble(expirationTimeSpan)),

            // Set SlidingExpiration=true to instruct the middleware to re-issue a new cookie
            // with a new expiration time any time it processes a request which is more than
            // halfway through the expiration window.
            SlidingExpiration = true,

            Provider = new CookieAuthenticationProvider
            {
                // This method is called every time the cookie is authenticated, which
                // is every time a request is made to the web app
                OnValidateIdentity = CookieAuthNotification.OnValidateIdentity
            }
        });

        app.UseOpenIdConnectAuthentication(
            new OpenIdConnectAuthenticationOptions
            {
                ClientId = clientId,
                Authority = authority,
                UseTokenLifetime = false,
                /*
                * Skipping the Home Realm Discovery Page in Azure AD
                * http://www.cloudidentity.com/blog/2014/11/17/skipping-the-home-realm-discovery-page-in-azure-ad/
                */
                Notifications = new OpenIdConnectAuthenticationNotifications
                {
                    RedirectToIdentityProvider = OpenIdConnectNotification.RedirectToIdentityProvider,
                    MessageReceived = OpenIdConnectNotification.MessageReceived,
                    SecurityTokenReceived = OpenIdConnectNotification.SecurityTokenReceived,
                    SecurityTokenValidated = OpenIdConnectNotification.SecurityTokenValidated,
                    AuthorizationCodeReceived = OpenIdConnectNotification.AuthorizationCodeReceived,
                    AuthenticationFailed = OpenIdConnectNotification.AuthenticationFailed
                },

            });
    }