尝试使用服务帐户进行身份验证时出现“401 Unauthorized”
getting "401 Unauthorized" while trying to authenticated with service account
问题来了:我最近做了 .
我提到的解决方案适用于一个令牌和一个 API,但是当我尝试使用两个令牌(gmail 和 Sheets API)处理两个 API 时,它失败了。
所以我现在要做的是让这两个工作,所以我告诉自己“嘿,让我们创建一个服务帐户”。即使我并不真正理解这两种方法之间的区别。服务帐户似乎无法显示同意屏幕(我说得对吗?)。
我已经在网上抓取了答案,但似乎都失败了。
我刷新了令牌,使用 GoogleCredential
而不是 Credential
,创建了新密钥等...虽然我没有尝试过,但有一件事是使用 Gsuite 帐户,我正在使用基本账户。
所以现在我创建了一个新的 p12 文件,但我立即收到 401
错误。我将分享我的代码以便更好地理解。
我的邮箱class
public class mailService {
private static final String APPLICATION_NAME = "AHS";
private static final JsonFactory JSON_FACTORY = JacksonFactory.getDefaultInstance();
private static final String TOKENS_DIRECTORY_PATH = "tokens";
//I've added sheet scope as it is activated in my project
private static final Collection<String> SCOPES = Arrays.asList(GmailScopes.GMAIL_SEND, GmailScopes.GMAIL_LABELS, SheetsScopes.SPREADSHEETS);
private static Credential getCredentials(final NetHttpTransport HTTP_TRANSPORT) throws IOException, GeneralSecurityException {
File sa = new File("WEB-INF/mykeyfile.p12");
Credential credential = new GoogleCredential.Builder()
.setTransport(HTTP_TRANSPORT)
.setJsonFactory(JSON_FACTORY)
.setServiceAccountId(
"myapp@appspot.gserviceaccount.com")
.setServiceAccountScopes(SCOPES)
.setServiceAccountPrivateKeyFromP12File(sa)
.setServiceAccountUser("myemailadress@gmail.com")
.build();
//credential.refreshToken();
return (credential);
}
public static Gmail getService() throws GeneralSecurityException, IOException {
final NetHttpTransport HTTP_TRANSPORT = GoogleNetHttpTransport.newTrustedTransport();
/*Gmail service = new Gmail.Builder(HTTP_TRANSPORT, JSON_FACTORY, getCredentials(HTTP_TRANSPORT))
.setApplicationName(APPLICATION_NAME)
.build();*/
Gmail service = new Gmail.Builder(HTTP_TRANSPORT, JSON_FACTORY, null)
.setHttpRequestInitializer(getCredentials(HTTP_TRANSPORT)).setApplicationName(APPLICATION_NAME).build();
return (service);
}
...
为了让您更好地理解,我正在使用 Angular 和 Google 应用引擎创建一个网络应用。我不想使用 Gmail API 从我的帐户发送邮件,我还使用工作表 API 到 read/write from/to 电子表格。需要说明的是,我有一个用于 google 登录的秘密文件(供网络应用程序的用户使用),但这是服务器端代码,我不希望用户看到同意屏幕.
我也在问自己是否需要使用 gcloud
来激活服务帐户。
我 运行(目前)我的服务器在本地使用 Eclipse 和 google 应用程序引擎插件。
如果您需要其他代码或精度来更好地理解问题,请告诉我
Gmail dosent 支持服务帐户,除非它是 gsuite 帐户并且您设置了域范围委托。
如果您查看文档,您只会看到有关使用 Oauth2 而不是服务器帐户的信息,这是因为 Google 仅记录受支持的内容,而不是不支持的内容。
Sheets确实支持服务账号,刚想起来需要在服务上预授权一个账号。这是通过与服务帐户共享 sheet 来完成的,就像您与任何其他使用服务帐户电子邮件地址的用户一样。
问题来了:我最近做了
我提到的解决方案适用于一个令牌和一个 API,但是当我尝试使用两个令牌(gmail 和 Sheets API)处理两个 API 时,它失败了。
所以我现在要做的是让这两个工作,所以我告诉自己“嘿,让我们创建一个服务帐户”。即使我并不真正理解这两种方法之间的区别。服务帐户似乎无法显示同意屏幕(我说得对吗?)。
我已经在网上抓取了答案,但似乎都失败了。
我刷新了令牌,使用 GoogleCredential
而不是 Credential
,创建了新密钥等...虽然我没有尝试过,但有一件事是使用 Gsuite 帐户,我正在使用基本账户。
所以现在我创建了一个新的 p12 文件,但我立即收到 401
错误。我将分享我的代码以便更好地理解。
我的邮箱class
public class mailService {
private static final String APPLICATION_NAME = "AHS";
private static final JsonFactory JSON_FACTORY = JacksonFactory.getDefaultInstance();
private static final String TOKENS_DIRECTORY_PATH = "tokens";
//I've added sheet scope as it is activated in my project
private static final Collection<String> SCOPES = Arrays.asList(GmailScopes.GMAIL_SEND, GmailScopes.GMAIL_LABELS, SheetsScopes.SPREADSHEETS);
private static Credential getCredentials(final NetHttpTransport HTTP_TRANSPORT) throws IOException, GeneralSecurityException {
File sa = new File("WEB-INF/mykeyfile.p12");
Credential credential = new GoogleCredential.Builder()
.setTransport(HTTP_TRANSPORT)
.setJsonFactory(JSON_FACTORY)
.setServiceAccountId(
"myapp@appspot.gserviceaccount.com")
.setServiceAccountScopes(SCOPES)
.setServiceAccountPrivateKeyFromP12File(sa)
.setServiceAccountUser("myemailadress@gmail.com")
.build();
//credential.refreshToken();
return (credential);
}
public static Gmail getService() throws GeneralSecurityException, IOException {
final NetHttpTransport HTTP_TRANSPORT = GoogleNetHttpTransport.newTrustedTransport();
/*Gmail service = new Gmail.Builder(HTTP_TRANSPORT, JSON_FACTORY, getCredentials(HTTP_TRANSPORT))
.setApplicationName(APPLICATION_NAME)
.build();*/
Gmail service = new Gmail.Builder(HTTP_TRANSPORT, JSON_FACTORY, null)
.setHttpRequestInitializer(getCredentials(HTTP_TRANSPORT)).setApplicationName(APPLICATION_NAME).build();
return (service);
}
...
为了让您更好地理解,我正在使用 Angular 和 Google 应用引擎创建一个网络应用。我不想使用 Gmail API 从我的帐户发送邮件,我还使用工作表 API 到 read/write from/to 电子表格。需要说明的是,我有一个用于 google 登录的秘密文件(供网络应用程序的用户使用),但这是服务器端代码,我不希望用户看到同意屏幕.
我也在问自己是否需要使用 gcloud
来激活服务帐户。
我 运行(目前)我的服务器在本地使用 Eclipse 和 google 应用程序引擎插件。
如果您需要其他代码或精度来更好地理解问题,请告诉我
Gmail dosent 支持服务帐户,除非它是 gsuite 帐户并且您设置了域范围委托。
如果您查看文档,您只会看到有关使用 Oauth2 而不是服务器帐户的信息,这是因为 Google 仅记录受支持的内容,而不是不支持的内容。
Sheets确实支持服务账号,刚想起来需要在服务上预授权一个账号。这是通过与服务帐户共享 sheet 来完成的,就像您与任何其他使用服务帐户电子邮件地址的用户一样。