0

我有一个脚本,可以将我的所有工作表导出到单独的 PDF 文档到名为 Invoices 的文件夹中,但是它正在创建 6 个文档,之后我收到下面的错误代码,这也已在 Gsuite 帐户上尝试过,我'我得到同样的错误:

异常:对https://docs.google.com的请求失败,返回代码 429。截断的服务器响应:<meta name="viewport" c...(使用 muteHttpExceptions 选项检查完整响应)

function savePDFs( optSSId, optSheetId ) {
  var ss = (optSSId) ? SpreadsheetApp.openById(optSSId) : SpreadsheetApp.getActiveSpreadsheet();
  var url = ss.getUrl().replace(/edit$/,'');
  var parents = DriveApp.getFileById(ss.getId()).getParents();
  
  var folders = DriveApp.getFoldersByName('Invoices');  
  var folder = folders.hasNext() ? folders.next() : parents.next(); 

  var sheets = ss.getSheets();
  for (var i=0; i<sheets.length; i++) {
    var sheet = sheets[i];
    if (optSheetId && optSheetId !== sheet.getSheetId()) continue; 
    var url_ext = 'export?exportFormat=pdf&format=pdf'   //export as pdf
      + '&gid=' + sheet.getSheetId()   //the sheet's Id
      // following parameters are optional...
      + '&size=letter'      // paper size
      + '&portrait=true'    // orientation, false for landscape
      + '&fitw=true'        // fit to width, false for actual size
      + '&sheetnames=false&printtitle=false&pagenumbers=false'  //hide optional headers and footers
      + '&gridlines=false'  // hide gridlines
      + '&fzr=false';       // do not repeat row headers (frozen rows) on each page
    var options = {headers: {'Authorization': 'Bearer ' +  ScriptApp.getOAuthToken()}}
    var response = UrlFetchApp.fetch(url + url_ext, options);
    var valor = sheet.getRange('D5').getValue();  // Modified
    var blob = response.getBlob().setName(valor + '.pdf');
    folder.createFile(blob);
  }
}

有什么想法吗?

4

2 回答 2

3

如果执行时间是这里的一个重要因素,请参阅我很长的答案以及更复杂的解决方法

更简单的解决方法:

在上面链接中我的答案的底部,您将看到对我能够延迟函数的递归部分以在 8 秒内无限期地从工作表中提取任意数量的 PDF 的秒数的引用,所以所有您需要做的是在 for 循环末尾添加以下代码:

Utilities.sleep(8000)

整个代码将是:

function savePDFs( optSSId, optSheetId ) {
  var ss = (optSSId) ? SpreadsheetApp.openById(optSSId) : SpreadsheetApp.getActiveSpreadsheet();
  var url = ss.getUrl().replace(/edit$/,'');
  var parents = DriveApp.getFileById(ss.getId()).getParents();
  
  var folders = DriveApp.getFoldersByName('Invoices');  
  var folder = folders.hasNext() ? folders.next() : parents.next(); 

  var sheets = ss.getSheets();
  for (var i=0; i<sheets.length; i++) {
    var sheet = sheets[i];
    if (optSheetId && optSheetId !== sheet.getSheetId()) continue; 
    var url_ext = 'export?exportFormat=pdf&format=pdf'   //export as pdf
      + '&gid=' + sheet.getSheetId()   //the sheet's Id
      // following parameters are optional...
      + '&size=letter'      // paper size
      + '&portrait=true'    // orientation, false for landscape
      + '&fitw=true'        // fit to width, false for actual size
      + '&sheetnames=false&printtitle=false&pagenumbers=false'  //hide optional headers and footers
      + '&gridlines=false'  // hide gridlines
      + '&fzr=false';       // do not repeat row headers (frozen rows) on each page
    var options = {headers: {'Authorization': 'Bearer ' +  ScriptApp.getOAuthToken()}}
    var response = UrlFetchApp.fetch(url + url_ext, options);
    var valor = sheet.getRange('D5').getValue();  // Modified
    var blob = response.getBlob().setName(valor + '.pdf');
    folder.createFile(blob);
    Utilities.sleep(8000);
  }
}
于 2020-07-11T04:43:31.227 回答
1

我发现另一种解决方法在我的情况下效果很好,它比Utilities.sleep方法快得多。

我最终做的是使用 try and catch 结构。在每条语句下,我添加了相同的函数来生成 pdf。像这样的东西:

    try{
        var pdf = generatePDF(sheet);
      } catch(e){
        var pdf = generatePDF(sheet);
      }

所以最终发生的情况是,如果您收到 429 错误,代码将再次尝试运行 generatePDF 函数,并且该过程将花费足够的时间,这样您就不会再次收到太多的请求消息。

于 2021-01-28T14:31:06.703 回答