Google Apps Script API یک روش scripts.run
را ارائه می دهد که از راه دور یک تابع Apps Script مشخص شده را اجرا می کند. می توانید از این روش در یک برنامه فراخوانی برای اجرای یک تابع در یکی از پروژه های اسکریپت خود از راه دور و دریافت پاسخ استفاده کنید.
الزامات
قبل از اینکه یک برنامه فراخوانی بتواند از روش scripts.run
استفاده کند، باید شرایط زیر را برآورده کنید. شما باید :
پروژه اسکریپت را به عنوان یک API اجرایی اجرا کنید. میتوانید پروژهها را در صورت نیاز مستقر، undeploy و مجدداً مستقر کنید.
یک نشانه OAuth با محدوده مناسب برای اجرا ارائه دهید. این نشانه OAuth باید تمام حوزه های مورد استفاده اسکریپت را پوشش دهد، نه فقط مواردی که توسط تابع فراخوانی شده استفاده می شود. لیست کامل دامنه های مجوز را در مرجع روش مشاهده کنید.
اطمینان حاصل کنید که اسکریپت و سرویس گیرنده OAuth2 برنامه تماس گیرنده یک پروژه مشترک Google Cloud را به اشتراک می گذارند. پروژه Cloud باید یک پروژه استاندارد Cloud باشد. پروژه های پیش فرض ایجاد شده برای پروژه های Apps Script کافی نیستند. می توانید از یک پروژه استاندارد جدید Cloud یا یک پروژه موجود استفاده کنید.
روش scripts.run
متد scripts.run
برای اجرا به اطلاعات شناسایی کلیدی نیاز دارد:
- شناسه پروژه اسکریپت .
- نام تابعی که باید اجرا شود.
- لیست پارامترهای مورد نیاز تابع (در صورت وجود).
شما می توانید به صورت اختیاری اسکریپت خود را برای اجرا در حالت توسعه پیکربندی کنید. این حالت با آخرین نسخه ذخیره شده پروژه اسکریپت به جای جدیدترین نسخه اجرا شده اجرا می شود. این کار را با تنظیم boolean devMode
در بدنه درخواست روی true
انجام دهید. فقط صاحب اسکریپت می تواند آن را در حالت توسعه اجرا کند.
مدیریت انواع داده های پارامتر
استفاده از روش Apps Script API scripts.run
معمولاً شامل ارسال داده به Apps Script به عنوان پارامترهای تابع و بازگرداندن داده ها به عنوان مقادیر بازگشتی تابع است. API فقط می تواند مقادیر را با انواع اصلی بگیرد و برگرداند: رشته ها، آرایه ها، اشیاء، اعداد و بولی. اینها شبیه به انواع پایه در جاوا اسکریپت هستند. اشیاء پیچیدهتر Apps Script مانند Document یا Sheet نمیتوانند توسط API به پروژه اسکریپت منتقل شوند یا از آن خارج شوند.
هنگامی که برنامه تماس شما به زبانی با نوع قوی مانند جاوا نوشته می شود، پارامترها را به عنوان لیست یا آرایه ای از اشیاء عمومی متناظر با این انواع اولیه ارسال می کند. در بسیاری از موارد، می توانید تبدیل نوع ساده را به صورت خودکار اعمال کنید. به عنوان مثال، تابعی که پارامتر عددی را می گیرد، می تواند یک شیء جاوا Double
یا Integer
یا Long
به عنوان پارامتر بدون پردازش اضافی داده شود.
هنگامی که API پاسخ تابع را برمیگرداند، اغلب باید قبل از استفاده، مقدار بازگشتی را به نوع صحیح ارسال کنید. در اینجا چند نمونه مبتنی بر جاوا آورده شده است:
- اعدادی که توسط API به یک برنامه جاوا بازگردانده میشوند بهعنوان آبجکتهای
java.math.BigDecimal
میآیند و ممکن است در صورت نیاز بهDoubles
یاint
تبدیل شوند. اگر تابع Apps Script آرایهای از رشتهها را برمیگرداند، یک برنامه جاوا پاسخ را به یک شی
List<String>
میفرستد:List<String> mylist = (List<String>)(op.getResponse().get("result"));
اگر به دنبال برگرداندن آرایه ای از
Bytes
هستید، ممکن است برای شما راحت باشد که آرایه را به عنوان یک رشته base64 در تابع Apps Script رمزگذاری کنید و در عوض آن رشته را برگردانید:return Utilities.base64Encode(myByteArray); // returns a String.
نمونه کدهای مثال زیر روش های تفسیر پاسخ API را نشان می دهد.
رویه عمومی
در زیر روش کلی استفاده از Apps Script API برای اجرای توابع Apps Script توضیح داده شده است:
مرحله 1: پروژه مشترک Cloud را راه اندازی کنید
هم اسکریپت شما و هم برنامه فراخوانی باید پروژه Cloud یکسانی را به اشتراک بگذارند. این پروژه Cloud می تواند یک پروژه موجود یا یک پروژه جدید باشد که برای این منظور ایجاد شده است. هنگامی که یک پروژه Cloud دارید، باید پروژه اسکریپت خود را برای استفاده از آن تغییر دهید .
مرحله 2: اسکریپت را به عنوان یک فایل اجرایی API اجرا کنید
- پروژه Apps Script را با توابعی که می خواهید استفاده کنید باز کنید.
- در بالا سمت چپ، روی Deploy > New Deployment کلیک کنید.
- در گفتگوی باز شده، روی Enable Deployment Types کلیک کنید > API اجرایی
- در منوی کشویی «چه کسی دسترسی دارد»، کاربرانی را انتخاب کنید که مجاز به فراخوانی عملکردهای اسکریپت با استفاده از Apps Script API هستند.
- روی Deploy کلیک کنید.
مرحله 3: برنامه تماس را پیکربندی کنید
برنامه تماس گیرنده باید قبل از استفاده، Apps Script API را فعال کند و اعتبار OAuth را ایجاد کند. برای انجام این کار باید به پروژه Cloud دسترسی داشته باشید.
- پروژه Cloud را که برنامه و اسکریپت تماس شما از آن استفاده می کند، پیکربندی کنید. با مراحل زیر می توانید این کار را انجام دهید:
- پروژه اسکریپت را باز کنید و در سمت چپ روی Overview کلیک کنید .
- در قسمت Project Oauth scopes ، تمام محدوده هایی را که اسکریپت نیاز دارد، ضبط کنید.
در کد برنامه فراخوانی، یک نشانه دسترسی اسکریپت OAuth برای تماس API ایجاد کنید. این نشانه ای نیست که خود API از آن استفاده می کند، بلکه یکی از نشانه هایی است که اسکریپت هنگام اجرا به آن نیاز دارد. باید با استفاده از شناسه مشتری پروژه Cloud و محدودههای اسکریپتی که ضبط کردهاید ساخته شود.
کتابخانههای سرویس گیرنده Google میتوانند در ساختن این توکن و مدیریت OAuth برای برنامه بسیار کمک کنند، معمولاً به شما این امکان را میدهند که در عوض با استفاده از محدودههای اسکریپت یک شیء سطح بالاتر «معتبر» بسازید. برای نمونههایی از ساختن یک شی اعتبارنامه از فهرستی از حوزهها، شروع سریع Apps Script API را ببینید.
مرحله 4: درخواست script.run
را ارسال کنید
هنگامی که برنامه تماس پیکربندی شد، می توانید تماس های scripts.run
برقرار کنید. هر فراخوانی API شامل مراحل زیر است:
- با استفاده از شناسه اسکریپت، نام تابع و هر پارامتر مورد نیاز، یک درخواست API بسازید.
- فراخوانی
scripts.run
را انجام دهید و توکن اسکریپت OAuth را که ساخته اید در هدر قرار دهید (اگر از یک درخواست اولیهPOST
استفاده می کنید) یا در غیر این صورت از یک شی اعتباری که با دامنه های اسکریپت ساخته اید استفاده کنید. - اجازه دهید اسکریپت اجرا شود. اسکریپت ها مجاز هستند تا شش دقیقه زمان اجرا داشته باشند، بنابراین برنامه شما باید این اجازه را بدهد.
- پس از اتمام، تابع اسکریپت ممکن است مقداری را برگرداند که اگر مقدار از نوع پشتیبانی شده باشد، API آن را به برنامه بازگرداند.
میتوانید نمونههایی از فراخوانهای API script.run
را در زیر بیابید.
نمونه های درخواست API
مثالهای زیر نشان میدهند که چگونه میتوان یک درخواست اجرای Apps Script API را به زبانهای مختلف ایجاد کرد، با فراخوانی یک تابع Apps Script برای چاپ فهرستی از پوشهها در فهرست اصلی کاربر. شناسه اسکریپت پروژه Apps Script حاوی تابع اجرا شده باید در جایی که با ENTER_YOUR_SCRIPT_ID_HERE
نشان داده شده است، مشخص شود. نمونهها به کتابخانههای Google API Client برای زبان مربوطه خود متکی هستند.
اسکریپت هدف
تابع در این اسکریپت از Drive API استفاده می کند.
باید Drive API را در پروژه میزبان اسکریپت فعال کنید .
بهعلاوه، برنامههای فراخوانی باید اعتبارنامههای OAuth را ارسال کنند که شامل حوزه Drive زیر است:
-
https://www.googleapis.com/auth/drive
برنامه های کاربردی مثال در اینجا از کتابخانه های سرویس گیرنده Google برای ساخت اشیاء اعتبار برای OAuth با استفاده از این محدوده استفاده می کنند.
/**
* Return the set of folder names contained in the user's root folder as an
* object (with folder IDs as keys).
* @return {Object} A set of folder names keyed by folder ID.
*/
function getFoldersUnderRoot() {
const root = DriveApp.getRootFolder();
const folders = root.getFolders();
const folderSet = {};
while (folders.hasNext()) {
const folder = folders.next();
folderSet[folder.getId()] = folder.getName();
}
return folderSet;
}
جاوا
/**
* Create a HttpRequestInitializer from the given one, except set
* the HTTP read timeout to be longer than the default (to allow
* called scripts time to execute).
*
* @param {HttpRequestInitializer} requestInitializer the initializer
* to copy and adjust; typically a Credential object.
* @return an initializer with an extended read timeout.
*/
private static HttpRequestInitializer setHttpTimeout(
final HttpRequestInitializer requestInitializer) {
return new HttpRequestInitializer() {
@Override
public void initialize(HttpRequest httpRequest) throws IOException {
requestInitializer.initialize(httpRequest);
// This allows the API to call (and avoid timing out on)
// functions that take up to 6 minutes to complete (the maximum
// allowed script run time), plus a little overhead.
httpRequest.setReadTimeout(380000);
}
};
}
/**
* Build and return an authorized Script client service.
*
* @param {Credential} credential an authorized Credential object
* @return an authorized Script client service
*/
public static Script getScriptService() throws IOException {
Credential credential = authorize();
return new Script.Builder(
HTTP_TRANSPORT, JSON_FACTORY, setHttpTimeout(credential))
.setApplicationName(APPLICATION_NAME)
.build();
}
/**
* Interpret an error response returned by the API and return a String
* summary.
*
* @param {Operation} op the Operation returning an error response
* @return summary of error response, or null if Operation returned no
* error
*/
public static String getScriptError(Operation op) {
if (op.getError() == null) {
return null;
}
// Extract the first (and only) set of error details and cast as a Map.
// The values of this map are the script's 'errorMessage' and
// 'errorType', and an array of stack trace elements (which also need to
// be cast as Maps).
Map<String, Object> detail = op.getError().getDetails().get(0);
List<Map<String, Object>> stacktrace =
(List<Map<String, Object>>) detail.get("scriptStackTraceElements");
java.lang.StringBuilder sb =
new StringBuilder("\nScript error message: ");
sb.append(detail.get("errorMessage"));
sb.append("\nScript error type: ");
sb.append(detail.get("errorType"));
if (stacktrace != null) {
// There may not be a stacktrace if the script didn't start
// executing.
sb.append("\nScript error stacktrace:");
for (Map<String, Object> elem : stacktrace) {
sb.append("\n ");
sb.append(elem.get("function"));
sb.append(":");
sb.append(elem.get("lineNumber"));
}
}
sb.append("\n");
return sb.toString();
}
public static void main(String[] args) throws IOException {
// ID of the script to call. Acquire this from the Apps Script editor,
// under Publish > Deploy as API executable.
String scriptId = "ENTER_YOUR_SCRIPT_ID_HERE";
Script service = getScriptService();
// Create an execution request object.
ExecutionRequest request = new ExecutionRequest()
.setFunction("getFoldersUnderRoot");
try {
// Make the API request.
Operation op =
service.scripts().run(scriptId, request).execute();
// Print results of request.
if (op.getError() != null) {
// The API executed, but the script returned an error.
System.out.println(getScriptError(op));
} else {
// The result provided by the API needs to be cast into
// the correct type, based upon what types the Apps
// Script function returns. Here, the function returns
// an Apps Script Object with String keys and values,
// so must be cast into a Java Map (folderSet).
Map<String, String> folderSet =
(Map<String, String>) (op.getResponse().get("result"));
if (folderSet.size() == 0) {
System.out.println("No folders returned!");
} else {
System.out.println("Folders under your root folder:");
for (String id : folderSet.keySet()) {
System.out.printf(
"\t%s (%s)\n", folderSet.get(id), id);
}
}
}
} catch (GoogleJsonResponseException e) {
// The API encountered a problem before the script was called.
e.printStackTrace(System.out);
}
}
جاوا اسکریپت
/**
* Load the API and make an API call. Display the results on the screen.
*/
function callScriptFunction() {
const scriptId = '<ENTER_YOUR_SCRIPT_ID_HERE>';
// Call the Apps Script API run method
// 'scriptId' is the URL parameter that states what script to run
// 'resource' describes the run request body (with the function name
// to execute)
try {
gapi.client.script.scripts.run({
'scriptId': scriptId,
'resource': {
'function': 'getFoldersUnderRoot',
},
}).then(function(resp) {
const result = resp.result;
if (result.error && result.error.status) {
// The API encountered a problem before the script
// started executing.
appendPre('Error calling API:');
appendPre(JSON.stringify(result, null, 2));
} else if (result.error) {
// The API executed, but the script returned an error.
// Extract the first (and only) set of error details.
// The values of this object are the script's 'errorMessage' and
// 'errorType', and an array of stack trace elements.
const error = result.error.details[0];
appendPre('Script error message: ' + error.errorMessage);
if (error.scriptStackTraceElements) {
// There may not be a stacktrace if the script didn't start
// executing.
appendPre('Script error stacktrace:');
for (let i = 0; i < error.scriptStackTraceElements.length; i++) {
const trace = error.scriptStackTraceElements[i];
appendPre('\t' + trace.function + ':' + trace.lineNumber);
}
}
} else {
// The structure of the result will depend upon what the Apps
// Script function returns. Here, the function returns an Apps
// Script Object with String keys and values, and so the result
// is treated as a JavaScript object (folderSet).
const folderSet = result.response.result;
if (Object.keys(folderSet).length == 0) {
appendPre('No folders returned!');
} else {
appendPre('Folders under your root folder:');
Object.keys(folderSet).forEach(function(id) {
appendPre('\t' + folderSet[id] + ' (' + id + ')');
});
}
}
});
} catch (err) {
document.getElementById('content').innerText = err.message;
return;
}
}
Node.js
/**
* Call an Apps Script function to list the folders in the user's root Drive
* folder.
*
*/
async function callAppsScript() {
const scriptId = '1xGOh6wCm7hlIVSVPKm0y_dL-YqetspS5DEVmMzaxd_6AAvI-_u8DSgBT';
const {GoogleAuth} = require('google-auth-library');
const {google} = require('googleapis');
// Get credentials and build service
// TODO (developer) - Use appropriate auth mechanism for your app
const auth = new GoogleAuth({
scopes: 'https://www.googleapis.com/auth/drive',
});
const script = google.script({version: 'v1', auth});
try {
// Make the API request. The request object is included here as 'resource'.
const resp = await script.scripts.run({
auth: auth,
resource: {
function: 'getFoldersUnderRoot',
},
scriptId: scriptId,
});
if (resp.error) {
// The API executed, but the script returned an error.
// Extract the first (and only) set of error details. The values of this
// object are the script's 'errorMessage' and 'errorType', and an array
// of stack trace elements.
const error = resp.error.details[0];
console.log('Script error message: ' + error.errorMessage);
console.log('Script error stacktrace:');
if (error.scriptStackTraceElements) {
// There may not be a stacktrace if the script didn't start executing.
for (let i = 0; i < error.scriptStackTraceElements.length; i++) {
const trace = error.scriptStackTraceElements[i];
console.log('\t%s: %s', trace.function, trace.lineNumber);
}
}
} else {
// The structure of the result will depend upon what the Apps Script
// function returns. Here, the function returns an Apps Script Object
// with String keys and values, and so the result is treated as a
// Node.js object (folderSet).
const folderSet = resp.response.result;
if (Object.keys(folderSet).length == 0) {
console.log('No folders returned!');
} else {
console.log('Folders under your root folder:');
Object.keys(folderSet).forEach(function(id) {
console.log('\t%s (%s)', folderSet[id], id);
});
}
}
} catch (err) {
// TODO(developer) - Handle error
throw err;
}
}
پایتون
import google.auth
from googleapiclient.discovery import build
from googleapiclient.errors import HttpError
def main():
"""Runs the sample."""
# pylint: disable=maybe-no-member
script_id = "1VFBDoJFy6yb9z7-luOwRv3fCmeNOzILPnR4QVmR0bGJ7gQ3QMPpCW-yt"
creds, _ = google.auth.default()
service = build("script", "v1", credentials=creds)
# Create an execution request object.
request = {"function": "getFoldersUnderRoot"}
try:
# Make the API request.
response = service.scripts().run(scriptId=script_id, body=request).execute()
if "error" in response:
# The API executed, but the script returned an error.
# Extract the first (and only) set of error details. The values of
# this object are the script's 'errorMessage' and 'errorType', and
# a list of stack trace elements.
error = response["error"]["details"][0]
print(f"Script error message: {0}.{format(error['errorMessage'])}")
if "scriptStackTraceElements" in error:
# There may not be a stacktrace if the script didn't start
# executing.
print("Script error stacktrace:")
for trace in error["scriptStackTraceElements"]:
print(f"\t{0}: {1}.{format(trace['function'], trace['lineNumber'])}")
else:
# The structure of the result depends upon what the Apps Script
# function returns. Here, the function returns an Apps Script
# Object with String keys and values, and so the result is
# treated as a Python dictionary (folder_set).
folder_set = response["response"].get("result", {})
if not folder_set:
print("No folders returned!")
else:
print("Folders under your root folder:")
for folder_id, folder in folder_set.items():
print(f"\t{0} ({1}).{format(folder, folder_id)}")
except HttpError as error:
# The API encountered a problem before the script started executing.
print(f"An error occurred: {error}")
print(error.content)
if __name__ == "__main__":
main()
محدودیت ها
Apps Script API چندین محدودیت دارد:
یک پروژه Cloud مشترک اسکریپت در حال فراخوانی و برنامه فراخوانی باید یک پروژه Cloud را به اشتراک بگذارند. پروژه Cloud باید یک پروژه استاندارد Cloud باشد. پروژه های پیش فرض ایجاد شده برای پروژه های Apps Script کافی نیستند. پروژه استاندارد Cloud می تواند یک پروژه جدید یا موجود باشد.
پارامتر اصلی و انواع بازگشت API نمیتواند اشیاء خاص Apps Script (مانند اسناد ، حبابها ، تقویمها ، فایلهای Drive ، و غیره) را به برنامه ارسال یا برگرداند. فقط انواع اولیه مانند رشته ها، آرایه ها، اشیاء، اعداد و بولی ها را می توان ارسال و برگرداند.
دامنه های OAuth . API فقط می تواند اسکریپت هایی را اجرا کند که حداقل یک محدوده مورد نیاز دارند. این بدان معناست که شما نمی توانید از API برای فراخوانی اسکریپتی استفاده کنید که به مجوز یک یا چند سرویس نیاز ندارد.
هیچ راهاندازی وجود ندارد . API نمیتواند محرکهای Apps Script را ایجاد کند.