无权访问此资源/API (GCP)

Posted

技术标签:

【中文标题】无权访问此资源/API (GCP)【英文标题】:Not Authorized To Access This Resource/API (GCP) 【发布时间】:2021-02-15 20:02:51 【问题描述】:

我设置了一个具有域范围委派的服务帐户,我将客户端电子邮件、私钥、范围和用户电子邮件传递给 JWT 方法以模拟 G-Suite 用户。然后,我从 Admin API 获取特定的用户信息,并使用它来创建电子邮件签名并将其推送到 Gmail API。如果我传递给 JWT 方法的用户电子邮件是超级管理员,它会很好,但如果我尝试传递任何其他用户,我会收到错误响应,“未授权访问此资源/api”。关于如何让它在我的域中使用常规用户帐户的任何想法?

这里是代码。 (genSignature.js)

const  google  = require('googleapis');
const privatekey = require('../private-key.json');

const scopes = [
    'https://www.googleapis.com/auth/gmail.settings.basic',
    'https://www.googleapis.com/auth/gmail.settings.sharing', 
    'https://www.googleapis.com/auth/admin.directory.user', 
    'https://www.googleapis.com/auth/admin.directory.user.readonly'
];

const auth = async (user) => 
    try 
        const jwtClient = new google.auth.JWT(
            privatekey.client_email,
            null, 
            privatekey.private_key,
            scopes,
            user // User who will be impersonated using the JWT client.
        );

        await jwtClient.authorize();
        return jwtClient;

     catch (err) 

        console.log(err.message);

    ;
;

function genSig(e) 

    auth(e).then((jwtClient) => 

        // Authenticate with the gmail API.
        const gmail = google.gmail(
            version: 'v1', 
            auth: jwtClient
        ); 

        // Authenticate with the admin API.
        const dir = google.admin(
            version: 'directory_v1', 
            auth: jwtClient
        );


        // Get users contact and job data from the directory. This data will be used as variables in their email signature. 
        dir.users.get( userKey: e , (err, response) => 
            if (err) 
                console.log(err.message);
             else 
                let phones = response.data.phones;
                let workPhone = '';

                if (phones) 
                    for (i = 0; i < phones.length; i++) 
                        if (phones[i].type == 'work') 
                            workPhone = phones[i].value;
                        ;
                    ;
                ;

                function getUserData() 
                    let userData = 
                        name: response.data.name.fullName, 
                        email: response.data.primaryEmail, 
                        phone: workPhone, 
                        avatar: response.data.thumbnailPhotoUrl, 
                        department: response.data.organizations[0].department, 
                        title: response.data.organizations[0].title
                    ; 
                    return userData;
                ;

                let requestBody = 
                    signature: 'Test' 
                ;

                // Update the users email signature for their primary email.
                gmail.users.settings.sendAs.update( userId: e, sendAsEmail: e, requestBody , (err, response) => 
                    if (err) 
                        console.log(err.message);
                     else 
                        console.log(response.data);
                    ;
                );
            ;
        );
    );


module.exports = genSig;

(signatures.js)

const express = require('express');
const router = express.Router();
const genSig = require('../../functions/genSignature');

// Get webhooks from Google.
router.post('/', (req, res) => 

    let email = req.body.email;
    let emailStr = email.toString();
    console.log(emailStr);
    genSig(emailStr);


    res.status(200).json(
        "msg": "data recieved..."
    );
);

module.exports = router;

(index.js)

const express = require('express');
const app = express();

app.use(express.json());
app.use('/email-signature', require('./routes/api/signatures'));

const PORT = process.env.PORT || 6000;

app.listen(PORT, () => console.log(`Server is running on port $PORT`));

这里有一些截图。

API configuration on G-Suite

Service Account Setup

Successful request vs unsuccessful request

【问题讨论】:

【参考方案1】:

你需要冒充管理员:

只有具有用户管理权限的帐户(如超级管理员或用户管理管理员)才能访问Users: get。您必须考虑到这是Admin SDK 的一部分,供管理员帐户使用。

如果您尝试通过Try this API on the reference docs 调用它,您也可以检查这是不可能的(您将收到相同的消息:Not Authorized to access this resource/api)。

使用具有域范围权限的服务帐户并不重要:当服务帐户冒充另一个用户时,它只能访问该用户可以访问的资源

解决方案:

在这种情况下,如果您想从 Admin SDK 检索用户数据,被模拟的帐户应该具有用户管理权限。

但由于调用 Gmail API 方法不需要这些权限,因此您可以在调用 Users: get 时模拟管理员帐户,在调用 users.settings.sendAs.update 时模拟普通帐户。

参考:

Admin privileges definitions Pre-built administrator roles

【讨论】:

这很有意义,感谢您为我澄清。所以从技术上讲,服务帐户电子邮件被认为是管理员帐户对吗?也许我可以使用该帐户从 Admin SDK 中检索有关另一个用户的特定信息,然后更新他们的电子邮件签名。这个想法是有一个云功能来监听来自 Admin SDK 的事件并更新电子邮件签名以反映这些变化。 @RainePetersen 不,服务帐户在技术上甚至不是域的一部分,它无法自行访问这些数据;它需要模拟具有访问权限的帐户。如果你想使用 Admin SDK,你应该模拟一个具有管理员权限的普通帐户。 @RainePetersen 另外,如果它解决了您的问题,请考虑接受这个答案。【参考方案2】:

这不是一个新帖子。但是,我面对它并找到了解决方案。

您可以通过分配角色来使用服务帐号。请参阅Assign specific admin role 中的“为服务帐户分配角色”。 updates blog post.有详细介绍

首先,您需要在 Google Workspace 管理控制台中创建一个自定义管理员角色。您可以使用电子邮件地址将服务帐户分配给自定义管理员角色。

它在我的环境中适用于 Google Cloud Functions。

【讨论】:

以上是关于无权访问此资源/API (GCP)的主要内容,如果未能解决你的问题,请参考以下文章

F400813: 用户 '' 无权访问此资源

Digital Ocean Forbidden 您无权访问此资源

部署 Django 应用程序:禁止您无权访问此资源

AWS 用户无权通过显式拒绝访问此资源

此 IP、站点或移动应用程序无权使用此 API 密钥

使用GCP API密钥限制对特定GCP App Engine服务的访问?