+2

Access Google Sheets API trong Kotlin sử dụng Service Account

Chúng ta ắt hẳn đều thường xuyên làm việc với Google sheet hàng ngày, trong bài viết này, hãy cùng mình tìm hiểu các để tự động hóa những công việc trên Google Sheet trong Kotlin nhé!

Đầu tiên hãy nghía qua Google Sheets API Docs một chút.

Trước tiên hãy xem chúng ta có những cách nào để truy cập Google Sheets API và hãy chọn cho mình một phương pháp phù hợp nhé:

  • API keys
  • Service Accounts
  • OAuth 2.0 Client IDs

Trong bài viết này mình sẽ giới thiệu cho các bạn cách số 2 dùng Service Accounts. Còn 2 cách còn lại các bạn có thể tham khảo tại đây

Truy cập Google Sheets API bằng cách sử dụng Service Accounts

Bước 1: Google Cloud

Đi đến trang Google Cloud Console và cài đặt project mới cho riêng bạn.

Bước 2: Kích hoạt Google Sheets

Đầu tiên, click vào search bar và tìm kiếm "Google Sheets API"

image.png

Tiếp theo chọn "Enable"

image.png

Bước 3: Tạo Service Account

Bây giờ, hãy tạo Service Account để chúng ta có thể xác nhận ứng dụng của chúng ta sau này.

image.png

Ở mục lựa chọn đầu tiên, hãy chọn "Google Sheets API".

Ta sẽ truy cập vào data nào? Hãy ghi nhớ rằng service account sẽ sao chép các hành động của người dùng. Nó không cho phép bạn truy cập vào dữ liệu của người dùng.

image.png

Dữ liệu thuộc về người dùng Google, như địa chỉ email hoặc tuổi. Cần có sự đồng ý của người dùng. Điều này sẽ tạo ra một client OAuth. Bạn có thể sử dụng GCE, GKE hay các sản phẩm khác của Google .. tùy ý.

image.png

Chọn Next

image.png

Tìm trong "Detail" và chọn "CREATE AND CONTINUE"

image.png

Chọn Owner hoặc Editor. Phần còn lại thì cứ để như tùy chỉnh default.

Bước 4: Lấy credentials locally

image.png

Đơn giản chỉ cần copy email ID và lưu nó ở đâu đó. Sau đấy thì click vào đây

image.png

Tại đây, chọn "KEYS" và "ADD KEYS".

Chọn "CREATE NEW KEY" và chọn "JSON":

image.png

Tải và lưu file JSON vào trong folder app của bạn. Sau đó đổi tên file thành "cred.json"

Bước 5: Chia sẻ Sheet cho Service Account

Cũng giống như bất kỳ Google Account nào khác, Service Account cần quyền truy cập vào bất kỳ tệp nào từ Tài khoản Google của bạn. Nó hoạt động như một con người riêng biệt hay nói cách khác nó là một con BOT. Vì vậy, mình hy vọng bạn vẫn lưu lại ID email ở trên. Bây giờ hãy dùng nó để chia sẻ file cho service account.

Để làm điều này, ta cần mở Google Sheet mà bạn muốn cấp quyền cho Service Account, chọn "share"

image.png

Hãy dán Service Account ID vào

image.png

Chọn "Editor" và "Send"

Lưu ý là id của Sheet rất quan trọng. Chúng ta sẽ sử dụng id của Sheet trong Script. Bạn có thể sử dụng Service Account cho nhiều Google Sheeet và xác định mỗi Sheet theo id thực của nó trong code.

Bước 6: Code

Đầu tiên thêm package Google api client.

// Google
implementation("com.google.api-client:google-api-client:1.30.4")
implementation("com.google.apis:google-api-services-sheets:v4-rev581-1.25.0")

Thêm file cred.json vào folder resources và code như ví dụ sau:

class GoogleService {
    private val spreadsheetId = "1ncYGCMaF-lYdSuUv0TAD_dhbnhCEU9WHW4U2NvUavd8" // id của file google của bạn 
    private val range = "'Trang tính2'!B2:G5" // Phạm vi bạn muốn đọc dữ liệu ở sheet google
    private val scopes = Collections.singletonList(SheetsScopes.SPREADSHEETS_READONLY)

    fun getData(): List<Any>? {
        val jsonFactory = JacksonFactory.getDefaultInstance()

        val resourceAsStream: InputStream = GoogleService::class.java.getResourceAsStream("/cred.json")
            ?: throw Exception()

        val credential = fromStream(resourceAsStream).createScoped(scopes)

        return try {
            credential.refreshToken()

            val service = Sheets.Builder(GoogleNetHttpTransport.newTrustedTransport(), jsonFactory, credential)
            val response = service.build().spreadsheets().values()
                .get(spreadsheetId, range).execute()
           return response.getValues()
        } catch (e: Exception) {
            // create log error
        }
    }
}

Mình sẽ giải thích cụ thể về đoạn code trên:

  • Scopes: Ở đây chúng ta chỉ cần đọc dữ liệu của google sheet nên scopes chỉ cần read only. cụ thể hơn là 1 mảng 1 phần tử ("https://www.googleapis.com/auth/spreadsheets.readonly"). Khi chúng ta cần ghi vào google sheet thì chúng ta sẽ thêm các scope khác. Cụ thể về các loại scope các bạn xem thêm tại đây

  • credential: Sau khi đọc data từ file cred.json và chuyển thành GoogleCredential. kết quả như ảnh đính kèm sau image.png

  • refreshToken: Nếu bạn muốn access vào google sheet, bạn cần có 1 access_token khi call api get data google sheet. credential.refreshToken() sẽ dùng các thông tin credential và thực hiện call api "https://oauth2.googleapis.com/token" để lấy access_token, nếu thành công sẽ return true. Khi đó credential sẽ như ảnh sau: image.png

    Thường nếu không setting gì thì expired của token sẽ hết hạn sau 1 tiếng.

  • Các bước tiếp theo, chúng ta dùng credential để get data của sheet phụ thuộc và sheetId và range đã được chỉ định trước đó.

Trường hợp cần bảo mật các thông tin ở trong file cred.json chúng ta không nên thêm file này vào trong folder của project, vì thế bạn có thể chuyển đổi file thành kiểu chuỗi String Base64 sau đó lưu biến này vào file .env Khi đó code sẽ như sau:

class GoogleServiceImpl {
    private val credentialBase64 = "example" // Bạn phải convert file cred.json thành kiểu Base64 sau đó lưu vào biến .env hoặc props ở file *.yml
    private val spreadsheetId = "1ncYGCMaF-lYdSuUv0TAD_dhbnhCEU9WHW4U2NvUavd8" // id của file google của bạn 
    private val range = "'Trang tính2'!B2:G5" // Phạm vi bạn muốn đọc dữ liệu ở sheet google
    private val scopes = Collections.singletonList(SheetsScopes.SPREADSHEETS_READONLY)

    fun getData(): List<Any>? {
        val jsonFactory = JacksonFactory.getDefaultInstance()

        val credential = try {
            fromStream(ByteArrayInputStream(Base64.decodeBase64(credentialBase64))).createScoped(scopes)
        } catch (e: Exception) {
            // create log error
        }

        return try {
            credential.refreshToken()

            val service = Sheets.Builder(GoogleNetHttpTransport.newTrustedTransport(), jsonFactory, credential)
            val response = service.build().spreadsheets().values()
                .get(spreadsheetId, range).execute()
           return response.getValues()
        } catch (e: Exception) {
            // create log error
        }
    }
}

Note: Bạn có thể tìm hiểu thêm ở cách diễn đạt dùng HTTP/REST tại đây

Bài viết của mình đến đây là hết, nếu có thắc mắc gì hãy để lại comment bên dưới nhé.

Bài viết sử dụng ảnh và các bước tạo service account từ nguồn!


All rights reserved

Viblo
Hãy đăng ký một tài khoản Viblo để nhận được nhiều bài viết thú vị hơn.
Đăng kí