mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-29 22:02:01 +00:00
Compare commits
17 Commits
refresh-ar
...
users/aisa
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
3ea5a059b0 | ||
|
|
37a5663bad | ||
|
|
31c9574df4 | ||
|
|
2e8e02f75b | ||
|
|
8809ba6cd2 | ||
|
|
592f2cd7e6 | ||
|
|
d966f4f2a7 | ||
|
|
42e230b88b | ||
|
|
6196ba4722 | ||
|
|
4801aae754 | ||
|
|
2a02112d87 | ||
|
|
bbfff77495 | ||
|
|
f695b42071 | ||
|
|
a8a96e22b4 | ||
|
|
a1b026544d | ||
|
|
a912233b33 | ||
|
|
487130f6e3 |
@@ -128,7 +128,7 @@
|
|||||||
@provisionDatabaseThroughputInfo: 200px;
|
@provisionDatabaseThroughputInfo: 200px;
|
||||||
|
|
||||||
//tabs container
|
//tabs container
|
||||||
@ActiveTabHeight: 31px;
|
@ActiveTabHeight: 32px;
|
||||||
@ActiveTabWidth: 141px;
|
@ActiveTabWidth: 141px;
|
||||||
@TabsHeight: 30px;
|
@TabsHeight: 30px;
|
||||||
@TabsWidth: 140px;
|
@TabsWidth: 140px;
|
||||||
|
|||||||
@@ -2643,7 +2643,7 @@ a:link {
|
|||||||
|
|
||||||
.tabPanesContainer {
|
.tabPanesContainer {
|
||||||
flex-grow: 1;
|
flex-grow: 1;
|
||||||
overflow-y: scroll;
|
overflow: hidden;
|
||||||
background-color: var(--colorNeutralBackground1);
|
background-color: var(--colorNeutralBackground1);
|
||||||
color: var(--colorNeutralForeground1);
|
color: var(--colorNeutralForeground1);
|
||||||
}
|
}
|
||||||
@@ -2651,6 +2651,7 @@ a:link {
|
|||||||
.tabs-container {
|
.tabs-container {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.paddingspan4 {
|
.paddingspan4 {
|
||||||
@@ -2677,7 +2678,7 @@ a:link {
|
|||||||
width: @ActiveTabWidth;
|
width: @ActiveTabWidth;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-tabs > li.active .contentWrapper {
|
.nav-tabs > li.active .contentWrapper .tabNavText {
|
||||||
border-bottom: 2px solid var(--colorCompoundBrandBackground);
|
border-bottom: 2px solid var(--colorCompoundBrandBackground);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ html {
|
|||||||
body {
|
body {
|
||||||
font-family: @FabricFont;
|
font-family: @FabricFont;
|
||||||
background-color: #f5f5f5;
|
background-color: #f5f5f5;
|
||||||
|
--colorCompoundBrandBackground: @FabricAccentMedium;
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
a {
|
||||||
@@ -41,7 +42,7 @@ a:focus {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.nav-tabs-margin {
|
.nav-tabs-margin {
|
||||||
padding-top: 5px;
|
padding-top: 0px;
|
||||||
background-color: #ffffff;
|
background-color: #ffffff;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -68,17 +69,20 @@ a:focus {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.nav-tabs > li > .tabNavContentContainer > .tab_Content:hover {
|
.nav-tabs > li > .tabNavContentContainer > .tab_Content:hover {
|
||||||
border-bottom: 2px solid #e0e0e0;
|
border-bottom: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content,
|
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content,
|
||||||
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content:hover {
|
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content:hover {
|
||||||
border-bottom: 2px solid @FabricAccentMedium;
|
border-bottom: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content > .contentWrapper > .tabNavText {
|
.nav-tabs > li.active > .tabNavContentContainer > .tab_Content > .contentWrapper > .tabNavText {
|
||||||
border-bottom: 0px none transparent;
|
border-bottom: 0px none transparent;
|
||||||
}
|
}
|
||||||
|
.nav-tabs > li.active .contentWrapper .tabNavText {
|
||||||
|
border-bottom: 2px solid @FabricAccentMedium;
|
||||||
|
}
|
||||||
|
|
||||||
.tabNavContentContainer {
|
.tabNavContentContainer {
|
||||||
padding: @SmallSpace 0px @SmallSpace 0px;
|
padding: @SmallSpace 0px @SmallSpace 0px;
|
||||||
|
|||||||
@@ -155,7 +155,12 @@ export class ComputedPropertiesComponent extends React.Component<
|
|||||||
</Link>
|
</Link>
|
||||||
  about how to define computed properties and how to use them.
|
  about how to define computed properties and how to use them.
|
||||||
</Text>
|
</Text>
|
||||||
<div className="settingsV2Editor" tabIndex={0} ref={this.computedPropertiesDiv}></div>
|
<div
|
||||||
|
className="settingsV2Editor"
|
||||||
|
tabIndex={0}
|
||||||
|
ref={this.computedPropertiesDiv}
|
||||||
|
data-test="computed-properties-editor"
|
||||||
|
></div>
|
||||||
</Stack>
|
</Stack>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -302,8 +302,8 @@ export class SubSettingsComponent extends React.Component<SubSettingsComponentPr
|
|||||||
);
|
);
|
||||||
|
|
||||||
private geoSpatialConfigTypeChoiceGroupOptions: IChoiceGroupOption[] = [
|
private geoSpatialConfigTypeChoiceGroupOptions: IChoiceGroupOption[] = [
|
||||||
{ key: GeospatialConfigType.Geography, text: "Geography" },
|
{ key: GeospatialConfigType.Geography, text: "Geography", ariaLabel: "geography-option" },
|
||||||
{ key: GeospatialConfigType.Geometry, text: "Geometry" },
|
{ key: GeospatialConfigType.Geometry, text: "Geometry", ariaLabel: "geometry-option" },
|
||||||
];
|
];
|
||||||
|
|
||||||
private getGeoSpatialComponent = (): JSX.Element => (
|
private getGeoSpatialComponent = (): JSX.Element => (
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ exports[`ComputedPropertiesComponent renders 1`] = `
|
|||||||
</Text>
|
</Text>
|
||||||
<div
|
<div
|
||||||
className="settingsV2Editor"
|
className="settingsV2Editor"
|
||||||
|
data-test="computed-properties-editor"
|
||||||
tabIndex={0}
|
tabIndex={0}
|
||||||
/>
|
/>
|
||||||
</Stack>
|
</Stack>
|
||||||
|
|||||||
@@ -167,10 +167,12 @@ exports[`SubSettingsComponent analyticalTimeToLive hidden 1`] = `
|
|||||||
options={
|
options={
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geography-option",
|
||||||
"key": "Geography",
|
"key": "Geography",
|
||||||
"text": "Geography",
|
"text": "Geography",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geometry-option",
|
||||||
"key": "Geometry",
|
"key": "Geometry",
|
||||||
"text": "Geometry",
|
"text": "Geometry",
|
||||||
},
|
},
|
||||||
@@ -652,10 +654,12 @@ exports[`SubSettingsComponent analyticalTimeToLiveSeconds hidden 1`] = `
|
|||||||
options={
|
options={
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geography-option",
|
||||||
"key": "Geography",
|
"key": "Geography",
|
||||||
"text": "Geography",
|
"text": "Geography",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geometry-option",
|
||||||
"key": "Geometry",
|
"key": "Geometry",
|
||||||
"text": "Geometry",
|
"text": "Geometry",
|
||||||
},
|
},
|
||||||
@@ -1224,10 +1228,12 @@ exports[`SubSettingsComponent changeFeedPolicy hidden 1`] = `
|
|||||||
options={
|
options={
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geography-option",
|
||||||
"key": "Geography",
|
"key": "Geography",
|
||||||
"text": "Geography",
|
"text": "Geography",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geometry-option",
|
||||||
"key": "Geometry",
|
"key": "Geometry",
|
||||||
"text": "Geometry",
|
"text": "Geometry",
|
||||||
},
|
},
|
||||||
@@ -1760,10 +1766,12 @@ exports[`SubSettingsComponent renders 1`] = `
|
|||||||
options={
|
options={
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geography-option",
|
||||||
"key": "Geography",
|
"key": "Geography",
|
||||||
"text": "Geography",
|
"text": "Geography",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geometry-option",
|
||||||
"key": "Geometry",
|
"key": "Geometry",
|
||||||
"text": "Geometry",
|
"text": "Geometry",
|
||||||
},
|
},
|
||||||
@@ -2330,10 +2338,12 @@ exports[`SubSettingsComponent timeToLiveSeconds hidden 1`] = `
|
|||||||
options={
|
options={
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geography-option",
|
||||||
"key": "Geography",
|
"key": "Geography",
|
||||||
"text": "Geography",
|
"text": "Geography",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
"ariaLabel": "geometry-option",
|
||||||
"key": "Geometry",
|
"key": "Geometry",
|
||||||
"text": "Geometry",
|
"text": "Geometry",
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -205,7 +205,7 @@ export const UploadItemsPane: FunctionComponent<UploadItemsPaneProps> = ({ onUpl
|
|||||||
tooltip="Select one or more JSON files to upload. Each file can contain a single JSON document or an array of JSON documents. The combined size of all files in an individual upload operation must be less than 2 MB. You can perform multiple upload operations for larger data sets."
|
tooltip="Select one or more JSON files to upload. Each file can contain a single JSON document or an array of JSON documents. The combined size of all files in an individual upload operation must be less than 2 MB. You can perform multiple upload operations for larger data sets."
|
||||||
/>
|
/>
|
||||||
{uploadFileData?.length > 0 && (
|
{uploadFileData?.length > 0 && (
|
||||||
<div className="fileUploadSummaryContainer">
|
<div className="fileUploadSummaryContainer" data-test="file-upload-status">
|
||||||
<b style={{ color: "var(--colorNeutralForeground1)" }}>File upload status</b>
|
<b style={{ color: "var(--colorNeutralForeground1)" }}>File upload status</b>
|
||||||
<DetailsList
|
<DetailsList
|
||||||
items={uploadFileData}
|
items={uploadFileData}
|
||||||
|
|||||||
@@ -61,7 +61,8 @@ const useStyles = makeStyles({
|
|||||||
display: "flex",
|
display: "flex",
|
||||||
flexDirection: "column",
|
flexDirection: "column",
|
||||||
alignItems: "center",
|
alignItems: "center",
|
||||||
minHeight: "100vh",
|
height: "100%",
|
||||||
|
overflowY: "auto",
|
||||||
backgroundColor: "var(--colorNeutralBackground1)",
|
backgroundColor: "var(--colorNeutralBackground1)",
|
||||||
color: "var(--colorNeutralForeground1)",
|
color: "var(--colorNeutralForeground1)",
|
||||||
},
|
},
|
||||||
@@ -73,20 +74,19 @@ const useStyles = makeStyles({
|
|||||||
},
|
},
|
||||||
title: {
|
title: {
|
||||||
fontSize: "48px",
|
fontSize: "48px",
|
||||||
fontWeight: "500",
|
fontWeight: "400",
|
||||||
margin: "16px auto",
|
margin: "16px auto",
|
||||||
color: "var(--colorNeutralForeground1)",
|
color: "var(--colorNeutralForeground1)",
|
||||||
},
|
},
|
||||||
subtitle: {
|
subtitle: {
|
||||||
fontSize: "18px",
|
fontSize: "18px",
|
||||||
marginBottom: "40px",
|
|
||||||
color: "var(--colorNeutralForeground2)",
|
color: "var(--colorNeutralForeground2)",
|
||||||
},
|
},
|
||||||
cardContainer: {
|
cardContainer: {
|
||||||
display: "grid",
|
display: "grid",
|
||||||
gridTemplateColumns: "repeat(2, 1fr)",
|
gridTemplateColumns: "repeat(2, 1fr)",
|
||||||
gap: "16px",
|
gap: "16px",
|
||||||
width: "66%",
|
width: "60%",
|
||||||
margin: "0 auto",
|
margin: "0 auto",
|
||||||
backgroundColor: "var(--colorNeutralBackground1)",
|
backgroundColor: "var(--colorNeutralBackground1)",
|
||||||
color: "var(--colorNeutralForeground1)",
|
color: "var(--colorNeutralForeground1)",
|
||||||
@@ -100,7 +100,7 @@ const useStyles = makeStyles({
|
|||||||
color: "var(--colorNeutralForeground1)",
|
color: "var(--colorNeutralForeground1)",
|
||||||
border: "1px solid var(--colorNeutralStroke1)",
|
border: "1px solid var(--colorNeutralStroke1)",
|
||||||
borderRadius: "4px",
|
borderRadius: "4px",
|
||||||
boxShadow: "var(--shadow4)",
|
boxShadow: "rgba(0, 0, 0, 0.25) 0px 4px 4px",
|
||||||
cursor: "pointer",
|
cursor: "pointer",
|
||||||
minHeight: "150px",
|
minHeight: "150px",
|
||||||
"&:hover": {
|
"&:hover": {
|
||||||
@@ -128,11 +128,10 @@ const useStyles = makeStyles({
|
|||||||
textAlign: "left",
|
textAlign: "left",
|
||||||
},
|
},
|
||||||
moreStuffContainer: {
|
moreStuffContainer: {
|
||||||
display: "grid",
|
display: "flex",
|
||||||
gridTemplateColumns: "repeat(3, 1fr)",
|
justifyContent: "space-between",
|
||||||
gap: "32px",
|
gap: "32px",
|
||||||
width: "66%",
|
width: "90%",
|
||||||
margin: "40px auto",
|
|
||||||
},
|
},
|
||||||
moreStuffColumn: {
|
moreStuffColumn: {
|
||||||
display: "flex",
|
display: "flex",
|
||||||
@@ -227,7 +226,7 @@ export const SplashScreen: React.FC<SplashScreenProps> = ({ explorer }) => {
|
|||||||
return (
|
return (
|
||||||
<Stack
|
<Stack
|
||||||
className="splashStackContainer"
|
className="splashStackContainer"
|
||||||
style={{ width: "66%", cursor: "pointer", margin: "40px auto" }}
|
style={{ width: "60%", cursor: "pointer", margin: "40px auto" }}
|
||||||
tokens={{ childrenGap: 16 }}
|
tokens={{ childrenGap: 16 }}
|
||||||
>
|
>
|
||||||
<Stack className="splashStackRow" horizontal>
|
<Stack className="splashStackRow" horizontal>
|
||||||
@@ -903,9 +902,9 @@ export const SplashScreen: React.FC<SplashScreenProps> = ({ explorer }) => {
|
|||||||
return (
|
return (
|
||||||
<div className={styles.splashScreenContainer}>
|
<div className={styles.splashScreenContainer}>
|
||||||
<div className={styles.splashScreen}>
|
<div className={styles.splashScreen}>
|
||||||
<h1 className={styles.title} role="heading" aria-label="Welcome to Azure Cosmos DB">
|
<h2 className={styles.title} role="heading" aria-label="Welcome to Azure Cosmos DB">
|
||||||
Welcome to Azure Cosmos DB<span className="activePatch"></span>
|
Welcome to Azure Cosmos DB<span className="activePatch"></span>
|
||||||
</h1>
|
</h2>
|
||||||
<div className={styles.subtitle}>Globally distributed, multi-model database service for any scale</div>
|
<div className={styles.subtitle}>Globally distributed, multi-model database service for any scale</div>
|
||||||
{getSplashScreenButtons()}
|
{getSplashScreenButtons()}
|
||||||
{useCarousel.getState().showCoachMark && (
|
{useCarousel.getState().showCoachMark && (
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ const useStyles = makeStyles({
|
|||||||
button: {
|
button: {
|
||||||
border: "1px solid var(--colorNeutralStroke1)",
|
border: "1px solid var(--colorNeutralStroke1)",
|
||||||
boxSizing: "border-box",
|
boxSizing: "border-box",
|
||||||
boxShadow: "var(--shadow4)",
|
boxShadow: "rgba(0, 0, 0, 0.25) 0px 4px 4px",
|
||||||
borderRadius: "4px",
|
borderRadius: "4px",
|
||||||
padding: "32px 16px",
|
padding: "32px 16px",
|
||||||
backgroundColor: "var(--colorNeutralBackground1)",
|
backgroundColor: "var(--colorNeutralBackground1)",
|
||||||
|
|||||||
@@ -326,6 +326,7 @@ type PanelOpenOptions = {
|
|||||||
export enum CommandBarButton {
|
export enum CommandBarButton {
|
||||||
Save = "Save",
|
Save = "Save",
|
||||||
ExecuteQuery = "Execute Query",
|
ExecuteQuery = "Execute Query",
|
||||||
|
UploadItem = "Upload Item",
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Helper class that provides locator methods for DataExplorer components, on top of a Frame */
|
/** Helper class that provides locator methods for DataExplorer components, on top of a Frame */
|
||||||
|
|||||||
@@ -1,7 +1,17 @@
|
|||||||
import { expect, test } from "@playwright/test";
|
import { expect, test } from "@playwright/test";
|
||||||
|
|
||||||
import { DataExplorer, DocumentsTab, TestAccount } from "../fx";
|
import { existsSync, unlinkSync, writeFileSync } from "fs";
|
||||||
import { retry, setPartitionKeys } from "../testData";
|
import path from "path";
|
||||||
|
import { CommandBarButton, DataExplorer, DocumentsTab, ONE_MINUTE_MS, TestAccount } from "../fx";
|
||||||
|
import {
|
||||||
|
createTestSQLContainer,
|
||||||
|
itemsPerPartition,
|
||||||
|
partitionCount,
|
||||||
|
retry,
|
||||||
|
setPartitionKeys,
|
||||||
|
TestContainerContext,
|
||||||
|
TestData,
|
||||||
|
} from "../testData";
|
||||||
import { documentTestCases } from "./testCases";
|
import { documentTestCases } from "./testCases";
|
||||||
|
|
||||||
let explorer: DataExplorer = null!;
|
let explorer: DataExplorer = null!;
|
||||||
@@ -95,3 +105,108 @@ for (const { name, databaseId, containerId, documents } of documentTestCases) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
test.describe.serial("Upload Item", () => {
|
||||||
|
let context: TestContainerContext = null!;
|
||||||
|
const uploadDocumentFilePath: string = path.join(__dirname, "uploadDocument.json");
|
||||||
|
|
||||||
|
test.beforeEach("Create Test Database and Open documents tab", async ({ page }) => {
|
||||||
|
context = await createTestSQLContainer();
|
||||||
|
explorer = await DataExplorer.open(page, TestAccount.SQL);
|
||||||
|
|
||||||
|
const containerNode = await explorer.waitForContainerNode(context.database.id, context.container.id);
|
||||||
|
await containerNode.expand();
|
||||||
|
|
||||||
|
const containerMenuNode = await explorer.waitForContainerItemsNode(context.database.id, context.container.id);
|
||||||
|
await containerMenuNode.element.click();
|
||||||
|
});
|
||||||
|
|
||||||
|
test.afterEach("Delete Test Database and uploadDocument.json", async () => {
|
||||||
|
if (existsSync(uploadDocumentFilePath)) {
|
||||||
|
unlinkSync(uploadDocumentFilePath);
|
||||||
|
}
|
||||||
|
await context?.dispose();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("upload document", async () => {
|
||||||
|
// Create file to upload
|
||||||
|
const TestDataJsonString: string = JSON.stringify(TestData, null, 2);
|
||||||
|
writeFileSync(uploadDocumentFilePath, TestDataJsonString);
|
||||||
|
|
||||||
|
const uploadItemCommandBar = explorer.commandBarButton(CommandBarButton.UploadItem);
|
||||||
|
await uploadItemCommandBar.click();
|
||||||
|
|
||||||
|
// Select file to upload
|
||||||
|
await explorer.frame.setInputFiles("#importFileInput", uploadDocumentFilePath);
|
||||||
|
|
||||||
|
const uploadButton = explorer.frame.getByTestId("Panel/OkButton");
|
||||||
|
await uploadButton.click();
|
||||||
|
|
||||||
|
// Verify upload success message
|
||||||
|
const fileUploadStatusExpected: string = `${partitionCount * itemsPerPartition} created, 0 throttled, 0 errors`;
|
||||||
|
const fileUploadStatus = explorer.frame.getByTestId("file-upload-status");
|
||||||
|
await expect(fileUploadStatus).toContainText(fileUploadStatusExpected, {
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("upload same document twice", async () => {
|
||||||
|
// Create file to upload
|
||||||
|
const TestDataJsonString: string = JSON.stringify(TestData, null, 2);
|
||||||
|
writeFileSync(uploadDocumentFilePath, TestDataJsonString);
|
||||||
|
|
||||||
|
const uploadItemCommandBar = explorer.commandBarButton(CommandBarButton.UploadItem);
|
||||||
|
await uploadItemCommandBar.click();
|
||||||
|
|
||||||
|
// Select file to upload
|
||||||
|
await explorer.frame.setInputFiles("#importFileInput", uploadDocumentFilePath);
|
||||||
|
|
||||||
|
const uploadButton = explorer.frame.getByTestId("Panel/OkButton");
|
||||||
|
await uploadButton.click();
|
||||||
|
|
||||||
|
// Verify upload success message
|
||||||
|
const fileUploadStatusExpected: string = `${partitionCount * itemsPerPartition} created, 0 throttled, 0 errors`;
|
||||||
|
const fileUploadStatus = explorer.frame.getByTestId("file-upload-status");
|
||||||
|
await expect(fileUploadStatus).toContainText(fileUploadStatusExpected, {
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Select file to upload again
|
||||||
|
await explorer.frame.setInputFiles("#importFileInput", uploadDocumentFilePath);
|
||||||
|
await uploadButton.click();
|
||||||
|
|
||||||
|
// Verify upload failure message
|
||||||
|
const errorIcon = explorer.frame.getByRole("img", { name: "error" });
|
||||||
|
await expect(errorIcon).toBeVisible({ timeout: ONE_MINUTE_MS });
|
||||||
|
await expect(fileUploadStatus).toContainText(
|
||||||
|
`0 created, 0 throttled, ${partitionCount * itemsPerPartition} errors`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("upload invalid json", async () => {
|
||||||
|
// Create file to upload
|
||||||
|
let TestDataJsonString: string = JSON.stringify(TestData, null, 2);
|
||||||
|
// Remove the first '[' so that it becomes invalid json
|
||||||
|
TestDataJsonString = TestDataJsonString.substring(1);
|
||||||
|
writeFileSync(uploadDocumentFilePath, TestDataJsonString);
|
||||||
|
|
||||||
|
const uploadItemCommandBar = explorer.commandBarButton(CommandBarButton.UploadItem);
|
||||||
|
await uploadItemCommandBar.click();
|
||||||
|
|
||||||
|
// Select file to upload
|
||||||
|
await explorer.frame.setInputFiles("#importFileInput", uploadDocumentFilePath);
|
||||||
|
|
||||||
|
const uploadButton = explorer.frame.getByTestId("Panel/OkButton");
|
||||||
|
await uploadButton.click();
|
||||||
|
|
||||||
|
// Verify upload failure message
|
||||||
|
const fileUploadStatusExpected: string = "Unexpected non-whitespace character after JSON";
|
||||||
|
const fileUploadErrorList = explorer.frame.getByLabel("error list");
|
||||||
|
await expect(fileUploadErrorList).toContainText(fileUploadStatusExpected, {
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
103
test/sql/scaleAndSettings/computedProperties.spec.ts
Normal file
103
test/sql/scaleAndSettings/computedProperties.spec.ts
Normal file
@@ -0,0 +1,103 @@
|
|||||||
|
import { expect, test } from "@playwright/test";
|
||||||
|
import * as DataModels from "../../../src/Contracts/DataModels";
|
||||||
|
import { CommandBarButton, DataExplorer, ONE_MINUTE_MS, TestAccount } from "../../fx";
|
||||||
|
import { createTestSQLContainer, TestContainerContext } from "../../testData";
|
||||||
|
|
||||||
|
test.describe("Computed Properties", () => {
|
||||||
|
let context: TestContainerContext = null!;
|
||||||
|
let explorer: DataExplorer = null!;
|
||||||
|
|
||||||
|
test.beforeAll("Create Test Database", async () => {
|
||||||
|
context = await createTestSQLContainer(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test.beforeEach("Open Settings tab under Scale & Settings", async ({ page }) => {
|
||||||
|
explorer = await DataExplorer.open(page, TestAccount.SQL);
|
||||||
|
const containerNode = await explorer.waitForContainerNode(context.database.id, context.container.id);
|
||||||
|
await containerNode.expand();
|
||||||
|
|
||||||
|
// Click Scale & Settings and open Settings tab
|
||||||
|
await explorer.openScaleAndSettings(context);
|
||||||
|
const computedPropertiesTab = explorer.frame.getByTestId("settings-tab-header/ComputedPropertiesTab");
|
||||||
|
await computedPropertiesTab.click();
|
||||||
|
});
|
||||||
|
|
||||||
|
test.afterAll("Delete Test Database", async () => {
|
||||||
|
await context?.dispose();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Add valid computed property", async ({ page }) => {
|
||||||
|
await clearComputedPropertiesTextBoxContent({ page });
|
||||||
|
|
||||||
|
// Create computed property
|
||||||
|
const computedProperties: DataModels.ComputedProperties = [
|
||||||
|
{
|
||||||
|
name: "cp_lowerName",
|
||||||
|
query: "SELECT VALUE LOWER(c.name) FROM c",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const computedPropertiesString: string = JSON.stringify(computedProperties);
|
||||||
|
await page.keyboard.type(computedPropertiesString);
|
||||||
|
|
||||||
|
// Save changes
|
||||||
|
const saveButton = explorer.commandBarButton(CommandBarButton.Save);
|
||||||
|
await expect(saveButton).toBeEnabled();
|
||||||
|
await saveButton.click();
|
||||||
|
await expect(explorer.getConsoleMessage()).toContainText(`Successfully updated container ${context.container.id}`, {
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Add computed property with invalid query", async ({ page }) => {
|
||||||
|
await clearComputedPropertiesTextBoxContent({ page });
|
||||||
|
|
||||||
|
// Create computed property with no VALUE keyword in query
|
||||||
|
const computedProperties: DataModels.ComputedProperties = [
|
||||||
|
{
|
||||||
|
name: "cp_lowerName",
|
||||||
|
query: "SELECT LOWER(c.name) FROM c",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const computedPropertiesString: string = JSON.stringify(computedProperties);
|
||||||
|
await page.keyboard.type(computedPropertiesString);
|
||||||
|
|
||||||
|
// Save changes
|
||||||
|
const saveButton = explorer.commandBarButton(CommandBarButton.Save);
|
||||||
|
await expect(saveButton).toBeEnabled();
|
||||||
|
await saveButton.click();
|
||||||
|
await expect(explorer.getConsoleMessage()).toContainText(`Failed to update container ${context.container.id}`, {
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Add computed property with invalid json", async ({ page }) => {
|
||||||
|
await clearComputedPropertiesTextBoxContent({ page });
|
||||||
|
|
||||||
|
// Create computed property with no VALUE keyword in query
|
||||||
|
const computedProperties: DataModels.ComputedProperties = [
|
||||||
|
{
|
||||||
|
name: "cp_lowerName",
|
||||||
|
query: "SELECT LOWER(c.name) FROM c",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
const computedPropertiesString: string = JSON.stringify(computedProperties);
|
||||||
|
await page.keyboard.type(computedPropertiesString + "]");
|
||||||
|
|
||||||
|
// Save button should remain disabled due to invalid json
|
||||||
|
const saveButton = explorer.commandBarButton(CommandBarButton.Save);
|
||||||
|
await expect(saveButton).toBeDisabled();
|
||||||
|
});
|
||||||
|
|
||||||
|
const clearComputedPropertiesTextBoxContent = async ({ page }): Promise<void> => {
|
||||||
|
// Get computed properties text box
|
||||||
|
const computedPropertiesTextBox = explorer.frame.getByRole("textbox", { name: "Computed properties" });
|
||||||
|
await computedPropertiesTextBox.waitFor();
|
||||||
|
const computedPropertiesEditor = explorer.frame.getByTestId("computed-properties-editor");
|
||||||
|
await computedPropertiesEditor.click();
|
||||||
|
|
||||||
|
// Clear existing content
|
||||||
|
const isMac: boolean = process.platform === "darwin";
|
||||||
|
await page.keyboard.press(isMac ? "Meta+A" : "Control+A");
|
||||||
|
await page.keyboard.press("Backspace");
|
||||||
|
};
|
||||||
|
});
|
||||||
@@ -15,7 +15,7 @@ test.describe("Settings under Scale & Settings", () => {
|
|||||||
const containerNode = await explorer.waitForContainerNode(context.database.id, context.container.id);
|
const containerNode = await explorer.waitForContainerNode(context.database.id, context.container.id);
|
||||||
await containerNode.expand();
|
await containerNode.expand();
|
||||||
|
|
||||||
// Click Scale & Settings and open Scale tab
|
// Click Scale & Settings and open Settings tab
|
||||||
await explorer.openScaleAndSettings(context);
|
await explorer.openScaleAndSettings(context);
|
||||||
const settingsTab = explorer.frame.getByTestId("settings-tab-header/SubSettingsTab");
|
const settingsTab = explorer.frame.getByTestId("settings-tab-header/SubSettingsTab");
|
||||||
await settingsTab.click();
|
await settingsTab.click();
|
||||||
@@ -25,46 +25,86 @@ test.describe("Settings under Scale & Settings", () => {
|
|||||||
await context?.dispose();
|
await context?.dispose();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("Update TTL to On (no default)", async () => {
|
test.describe("Set TTL", () => {
|
||||||
const ttlOnNoDefaultRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-no-default-option" });
|
test("Update TTL to On (no default)", async () => {
|
||||||
await ttlOnNoDefaultRadioButton.click();
|
const ttlOnNoDefaultRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-no-default-option" });
|
||||||
|
await ttlOnNoDefaultRadioButton.click();
|
||||||
|
|
||||||
await explorer.commandBarButton(CommandBarButton.Save).click();
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
await expect(explorer.getConsoleMessage()).toContainText(`Successfully updated container ${context.container.id}`, {
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
timeout: ONE_MINUTE_MS,
|
`Successfully updated container ${context.container.id}`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Update TTL to On (with user entry)", async () => {
|
||||||
|
const ttlOnRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-option" });
|
||||||
|
await ttlOnRadioButton.click();
|
||||||
|
|
||||||
|
// Enter TTL seconds
|
||||||
|
const ttlInput = explorer.frame.getByTestId("ttl-input");
|
||||||
|
await ttlInput.fill("30000");
|
||||||
|
|
||||||
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
|
`Successfully updated container ${context.container.id}`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("Update TTL to Off", async () => {
|
||||||
|
// By default TTL is set to off so we need to first set it to On
|
||||||
|
const ttlOnNoDefaultRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-no-default-option" });
|
||||||
|
await ttlOnNoDefaultRadioButton.click();
|
||||||
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
|
`Successfully updated container ${context.container.id}`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
// Set it to Off
|
||||||
|
const ttlOffRadioButton = explorer.frame.getByRole("radio", { name: "ttl-off-option" });
|
||||||
|
await ttlOffRadioButton.click();
|
||||||
|
|
||||||
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
|
`Successfully updated container ${context.container.id}`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
test("Update TTL to On (with user entry)", async () => {
|
test.describe("Set Geospatial Config", () => {
|
||||||
const ttlOnRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-option" });
|
test("Set Geospatial Config to Geometry then Geography", async () => {
|
||||||
await ttlOnRadioButton.click();
|
const geometryRadioButton = explorer.frame.getByRole("radio", { name: "geometry-option" });
|
||||||
|
await geometryRadioButton.click();
|
||||||
|
|
||||||
// Enter TTL seconds
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
const ttlInput = explorer.frame.getByTestId("ttl-input");
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
await ttlInput.fill("30000");
|
`Successfully updated container ${context.container.id}`,
|
||||||
|
{
|
||||||
|
timeout: ONE_MINUTE_MS,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
await explorer.commandBarButton(CommandBarButton.Save).click();
|
const geographyRadioButton = explorer.frame.getByRole("radio", { name: "geography-option" });
|
||||||
await expect(explorer.getConsoleMessage()).toContainText(`Successfully updated container ${context.container.id}`, {
|
await geographyRadioButton.click();
|
||||||
timeout: ONE_MINUTE_MS,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
test("Update TTL to Off", async () => {
|
await explorer.commandBarButton(CommandBarButton.Save).click();
|
||||||
// By default TTL is set to off so we need to first set it to On
|
await expect(explorer.getConsoleMessage()).toContainText(
|
||||||
const ttlOnNoDefaultRadioButton = explorer.frame.getByRole("radio", { name: "ttl-on-no-default-option" });
|
`Successfully updated container ${context.container.id}`,
|
||||||
await ttlOnNoDefaultRadioButton.click();
|
{
|
||||||
await explorer.commandBarButton(CommandBarButton.Save).click();
|
timeout: ONE_MINUTE_MS,
|
||||||
await expect(explorer.getConsoleMessage()).toContainText(`Successfully updated container ${context.container.id}`, {
|
},
|
||||||
timeout: ONE_MINUTE_MS,
|
);
|
||||||
});
|
|
||||||
|
|
||||||
// Set it to Off
|
|
||||||
const ttlOffRadioButton = explorer.frame.getByRole("radio", { name: "ttl-off-option" });
|
|
||||||
await ttlOffRadioButton.click();
|
|
||||||
|
|
||||||
await explorer.commandBarButton(CommandBarButton.Save).click();
|
|
||||||
await expect(explorer.getConsoleMessage()).toContainText(`Successfully updated container ${context.container.id}`, {
|
|
||||||
timeout: ONE_MINUTE_MS,
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -37,27 +37,35 @@ export interface PartitionKey {
|
|||||||
value: string | null;
|
value: string | null;
|
||||||
}
|
}
|
||||||
|
|
||||||
const partitionCount = 4;
|
export const partitionCount = 4;
|
||||||
|
|
||||||
// If we increase this number, we need to split bulk creates into multiple batches.
|
// If we increase this number, we need to split bulk creates into multiple batches.
|
||||||
// Bulk operations are limited to 100 items per partition.
|
// Bulk operations are limited to 100 items per partition.
|
||||||
const itemsPerPartition = 100;
|
export const itemsPerPartition = 100;
|
||||||
|
|
||||||
function createTestItems(): TestItem[] {
|
function createTestItems(): TestItem[] {
|
||||||
const items: TestItem[] = [];
|
const items: TestItem[] = [];
|
||||||
for (let i = 0; i < partitionCount; i++) {
|
for (let i = 0; i < partitionCount; i++) {
|
||||||
for (let j = 0; j < itemsPerPartition; j++) {
|
for (let j = 0; j < itemsPerPartition; j++) {
|
||||||
const id = crypto.randomBytes(32).toString("base64");
|
const id = createSafeRandomString(32);
|
||||||
items.push({
|
items.push({
|
||||||
id,
|
id,
|
||||||
partitionKey: `partition_${i}`,
|
partitionKey: `partition_${i}`,
|
||||||
randomData: crypto.randomBytes(32).toString("base64"),
|
randomData: createSafeRandomString(32),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return items;
|
return items;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Document IDs cannot contain '/', '\', or '#'
|
||||||
|
function createSafeRandomString(byteLength: number): string {
|
||||||
|
return crypto
|
||||||
|
.randomBytes(byteLength)
|
||||||
|
.toString("base64")
|
||||||
|
.replace(/[/\\#]/g, "_");
|
||||||
|
}
|
||||||
|
|
||||||
export const TestData: TestItem[] = createTestItems();
|
export const TestData: TestItem[] = createTestItems();
|
||||||
|
|
||||||
export class TestContainerContext {
|
export class TestContainerContext {
|
||||||
|
|||||||
Reference in New Issue
Block a user