Xử lý trường hợp đăng nhập nhiều lần

Đây là hướng dẫn thứ ba trong chuỗi hướng dẫn từng bước về tiện ích bổ sung trong Lớp học.

Trong hướng dẫn này, bạn sẽ xử lý các lượt truy cập nhiều lần vào tiện ích bổ sung của chúng tôi bằng cách tự động truy xuất thông tin đăng nhập đã cấp trước đó của người dùng. Sau đó, bạn sẽ chuyển người dùng đến các trang mà họ có thể gửi yêu cầu API ngay lập tức. Đây là hành vi bắt buộc đối với các tiện ích bổ sung của Lớp học.

Trong quá trình hướng dẫn này, bạn sẽ hoàn thành các bước sau:

  • Triển khai bộ nhớ liên tục cho thông tin đăng nhập của người dùng.
  • Truy xuất và đánh giá các tham số truy vấn của tiện ích bổ sung sau đây:
    • login_hint: Mã nhận dạng trên Google của người dùng đã đăng nhập.
    • hd: Miền của người dùng đã đăng nhập.

Xin lưu ý rằng chỉ một trong số các mã này được gửi. API Lớp học sẽ gửi tham số hd nếu người dùng KHÔNG uỷ quyền cho ứng dụng của bạn. Nếu không, API sẽ gửi login_hint. Danh sách đầy đủ các tham số truy vấn có sẵn trong trang hướng dẫn về iframe.

Sau khi hoàn tất, bạn có toàn quyền uỷ quyền cho người dùng trong ứng dụng web của mình và thực hiện lệnh gọi đến các API của Google.

Tìm hiểu về các tham số truy vấn iframe

Lớp học sẽ tải URI thiết lập tệp đính kèm của tiện ích bổ sung khi mở. Lớp học sẽ thêm một số tham số truy vấn GET vào URI; các tham số này chứa thông tin hữu ích về ngữ cảnh. Chẳng hạn, nếu URI phát hiện tệp đính kèm là https://example.com/addon, Lớp học sẽ tạo iframe với URL nguồn được đặt thành https://example.com/addon?courseId=XXX&postId=YYY&addOnToken=ZZZ, trong đó XXX, YYYZZZ là các mã nhận dạng chuỗi. Xem hướng dẫn về iframe để biết nội dung mô tả chi tiết về trường hợp này.

Có thể có 5 tham số truy vấn cho URL khám phá:

  • courseId: Mã của khoá học hiện tại trên Lớp học.
  • postId: Mã của bài đăng về bài tập mà người dùng đang chỉnh sửa hoặc tạo.
  • addOnToken: Mã thông báo dùng để uỷ quyền một số thao tác nhất định của tiện ích bổ sung cho Lớp học.
  • login_hint: Mã nhận dạng trên Google của người dùng hiện tại.
  • hd: Miền lưu trữ cho người dùng hiện tại, chẳng hạn như example.com.

Hướng dẫn này hướng dẫn hdlogin_hint. Người dùng được định tuyến dựa trên bất kỳ tham số truy vấn nào được cung cấp, vào quy trình uỷ quyền nếu là hd hoặc đến trang khám phá tiện ích bổ sung nếu login_hint.

Truy cập vào tham số truy vấn

Như đã mô tả ở trên, các tham số truy vấn được truyền đến ứng dụng web của bạn trong chuỗi URI. Lưu trữ các giá trị này trong phiên của bạn; chúng được dùng trong quy trình uỷ quyền, cũng như để lưu trữ và truy xuất thông tin về người dùng. Các tham số truy vấn này chỉ được truyền khi tiện ích bổ sung được mở lần đầu tiên.

Python

Chuyển đến phần định nghĩa của các tuyến trong Flask (routes.py nếu bạn đang làm theo ví dụ mà chúng tôi cung cấp). Ở đầu tuyến đích của tiện ích bổ sung (/classroom-addon trong ví dụ của chúng tôi), hãy truy xuất và lưu trữ các tham số truy vấn login_hinthd:

# Retrieve the login_hint and hd query parameters.
login_hint = flask.request.args.get("login_hint")
hd = flask.request.args.get("hd")

Đảm bảo login_hinthd được lưu trữ trong phiên. Đây là nơi thích hợp để lưu trữ các giá trị này; chúng mang tính tạm thời và bạn sẽ nhận được các giá trị mới khi tiện ích bổ sung được mở.

# It's possible that we might return to this route later, in which case the
# parameters will not be passed in. Instead, use the values cached in the
# session.

# If neither query parameter is available, use the values in the session.
if login_hint is None and hd is None:
    login_hint = flask.session.get("login_hint")
    hd = flask.session.get("hd")

# If there's no login_hint query parameter, then check for hd.
# Send the user to the sign in page.
elif hd is not None:
    flask.session["hd"] = hd
    return start_auth_flow()

# If the login_hint query parameter is available, we'll store it in the
# session.
else:
    flask.session["login_hint"] = login_hint

Java

Chuyển đến tuyến đích của tiện ích bổ sung trong lớp tay điều khiển (/addon-discovery trong AuthController.java trong ví dụ được cung cấp). Ở đầu tuyến này, hãy truy xuất và lưu trữ các tham số truy vấn login_hinthd.

/** Retrieve the login_hint or hd query parameters from the request URL. */
String login_hint = request.getParameter("login_hint");
String hd = request.getParameter("hd");

Đảm bảo login_hinthd được lưu trữ trong phiên. Đây là nơi thích hợp để lưu trữ các giá trị này; chúng mang tính tạm thời và bạn sẽ nhận được các giá trị mới khi tiện ích bổ sung được mở.

/** If neither query parameter is sent, use the values in the session. */
if (login_hint == null && hd == null) {
    login_hint = (String) session.getAttribute("login_hint");
    hd = (String) session.getAttribute("hd");
}

/** If the hd query parameter is provided, add hd to the session and route
*   the user to the authorization page. */
else if (hd != null) {
    session.setAttribute("hd", hd);
    return startAuthFlow(model);
}

/** If the login_hint query parameter is provided, add it to the session. */
else if (login_hint != null) {
    session.setAttribute("login_hint", login_hint);
}

Thêm tham số truy vấn vào quy trình uỷ quyền

Các tham số login_hinthd cũng phải được truyền đến máy chủ xác thực của Google. Các phương thức này hỗ trợ quá trình xác thực; nếu ứng dụng của bạn biết người dùng nào đang cố gắng xác thực, thì máy chủ sẽ sử dụng gợi ý để đơn giản hoá quy trình đăng nhập bằng cách điền sẵn trường email trong biểu mẫu đăng nhập.

Python

Chuyển đến tuyến uỷ quyền trong tệp máy chủ Flask (/authorize trong ví dụ của chúng tôi). Thêm các đối số login_hinthd vào lệnh gọi đến flow.authorization_url.

authorization_url, state = flow.authorization_url(
    # Enable offline access so that you can refresh an access token without
    # re-prompting the user for permission. Recommended for web server apps.
    access_type="offline",
    # Enable incremental authorization. Recommended as a best practice.
    include_granted_scopes="true",
    # The user will automatically be selected if we have the login_hint.
    login_hint=flask.session.get("login_hint"),
    # If we don't have login_hint, passing hd will reduce the list of
    # accounts in the account chooser to only those with the same domain.
    hd=flask.session.get("hd"))

Java

Chuyển đến phương thức authorize() trong lớp AuthService.java. Thêm login_hinthd làm tham số vào phương thức, đồng thời thêm các đối số login_hinthd vào trình tạo URL uỷ quyền.

String authUrl = flow
    .newAuthorizationUrl()
    .setState(state)
    .set("login_hint", login_hint)
    .set("hd", hd)
    .setRedirectUri(REDIRECT_URI)
    .build();

Thêm bộ nhớ ổn định cho thông tin đăng nhập của người dùng

Nếu bạn nhận được login_hint dưới dạng tham số truy vấn khi tiện ích bổ sung tải, thì đó là dấu hiệu cho biết người dùng đã hoàn tất quy trình uỷ quyền cho ứng dụng của chúng ta. Bạn nên truy xuất thông tin xác thực trước đó của trẻ thay vì buộc trẻ đăng nhập lại.

Hãy nhớ rằng bạn đã nhận được một mã làm mới sau khi hoàn tất quy trình uỷ quyền. Lưu mã thông báo này; bạn có thể sử dụng lại mã này để lấy mã truy cập. Mã này là một mã ngắn hạn và cần thiết để sử dụng các API của Google. Trước đây, bạn đã lưu các thông tin đăng nhập này trong phiên, nhưng bạn cần lưu trữ thông tin đăng nhập để xử lý các lượt truy cập lặp lại.

Xác định giản đồ người dùng và thiết lập cơ sở dữ liệu

Thiết lập giản đồ cơ sở dữ liệu cho User.

Python

Xác định Giản đồ người dùng

User chứa các thuộc tính sau:

  • id: Mã nhận dạng Google của người dùng. Giá trị này phải khớp với các giá trị được cung cấp trong tham số truy vấn login_hint.
  • display_name: Họ và tên của người dùng, chẳng hạn như "Alex Smith".
  • email: Địa chỉ email của người dùng.
  • portrait_url: URL ảnh hồ sơ của người dùng.
  • refresh_token: Mã làm mới đã nhận trước đó.

Ví dụ này triển khai việc lưu trữ bằng SQLite vốn được Python hỗ trợ. Công cụ này sử dụng mô-đun flask_sqlalchemy để hỗ trợ việc quản lý cơ sở dữ liệu của chúng tôi.

Thiết lập cơ sở dữ liệu

Trước tiên, hãy chỉ định vị trí tệp cho cơ sở dữ liệu của chúng ta. Chuyển đến tệp cấu hình máy chủ (số config.py trong ví dụ của chúng tôi) và thêm đoạn mã sau.

import os

# Point to a database file in the project root.
DATABASE_FILE_NAME = os.path.join(
    os.path.abspath(os.path.dirname(__file__)), 'data.sqlite')

class Config(object):
    SQLALCHEMY_DATABASE_URI = f"sqlite:///{DATABASE_FILE_NAME}"
    SQLALCHEMY_TRACK_MODIFICATIONS = False

Thao tác này sẽ trỏ Flask đến tệp data.sqlite trong cùng thư mục với tệp main.py.

Tiếp theo, hãy chuyển đến thư mục mô-đun và tạo tệp models.py mới. Đây là giá trị webapp/models.py nếu bạn đang làm theo ví dụ mà chúng tôi cung cấp. Thêm nội dung sau vào tệp mới để xác định bảng User, thay thế tên mô-đun bằng webapp nếu khác.

from webapp import db

# Database model to represent a user.
class User(db.Model):
    # The user's identifying information:
    id = db.Column(db.String(120), primary_key=True)
    display_name = db.Column(db.String(80))
    email = db.Column(db.String(120), unique=True)
    portrait_url = db.Column(db.Text())

    # The user's refresh token, which will be used to obtain an access token.
    # Note that refresh tokens will become invalid if:
    # - The refresh token has not been used for six months.
    # - The user revokes your app's access permissions.
    # - The user changes passwords.
    # - The user belongs to a Google Cloud organization
    #   that has session control policies in effect.
    refresh_token = db.Column(db.Text())

Cuối cùng, trong tệp __init__.py của mô-đun, hãy thêm đoạn mã sau để nhập các mô hình mới và tạo cơ sở dữ liệu.

from webapp import models
from os import path
from flask_sqlalchemy import SQLAlchemy

db = SQLAlchemy(app)

# Initialize the database file if not created.
if not path.exists(config.DATABASE_FILE_NAME):
    db.create_all()

Java

Xác định Giản đồ người dùng

User chứa các thuộc tính sau:

  • id: Mã nhận dạng Google của người dùng. Giá trị này phải khớp với giá trị được cung cấp trong tham số truy vấn login_hint.
  • email: Địa chỉ email của người dùng.

Tạo tệp schema.sql trong thư mục resources của mô-đun. Spring đọc tệp này và tạo giản đồ cho cơ sở dữ liệu tương ứng. Xác định bảng bằng tên bảng là users và các cột biểu thị các thuộc tính User, idemail.

CREATE TABLE IF NOT EXISTS users (
    id VARCHAR(255) PRIMARY KEY, -- user's unique Google ID
    email VARCHAR(255), -- user's email address
);

Tạo một lớp Java để định nghĩa mô hình User cho cơ sở dữ liệu. Giá trị này là User.java trong ví dụ đã cho.

Thêm chú giải @Entity để cho biết rằng đây là POJO có thể được lưu vào cơ sở dữ liệu. Thêm chú giải @Table có tên bảng tương ứng mà bạn đã định cấu hình trong schema.sql.

Lưu ý rằng ví dụ về mã bao gồm các hàm khởi tạo và phương thức setter cho hai thuộc tính. Hàm khởi tạo và phương thức setter được sử dụng trong AuthController.java để tạo hoặc cập nhật người dùng trong cơ sở dữ liệu. Bạn cũng có thể bao gồm phương thức getter và phương thức toString nếu thấy phù hợp, nhưng đối với hướng dẫn cụ thể này, các phương thức này không được sử dụng và bị xoá khỏi ví dụ về mã trên trang này cho ngắn gọn.

/** An entity class that provides a model to store user information. */
@Entity
@Table(name = "users")
public class User {
    /** The user's unique Google ID. The @Id annotation specifies that this
     *   is the primary key. */
    @Id
    @Column
    private String id;

    /** The user's email address. */
    @Column
    private String email;

    /** Required User class no args constructor. */
    public User() {
    }

    /** The User class constructor that creates a User object with the
    *   specified parameters.
    *   @param id the user's unique Google ID
    *   @param email the user's email address
    */
    public User(String id, String email) {
        this.id = id;
        this.email = email;
    }

    public void setId(String id) { this.id = id; }

    public void setEmail(String email) { this.email = email; }
}

Tạo một giao diện có tên là UserRepository.java để xử lý các thao tác CRUD cho cơ sở dữ liệu. Giao diện này mở rộng giao diện CrudRepository.

/** Provides CRUD operations for the User class by extending the
 *   CrudRepository interface. */
@Repository
public interface UserRepository extends CrudRepository<User, String> {
}

Lớp trình điều khiển hỗ trợ hoạt động giao tiếp giữa ứng dụng và kho lưu trữ. Do đó, hãy cập nhật hàm khởi tạo lớp tay điều khiển để chèn lớp UserRepository.

/** Declare UserRepository to be used in the Controller class constructor. */
private final UserRepository userRepository;

/**
*   ...
*   @param userRepository the class that interacts with User objects stored in
*   persistent storage.
*/
public AuthController(AuthService authService, UserRepository userRepository) {
    this.authService = authService;
    this.userRepository = userRepository;
}

Thiết lập cơ sở dữ liệu

Để lưu trữ thông tin liên quan đến người dùng, hãy sử dụng cơ sở dữ liệu H2 về bản chất được hỗ trợ trong Spring Boot. Cơ sở dữ liệu này cũng được dùng trong các hướng dẫn tiếp theo để lưu trữ các thông tin khác liên quan đến Lớp học. Để thiết lập cơ sở dữ liệu H2, bạn cần thêm cấu hình sau vào application.properties.

# Enable configuration for persistent storage using an H2 database
spring.datasource.driver-class-name=org.h2.Driver
spring.datasource.url=jdbc:h2:file:./h2/userdb
spring.datasource.username=<USERNAME>
spring.datasource.password=<PASSWORD>
spring.jpa.hibernate.ddl-auto=update
spring.jpa.open-in-view=false

Cấu hình spring.datasource.url tạo một thư mục có tên là h2 chứa tệp userdb được lưu trữ bên trong. Thêm đường dẫn đến cơ sở dữ liệu H2 vào .gitignore. Bạn phải cập nhật spring.datasource.usernamespring.datasource.password trước khi chạy ứng dụng để đặt cơ sở dữ liệu bằng tên người dùng và mật khẩu do bạn chọn. Để cập nhật tên người dùng và mật khẩu cho cơ sở dữ liệu sau khi chạy ứng dụng, hãy xoá thư mục h2 đã tạo, cập nhật cấu hình rồi chạy lại ứng dụng.

Việc đặt cấu hình spring.jpa.hibernate.ddl-auto thành update đảm bảo rằng dữ liệu lưu trữ trong cơ sở dữ liệu sẽ được giữ nguyên khi khởi động lại ứng dụng. Để xoá cơ sở dữ liệu mỗi khi khởi động lại ứng dụng, hãy đặt cấu hình này thành create.

Đặt cấu hình spring.jpa.open-in-view thành false. Cấu hình này được bật theo mặc định và có thể dẫn đến các vấn đề về hiệu suất khó chẩn đoán trong giai đoạn phát hành chính thức.

Như đã mô tả trước đó, bạn phải truy xuất được thông tin xác thực của một người dùng sử dụng nhiều lần. Việc này được hỗ trợ bởi tính năng hỗ trợ kho thông tin xác thực tích hợp sẵn do GoogleAuthorizationCodeFlow cung cấp.

Trong lớp AuthService.java, hãy xác định một đường dẫn đến tệp lưu trữ lớp thông tin xác thực. Trong ví dụ này, tệp được tạo trong thư mục /credentialStore. Thêm đường dẫn đến kho thông tin xác thực vào .gitignore. Thư mục này được tạo khi người dùng bắt đầu quy trình uỷ quyền.

private static final File dataDirectory = new File("credentialStore");

Tiếp theo, hãy tạo một phương thức trong tệp AuthService.java để tạo và trả về đối tượng FileDataStoreFactory. Đây là kho dữ liệu lưu trữ thông tin xác thực.

/** Creates and returns FileDataStoreFactory object to store credentials.
 *   @return FileDataStoreFactory dataStore used to save and obtain users ids
 *   mapped to Credentials.
 *   @throws IOException if creating the dataStore is unsuccessful.
 */
public FileDataStoreFactory getCredentialDataStore() throws IOException {
    FileDataStoreFactory dataStore = new FileDataStoreFactory(dataDirectory);
    return dataStore;
}

Cập nhật phương thức getFlow() trong AuthService.java để đưa setDataStoreFactory vào phương thức GoogleAuthorizationCodeFlow Builder() và gọi getCredentialDataStore() để đặt kho dữ liệu.

GoogleAuthorizationCodeFlow authorizationCodeFlow =
    new GoogleAuthorizationCodeFlow.Builder(
        HTTP_TRANSPORT,
        JSON_FACTORY,
        getClientSecrets(),
        getScopes())
    .setAccessType("offline")
    .setDataStoreFactory(getCredentialDataStore())
    .build();

Tiếp theo, hãy cập nhật phương thức getAndSaveCredentials(String authorizationCode). Trước đây, phương thức này lấy thông tin xác thực mà không cần lưu trữ ở bất cứ đâu. Cập nhật phương thức để lưu trữ thông tin xác thực trong kho dữ liệu được lập chỉ mục theo mã nhận dạng người dùng.

Bạn có thể lấy mã nhận dạng người dùng từ đối tượng TokenResponse bằng cách sử dụng id_token, nhưng trước tiên, mã nhận dạng này phải được xác minh. Nếu không, các ứng dụng khách có thể mạo danh người dùng bằng cách gửi mã nhận dạng người dùng đã sửa đổi tới máy chủ. Bạn nên sử dụng thư viện ứng dụng API của Google để xác thực id_token. Hãy xem [Trang thông tin nhận dạng của Google về cách xác minh mã thông báo mã nhận dạng trên Google] để biết thêm thông tin.

// Obtaining the id_token will help determine which user signed in to the application.
String idTokenString = tokenResponse.get("id_token").toString();

// Validate the id_token using the GoogleIdTokenVerifier object.
GoogleIdTokenVerifier googleIdTokenVerifier = new GoogleIdTokenVerifier.Builder(
        HTTP_TRANSPORT,
        JSON_FACTORY)
    .setAudience(Collections.singletonList(
        googleClientSecrets.getWeb().getClientId()))
    .build();

GoogleIdToken idToken = googleIdTokenVerifier.verify(idTokenString);

if (idToken == null) {
    throw new Exception("Invalid ID token.");
}

Sau khi xác minh id_token, hãy lấy userId để lưu trữ cùng với thông tin đăng nhập đã thu được.

// Obtain the user id from the id_token.
Payload payload = idToken.getPayload();
String userId = payload.getSubject();

Cập nhật lệnh gọi thành flow.createAndStoreCredential để đưa userId vào.

// Save the user id and credentials to the configured FileDataStoreFactory.
Credential credential = flow.createAndStoreCredential(tokenResponse, userId);

Thêm một phương thức vào lớp AuthService.java để trả về thông tin xác thực cho một người dùng cụ thể nếu người dùng đó tồn tại trong kho dữ liệu.

/** Find credentials in the datastore based on a specific user id.
*   @param userId key to find in the file datastore.
*   @return Credential object to be returned if a matching key is found in the datastore. Null if
*   the key doesn't exist.
*   @throws Exception if building flow object or checking for userId key is unsuccessful. */
public Credential loadFromCredentialDataStore(String userId) throws Exception {
    try {
        GoogleAuthorizationCodeFlow flow = getFlow();
        Credential credential = flow.loadCredential(userId);
        return credential;
    } catch (Exception e) {
        e.printStackTrace();
        throw e;
    }
}

Truy xuất thông tin xác thực

Xác định phương thức để tìm nạp Users. Bạn được cung cấp một id trong tham số truy vấn login_hint. Bạn có thể sử dụng tham số này để truy xuất một bản ghi người dùng cụ thể.

Python

def get_credentials_from_storage(id):
    """
    Retrieves credentials from the storage and returns them as a dictionary.
    """
    return User.query.get(id)

Java

Trong lớp AuthController.java, hãy xác định một phương thức để truy xuất người dùng từ cơ sở dữ liệu dựa trên mã nhận dạng người dùng của họ.

/** Retrieves stored credentials based on the user id.
*   @param id the id of the current user
*   @return User the database entry corresponding to the current user or null
*   if the user doesn't exist in the database.
*/
public User getUser(String id) {
    if (id != null) {
        Optional<User> user = userRepository.findById(id);
        if (user.isPresent()) {
            return user.get();
        }
    }
    return null;
}

Thông tin đăng nhập lưu trữ

Có hai trường hợp lưu trữ thông tin xác thực. Nếu id của người dùng đã có trong cơ sở dữ liệu, hãy cập nhật mọi giá trị mới cho bản ghi hiện có. Nếu không, hãy tạo một bản ghi User mới và thêm bản ghi đó vào cơ sở dữ liệu.

Python

Trước tiên, hãy xác định một phương thức tiện ích sẽ triển khai hành vi lưu trữ hoặc cập nhật.

def save_user_credentials(credentials=None, user_info=None):
    """
    Updates or adds a User to the database. A new user is added only if both
    credentials and user_info are provided.

    Args:
        credentials: An optional Credentials object.
        user_info: An optional dict containing user info returned by the
            OAuth 2.0 API.
    """

    existing_user = get_credentials_from_storage(
        flask.session.get("login_hint"))

    if existing_user:
        if user_info:
            existing_user.id = user_info.get("id")
            existing_user.display_name = user_info.get("name")
            existing_user.email = user_info.get("email")
            existing_user.portrait_url = user_info.get("picture")

        if credentials and credentials.refresh_token is not None:
            existing_user.refresh_token = credentials.refresh_token

    elif credentials and user_info:
        new_user = User(id=user_info.get("id"),
                        display_name=user_info.get("name"),
                        email=user_info.get("email"),
                        portrait_url=user_info.get("picture"),
                        refresh_token=credentials.refresh_token)

        db.session.add(new_user)

    db.session.commit()

Có 2 trường hợp bạn có thể lưu thông tin xác thực vào cơ sở dữ liệu của mình: khi người dùng quay lại ứng dụng ở cuối quy trình uỷ quyền và khi thực hiện lệnh gọi API. Đây là nơi trước đây chúng ta đã đặt khoá credentials của phiên.

Gọi save_user_credentials ở cuối tuyến đường callback. Giữ lại đối tượng user_info thay vì chỉ trích xuất tên của người dùng.

# The flow is complete! We'll use the credentials to fetch the user's info.
user_info_service = googleapiclient.discovery.build(
    serviceName="oauth2", version="v2", credentials=credentials)

user_info = user_info_service.userinfo().get().execute()

flask.session["username"] = user_info.get("name")

save_user_credentials(credentials, user_info)

Bạn cũng nên cập nhật thông tin đăng nhập sau các lệnh gọi đến API. Trong trường hợp này, bạn có thể cung cấp thông tin xác thực đã cập nhật làm đối số cho phương thức save_user_credentials.

# Save credentials in case access token was refreshed.
flask.session["credentials"] = credentials_to_dict(credentials)
save_user_credentials(credentials)

Java

Trước tiên, hãy xác định một phương thức lưu trữ hoặc cập nhật đối tượng User trong cơ sở dữ liệu H2.

/** Adds or updates a user in the database.
*   @param credential the credentials object to save or update in the database.
*   @param userinfo the userinfo object to save or update in the database.
*   @param session the current session.
*/
public void saveUser(Credential credential, Userinfo userinfo, HttpSession session) {
    User storedUser = null;
    if (session != null && session.getAttribute("login_hint") != null) {
        storedUser = getUser(session.getAttribute("login_hint").toString());
    }

    if (storedUser != null) {
        if (userinfo != null) {
            storedUser.setId(userinfo.getId());
            storedUser.setEmail(userinfo.getEmail());
        }
        userRepository.save(storedUser);
    } else if (credential != null && userinfo != null) {
        User newUser = new User(
            userinfo.getId(),
            userinfo.getEmail(),
        );
        userRepository.save(newUser);
    }
}

Có 2 trường hợp bạn có thể lưu thông tin xác thực vào cơ sở dữ liệu của mình: khi người dùng quay lại ứng dụng ở cuối quy trình uỷ quyền và khi thực hiện lệnh gọi API. Đây là nơi trước đây chúng ta đã đặt khoá credentials của phiên.

Gọi saveUser ở cuối tuyến /callback. Bạn nên giữ lại đối tượng user_info thay vì chỉ trích xuất email của người dùng.

/** This is the end of the auth flow. We should save user info to the database. */
Userinfo userinfo = authService.getUserInfo(credentials);
saveUser(credentials, userinfo, session);

Bạn cũng nên cập nhật thông tin đăng nhập sau các lệnh gọi đến API. Trong trường hợp này, bạn có thể cung cấp thông tin xác thực đã cập nhật làm đối số cho phương thức saveUser.

/** Save credentials in case access token was refreshed. */
saveUser(credentials, null, session);

Thông tin đăng nhập đã hết hạn

Xin lưu ý rằng có một vài lý do khiến mã làm mới có thể trở nên không hợp lệ. Các cấu hình này bao gồm:

  • Bạn đã không sử dụng mã làm mới trong 6 tháng.
  • Người dùng thu hồi quyền truy cập của ứng dụng.
  • Người dùng đổi mật khẩu.
  • Người dùng thuộc một tổ chức Google Cloud có các chính sách kiểm soát phiên đang có hiệu lực.

Lấy mã thông báo mới bằng cách gửi lại cho người dùng thông qua quy trình uỷ quyền nếu thông tin xác thực của họ không hợp lệ.

Tự động định tuyến người dùng

Sửa đổi tuyến đích của tiện ích bổ sung để phát hiện xem người dùng đã từng cho phép ứng dụng của chúng ta hay chưa. Nếu có, hãy chuyển họ đến trang tiện ích bổ sung chính của chúng tôi. Nếu không, hãy nhắc chúng đăng nhập.

Python

Đảm bảo rằng tệp cơ sở dữ liệu đã được tạo khi ứng dụng khởi chạy. Chèn nội dung sau vào trình khởi tạo mô-đun (chẳng hạn như webapp/__init__.py trong ví dụ đã cung cấp của chúng tôi) hoặc vào phương thức chính để khởi chạy máy chủ.

# Initialize the database file if not created.
if not os.path.exists(DATABASE_FILE_NAME):
    db.create_all()

Sau đó, phương thức của bạn phải xử lý các tham số truy vấn login_hinthd như thảo luận ở trên. Sau đó, tải thông tin đăng nhập cửa hàng nếu đây là khách truy cập lặp lại. Bạn biết đó là khách truy cập thường xuyên nếu bạn nhận được login_hint. Truy xuất mọi thông tin xác thực đã lưu trữ cho người dùng này và tải các thông tin đó vào phiên.

stored_credentials = get_credentials_from_storage(login_hint)

# If we have stored credentials, store them in the session.
if stored_credentials:
    # Load the client secrets file contents.
    client_secrets_dict = json.load(
        open(CLIENT_SECRETS_FILE)).get("web")

    # Update the credentials in the session.
    if not flask.session.get("credentials"):
        flask.session["credentials"] = {}

    flask.session["credentials"] = {
        "token": stored_credentials.access_token,
        "refresh_token": stored_credentials.refresh_token,
        "token_uri": client_secrets_dict["token_uri"],
        "client_id": client_secrets_dict["client_id"],
        "client_secret": client_secrets_dict["client_secret"],
        "scopes": SCOPES
    }

    # Set the username in the session.
    flask.session["username"] = stored_credentials.display_name

Cuối cùng, hãy chuyển người dùng đến trang đăng nhập nếu chúng tôi không có thông tin đăng nhập của họ. Nếu chúng tôi thực hiện, hãy chuyển họ đến trang tiện ích bổ sung chính.

if "credentials" not in flask.session or \
    flask.session["credentials"]["refresh_token"] is None:
    return flask.render_template("authorization.html")

return flask.render_template(
    "addon-discovery.html",
    message="You've reached the addon discovery page.")

Java

Chuyển đến tuyến đích của tiện ích bổ sung (trong ví dụ được cung cấp là /addon-discovery). Như thảo luận ở trên, đây là nơi bạn xử lý các tham số truy vấn login_hinthd.

Trước tiên, hãy kiểm tra xem có thông tin đăng nhập trong phiên đó hay không. Nếu không, hãy định tuyến người dùng thông qua quy trình xác thực bằng cách gọi phương thức startAuthFlow.

/** Check if the credentials exist in the session. The session could have
 *   been cleared when the user clicked the Sign-Out button, and the expected
 *   behavior after sign-out would be to display the sign-in page when the
 *   iframe is opened again. */
if (session.getAttribute("credentials") == null) {
    return startAuthFlow(model);
}

Sau đó, tải người dùng từ cơ sở dữ liệu H2 nếu đây là khách truy cập thường xuyên. Đây là một khách truy cập thường xuyên nếu bạn nhận được tham số truy vấn login_hint. Nếu người dùng tồn tại trong cơ sở dữ liệu H2, hãy tải thông tin xác thực từ kho dữ liệu thông tin xác thực đã thiết lập trước đó rồi đặt thông tin xác thực trong phiên. Nếu không lấy thông tin xác thực từ kho dữ liệu thông tin xác thực, hãy định tuyến người dùng thông qua quy trình xác thực bằng cách gọi startAuthFlow.

/** At this point, we know that credentials exist in the session, but we
 *   should update the session credentials with the credentials in persistent
 *   storage in case they were refreshed. If the credentials in persistent
 *   storage are null, we should navigate the user to the authorization flow
 *   to obtain persisted credentials. */

User storedUser = getUser(login_hint);

if (storedUser != null) {
    Credential credential = authService.loadFromCredentialDataStore(login_hint);
    if (credential != null) {
        session.setAttribute("credentials", credential);
    } else {
        return startAuthFlow(model);
    }
}

Cuối cùng, hãy chuyển người dùng đến trang đích của tiện ích bổ sung.

/** Finally, if there are credentials in the session and in persistent
 *   storage, direct the user to the addon-discovery page. */
return "addon-discovery";

Kiểm thử tiện ích bổ sung

Đăng nhập vào Google Lớp học với tư cách là một trong những người dùng kiểm tra cho Giáo viên. Chuyển đến thẻ Bài tập trên lớp rồi tạo một Bài tập mới. Nhấp vào nút Tiện ích bổ sung bên dưới vùng văn bản rồi chọn tiện ích bổ sung của bạn. Iframe sẽ mở ra và tiện ích bổ sung sẽ tải URI thiết lập tệp đính kèm mà bạn đã chỉ định trên trang Cấu hình ứng dụng của SDK GWM.

Xin chúc mừng! Bạn đã sẵn sàng để chuyển sang bước tiếp theo: tạo tệp đính kèm và xác định vai trò của người dùng.