本教程详细介绍了如何使用PHP与Google API客户端库,通过服务账户(Service Account)和域级授权(Domain-Wide Delegation)来获取Google Workspace中特定用户所属的群组列表。文章涵盖了Google API项目配置、PHP客户端库的初始化、用户身份验证流程,以及核心的服务账户授权与用户模拟机制,并提供了完整的示例代码与注意事项,旨在帮助开发者解决在集成Google Admin SDK Directory API时遇到的权限问题。
在Google Workspace环境中,若要通过编程方式获取组织内用户的群组信息,通常需要具备管理员权限。直接使用用户的OAuth 2.0凭据通常无法完成此类域级管理操作。此时,Google的服务账户(Service Account)结合域级授权(Domain-Wide Delegation)成为最佳实践。通过这种方式,服务账户可以模拟(impersonate)域内的任何用户,执行该用户被授权的操作,甚至执行管理员级别的操作,前提是服务账户本身已被委派了相应的权限。
准备工作:
使用Composer安装Google API PHP客户端库:
composer require google/apiclient:^2.0
以下代码示例展示了如何结合用户OAuth认证和服务账户域级授权来获取当前登录用户的群组列表。
setClientId('YOUR_CLIENT_ID.apps.googleusercontent.com');
$client->setClientSecret('YOUR_CLIENT_SECRET');
// 替换为您的重定向URI,必须与Google Cloud控制台中配置的完全一致
$client->setRedirectUri('YOUR_REDIRECT_URI');
$client->setApplicationName('My Google Workspace App');
// 为用户OAuth认证添加必要的范围
// 这些范围将请求用户授权应用程序访问其电子邮件和个人资料
$client->addScope('https://www.googleapis.com/auth/userinfo.email');
$client->addScope('https://www.googleapis.com/auth/userinfo.profile');
// 检查是否存在授权码,表示用户已从Google认证服务器重定向回来
if (isset($_GET['code'])) {
try {
// 使用授权码获取访问令牌
$token = $client->fetchAccessTokenWithAuthCode($_GET['code']);
// 检查是否存在错误
if (isset($token["error"])) {
echo "Error fetching access token: " . $token["error_description"];
exit();
}
// 设置客户端的访问令牌
$client->setAccessToken($token['access_token']);
// 使用OAuth2服务获取当前登录用户的基本信息
$google_oauth = new Google_Service_Oauth2($client);
$google_account_info = $google_oauth->userinfo->get();
$userEmail = $google_account_info->email;
echo "当前登录用户: " . htmlspecialchars($userEmail) . "
";
// -------------------------------------------------------------------------
// 步骤2: 重新配置客户端以使用服务账户进行域级授权和用户模拟
// -------------------------------------------------------------------------
// 加载服务账户的JSON密钥文件
// 确保此文件路径正确,且文件安全存储
$client->setAuthConfig('path/to/your-service-account-key.json');
// 设置要模拟的用户邮箱。这里我们模拟当前登录的用户。
// 服务账户必须已通过域级授权,并被授予了对admin.directory.group.readonly范围的权限。
$client->setSubject($userEmail);
// 初始化Directory服务。此时,客户端将使用服务账户的凭据
// 并模拟由setSubject指定的用户来执行Admin SDK操作。
$adminService = new Google_Service_Directory($client);
// -------------------------------------------------------------------------
// 步骤3: 使用Directory服务获取模拟用户所属的群组列表
// -------------------------------------------------------------------------
$optParams = array(
'domain' => 'yourdomain.com', // 替换为您的Google Workspace域名
'userKey' => $userEmail // 指定要查询群组的用户邮箱
);
// 调用listGroups方法获取群组列表
$googleGroups = $adminService->groups->listGroups($optParams);
$groups = $googleGroups->getGroups();
if (!empty($groups)) {
echo "" . htmlspecialchars($userEmail) . " 所属群组:
";
echo "未找到 " . htmlspecialchars($userEmail) . " 所属的群组。
"; } } catch (Google_Service_Exception $e) { // 捕获Google API服务异常,通常是权限不足或配置错误 echo "错误代码: " . $e->getCode() . "
"; echo "错误信息: " . $e->getMessage() . "
"; // 可以根据错误信息进一步调试,例如检查域级授权配置 } catch (Exception $e) { // 捕获其他通用异常 echo "" . $e->getMessage() . "
"; } } else { // 如果没有授权码,则生成授权URL并引导用户进行认证 $authUrl = $client->createAuthUrl(); echo "点击此处登录Goo
gle并查看您的群组