(core) Add AI assistant usage banners

Summary:
Banners are now shown when there are low or no AI assistant
credits remaining.

Test Plan: Browser tests.

Reviewers: jarek

Reviewed By: jarek

Differential Revision: https://phab.getgrist.com/D4018
This commit is contained in:
George Gevoian 2023-08-30 11:58:18 -04:00
parent 98068cb86c
commit 70feb336d9
6 changed files with 178 additions and 69 deletions

View File

@ -134,6 +134,16 @@ const cssBanner = styled('div', `
} }
`); `);
export const cssBannerLink = styled('span', `
cursor: pointer;
color: unset;
text-decoration: underline;
&:hover, &:focus {
color: unset;
}
`);
const cssButtons = styled('div', ` const cssButtons = styled('div', `
display: flex; display: flex;
gap: 16px; gap: 16px;

View File

@ -1,3 +1,4 @@
import {cssBannerLink} from 'app/client/components/Banner';
import {DocPageModel} from 'app/client/models/DocPageModel'; import {DocPageModel} from 'app/client/models/DocPageModel';
import {urlState} from 'app/client/models/gristUrlState'; import {urlState} from 'app/client/models/gristUrlState';
import {docListHeader} from 'app/client/ui/DocMenuCss'; import {docListHeader} from 'app/client/ui/DocMenuCss';
@ -253,11 +254,11 @@ export function buildUpgradeMessage(
} }
function buildUpgradeLink(linkText: string, onClick: () => void) { function buildUpgradeLink(linkText: string, onClick: () => void) {
return cssUnderlinedLink(linkText, dom.on('click', () => onClick())); return cssBannerLink(linkText, dom.on('click', () => onClick()));
} }
function buildRawDataPageLink(linkText: string) { function buildRawDataPageLink(linkText: string) {
return cssUnderlinedLink(linkText, urlState().setLinkUrl({docPage: 'data'})); return cssBannerLink(linkText, urlState().setLinkUrl({docPage: 'data'}));
} }
interface MetricOptions { interface MetricOptions {
@ -377,16 +378,6 @@ const cssHeader = styled(docListHeader, `
margin-bottom: 0px; margin-bottom: 0px;
`); `);
const cssUnderlinedLink = styled('span', `
cursor: pointer;
color: unset;
text-decoration: underline;
&:hover, &:focus {
color: unset;
}
`);
const cssUsageMetrics = styled('div', ` const cssUsageMetrics = styled('div', `
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;

View File

@ -1,7 +1,8 @@
import {Banner, buildBannerMessage, cssBannerLink} from 'app/client/components/Banner';
import * as commands from 'app/client/components/commands'; import * as commands from 'app/client/components/commands';
import {GristDoc} from 'app/client/components/GristDoc'; import {GristDoc} from 'app/client/components/GristDoc';
import {makeT} from 'app/client/lib/localization'; import {makeT} from 'app/client/lib/localization';
import {localStorageBoolObs} from 'app/client/lib/localStorageObs'; import {localStorageBoolObs, sessionStorageBoolObs} from 'app/client/lib/localStorageObs';
import {movable} from 'app/client/lib/popupUtils'; import {movable} from 'app/client/lib/popupUtils';
import {logTelemetryEvent} from 'app/client/lib/telemetry'; import {logTelemetryEvent} from 'app/client/lib/telemetry';
import {ColumnRec, ViewFieldRec} from 'app/client/models/DocModel'; import {ColumnRec, ViewFieldRec} from 'app/client/models/DocModel';
@ -19,7 +20,9 @@ import {cssLink} from 'app/client/ui2018/links';
import {loadingDots} from 'app/client/ui2018/loaders'; import {loadingDots} from 'app/client/ui2018/loaders';
import {menu, menuCssClass, menuItem} from 'app/client/ui2018/menus'; import {menu, menuCssClass, menuItem} from 'app/client/ui2018/menus';
import {FormulaEditor} from 'app/client/widgets/FormulaEditor'; import {FormulaEditor} from 'app/client/widgets/FormulaEditor';
import {ApiError} from 'app/common/ApiError';
import {AssistanceResponse, AssistanceState, FormulaAssistanceContext} from 'app/common/AssistancePrompts'; import {AssistanceResponse, AssistanceState, FormulaAssistanceContext} from 'app/common/AssistancePrompts';
import {isFreePlan} from 'app/common/Features';
import {commonUrls} from 'app/common/gristUrls'; import {commonUrls} from 'app/common/gristUrls';
import {TelemetryEvent, TelemetryMetadata} from 'app/common/Telemetry'; import {TelemetryEvent, TelemetryMetadata} from 'app/common/Telemetry';
import {getGristConfig} from 'app/common/urlUtils'; import {getGristConfig} from 'app/common/urlUtils';
@ -33,6 +36,8 @@ import {v4 as uuidv4} from 'uuid';
const t = makeT('FormulaEditor'); const t = makeT('FormulaEditor');
const testId = makeTestId('test-formula-editor-'); const testId = makeTestId('test-formula-editor-');
const LOW_CREDITS_WARNING_BANNER_THRESHOLD = 10;
/** /**
* An extension or the FormulaEditor that provides assistance for writing formulas. * An extension or the FormulaEditor that provides assistance for writing formulas.
* It renders itself in the detached FormulaEditor and adds some extra UI elements. * It renders itself in the detached FormulaEditor and adds some extra UI elements.
@ -43,6 +48,7 @@ const testId = makeTestId('test-formula-editor-');
*/ */
export class FormulaAssistant extends Disposable { export class FormulaAssistant extends Disposable {
private _gristDoc = this._options.gristDoc; private _gristDoc = this._options.gristDoc;
private _appModel = this._gristDoc.appModel;
/** Chat component */ /** Chat component */
private _chat: ChatHistory; private _chat: ChatHistory;
/** State of the user input */ /** State of the user input */
@ -52,7 +58,7 @@ export class FormulaAssistant extends Disposable {
private _input: HTMLTextAreaElement; private _input: HTMLTextAreaElement;
/** Is the formula assistant expanded */ /** Is the formula assistant expanded */
private _assistantExpanded = this.autoDispose(localStorageBoolObs( private _assistantExpanded = this.autoDispose(localStorageBoolObs(
`u:${this._options.gristDoc.appModel.currentUser?.id ?? 0};formulaAssistantExpanded`, true)); `u:${this._appModel.currentUser?.id ?? 0};formulaAssistantExpanded`, true));
/** Is the request pending */ /** Is the request pending */
private _waiting = Observable.create(this, false); private _waiting = Observable.create(this, false);
/** Is assistant features are enabled */ /** Is assistant features are enabled */
@ -73,8 +79,8 @@ export class FormulaAssistant extends Disposable {
private _chatPanelBody: HTMLElement; private _chatPanelBody: HTMLElement;
/** Client height of the chat panel body element. */ /** Client height of the chat panel body element. */
private _chatPanelBodyClientHeight = Observable.create<number>(this, 0); private _chatPanelBodyClientHeight = Observable.create<number>(this, 0);
/** Set to true once the panel has been expanded (including by default). */ /** Set to true the first time the panel has been expanded (including by default). */
private _hasExpanded = false; private _hasExpandedOnce = false;
/** /**
* Last known height of the chat panel. * Last known height of the chat panel.
* *
@ -84,6 +90,15 @@ export class FormulaAssistant extends Disposable {
private _lastChatPanelHeight: number|undefined; private _lastChatPanelHeight: number|undefined;
/** True if the chat panel is being resized via dragging. */ /** True if the chat panel is being resized via dragging. */
private _isResizing = Observable.create(this, false); private _isResizing = Observable.create(this, false);
/** Whether the low credit limit banner should be shown. */
private _showApproachingLimitBanner = this.autoDispose(
sessionStorageBoolObs(
`org:${this._appModel.currentOrg?.id ?? 0};formulaAssistantShowApproachingLimitBanner`,
true
));
/** Number of remaining credits. If null, assistant usage is unlimited. */
private _numRemainingCredits = Observable.create<number|null>(this, null);
/** /**
* Debounced version of the method that will force parent editor to resize, we call it often * Debounced version of the method that will force parent editor to resize, we call it often
* as we have an ability to resize the chat window. * as we have an ability to resize the chat window.
@ -156,7 +171,7 @@ export class FormulaAssistant extends Disposable {
this._triggerFinalize = bundleInfo.triggerFinalize; this._triggerFinalize = bundleInfo.triggerFinalize;
this.onDispose(() => { this.onDispose(() => {
if (this._hasExpanded) { if (this._hasExpandedOnce) {
this._logTelemetryEvent('assistantClose', false, { this._logTelemetryEvent('assistantClose', false, {
suggestionApplied: this._chat.conversationSuggestedFormulas.get() suggestionApplied: this._chat.conversationSuggestedFormulas.get()
.includes(this._options.column.formula.peek()), .includes(this._options.column.formula.peek()),
@ -204,7 +219,7 @@ export class FormulaAssistant extends Disposable {
if (this._assistantEnabled && this._assistantExpanded.get()) { if (this._assistantEnabled && this._assistantExpanded.get()) {
this._logTelemetryEvent('assistantOpen', true); this._logTelemetryEvent('assistantOpen', true);
this._hasExpanded = true; this._hasExpandedOnce = true;
} }
return this._domElement; return this._domElement;
@ -294,8 +309,9 @@ export class FormulaAssistant extends Disposable {
this._chatPanelBody = cssChatPanelBody( this._chatPanelBody = cssChatPanelBody(
dom.onDispose(() => observer.disconnect()), dom.onDispose(() => observer.disconnect()),
testId('ai-assistant-chat-panel'), testId('ai-assistant-chat-panel'),
this._buildChatPanelBanner(),
this._chat.buildDom(), this._chat.buildDom(),
this._gristDoc.appModel.currentValidUser ? this._buildChatInput() : this._buildSignupNudge(), this._appModel.currentValidUser ? this._buildChatInput() : this._buildSignupNudge(),
cssChatPanelBody.cls('-resizing', this._isResizing), cssChatPanelBody.cls('-resizing', this._isResizing),
// Stop propagation of mousedown events, as the formula editor will still focus. // Stop propagation of mousedown events, as the formula editor will still focus.
dom.on('mousedown', (ev) => ev.stopPropagation()), dom.on('mousedown', (ev) => ev.stopPropagation()),
@ -306,11 +322,70 @@ export class FormulaAssistant extends Disposable {
return this._chatPanelBody; return this._chatPanelBody;
} }
private _buildChatPanelBanner() {
return dom.domComputed(use => {
const numCredits = use(this._numRemainingCredits);
if (
numCredits === null ||
numCredits > LOW_CREDITS_WARNING_BANNER_THRESHOLD
) {
return null;
} else if (numCredits === 0) {
return dom.create(Banner, {
content: buildBannerMessage(
t('You have used all available credits.'),
' ',
this._buildBannerUpgradeMessage(),
testId('ai-assistant-banner-message'),
),
style: 'error',
bannerCssClass: cssBanner.className,
});
} else {
const showBanner = use(this._showApproachingLimitBanner);
if (!showBanner) { return null; }
return dom.create(Banner, {
content: buildBannerMessage(
t('You have {{numCredits}} remaining credits.', {numCredits}),
' ',
this._buildBannerUpgradeMessage(),
testId('ai-assistant-banner-message'),
),
style: 'warning',
showCloseButton: true,
onClose: () => { this._showApproachingLimitBanner.set(false); },
bannerCssClass: cssBanner.className,
});
}
});
}
private _buildBannerUpgradeMessage() {
const canUpgradeSite = this._appModel.isOwner()
&& Boolean(this._appModel.planName && isFreePlan(this._appModel.planName));
const isBillingManager = this._appModel.isBillingManager() || this._appModel.isSupport();
if (!canUpgradeSite && !isBillingManager) {
return t('For higher limits, contact the site owner.');
}
return t('For higher limits, {{upgradeNudge}}.', {upgradeNudge: cssBannerLink(
canUpgradeSite ? t('upgrade to the Pro Team plan') : t('upgrade your plan'),
dom.on('click', async () => {
if (canUpgradeSite) {
this._gristDoc.appModel.showUpgradeModal();
} else {
await urlState().pushUrl({billing: 'billing'});
}
}))
});
}
/** /**
* Save button handler. We just store the action and wait for the bundler to finalize. * Save button handler. We just store the action and wait for the bundler to finalize.
*/ */
private _saveOrClose() { private _saveOrClose() {
if (this._hasExpanded) { if (this._hasExpandedOnce) {
this._logTelemetryEvent('assistantSave', true, { this._logTelemetryEvent('assistantSave', true, {
oldFormula: this._options.column.formula.peek(), oldFormula: this._options.column.formula.peek(),
newFormula: this._options.editor.getTextValue(), newFormula: this._options.editor.getTextValue(),
@ -324,7 +399,7 @@ export class FormulaAssistant extends Disposable {
* Cancel button handler. * Cancel button handler.
*/ */
private _cancel() { private _cancel() {
if (this._hasExpanded) { if (this._hasExpandedOnce) {
this._logTelemetryEvent('assistantCancel', true); this._logTelemetryEvent('assistantCancel', true);
} }
this._action = 'cancel'; this._action = 'cancel';
@ -424,6 +499,8 @@ export class FormulaAssistant extends Disposable {
} }
private _collapseChatPanel() { private _collapseChatPanel() {
if (!this._assistantExpanded.get()) { return; }
this._assistantExpanded.set(false); this._assistantExpanded.set(false);
// The panel's height and client height may differ; to ensure the collapse transition // The panel's height and client height may differ; to ensure the collapse transition
// appears linear, temporarily disable the transition and sync the height and client // appears linear, temporarily disable the transition and sync the height and client
@ -438,10 +515,11 @@ export class FormulaAssistant extends Disposable {
} }
private _expandChatPanel() { private _expandChatPanel() {
if (!this._hasExpanded) { if (!this._hasExpandedOnce) {
this._logTelemetryEvent('assistantOpen', true); this._logTelemetryEvent('assistantOpen', true);
this._hasExpanded = true; this._hasExpandedOnce = true;
} }
if (this._assistantExpanded.get()) { return; }
this._assistantExpanded.set(true); this._assistantExpanded.set(true);
const editor = this._options.editor.getDom(); const editor = this._options.editor.getDom();
@ -492,13 +570,9 @@ export class FormulaAssistant extends Disposable {
const collapseThreshold = 78; const collapseThreshold = 78;
if (newChatPanelBodyHeight < collapseThreshold) { if (newChatPanelBodyHeight < collapseThreshold) {
if (this._assistantExpanded.get()) {
this._collapseChatPanel(); this._collapseChatPanel();
}
} else { } else {
if (!this._assistantExpanded.get()) {
this._expandChatPanel(); this._expandChatPanel();
}
const calculatedHeight = Math.max( const calculatedHeight = Math.max(
MIN_CHAT_PANEL_BODY_HEIGHT_PX, MIN_CHAT_PANEL_BODY_HEIGHT_PX,
Math.min(total - MIN_FORMULA_EDITOR_HEIGHT_PX, newChatPanelBodyHeight) Math.min(total - MIN_FORMULA_EDITOR_HEIGHT_PX, newChatPanelBodyHeight)
@ -626,12 +700,17 @@ export class FormulaAssistant extends Disposable {
// Get the state of the chat from the column. // Get the state of the chat from the column.
const conversationId = this._chat.conversationId.get(); const conversationId = this._chat.conversationId.get();
const prevState = column.chatHistory.peek().get().state; const prevState = column.chatHistory.peek().get().state;
const {reply, suggestedActions, suggestedFormula, state} = await askAI(gristDoc, { const {reply, suggestedActions, suggestedFormula, state, limit} = await askAI(gristDoc, {
conversationId, conversationId,
column, column,
description, description,
state: prevState, state: prevState,
}); });
if (limit && limit.limit >= 0) {
this._numRemainingCredits.set(Math.max(limit.limit - limit.usage, 0));
} else {
this._numRemainingCredits.set(null);
}
console.debug('received formula assistant response: ', {suggestedActions, suggestedFormula, reply, state}); console.debug('received formula assistant response: ', {suggestedActions, suggestedFormula, reply, state});
// If back-end is capable of conversation, keep its state. // If back-end is capable of conversation, keep its state.
const chatHistoryNew = column.chatHistory.peek(); const chatHistoryNew = column.chatHistory.peek();
@ -682,10 +761,18 @@ export class FormulaAssistant extends Disposable {
try { try {
const response = await this._sendMessage(message); const response = await this._sendMessage(message);
this._chat.addResponse(response); this._chat.addResponse(response);
} catch(err) { } catch (err: unknown) {
this._chat.thinking(false); if (err instanceof ApiError && err.status === 429 && err.details?.limit) {
const {projectedValue, maximum} = err.details.limit;
if (projectedValue >= maximum) {
this._numRemainingCredits.set(0);
return;
}
}
throw err; throw err;
} finally { } finally {
this._chat.thinking(false);
this._waiting.set(false); this._waiting.set(false);
} }
} }
@ -998,9 +1085,9 @@ const MIN_FORMULA_EDITOR_HEIGHT_PX = 100;
const FORMULA_EDITOR_BUTTONS_HEIGHT_PX = 42; const FORMULA_EDITOR_BUTTONS_HEIGHT_PX = 42;
const MIN_CHAT_HISTORY_HEIGHT_PX = 100; const MIN_CHAT_HISTORY_HEIGHT_PX = 160;
const MIN_CHAT_PANEL_BODY_HEIGHT_PX = 120; const MIN_CHAT_PANEL_BODY_HEIGHT_PX = 180;
const CHAT_PANEL_HEADER_HEIGHT_PX = 30; const CHAT_PANEL_HEADER_HEIGHT_PX = 30;
@ -1085,7 +1172,6 @@ const cssHContainer = styled('div', `
margin-top: auto; margin-top: auto;
padding-left: 18px; padding-left: 18px;
padding-right: 18px; padding-right: 18px;
min-height: ${MIN_CHAT_PANEL_BODY_HEIGHT_PX}px;
display: flex; display: flex;
flex-shrink: 0; flex-shrink: 0;
flex-direction: column; flex-direction: column;
@ -1286,7 +1372,6 @@ const cssSignupNudgeWrapper = styled('div', `
border-top: 1px solid ${theme.formulaAssistantBorder}; border-top: 1px solid ${theme.formulaAssistantBorder};
padding: 16px; padding: 16px;
margin-top: auto; margin-top: auto;
min-height: ${MIN_CHAT_PANEL_BODY_HEIGHT_PX}px;
display: flex; display: flex;
flex-shrink: 0; flex-shrink: 0;
flex-direction: column; flex-direction: column;
@ -1308,3 +1393,7 @@ const cssSignupNudgeButtonsRow = styled('div', `
display: flex; display: flex;
justify-content: center; justify-content: center;
`); `);
const cssBanner = styled('div', `
padding: 6px 8px 6px 8px;
`);

View File

@ -55,4 +55,10 @@ export interface AssistanceResponse {
// If the model can be trusted to issue a self-contained // If the model can be trusted to issue a self-contained
// markdown-friendly string, it can be included here. // markdown-friendly string, it can be included here.
reply?: string; reply?: string;
limit?: AssistanceLimit;
}
export interface AssistanceLimit {
usage: number;
limit: number;
} }

View File

@ -1,4 +1,4 @@
import {ApiError, ApiErrorDetails, LimitType} from 'app/common/ApiError'; import {ApiError, LimitType} from 'app/common/ApiError';
import {mapGetOrSet, mapSetOrClear, MapWithTTL} from 'app/common/AsyncCreate'; import {mapGetOrSet, mapSetOrClear, MapWithTTL} from 'app/common/AsyncCreate';
import {getDataLimitStatus} from 'app/common/DocLimits'; import {getDataLimitStatus} from 'app/common/DocLimits';
import {createEmptyOrgUsageSummary, DocumentUsage, OrgUsageSummary} from 'app/common/DocUsage'; import {createEmptyOrgUsageSummary, DocumentUsage, OrgUsageSummary} from 'app/common/DocUsage';
@ -2943,14 +2943,19 @@ export class HomeDBManager extends EventEmitter {
} }
/** /**
* Increases the usage of a limit for a given org. If the limit doesn't exist, it will be created. * Increases the usage of a limit for a given org, and returns it.
* Pass `dryRun: true` to check if the limit can be increased without actually increasing it. *
* If a limit doesn't exist, but the product associated with the org
* has limits for the given `limitType`, one will be created.
*
* Pass `dryRun: true` to check if a limit can be increased without
* actually increasing it.
*/ */
public async increaseUsage(scope: Scope, limitType: LimitType, options: { public async increaseUsage(scope: Scope, limitType: LimitType, options: {
delta: number, delta: number,
dryRun?: boolean, dryRun?: boolean,
}): Promise<void> { }): Promise<Limit|null> {
const limitError = await this._connection.transaction(async manager => { const limitOrError: Limit|ApiError|null = await this._connection.transaction(async manager => {
const org = await this._org(scope, false, scope.org ?? null, {manager, needRealOrg: true}) const org = await this._org(scope, false, scope.org ?? null, {manager, needRealOrg: true})
.innerJoinAndSelect('orgs.billingAccount', 'billing_account') .innerJoinAndSelect('orgs.billingAccount', 'billing_account')
.innerJoinAndSelect('billing_account.product', 'product') .innerJoinAndSelect('billing_account.product', 'product')
@ -2970,7 +2975,7 @@ export class HomeDBManager extends EventEmitter {
if (product.features.baseMaxAssistantCalls === undefined) { if (product.features.baseMaxAssistantCalls === undefined) {
// If the product has no assistantLimit, then it is not billable yet, and we don't need to // If the product has no assistantLimit, then it is not billable yet, and we don't need to
// track usage as it is basically unlimited. // track usage as it is basically unlimited.
return; return null;
} }
existing = new Limit(); existing = new Limit();
existing.billingAccountId = org.billingAccountId; existing.billingAccountId = org.billingAccountId;
@ -2979,36 +2984,38 @@ export class HomeDBManager extends EventEmitter {
existing.usage = 0; existing.usage = 0;
} }
const limitLess = existing.limit === -1; // -1 means no limit, it is not possible to do in stripe. const limitLess = existing.limit === -1; // -1 means no limit, it is not possible to do in stripe.
const usageAfter = existing.usage + options.delta; const projectedValue = existing.usage + options.delta;
if (!limitLess && usageAfter > existing.limit) { if (!limitLess && projectedValue > existing.limit) {
const billable = Boolean(org?.billingAccount?.stripeCustomerId); return new ApiError(
return { `Your ${limitType} limit has been reached. Please upgrade your plan to increase your limit.`,
429,
{
limit: { limit: {
maximum: existing.limit, maximum: existing.limit,
projectedValue: existing.usage + options.delta, projectedValue,
quantity: limitType, quantity: limitType,
value: existing.usage, value: existing.usage,
}, },
tips: [{ tips: [{
// For non-billable accounts, suggest getting a plan, otherwise suggest visiting the billing page. // For non-billable accounts, suggest getting a plan, otherwise suggest visiting the billing page.
action: billable ? 'manage' : 'upgrade', action: org?.billingAccount?.stripeCustomerId ? 'manage' : 'upgrade',
message: `Upgrade to a paid plan to increase your ${limitType} limit.`, message: `Upgrade to a paid plan to increase your ${limitType} limit.`,
}] }],
} as ApiErrorDetails; }
);
} }
existing.usage += options.delta; existing.usage += options.delta;
existing.usedAt = new Date(); existing.usedAt = new Date();
if (!options.dryRun) { if (!options.dryRun) {
await manager.save(existing); await manager.save(existing);
} }
return existing;
}); });
if (limitError) { if (limitOrError instanceof ApiError) {
let message = `Your ${limitType} limit has been reached. Please upgrade your plan to increase your limit.`; throw limitOrError;
if (limitType === 'assistant') {
message = 'You used all available credits. For a bigger limit upgrade you Assistant plan.';
}
throw new ApiError(message, 429, limitError);
} }
return limitOrError;
} }
private async _getOrCreateLimit(accountId: number, limitType: LimitType, force: boolean): Promise<Limit|null> { private async _getOrCreateLimit(accountId: number, limitType: LimitType, force: boolean): Promise<Limit|null> {

View File

@ -1173,8 +1173,14 @@ export class DocWorkerApi {
const docSession = docSessionFromRequest(req); const docSession = docSessionFromRequest(req);
const request = req.body; const request = req.body;
const result = await sendForCompletion(docSession, activeDoc, request); const result = await sendForCompletion(docSession, activeDoc, request);
await this._increaseLimit('assistant', req); const limit = await this._increaseLimit('assistant', req);
res.json(result); res.json({
...result,
limit: !limit ? undefined : {
usage: limit.usage,
limit: limit.limit,
},
});
}) })
); );
@ -1370,7 +1376,7 @@ export class DocWorkerApi {
* Increases the current usage of a limit by 1. * Increases the current usage of a limit by 1.
*/ */
private async _increaseLimit(limit: LimitType, req: Request) { private async _increaseLimit(limit: LimitType, req: Request) {
await this._dbManager.increaseUsage(getDocScope(req), limit, {delta: 1}); return await this._dbManager.increaseUsage(getDocScope(req), limit, {delta: 1});
} }
private async _assertAccess(role: 'viewers'|'editors'|'owners'|null, allowRemoved: boolean, private async _assertAccess(role: 'viewers'|'editors'|'owners'|null, allowRemoved: boolean,