import { Link, MessageBar, MessageBarType, Stack, Text, TextField } from "@fluentui/react"; import * as React from "react"; import * as Constants from "../../../../Common/Constants"; import { Platform, configContext } from "../../../../ConfigContext"; import * as DataModels from "../../../../Contracts/DataModels"; import * as ViewModels from "../../../../Contracts/ViewModels"; import * as SharedConstants from "../../../../Shared/Constants"; import { userContext } from "../../../../UserContext"; import * as AutoPilotUtils from "../../../../Utils/AutoPilotUtils"; import { isRunningOnNationalCloud } from "../../../../Utils/CloudUtils"; import { getTextFieldStyles, getThroughputApplyShortDelayMessage, subComponentStackProps, throughputUnit, titleAndInputStackProps, } from "../SettingsRenderUtils"; import { hasDatabaseSharedThroughput } from "../SettingsUtils"; import { ThroughputBucketsComponent } from "./ThroughputInputComponents/ThroughputBucketsComponent"; import { ThroughputInputAutoPilotV3Component } from "./ThroughputInputComponents/ThroughputInputAutoPilotV3Component"; export interface ScaleComponentProps { collection: ViewModels.Collection; database: ViewModels.Database; isFixedContainer: boolean; onThroughputChange: (newThroughput: number) => void; throughput: number; throughputBaseline: number; autoPilotThroughput: number; autoPilotThroughputBaseline: number; isAutoPilotSelected: boolean; wasAutopilotOriginallySet: boolean; onAutoPilotSelected: (isAutoPilotSelected: boolean) => void; onMaxAutoPilotThroughputChange: (newThroughput: number) => void; onScaleSaveableChange: (isScaleSaveable: boolean) => void; onScaleDiscardableChange: (isScaleDiscardable: boolean) => void; throughputBuckets: DataModels.ThroughputBucket[]; throughputBucketsBaseline: DataModels.ThroughputBucket[]; enableThroughputBuckets: boolean; onThroughputBucketChange: (throughputBuckets: DataModels.ThroughputBucket[]) => void; throughputError?: string; } interface ScaleComponentState { isThroughputSaveable: boolean; isBucketsSaveable: boolean; isThroughputDiscardable: boolean; isBucketsDiscardable: boolean; } export class ScaleComponent extends React.Component { private isEmulator: boolean; private offer: DataModels.Offer; private databaseId: string; private collectionId: string; constructor(props: ScaleComponentProps) { super(props); this.isEmulator = configContext.platform === Platform.Emulator; this.offer = this.props.database?.offer() || this.props.collection?.offer(); this.databaseId = this.props.database?.id() || this.props.collection.databaseId; this.collectionId = this.props.collection?.id(); this.state = { isThroughputSaveable: false, isBucketsSaveable: false, isThroughputDiscardable: false, isBucketsDiscardable: false, }; console.log(this.offer); } public isAutoScaleEnabled = (): boolean => { const accountCapabilities: DataModels.Capability[] = userContext?.databaseAccount?.properties?.capabilities; const enableAutoScaleCapability = accountCapabilities && accountCapabilities.find((capability: DataModels.Capability) => { return ( capability && capability.name && capability.name.toLowerCase() === Constants.CapabilityNames.EnableAutoScale.toLowerCase() ); }); return !!enableAutoScaleCapability; }; public getMaxRUs = (): number => { if (userContext.isTryCosmosDBSubscription) { return Constants.TryCosmosExperience.maxRU; } if (this.props.isFixedContainer) { return SharedConstants.CollectionCreation.MaxRUPerPartition; } return SharedConstants.CollectionCreation.DefaultCollectionRUs1Million; }; public getMinRUs = (): number => { if (userContext.isTryCosmosDBSubscription) { return SharedConstants.CollectionCreation.DefaultCollectionRUs400; } return this.offer?.minimumThroughput || SharedConstants.CollectionCreation.DefaultCollectionRUs400; }; public getThroughputTitle = (): string => { if (this.props.isAutoPilotSelected) { return AutoPilotUtils.getAutoPilotHeaderText(); } const minThroughput: string = this.getMinRUs().toLocaleString(); const maxThroughput: string = !this.props.isFixedContainer ? "unlimited" : this.getMaxRUs().toLocaleString(); return `Throughput (${minThroughput} - ${maxThroughput} RU/s)`; }; public canThroughputExceedMaximumValue = (): boolean => { return !this.props.isFixedContainer && configContext.platform === Platform.Portal && !isRunningOnNationalCloud(); }; public getInitialNotificationElement = (): JSX.Element => { if (this.offer?.offerReplacePending) { const throughput = this.offer.manualThroughput || this.offer.autoscaleMaxThroughput; return getThroughputApplyShortDelayMessage( this.props.isAutoPilotSelected, throughput, throughputUnit, this.databaseId, this.collectionId, ); } return undefined; }; private getThroughputInputComponent = (): JSX.Element => ( ); private isFreeTierAccount(): boolean { return userContext?.databaseAccount?.properties?.enableFreeTier; } private getFreeTierInfoMessage(): JSX.Element { const freeTierLimits = SharedConstants.FreeTierLimits; return ( With free tier, you will get the first {freeTierLimits.RU} RU/s and {freeTierLimits.Storage} GB of storage in this account for free. To keep your account free, keep the total RU/s across all resources in the account to{" "} {freeTierLimits.RU} RU/s. Learn more. ); } private updateScaleSettingsState = (updates: Partial) => { this.setState( (prevState) => { const hasChanges = Object.keys(updates).some( (key) => prevState[key as keyof ScaleComponentState] !== updates[key as keyof ScaleComponentState], ); return hasChanges ? { ...prevState, ...updates } : null; }, () => { const isSaveable = this.state.isThroughputSaveable || this.state.isBucketsSaveable; const isDiscardable = this.state.isThroughputDiscardable || this.state.isBucketsDiscardable; this.props.onScaleSaveableChange(isSaveable); this.props.onScaleDiscardableChange(isDiscardable); }, ); }; private handleThroughputSaveableChange = (isSaveable: boolean) => { this.updateScaleSettingsState({ isThroughputSaveable: isSaveable }); }; private handleThroughputDiscardableChange = (isDiscardable: boolean) => { this.updateScaleSettingsState({ isThroughputDiscardable: isDiscardable }); }; private handleBucketsSaveableChange = (isSaveable: boolean) => { this.updateScaleSettingsState({ isBucketsSaveable: isSaveable }); }; private handleBucketsDiscardableChange = (isDiscardable: boolean) => { this.updateScaleSettingsState({ isBucketsDiscardable: isDiscardable }); }; public render(): JSX.Element { return ( {this.isFreeTierAccount() && ( {this.getFreeTierInfoMessage()} )} {this.getInitialNotificationElement() && ( {this.getInitialNotificationElement()} )} {!this.isAutoScaleEnabled() && {this.getThroughputInputComponent()}} {this.props.enableThroughputBuckets && ( )} {/* TODO: Replace link with call to the Azure Support blade */} {this.isAutoScaleEnabled() && ( Throughput (RU/s) Your account has custom settings that prevents setting throughput at the container level. Please work with your Cosmos DB engineering team point of contact to make changes. )} ); } }