Beanstalk / App Runnerとは
.Net CoreのウェブアプリケーションをAWSにホストするサービスとしてApp Runner / Beanstalkがある。
App Runnerは、Visual StudioからAWS Toolkitでデプロイするだけで、Load Balancerの設定など一通りの設定が自動で完了する。
BeanstalkもAWS Toolkitでデプロイすれば諸々の設定を行ってくれるが、もう少し設定する(できる)ことが増える。
ウェブアプリケーションにアクセスするときに一番最初に気がつく点は、Beanstalkはhttpがデフォルトに対し、App Runnerはhttpsでアクセスを行う。(Cognitoを使用して認証を行うときは、ここは重要。なぜならCognitoは、戻りURLとしてhttpsしか受け付けてくれない。)
Load Balancerから先の事
BeanstalkもApp RunnerもLoad Balancerが最初のアクセスポイントとなり、Load Balancerとウェブサーバ間の処理はhttpで行なわれる。(おそらくこの部分は、変更できないのではないかと思う。)
Beanstalk / App Runnerでホストされている.Net Coreのウェブアプリがホストされているウェブサーバには、httpでアクセスされることとなる。
Load Balancerにhttpでアクセスしても、httpsでアクセスしても、.Net的には、httpでアクセスされていると認識され、httpの処理となる。よって、Cognitoに接続する際に渡されるredirect urlもhttpとなってしまう。
※Visual Studioでローカル環境でデバッグする環境はhttps://localhost:xxxxのようなLoad Balancerが登場しない環境では、ウェブサーバにhttpsでアクセスし、ウェブサーバからCognitoへの接続時にもredirect urlはhttpsとして渡されるため、ローカルではうまくいくが、AWSにデプロイするとうまくいかないことがわかる。
簡単な図を書くとこのような流れとなる。(絵はBeanstalkだがApp Runnerも同様)
data:image/s3,"s3://crabby-images/b158e/b158e40182c5b0d477151009b5d717502a614b4c" alt=""
Cognitoにhttpsのredirect urlを渡す
Load Balancerから先をhttpsに設定できれば、そのままCognitoにhttpsのredirect urlが渡るが、それができないため、.Netの方で、強制的にredirect urlがhttpであれば、httpsにする設定を行う。
絵の中にはしれっと書いてある Cookieもその設定を行う必要がある。(おそらく認証時に発行されるTokenはCookieの中に保存される)
同じサイト内であっても、httpのページとhttpsのページでは、Cookieを共有することができない。そのため、Cookieを常にhttpsのページとして使用するように設定する。
上記絵の中では、Cookieがhttpsのサイトとして処理され、redirect urlがhttpsと強制的に変更する。
最後に
上記のよう.Net側の変更を行うことでCognitoでの認証が可能となる。App Runnerでホストされている場合は、ここでおしまいだが、Beanstalkでホストされている場合は、Load BalancerのListenerの設定やInbound/Outboundルールの設定が少しだが、必要となる。
以下、設定周りのスクリーンショット。
設定内容
Cognito
Cognitoの設定はこちらを参照できる。.Net CoreのApp Settingsの設定も後半に登場する。
data:image/s3,"s3://crabby-images/08570/0857069ab004288a1fb6f87363404738702dadc5" alt=""
data:image/s3,"s3://crabby-images/09b45/09b450e4c04faf73830baaebf87aaf931956b690" alt=""
data:image/s3,"s3://crabby-images/80514/8051451f9983ba4ac4a0aa22f3f0b3aaa13ca04e" alt=""
data:image/s3,"s3://crabby-images/76068/760683b49026cdc27b89cd505078ba4d940e19a0" alt=""
data:image/s3,"s3://crabby-images/e0204/e02041711a1f365092c8d47be1517412aca64351" alt=""
data:image/s3,"s3://crabby-images/1b4b7/1b4b72eecf81ce919301caf5a6c42018d1744cf5" alt=""
data:image/s3,"s3://crabby-images/ef052/ef0527fa1eb0cc27336dcc05d0ea7313014c39a0" alt=""
data:image/s3,"s3://crabby-images/29509/295092bc42e1fe333020572c33e4fab4ee1653ac" alt=""
data:image/s3,"s3://crabby-images/04496/044968d421c05a4e5a93cf6e8863259b64b01cc0" alt=""
data:image/s3,"s3://crabby-images/47bc3/47bc38f28c2168f287c51cf8b42602404bbd19ed" alt=""
data:image/s3,"s3://crabby-images/e1a19/e1a192cb9f7f5ae45c07d7a625ffd6a3e206cbf1" alt=""
data:image/s3,"s3://crabby-images/b6625/b6625352d3b96ef57d4a376208e88c8e9f098358" alt=""
data:image/s3,"s3://crabby-images/8f17a/8f17a0b449a18f11056ea4dcb74014d29fc5c278" alt=""
data:image/s3,"s3://crabby-images/4caf6/4caf65e399ff5df85f6bdddc0e958d5964453dff" alt=""
Visual StudioプロジェクトのApp Settingsの設定(上記Cognitoサイトで作成された情報を使用する)
下記AppSettings.jsonの内容となる。これらの<>の中を埋めていく。
{
"Logging": {
"LogLevel": {
"Default": "Warning"
}
},
"AllowedHosts": "*",
"Authentication": {
"Cognito": {
"ClientId": "<app client id from AWS Cognito>",
"IncludeErrorDetails": true,
"MetadataAddress": "https://cognito-idp.<your region>.amazonaws.com/<your-pool id>/.well-known/openid-configuration",
"RequireHttpsMetadata": false,
"ResponseType": "code",
"SaveToken": true,
"TokenValidationParameters": {
"ValidateIssuer": true
},
"AppSignOutUrl": "",
"CognitoDomain": "https://<custom domain name>.auth.us-east-1.amazoncognito.com"
}
}
}
data:image/s3,"s3://crabby-images/f4309/f4309e40a16ca6553e50361c4be402599ba32fdf" alt=""
data:image/s3,"s3://crabby-images/78b8c/78b8ce6ad08b997e6bb1b0fe51fa1b56da61483b" alt=""
data:image/s3,"s3://crabby-images/a6993/a699346d36c8acaee9d71e58b326995f15bf3e91" alt=""
AWS Toolkitからデプロイ
data:image/s3,"s3://crabby-images/a7651/a76515f43a89cd345e4466256f78dc57710a2d7f" alt=""
data:image/s3,"s3://crabby-images/5c677/5c6771f0c615dd34e088b7190c36b98191b72d5d" alt=""
{
"DetailedErrors": true,
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
},
"AllowedHosts": "*",
"Authentication": {
"Cognito": {
"ClientId": "75gkrda7e6nnn85s99ssdhs5o5",
"IncludeErrorDetails": true,
"MetadataAddress": https://cognito-idp.us-east-1.amazonaws.com/us-east-1_PeCdkl5kq/.well-known/openid-configuration,
"RequireHttpsMetadata": false,
"ResponseType": "code",
"SaveToken": true,
"TokenValidationParameters": {
"ValidateIssuer": true
},
"AppSignOutUrl": "",
"CognitoDomain": https://test-demo-domain-101.auth.us-east-1.amazoncognito.com
}
}
}
Beanstalkへデプロイ
data:image/s3,"s3://crabby-images/0bdd3/0bdd30dcd0c30b46e2ce929e1859dbe84ce0c933" alt=""
data:image/s3,"s3://crabby-images/cae05/cae05d0c6cbf64bcbcb30778e17357c696165066" alt=""
Load Balancerの設定
data:image/s3,"s3://crabby-images/b6797/b6797ff0d5de4512ba1918d722ad6ce358143f59" alt=""
data:image/s3,"s3://crabby-images/e94d7/e94d7b820cdf42f926109baffc9dd5a969077796" alt=""
data:image/s3,"s3://crabby-images/b88e4/b88e4b37770f6a1c60f196a87084d66a70f444e9" alt=""
data:image/s3,"s3://crabby-images/7d072/7d072f65e86968bb18a9cb60a96e4ab3de989a3d" alt=""
data:image/s3,"s3://crabby-images/e7999/e7999351977e94e3173f837955509a934b5c8daf" alt=""
data:image/s3,"s3://crabby-images/e4589/e4589ce5473b30aa36be0fbf28344f000aeb89b3" alt=""
data:image/s3,"s3://crabby-images/2b6b9/2b6b94ae630fb5921fb2bda4f4af5c68d6558ea1" alt=""
Cognitoの設定(App client settings > Callback URLs)
data:image/s3,"s3://crabby-images/2c402/2c402a4983e902d290769f396cd0233bf6180830" alt=""
Visual Studioプロジェクト Program.csの変更(追加)
using Microsoft.AspNetCore.Authentication.Cookies;
using Microsoft.AspNetCore.Authentication.OpenIdConnect;
var builder = WebApplication.CreateBuilder(args);
builder.Services.AddAuthentication(options =>
{
options.DefaultAuthenticateScheme = CookieAuthenticationDefaults.AuthenticationScheme;
options.DefaultSignInScheme = CookieAuthenticationDefaults.AuthenticationScheme;
options.DefaultChallengeScheme = OpenIdConnectDefaults.AuthenticationScheme;
})
.AddCookie()
.AddOpenIdConnect(options =>
{
options.CorrelationCookie.SecurePolicy = CookieSecurePolicy.Always;
options.NonceCookie.SecurePolicy = CookieSecurePolicy.Always;
options.ResponseType = builder.Configuration["Authentication:Cognito:ResponseType"];
options.MetadataAddress = builder.Configuration["Authentication:Cognito:MetadataAddress"];
options.ClientId = builder.Configuration["Authentication:Cognito:ClientId"];
options.Events = new OpenIdConnectEvents()
{
//OnRedirectToIdentityProviderForSignOut = OnRedirectToIdentityProviderForSignOut
OnRedirectToIdentityProvider = (context) =>
{
// force https for the redirect url.
if (context.ProtocolMessage.RedirectUri.StartsWith([http://]http://))
{
context.ProtocolMessage.RedirectUri = context.ProtocolMessage.RedirectUri.Replace([http://]http://, [https://]https://);
}
return Task.FromResult(0);
}
};
});
// Add services to the container.
builder.Services.AddRazorPages();
var app = builder.Build();
// Configure the HTTP request pipeline.
if (!app.Environment.IsDevelopment())
{
app.UseExceptionHandler("/Error");
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
app.UseHsts();
}
app.UseHttpsRedirection();
app.UseStaticFiles();
app.UseRouting();
app.UseAuthentication();
app.UseAuthorization();
app.MapRazorPages();
app.Run();
data:image/s3,"s3://crabby-images/c5449/c5449bb805a23be70046fd2f3b3716e4b69901c4" alt=""
data:image/s3,"s3://crabby-images/ba158/ba158c0c8242d0315be7c52464559a015948f9c1" alt=""