6

我有一个使用 OAuth2 授权外部服务的 GMail AddOn。启动 AddOn 时,我通过调用来检查用户是否已经拥有访问令牌oauthService.hasAccess()。如果用户没有令牌,那么我会出示一张卡片,上面写着“您需要授权服务”。

该卡包含一个这样构建的链接:

section.addWidget(CardService.newTextButton()
      .setText("Authorize MyService")
      .setAuthorizationAction(
        CardService.newAuthorizationAction()
          .setAuthorizationUrl(authorizationUrl)
      ));

当用户单击“Authorize MyService”按钮时,会出现一个弹出窗口,并且旧卡中会出现一个微调器。然后用户登录到 MyService,授权访问,并AuthorizeCallback使用令牌调用 my。我存储令牌,并返回一个刚刚关闭的 HtmlPage。

原始卡中的旋转器停止旋转。但是,AddOn 不会重新加载。这表明当授权完成时,有一些东西阻止了 AuthorizationAction 卡重新加载。有什么建议吗?

我希望加载项重新加载,因为入口点函数会发现现在oauthService.hasAccess()返回true,然后将显示正常的“您已获得授权”卡(并开始处理发送给 MyService 的电子邮件)。

谢谢你的帮助!

4

2 回答 2

1

正如@AKT 的回答中提到的,这里的核心问题是授权回调将令牌保存到属性中,这些属性不会立即复制到所有 Apps 脚本后端。因此,尽管身份验证流程已经完成,但当加载项重新加载时,它仍然会看到陈旧的属性,表明它还没有。

解决此问题的一种方法是在关闭弹出窗口(并刷新附加侧边栏)之前等待更长时间。更好的选择似乎是确保您在 OAuth 服务上设置缓存:

.setCache(CacheService.getUserCache())

这增加了一个复制速度更快的额外缓存层,这似乎可以解决这个问题。

于 2018-11-20T21:17:26.473 回答
1

可能有更好的方法来做到这一点,但这就是我解决它的方法。

首先,不要使用setAuthorizationActionuse setOnClickAction并设置一个函数名。

section.addWidget(CardService.newTextButton()
.setText("Authorize MyService")
.setOnClickAction(CardService.newAction().setFunctionName('loginAction'));

loginFunction将返回一个ActionResponse. 它将有一个NavigationOpenLink与之关联。

function loginAction(){
    var nav = CardService.newNavigation().pushCard(getLoginStepsCard());
    return CardService.newActionResponseBuilder()
        .setStateChanged(true)
        .setNavigation(nav)
        .setOpenLink(
            CardService.newOpenLink().setUrl(getLoginUrl()  
        )
        .setOnClose(CardService.OnClose.RELOAD_ADD_ON)
        .setOpenAs(CardService.OpenAs.OVERLAY)).build();
}

这将打开登录弹出窗口。getLoginUrl将返回 OAuth 登录 URL。getLoginStepsCard将返回一张包含一些信息的卡片。在我的例子中,我展示了登录所需的步骤,展示你想要的任何东西。

在授权回调函数中,保存您的令牌并且不要立即关闭弹出窗口。在我测试(使用HEAD)时关闭弹出窗口有效,但它不适用于版本化部署。PropertiesService问题是保存令牌需要几秒钟。只需等待几秒钟(我使用setTimeout5s)并关闭弹出窗口或要求用户关闭它。附加组件将重新加载,PropertiesService 将拥有您的令牌。

我希望这有帮助。它对我有用,我已经发布了我的插件;)

于 2018-08-03T10:32:05.397 回答