Quicken google apps script so I can return success response within Shopify 5 second limit

前端 未结 1 2036
难免孤独
难免孤独 2021-01-27 03:33

I have this google apps script I wrote that I\'m using as a web app as an endpoint for a Shopify webhook.

The issue I\'m having is that Shopify has a 5 second limit to re

相关标签:
1条回答
  • 2021-01-27 04:32

    It's slower than using the PropertiesService, but I like using Sheets as a queue. (I use this with services that require responses within 3 seconds.) Not only is it easier to work with, but I've actually had issues with using Properties that are addressed with the appendRow() method:

    Appends a row to the spreadsheet. This operation is atomic; it prevents issues where a user asks for the last row, and then writes to that row, and an intervening mutation occurs between getting the last row and writing to it.

    When you receive the POST data, simply add it to the queue and terminate. Apps Script will send a 200 success response, so Shopify shouldn't send duplicate requests.

    Then have a time-driven trigger that runs a processQueue() function at the interval of your choice.

    function doPost(e) {
      const queue = new Queue(SpreadsheetApp.getActive().getId(), "Unprocessed", "Processed");
      queue.append(e.postData.contents, skipRefresh = true);
    }
    
    function processQueue() {
      const queue = new Queue(SpreadsheetApp.getActive().getId(), "Unprocessed", "Processed");
      while (queue.hasNext()) {
        try {
          const data = JSON.parse(queue.next());
          
          doSomethingWithShopifyData(data); // Process your data
          
          queue.moveToProcessed();
        } catch (error) {
          console.error(error);
          queue.skip();
        }
      }
    }
    
    function doSomethingWithShopifyData(data) { /* your existing code, but with appropriate modifications */ }
    

    Here's the class I use to abstract the spreadsheet into a queue. I have it setup to preserve all of the data moving it from an unprocessed to a processed sheet. You may prefer to simply delete the data once processed.

    /**
     * A spreadsheet is used as a makeshift queue for processing requests asynchronously.
     * @param {string} spreadsheetId - The ID of the spreadsheet to be used for the queue.
     * @param {string} unprocessedSheetName - The name of the sheet to be used for storing unprocessed items.
     * @param {string} processedSheetName - The name of the sheet to be used for storing processed items.
     */
    class Queue {
      constructor(spreadsheetId, unprocessedSheetName, processedSheetName) {
        this.index = 0;
        this.row = 1;
        this.spreadsheet = SpreadsheetApp.openById(spreadsheetId);
        this.unprocessedSheet = this.spreadsheet.getSheetByName(unprocessedSheetName);
        this.processedSheet = this.spreadsheet.getSheetByName(processedSheetName);
      }
      
      /**
       * Determines whether calling next() will return an item.
       * @returns {boolean}
       */
      hasNext() {
        if (this.unprocessedValues == null) { this.refreshUnprocessedValues(); }
        return this.index < this.unprocessedValues.length;
      }
      
      /**
       * Get and save the unprocessed element values to the queue.
       * @returns {object[]}
       */
      refreshUnprocessedValues() {
        try {
          const range =this.unprocessedSheet.getRange(1, 1, this.unprocessedSheet.getLastRow());
          this.unprocessedValues = range.getValues();
        } catch (error) {
          this.unprocessedValues = [];
        }
        return this.unprocessedValues;
      }
      
      /**
       * Get the next element from the queue.
       * @returns {string}
       */
      next() {
        return this.unprocessedValues[this.index++][0];
      }
      
      /**
       * Skip the current queue element. Update row property to maintain synchronization
       * with the spreadsheet range.
       */
      skip() {
        this.row++;
      }
      
      /**
       * Add new data to the queue for processing.
       * @param {string} data - The data to add to the queue.
       * @param {boolean} [skipRefresh] - Default: false. If true, will skip refreshing the queue values.
       */
      append(data, skipRefresh) {
        this.unprocessedSheet.appendRow([data]);
        if (!skipRefresh) { this.refreshUnprocessedValues(); }
      }
      
      /**
       * Move a payload out of the unprocessed sheet and into the processed sheet. Uses the payload
       * at the top of the unprocessed range.
       */
      moveToProcessed() {
        const cell = this.unprocessedSheet.getRange(this.row, 1);
        
        // Move into processed sheet
        this.processedSheet.appendRow([cell.getValue()]);
        
        // Move out of unprocessed sheet
        cell.deleteCells(SpreadsheetApp.Dimension.ROWS);
      }
    }
    
    0 讨论(0)
提交回复
热议问题