从Docker容器导航到Google身份验证页面时遇到问题。该应用程序使用asp.net Core 2.0制作,其目的是显示从Google Calendar API检索的即将发生的事件,并在需要时创建事件。
当应用程序未从docker容器运行时,应用程序按预期工作 生成授权代码请求URL,并根据环境(Windows,Linux或OSX),应用程序尝试在OpenBrowser方法中使用Process.start()打开URL。
private bool OpenBrowser(string url)
{
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
url = System.Text.RegularExpressions.Regex.Replace(url, @"(\\*)" + "\"", @"$1$1\" + "\"");
url = System.Text.RegularExpressions.Regex.Replace(url, @"(\\+)$", @"$1$1");
Process.Start(new ProcessStartInfo("cmd", $"/c start \"\" \"{url}\"") { CreateNoWindow = true });
return true;
}
if (RuntimeInformation.IsOSPlatform(OSPlatform.Linux))
{
Process.Start("xdg-open", url);
return true;
}
if (RuntimeInformation.IsOSPlatform(OSPlatform.OSX))
{
Process.Start("open", url);
return true;
}
return false;
}
成功打开浏览器后,用户需要输入其Google帐户凭据并登录才能接收用于创建令牌的授权码。
public async Task<AuthorizationCodeResponseUrl> ReceiveCodeAsync(AuthorizationCodeRequestUrl url, CancellationToken taskCancellationToken)
{
var authorizationUrl = url.Build().AbsoluteUri;
using (var listener = StartListener())
{
bool browserOpenOk;
try
{
browserOpenOk = OpenBrowser(authorizationUrl);
}
catch (Exception e)
{
Logger.Error(e, "Failed to launch browser with \"{0}\" for authorization", authorizationUrl);
throw new NotSupportedException(String.Format("Failed to launch browser with \"{0}\" for authorization. See inner exception for details.", authorizationUrl, e));
}
var ret = await GetResponseFromListener(listener, taskCancellationToken).ConfigureAwait(false);
s_receivedCallback = true;
return ret;
}
}
我的猜测是linux docker容器没有打开URL的必要工具,这就是导致问题的原因。我的问题是,有人可以告诉我如何将URL转发到主机而不是容器内部,或者如何在不尝试从docker容器中打开URL的情况下获取令牌?
Google默认使用这种方式对用户进行身份验证(通过使用Process.start打开url),并打开随机端口来侦听身份验证响应代码,因此我必须创建自己的类来实现ICodeReceiver,因为docker需要精确在容器内运行映像之前要指定的端口。
答案 0 :(得分:0)
我在编写python脚本并将其打包到docker镜像时遇到了同样的问题。
我找到的唯一解决方案是将以下行添加到参数noauth_local_webserver = True
。
这是python中的示例代码:
if not creds or creds.invalid:
flow = client.flow_from_clientsecrets('.client_secret.json', SCOPES)
args = tools.argparser.parse_args()
args.noauth_local_webserver = True
creds = tools.run_flow(flow, store, args)
service = build('sheets', 'v4', http=creds.authorize(Http()))
else:
service = build('sheets', 'v4', http=creds.authorize(Http()))
使用该选项,它会打印一个可以复制到任何Web浏览器的URL,并在验证后生成一个需要复制到终端的代码。