google-apps-script - 如何使用 google sheet api v 4 将 blob(图像)上传到 google sheet
问题描述
我有一个blob数据。我想使用 google sheet API v4 将此上传到 google sheet 中的单元格。
我在这里查看了文档。 https://developers.google.com/sheets/api/guides/values
我还在这里查看了 SO 问题。 使用 Google Sheets API 将图像插入 Google Sheets 单元格
result = service.spreadsheets().values().update(
spreadsheetId=spreadsheet_id, range=range_name,
valueInputOption=value_input_option, body=body).execute()
我没有看到任何描述将 blob 作为图像插入的服务。请帮忙。
根据以下建议,我们从这里实现了 Webapp -使用 Google Sheets API 将图像插入 Google Sheets 单元格
这就是我们从 Python 代码调用 Web 应用程序的方式
dropoff_signature = "ZGF0YT <clip > WVhSaA=="
web_app_url = "https://script.google.com/macros/s/A < clip > y/exec"
image_data = "data:image/png;base64," + dropoff_signature
data_to_post = {
'spreadsheetid' : spreadsheet_Id,
'sheetname' : 'Sheet1',
'imageurl' : image_data,
'column' : 5,
'row' : 5
}
encoded_data = urllib.urlencode(data_to_post)
# Send encoded data to application-2
url_result = urlfetch.fetch(web_app_url, encoded_data, method='POST')
我们在 Web 应用程序中看到以下错误。
result : 200 content : {"status":"error","defaultMessage":"Error retrieving image from URL or bad URL: data:image/png;base64, <clip> ","name":"Exception","fileName":"Code (Insert image into spreadsheet)","lineNumber":42,"stack":"\tat Code (Insert image into spreadsheet):42 (doPost)\n"}}
你能帮忙吗?
做了这个改变。仍然收到错误的 URL 错误。
dropoff_signature = "ZGF0YTpp<clip>WVhSaA=="
web_app_url = "https://script.google.com/macros/s/A<clip>y/exec"
image_data = "data:image/png;base64," + dropoff_signature
data_to_post = {
'spreadsheetid' : spreadsheet_Id,
'sheetname' : 'Sheet1',
'imageurl' : image_data,
'column' : 5,
'row' : 5
}
# encoded_data = urllib.urlencode(data_to_post)
# Send encoded data to application-2
# url_result = urlfetch.fetch(web_app_url, encoded_data, method='POST')
url_result = urlfetch.fetch(url=web_app_url, payload=json.dumps(data_to_post), method='POST', headers={'Content-type': 'application/json'})
result : 200 content : {"status":"error","defaultMessage":"Error retrieving
image from URL or bad URL:
data:image/png;base64,Z<clip>A==","error":
{"message":"Error retrieving image from URL or bad URL: data:image/png;base64,Z<clip>A==","name":"Exception","fileName":"Code (Insert image into spreadsheet)","lineNumber":42,"stack":"\tat Code (Insert image into spreadsheet):42 (doPost)\n"}}
这是我们正在使用的 Webapp。
function doGet(e) {
return ContentService.createTextOutput("Authorization: Bearer " +
ScriptApp.getOAuthToken())
}
//
// Example curl command to insert an image:
//
// curl -L -d '{ "spreadsheetid": "1xNDWJXOekpBBV2hPseQwCRR8Qs4LcLOcSLDadVqDA0E","sheetname": "Sheet1", "imageurl": "https://www.google.com/images/srpr/logo3w.png", "column": 1, "row": 1 }' \
// -H "Authorization: Bearer <INSERT TOKEN RETURNED FROM GET HERE>" \
// -H 'Content-Type: application/json' \
// https://script.google.com/a/tillerhq.com/macros/s/AKfycbzjFgIrgCfZTvOHImuX54G90VuAgmyfz2cmaKjrsNFrTzcLpNk0/exec
//
var REQUIRED_PARAMS = [
'spreadsheetid', // example: "1xNDWJXOekpBBV2hPseQwCRR8Qs4LcLOcSLDadVqDA0E"
'sheetname', // Case-sensitive; example: "Sheet1"
'imageurl', // Can be an url such as "https://www.google.com/images/srpr/logo3w.png"
// or alternately "data:image/png;base64,iVBOR...<snip>...gg=="
'column', // 1-based (i.e. top left corner is column 1)
'row' // 1-based (i.e. top left corner is row 1)
];
function doPost(e) {
var result = {
status: "ok",
defaultMessage: "Image inserted."
}
try {
var params = (e.postData && e.postData.type == "application/x-www-form-urlencoded") ? e.parameter
: (e.postData && e.postData.type == "application/json") ? JSON.parse(e.postData.contents)
: undefined;
if (!params) throw new Error('Unsupported content-type, must be either application/x-www-form-urlencoded or application/json.');
REQUIRED_PARAMS.forEach(function(requiredParam) {
if (!params[requiredParam]) throw new Error('Missing required parameter ' + requiredParam);
});
SpreadsheetApp.openById(params.spreadsheetid).getSheetByName(params.sheetname).insertImage(params.imageurl, params.column, params.row);
} catch(e) {
console.error(e);
result.status = "error";
result.error = e;
result.defaultMessage = e.message;
}
return ContentService.createTextOutput(JSON.stringify(result))
.setMimeType(ContentService.MimeType.JSON)
}
解决方案
解决方案1:
在您的 Python 应用程序中,您可以使用以下代码使用 Sheets API [1] 使用 IMAGE 公式设置图像。您需要输入电子表格 ID 并更改所需图像的范围。
spreadsheet_id = '[SPREADSHEET-ID]'
range_name = 'D13'
service = build('sheets', 'v4', credentials=creds)
values = [
[
'=IMAGE("https://google.com","google")'
]
]
body = {
'values': values
}
result = service.spreadsheets().values().update(
spreadsheetId=spreadsheet_id, range=range_name,
valueInputOption='USER_ENTERED', body=body).execute()
解决方案2:
相反,如果您想使用 Apps 脚本中的 insertImage 函数 [2],将网格图像插入到表格中,而不是链接到单元格的图像。您可以使用 doPost() 函数部署 Web 应用程序 [3],您可以在其中执行此操作并使用服务帐户凭据从 Python 应用程序调用 Web 应用程序。此外,您需要部署 Web 应用程序以作为“访问 Web 应用程序的用户”执行,因此您从 Web 应用程序执行的所有请求都将使用服务帐户凭据进行。
Python脚本:
from google.oauth2 import service_account
import requests
import json
import google.auth.transport.requests
SCOPES = ['https://www.googleapis.com/auth/drive', 'https://www.googleapis.com/auth/spreadsheets']
SERVICE_ACCOUNT_FILE = 'service_account.json'
credentials = service_account.Credentials.from_service_account_file(
SERVICE_ACCOUNT_FILE, scopes=SCOPES)
delegated_credentials = credentials.with_subject('[USER-EMAIL-TO-IMPERSONATE]')
delegated_credentials.refresh(google.auth.transport.requests.Request())
token = delegated_credentials.token
headers = {'content-type': 'application/json', 'Authorization': 'Bearer ' + token}
url = '[WEB-APP-URL]'
data = {"file": '[blob]'}
response = requests.post(url, data=json.dumps(data), headers=headers)
网络应用脚本:
function doPost(e) {
var ss = SpreadsheetApp.openById('[SPREADSHEET-ID]');
var sheet = ss.getSheets()[0];
var blob = DriveApp.getFileById("[IMAGE-ID]").getBlob();
sheet.insertImage(blob, 4, 14);
return ContentService.createTextOutput("Good");
}
我使用从 Web App 中的 Drive 获得的图像测试了我的代码。您可以跳过该部分并直接从数据负载中的 Python 应用程序发送 blob。
要使用服务帐户,请记住授予对所有所需范围的 API 访问权限,您需要转到 admin.google.com->security->设置->高级设置->管理 API 客户端访问权限并使用服务帐户客户端 ID。
推荐阅读
- batch-file - 如何在运行 VSTS 构建定义时将功能作为环境变量传递给批处理脚本?
- python - 在不合并列的情况下将 xlsx 转换为 csv - Python
- java - 自定义请求参数验证器在 SpringBoot 中不起作用
- rest - RequestMappingHandlerMapping afterPropertiesSet 方法导致二义映射
- angular - 在 ionic 3 工具提示中使用 HTML 代码
- azure-iot-hub - Azure IoT 中心指标数据
- android - 如何在 kotlin 的片段类中更改 ActionBar 标题?
- node.js - 真正删除所有全局 npm 模块
- android - 我想做一个 asynctask 类,asynctask 可以操作字符串、整数、URL
- c++ - 从 QDataStream 读取数据