mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
(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:
parent
98068cb86c
commit
70feb336d9
@ -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;
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
|
`);
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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> {
|
||||||
|
@ -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,
|
||||||
|
Loading…
Reference in New Issue
Block a user