Page Summary
-
Script projects accessing user data require authorization, initiating an authorization flow when a script runs for the first time.
-
OAuth scopes define the specific permissions a script needs, such as reading emails or creating calendar events.
-
For most scripts, Apps Script automatically detects necessary scopes, but for published applications like add-ons, you should explicitly set the narrowest scopes possible in the manifest file.
-
Granular OAuth permissions allow users to authorize specific scopes, and scripts should be designed to handle these permissions using
requireScopesorgetAuthorizationInfomethods. -
Scripts using sensitive or restricted OAuth scopes, especially for publicly published applications, may require OAuth client verification and adherence to additional data policies.
Users must authorize script projects that access their data or act on their behalf. When a user runs a script that requires authorization for the first time, the UI presents a prompt to start the authorization flow.
During this flow, the UI tells users which permissions the script requests. For example, a script might request permission to read email messages or create calendar events. The script project defines these individual permissions as OAuth scopes.
For most scripts, Apps Script automatically detects required scopes. You can view the scopes a script uses at any time. You can also set scopes explicitly in your manifest using URL strings. Published applications, such as add-ons, must use the narrowest scopes possible.
During the authorization flow, Apps Script presents
human-readable descriptions of the required scopes. For example, if your script
needs read-only access to spreadsheets, the manifest might include the scope
https://www.googleapis.com/auth/spreadsheets.readonly. The authorization
prompt asks the user to "View your Google Spreadsheets".
Some scopes include others. For example, authorized access to
https://www.googleapis.com/auth/spreadsheets allows read and write access to
spreadsheets.
For some surfaces, such as the Apps Script IDE, users see the granular OAuth consent screen. This screen lets users select specific permissions to grant rather than granting all permissions at once. Design your script to handle granular OAuth permissions.
View scopes
To see the scopes your script project requires:
- Open the script project.
- At the left, click Overview .
- View the scopes under Project OAuth Scopes.
Set explicit scopes
Apps Script automatically determines required scopes by scanning the code for function calls. While this is sufficient for most scripts, you must exercise more direct control for published add-ons, web apps, Chat apps, and calls to the Chat API.
Apps Script sometimes automatically assigns permissive scopes. This can mean your script asks users for more access than it needs. For published scripts, replace broad scopes with a limited set that covers the script's needs.
You can explicitly set the scopes your script project uses by editing its
manifest file. The oauthScopes manifest
field is an array of scopes used by the project. To set your project's scopes:
- Open the script project.
- At the left, click Project Settings .
- Select the Show "appsscript.json" manifest file in editor checkbox.
- At the left, click Editor .
- At the left, click the
appsscript.jsonfile. - Locate the top-level field labeled
oauthScopes. If it's not present, you can add it. - Replace the contents of the
oauthScopesarray with the scopes you want the project to use. For example:{ ... "oauthScopes": [ "https://www.googleapis.com/auth/spreadsheets.readonly", "https://www.googleapis.com/auth/userinfo.email" ], ... } - At the top, click Save .
Handle granular OAuth permissions
The granular OAuth consent screen first launched to the Apps Script IDE for users executing a script directly. The consent screen is progressively released to other surfaces, such as macros, triggers, and add-ons, over time. For more information, see Granular OAuth consent in Google Apps Script IDE executions.
The granular OAuth consent screen lets users specify which individual OAuth scopes to authorize. This gives users fine-grained control over what account data they share with each script. For example, if a script requests email and calendar scopes, users can choose to grant Calendar permission but not Gmail.
The following sections describe how to handle granular OAuth permissions.
Automatically require permission for necessary scopes
If an execution flow requires specific scopes, you can require users to grant those permissions. Your script can check for permissions and automatically ask for them if missing.
The following methods from the
ScriptApp class validate
permissions and render the authorization prompt:
requireScopes(authMode, oAuthScopes): Use this method for flows that rely on specific scopes.requireAllScopes(authMode): Use this method if an execution flow relies on all project scopes.
Example
The following example shows how to call requireScopes() and
requireAllScopes(). The script uses scopes for Gmail,
Sheets, and Calendar. The sendEmail() function
requires only the scopes for Gmail and Sheets
while the createEventSendEmail() function requires all scopes used by the
script.
// This function requires the Gmail and Sheets scopes.
function sendEmail() {
// Validates that the user has granted permission for the Gmail and Sheets scopes.
// If not, the execution ends and prompts the user for authorization.
ScriptApp.requireScopes(ScriptApp.AuthMode.FULL, [
'https://mail.google.com/',
'https://www.googleapis.com/auth/spreadsheets'
]);
// Sends an email.
GmailApp.sendEmail("dana@example.com", "Subject", "Body");
Logger.log("Email sent successfully!");
// Opens a spreadsheet and sheet to track the sent email.
const ss = SpreadsheetApp.openById("abc1234567");
const sheet = ss.getSheetByName("Email Tracker")
// Gets the last row of the sheet.
const lastRow = sheet.getLastRow();
// Adds "Sent" to column E of the last row of the spreadsheet.
sheet.getRange(lastRow, 5).setValue("Sent");
Logger.log("Sheet updated successfully!");
}
// This function requires all scopes used by the script (Gmail,
// Calendar, and Sheets).
function createEventSendEmail() {
// Validates that the user has granted permission for all scopes used by the
// script. If not, the execution ends and prompts the user for authorization.
ScriptApp.requireAllScopes(ScriptApp.AuthMode.FULL);
// Creates an event.
CalendarApp.getDefaultCalendar().createEvent(
"Meeting",
new Date("November 28, 2024 10:00:00"),
new Date("November 28, 2024 11:00:00")
);
Logger.log("Calendar event created successfully!");
// Sends an email.
GmailApp.sendEmail("dana@example.com", "Subject 2", "Body 2");
Logger.log("Email sent successfully!");
// Opens a spreadsheet and sheet to track the created meeting and sent email.
const ss = SpreadsheetApp.openById("abc1234567");
const sheet = ss.getSheetByName("Email and Meeting Tracker")
// Gets the last row
const lastRow = sheet.getLastRow();
// Adds "Sent" to column E of the last row
sheet.getRange(lastRow, 5).setValue("Sent");
// Adds "Meeting created" to column F of the last row
sheet.getRange(lastRow, 6).setValue("Meeting created");
Logger.log("Sheet updated successfully!");
}
Create a custom experience for missing scopes
You can retrieve the permission status of users and design custom experiences. For example, you might disable features that require missing permissions or display a dialog explaining the requirement. The following methods retrieve an object with the user's permission information that includes which scopes the user has authorized and a URL to request any missing scopes:
getAuthorizationInfo(authMode, oAuthScopes): Checks permission status for specific scopes.getAuthorizationInfo(authMode): Checks permission status for all project scopes.
To get the permission details from the authorization info object, such as the
list of authorized scopes and the URL to request missing permissions, use the
methods from the
AuthorizationInfo class.
Example
The following example shows how to use getAuthorizationInfo() to skip features
where users haven't granted the required scopes. This allows the rest of the
execution flow to continue without prompting for authorization of the missing
scopes.
// This function uses the Gmail scope and skips the email
// capabilities if the scope for Gmail hasn't been granted.
function myFunction() {
const authInfo = ScriptApp.getAuthorizationInfo(ScriptApp.AuthMode.FULL, ['https://mail.google.com/']);
if (authInfo.getAuthorizationStatus() === ScriptApp.AuthorizationStatus.NOT_REQUIRED) {
GmailApp.sendEmail("dana@example.com", "Subject", "Body");
Logger.log("Email sent successfully!");
} else {
const scopesGranted = ScriptApp.getAuthorizationInfo(ScriptApp.AuthMode.FULL).getAuthorizedScopes();
console.warn(`Authorized scopes: ${scopesGranted} not enough to send mail, skipping.`);
}
// Continue the rest of the execution flow...
}
Ensure that trigger executions have permissions
Functions associated with triggers run automatically, and users might not be
present to provide permissions. We recommend that you use
requireScopes(authMode, oAuthScopes)
before installing a trigger. This prompts the user for missing permissions
and doesn't allow the installation of the trigger without them.
Example
// This function requires scope Sheets.
function trackFormSubmissions(e){
// Opens a spreadsheet to track the sent email.
const ss = SpreadsheetApp.openById("abc1234567");
const sheet = ss.getSheetByName("Submission Tracker")
// Gets the last row of the sheet.
const lastRow = sheet.getLastRow();
// Adds email address of user that submitted the form
// to column E of the last row of the spreadsheet.
sheet.getRange(lastRow, 5).setValue(e.name);
Logger.log("Sheet updated successfully!");
}
function installTrigger(){
// Validates that the user has granted permissions for trigger
// installation and execution. If not, trigger doesn't get
// installed and prompts the user for authorization.
ScriptApp.requireScopes(ScriptApp.AuthMode.FULL, [
'https://www.googleapis.com/auth/script.scriptapp',
'https://www.googleapis.com/auth/spreadsheets',
'https://www.googleapis.com/auth/forms.currentonly'
]);
ScriptApp.newTrigger('trackFormSubmission')
.forForm(FormApp.getActiveForm())
.onFormSubmit()
.create();
}
OAuth verification
Certain OAuth scopes are sensitive because they allow access to Google User Data. If your script project uses scopes that allow access to user data, the project must go through OAuth client verification before you can publish it publicly as a web app or add-on. For more information, see the following guides:
- OAuth client verification for Apps Script
- Unverified apps
- OAuth verification FAQ
- Google APIs Service: User Data Policy
Restricted scopes
In addition to sensitive scopes, certain scopes are classified as restricted and subject to additional rules that help protect user data. If you publish an app that uses restricted scopes, it must comply with all specifications.
Review the full list of restricted scopes before publishing. Compliant apps must follow the Additional Requirements for Specific API scopes.
Avoid using restricted scopes if possible to simplify the review process. You can use restricted scopes freely for non-public apps.