Menjalankan Fungsi dengan Apps Script API

Google Apps Script API menyediakan metode scripts.run yang mengeksekusi fungsi Apps Script yang ditentukan dari jarak jauh. Anda dapat menggunakan metode ini di aplikasi panggilan untuk menjalankan fungsi di salah satu project skrip dari jarak jauh dan menerima respons.

Persyaratan

Anda harus memenuhi persyaratan berikut sebelum aplikasi panggilan dapat menggunakan metode scripts.run. Anda harus:

  • Deploy project skrip sebagai API yang dapat dieksekusi. Anda dapat men-deploy, membatalkan deployment, dan men-deploy ulang project sesuai kebutuhan.

  • Berikan token OAuth dengan cakupan yang benar untuk eksekusi. Token OAuth ini harus mencakup semua cakupan yang digunakan oleh skrip, bukan hanya yang digunakan oleh fungsi yang dipanggil. Lihat daftar lengkap cakupan otorisasi di referensi metode.

  • Pastikan skrip dan klien OAuth2 aplikasi panggilan menggunakan project Google Cloud yang sama. Project Cloud harus berupa project Cloud standar; project default yang dibuat untuk project Apps Script tidak memadai. Anda dapat menggunakan project Cloud standar baru atau yang sudah ada.

  • Aktifkan Google Apps Script API di project Cloud.

Metode scripts.run

Metode scripts.run memerlukan informasi pengidentifikasi kunci agar dapat dijalankan:

Secara opsional, Anda dapat mengonfigurasi skrip untuk dijalankan dalam mode pengembangan. Mode ini dijalankan dengan versi project skrip yang baru disimpan, bukan versi yang paling baru di-deploy. Lakukan hal ini dengan menetapkan boolean devMode dalam isi permintaan ke true. Hanya pemilik skrip yang dapat mengeksekusinya dalam mode pengembangan.

Menangani jenis data parameter

Penggunaan metode scripts.run Apps Script API biasanya melibatkan pengiriman data ke Apps Script sebagai parameter fungsi dan mengembalikan data sebagai nilai yang ditampilkan fungsi. API hanya dapat mengambil dan menampilkan nilai dengan jenis dasar: string, array, objek, angka, dan boolean. Jenis ini mirip dengan jenis dasar di JavaScript. Objek Apps Script yang lebih kompleks seperti Dokumen atau Sheet tidak dapat diteruskan ke atau dari project skrip oleh API.

Jika aplikasi panggilan Anda ditulis dalam bahasa berjenis kuat seperti Java, aplikasi tersebut akan meneruskan parameter sebagai daftar atau array objek generik yang sesuai dengan jenis dasar ini. Dalam banyak kasus, Anda dapat menerapkan konversi jenis sederhana secara otomatis. Misalnya, fungsi yang menggunakan parameter angka dapat diberi objek Double atau Integer atau Long Java sebagai parameter tanpa penanganan tambahan.

Saat API menampilkan respons fungsi, Anda sering kali perlu mentransmisikan nilai yang ditampilkan ke jenis yang benar sebelum dapat digunakan. Berikut beberapa contoh berbasis Java:

  • Angka yang ditampilkan oleh API ke aplikasi Java diterima sebagai objek java.math.BigDecimal, dan mungkin perlu dikonversi ke jenis Doubles atau int sesuai kebutuhan.
  • Jika fungsi Apps Script menampilkan array string, aplikasi Java akan mentransmisikan respons ke objek List<String>:

    List<String> mylist = (List<String>)(op.getResponse().get("result"));
    
  • Jika ingin menampilkan array Bytes, Anda dapat mengenkode array sebagai String base64 dalam fungsi Apps Script dan menampilkan String tersebut:

    return Utilities.base64Encode(myByteArray); // returns a String.
    

Contoh kode contoh di bawah menggambarkan cara menafsirkan respons API.

Prosedur umum

Berikut ini penjelasan prosedur umum penggunaan Apps Script API untuk menjalankan fungsi Apps Script:

Langkah 1: Menyiapkan project Cloud umum

Skrip Anda dan aplikasi panggilan harus menggunakan project Cloud yang sama. Project Cloud ini dapat berupa project yang sudah ada atau project baru yang dibuat untuk tujuan ini. Setelah memiliki project Cloud, Anda harus mengalihkan project skrip untuk menggunakannya.

Langkah 2: Deploy skrip sebagai API yang dapat dieksekusi

  1. Buka project Apps Script dengan fungsi yang ingin Anda gunakan.
  2. Di kanan atas, klik Deploy > New Deployment.
  3. Pada dialog yang terbuka, klik Enable deployment types > API Executable.
  4. Di menu drop-down "Siapa yang memiliki akses", pilih pengguna yang diizinkan untuk memanggil fungsi skrip menggunakan Apps Script API.
  5. Klik Deploy.

Langkah 3: Konfigurasikan aplikasi panggilan

Aplikasi panggilan harus mengaktifkan Apps Script API dan membuat Crendential OAuth sebelum dapat digunakan. Anda harus memiliki akses ke project Cloud untuk melakukannya.

  1. Konfigurasikan project Cloud yang digunakan oleh aplikasi dan skrip panggilan Anda. Anda dapat melakukannya dengan langkah-langkah berikut:
    1. Aktifkan Apps Script API di project Cloud.
    2. Konfigurasi layar izin OAuth.
    3. Buat kredensial OAuth.
  2. Buka project skrip, lalu klik Overview di sebelah kiri.
  3. Pada bagian Cakupan Project Oauth, catat semua cakupan yang diperlukan skrip.
  4. Pada kode aplikasi panggilan, buat token akses OAuth skrip untuk panggilan API. Ini bukan token yang digunakan API itu sendiri, melainkan token yang diperlukan skrip saat dieksekusi. Project ini harus dibuat menggunakan client ID project Cloud dan cakupan skrip yang Anda rekam.

    Library klien Google dapat sangat membantu dalam membangun token ini dan menangani OAuth untuk aplikasi, biasanya memungkinkan Anda untuk membuat objek "kredensial" tingkat tinggi menggunakan cakupan skrip. Baca panduan memulai Apps Script API untuk mengetahui contoh cara membuat objek kredensial dari daftar cakupan.

Langkah 4: Buat permintaan script.run

Setelah aplikasi panggilan dikonfigurasi, Anda dapat melakukan panggilan scripts.run. Setiap panggilan API terdiri dari langkah-langkah berikut:

  1. Buat permintaan API menggunakan ID skrip, nama fungsi, dan parameter yang diperlukan.
  2. Lakukan panggilan scripts.run dan sertakan token OAuth skrip yang Anda build di header (jika menggunakan permintaan POST dasar) atau gunakan objek kredensial yang Anda buat dengan cakupan skrip.
  3. Izinkan skrip menyelesaikan eksekusi. Skrip diizinkan membutuhkan waktu eksekusi hingga enam menit, jadi aplikasi Anda harus mengizinkannya.
  4. Setelah selesai, fungsi skrip dapat menampilkan nilai, yang akan dikirimkan kembali oleh API ke aplikasi jika nilai tersebut merupakan jenis yang didukung.

Anda dapat menemukan contoh panggilan API script.run di bawah.

Contoh permintaan API

Contoh berikut menunjukkan cara membuat permintaan eksekusi Apps Script API dalam berbagai bahasa, dengan memanggil fungsi Apps Script untuk mencetak daftar folder di direktori utama pengguna. ID skrip untuk project Apps Script yang berisi fungsi yang dieksekusi harus ditentukan, jika ditunjukkan dengan ENTER_YOUR_SCRIPT_ID_HERE. Contoh ini mengandalkan library Klien Google API untuk bahasanya masing-masing.

Skrip Target

Fungsi dalam skrip ini menggunakan Drive API.

Anda harus mengaktifkan Drive API dalam project yang menghosting skrip.

Selain itu, aplikasi yang dipanggil harus mengirimkan kredensial OAuth yang mencakup cakupan Drive berikut:

  • https://www.googleapis.com/auth/drive

Contoh aplikasi di sini menggunakan library klien Google guna mem-build objek kredensial untuk OAuth menggunakan cakupan ini.

/**
 * 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;
}

Java


/**
 * 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);
  }
}

JavaScript

/**
 * 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;
  }
}

Python

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()

Batasan

Apps Script API memiliki beberapa batasan:

  1. Project Cloud umum. Skrip yang dipanggil dan aplikasi panggilan harus menggunakan project Cloud yang sama. Project Cloud harus berupa project Cloud standar; project default yang dibuat untuk project Apps Script tidak memadai. Project Cloud standar dapat berupa project baru atau yang sudah ada.

  2. Parameter dasar dan jenis nilai yang ditampilkan. API tidak dapat meneruskan atau menampilkan objek khusus Apps Script (seperti Documents, Blob, Kalender, Drive Files, dll.) ke aplikasi. Hanya jenis dasar seperti string, array, objek, angka, dan boolean yang dapat diteruskan dan ditampilkan.

  3. Cakupan OAuth. API hanya dapat mengeksekusi skrip yang memiliki setidaknya satu cakupan yang diperlukan. Artinya, Anda tidak dapat menggunakan API untuk memanggil skrip yang tidak memerlukan otorisasi dari satu atau beberapa layanan.

  4. Tidak ada pemicu.API tidak dapat membuat pemicu Apps Script.