Nếu sử dụng tính năng Đăng nhập bằng Google thông qua một ứng dụng hoặc trang web giao tiếp với một máy chủ phụ trợ, thì bạn có thể cần xác định người dùng hiện đã đăng nhập trên máy chủ. Để thực hiện việc này một cách an toàn, sau khi người dùng đăng nhập thành công, hãy gửi mã thông báo nhận dạng của người dùng đến máy chủ của bạn bằng HTTPS. Sau đó, trên máy chủ, hãy xác minh tính toàn vẹn của mã thông báo nhận dạng và sử dụng thông tin người dùng có trong mã thông báo để thiết lập một phiên hoặc tạo một tài khoản mới.
Gửi mã thông báo nhận dạng đến máy chủ của bạn
Sau khi người dùng đăng nhập thành công, hãy lấy mã thông báo nhận dạng của người dùng:
Swift
GIDSignIn.sharedInstance.signIn(withPresenting: self) { signInResult, error in guard error == nil else { return } guard let signInResult = signInResult else { return } signInResult.user.refreshTokensIfNeeded { user, error in guard error == nil else { return } guard let user = user else { return } let idToken = user.idToken // Send ID token to backend (example below). } }
Objective-C
[GIDSignIn.sharedInstance signInWithPresentingViewController:self completion:^(GIDSignInResult * _Nullable signInResult, NSError * _Nullable error) { if (error) { return; } if (signInResult == nil) { return; } [signInResult.user refreshTokensIfNeededWithCompletion:^(GIDGoogleUser * _Nullable user, NSError * _Nullable error) { if (error) { return; } if (user == nil) { return; } NSString *idToken = user.idToken; // Send ID token to backend (example below). }]; }];
Sau đó, hãy gửi mã thông báo mã nhận dạng đến máy chủ của bạn bằng yêu cầu POST qua HTTPS:
Swift
func tokenSignInExample(idToken: String) { guard let authData = try? JSONEncoder().encode(["idToken": idToken]) else { return } let url = URL(string: "https://yourbackend.example.com/tokensignin")! var request = URLRequest(url: url) request.httpMethod = "POST" request.setValue("application/json", forHTTPHeaderField: "Content-Type") let task = URLSession.shared.uploadTask(with: request, from: authData) { data, response, error in // Handle response from your backend. } task.resume() }
Objective-C
NSString *signinEndpoint = @"https://yourbackend.example.com/tokensignin"; NSDictionary *params = @{@"idtoken": idToken}; NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:signinEndpoint]; [request setValue:@"application/x-www-form-urlencoded" forHTTPHeaderField:@"Content-Type"]; [request setHTTPMethod:@"POST"]; [request setHTTPBody:[self httpBodyForParamsDictionary:params]]; NSOperationQueue *queue = [[NSOperationQueue alloc] init]; [NSURLConnection sendAsynchronousRequest:request queue:queue completionHandler:^(NSURLResponse *response, NSData *data, NSError *error) { if (error) { NSLog(@"Error: %@", error.localizedDescription); } else { NSLog(@"Signed in as %@", data.bytes); } }];
Xác minh tính toàn vẹn của mã thông báo nhận dạng
Sau khi nhận được mã thông báo nhận dạng qua phương thức POST qua HTTPS, bạn phải xác minh tính toàn vẹn của mã thông báo.
Để xác minh rằng mã thông báo hợp lệ, hãy đảm bảo rằng bạn đáp ứng các tiêu chí sau:
- Mã thông báo nhận dạng được Google ký đúng cách. Sử dụng khoá công khai của Google (có ở định dạng JWK hoặc PEM) để xác minh chữ ký của mã thông báo. Các khoá này được xoay vòng thường xuyên; hãy kiểm tra tiêu đề
Cache-Controltrong phản hồi để xác định thời điểm bạn nên truy xuất lại các khoá này. - Giá trị của
audtrong mã thông báo nhận dạng bằng một trong các mã ứng dụng khách của ứng dụng. Bạn cần thực hiện bước kiểm tra này để ngăn chặn việc sử dụng mã nhận dạng (ID) được cấp cho một ứng dụng độc hại nhằm truy cập vào dữ liệu về cùng một người dùng trên máy chủ phụ trợ của ứng dụng. - Giá trị của
isstrong mã thông báo nhận dạng bằngaccounts.google.comhoặchttps://accounts.google.com. - Thời gian hết hạn (
exp) của mã thông báo nhận dạng chưa trôi qua. - Nếu cần xác thực rằng mã thông báo nhận dạng đại diện cho tài khoản tổ chức Google Workspace hoặc Cloud, bạn có thể kiểm tra khai báo
hd. Khai báo này cho biết miền được lưu trữ của người dùng. Bạn phải sử dụng tham số này khi chỉ cho phép thành viên của một số miền nhất định truy cập vào một tài nguyên. Việc thiếu xác nhận quyền sở hữu này cho biết rằng tài khoản không thuộc về một miền do Google lưu trữ.
Bằng cách sử dụng các trường email, email_verified và hd, bạn có thể xác định xem Google có lưu trữ và có quyền đối với một địa chỉ email hay không. Trong trường hợp Google là nguồn đáng tin cậy, người dùng được xác định là chủ sở hữu hợp pháp của tài khoản và bạn có thể bỏ qua mật khẩu hoặc các phương thức thử thách khác.
Các trường hợp Google là nguồn thông tin xác thực:
emailcó đuôi là@gmail.com, đây là tài khoản Gmail.email_verifiedlà true vàhdđược đặt, thì đây là tài khoản Google Workspace.
Người dùng có thể đăng ký Tài khoản Google mà không cần sử dụng Gmail hoặc Google Workspace. Khi email không chứa hậu tố @gmail.com và không có hd, Google không phải là nguồn có thẩm quyền và bạn nên dùng mật khẩu hoặc các phương thức thử thách khác để xác minh người dùng. email_verified cũng có thể đúng vì ban đầu Google đã xác minh người dùng khi tài khoản Google được tạo. Tuy nhiên, quyền sở hữu tài khoản email bên thứ ba có thể đã thay đổi kể từ đó.
Thay vì tự viết mã để thực hiện các bước xác minh này, bạn nên sử dụng một thư viện ứng dụng API của Google cho nền tảng của mình hoặc một thư viện JWT đa năng. Để phát triển và gỡ lỗi, bạn có thể gọi điểm cuối xác thực tokeninfo của chúng tôi.
Sử dụng Thư viện ứng dụng API của Google
Sử dụng một trong các Thư viện ứng dụng API của Google (ví dụ: Java, Node.js PHP, Python) bạn nên dùng để xác thực mã thông báo mã nhận dạng Google trong môi trường phát hành chính thức.
Để xác thực mã thông báo giá trị nhận dạng trong Java, hãy sử dụng GoogleIdTokenVerifier. Ví dụ:
import com.google.api.client.googleapis.auth.oauth2.GoogleIdToken; import com.google.api.client.googleapis.auth.oauth2.GoogleIdToken.Payload; import com.google.api.client.googleapis.auth.oauth2.GoogleIdTokenVerifier; ... GoogleIdTokenVerifier verifier = new GoogleIdTokenVerifier.Builder(transport, jsonFactory) // Specify the WEB_CLIENT_ID of the app that accesses the backend: .setAudience(Collections.singletonList(WEB_CLIENT_ID)) // Or, if multiple clients access the backend: //.setAudience(Arrays.asList(WEB_CLIENT_ID_1, WEB_CLIENT_ID_2, WEB_CLIENT_ID_3)) .build(); // (Receive idTokenString by HTTPS POST) GoogleIdToken idToken = verifier.verify(idTokenString); if (idToken != null) { Payload payload = idToken.getPayload(); // Print user identifier. This ID is unique to each Google Account, making it suitable for // use as a primary key during account lookup. Email is not a good choice because it can be // changed by the user. String userId = payload.getSubject(); System.out.println("User ID: " + userId); // Get profile information from payload String email = payload.getEmail(); boolean emailVerified = Boolean.valueOf(payload.getEmailVerified()); String name = (String) payload.get("name"); String pictureUrl = (String) payload.get("picture"); String locale = (String) payload.get("locale"); String familyName = (String) payload.get("family_name"); String givenName = (String) payload.get("given_name"); // Use or store profile information // ... } else { System.out.println("Invalid ID token."); }
Phương thức GoogleIdTokenVerifier.verify() xác minh JWT
chữ ký, xác nhận quyền sở hữu aud, xác nhận quyền sở hữu iss và
Thông báo xác nhận quyền sở hữu exp.
Trường hợp bạn cần xác thực rằng mã thông báo mã nhận dạng đại diện cho Google Workspace hoặc Cloud
tài khoản tổ chức, bạn có thể xác minh thông báo xác nhận quyền sở hữu hd bằng cách kiểm tra tên miền
được phương thức Payload.getHostedDomain() trả về. Miền của
Khiếu nại email là không đủ để đảm bảo rằng tài khoản này do một miền quản lý
hoặc tổ chức.
Để xác thực mã thông báo mã nhận dạng trong Node.js, hãy sử dụng Thư viện xác thực của Google cho Node.js. Cài đặt thư viện:
npm install google-auth-library --save
verifyIdToken(). Ví dụ:
const {OAuth2Client} = require('google-auth-library'); const client = new OAuth2Client(); async function verify() { const ticket = await client.verifyIdToken({ idToken: token, audience: WEB_CLIENT_ID, // Specify the WEB_CLIENT_ID of the app that accesses the backend // Or, if multiple clients access the backend: //[WEB_CLIENT_ID_1, WEB_CLIENT_ID_2, WEB_CLIENT_ID_3] }); const payload = ticket.getPayload(); // This ID is unique to each Google Account, making it suitable for use as a primary key // during account lookup. Email is not a good choice because it can be changed by the user. const userid = payload['sub']; // If the request specified a Google Workspace domain: // const domain = payload['hd']; } verify().catch(console.error);
Hàm verifyIdToken xác minh
chữ ký JWT, xác nhận quyền sở hữu aud, xác nhận quyền sở hữu exp,
và thông báo xác nhận quyền sở hữu iss.
Trường hợp bạn cần xác thực rằng mã thông báo mã nhận dạng đại diện cho Google Workspace hoặc Cloud
tài khoản tổ chức của mình, bạn có thể kiểm tra xác nhận quyền sở hữu hd, cho biết
của người dùng. Bạn phải sử dụng chế độ này khi hạn chế quyền truy cập vào tài nguyên chỉ cho thành viên
của một số miền nhất định. Việc không có xác nhận quyền sở hữu này cho thấy rằng tài khoản không thuộc về
tên miền được Google lưu trữ.
Để xác thực mã thông báo giá trị nhận dạng trong PHP, hãy sử dụng Thư viện ứng dụng API của Google cho PHP. Cài đặt thư viện (ví dụ: sử dụng Composer):
composer require google/apiclient
verifyIdToken(). Ví dụ:
require_once 'vendor/autoload.php'; // Get $id_token via HTTPS POST. $client = new Google_Client(['client_id' => $WEB_CLIENT_ID]); // Specify the WEB_CLIENT_ID of the app that accesses the backend $payload = $client->verifyIdToken($id_token); if ($payload) { // This ID is unique to each Google Account, making it suitable for use as a primary key // during account lookup. Email is not a good choice because it can be changed by the user. $userid = $payload['sub']; // If the request specified a Google Workspace domain //$domain = $payload['hd']; } else { // Invalid ID token }
Hàm verifyIdToken xác minh
chữ ký JWT, xác nhận quyền sở hữu aud, xác nhận quyền sở hữu exp,
và thông báo xác nhận quyền sở hữu iss.
Trường hợp bạn cần xác thực rằng mã thông báo mã nhận dạng đại diện cho Google Workspace hoặc Cloud
tài khoản tổ chức của mình, bạn có thể kiểm tra xác nhận quyền sở hữu hd, cho biết
của người dùng. Bạn phải sử dụng chế độ này khi hạn chế quyền truy cập vào tài nguyên chỉ cho thành viên
của một số miền nhất định. Việc không có xác nhận quyền sở hữu này cho thấy rằng tài khoản không thuộc về
tên miền được Google lưu trữ.
Để xác thực mã thông báo nhận dạng trong Python, hãy sử dụng verify_oauth2_token . Ví dụ:
from google.oauth2 import id_token from google.auth.transport import requests # (Receive token by HTTPS POST) # ... try: # Specify the WEB_CLIENT_ID of the app that accesses the backend: idinfo = id_token.verify_oauth2_token(token, requests.Request(), WEB_CLIENT_ID) # Or, if multiple clients access the backend server: # idinfo = id_token.verify_oauth2_token(token, requests.Request()) # if idinfo['aud'] not in [WEB_CLIENT_ID_1, WEB_CLIENT_ID_2, WEB_CLIENT_ID_3]: # raise ValueError('Could not verify audience.') # If the request specified a Google Workspace domain # if idinfo['hd'] != DOMAIN_NAME: # raise ValueError('Wrong domain name.') # ID token is valid. Get the user's Google Account ID from the decoded token. # This ID is unique to each Google Account, making it suitable for use as a primary key # during account lookup. Email is not a good choice because it can be changed by the user. userid = idinfo['sub'] except ValueError: # Invalid token pass
Hàm verify_oauth2_token xác minh JWT
chữ ký, xác nhận quyền sở hữu aud và xác nhận quyền sở hữu exp.
Bạn cũng phải xác minh hd
tuyên bố (nếu có) bằng cách kiểm tra đối tượng
Trả lại hàng với mức phí verify_oauth2_token. Nếu có nhiều khách hàng truy cập vào
máy chủ phụ trợ, cũng như xác minh thông báo xác nhận quyền sở hữu aud theo cách thủ công.
Calling the tokeninfo endpoint
An easy way to validate an ID token signature for debugging is to
use the tokeninfo endpoint. Calling this endpoint involves an
additional network request that does most of the validation for you while you test proper
validation and payload extraction in your own code. It is not suitable for use in production
code as requests may be throttled or otherwise subject to intermittent errors.
To validate an ID token using the tokeninfo endpoint, make an HTTPS
POST or GET request to the endpoint, and pass your ID token in the
id_token parameter.
For example, to validate the token "XYZ123", make the following GET request:
https://oauth2.googleapis.com/tokeninfo?id_token=XYZ123
If the token is properly signed and the iss and exp
claims have the expected values, you will get a HTTP 200 response, where the body
contains the JSON-formatted ID token claims.
Here's an example response:
{
// These six fields are included in all Google ID Tokens.
"iss": "https://accounts.google.com",
"sub": "110169484474386276334",
"azp": "1008719970978-hb24n2dstb40o45d4feuo2ukqmcc6381.apps.googleusercontent.com",
"aud": "1008719970978-hb24n2dstb40o45d4feuo2ukqmcc6381.apps.googleusercontent.com",
"iat": "1433978353",
"exp": "1433981953",
// These seven fields are only included when the user has granted the "profile" and
// "email" OAuth scopes to the application.
"email": "testuser@gmail.com",
"email_verified": "true",
"name" : "Test User",
"picture": "https://lh4.googleusercontent.com/-kYgzyAWpZzJ/ABCDEFGHI/AAAJKLMNOP/tIXL9Ir44LE/s99-c/photo.jpg",
"given_name": "Test",
"family_name": "User",
"locale": "en"
}If you need to validate that the ID token represents a Google Workspace account, you can check
the hd claim, which indicates the hosted domain of the user. This must be used when
restricting access to a resource to only members of certain domains. The absence of this claim
indicates that the account does not belong to a Google Workspace hosted domain.
Tạo tài khoản hoặc phiên
Sau khi bạn xác minh mã thông báo, hãy kiểm tra xem người dùng đã có trong cơ sở dữ liệu người dùng của bạn hay chưa. Nếu có, hãy thiết lập một phiên được xác thực cho người dùng. Nếu người dùng chưa có trong cơ sở dữ liệu người dùng, hãy tạo bản ghi người dùng mới từ thông tin trong tải trọng mã thông báo nhận dạng và thiết lập một phiên cho người dùng. Bạn có thể nhắc người dùng cung cấp bất kỳ thông tin hồ sơ bổ sung nào mà bạn yêu cầu khi phát hiện một người dùng mới tạo trong ứng dụng.
Bảo mật tài khoản của người dùng bằng tính năng Bảo vệ nhiều tài khoản
Khi dựa vào Google để đăng nhập cho người dùng, bạn sẽ tự động được hưởng lợi từ tất cả các tính năng bảo mật và cơ sở hạ tầng mà Google đã xây dựng để bảo vệ dữ liệu của người dùng. Tuy nhiên, trong trường hợp hiếm gặp là Tài khoản Google của người dùng bị xâm phạm hoặc sự kiện bảo mật quan trọng, ứng dụng của bạn cũng có thể dễ bị tấn công. Để bảo vệ tài khoản khỏi bất kỳ sự kiện bảo mật lớn nào, hãy sử dụng tính năng Nhiều tài khoản Tính năng bảo vệ nhận cảnh báo bảo mật từ Google. Khi nhận được những sự kiện này, bạn biết được những thay đổi quan trọng đối với tính bảo mật của Tài khoản Google của người dùng và từ đó, bạn có thể thực hiện hành động đối với dịch vụ đó để bảo mật tài khoản của mình.