mirror of
https://github.com/Azure/cosmos-explorer.git
synced 2025-12-23 10:51:30 +00:00
Compare commits
1 Commits
fixed-ts-s
...
tsStrict/f
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
17fa6aff31 |
15
.env.example
15
.env.example
@@ -1 +1,16 @@
|
|||||||
|
PORTAL_RUNNER_USERNAME=
|
||||||
|
PORTAL_RUNNER_PASSWORD=
|
||||||
|
PORTAL_RUNNER_SUBSCRIPTION=
|
||||||
|
PORTAL_RUNNER_RESOURCE_GROUP=
|
||||||
|
PORTAL_RUNNER_DATABASE_ACCOUNT=
|
||||||
|
PORTAL_RUNNER_DATABASE_ACCOUNT_KEY=
|
||||||
|
PORTAL_RUNNER_MONGO_DATABASE_ACCOUNT=
|
||||||
|
PORTAL_RUNNER_MONGO_DATABASE_ACCOUNT_KEY=
|
||||||
|
PORTAL_RUNNER_CONNECTION_STRING=
|
||||||
|
NOTEBOOKS_TEST_RUNNER_TENANT_ID=
|
||||||
|
NOTEBOOKS_TEST_RUNNER_CLIENT_ID=
|
||||||
|
NOTEBOOKS_TEST_RUNNER_CLIENT_SECRET=
|
||||||
|
CASSANDRA_CONNECTION_STRING=
|
||||||
|
MONGO_CONNECTION_STRING=
|
||||||
|
TABLES_CONNECTION_STRING=
|
||||||
DATA_EXPLORER_ENDPOINT=https://localhost:1234/hostedExplorer.html
|
DATA_EXPLORER_ENDPOINT=https://localhost:1234/hostedExplorer.html
|
||||||
154
.eslintignore
154
.eslintignore
@@ -21,8 +21,16 @@ src/Common/MongoUtility.ts
|
|||||||
src/Common/NotificationsClientBase.ts
|
src/Common/NotificationsClientBase.ts
|
||||||
src/Common/QueriesClient.ts
|
src/Common/QueriesClient.ts
|
||||||
src/Common/Splitter.ts
|
src/Common/Splitter.ts
|
||||||
|
src/Config.ts
|
||||||
|
src/Contracts/ActionContracts.ts
|
||||||
|
src/Contracts/DataModels.ts
|
||||||
|
src/Contracts/Diagnostics.ts
|
||||||
|
src/Contracts/ExplorerContracts.ts
|
||||||
|
src/Contracts/Versions.ts
|
||||||
|
src/Contracts/ViewModels.ts
|
||||||
src/Controls/Heatmap/Heatmap.test.ts
|
src/Controls/Heatmap/Heatmap.test.ts
|
||||||
src/Controls/Heatmap/Heatmap.ts
|
src/Controls/Heatmap/Heatmap.ts
|
||||||
|
src/Controls/Heatmap/HeatmapDatatypes.ts
|
||||||
src/Definitions/datatables.d.ts
|
src/Definitions/datatables.d.ts
|
||||||
src/Definitions/gif.d.ts
|
src/Definitions/gif.d.ts
|
||||||
src/Definitions/globals.d.ts
|
src/Definitions/globals.d.ts
|
||||||
@@ -36,14 +44,35 @@ src/Definitions/png.d.ts
|
|||||||
src/Definitions/svg.d.ts
|
src/Definitions/svg.d.ts
|
||||||
src/Explorer/ComponentRegisterer.test.ts
|
src/Explorer/ComponentRegisterer.test.ts
|
||||||
src/Explorer/ComponentRegisterer.ts
|
src/Explorer/ComponentRegisterer.ts
|
||||||
|
src/Explorer/ContextMenuButtonFactory.ts
|
||||||
|
src/Explorer/Controls/CollapsiblePanel/CollapsiblePanelComponent.ts
|
||||||
src/Explorer/Controls/DiffEditor/DiffEditorComponent.ts
|
src/Explorer/Controls/DiffEditor/DiffEditorComponent.ts
|
||||||
|
src/Explorer/Controls/DynamicList/DynamicList.test.ts
|
||||||
|
src/Explorer/Controls/DynamicList/DynamicListComponent.ts
|
||||||
src/Explorer/Controls/Editor/EditorComponent.ts
|
src/Explorer/Controls/Editor/EditorComponent.ts
|
||||||
|
src/Explorer/Controls/ErrorDisplayComponent/ErrorDisplayComponent.ts
|
||||||
|
src/Explorer/Controls/InputTypeahead/InputTypeahead.ts
|
||||||
src/Explorer/Controls/JsonEditor/JsonEditorComponent.ts
|
src/Explorer/Controls/JsonEditor/JsonEditorComponent.ts
|
||||||
|
src/Explorer/Controls/Notebook/NotebookAppMessageHandler.ts
|
||||||
|
src/Explorer/Controls/ThroughputInput/ThroughputInputComponent.ts
|
||||||
|
src/Explorer/Controls/ThroughputInput/ThroughputInputComponentAutoPilotV3.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarAction.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarDisplayable.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarDropDown.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarItem.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarSeperator.ts
|
||||||
|
src/Explorer/Controls/Toolbar/IToolbarToggle.ts
|
||||||
|
src/Explorer/Controls/Toolbar/KeyCodes.ts
|
||||||
|
src/Explorer/Controls/Toolbar/Toolbar.ts
|
||||||
|
src/Explorer/Controls/Toolbar/ToolbarAction.ts
|
||||||
|
src/Explorer/Controls/Toolbar/ToolbarDropDown.ts
|
||||||
|
src/Explorer/Controls/Toolbar/ToolbarToggle.ts
|
||||||
|
src/Explorer/Controls/Toolbar/Utilities.ts
|
||||||
src/Explorer/DataSamples/ContainerSampleGenerator.test.ts
|
src/Explorer/DataSamples/ContainerSampleGenerator.test.ts
|
||||||
src/Explorer/DataSamples/ContainerSampleGenerator.ts
|
src/Explorer/DataSamples/ContainerSampleGenerator.ts
|
||||||
src/Explorer/DataSamples/DataSamplesUtil.test.ts
|
src/Explorer/DataSamples/DataSamplesUtil.test.ts
|
||||||
src/Explorer/DataSamples/DataSamplesUtil.ts
|
src/Explorer/DataSamples/DataSamplesUtil.ts
|
||||||
|
src/Explorer/Explorer.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/ArraysByKeyCache.test.ts
|
src/Explorer/Graph/GraphExplorerComponent/ArraysByKeyCache.test.ts
|
||||||
src/Explorer/Graph/GraphExplorerComponent/ArraysByKeyCache.ts
|
src/Explorer/Graph/GraphExplorerComponent/ArraysByKeyCache.ts
|
||||||
src/Explorer/Graph/GraphExplorerComponent/D3ForceGraph.test.ts
|
src/Explorer/Graph/GraphExplorerComponent/D3ForceGraph.test.ts
|
||||||
@@ -55,6 +84,11 @@ src/Explorer/Graph/GraphExplorerComponent/GremlinClient.test.ts
|
|||||||
src/Explorer/Graph/GraphExplorerComponent/GremlinClient.ts
|
src/Explorer/Graph/GraphExplorerComponent/GremlinClient.ts
|
||||||
src/Explorer/Graph/GraphExplorerComponent/GremlinSimpleClient.test.ts
|
src/Explorer/Graph/GraphExplorerComponent/GremlinSimpleClient.test.ts
|
||||||
src/Explorer/Graph/GraphExplorerComponent/GremlinSimpleClient.ts
|
src/Explorer/Graph/GraphExplorerComponent/GremlinSimpleClient.ts
|
||||||
|
# src/Explorer/Graph/GraphStyleComponent/GraphStyle.test.ts
|
||||||
|
# src/Explorer/Graph/GraphStyleComponent/GraphStyleComponent.ts
|
||||||
|
|
||||||
|
src/Explorer/Menus/CommandBar/CommandBarComponentButtonFactory.test.ts
|
||||||
|
src/Explorer/Menus/CommandBar/CommandBarComponentButtonFactory.ts
|
||||||
src/Explorer/Menus/ContextMenu.ts
|
src/Explorer/Menus/ContextMenu.ts
|
||||||
src/Explorer/MostRecentActivity/MostRecentActivity.ts
|
src/Explorer/MostRecentActivity/MostRecentActivity.ts
|
||||||
src/Explorer/Notebook/NotebookClientV2.ts
|
src/Explorer/Notebook/NotebookClientV2.ts
|
||||||
@@ -71,11 +105,23 @@ src/Explorer/Notebook/NotebookContainerClient.ts
|
|||||||
src/Explorer/Notebook/NotebookContentClient.ts
|
src/Explorer/Notebook/NotebookContentClient.ts
|
||||||
src/Explorer/Notebook/NotebookContentItem.ts
|
src/Explorer/Notebook/NotebookContentItem.ts
|
||||||
src/Explorer/Notebook/NotebookUtil.ts
|
src/Explorer/Notebook/NotebookUtil.ts
|
||||||
|
src/Explorer/OpenActions.test.ts
|
||||||
|
src/Explorer/OpenActions.ts
|
||||||
src/Explorer/OpenActionsStubs.ts
|
src/Explorer/OpenActionsStubs.ts
|
||||||
|
src/Explorer/Panes/AddDatabasePane.ts
|
||||||
|
src/Explorer/Panes/AddDatabasePane.test.ts
|
||||||
|
src/Explorer/Panes/BrowseQueriesPane.ts
|
||||||
|
src/Explorer/Panes/ContextualPaneBase.ts
|
||||||
|
# src/Explorer/Panes/GraphStylingPane.ts
|
||||||
|
# src/Explorer/Panes/NewVertexPane.ts
|
||||||
|
src/Explorer/Panes/RenewAdHocAccessPane.ts
|
||||||
|
src/Explorer/Panes/SetupNotebooksPane.ts
|
||||||
|
src/Explorer/Panes/SwitchDirectoryPane.ts
|
||||||
src/Explorer/Panes/Tables/Validators/EntityPropertyNameValidator.ts
|
src/Explorer/Panes/Tables/Validators/EntityPropertyNameValidator.ts
|
||||||
src/Explorer/Panes/Tables/Validators/EntityPropertyValidationCommon.ts
|
src/Explorer/Panes/Tables/Validators/EntityPropertyValidationCommon.ts
|
||||||
src/Explorer/Panes/Tables/Validators/EntityPropertyValueValidator.ts
|
src/Explorer/Panes/Tables/Validators/EntityPropertyValueValidator.ts
|
||||||
src/Explorer/SplashScreen/SplashScreen.test.ts
|
src/Explorer/SplashScreen/SplashScreen.test.ts
|
||||||
|
src/Explorer/Tables/Constants.ts
|
||||||
src/Explorer/Tables/DataTable/CacheBase.ts
|
src/Explorer/Tables/DataTable/CacheBase.ts
|
||||||
src/Explorer/Tables/DataTable/DataTableBindingManager.ts
|
src/Explorer/Tables/DataTable/DataTableBindingManager.ts
|
||||||
src/Explorer/Tables/DataTable/DataTableBuilder.ts
|
src/Explorer/Tables/DataTable/DataTableBuilder.ts
|
||||||
@@ -92,6 +138,7 @@ src/Explorer/Tables/QueryBuilder/ClauseGroupViewModel.ts
|
|||||||
src/Explorer/Tables/QueryBuilder/CustomTimestampHelper.ts
|
src/Explorer/Tables/QueryBuilder/CustomTimestampHelper.ts
|
||||||
src/Explorer/Tables/QueryBuilder/QueryBuilderViewModel.ts
|
src/Explorer/Tables/QueryBuilder/QueryBuilderViewModel.ts
|
||||||
src/Explorer/Tables/QueryBuilder/QueryClauseViewModel.ts
|
src/Explorer/Tables/QueryBuilder/QueryClauseViewModel.ts
|
||||||
|
src/Explorer/Tables/QueryBuilder/QueryViewModel.ts
|
||||||
src/Explorer/Tables/TableDataClient.ts
|
src/Explorer/Tables/TableDataClient.ts
|
||||||
src/Explorer/Tables/TableEntityProcessor.ts
|
src/Explorer/Tables/TableEntityProcessor.ts
|
||||||
src/Explorer/Tables/Utilities.ts
|
src/Explorer/Tables/Utilities.ts
|
||||||
@@ -101,67 +148,170 @@ src/Explorer/Tabs/DocumentsTab.test.ts
|
|||||||
src/Explorer/Tabs/DocumentsTab.ts
|
src/Explorer/Tabs/DocumentsTab.ts
|
||||||
src/Explorer/Tabs/GraphTab.ts
|
src/Explorer/Tabs/GraphTab.ts
|
||||||
src/Explorer/Tabs/MongoDocumentsTab.ts
|
src/Explorer/Tabs/MongoDocumentsTab.ts
|
||||||
|
src/Explorer/Tabs/MongoQueryTab.ts
|
||||||
|
src/Explorer/Tabs/MongoShellTab.ts
|
||||||
src/Explorer/Tabs/NotebookV2Tab.ts
|
src/Explorer/Tabs/NotebookV2Tab.ts
|
||||||
|
src/Explorer/Tabs/QueryTab.test.ts
|
||||||
|
src/Explorer/Tabs/QueryTab.ts
|
||||||
|
src/Explorer/Tabs/QueryTablesTab.ts
|
||||||
src/Explorer/Tabs/ScriptTabBase.ts
|
src/Explorer/Tabs/ScriptTabBase.ts
|
||||||
|
src/Explorer/Tabs/StoredProcedureTab.ts
|
||||||
src/Explorer/Tabs/TabComponents.ts
|
src/Explorer/Tabs/TabComponents.ts
|
||||||
src/Explorer/Tabs/TabsBase.ts
|
src/Explorer/Tabs/TabsBase.ts
|
||||||
src/Explorer/Tabs/TriggerTab.ts
|
src/Explorer/Tabs/TriggerTab.ts
|
||||||
src/Explorer/Tabs/UserDefinedFunctionTab.ts
|
src/Explorer/Tabs/UserDefinedFunctionTab.ts
|
||||||
src/Explorer/Tree/AccessibleVerticalList.ts
|
src/Explorer/Tree/AccessibleVerticalList.ts
|
||||||
|
src/Explorer/Tree/Collection.test.ts
|
||||||
src/Explorer/Tree/Collection.ts
|
src/Explorer/Tree/Collection.ts
|
||||||
src/Explorer/Tree/ConflictId.ts
|
src/Explorer/Tree/ConflictId.ts
|
||||||
|
src/Explorer/Tree/Database.ts
|
||||||
src/Explorer/Tree/DocumentId.ts
|
src/Explorer/Tree/DocumentId.ts
|
||||||
src/Explorer/Tree/ObjectId.ts
|
src/Explorer/Tree/ObjectId.ts
|
||||||
src/Explorer/Tree/ResourceTokenCollection.ts
|
src/Explorer/Tree/ResourceTokenCollection.ts
|
||||||
src/Explorer/Tree/StoredProcedure.ts
|
src/Explorer/Tree/StoredProcedure.ts
|
||||||
src/Explorer/Tree/TreeComponents.ts
|
src/Explorer/Tree/TreeComponents.ts
|
||||||
src/Explorer/Tree/Trigger.ts
|
src/Explorer/Tree/Trigger.ts
|
||||||
|
src/Explorer/Tree/UserDefinedFunction.ts
|
||||||
src/Explorer/WaitsForTemplateViewModel.ts
|
src/Explorer/WaitsForTemplateViewModel.ts
|
||||||
src/GitHub/GitHubClient.test.ts
|
src/GitHub/GitHubClient.test.ts
|
||||||
src/GitHub/GitHubClient.ts
|
src/GitHub/GitHubClient.ts
|
||||||
src/GitHub/GitHubConnector.ts
|
src/GitHub/GitHubConnector.ts
|
||||||
|
src/GitHub/GitHubContentProvider.test.ts
|
||||||
|
src/GitHub/GitHubContentProvider.ts
|
||||||
src/GitHub/GitHubOAuthService.ts
|
src/GitHub/GitHubOAuthService.ts
|
||||||
|
src/HostedExplorer.ts
|
||||||
src/Index.ts
|
src/Index.ts
|
||||||
src/Juno/JunoClient.test.ts
|
src/Juno/JunoClient.test.ts
|
||||||
src/Juno/JunoClient.ts
|
src/Juno/JunoClient.ts
|
||||||
|
src/Main.ts
|
||||||
|
src/NotebookWorkspaceManager/NotebookWorkspaceManager.ts
|
||||||
|
src/NotebookWorkspaceManager/NotebookWorkspaceResourceProviderMockClients.ts
|
||||||
|
src/Platform/Emulator/DataAccessUtility.ts
|
||||||
|
src/Platform/Emulator/ExplorerFactory.ts
|
||||||
|
src/Platform/Emulator/Main.ts
|
||||||
|
src/Platform/Emulator/NotificationsClient.ts
|
||||||
|
src/Platform/Hosted/ArmResourceUtils.ts
|
||||||
src/Platform/Hosted/Authorization.ts
|
src/Platform/Hosted/Authorization.ts
|
||||||
|
src/Platform/Hosted/DataAccessUtility.ts
|
||||||
|
src/Platform/Hosted/ExplorerFactory.ts
|
||||||
|
src/Platform/Hosted/Helpers/ConnectionStringParser.test.ts
|
||||||
|
src/Platform/Hosted/Main.ts
|
||||||
|
src/Platform/Hosted/Maint.test.ts
|
||||||
|
src/Platform/Hosted/NotificationsClient.ts
|
||||||
|
src/Platform/Portal/DataAccessUtility.ts
|
||||||
|
src/Platform/Portal/ExplorerFactory.ts
|
||||||
|
src/Platform/Portal/Main.ts
|
||||||
|
src/Platform/Portal/NotificationsClient.ts
|
||||||
|
src/PlatformType.ts
|
||||||
src/ReactDevTools.ts
|
src/ReactDevTools.ts
|
||||||
|
src/ResourceProvider/IResourceProviderClient.test.ts
|
||||||
|
src/ResourceProvider/IResourceProviderClient.ts
|
||||||
|
src/ResourceProvider/ResourceProviderClient.ts
|
||||||
|
src/ResourceProvider/ResourceProviderClientFactory.ts
|
||||||
|
src/RouteHandlers/RouteHandler.ts
|
||||||
|
src/RouteHandlers/TabRouteHandler.test.ts
|
||||||
|
src/RouteHandlers/TabRouteHandler.ts
|
||||||
src/Shared/Constants.ts
|
src/Shared/Constants.ts
|
||||||
src/Shared/DefaultExperienceUtility.test.ts
|
src/Shared/DefaultExperienceUtility.test.ts
|
||||||
src/Shared/DefaultExperienceUtility.ts
|
src/Shared/DefaultExperienceUtility.ts
|
||||||
|
src/Shared/ExplorerSettings.ts
|
||||||
|
src/Shared/PriceEstimateCalculator.ts
|
||||||
|
src/Shared/StorageUtility.test.ts
|
||||||
|
src/Shared/StorageUtility.ts
|
||||||
src/Shared/appInsights.ts
|
src/Shared/appInsights.ts
|
||||||
src/SparkClusterManager/ArcadiaResourceManager.ts
|
src/SparkClusterManager/ArcadiaResourceManager.ts
|
||||||
src/SparkClusterManager/SparkClusterManager.ts
|
src/SparkClusterManager/SparkClusterManager.ts
|
||||||
src/Terminal/JupyterLabAppFactory.ts
|
src/Terminal/JupyterLabAppFactory.ts
|
||||||
src/Terminal/NotebookAppContracts.d.ts
|
src/Terminal/NotebookAppContracts.d.ts
|
||||||
|
src/Terminal/index.ts
|
||||||
|
src/TokenProviders/PortalTokenProvider.ts
|
||||||
|
src/TokenProviders/TokenProviderFactory.ts
|
||||||
|
src/Utils/PricingUtils.test.ts
|
||||||
|
src/Utils/QueryUtils.test.ts
|
||||||
src/applyExplorerBindings.ts
|
src/applyExplorerBindings.ts
|
||||||
src/global.d.ts
|
src/global.d.ts
|
||||||
src/setupTests.ts
|
src/setupTests.ts
|
||||||
|
src/Explorer/Controls/AccessibleElement/AccessibleElement.tsx
|
||||||
|
src/Explorer/Controls/Accordion/AccordionComponent.tsx
|
||||||
|
src/Explorer/Controls/AccountSwitch/AccountSwitchComponent.test.tsx
|
||||||
|
src/Explorer/Controls/AccountSwitch/AccountSwitchComponent.tsx
|
||||||
|
src/Explorer/Controls/AccountSwitch/AccountSwitchComponentAdapter.tsx
|
||||||
|
src/Explorer/Controls/Arcadia/ArcadiaMenuPicker.tsx
|
||||||
|
src/Explorer/Controls/CollapsiblePanel/CollapsiblePanel.tsx
|
||||||
|
src/Explorer/Controls/CommandButton/CommandButtonComponent.tsx
|
||||||
|
src/Explorer/Controls/DialogReactComponent/DialogComponent.tsx
|
||||||
|
src/Explorer/Controls/DialogReactComponent/DialogComponentAdapter.tsx
|
||||||
|
src/Explorer/Controls/Directory/DefaultDirectoryDropdownComponent.test.tsx
|
||||||
|
src/Explorer/Controls/Directory/DefaultDirectoryDropdownComponent.tsx
|
||||||
|
src/Explorer/Controls/Directory/DirectoryComponentAdapter.tsx
|
||||||
|
src/Explorer/Controls/Directory/DirectoryListComponent.test.tsx
|
||||||
|
src/Explorer/Controls/Directory/DirectoryListComponent.tsx
|
||||||
|
src/Explorer/Controls/Editor/EditorReact.tsx
|
||||||
src/Explorer/Controls/InputTypeahead/InputTypeaheadComponent.tsx
|
src/Explorer/Controls/InputTypeahead/InputTypeaheadComponent.tsx
|
||||||
src/Explorer/Controls/Notebook/NotebookTerminalComponent.test.tsx
|
src/Explorer/Controls/Notebook/NotebookTerminalComponent.test.tsx
|
||||||
src/Explorer/Controls/Notebook/NotebookTerminalComponent.tsx
|
src/Explorer/Controls/Notebook/NotebookTerminalComponent.tsx
|
||||||
|
src/Explorer/Controls/NotebookViewer/NotebookMetadataComponent.tsx
|
||||||
|
src/NotebookViewer/NotebookViewer.tsx
|
||||||
src/Explorer/Controls/NotebookViewer/NotebookViewerComponent.tsx
|
src/Explorer/Controls/NotebookViewer/NotebookViewerComponent.tsx
|
||||||
|
src/Explorer/Controls/QueriesGridReactComponent/QueriesGridComponent.tsx
|
||||||
|
src/Explorer/Controls/QueriesGridReactComponent/QueriesGridComponentAdapter.tsx
|
||||||
|
src/Explorer/Controls/ResizeSensorReactComponent/ResizeSensorComponent.tsx
|
||||||
|
src/Explorer/Controls/Spark/ClusterSettingsComponent.tsx
|
||||||
|
src/Explorer/Controls/Spark/ClusterSettingsComponentAdapter.tsx
|
||||||
|
src/Explorer/Controls/Tabs/TabComponent.tsx
|
||||||
|
src/Explorer/Controls/TreeComponent/TreeComponent.test.tsx
|
||||||
src/Explorer/Controls/TreeComponent/TreeComponent.tsx
|
src/Explorer/Controls/TreeComponent/TreeComponent.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/EditorNeighborsComponent.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/EditorNodePropertiesComponent.test.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/EditorNodePropertiesComponent.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/GraphExplorer.test.tsx
|
src/Explorer/Graph/GraphExplorerComponent/GraphExplorer.test.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/GraphExplorer.tsx
|
src/Explorer/Graph/GraphExplorerComponent/GraphExplorer.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/GraphExplorerAdapter.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/GraphVizComponent.tsx
|
src/Explorer/Graph/GraphExplorerComponent/GraphVizComponent.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/LeftPaneComponent.tsx
|
src/Explorer/Graph/GraphExplorerComponent/LeftPaneComponent.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/MiddlePaneComponent.tsx
|
src/Explorer/Graph/GraphExplorerComponent/MiddlePaneComponent.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/NodePropertiesComponent.test.tsx
|
src/Explorer/Graph/GraphExplorerComponent/NodePropertiesComponent.test.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/NodePropertiesComponent.tsx
|
src/Explorer/Graph/GraphExplorerComponent/NodePropertiesComponent.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/QueryContainerComponent.tsx
|
||||||
|
src/Explorer/Graph/GraphExplorerComponent/ReadOnlyNeighborsComponent.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/ReadOnlyNodePropertiesComponent.test.tsx
|
src/Explorer/Graph/GraphExplorerComponent/ReadOnlyNodePropertiesComponent.test.tsx
|
||||||
src/Explorer/Graph/GraphExplorerComponent/ReadOnlyNodePropertiesComponent.tsx
|
src/Explorer/Graph/GraphExplorerComponent/ReadOnlyNodePropertiesComponent.tsx
|
||||||
src/Explorer/Menus/CommandBar/CommandBarUtil.tsx
|
src/Explorer/Menus/CommandBar/CommandBarUtil.tsx
|
||||||
|
src/Explorer/Menus/NotificationConsole/NotificationConsoleComponent.test.tsx
|
||||||
|
src/Explorer/Notebook/NotebookComponent/NotebookComponent.tsx
|
||||||
src/Explorer/Notebook/NotebookComponent/NotebookComponentAdapter.tsx
|
src/Explorer/Notebook/NotebookComponent/NotebookComponentAdapter.tsx
|
||||||
src/Explorer/Notebook/NotebookComponent/NotebookComponentBootstrapper.tsx
|
src/Explorer/Notebook/NotebookComponent/NotebookComponentBootstrapper.tsx
|
||||||
src/Explorer/Notebook/NotebookComponent/VirtualCommandBarComponent.tsx
|
src/Explorer/Notebook/NotebookComponent/VirtualCommandBarComponent.tsx
|
||||||
|
src/Explorer/Notebook/NotebookComponent/contents/file/index.tsx
|
||||||
|
src/Explorer/Notebook/NotebookComponent/contents/file/text-file.tsx
|
||||||
src/Explorer/Notebook/NotebookComponent/contents/index.tsx
|
src/Explorer/Notebook/NotebookComponent/contents/index.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/AzureTheme.tsx
|
||||||
src/Explorer/Notebook/NotebookRenderer/NotebookReadOnlyRenderer.tsx
|
src/Explorer/Notebook/NotebookRenderer/NotebookReadOnlyRenderer.tsx
|
||||||
src/Explorer/Notebook/NotebookRenderer/NotebookRenderer.tsx
|
src/Explorer/Notebook/NotebookRenderer/NotebookRenderer.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/Prompt.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/PromptContent.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/StatusBar.test.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/StatusBar.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/Toolbar.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/decorators/CellCreator.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/decorators/CellLabeler.tsx
|
||||||
|
src/Explorer/Notebook/NotebookRenderer/decorators/HoverableCell.tsx
|
||||||
src/Explorer/Notebook/NotebookRenderer/decorators/draggable/index.tsx
|
src/Explorer/Notebook/NotebookRenderer/decorators/draggable/index.tsx
|
||||||
src/Explorer/Notebook/NotebookRenderer/decorators/hijack-scroll/index.tsx
|
src/Explorer/Notebook/NotebookRenderer/decorators/hijack-scroll/index.tsx
|
||||||
src/Explorer/Notebook/NotebookRenderer/decorators/kbd-shortcuts/index.tsx
|
src/Explorer/Notebook/NotebookRenderer/decorators/kbd-shortcuts/index.tsx
|
||||||
src/Explorer/Notebook/temp/inputs/connected-editors/codemirror.tsx
|
src/Explorer/Notebook/temp/inputs/connected-editors/codemirror.tsx
|
||||||
|
src/Explorer/Notebook/temp/inputs/editor.tsx
|
||||||
|
src/Explorer/Notebook/temp/markdown-cell.tsx
|
||||||
|
src/Explorer/Notebook/temp/source.tsx
|
||||||
|
src/Explorer/Notebook/temp/syntax-highlighter/index.tsx
|
||||||
|
src/Explorer/SplashScreen/SplashScreen.tsx
|
||||||
|
src/Explorer/Tabs/GalleryTab.tsx
|
||||||
|
src/Explorer/Tabs/NotebookViewerTab.tsx
|
||||||
|
src/Explorer/Tabs/TerminalTab.tsx
|
||||||
src/Explorer/Tree/ResourceTreeAdapter.tsx
|
src/Explorer/Tree/ResourceTreeAdapter.tsx
|
||||||
|
src/Explorer/Tree/ResourceTreeAdapterForResourceToken.tsx
|
||||||
|
src/GalleryViewer/Cards/GalleryCardComponent.tsx
|
||||||
|
src/GalleryViewer/GalleryViewer.tsx
|
||||||
|
src/GalleryViewer/GalleryViewerComponent.tsx
|
||||||
__mocks__/monaco-editor.ts
|
__mocks__/monaco-editor.ts
|
||||||
src/Explorer/Tree/ResourceTree.tsx
|
src/Explorer/Tree/ResourceTreeAdapterForResourceToken.test.tsx
|
||||||
6
.github/workflows/ci.yml
vendored
6
.github/workflows/ci.yml
vendored
@@ -92,11 +92,11 @@ jobs:
|
|||||||
name: dist
|
name: dist
|
||||||
path: dist/
|
path: dist/
|
||||||
- name: Upload build to preview blob storage
|
- name: Upload build to preview blob storage
|
||||||
run: az storage blob upload-batch -d '$web' -s 'dist' --account-name cosmosexplorerpreview --subscription cosmosdb-portalteam-generaldemo --destination-path "${{github.event.pull_request.head.sha || github.sha}}" --account-key="${PREVIEW_STORAGE_KEY}"
|
run: az storage blob upload-batch -d '$web' -s 'dist' --account-name cosmosexplorerpreview --subscription cosmosdb-portalteam-generaldemo --destination-path "${{github.event.pull_request.head.sha}}" --account-key="${PREVIEW_STORAGE_KEY}"
|
||||||
env:
|
env:
|
||||||
PREVIEW_STORAGE_KEY: ${{ secrets.PREVIEW_STORAGE_KEY }}
|
PREVIEW_STORAGE_KEY: ${{ secrets.PREVIEW_STORAGE_KEY }}
|
||||||
- name: Upload preview config to blob storage
|
- name: Upload preview config to blob storage
|
||||||
run: az storage blob upload -c '$web' -f ./preview/config.json --account-name cosmosexplorerpreview --subscription cosmosdb-portalteam-generaldemo --name "${{github.event.pull_request.head.sha || github.sha}}/config.json" --account-key="${PREVIEW_STORAGE_KEY}"
|
run: az storage blob upload -c '$web' -f ./preview/config.json --account-name cosmosexplorerpreview --subscription cosmosdb-portalteam-generaldemo --name "${{github.event.pull_request.head.sha}}/config.json" --account-key="${PREVIEW_STORAGE_KEY}"
|
||||||
env:
|
env:
|
||||||
PREVIEW_STORAGE_KEY: ${{ secrets.PREVIEW_STORAGE_KEY }}
|
PREVIEW_STORAGE_KEY: ${{ secrets.PREVIEW_STORAGE_KEY }}
|
||||||
endtoendemulator:
|
endtoendemulator:
|
||||||
@@ -143,7 +143,7 @@ jobs:
|
|||||||
- ./test/mongo/container.spec.ts
|
- ./test/mongo/container.spec.ts
|
||||||
- ./test/mongo/container32.spec.ts
|
- ./test/mongo/container32.spec.ts
|
||||||
- ./test/selfServe/selfServeExample.spec.ts
|
- ./test/selfServe/selfServeExample.spec.ts
|
||||||
# - ./test/notebooks/upload.spec.ts // TEMP disabled since notebooks service is off
|
- ./test/notebooks/upload.spec.ts
|
||||||
- ./test/sql/resourceToken.spec.ts
|
- ./test/sql/resourceToken.spec.ts
|
||||||
- ./test/tables/container.spec.ts
|
- ./test/tables/container.spec.ts
|
||||||
steps:
|
steps:
|
||||||
|
|||||||
2
.github/workflows/cleanup.yml
vendored
2
.github/workflows/cleanup.yml
vendored
@@ -7,7 +7,7 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
schedule:
|
schedule:
|
||||||
# Once every hour
|
# Once every hour
|
||||||
- cron: "0 15 * * *"
|
- cron: "0 * * * *"
|
||||||
|
|
||||||
# A workflow run is made up of one or more jobs that can run sequentially or in parallel
|
# A workflow run is made up of one or more jobs that can run sequentially or in parallel
|
||||||
jobs:
|
jobs:
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
{
|
{
|
||||||
"JUNO_ENDPOINT": "https://tools-staging.cosmos.azure.com"
|
"JUNO_ENDPOINT": "https://tools-staging.cosmos.azure.com",
|
||||||
|
"enableSchemaAnalyzer": true
|
||||||
}
|
}
|
||||||
|
Before Width: | Height: | Size: 842 B After Width: | Height: | Size: 842 B |
|
Before Width: | Height: | Size: 371 B After Width: | Height: | Size: 371 B |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.3 KiB |
@@ -9,7 +9,6 @@
|
|||||||
|
|
||||||
@DataExplorerFont: wf_segoe-ui_normal, "Segoe UI", "Segoe WP", Tahoma, Arial, sans-serif;
|
@DataExplorerFont: wf_segoe-ui_normal, "Segoe UI", "Segoe WP", Tahoma, Arial, sans-serif;
|
||||||
@SemiboldFont: "Segoe UI Semibold", "Segoe UI", "Segoe WP", Tahoma, Arial, sans-serif;
|
@SemiboldFont: "Segoe UI Semibold", "Segoe UI", "Segoe WP", Tahoma, Arial, sans-serif;
|
||||||
@GrayScale: "grayscale()";
|
|
||||||
|
|
||||||
@xSmallFontSize: 4px;
|
@xSmallFontSize: 4px;
|
||||||
@smallFontSize: 8px;
|
@smallFontSize: 8px;
|
||||||
|
|||||||
@@ -724,24 +724,45 @@ execute-sproc-params-pane {
|
|||||||
|
|
||||||
.results-container,
|
.results-container,
|
||||||
.errors-container {
|
.errors-container {
|
||||||
|
padding: @MediumSpace 0px 0px @MediumSpace;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
.flex-display();
|
.flex-display();
|
||||||
.flex-direction();
|
.flex-direction();
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
|
||||||
.enterInputParameters {
|
.toggles {
|
||||||
padding: @LargeSpace @MediumSpace;
|
height: @ToggleHeight;
|
||||||
|
width: @ToggleWidth;
|
||||||
|
margin-left: @MediumSpace;
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab {
|
||||||
|
margin-right: @MediumSpace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toggleSwitch {
|
||||||
|
.toggleSwitch();
|
||||||
|
}
|
||||||
|
|
||||||
|
.selectedToggle {
|
||||||
|
.selectedToggle();
|
||||||
|
}
|
||||||
|
|
||||||
|
.unselectedToggle {
|
||||||
|
.unselectedToggle();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
div[role="tabpanel"] {
|
.enterInputParameters {
|
||||||
height: 100%;
|
padding: @LargeSpace @MediumSpace;
|
||||||
padding-bottom: 50px;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.errors-container {
|
.errors-container {
|
||||||
padding-left: (2 * @MediumSpace);
|
padding-left: (2 * @MediumSpace);
|
||||||
padding: @MediumSpace 0px 0px @MediumSpace;
|
|
||||||
.errors-header {
|
.errors-header {
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
font-size: @DefaultFontSize;
|
font-size: @DefaultFontSize;
|
||||||
@@ -3064,14 +3085,3 @@ settings-pane {
|
|||||||
padding-left: @SmallSpace;
|
padding-left: @SmallSpace;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.hiddenMain {
|
|
||||||
display: none;
|
|
||||||
height: 0px;
|
|
||||||
}
|
|
||||||
.spinner {
|
|
||||||
width: 100%;
|
|
||||||
position: absolute;
|
|
||||||
z-index: 1;
|
|
||||||
background: white;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -201,11 +201,3 @@
|
|||||||
.migration:disabled {
|
.migration:disabled {
|
||||||
background-color: #ccc;
|
background-color: #ccc;
|
||||||
}
|
}
|
||||||
|
|
||||||
.trigger-field {
|
|
||||||
width: 40%;
|
|
||||||
margin-top: 10px
|
|
||||||
}
|
|
||||||
.trigger-form {
|
|
||||||
padding: 10px 30px 10px 30px;
|
|
||||||
}
|
|
||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
.dataResourceTree {
|
.dataResourceTree {
|
||||||
margin-left: @MediumSpace;
|
margin-left: @MediumSpace;
|
||||||
overflow: auto;
|
|
||||||
|
|
||||||
.databaseHeader {
|
.databaseHeader {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
@@ -19,10 +18,6 @@
|
|||||||
.notebookHeader {
|
.notebookHeader {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.clickDisabled {
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
268
less/tree.less
268
less/tree.less
@@ -1,270 +1,272 @@
|
|||||||
@import "./Common/Constants";
|
@import "./Common/Constants";
|
||||||
|
|
||||||
|
|
||||||
.resourceTree {
|
.resourceTree {
|
||||||
height: 100%;
|
|
||||||
flex: 0 0 auto;
|
|
||||||
.main {
|
|
||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
flex: 0 0 auto;
|
||||||
|
.main {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.resourceTreeScroll {
|
.resourceTreeScroll {
|
||||||
height: 100%;
|
height: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
padding-right: 10px;
|
padding-right: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.userSelectNone {
|
.userSelectNone {
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
-moz-user-select: none;
|
-moz-user-select: none;
|
||||||
-ms-user-select: none;
|
-ms-user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.treeHovermargin {
|
.treeHovermargin {
|
||||||
margin-left: 16px;
|
margin-left: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.highlight {
|
.highlight {
|
||||||
padding: @SmallSpace 2px;
|
padding: @SmallSpace 2px;
|
||||||
outline: 0;
|
outline: 0;
|
||||||
|
|
||||||
&:hover {
|
&:hover {
|
||||||
.hover();
|
.hover();
|
||||||
}
|
}
|
||||||
|
|
||||||
&:active {
|
&:active {
|
||||||
.active();
|
.active();
|
||||||
}
|
}
|
||||||
|
|
||||||
&:focus {
|
&:focus {
|
||||||
.focus();
|
.focus();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.contextmenushowing {
|
.contextmenushowing {
|
||||||
background-color: #eee;
|
background-color: #EEE;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionstree {
|
.collectionstree {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin-top: @DefaultSpace;
|
margin-top: @DefaultSpace;
|
||||||
|
|
||||||
.databaseList {
|
|
||||||
list-style-type: none;
|
|
||||||
padding-left: 0px;
|
|
||||||
|
|
||||||
.collectionList {
|
.databaseList {
|
||||||
padding-left: (2 * @MediumSpace);
|
list-style-type: none;
|
||||||
|
padding-left: 0px;
|
||||||
|
|
||||||
|
.collectionList {
|
||||||
|
padding-left:(2 * @MediumSpace);
|
||||||
|
}
|
||||||
|
|
||||||
|
.collectionChildList {
|
||||||
|
padding-left: @LargeSpace;
|
||||||
|
}
|
||||||
|
|
||||||
|
.databaseDocuments {
|
||||||
|
padding-left: (5 * @MediumSpace);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionChildList {
|
|
||||||
padding-left: @LargeSpace;
|
|
||||||
}
|
|
||||||
|
|
||||||
.databaseDocuments {
|
|
||||||
padding-left: (5 * @MediumSpace);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.pointerCursor {
|
.pointerCursor {
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.menuEllipsis {
|
.menuEllipsis {
|
||||||
padding-right: 6px;
|
padding-right: 6px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
position: relative;
|
position: relative;
|
||||||
top: -5px;
|
top: -5px;
|
||||||
left: 0px;
|
left: 0px;
|
||||||
float: right;
|
float: right;
|
||||||
display: none;
|
display: none;
|
||||||
padding-left: 6px !important;
|
padding-left: 6px!important;
|
||||||
line-height: @TreeLineHeight;
|
line-height: @TreeLineHeight;
|
||||||
}
|
}
|
||||||
|
|
||||||
.databaseMenu {
|
.databaseMenu {
|
||||||
.flex-display();
|
.flex-display();
|
||||||
}
|
}
|
||||||
|
|
||||||
.databaseMenu:hover .menuEllipsis,
|
.databaseMenu:hover .menuEllipsis,
|
||||||
.databaseMenu:focus .menuEllipsis {
|
.databaseMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.databaseCollChildTextOverflow {
|
.databaseCollChildTextOverflow {
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionMenu {
|
.collectionMenu {
|
||||||
.flex-display();
|
.flex-display();
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionMenu:hover .menuEllipsis,
|
.collectionMenu:hover .menuEllipsis,
|
||||||
.collectionMenu:focus .menuEllipsis {
|
.collectionMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.documentsMenu:hover .menuEllipsis,
|
.documentsMenu:hover .menuEllipsis,
|
||||||
.documentsMenu:focus .menuEllipsis {
|
.documentsMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.treeChildMenu {
|
.treeChildMenu {
|
||||||
display: flex;
|
display: flex;
|
||||||
}
|
}
|
||||||
|
|
||||||
.storedProcedureMenu:hover .menuEllipsis,
|
.storedProcedureMenu:hover .menuEllipsis,
|
||||||
.storedProcedureMenu:focus .menuEllipsis {
|
.storedProcedureMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.childMenu {
|
.childMenu {
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
padding-left: (6 * @MediumSpace);
|
padding-left: (6 * @MediumSpace);
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.storedChildMenu:hover .menuEllipsis,
|
.storedChildMenu:hover .menuEllipsis,
|
||||||
.storedChildMenu:focus .menuEllipsis {
|
.storedChildMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.contextmenu6 {
|
.contextmenu6 {
|
||||||
top: -29px;
|
top: -29px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.userDefinedMenu:hover .contextmenu6 {
|
.userDefinedMenu:hover .contextmenu6 {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.userDefinedchildMenu:hover .menuEllipsis,
|
.userDefinedchildMenu:hover .menuEllipsis,
|
||||||
.userDefinedchildMenu:focus .menuEllipsis {
|
.userDefinedchildMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.triggersMenu:hover .menuEllipsis,
|
.triggersMenu:hover .menuEllipsis,
|
||||||
.triggersMenu:focus .menuEllipsis {
|
.triggersMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.triggersChildMenu:hover .menuEllipsis,
|
.triggersChildMenu:hover .menuEllipsis,
|
||||||
.triggersChildMenu:focus .menuEllipsis {
|
.triggersChildMenu:focus .menuEllipsis {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.databaseId {
|
.databaseId {
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.storedUdfTriggerMenu {
|
.storedUdfTriggerMenu {
|
||||||
padding-left: 0px;
|
padding-left: 0px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionstree img {
|
.collectionstree img {
|
||||||
width: 16px;
|
width: 16px;
|
||||||
height: 16px;
|
height: 16px;
|
||||||
vertical-align: text-top;
|
vertical-align: text-top;
|
||||||
}
|
}
|
||||||
|
|
||||||
img.collectionsTreeCollapseExpand {
|
img.collectionsTreeCollapseExpand {
|
||||||
width: 10px;
|
width: 10px;
|
||||||
height: 10px;
|
height: 10px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
margin-bottom: 5px;
|
margin-bottom: 5px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collapsed::before {
|
.collapsed::before {
|
||||||
content: "\23F5";
|
content: "\23F5";
|
||||||
margin-left: 0px;
|
margin-left: 0px;
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.expanded::before {
|
.expanded::before {
|
||||||
content: "\23F7";
|
content: '\23F7';
|
||||||
margin-left: 0px;
|
margin-left: 0px;
|
||||||
font-size: 15px;
|
font-size: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.collectionMenuChildren {
|
.collectionMenuChildren {
|
||||||
padding-left: 42px;
|
padding-left: 42px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.main-nav {
|
.main-nav {
|
||||||
width: 100vh;
|
width: 100vh;
|
||||||
height: 40px;
|
height: 40px;
|
||||||
background: white;
|
background: white;
|
||||||
transform-origin: left top;
|
transform-origin: left top;
|
||||||
-webkit-transform-origin: left top;
|
-webkit-transform-origin: left top;
|
||||||
-ms-transform-origin: left top;
|
-ms-transform-origin: left top;
|
||||||
transform: rotate(-90deg) translateX(-100%);
|
transform: rotate(-90deg) translateX(-100%);
|
||||||
-webkit-transform: rotate(-90deg) translateX(-100%);
|
-webkit-transform: rotate(-90deg) translateX(-100%);
|
||||||
-ms-transform: rotate(-90deg) translateX(-100%);
|
-ms-transform: rotate(-90deg) translateX(-100%);
|
||||||
border-bottom: 1px solid #ccc;
|
border-bottom: 1px solid #CCC;
|
||||||
}
|
}
|
||||||
|
|
||||||
.main-nav-img {
|
.main-nav-img {
|
||||||
width: 16px;
|
width: 16px;
|
||||||
height: 16px;
|
height: 16px;
|
||||||
margin: -32px 0 0 0;
|
margin: -32px 0 0 0;
|
||||||
transform: rotate(-90deg) translateX(-100%);
|
transform: rotate(-90deg) translateX(-100%);
|
||||||
-webkit-transform: rotate(-90deg) translateX(-100%);
|
-webkit-transform: rotate(-90deg) translateX(-100%);
|
||||||
-ms-transform: rotate(-90deg) translateX(-100%);
|
-ms-transform: rotate(-90deg) translateX(-100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.main-nav-img.main-nav-sub-img {
|
.main-nav-img.main-nav-sub-img {
|
||||||
width: 16px;
|
width: 16px;
|
||||||
height: 16px;
|
height: 16px;
|
||||||
margin: 0px 0px 0 0;
|
margin: 0px 0px 0 0;
|
||||||
transform: rotate(180deg) translateX(0%);
|
transform: rotate(180deg) translateX(0%);
|
||||||
-webkit-transform: rotate(180deg) translateX(0%);
|
-webkit-transform: rotate(180deg) translateX(0%);
|
||||||
-ms-transform: rotate(180deg) translateX(0%);
|
-ms-transform: rotate(180deg) translateX(0%);
|
||||||
position: absolute;
|
position: absolute;
|
||||||
right: -8px;
|
right: -8px;
|
||||||
top: 16px;
|
top: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
ul.nav {
|
ul.nav {
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
margin-top: 0px;
|
margin-top: 0px;
|
||||||
margin-left: 0px;
|
margin-left: 0px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.mini ul.nav li {
|
.mini ul.nav li {
|
||||||
float: right;
|
float: right;
|
||||||
line-height: 25px;
|
line-height: 25px;
|
||||||
height: auto;
|
height: auto;
|
||||||
margin-top: 3px;
|
margin-top: 3px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.spancolchildstyle {
|
.spancolchildstyle {
|
||||||
padding: 4px;
|
padding: 4px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.contextmenubutton {
|
.contextmenubutton {
|
||||||
float: right;
|
float: right;
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.highlight:hover > .contextmenubutton {
|
.highlight:hover>.contextmenubutton {
|
||||||
display: unset;
|
display: unset;
|
||||||
}
|
}
|
||||||
|
|
||||||
.highlight:hover > .contextmenubutton::after {
|
.highlight:hover>.contextmenubutton::after {
|
||||||
content: "\2026";
|
content: "\2026";
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.showEllipsis {
|
.showEllipsis {
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
4802
package-lock.json
generated
4802
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
47
package.json
47
package.json
@@ -22,7 +22,7 @@
|
|||||||
"@nteract/data-explorer": "8.0.3",
|
"@nteract/data-explorer": "8.0.3",
|
||||||
"@nteract/directory-listing": "2.0.6",
|
"@nteract/directory-listing": "2.0.6",
|
||||||
"@nteract/dropdown-menu": "1.0.1",
|
"@nteract/dropdown-menu": "1.0.1",
|
||||||
"@nteract/editor": "10.1.12",
|
"@nteract/editor": "10.1.2",
|
||||||
"@nteract/fixtures": "2.3.0",
|
"@nteract/fixtures": "2.3.0",
|
||||||
"@nteract/iron-icons": "1.0.0",
|
"@nteract/iron-icons": "1.0.0",
|
||||||
"@nteract/jupyter-widgets": "2.0.0",
|
"@nteract/jupyter-widgets": "2.0.0",
|
||||||
@@ -42,15 +42,14 @@
|
|||||||
"@octokit/rest": "17.9.2",
|
"@octokit/rest": "17.9.2",
|
||||||
"@phosphor/widgets": "1.9.3",
|
"@phosphor/widgets": "1.9.3",
|
||||||
"@testing-library/jest-dom": "5.11.9",
|
"@testing-library/jest-dom": "5.11.9",
|
||||||
"@types/lodash": "4.14.171",
|
|
||||||
"@types/mkdirp": "1.0.1",
|
"@types/mkdirp": "1.0.1",
|
||||||
"@types/node-fetch": "2.5.7",
|
"@types/node-fetch": "2.5.7",
|
||||||
"applicationinsights": "1.8.0",
|
"applicationinsights": "1.8.0",
|
||||||
"bootstrap": "3.4.1",
|
"bootstrap": "3.4.1",
|
||||||
"canvas": "file:./canvas",
|
"canvas": "file:./canvas",
|
||||||
"clean-webpack-plugin": "3.0.0",
|
"clean-webpack-plugin": "0.1.19",
|
||||||
"clipboard-copy": "4.0.1",
|
"clipboard-copy": "4.0.1",
|
||||||
"copy-webpack-plugin": "9.0.1",
|
"copy-webpack-plugin": "6.0.2",
|
||||||
"crossroads": "0.12.2",
|
"crossroads": "0.12.2",
|
||||||
"css-element-queries": "1.1.1",
|
"css-element-queries": "1.1.1",
|
||||||
"d3": "6.1.1",
|
"d3": "6.1.1",
|
||||||
@@ -81,16 +80,15 @@
|
|||||||
"plotly.js-cartesian-dist-min": "1.52.3",
|
"plotly.js-cartesian-dist-min": "1.52.3",
|
||||||
"post-robot": "10.0.42",
|
"post-robot": "10.0.42",
|
||||||
"q": "1.5.1",
|
"q": "1.5.1",
|
||||||
"react": "16.14.0",
|
"react": "16.13.1",
|
||||||
"react-animate-height": "2.0.8",
|
"react-animate-height": "2.0.8",
|
||||||
"react-dnd": "14.0.2",
|
"react-dnd": "9.4.0",
|
||||||
"react-dnd-html5-backend": "14.0.0",
|
"react-dnd-html5-backend": "9.4.0",
|
||||||
"react-dom": "16.13.1",
|
"react-dom": "16.13.1",
|
||||||
"react-hotkeys": "2.0.0",
|
"react-hotkeys": "2.0.0",
|
||||||
"react-i18next": "11.8.5",
|
"react-i18next": "11.8.5",
|
||||||
"react-notification-system": "0.2.17",
|
"react-notification-system": "0.2.17",
|
||||||
"react-redux": "7.1.3",
|
"react-redux": "7.1.3",
|
||||||
"react-splitter-layout": "4.0.0",
|
|
||||||
"redux": "4.0.4",
|
"redux": "4.0.4",
|
||||||
"reflect-metadata": "0.1.13",
|
"reflect-metadata": "0.1.13",
|
||||||
"rx-jupyter": "5.5.12",
|
"rx-jupyter": "5.5.12",
|
||||||
@@ -98,7 +96,7 @@
|
|||||||
"sanitize-html": "2.3.3",
|
"sanitize-html": "2.3.3",
|
||||||
"styled-components": "4.3.2",
|
"styled-components": "4.3.2",
|
||||||
"swr": "0.4.0",
|
"swr": "0.4.0",
|
||||||
"terser-webpack-plugin": "5.1.4",
|
"terser-webpack-plugin": "3.1.0",
|
||||||
"underscore": "1.9.1",
|
"underscore": "1.9.1",
|
||||||
"utility-types": "3.10.0",
|
"utility-types": "3.10.0",
|
||||||
"zustand": "3.5.0"
|
"zustand": "3.5.0"
|
||||||
@@ -125,14 +123,12 @@
|
|||||||
"@types/react-dom": "17.0.3",
|
"@types/react-dom": "17.0.3",
|
||||||
"@types/react-notification-system": "0.2.39",
|
"@types/react-notification-system": "0.2.39",
|
||||||
"@types/react-redux": "7.1.7",
|
"@types/react-redux": "7.1.7",
|
||||||
"@types/react-splitter-layout": "3.0.1",
|
|
||||||
"@types/sanitize-html": "1.27.2",
|
"@types/sanitize-html": "1.27.2",
|
||||||
"@types/sinon": "2.3.3",
|
"@types/sinon": "2.3.3",
|
||||||
"@types/styled-components": "5.1.1",
|
"@types/styled-components": "5.1.1",
|
||||||
"@types/underscore": "1.7.36",
|
"@types/underscore": "1.7.36",
|
||||||
"@typescript-eslint/eslint-plugin": "4.22.0",
|
"@typescript-eslint/eslint-plugin": "4.22.0",
|
||||||
"@typescript-eslint/parser": "4.22.0",
|
"@typescript-eslint/parser": "4.22.0",
|
||||||
"@webpack-cli/serve": "1.5.2",
|
|
||||||
"babel-jest": "24.9.0",
|
"babel-jest": "24.9.0",
|
||||||
"babel-loader": "8.1.0",
|
"babel-loader": "8.1.0",
|
||||||
"buffer": "5.1.0",
|
"buffer": "5.1.0",
|
||||||
@@ -154,45 +150,44 @@
|
|||||||
"html-inline-css-webpack-plugin": "1.11.0",
|
"html-inline-css-webpack-plugin": "1.11.0",
|
||||||
"html-loader": "0.5.5",
|
"html-loader": "0.5.5",
|
||||||
"html-loader-jest": "0.2.1",
|
"html-loader-jest": "0.2.1",
|
||||||
"html-webpack-plugin": "5.3.2",
|
"html-webpack-plugin": "4.5.2",
|
||||||
"jest": "26.6.3",
|
"jest": "25.5.4",
|
||||||
"jest-canvas-mock": "2.3.1",
|
"jest-canvas-mock": "2.1.0",
|
||||||
"jest-playwright-preset": "1.5.1",
|
"jest-playwright-preset": "1.5.1",
|
||||||
"jest-trx-results-processor": "0.0.7",
|
"jest-trx-results-processor": "0.0.7",
|
||||||
"less": "3.8.1",
|
"less": "3.8.1",
|
||||||
"less-loader": "4.1.0",
|
"less-loader": "4.1.0",
|
||||||
"less-vars-loader": "1.1.0",
|
"less-vars-loader": "1.1.0",
|
||||||
"mini-css-extract-plugin": "2.1.0",
|
"mini-css-extract-plugin": "0.4.3",
|
||||||
"monaco-editor-webpack-plugin": "1.7.0",
|
"monaco-editor-webpack-plugin": "1.7.0",
|
||||||
"node-fetch": "2.6.1",
|
"node-fetch": "2.6.1",
|
||||||
"playwright": "1.13.0",
|
"playwright": "1.10.0",
|
||||||
"prettier": "2.2.1",
|
"prettier": "2.2.1",
|
||||||
"process": "0.11.10",
|
|
||||||
"raw-loader": "0.5.1",
|
"raw-loader": "0.5.1",
|
||||||
"react-dev-utils": "11.0.4",
|
"react-dev-utils": "11.0.4",
|
||||||
"rimraf": "3.0.0",
|
"rimraf": "3.0.0",
|
||||||
"sinon": "3.2.1",
|
"sinon": "3.2.1",
|
||||||
"style-loader": "0.23.0",
|
"style-loader": "0.23.0",
|
||||||
"ts-loader": "9.2.4",
|
"ts-loader": "6.2.2",
|
||||||
"tslint": "5.11.0",
|
"tslint": "5.11.0",
|
||||||
"tslint-microsoft-contrib": "6.0.0",
|
"tslint-microsoft-contrib": "6.0.0",
|
||||||
"typedoc": "0.20.36",
|
"typedoc": "0.20.36",
|
||||||
"typescript": "4.3.4",
|
"typescript": "4.2.4",
|
||||||
"url-loader": "1.1.1",
|
"url-loader": "1.1.1",
|
||||||
"wait-on": "4.0.2",
|
"wait-on": "4.0.2",
|
||||||
"webpack": "5.47.0",
|
"webpack": "4.46.0",
|
||||||
"webpack-bundle-analyzer": "4.4.2",
|
"webpack-bundle-analyzer": "3.6.1",
|
||||||
"webpack-cli": "4.7.2",
|
"webpack-cli": "3.3.10",
|
||||||
"webpack-dev-server": "3.11.2"
|
"webpack-dev-server": "3.11.0"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "webpack serve --mode development",
|
"start": "node --max-old-space-size=10196 node_modules/webpack-dev-server/bin/webpack-dev-server.js",
|
||||||
"dev": "echo \"WARNING: npm run dev has been deprecated\" && npm run build",
|
"dev": "echo \"WARNING: npm run dev has been deprecated\" && npm run build",
|
||||||
"build:dataExplorer:ci": "npm run build:ci",
|
"build:dataExplorer:ci": "npm run build:ci",
|
||||||
"build": "npm run format:check && npm run lint && npm run compile && npm run compile:strict && npm run pack:prod && npm run copyToConsumers",
|
"build": "npm run format:check && npm run lint && npm run compile && npm run compile:strict && npm run pack:prod && npm run copyToConsumers",
|
||||||
"build:ci": "npm run format:check && npm run lint && npm run compile && npm run compile:strict && npm run pack:fast",
|
"build:ci": "npm run format:check && npm run lint && npm run compile && npm run compile:strict && npm run pack:fast",
|
||||||
"pack:prod": "webpack --mode production",
|
"pack:prod": "node --max_old_space_size=10196 ./node_modules/webpack/bin/webpack.js --mode production",
|
||||||
"pack:fast": "webpack --mode development --progress",
|
"pack:fast": "node --max_old_space_size=10196 ./node_modules/webpack/bin/webpack.js --mode development --progress",
|
||||||
"copyToConsumers": "node copyToConsumers",
|
"copyToConsumers": "node copyToConsumers",
|
||||||
"test": "rimraf coverage && jest",
|
"test": "rimraf coverage && jest",
|
||||||
"test:e2e": "jest -c ./jest.config.playwright.js --detectOpenHandles",
|
"test:e2e": "jest -c ./jest.config.playwright.js --detectOpenHandles",
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
{
|
{
|
||||||
"PROXY_PATH": "/proxy",
|
"PROXY_PATH": "/proxy"
|
||||||
"msalRedirectURI": "https://cosmos-explorer-preview.azurewebsites.net/"
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -62,17 +62,6 @@ app.get("/pull/:pr(\\d+)", (req, res) => {
|
|||||||
})
|
})
|
||||||
.catch(() => res.sendStatus(500));
|
.catch(() => res.sendStatus(500));
|
||||||
});
|
});
|
||||||
app.get("/", (req, res) => {
|
|
||||||
fetch("https://api.github.com/repos/Azure/cosmos-explorer/branches/master")
|
|
||||||
.then((response) => response.json())
|
|
||||||
.then(({ commit: { sha } }) => {
|
|
||||||
const explorer = new URL(
|
|
||||||
"https://cosmos-explorer-preview.azurewebsites.net/commit/" + sha + "/hostedExplorer.html"
|
|
||||||
);
|
|
||||||
return res.redirect(explorer.href);
|
|
||||||
})
|
|
||||||
.catch(() => res.sendStatus(500));
|
|
||||||
});
|
|
||||||
|
|
||||||
app.listen(port, () => {
|
app.listen(port, () => {
|
||||||
console.log(`Example app listening on port: ${port}`);
|
console.log(`Example app listening on port: ${port}`);
|
||||||
|
|||||||
@@ -1,36 +0,0 @@
|
|||||||
import React, { FunctionComponent } from "react";
|
|
||||||
import arrowLeftImg from "../../images/imgarrowlefticon.svg";
|
|
||||||
import { userContext } from "../UserContext";
|
|
||||||
|
|
||||||
export interface CollapsedResourceTreeProps {
|
|
||||||
toggleLeftPaneExpanded: () => void;
|
|
||||||
isLeftPaneExpanded: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const CollapsedResourceTree: FunctionComponent<CollapsedResourceTreeProps> = ({
|
|
||||||
toggleLeftPaneExpanded,
|
|
||||||
isLeftPaneExpanded,
|
|
||||||
}: CollapsedResourceTreeProps): JSX.Element => {
|
|
||||||
return (
|
|
||||||
<div id="mini" className={!isLeftPaneExpanded ? "mini toggle-mini" : "hiddenMain"}>
|
|
||||||
<div className="main-nav nav">
|
|
||||||
<ul className="nav">
|
|
||||||
<li
|
|
||||||
className="resourceTreeCollapse"
|
|
||||||
id="collapseToggleLeftPaneButton"
|
|
||||||
role="button"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label="Expand Tree"
|
|
||||||
>
|
|
||||||
<span className="leftarrowCollapsed" onClick={toggleLeftPaneExpanded}>
|
|
||||||
<img className="arrowCollapsed" src={arrowLeftImg} alt="Expand" />
|
|
||||||
</span>
|
|
||||||
<span className="collectionCollapsed" onClick={toggleLeftPaneExpanded}>
|
|
||||||
<span>{userContext.apiType} API</span>
|
|
||||||
</span>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -94,8 +94,7 @@ export class Flights {
|
|||||||
public static readonly MongoIndexEditor = "mongoindexeditor";
|
public static readonly MongoIndexEditor = "mongoindexeditor";
|
||||||
public static readonly MongoIndexing = "mongoindexing";
|
public static readonly MongoIndexing = "mongoindexing";
|
||||||
public static readonly AutoscaleTest = "autoscaletest";
|
public static readonly AutoscaleTest = "autoscaletest";
|
||||||
public static readonly PartitionKeyTest = "partitionkeytest";
|
public static readonly SchemaAnalyzer = "schemaanalyzer";
|
||||||
public static readonly PKPartitionKeyTest = "pkpartitionkeytest";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class AfecFeatures {
|
export class AfecFeatures {
|
||||||
@@ -159,6 +158,16 @@ export class DocumentsGridMetrics {
|
|||||||
public static DocumentEditorMaxWidthRatio: number = 0.4;
|
public static DocumentEditorMaxWidthRatio: number = 0.4;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export class ExplorerMetrics {
|
||||||
|
public static SplitterMinWidth: number = 240;
|
||||||
|
public static SplitterMaxWidth: number = 400;
|
||||||
|
public static CollapsedResourceTreeWidth: number = 36;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class SplitterMetrics {
|
||||||
|
public static CollapsedPositionLeft: number = ExplorerMetrics.CollapsedResourceTreeWidth;
|
||||||
|
}
|
||||||
|
|
||||||
export class Areas {
|
export class Areas {
|
||||||
public static ResourceTree: string = "Resource Tree";
|
public static ResourceTree: string = "Resource Tree";
|
||||||
public static ContextualPane: string = "Contextual Pane";
|
public static ContextualPane: string = "Contextual Pane";
|
||||||
@@ -350,11 +359,6 @@ export class Notebook {
|
|||||||
public static readonly kernelRestartInitialDelayMs = 1000;
|
public static readonly kernelRestartInitialDelayMs = 1000;
|
||||||
public static readonly kernelRestartMaxDelayMs = 20000;
|
public static readonly kernelRestartMaxDelayMs = 20000;
|
||||||
public static readonly autoSaveIntervalMs = 120000;
|
public static readonly autoSaveIntervalMs = 120000;
|
||||||
public static readonly temporarilyDownMsg = "Notebooks is currently not available. We are working on it.";
|
|
||||||
public static readonly mongoShellTemporarilyDownMsg =
|
|
||||||
"We have identified an issue with the Mongo Shell and it is unavailable right now. We are actively working on the mitigation.";
|
|
||||||
public static readonly cassandraShellTemporarilyDownMsg =
|
|
||||||
"We have identified an issue with the Cassandra Shell and it is unavailable right now. We are actively working on the mitigation.";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export class SparkLibrary {
|
export class SparkLibrary {
|
||||||
|
|||||||
@@ -83,7 +83,7 @@ export function client(): Cosmos.CosmosClient {
|
|||||||
if (_client) return _client;
|
if (_client) return _client;
|
||||||
const options: Cosmos.CosmosClientOptions = {
|
const options: Cosmos.CosmosClientOptions = {
|
||||||
endpoint: endpoint() || "https://cosmos.azure.com", // CosmosClient gets upset if we pass a bad URL. This should never actually get called
|
endpoint: endpoint() || "https://cosmos.azure.com", // CosmosClient gets upset if we pass a bad URL. This should never actually get called
|
||||||
key: userContext.masterKey,
|
...(!userContext.features.enableAadDataPlane && { key: userContext.masterKey }),
|
||||||
tokenProvider,
|
tokenProvider,
|
||||||
connectionPolicy: {
|
connectionPolicy: {
|
||||||
enableEndpointDiscovery: false,
|
enableEndpointDiscovery: false,
|
||||||
|
|||||||
@@ -1,17 +0,0 @@
|
|||||||
import { userContext } from "../UserContext";
|
|
||||||
|
|
||||||
function isVirtualNetworkFilterEnabled() {
|
|
||||||
return userContext.databaseAccount?.properties?.isVirtualNetworkFilterEnabled;
|
|
||||||
}
|
|
||||||
|
|
||||||
function isIpRulesEnabled() {
|
|
||||||
return userContext.databaseAccount?.properties?.ipRules?.length > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
function isPrivateEndpointConnectionsEnabled() {
|
|
||||||
return userContext.databaseAccount?.properties?.privateEndpointConnections?.length > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
export function isPublicInternetAccessAllowed(): boolean {
|
|
||||||
return !isVirtualNetworkFilterEnabled() && !isIpRulesEnabled() && !isPrivateEndpointConnectionsEnabled();
|
|
||||||
}
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as ExplorerSettings from "../Shared/ExplorerSettings";
|
|
||||||
import { LocalStorageUtility, StorageKey } from "../Shared/StorageUtility";
|
|
||||||
import * as HeadersUtility from "./HeadersUtility";
|
import * as HeadersUtility from "./HeadersUtility";
|
||||||
|
import { ExplorerSettings } from "../Shared/ExplorerSettings";
|
||||||
|
import { LocalStorageUtility, StorageKey } from "../Shared/StorageUtility";
|
||||||
|
|
||||||
describe("Headers Utility", () => {
|
describe("Headers Utility", () => {
|
||||||
describe("shouldEnableCrossPartitionKeyForResourceWithPartitionKey()", () => {
|
describe("shouldEnableCrossPartitionKeyForResourceWithPartitionKey()", () => {
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
jest.mock("./MessageHandler");
|
jest.mock("./MessageHandler");
|
||||||
|
import { LogEntryLevel } from "../Contracts/Diagnostics";
|
||||||
import * as Logger from "./Logger";
|
import * as Logger from "./Logger";
|
||||||
|
import { MessageTypes } from "../Contracts/ExplorerContracts";
|
||||||
import { sendMessage } from "./MessageHandler";
|
import { sendMessage } from "./MessageHandler";
|
||||||
|
|
||||||
describe("Logger", () => {
|
describe("Logger", () => {
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { Collection } from "../Contracts/ViewModels";
|
|||||||
import DocumentId from "../Explorer/Tree/DocumentId";
|
import DocumentId from "../Explorer/Tree/DocumentId";
|
||||||
import { updateUserContext } from "../UserContext";
|
import { updateUserContext } from "../UserContext";
|
||||||
import { deleteDocument, getEndpoint, queryDocuments, readDocument, updateDocument } from "./MongoProxyClient";
|
import { deleteDocument, getEndpoint, queryDocuments, readDocument, updateDocument } from "./MongoProxyClient";
|
||||||
|
jest.mock("../ResourceProvider/ResourceProviderClient.ts");
|
||||||
|
|
||||||
const databaseId = "testDB";
|
const databaseId = "testDB";
|
||||||
|
|
||||||
|
|||||||
@@ -111,7 +111,7 @@ export function queryDocuments(
|
|||||||
headers: response.headers,
|
headers: response.headers,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
await errorHandling(response, "querying documents", params);
|
errorHandling(response, "querying documents", params);
|
||||||
return undefined;
|
return undefined;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -153,11 +153,11 @@ export function readDocument(
|
|||||||
),
|
),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(async (response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
return await errorHandling(response, "reading document", params);
|
return errorHandling(response, "reading document", params);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -192,11 +192,11 @@ export function createDocument(
|
|||||||
...authHeaders(),
|
...authHeaders(),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(async (response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
return await errorHandling(response, "creating document", params);
|
return errorHandling(response, "creating document", params);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -238,11 +238,11 @@ export function updateDocument(
|
|||||||
[CosmosSDKConstants.HttpHeaders.PartitionKey]: JSON.stringify(documentId.partitionKeyHeader()),
|
[CosmosSDKConstants.HttpHeaders.PartitionKey]: JSON.stringify(documentId.partitionKeyHeader()),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(async (response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
return await errorHandling(response, "updating document", params);
|
return errorHandling(response, "updating document", params);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -278,11 +278,11 @@ export function deleteDocument(databaseId: string, collection: Collection, docum
|
|||||||
[CosmosSDKConstants.HttpHeaders.PartitionKey]: JSON.stringify(documentId.partitionKeyHeader()),
|
[CosmosSDKConstants.HttpHeaders.PartitionKey]: JSON.stringify(documentId.partitionKeyHeader()),
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then(async (response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
return await errorHandling(response, "deleting document", params);
|
return errorHandling(response, "deleting document", params);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -325,11 +325,11 @@ export function createMongoCollectionWithProxy(
|
|||||||
},
|
},
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
.then(async (response) => {
|
.then((response) => {
|
||||||
if (response.ok) {
|
if (response.ok) {
|
||||||
return response.json();
|
return response.json();
|
||||||
}
|
}
|
||||||
return await errorHandling(response, "creating collection", mongoParams);
|
return errorHandling(response, "creating collection", mongoParams);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,17 +1,19 @@
|
|||||||
|
import { ItemDefinition, QueryIterator, Resource } from "@azure/cosmos";
|
||||||
import * as _ from "underscore";
|
import * as _ from "underscore";
|
||||||
import * as DataModels from "../Contracts/DataModels";
|
import * as DataModels from "../Contracts/DataModels";
|
||||||
import * as ViewModels from "../Contracts/ViewModels";
|
import * as ViewModels from "../Contracts/ViewModels";
|
||||||
import Explorer from "../Explorer/Explorer";
|
import Explorer from "../Explorer/Explorer";
|
||||||
import DocumentsTab from "../Explorer/Tabs/DocumentsTab";
|
import DocumentsTab from "../Explorer/Tabs/DocumentsTab";
|
||||||
import DocumentId from "../Explorer/Tree/DocumentId";
|
import DocumentId from "../Explorer/Tree/DocumentId";
|
||||||
import { useDatabases } from "../Explorer/useDatabases";
|
|
||||||
import { userContext } from "../UserContext";
|
import { userContext } from "../UserContext";
|
||||||
import * as NotificationConsoleUtils from "../Utils/NotificationConsoleUtils";
|
import * as NotificationConsoleUtils from "../Utils/NotificationConsoleUtils";
|
||||||
|
import * as QueryUtils from "../Utils/QueryUtils";
|
||||||
import { BackendDefaults, HttpStatusCodes, SavedQueries } from "./Constants";
|
import { BackendDefaults, HttpStatusCodes, SavedQueries } from "./Constants";
|
||||||
import { createCollection } from "./dataAccess/createCollection";
|
import { createCollection } from "./dataAccess/createCollection";
|
||||||
import { createDocument } from "./dataAccess/createDocument";
|
import { createDocument } from "./dataAccess/createDocument";
|
||||||
import { deleteDocument } from "./dataAccess/deleteDocument";
|
import { deleteDocument } from "./dataAccess/deleteDocument";
|
||||||
import { queryDocuments } from "./dataAccess/queryDocuments";
|
import { queryDocuments } from "./dataAccess/queryDocuments";
|
||||||
|
import { queryDocumentsPage } from "./dataAccess/queryDocumentsPage";
|
||||||
import { handleError } from "./ErrorHandlingUtils";
|
import { handleError } from "./ErrorHandlingUtils";
|
||||||
|
|
||||||
export class QueriesClient {
|
export class QueriesClient {
|
||||||
@@ -98,35 +100,45 @@ export class QueriesClient {
|
|||||||
|
|
||||||
const options: any = { enableCrossPartitionQuery: true };
|
const options: any = { enableCrossPartitionQuery: true };
|
||||||
const clearMessage = NotificationConsoleUtils.logConsoleProgress("Fetching saved queries");
|
const clearMessage = NotificationConsoleUtils.logConsoleProgress("Fetching saved queries");
|
||||||
const results = await queryDocuments(
|
const queryIterator: QueryIterator<ItemDefinition & Resource> = queryDocuments(
|
||||||
SavedQueries.DatabaseName,
|
SavedQueries.DatabaseName,
|
||||||
SavedQueries.CollectionName,
|
SavedQueries.CollectionName,
|
||||||
this.fetchQueriesQuery(),
|
this.fetchQueriesQuery(),
|
||||||
options
|
options
|
||||||
).fetchAll();
|
);
|
||||||
|
const fetchQueries = async (firstItemIndex: number): Promise<ViewModels.QueryResults> =>
|
||||||
let queries: DataModels.Query[] = _.map(results.resources, (document: DataModels.Query) => {
|
await queryDocumentsPage(queriesCollection.id(), queryIterator, firstItemIndex);
|
||||||
if (!document) {
|
return QueryUtils.queryAllPages(fetchQueries)
|
||||||
return undefined;
|
.then(
|
||||||
}
|
(results: ViewModels.QueryResults) => {
|
||||||
const { id, resourceId, query, queryName } = document;
|
let queries: DataModels.Query[] = _.map(results.documents, (document: DataModels.Query) => {
|
||||||
const parsedQuery: DataModels.Query = {
|
if (!document) {
|
||||||
resourceId: resourceId,
|
return undefined;
|
||||||
queryName: queryName,
|
}
|
||||||
query: query,
|
const { id, resourceId, query, queryName } = document;
|
||||||
id: id,
|
const parsedQuery: DataModels.Query = {
|
||||||
};
|
resourceId: resourceId,
|
||||||
try {
|
queryName: queryName,
|
||||||
this.validateQuery(parsedQuery);
|
query: query,
|
||||||
return parsedQuery;
|
id: id,
|
||||||
} catch (error) {
|
};
|
||||||
return undefined;
|
try {
|
||||||
}
|
this.validateQuery(parsedQuery);
|
||||||
});
|
return parsedQuery;
|
||||||
queries = _.reject(queries, (parsedQuery: DataModels.Query) => !parsedQuery);
|
} catch (error) {
|
||||||
NotificationConsoleUtils.logConsoleInfo("Successfully fetched saved queries");
|
return undefined;
|
||||||
clearMessage();
|
}
|
||||||
return queries;
|
});
|
||||||
|
queries = _.reject(queries, (parsedQuery: DataModels.Query) => !parsedQuery);
|
||||||
|
NotificationConsoleUtils.logConsoleInfo("Successfully fetched saved queries");
|
||||||
|
return Promise.resolve(queries);
|
||||||
|
},
|
||||||
|
(error: any) => {
|
||||||
|
handleError(error, "getSavedQueries", "Failed to fetch saved queries");
|
||||||
|
return Promise.reject(error);
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.finally(() => clearMessage());
|
||||||
}
|
}
|
||||||
|
|
||||||
public async deleteQuery(query: DataModels.Query): Promise<void> {
|
public async deleteQuery(query: DataModels.Query): Promise<void> {
|
||||||
@@ -177,7 +189,7 @@ export class QueriesClient {
|
|||||||
|
|
||||||
private findQueriesCollection(): ViewModels.Collection {
|
private findQueriesCollection(): ViewModels.Collection {
|
||||||
const queriesDatabase: ViewModels.Database = _.find(
|
const queriesDatabase: ViewModels.Database = _.find(
|
||||||
useDatabases.getState().databases,
|
this.container.databases(),
|
||||||
(database: ViewModels.Database) => database.id() === SavedQueries.DatabaseName
|
(database: ViewModels.Database) => database.id() === SavedQueries.DatabaseName
|
||||||
);
|
);
|
||||||
if (!queriesDatabase) {
|
if (!queriesDatabase) {
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
import React, { FunctionComponent } from "react";
|
|
||||||
import arrowLeftImg from "../../images/imgarrowlefticon.svg";
|
|
||||||
import refreshImg from "../../images/refresh-cosmos.svg";
|
|
||||||
import { AuthType } from "../AuthType";
|
|
||||||
import Explorer from "../Explorer/Explorer";
|
|
||||||
import { ResourceTokenTree } from "../Explorer/Tree/ResourceTokenTree";
|
|
||||||
import { ResourceTree } from "../Explorer/Tree/ResourceTree";
|
|
||||||
import { userContext } from "../UserContext";
|
|
||||||
|
|
||||||
export interface ResourceTreeContainerProps {
|
|
||||||
toggleLeftPaneExpanded: () => void;
|
|
||||||
isLeftPaneExpanded: boolean;
|
|
||||||
container: Explorer;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const ResourceTreeContainer: FunctionComponent<ResourceTreeContainerProps> = ({
|
|
||||||
toggleLeftPaneExpanded,
|
|
||||||
isLeftPaneExpanded,
|
|
||||||
container,
|
|
||||||
}: ResourceTreeContainerProps): JSX.Element => {
|
|
||||||
return (
|
|
||||||
<div id="main" className={isLeftPaneExpanded ? "main" : "hiddenMain"}>
|
|
||||||
{/* Collections Window - - Start */}
|
|
||||||
<div id="mainslide" className="flexContainer">
|
|
||||||
{/* Collections Window Title/Command Bar - Start */}
|
|
||||||
<div className="collectiontitle">
|
|
||||||
<div className="coltitle">
|
|
||||||
<span className="titlepadcol">{userContext.apiType} API</span>
|
|
||||||
<div className="float-right">
|
|
||||||
<span
|
|
||||||
className="padimgcolrefresh"
|
|
||||||
data-test="refreshTree"
|
|
||||||
role="button"
|
|
||||||
data-bind="click: onRefreshResourcesClick, clickBubble: false, event: { keypress: onRefreshDatabasesKeyPress }"
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label="Refresh tree"
|
|
||||||
title="Refresh tree"
|
|
||||||
>
|
|
||||||
<img className="refreshcol" src={refreshImg} alt="Refresh Tree" />
|
|
||||||
</span>
|
|
||||||
<span
|
|
||||||
className="padimgcolrefresh1"
|
|
||||||
id="expandToggleLeftPaneButton"
|
|
||||||
role="button"
|
|
||||||
onClick={toggleLeftPaneExpanded}
|
|
||||||
tabIndex={0}
|
|
||||||
aria-label="Collapse Tree"
|
|
||||||
title="Collapse Tree"
|
|
||||||
>
|
|
||||||
<img className="refreshcol1" src={arrowLeftImg} alt="Hide" />
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{userContext.authType === AuthType.ResourceToken ? (
|
|
||||||
<ResourceTokenTree />
|
|
||||||
) : userContext.features.enableKoResourceTree ? (
|
|
||||||
<div style={{ overflowY: "auto" }} data-bind="react:resourceTree" />
|
|
||||||
) : (
|
|
||||||
<ResourceTree container={container} />
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
{/* Collections Window - End */}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,3 +1,7 @@
|
|||||||
|
import * as ko from "knockout";
|
||||||
|
|
||||||
|
import { SplitterMetrics } from "./Constants";
|
||||||
|
|
||||||
export enum SplitterDirection {
|
export enum SplitterDirection {
|
||||||
Horizontal = "horizontal",
|
Horizontal = "horizontal",
|
||||||
Vertical = "vertical",
|
Vertical = "vertical",
|
||||||
@@ -24,12 +28,14 @@ export class Splitter {
|
|||||||
public lastX!: number;
|
public lastX!: number;
|
||||||
public lastWidth!: number;
|
public lastWidth!: number;
|
||||||
|
|
||||||
|
private isCollapsed: ko.Observable<boolean>;
|
||||||
private bounds: SplitterBounds;
|
private bounds: SplitterBounds;
|
||||||
private direction: SplitterDirection;
|
private direction: SplitterDirection;
|
||||||
|
|
||||||
constructor(options: SplitterOptions) {
|
constructor(options: SplitterOptions) {
|
||||||
this.splitterId = options.splitterId;
|
this.splitterId = options.splitterId;
|
||||||
this.leftSideId = options.leftId;
|
this.leftSideId = options.leftId;
|
||||||
|
this.isCollapsed = ko.observable<boolean>(false);
|
||||||
this.bounds = options.bounds;
|
this.bounds = options.bounds;
|
||||||
this.direction = options.direction;
|
this.direction = options.direction;
|
||||||
this.initialize();
|
this.initialize();
|
||||||
@@ -77,4 +83,23 @@ export class Splitter {
|
|||||||
};
|
};
|
||||||
|
|
||||||
private onResizeStop: JQueryUI.ResizableEvent = () => $("iframe").css("pointer-events", "auto");
|
private onResizeStop: JQueryUI.ResizableEvent = () => $("iframe").css("pointer-events", "auto");
|
||||||
|
|
||||||
|
public collapseLeft() {
|
||||||
|
this.lastX = $(this.splitter).position().left;
|
||||||
|
this.lastWidth = $(this.leftSide).width();
|
||||||
|
$(this.splitter).css("left", SplitterMetrics.CollapsedPositionLeft);
|
||||||
|
$(this.leftSide).css("width", "");
|
||||||
|
$(this.leftSide).resizable("option", "disabled", true).removeClass("ui-resizable-disabled"); // remove class so splitter is visible
|
||||||
|
$(this.splitter).removeClass("ui-resizable-e");
|
||||||
|
this.isCollapsed(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
public expandLeft() {
|
||||||
|
$(this.splitter).addClass("ui-resizable-e");
|
||||||
|
$(this.leftSide).css("width", this.lastWidth);
|
||||||
|
$(this.splitter).css("left", this.lastX);
|
||||||
|
$(this.splitter).css("left", ""); // this ensures the splitter's position is not fixed and enables movement during resizing
|
||||||
|
$(this.leftSide).resizable("enable");
|
||||||
|
this.isCollapsed(false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,7 @@
|
|||||||
jest.mock("../../Utils/arm/request");
|
jest.mock("../../Utils/arm/request");
|
||||||
jest.mock("../CosmosClient");
|
jest.mock("../CosmosClient");
|
||||||
import ko from "knockout";
|
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import { CreateCollectionParams, DatabaseAccount } from "../../Contracts/DataModels";
|
import { CreateCollectionParams, DatabaseAccount } from "../../Contracts/DataModels";
|
||||||
import { Database } from "../../Contracts/ViewModels";
|
|
||||||
import { useDatabases } from "../../Explorer/useDatabases";
|
|
||||||
import { updateUserContext } from "../../UserContext";
|
import { updateUserContext } from "../../UserContext";
|
||||||
import { armRequest } from "../../Utils/arm/request";
|
import { armRequest } from "../../Utils/arm/request";
|
||||||
import { client } from "../CosmosClient";
|
import { client } from "../CosmosClient";
|
||||||
@@ -26,15 +23,6 @@ describe("createCollection", () => {
|
|||||||
} as DatabaseAccount,
|
} as DatabaseAccount,
|
||||||
apiType: "SQL",
|
apiType: "SQL",
|
||||||
});
|
});
|
||||||
useDatabases.setState({
|
|
||||||
databases: [
|
|
||||||
{
|
|
||||||
id: ko.observable("testDatabase"),
|
|
||||||
loadCollections: () => undefined,
|
|
||||||
collections: ko.observableArray([]),
|
|
||||||
} as Database,
|
|
||||||
],
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should call ARM if logged in with AAD", async () => {
|
it("should call ARM if logged in with AAD", async () => {
|
||||||
|
|||||||
@@ -4,16 +4,20 @@ import { ContainerRequest } from "@azure/cosmos/dist-esm/client/Container/Contai
|
|||||||
import { DatabaseRequest } from "@azure/cosmos/dist-esm/client/Database/DatabaseRequest";
|
import { DatabaseRequest } from "@azure/cosmos/dist-esm/client/Database/DatabaseRequest";
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
import { useDatabases } from "../../Explorer/useDatabases";
|
|
||||||
import { Action, ActionModifiers } from "../../Shared/Telemetry/TelemetryConstants";
|
import { Action, ActionModifiers } from "../../Shared/Telemetry/TelemetryConstants";
|
||||||
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
import * as TelemetryProcessor from "../../Shared/Telemetry/TelemetryProcessor";
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
import { getCollectionName } from "../../Utils/APITypeUtils";
|
import {
|
||||||
import { createUpdateCassandraTable } from "../../Utils/arm/generatedClients/cosmos/cassandraResources";
|
createUpdateCassandraTable,
|
||||||
import { createUpdateGremlinGraph } from "../../Utils/arm/generatedClients/cosmos/gremlinResources";
|
getCassandraTable,
|
||||||
import { createUpdateMongoDBCollection } from "../../Utils/arm/generatedClients/cosmos/mongoDBResources";
|
} from "../../Utils/arm/generatedClients/cosmos/cassandraResources";
|
||||||
import { createUpdateSqlContainer } from "../../Utils/arm/generatedClients/cosmos/sqlResources";
|
import { createUpdateGremlinGraph, getGremlinGraph } from "../../Utils/arm/generatedClients/cosmos/gremlinResources";
|
||||||
import { createUpdateTable } from "../../Utils/arm/generatedClients/cosmos/tableResources";
|
import {
|
||||||
|
createUpdateMongoDBCollection,
|
||||||
|
getMongoDBCollection,
|
||||||
|
} from "../../Utils/arm/generatedClients/cosmos/mongoDBResources";
|
||||||
|
import { createUpdateSqlContainer, getSqlContainer } from "../../Utils/arm/generatedClients/cosmos/sqlResources";
|
||||||
|
import { createUpdateTable, getTable } from "../../Utils/arm/generatedClients/cosmos/tableResources";
|
||||||
import * as ARMTypes from "../../Utils/arm/generatedClients/cosmos/types";
|
import * as ARMTypes from "../../Utils/arm/generatedClients/cosmos/types";
|
||||||
import { logConsoleInfo, logConsoleProgress } from "../../Utils/NotificationConsoleUtils";
|
import { logConsoleInfo, logConsoleProgress } from "../../Utils/NotificationConsoleUtils";
|
||||||
import { client } from "../CosmosClient";
|
import { client } from "../CosmosClient";
|
||||||
@@ -55,16 +59,6 @@ export const createCollection = async (params: DataModels.CreateCollectionParams
|
|||||||
};
|
};
|
||||||
|
|
||||||
const createCollectionWithARM = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createCollectionWithARM = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
if (!params.createNewDatabase) {
|
|
||||||
const isValid = await useDatabases.getState().validateCollectionId(params.databaseId, params.collectionId);
|
|
||||||
if (!isValid) {
|
|
||||||
const collectionName = getCollectionName().toLocaleLowerCase();
|
|
||||||
throw new Error(
|
|
||||||
`Create ${collectionName} failed: ${collectionName} with id ${params.collectionId} already exists`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const { apiType } = userContext;
|
const { apiType } = userContext;
|
||||||
switch (apiType) {
|
switch (apiType) {
|
||||||
case "SQL":
|
case "SQL":
|
||||||
@@ -83,6 +77,23 @@ const createCollectionWithARM = async (params: DataModels.CreateCollectionParams
|
|||||||
};
|
};
|
||||||
|
|
||||||
const createSqlContainer = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createSqlContainer = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
|
try {
|
||||||
|
const getResponse = await getSqlContainer(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId,
|
||||||
|
params.collectionId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create container failed: container with id ${params.collectionId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
||||||
const resource: ARMTypes.SqlContainerResource = {
|
const resource: ARMTypes.SqlContainerResource = {
|
||||||
id: params.collectionId,
|
id: params.collectionId,
|
||||||
@@ -120,6 +131,23 @@ const createSqlContainer = async (params: DataModels.CreateCollectionParams): Pr
|
|||||||
|
|
||||||
const createMongoCollection = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createMongoCollection = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
const mongoWildcardIndexOnAllFields: ARMTypes.MongoIndex[] = [{ key: { keys: ["$**"] } }, { key: { keys: ["_id"] } }];
|
const mongoWildcardIndexOnAllFields: ARMTypes.MongoIndex[] = [{ key: { keys: ["$**"] } }, { key: { keys: ["_id"] } }];
|
||||||
|
try {
|
||||||
|
const getResponse = await getMongoDBCollection(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId,
|
||||||
|
params.collectionId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create collection failed: collection with id ${params.collectionId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
||||||
const resource: ARMTypes.MongoDBCollectionResource = {
|
const resource: ARMTypes.MongoDBCollectionResource = {
|
||||||
id: params.collectionId,
|
id: params.collectionId,
|
||||||
@@ -161,6 +189,23 @@ const createMongoCollection = async (params: DataModels.CreateCollectionParams):
|
|||||||
};
|
};
|
||||||
|
|
||||||
const createCassandraTable = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createCassandraTable = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
|
try {
|
||||||
|
const getResponse = await getCassandraTable(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId,
|
||||||
|
params.collectionId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create table failed: table with id ${params.collectionId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
||||||
const resource: ARMTypes.CassandraTableResource = {
|
const resource: ARMTypes.CassandraTableResource = {
|
||||||
id: params.collectionId,
|
id: params.collectionId,
|
||||||
@@ -188,6 +233,23 @@ const createCassandraTable = async (params: DataModels.CreateCollectionParams):
|
|||||||
};
|
};
|
||||||
|
|
||||||
const createGraph = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createGraph = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
|
try {
|
||||||
|
const getResponse = await getGremlinGraph(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId,
|
||||||
|
params.collectionId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create graph failed: graph with id ${params.collectionId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
||||||
const resource: ARMTypes.GremlinGraphResource = {
|
const resource: ARMTypes.GremlinGraphResource = {
|
||||||
id: params.collectionId,
|
id: params.collectionId,
|
||||||
@@ -222,6 +284,22 @@ const createGraph = async (params: DataModels.CreateCollectionParams): Promise<D
|
|||||||
};
|
};
|
||||||
|
|
||||||
const createTable = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
const createTable = async (params: DataModels.CreateCollectionParams): Promise<DataModels.Collection> => {
|
||||||
|
try {
|
||||||
|
const getResponse = await getTable(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.collectionId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create table failed: table with id ${params.collectionId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
const options: ARMTypes.CreateUpdateOptions = constructRpOptions(params);
|
||||||
const resource: ARMTypes.TableResource = {
|
const resource: ARMTypes.TableResource = {
|
||||||
id: params.collectionId,
|
id: params.collectionId,
|
||||||
|
|||||||
@@ -2,13 +2,20 @@ import { DatabaseResponse } from "@azure/cosmos";
|
|||||||
import { DatabaseRequest } from "@azure/cosmos/dist-esm/client/Database/DatabaseRequest";
|
import { DatabaseRequest } from "@azure/cosmos/dist-esm/client/Database/DatabaseRequest";
|
||||||
import { AuthType } from "../../AuthType";
|
import { AuthType } from "../../AuthType";
|
||||||
import * as DataModels from "../../Contracts/DataModels";
|
import * as DataModels from "../../Contracts/DataModels";
|
||||||
import { useDatabases } from "../../Explorer/useDatabases";
|
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
import { getDatabaseName } from "../../Utils/APITypeUtils";
|
import {
|
||||||
import { createUpdateCassandraKeyspace } from "../../Utils/arm/generatedClients/cosmos/cassandraResources";
|
createUpdateCassandraKeyspace,
|
||||||
import { createUpdateGremlinDatabase } from "../../Utils/arm/generatedClients/cosmos/gremlinResources";
|
getCassandraKeyspace,
|
||||||
import { createUpdateMongoDBDatabase } from "../../Utils/arm/generatedClients/cosmos/mongoDBResources";
|
} from "../../Utils/arm/generatedClients/cosmos/cassandraResources";
|
||||||
import { createUpdateSqlDatabase } from "../../Utils/arm/generatedClients/cosmos/sqlResources";
|
import {
|
||||||
|
createUpdateGremlinDatabase,
|
||||||
|
getGremlinDatabase,
|
||||||
|
} from "../../Utils/arm/generatedClients/cosmos/gremlinResources";
|
||||||
|
import {
|
||||||
|
createUpdateMongoDBDatabase,
|
||||||
|
getMongoDBDatabase,
|
||||||
|
} from "../../Utils/arm/generatedClients/cosmos/mongoDBResources";
|
||||||
|
import { createUpdateSqlDatabase, getSqlDatabase } from "../../Utils/arm/generatedClients/cosmos/sqlResources";
|
||||||
import {
|
import {
|
||||||
CassandraKeyspaceCreateUpdateParameters,
|
CassandraKeyspaceCreateUpdateParameters,
|
||||||
CreateUpdateOptions,
|
CreateUpdateOptions,
|
||||||
@@ -41,11 +48,6 @@ export async function createDatabase(params: DataModels.CreateDatabaseParams): P
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createDatabaseWithARM(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
async function createDatabaseWithARM(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
||||||
if (!useDatabases.getState().validateDatabaseId(params.databaseId)) {
|
|
||||||
const databaseName = getDatabaseName().toLocaleLowerCase();
|
|
||||||
throw new Error(`Create ${databaseName} failed: ${databaseName} with id ${params.databaseId} already exists`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const { apiType } = userContext;
|
const { apiType } = userContext;
|
||||||
|
|
||||||
switch (apiType) {
|
switch (apiType) {
|
||||||
@@ -63,6 +65,22 @@ async function createDatabaseWithARM(params: DataModels.CreateDatabaseParams): P
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createSqlDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
async function createSqlDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
||||||
|
try {
|
||||||
|
const getResponse = await getSqlDatabase(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create database failed: database with id ${params.databaseId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: CreateUpdateOptions = constructRpOptions(params);
|
const options: CreateUpdateOptions = constructRpOptions(params);
|
||||||
const rpPayload: SqlDatabaseCreateUpdateParameters = {
|
const rpPayload: SqlDatabaseCreateUpdateParameters = {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -83,6 +101,22 @@ async function createSqlDatabase(params: DataModels.CreateDatabaseParams): Promi
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createMongoDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
async function createMongoDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
||||||
|
try {
|
||||||
|
const getResponse = await getMongoDBDatabase(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create database failed: database with id ${params.databaseId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: CreateUpdateOptions = constructRpOptions(params);
|
const options: CreateUpdateOptions = constructRpOptions(params);
|
||||||
const rpPayload: MongoDBDatabaseCreateUpdateParameters = {
|
const rpPayload: MongoDBDatabaseCreateUpdateParameters = {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -103,6 +137,22 @@ async function createMongoDatabase(params: DataModels.CreateDatabaseParams): Pro
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createCassandraKeyspace(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
async function createCassandraKeyspace(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
||||||
|
try {
|
||||||
|
const getResponse = await getCassandraKeyspace(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create database failed: database with id ${params.databaseId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: CreateUpdateOptions = constructRpOptions(params);
|
const options: CreateUpdateOptions = constructRpOptions(params);
|
||||||
const rpPayload: CassandraKeyspaceCreateUpdateParameters = {
|
const rpPayload: CassandraKeyspaceCreateUpdateParameters = {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -123,6 +173,22 @@ async function createCassandraKeyspace(params: DataModels.CreateDatabaseParams):
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function createGremlineDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
async function createGremlineDatabase(params: DataModels.CreateDatabaseParams): Promise<DataModels.Database> {
|
||||||
|
try {
|
||||||
|
const getResponse = await getGremlinDatabase(
|
||||||
|
userContext.subscriptionId,
|
||||||
|
userContext.resourceGroup,
|
||||||
|
userContext.databaseAccount.name,
|
||||||
|
params.databaseId
|
||||||
|
);
|
||||||
|
if (getResponse?.properties?.resource) {
|
||||||
|
throw new Error(`Create database failed: database with id ${params.databaseId} already exists`);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
if (error.code !== "NotFound") {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const options: CreateUpdateOptions = constructRpOptions(params);
|
const options: CreateUpdateOptions = constructRpOptions(params);
|
||||||
const rpPayload: GremlinDatabaseCreateUpdateParameters = {
|
const rpPayload: GremlinDatabaseCreateUpdateParameters = {
|
||||||
properties: {
|
properties: {
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ export interface ConfigContext {
|
|||||||
hostedExplorerURL: string;
|
hostedExplorerURL: string;
|
||||||
armAPIVersion?: string;
|
armAPIVersion?: string;
|
||||||
allowedJunoOrigins: string[];
|
allowedJunoOrigins: string[];
|
||||||
msalRedirectURI?: string;
|
enableSchemaAnalyzer: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Default configuration
|
// Default configuration
|
||||||
@@ -62,6 +62,7 @@ let configContext: Readonly<ConfigContext> = {
|
|||||||
"https://tools-staging.cosmos.azure.com",
|
"https://tools-staging.cosmos.azure.com",
|
||||||
"https://localhost",
|
"https://localhost",
|
||||||
],
|
],
|
||||||
|
enableSchemaAnalyzer: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export function resetConfigContext(): void {
|
export function resetConfigContext(): void {
|
||||||
@@ -118,14 +119,6 @@ export async function initializeConfiguration(): Promise<ConfigContext> {
|
|||||||
const armAPIVersion = params.get("armAPIVersion") || "";
|
const armAPIVersion = params.get("armAPIVersion") || "";
|
||||||
updateConfigContext({ armAPIVersion });
|
updateConfigContext({ armAPIVersion });
|
||||||
}
|
}
|
||||||
if (params.has("armEndpoint")) {
|
|
||||||
const ARM_ENDPOINT = params.get("armEndpoint") || "";
|
|
||||||
updateConfigContext({ ARM_ENDPOINT });
|
|
||||||
}
|
|
||||||
if (params.has("aadEndpoint")) {
|
|
||||||
const AAD_ENDPOINT = params.get("aadEndpoint") || "";
|
|
||||||
updateConfigContext({ AAD_ENDPOINT });
|
|
||||||
}
|
|
||||||
if (params.has("platform")) {
|
if (params.has("platform")) {
|
||||||
const platform = params.get("platform");
|
const platform = params.get("platform");
|
||||||
switch (platform) {
|
switch (platform) {
|
||||||
|
|||||||
@@ -9,7 +9,6 @@ export interface DatabaseAccount {
|
|||||||
|
|
||||||
export interface DatabaseAccountExtendedProperties {
|
export interface DatabaseAccountExtendedProperties {
|
||||||
documentEndpoint?: string;
|
documentEndpoint?: string;
|
||||||
disableLocalAuth?: boolean;
|
|
||||||
tableEndpoint?: string;
|
tableEndpoint?: string;
|
||||||
gremlinEndpoint?: string;
|
gremlinEndpoint?: string;
|
||||||
cassandraEndpoint?: string;
|
cassandraEndpoint?: string;
|
||||||
@@ -23,7 +22,6 @@ export interface DatabaseAccountExtendedProperties {
|
|||||||
enableAnalyticalStorage?: boolean;
|
enableAnalyticalStorage?: boolean;
|
||||||
isVirtualNetworkFilterEnabled?: boolean;
|
isVirtualNetworkFilterEnabled?: boolean;
|
||||||
ipRules?: IpRule[];
|
ipRules?: IpRule[];
|
||||||
privateEndpointConnections?: unknown[];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DatabaseAccountResponseLocation {
|
export interface DatabaseAccountResponseLocation {
|
||||||
@@ -393,6 +391,16 @@ export interface GeospatialConfig {
|
|||||||
type: string;
|
type: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface GatewayDatabaseAccount {
|
||||||
|
MediaLink: string;
|
||||||
|
DatabasesLink: string;
|
||||||
|
MaxMediaStorageUsageInMB: number;
|
||||||
|
CurrentMediaStorageUsageInMB: number;
|
||||||
|
EnableMultipleWriteLocations?: boolean;
|
||||||
|
WritableLocations: RegionEndpoint[];
|
||||||
|
ReadableLocations: RegionEndpoint[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface RegionEndpoint {
|
export interface RegionEndpoint {
|
||||||
name: string;
|
name: string;
|
||||||
documentAccountEndpoint: string;
|
documentAccountEndpoint: string;
|
||||||
@@ -413,6 +421,13 @@ export interface AccountKeys {
|
|||||||
secondaryReadonlyMasterKey: string;
|
secondaryReadonlyMasterKey: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface AfecFeature {
|
||||||
|
id: string;
|
||||||
|
name: string;
|
||||||
|
properties: { state: string };
|
||||||
|
type: string;
|
||||||
|
}
|
||||||
|
|
||||||
export interface OperationStatus {
|
export interface OperationStatus {
|
||||||
status: string;
|
status: string;
|
||||||
id?: string;
|
id?: string;
|
||||||
@@ -492,6 +507,91 @@ export interface MongoParameters extends RpParameters {
|
|||||||
analyticalStorageTtl?: number;
|
analyticalStorageTtl?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface SparkClusterLibrary {
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface Library extends SparkClusterLibrary {
|
||||||
|
properties: {
|
||||||
|
kind: "Jar";
|
||||||
|
source: {
|
||||||
|
kind: "HttpsUri";
|
||||||
|
uri: string;
|
||||||
|
libraryFileName: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LibraryFeedResponse {
|
||||||
|
value: Library[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArmResource {
|
||||||
|
id: string;
|
||||||
|
location: string;
|
||||||
|
name: string;
|
||||||
|
type: string;
|
||||||
|
tags: { [key: string]: string };
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArcadiaWorkspaceIdentity {
|
||||||
|
type: string;
|
||||||
|
principalId: string;
|
||||||
|
tenantId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArcadiaWorkspaceProperties {
|
||||||
|
managedResourceGroupName: string;
|
||||||
|
provisioningState: string;
|
||||||
|
sqlAdministratorLogin: string;
|
||||||
|
connectivityEndpoints: {
|
||||||
|
artifacts: string;
|
||||||
|
dev: string;
|
||||||
|
spark: string;
|
||||||
|
sql: string;
|
||||||
|
web: string;
|
||||||
|
};
|
||||||
|
defaultDataLakeStorage: {
|
||||||
|
accountUrl: string;
|
||||||
|
filesystem: string;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArcadiaWorkspaceFeedResponse {
|
||||||
|
value: ArcadiaWorkspace[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArcadiaWorkspace extends ArmResource {
|
||||||
|
identity: ArcadiaWorkspaceIdentity;
|
||||||
|
properties: ArcadiaWorkspaceProperties;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SparkPoolFeedResponse {
|
||||||
|
value: SparkPool[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SparkPoolProperties {
|
||||||
|
creationDate: string;
|
||||||
|
sparkVersion: string;
|
||||||
|
nodeCount: number;
|
||||||
|
nodeSize: string;
|
||||||
|
nodeSizeFamily: string;
|
||||||
|
provisioningState: string;
|
||||||
|
autoScale: {
|
||||||
|
enabled: boolean;
|
||||||
|
minNodeCount: number;
|
||||||
|
maxNodeCount: number;
|
||||||
|
};
|
||||||
|
autoPause: {
|
||||||
|
enabled: boolean;
|
||||||
|
delayInMinutes: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SparkPool extends ArmResource {
|
||||||
|
properties: SparkPoolProperties;
|
||||||
|
}
|
||||||
|
|
||||||
export interface MemoryUsageInfo {
|
export interface MemoryUsageInfo {
|
||||||
freeKB: number;
|
freeKB: number;
|
||||||
totalKB: number;
|
totalKB: number;
|
||||||
|
|||||||
@@ -5,8 +5,9 @@ import {
|
|||||||
TriggerDefinition,
|
TriggerDefinition,
|
||||||
UserDefinedFunctionDefinition,
|
UserDefinedFunctionDefinition,
|
||||||
} from "@azure/cosmos";
|
} from "@azure/cosmos";
|
||||||
|
import { CommandButtonComponentProps } from "../Explorer/Controls/CommandButton/CommandButtonComponent";
|
||||||
import Explorer from "../Explorer/Explorer";
|
import Explorer from "../Explorer/Explorer";
|
||||||
import { ConsoleData } from "../Explorer/Menus/NotificationConsole/ConsoleData";
|
import { ConsoleData } from "../Explorer/Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
import { CassandraTableKey, CassandraTableKeys } from "../Explorer/Tables/TableDataClient";
|
import { CassandraTableKey, CassandraTableKeys } from "../Explorer/Tables/TableDataClient";
|
||||||
import ConflictId from "../Explorer/Tree/ConflictId";
|
import ConflictId from "../Explorer/Tree/ConflictId";
|
||||||
import DocumentId from "../Explorer/Tree/DocumentId";
|
import DocumentId from "../Explorer/Tree/DocumentId";
|
||||||
@@ -14,7 +15,6 @@ import StoredProcedure from "../Explorer/Tree/StoredProcedure";
|
|||||||
import Trigger from "../Explorer/Tree/Trigger";
|
import Trigger from "../Explorer/Tree/Trigger";
|
||||||
import UserDefinedFunction from "../Explorer/Tree/UserDefinedFunction";
|
import UserDefinedFunction from "../Explorer/Tree/UserDefinedFunction";
|
||||||
import { SelfServeType } from "../SelfServe/SelfServeUtils";
|
import { SelfServeType } from "../SelfServe/SelfServeUtils";
|
||||||
import { CollectionCreationDefaults } from "../UserContext";
|
|
||||||
import { SqlTriggerResource } from "../Utils/arm/generatedClients/cosmos/types";
|
import { SqlTriggerResource } from "../Utils/arm/generatedClients/cosmos/types";
|
||||||
import * as DataModels from "./DataModels";
|
import * as DataModels from "./DataModels";
|
||||||
import { SubscriptionType } from "./SubscriptionType";
|
import { SubscriptionType } from "./SubscriptionType";
|
||||||
@@ -89,6 +89,7 @@ export interface Database extends TreeNode {
|
|||||||
|
|
||||||
selectedSubnodeKind: ko.Observable<CollectionTabKind>;
|
selectedSubnodeKind: ko.Observable<CollectionTabKind>;
|
||||||
|
|
||||||
|
selectDatabase(): void;
|
||||||
expandDatabase(): Promise<void>;
|
expandDatabase(): Promise<void>;
|
||||||
collapseDatabase(): void;
|
collapseDatabase(): void;
|
||||||
|
|
||||||
@@ -274,6 +275,8 @@ export interface TabOptions {
|
|||||||
tabKind: CollectionTabKind;
|
tabKind: CollectionTabKind;
|
||||||
title: string;
|
title: string;
|
||||||
tabPath: string;
|
tabPath: string;
|
||||||
|
hashLocation: string;
|
||||||
|
onUpdateTabsButtons: (buttons: CommandButtonComponentProps[]) => void;
|
||||||
isTabsContentExpanded?: ko.Observable<boolean>;
|
isTabsContentExpanded?: ko.Observable<boolean>;
|
||||||
onLoadStartKey?: number;
|
onLoadStartKey?: number;
|
||||||
|
|
||||||
@@ -284,7 +287,6 @@ export interface TabOptions {
|
|||||||
rid?: string;
|
rid?: string;
|
||||||
node?: TreeNode;
|
node?: TreeNode;
|
||||||
theme?: string;
|
theme?: string;
|
||||||
index?: number;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DocumentsTabOptions extends TabOptions {
|
export interface DocumentsTabOptions extends TabOptions {
|
||||||
@@ -409,6 +411,25 @@ export interface SelfServeFrameInputs {
|
|||||||
flights?: readonly string[];
|
flights?: readonly string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface CollectionCreationDefaults {
|
||||||
|
storage: string;
|
||||||
|
throughput: ThroughputDefaults;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ThroughputDefaults {
|
||||||
|
fixed: number;
|
||||||
|
unlimited:
|
||||||
|
| number
|
||||||
|
| {
|
||||||
|
collectionThreshold: number;
|
||||||
|
lessThanOrEqualToThreshold: number;
|
||||||
|
greatThanThreshold: number;
|
||||||
|
};
|
||||||
|
unlimitedmax: number;
|
||||||
|
unlimitedmin: number;
|
||||||
|
shared: number;
|
||||||
|
}
|
||||||
|
|
||||||
export class MonacoEditorSettings {
|
export class MonacoEditorSettings {
|
||||||
public readonly language: string;
|
public readonly language: string;
|
||||||
public readonly readOnly: boolean;
|
public readonly readOnly: boolean;
|
||||||
|
|||||||
@@ -22,8 +22,8 @@ describe("The Heatmap Control", () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let heatmap: Heatmap;
|
let heatmap: Heatmap;
|
||||||
const theme: PortalTheme = 1;
|
let theme: PortalTheme = 1;
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
const divElement: string = `<div id="${Heatmap.elementId}"></div>`;
|
||||||
|
|
||||||
describe("drawHeatmap rendering", () => {
|
describe("drawHeatmap rendering", () => {
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
@@ -100,7 +100,7 @@ describe("iframe rendering when there is no data", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("should show a no data message with a dark theme", () => {
|
it("should show a no data message with a dark theme", () => {
|
||||||
const data = {
|
let data = {
|
||||||
data: {
|
data: {
|
||||||
signature: "pcIframe",
|
signature: "pcIframe",
|
||||||
data: {
|
data: {
|
||||||
@@ -111,7 +111,7 @@ describe("iframe rendering when there is no data", () => {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
const divElement: string = `<div id="${Heatmap.elementId}"></div>`;
|
||||||
document.body.innerHTML = divElement;
|
document.body.innerHTML = divElement;
|
||||||
|
|
||||||
handleMessage(data as MessageEvent);
|
handleMessage(data as MessageEvent);
|
||||||
@@ -120,7 +120,7 @@ describe("iframe rendering when there is no data", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("should show a no data message with a white theme", () => {
|
it("should show a no data message with a white theme", () => {
|
||||||
const data = {
|
let data = {
|
||||||
data: {
|
data: {
|
||||||
signature: "pcIframe",
|
signature: "pcIframe",
|
||||||
data: {
|
data: {
|
||||||
@@ -131,7 +131,7 @@ describe("iframe rendering when there is no data", () => {
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const divElement = `<div id="${Heatmap.elementId}"></div>`;
|
const divElement: string = `<div id="${Heatmap.elementId}"></div>`;
|
||||||
document.body.innerHTML = divElement;
|
document.body.innerHTML = divElement;
|
||||||
|
|
||||||
handleMessage(data as MessageEvent);
|
handleMessage(data as MessageEvent);
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ export class Heatmap {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _getFontStyles(size: number = StyleConstants.MediumFontSize, color = "#838383"): FontSettings {
|
private _getFontStyles(size: number = StyleConstants.MediumFontSize, color: string = "#838383"): FontSettings {
|
||||||
return {
|
return {
|
||||||
family: StyleConstants.DataExplorerFont,
|
family: StyleConstants.DataExplorerFont,
|
||||||
size,
|
size,
|
||||||
@@ -78,9 +78,9 @@ export class Heatmap {
|
|||||||
// go thru all rows and create 2d matrix for heatmap...
|
// go thru all rows and create 2d matrix for heatmap...
|
||||||
for (let i = 0; i < rows.length; i++) {
|
for (let i = 0; i < rows.length; i++) {
|
||||||
output.yAxisPoints.push(rows[i]);
|
output.yAxisPoints.push(rows[i]);
|
||||||
const dataPoints: number[] = [];
|
let dataPoints: number[] = [];
|
||||||
for (let a = 0; a < output.xAxisPoints.length; a++) {
|
for (let a = 0; a < output.xAxisPoints.length; a++) {
|
||||||
const row: PartitionTimeStampToData = data[rows[i]];
|
let row: PartitionTimeStampToData = data[rows[i]];
|
||||||
dataPoints.push(row[output.xAxisPoints[a]]["Normalized Throughput"]);
|
dataPoints.push(row[output.xAxisPoints[a]]["Normalized Throughput"]);
|
||||||
}
|
}
|
||||||
output.dataPoints.push(dataPoints);
|
output.dataPoints.push(dataPoints);
|
||||||
@@ -193,7 +193,7 @@ export class Heatmap {
|
|||||||
this._getLayoutSettings(),
|
this._getLayoutSettings(),
|
||||||
this._getChartDisplaySettings()
|
this._getChartDisplaySettings()
|
||||||
);
|
);
|
||||||
const plotDiv: any = document.getElementById(Heatmap.elementId);
|
let plotDiv: any = document.getElementById(Heatmap.elementId);
|
||||||
plotDiv.on("plotly_click", (data: any) => {
|
plotDiv.on("plotly_click", (data: any) => {
|
||||||
let timeSelected: string = data.points[0].x;
|
let timeSelected: string = data.points[0].x;
|
||||||
timeSelected = timeSelected.replace(" ", "T");
|
timeSelected = timeSelected.replace(" ", "T");
|
||||||
@@ -205,7 +205,7 @@ export class Heatmap {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const output = [];
|
let output = [];
|
||||||
for (let i = 0; i < this._chartData.dataPoints.length; i++) {
|
for (let i = 0; i < this._chartData.dataPoints.length; i++) {
|
||||||
output.push(this._chartData.dataPoints[i][xAxisIndex]);
|
output.push(this._chartData.dataPoints[i][xAxisIndex]);
|
||||||
}
|
}
|
||||||
|
|||||||
14
src/Explorer/ComponentRegisterer.test.ts
Normal file
14
src/Explorer/ComponentRegisterer.test.ts
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
jest.mock("monaco-editor");
|
||||||
|
|
||||||
|
import * as ko from "knockout";
|
||||||
|
import "./ComponentRegisterer";
|
||||||
|
|
||||||
|
describe("Component Registerer", () => {
|
||||||
|
it("should register json-editor component", () => {
|
||||||
|
expect(ko.components.isRegistered("json-editor")).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should register dynamic-list component", () => {
|
||||||
|
expect(ko.components.isRegistered("dynamic-list")).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -1,8 +1,12 @@
|
|||||||
import * as ko from "knockout";
|
import * as ko from "knockout";
|
||||||
import { DiffEditorComponent } from "./Controls/DiffEditor/DiffEditorComponent";
|
import { DiffEditorComponent } from "./Controls/DiffEditor/DiffEditorComponent";
|
||||||
|
import { DynamicListComponent } from "./Controls/DynamicList/DynamicListComponent";
|
||||||
import { EditorComponent } from "./Controls/Editor/EditorComponent";
|
import { EditorComponent } from "./Controls/Editor/EditorComponent";
|
||||||
import { JsonEditorComponent } from "./Controls/JsonEditor/JsonEditorComponent";
|
import { JsonEditorComponent } from "./Controls/JsonEditor/JsonEditorComponent";
|
||||||
|
import { ThroughputInputComponentAutoPilotV3 } from "./Controls/ThroughputInput/ThroughputInputComponentAutoPilotV3";
|
||||||
|
|
||||||
ko.components.register("editor", new EditorComponent());
|
ko.components.register("editor", new EditorComponent());
|
||||||
ko.components.register("json-editor", new JsonEditorComponent());
|
ko.components.register("json-editor", new JsonEditorComponent());
|
||||||
ko.components.register("diff-editor", new DiffEditorComponent());
|
ko.components.register("diff-editor", new DiffEditorComponent());
|
||||||
|
ko.components.register("dynamic-list", DynamicListComponent);
|
||||||
|
ko.components.register("throughput-input-autopilot-v3", ThroughputInputComponentAutoPilotV3);
|
||||||
|
|||||||
172
src/Explorer/ContextMenuButtonFactory.ts
Normal file
172
src/Explorer/ContextMenuButtonFactory.ts
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
import AddCollectionIcon from "../../images/AddCollection.svg";
|
||||||
|
import AddSqlQueryIcon from "../../images/AddSqlQuery_16x16.svg";
|
||||||
|
import AddStoredProcedureIcon from "../../images/AddStoredProcedure.svg";
|
||||||
|
import AddTriggerIcon from "../../images/AddTrigger.svg";
|
||||||
|
import AddUdfIcon from "../../images/AddUdf.svg";
|
||||||
|
import DeleteCollectionIcon from "../../images/DeleteCollection.svg";
|
||||||
|
import DeleteDatabaseIcon from "../../images/DeleteDatabase.svg";
|
||||||
|
import DeleteSprocIcon from "../../images/DeleteSproc.svg";
|
||||||
|
import DeleteTriggerIcon from "../../images/DeleteTrigger.svg";
|
||||||
|
import DeleteUDFIcon from "../../images/DeleteUDF.svg";
|
||||||
|
import HostedTerminalIcon from "../../images/Hosted-Terminal.svg";
|
||||||
|
import * as ViewModels from "../Contracts/ViewModels";
|
||||||
|
import { userContext } from "../UserContext";
|
||||||
|
import { TreeNodeMenuItem } from "./Controls/TreeComponent/TreeComponent";
|
||||||
|
import Explorer from "./Explorer";
|
||||||
|
import StoredProcedure from "./Tree/StoredProcedure";
|
||||||
|
import Trigger from "./Tree/Trigger";
|
||||||
|
import UserDefinedFunction from "./Tree/UserDefinedFunction";
|
||||||
|
|
||||||
|
export interface CollectionContextMenuButtonParams {
|
||||||
|
databaseId: string;
|
||||||
|
collectionId: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface DatabaseContextMenuButtonParams {
|
||||||
|
databaseId: string;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* New resource tree (in ReactJS)
|
||||||
|
*/
|
||||||
|
export class ResourceTreeContextMenuButtonFactory {
|
||||||
|
public static createDatabaseContextMenu(container: Explorer, databaseId: string): TreeNodeMenuItem[] {
|
||||||
|
const items: TreeNodeMenuItem[] = [
|
||||||
|
{
|
||||||
|
iconSrc: AddCollectionIcon,
|
||||||
|
onClick: () => container.onNewCollectionClicked(databaseId),
|
||||||
|
label: container.addCollectionText(),
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
if (userContext.apiType !== "Tables") {
|
||||||
|
items.push({
|
||||||
|
iconSrc: DeleteDatabaseIcon,
|
||||||
|
onClick: () => container.openDeleteDatabaseConfirmationPane(),
|
||||||
|
label: container.deleteDatabaseText(),
|
||||||
|
styleClass: "deleteDatabaseMenuItem",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static createCollectionContextMenuButton(
|
||||||
|
container: Explorer,
|
||||||
|
selectedCollection: ViewModels.Collection
|
||||||
|
): TreeNodeMenuItem[] {
|
||||||
|
const items: TreeNodeMenuItem[] = [];
|
||||||
|
if (userContext.apiType === "SQL" || userContext.apiType === "Gremlin") {
|
||||||
|
items.push({
|
||||||
|
iconSrc: AddSqlQueryIcon,
|
||||||
|
onClick: () => selectedCollection && selectedCollection.onNewQueryClick(selectedCollection, null),
|
||||||
|
label: "New SQL Query",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (userContext.apiType === "Mongo") {
|
||||||
|
items.push({
|
||||||
|
iconSrc: AddSqlQueryIcon,
|
||||||
|
onClick: () => selectedCollection && selectedCollection.onNewMongoQueryClick(selectedCollection, null),
|
||||||
|
label: "New Query",
|
||||||
|
});
|
||||||
|
|
||||||
|
items.push({
|
||||||
|
iconSrc: HostedTerminalIcon,
|
||||||
|
onClick: () => {
|
||||||
|
const selectedCollection: ViewModels.Collection = container.findSelectedCollection();
|
||||||
|
if (container.isShellEnabled()) {
|
||||||
|
container.openNotebookTerminal(ViewModels.TerminalKind.Mongo);
|
||||||
|
} else {
|
||||||
|
selectedCollection && selectedCollection.onNewMongoShellClick();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
label: container.isShellEnabled() ? "Open Mongo Shell" : "New Shell",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (userContext.apiType === "SQL" || userContext.apiType === "Gremlin") {
|
||||||
|
items.push({
|
||||||
|
iconSrc: AddStoredProcedureIcon,
|
||||||
|
onClick: () => {
|
||||||
|
const selectedCollection: ViewModels.Collection = container.findSelectedCollection();
|
||||||
|
selectedCollection && selectedCollection.onNewStoredProcedureClick(selectedCollection, null);
|
||||||
|
},
|
||||||
|
label: "New Stored Procedure",
|
||||||
|
});
|
||||||
|
|
||||||
|
items.push({
|
||||||
|
iconSrc: AddUdfIcon,
|
||||||
|
onClick: () => {
|
||||||
|
const selectedCollection: ViewModels.Collection = container.findSelectedCollection();
|
||||||
|
selectedCollection && selectedCollection.onNewUserDefinedFunctionClick(selectedCollection, null);
|
||||||
|
},
|
||||||
|
label: "New UDF",
|
||||||
|
});
|
||||||
|
|
||||||
|
items.push({
|
||||||
|
iconSrc: AddTriggerIcon,
|
||||||
|
onClick: () => {
|
||||||
|
const selectedCollection: ViewModels.Collection = container.findSelectedCollection();
|
||||||
|
selectedCollection && selectedCollection.onNewTriggerClick(selectedCollection, null);
|
||||||
|
},
|
||||||
|
label: "New Trigger",
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
items.push({
|
||||||
|
iconSrc: DeleteCollectionIcon,
|
||||||
|
onClick: () => container.openDeleteCollectionConfirmationPane(),
|
||||||
|
label: container.deleteCollectionText(),
|
||||||
|
styleClass: "deleteCollectionMenuItem",
|
||||||
|
});
|
||||||
|
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static createStoreProcedureContextMenuItems(
|
||||||
|
container: Explorer,
|
||||||
|
storedProcedure: StoredProcedure
|
||||||
|
): TreeNodeMenuItem[] {
|
||||||
|
if (userContext.apiType === "Cassandra") {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
iconSrc: DeleteSprocIcon,
|
||||||
|
onClick: () => storedProcedure.delete(),
|
||||||
|
label: "Delete Store Procedure",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static createTriggerContextMenuItems(container: Explorer, trigger: Trigger): TreeNodeMenuItem[] {
|
||||||
|
if (userContext.apiType === "Cassandra") {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
iconSrc: DeleteTriggerIcon,
|
||||||
|
onClick: () => trigger.delete(),
|
||||||
|
label: "Delete Trigger",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static createUserDefinedFunctionContextMenuItems(
|
||||||
|
container: Explorer,
|
||||||
|
userDefinedFunction: UserDefinedFunction
|
||||||
|
): TreeNodeMenuItem[] {
|
||||||
|
if (userContext.apiType === "Cassandra") {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
{
|
||||||
|
iconSrc: DeleteUDFIcon,
|
||||||
|
onClick: () => userDefinedFunction.delete(),
|
||||||
|
label: "Delete User Defined Function",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,190 +0,0 @@
|
|||||||
import React from "react";
|
|
||||||
import AddCollectionIcon from "../../images/AddCollection.svg";
|
|
||||||
import AddSqlQueryIcon from "../../images/AddSqlQuery_16x16.svg";
|
|
||||||
import AddStoredProcedureIcon from "../../images/AddStoredProcedure.svg";
|
|
||||||
import AddTriggerIcon from "../../images/AddTrigger.svg";
|
|
||||||
import AddUdfIcon from "../../images/AddUdf.svg";
|
|
||||||
import DeleteCollectionIcon from "../../images/DeleteCollection.svg";
|
|
||||||
import DeleteDatabaseIcon from "../../images/DeleteDatabase.svg";
|
|
||||||
import DeleteSprocIcon from "../../images/DeleteSproc.svg";
|
|
||||||
import DeleteTriggerIcon from "../../images/DeleteTrigger.svg";
|
|
||||||
import DeleteUDFIcon from "../../images/DeleteUDF.svg";
|
|
||||||
import HostedTerminalIcon from "../../images/Hosted-Terminal.svg";
|
|
||||||
import * as ViewModels from "../Contracts/ViewModels";
|
|
||||||
import { useSidePanel } from "../hooks/useSidePanel";
|
|
||||||
import { userContext } from "../UserContext";
|
|
||||||
import { getCollectionName, getDatabaseName } from "../Utils/APITypeUtils";
|
|
||||||
import { TreeNodeMenuItem } from "./Controls/TreeComponent/TreeComponent";
|
|
||||||
import Explorer from "./Explorer";
|
|
||||||
import { useNotebook } from "./Notebook/useNotebook";
|
|
||||||
import { DeleteCollectionConfirmationPane } from "./Panes/DeleteCollectionConfirmationPane/DeleteCollectionConfirmationPane";
|
|
||||||
import { DeleteDatabaseConfirmationPanel } from "./Panes/DeleteDatabaseConfirmationPanel";
|
|
||||||
import StoredProcedure from "./Tree/StoredProcedure";
|
|
||||||
import Trigger from "./Tree/Trigger";
|
|
||||||
import UserDefinedFunction from "./Tree/UserDefinedFunction";
|
|
||||||
import { useSelectedNode } from "./useSelectedNode";
|
|
||||||
|
|
||||||
export interface CollectionContextMenuButtonParams {
|
|
||||||
databaseId: string;
|
|
||||||
collectionId: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface DatabaseContextMenuButtonParams {
|
|
||||||
databaseId: string;
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* New resource tree (in ReactJS)
|
|
||||||
*/
|
|
||||||
export const createDatabaseContextMenu = (container: Explorer, databaseId: string): TreeNodeMenuItem[] => {
|
|
||||||
const items: TreeNodeMenuItem[] = [
|
|
||||||
{
|
|
||||||
iconSrc: AddCollectionIcon,
|
|
||||||
onClick: () => container.onNewCollectionClicked(databaseId),
|
|
||||||
label: `New ${getCollectionName()}`,
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
if (userContext.apiType !== "Tables") {
|
|
||||||
items.push({
|
|
||||||
iconSrc: DeleteDatabaseIcon,
|
|
||||||
onClick: () =>
|
|
||||||
useSidePanel
|
|
||||||
.getState()
|
|
||||||
.openSidePanel(
|
|
||||||
"Delete " + getDatabaseName(),
|
|
||||||
<DeleteDatabaseConfirmationPanel refreshDatabases={() => container.refreshAllDatabases()} />
|
|
||||||
),
|
|
||||||
label: `Delete ${getDatabaseName()}`,
|
|
||||||
styleClass: "deleteDatabaseMenuItem",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return items;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createCollectionContextMenuButton = (
|
|
||||||
container: Explorer,
|
|
||||||
selectedCollection: ViewModels.Collection
|
|
||||||
): TreeNodeMenuItem[] => {
|
|
||||||
const items: TreeNodeMenuItem[] = [];
|
|
||||||
if (userContext.apiType === "SQL" || userContext.apiType === "Gremlin") {
|
|
||||||
items.push({
|
|
||||||
iconSrc: AddSqlQueryIcon,
|
|
||||||
onClick: () => selectedCollection && selectedCollection.onNewQueryClick(selectedCollection, undefined),
|
|
||||||
label: "New SQL Query",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (userContext.apiType === "Mongo") {
|
|
||||||
items.push({
|
|
||||||
iconSrc: AddSqlQueryIcon,
|
|
||||||
onClick: () => selectedCollection && selectedCollection.onNewMongoQueryClick(selectedCollection, undefined),
|
|
||||||
label: "New Query",
|
|
||||||
});
|
|
||||||
|
|
||||||
items.push({
|
|
||||||
iconSrc: HostedTerminalIcon,
|
|
||||||
isDisabled: useNotebook.getState().isShellEnabled && userContext.features.notebooksTemporarilyDown,
|
|
||||||
onClick: () => {
|
|
||||||
const selectedCollection: ViewModels.Collection = useSelectedNode.getState().findSelectedCollection();
|
|
||||||
if (useNotebook.getState().isShellEnabled) {
|
|
||||||
container.openNotebookTerminal(ViewModels.TerminalKind.Mongo);
|
|
||||||
} else {
|
|
||||||
selectedCollection && selectedCollection.onNewMongoShellClick();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
label: useNotebook.getState().isShellEnabled ? "Open Mongo Shell" : "New Shell",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (userContext.apiType === "SQL" || userContext.apiType === "Gremlin") {
|
|
||||||
items.push({
|
|
||||||
iconSrc: AddStoredProcedureIcon,
|
|
||||||
onClick: () => {
|
|
||||||
const selectedCollection: ViewModels.Collection = useSelectedNode.getState().findSelectedCollection();
|
|
||||||
selectedCollection && selectedCollection.onNewStoredProcedureClick(selectedCollection, undefined);
|
|
||||||
},
|
|
||||||
label: "New Stored Procedure",
|
|
||||||
});
|
|
||||||
|
|
||||||
items.push({
|
|
||||||
iconSrc: AddUdfIcon,
|
|
||||||
onClick: () => {
|
|
||||||
const selectedCollection: ViewModels.Collection = useSelectedNode.getState().findSelectedCollection();
|
|
||||||
selectedCollection && selectedCollection.onNewUserDefinedFunctionClick(selectedCollection);
|
|
||||||
},
|
|
||||||
label: "New UDF",
|
|
||||||
});
|
|
||||||
|
|
||||||
items.push({
|
|
||||||
iconSrc: AddTriggerIcon,
|
|
||||||
onClick: () => {
|
|
||||||
const selectedCollection: ViewModels.Collection = useSelectedNode.getState().findSelectedCollection();
|
|
||||||
selectedCollection && selectedCollection.onNewTriggerClick(selectedCollection, undefined);
|
|
||||||
},
|
|
||||||
label: "New Trigger",
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
items.push({
|
|
||||||
iconSrc: DeleteCollectionIcon,
|
|
||||||
onClick: () =>
|
|
||||||
useSidePanel
|
|
||||||
.getState()
|
|
||||||
.openSidePanel(
|
|
||||||
"Delete " + getCollectionName(),
|
|
||||||
<DeleteCollectionConfirmationPane refreshDatabases={() => container.refreshAllDatabases()} />
|
|
||||||
),
|
|
||||||
label: `Delete ${getCollectionName()}`,
|
|
||||||
styleClass: "deleteCollectionMenuItem",
|
|
||||||
});
|
|
||||||
|
|
||||||
return items;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createStoreProcedureContextMenuItems = (
|
|
||||||
container: Explorer,
|
|
||||||
storedProcedure: StoredProcedure
|
|
||||||
): TreeNodeMenuItem[] => {
|
|
||||||
if (userContext.apiType === "Cassandra") {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
iconSrc: DeleteSprocIcon,
|
|
||||||
onClick: () => storedProcedure.delete(),
|
|
||||||
label: "Delete Store Procedure",
|
|
||||||
},
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createTriggerContextMenuItems = (container: Explorer, trigger: Trigger): TreeNodeMenuItem[] => {
|
|
||||||
if (userContext.apiType === "Cassandra") {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
iconSrc: DeleteTriggerIcon,
|
|
||||||
onClick: () => trigger.delete(),
|
|
||||||
label: "Delete Trigger",
|
|
||||||
},
|
|
||||||
];
|
|
||||||
};
|
|
||||||
|
|
||||||
export const createUserDefinedFunctionContextMenuItems = (
|
|
||||||
container: Explorer,
|
|
||||||
userDefinedFunction: UserDefinedFunction
|
|
||||||
): TreeNodeMenuItem[] => {
|
|
||||||
if (userContext.apiType === "Cassandra") {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
return [
|
|
||||||
{
|
|
||||||
iconSrc: DeleteUDFIcon,
|
|
||||||
onClick: () => userDefinedFunction.delete(),
|
|
||||||
label: "Delete User Defined Function",
|
|
||||||
},
|
|
||||||
];
|
|
||||||
};
|
|
||||||
@@ -8,9 +8,7 @@ import TriangleDownIcon from "../../../../images/Triangle-down.svg";
|
|||||||
import TriangleRightIcon from "../../../../images/Triangle-right.svg";
|
import TriangleRightIcon from "../../../../images/Triangle-right.svg";
|
||||||
import * as Constants from "../../../Common/Constants";
|
import * as Constants from "../../../Common/Constants";
|
||||||
|
|
||||||
export interface AccordionComponentProps {
|
export interface AccordionComponentProps {}
|
||||||
children: React.ReactNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
export class AccordionComponent extends React.Component<AccordionComponentProps> {
|
export class AccordionComponent extends React.Component<AccordionComponentProps> {
|
||||||
public render(): JSX.Element {
|
public render(): JSX.Element {
|
||||||
@@ -80,7 +78,7 @@ export class AccordionItemComponent extends React.Component<AccordionItemCompone
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private onHeaderClick = (): void => {
|
private onHeaderClick = (_event: React.MouseEvent<HTMLDivElement>): void => {
|
||||||
this.setState({ isExpanded: !this.state.isExpanded });
|
this.setState({ isExpanded: !this.state.isExpanded });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
142
src/Explorer/Controls/Arcadia/ArcadiaMenuPicker.tsx
Normal file
142
src/Explorer/Controls/Arcadia/ArcadiaMenuPicker.tsx
Normal file
@@ -0,0 +1,142 @@
|
|||||||
|
import { DefaultButton, IButtonStyles, IContextualMenuItem, IContextualMenuProps } from "@fluentui/react";
|
||||||
|
import * as React from "react";
|
||||||
|
import { getErrorMessage } from "../../../Common/ErrorHandlingUtils";
|
||||||
|
import * as Logger from "../../../Common/Logger";
|
||||||
|
import { ArcadiaWorkspace, SparkPool } from "../../../Contracts/DataModels";
|
||||||
|
|
||||||
|
export interface ArcadiaMenuPickerProps {
|
||||||
|
selectText?: string;
|
||||||
|
disableSubmenu?: boolean;
|
||||||
|
selectedSparkPool: string;
|
||||||
|
workspaces: ArcadiaWorkspaceItem[];
|
||||||
|
onSparkPoolSelect: (
|
||||||
|
e: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>,
|
||||||
|
item: IContextualMenuItem
|
||||||
|
) => boolean | void;
|
||||||
|
onCreateNewWorkspaceClicked: () => boolean | void;
|
||||||
|
onCreateNewSparkPoolClicked: (workspaceResourceId: string) => boolean | void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface ArcadiaMenuPickerStates {
|
||||||
|
selectedSparkPool: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ArcadiaWorkspaceItem extends ArcadiaWorkspace {
|
||||||
|
sparkPools: SparkPool[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ArcadiaMenuPicker extends React.Component<ArcadiaMenuPickerProps, ArcadiaMenuPickerStates> {
|
||||||
|
constructor(props: ArcadiaMenuPickerProps) {
|
||||||
|
super(props);
|
||||||
|
this.state = {
|
||||||
|
selectedSparkPool: props.selectedSparkPool,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onSparkPoolClicked = (
|
||||||
|
e: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>,
|
||||||
|
item: IContextualMenuItem
|
||||||
|
): boolean | void => {
|
||||||
|
try {
|
||||||
|
this.props.onSparkPoolSelect(e, item);
|
||||||
|
this.setState({
|
||||||
|
selectedSparkPool: item.text,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
Logger.logError(getErrorMessage(error), "ArcadiaMenuPicker/_onSparkPoolClicked");
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
private _onCreateNewWorkspaceClicked = (
|
||||||
|
e: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>,
|
||||||
|
item: IContextualMenuItem
|
||||||
|
): boolean | void => {
|
||||||
|
this.props.onCreateNewWorkspaceClicked();
|
||||||
|
};
|
||||||
|
|
||||||
|
private _onCreateNewSparkPoolClicked = (
|
||||||
|
e: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>,
|
||||||
|
item: IContextualMenuItem
|
||||||
|
): boolean | void => {
|
||||||
|
this.props.onCreateNewSparkPoolClicked(item.key);
|
||||||
|
};
|
||||||
|
|
||||||
|
public render() {
|
||||||
|
const { workspaces } = this.props;
|
||||||
|
let workspaceMenuItems: IContextualMenuItem[] = workspaces.map((workspace) => {
|
||||||
|
let sparkPoolsMenuProps: IContextualMenuProps = {
|
||||||
|
items: workspace.sparkPools.map(
|
||||||
|
(sparkpool): IContextualMenuItem => ({
|
||||||
|
key: sparkpool.id,
|
||||||
|
text: sparkpool.name,
|
||||||
|
onClick: this._onSparkPoolClicked,
|
||||||
|
})
|
||||||
|
),
|
||||||
|
};
|
||||||
|
if (!sparkPoolsMenuProps.items.length) {
|
||||||
|
sparkPoolsMenuProps.items.push({
|
||||||
|
key: workspace.id,
|
||||||
|
text: "Create new spark pool",
|
||||||
|
onClick: this._onCreateNewSparkPoolClicked,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
key: workspace.id,
|
||||||
|
text: workspace.name,
|
||||||
|
subMenuProps: this.props.disableSubmenu ? undefined : sparkPoolsMenuProps,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!workspaceMenuItems.length) {
|
||||||
|
workspaceMenuItems.push({
|
||||||
|
key: "create_workspace",
|
||||||
|
text: "Create new workspace",
|
||||||
|
onClick: this._onCreateNewWorkspaceClicked,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const dropdownStyle: IButtonStyles = {
|
||||||
|
root: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
margin: "auto 5px",
|
||||||
|
padding: "0",
|
||||||
|
border: "0",
|
||||||
|
},
|
||||||
|
rootHovered: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
rootChecked: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
rootFocused: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
rootExpanded: {
|
||||||
|
backgroundColor: "transparent",
|
||||||
|
},
|
||||||
|
flexContainer: {
|
||||||
|
height: "30px",
|
||||||
|
border: "1px solid #a6a6a6",
|
||||||
|
padding: "0 8px",
|
||||||
|
},
|
||||||
|
label: {
|
||||||
|
fontWeight: "400",
|
||||||
|
fontSize: "12px",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DefaultButton
|
||||||
|
text={this.state.selectedSparkPool || this.props.selectText || "Select a Spark pool"}
|
||||||
|
persistMenu={true}
|
||||||
|
className="arcadia-menu-picker"
|
||||||
|
menuProps={{
|
||||||
|
items: workspaceMenuItems,
|
||||||
|
}}
|
||||||
|
styles={dropdownStyle}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,12 +1,15 @@
|
|||||||
|
import * as StringUtils from "../../../Utils/StringUtils";
|
||||||
|
import { KeyCodes } from "../../../Common/Constants";
|
||||||
|
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
import { Action, ActionModifiers } from "../../../Shared/Telemetry/TelemetryConstants";
|
||||||
|
import CollapseChevronDownIcon from "../../../../images/QueryBuilder/CollapseChevronDown_16x.png";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* React component for Command button component.
|
* React component for Command button component.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import CollapseChevronDownIcon from "../../../../images/QueryBuilder/CollapseChevronDown_16x.png";
|
import { ArcadiaMenuPickerProps } from "../Arcadia/ArcadiaMenuPicker";
|
||||||
import { KeyCodes } from "../../../Common/Constants";
|
|
||||||
import { Action, ActionModifiers } from "../../../Shared/Telemetry/TelemetryConstants";
|
|
||||||
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
|
||||||
import * as StringUtils from "../../../Utils/StringUtils";
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Options for this component
|
* Options for this component
|
||||||
@@ -111,6 +114,15 @@ export interface CommandButtonComponentProps {
|
|||||||
* Aria-label for the button
|
* Aria-label for the button
|
||||||
*/
|
*/
|
||||||
ariaLabel: string;
|
ariaLabel: string;
|
||||||
|
//TODO: generalize customized command bar
|
||||||
|
/**
|
||||||
|
* If set to true, will render arcadia picker
|
||||||
|
*/
|
||||||
|
isArcadiaPicker?: boolean;
|
||||||
|
/**
|
||||||
|
* props to render arcadia picker
|
||||||
|
*/
|
||||||
|
arcadiaProps?: ArcadiaMenuPickerProps;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class CommandButtonComponent extends React.Component<CommandButtonComponentProps> {
|
export class CommandButtonComponent extends React.Component<CommandButtonComponentProps> {
|
||||||
@@ -121,7 +133,8 @@ export class CommandButtonComponent extends React.Component<CommandButtonCompone
|
|||||||
if (!this.dropdownElt || !this.expandButtonElt) {
|
if (!this.dropdownElt || !this.expandButtonElt) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
$(this.dropdownElt).offset({ left: $(this.expandButtonElt).offset().left });
|
|
||||||
|
const dropdownElt = $(this.dropdownElt).offset({ left: $(this.expandButtonElt).offset().left });
|
||||||
}
|
}
|
||||||
|
|
||||||
private onKeyPress(event: React.KeyboardEvent): boolean {
|
private onKeyPress(event: React.KeyboardEvent): boolean {
|
||||||
|
|||||||
@@ -23,75 +23,13 @@ export interface DialogState {
|
|||||||
dialogProps?: DialogProps;
|
dialogProps?: DialogProps;
|
||||||
openDialog: (props: DialogProps) => void;
|
openDialog: (props: DialogProps) => void;
|
||||||
closeDialog: () => void;
|
closeDialog: () => void;
|
||||||
showOkCancelModalDialog: (
|
|
||||||
title: string,
|
|
||||||
subText: string,
|
|
||||||
okLabel: string,
|
|
||||||
onOk: () => void,
|
|
||||||
cancelLabel: string,
|
|
||||||
onCancel: () => void,
|
|
||||||
choiceGroupProps?: IChoiceGroupProps,
|
|
||||||
textFieldProps?: TextFieldProps,
|
|
||||||
primaryButtonDisabled?: boolean
|
|
||||||
) => void;
|
|
||||||
showOkModalDialog: (title: string, subText: string) => void;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const useDialog: UseStore<DialogState> = create((set, get) => ({
|
export const useDialog: UseStore<DialogState> = create((set) => ({
|
||||||
visible: false,
|
visible: false,
|
||||||
openDialog: (props: DialogProps) => set(() => ({ visible: true, dialogProps: props })),
|
openDialog: (props: DialogProps) => set(() => ({ visible: true, dialogProps: props })),
|
||||||
closeDialog: () =>
|
closeDialog: () =>
|
||||||
set(
|
set((state) => ({ visible: false, openDialog: state.openDialog, closeDialog: state.closeDialog }), true),
|
||||||
(state) => ({
|
|
||||||
visible: false,
|
|
||||||
openDialog: state.openDialog,
|
|
||||||
closeDialog: state.closeDialog,
|
|
||||||
showOkCancelModalDialog: state.showOkCancelModalDialog,
|
|
||||||
showOkModalDialog: state.showOkModalDialog,
|
|
||||||
}),
|
|
||||||
true // TODO: This probably should not be true but its causing a prod bug so easier to just set the proper state above
|
|
||||||
),
|
|
||||||
showOkCancelModalDialog: (
|
|
||||||
title: string,
|
|
||||||
subText: string,
|
|
||||||
okLabel: string,
|
|
||||||
onOk: () => void,
|
|
||||||
cancelLabel: string,
|
|
||||||
onCancel: () => void,
|
|
||||||
choiceGroupProps?: IChoiceGroupProps,
|
|
||||||
textFieldProps?: TextFieldProps,
|
|
||||||
primaryButtonDisabled?: boolean
|
|
||||||
): void =>
|
|
||||||
get().openDialog({
|
|
||||||
isModal: true,
|
|
||||||
title,
|
|
||||||
subText,
|
|
||||||
primaryButtonText: okLabel,
|
|
||||||
secondaryButtonText: cancelLabel,
|
|
||||||
onPrimaryButtonClick: () => {
|
|
||||||
get().closeDialog();
|
|
||||||
onOk && onOk();
|
|
||||||
},
|
|
||||||
onSecondaryButtonClick: () => {
|
|
||||||
get().closeDialog();
|
|
||||||
onCancel && onCancel();
|
|
||||||
},
|
|
||||||
choiceGroupProps,
|
|
||||||
textFieldProps,
|
|
||||||
primaryButtonDisabled,
|
|
||||||
}),
|
|
||||||
showOkModalDialog: (title: string, subText: string): void =>
|
|
||||||
get().openDialog({
|
|
||||||
isModal: true,
|
|
||||||
title,
|
|
||||||
subText,
|
|
||||||
primaryButtonText: "Close",
|
|
||||||
secondaryButtonText: undefined,
|
|
||||||
onPrimaryButtonClick: () => {
|
|
||||||
get().closeDialog();
|
|
||||||
},
|
|
||||||
onSecondaryButtonClick: undefined,
|
|
||||||
}),
|
|
||||||
}));
|
}));
|
||||||
|
|
||||||
export interface TextFieldProps extends ITextFieldProps {
|
export interface TextFieldProps extends ITextFieldProps {
|
||||||
|
|||||||
64
src/Explorer/Controls/DynamicList/DynamicList.test.ts
Normal file
64
src/Explorer/Controls/DynamicList/DynamicList.test.ts
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
import * as ko from "knockout";
|
||||||
|
import { DynamicListComponent, DynamicListParams, DynamicListItem } from "./DynamicListComponent";
|
||||||
|
|
||||||
|
const $ = (selector: string) => document.querySelector(selector) as HTMLElement;
|
||||||
|
|
||||||
|
function buildComponent(buttonOptions: any) {
|
||||||
|
document.body.innerHTML = DynamicListComponent.template as any;
|
||||||
|
const vm = new DynamicListComponent.viewModel(buttonOptions);
|
||||||
|
ko.applyBindings(vm);
|
||||||
|
}
|
||||||
|
|
||||||
|
describe("Dynamic List Component", () => {
|
||||||
|
const mockPlaceHolder = "Write here";
|
||||||
|
const mockButton = "Add something";
|
||||||
|
const mockValue = "/someText";
|
||||||
|
const mockAriaLabel = "Add ariaLabel";
|
||||||
|
const items: ko.ObservableArray<DynamicListItem> = ko.observableArray<DynamicListItem>();
|
||||||
|
|
||||||
|
function buildListOptions(
|
||||||
|
items: ko.ObservableArray<DynamicListItem>,
|
||||||
|
placeholder?: string,
|
||||||
|
mockButton?: string
|
||||||
|
): DynamicListParams {
|
||||||
|
return {
|
||||||
|
placeholder: placeholder,
|
||||||
|
listItems: items,
|
||||||
|
buttonText: mockButton,
|
||||||
|
ariaLabel: mockAriaLabel,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
ko.cleanNode(document);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Rendering", () => {
|
||||||
|
it("should display button text", () => {
|
||||||
|
const params = buildListOptions(items, mockPlaceHolder, mockButton);
|
||||||
|
buildComponent(params);
|
||||||
|
expect($(".dynamicListItemAdd").textContent).toContain(mockButton);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Behavior", () => {
|
||||||
|
it("should add items to the list", () => {
|
||||||
|
const params = buildListOptions(items, mockPlaceHolder, mockButton);
|
||||||
|
buildComponent(params);
|
||||||
|
$(".dynamicListItemAdd").click();
|
||||||
|
expect(items().length).toBe(1);
|
||||||
|
const input = document.getElementsByClassName("dynamicListItem").item(0).children[0];
|
||||||
|
input.setAttribute("value", mockValue);
|
||||||
|
input.dispatchEvent(new Event("change"));
|
||||||
|
input.dispatchEvent(new Event("blur"));
|
||||||
|
expect(items()[0].value()).toBe(mockValue);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should remove items from the list", () => {
|
||||||
|
const params = buildListOptions(items, mockPlaceHolder);
|
||||||
|
buildComponent(params);
|
||||||
|
$(".dynamicListItemDelete").click();
|
||||||
|
expect(items().length).toBe(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
59
src/Explorer/Controls/DynamicList/DynamicListComponent.less
Normal file
59
src/Explorer/Controls/DynamicList/DynamicListComponent.less
Normal file
@@ -0,0 +1,59 @@
|
|||||||
|
@import "../../../../less/Common/Constants";
|
||||||
|
|
||||||
|
.dynamicList {
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.dynamicListContainer {
|
||||||
|
.dynamicListItem {
|
||||||
|
justify-content: space-around;
|
||||||
|
margin-bottom: @MediumSpace;
|
||||||
|
|
||||||
|
input {
|
||||||
|
width: @newCollectionPaneInputWidth;
|
||||||
|
margin: auto;
|
||||||
|
font-size: @mediumFontSize;
|
||||||
|
padding: @SmallSpace @DefaultSpace;
|
||||||
|
color: @BaseDark;
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamicListItemDelete {
|
||||||
|
padding: @SmallSpace @SmallSpace @DefaultSpace;
|
||||||
|
margin-left: @SmallSpace;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
.hover();
|
||||||
|
}
|
||||||
|
|
||||||
|
&:active {
|
||||||
|
.active();
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
.dataExplorerIcons();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.dynamicListItemNew {
|
||||||
|
margin-top: @LargeSpace;
|
||||||
|
|
||||||
|
.dynamicListItemAdd {
|
||||||
|
padding: @DefaultSpace;
|
||||||
|
cursor: pointer;
|
||||||
|
|
||||||
|
&:hover {
|
||||||
|
.hover();
|
||||||
|
}
|
||||||
|
|
||||||
|
&:active {
|
||||||
|
.active();
|
||||||
|
}
|
||||||
|
|
||||||
|
img {
|
||||||
|
.dataExplorerIcons();
|
||||||
|
margin: 0px @SmallSpace @SmallSpace 0px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
117
src/Explorer/Controls/DynamicList/DynamicListComponent.ts
Normal file
117
src/Explorer/Controls/DynamicList/DynamicListComponent.ts
Normal file
@@ -0,0 +1,117 @@
|
|||||||
|
/**
|
||||||
|
* Dynamic list:
|
||||||
|
*
|
||||||
|
* Creates a list of dynamic inputs that can be populated and deleted.
|
||||||
|
*
|
||||||
|
* How to use in your markup:
|
||||||
|
* <dynamic-list params="{ listItems: anObservableArrayOfDynamicListItem, placeholder: 'Text to display in placeholder', ariaLabel: 'Text for aria-label', buttonText: 'Add item' }">
|
||||||
|
* </dynamic-list>
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
import * as ko from "knockout";
|
||||||
|
import { WaitsForTemplateViewModel } from "../../WaitsForTemplateViewModel";
|
||||||
|
import { KeyCodes } from "../../../Common/Constants";
|
||||||
|
import template from "./dynamic-list.html";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parameters for this component
|
||||||
|
*/
|
||||||
|
export interface DynamicListParams {
|
||||||
|
/**
|
||||||
|
* Observable list of items to update
|
||||||
|
*/
|
||||||
|
listItems: ko.ObservableArray<DynamicListItem>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Placeholder text to use on inputs
|
||||||
|
*/
|
||||||
|
placeholder?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text to use as aria-label
|
||||||
|
*/
|
||||||
|
ariaLabel: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text for the button to add items
|
||||||
|
*/
|
||||||
|
buttonText?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Callback triggered when the template is bound to the component (for testing purposes)
|
||||||
|
*/
|
||||||
|
onTemplateReady?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Item in the dynamic list
|
||||||
|
*/
|
||||||
|
export interface DynamicListItem {
|
||||||
|
value: ko.Observable<string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class DynamicListViewModel extends WaitsForTemplateViewModel {
|
||||||
|
public placeholder: string;
|
||||||
|
public ariaLabel: string;
|
||||||
|
public buttonText: string;
|
||||||
|
public newItem: ko.Observable<string>;
|
||||||
|
public isTemplateReady: ko.Observable<boolean>;
|
||||||
|
public listItems: ko.ObservableArray<DynamicListItem>;
|
||||||
|
|
||||||
|
public constructor(options: DynamicListParams) {
|
||||||
|
super();
|
||||||
|
super.onTemplateReady((isTemplateReady: boolean) => {
|
||||||
|
if (isTemplateReady && options.onTemplateReady) {
|
||||||
|
options.onTemplateReady();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const params: DynamicListParams = options;
|
||||||
|
const paramsPlaceholder: string = params.placeholder;
|
||||||
|
const paramsButtonText: string = params.buttonText;
|
||||||
|
this.placeholder = paramsPlaceholder || "Write a value";
|
||||||
|
this.ariaLabel = "Unique keys";
|
||||||
|
this.buttonText = paramsButtonText || "Add item";
|
||||||
|
this.listItems = params.listItems || ko.observableArray<DynamicListItem>();
|
||||||
|
this.newItem = ko.observable("");
|
||||||
|
}
|
||||||
|
|
||||||
|
public removeItem = (data: any, event: MouseEvent | KeyboardEvent): void => {
|
||||||
|
const context = ko.contextFor(event.target as Node);
|
||||||
|
this.listItems.splice(context.$index(), 1);
|
||||||
|
document.getElementById("addUniqueKeyBtn").focus();
|
||||||
|
};
|
||||||
|
|
||||||
|
public onRemoveItemKeyPress = (data: any, event: KeyboardEvent, source: any): boolean => {
|
||||||
|
if (event.keyCode === KeyCodes.Enter || event.keyCode === KeyCodes.Space) {
|
||||||
|
this.removeItem(data, event);
|
||||||
|
(document.querySelector(".dynamicListItem:last-of-type input") as HTMLElement).focus();
|
||||||
|
event.stopPropagation();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
public addItem(): void {
|
||||||
|
this.listItems.push({ value: ko.observable("") });
|
||||||
|
(document.querySelector(".dynamicListItem:last-of-type input") as HTMLElement).focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
public onAddItemKeyPress = (source: any, event: KeyboardEvent): boolean => {
|
||||||
|
if (event.keyCode === KeyCodes.Enter || event.keyCode === KeyCodes.Space) {
|
||||||
|
this.addItem();
|
||||||
|
event.stopPropagation();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Helper class for ko component registration
|
||||||
|
*/
|
||||||
|
export const DynamicListComponent = {
|
||||||
|
viewModel: DynamicListViewModel,
|
||||||
|
template,
|
||||||
|
};
|
||||||
34
src/Explorer/Controls/DynamicList/dynamic-list.html
Normal file
34
src/Explorer/Controls/DynamicList/dynamic-list.html
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
<div class="dynamicList" data-bind="setTemplateReady: true">
|
||||||
|
<div class="dynamicListContainer" data-bind="foreach: listItems">
|
||||||
|
<div class="dynamicListItem">
|
||||||
|
<input
|
||||||
|
id="uniqueKeyItems"
|
||||||
|
type="text"
|
||||||
|
autocomplete="off"
|
||||||
|
data-bind="value: value, attr: {placeholder: $parent.placeholder, 'aria-label': $parent.ariaLabel}"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
class="dynamicListItemDelete"
|
||||||
|
title="Remove item"
|
||||||
|
role="button"
|
||||||
|
aria-label="Remove item"
|
||||||
|
tabindex="0"
|
||||||
|
data-bind="click: $parent.removeItem, event: { keydown: $parent.onRemoveItemKeyPress }"
|
||||||
|
>
|
||||||
|
<img src="/delete.svg" alt="Remove item" />
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dynamicListItemNew">
|
||||||
|
<span
|
||||||
|
class="dynamicListItemAdd"
|
||||||
|
id="addUniqueKeyBtn"
|
||||||
|
role="button"
|
||||||
|
aria-label="Add unique key"
|
||||||
|
tabindex="0"
|
||||||
|
data-bind="click: addItem, event: { keydown: onAddItemKeyPress }"
|
||||||
|
>
|
||||||
|
<img src="/Add-property.svg" data-bind="attr: {alt: buttonText}" /> <span data-bind="text: buttonText"></span>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -1,11 +1,6 @@
|
|||||||
import { Spinner, SpinnerSize } from "@fluentui/react";
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { loadMonaco, monaco } from "../../LazyMonaco";
|
import { loadMonaco, monaco } from "../../LazyMonaco";
|
||||||
// import "./EditorReact.less";
|
|
||||||
|
|
||||||
interface EditorReactStates {
|
|
||||||
showEditor: boolean;
|
|
||||||
}
|
|
||||||
export interface EditorReactProps {
|
export interface EditorReactProps {
|
||||||
language: string;
|
language: string;
|
||||||
content: string;
|
content: string;
|
||||||
@@ -17,26 +12,22 @@ export interface EditorReactProps {
|
|||||||
theme?: string; // Monaco editor theme
|
theme?: string; // Monaco editor theme
|
||||||
}
|
}
|
||||||
|
|
||||||
export class EditorReact extends React.Component<EditorReactProps, EditorReactStates> {
|
export class EditorReact extends React.Component<EditorReactProps> {
|
||||||
private rootNode: HTMLElement;
|
private rootNode: HTMLElement;
|
||||||
private editor: monaco.editor.IStandaloneCodeEditor;
|
private editor: monaco.editor.IStandaloneCodeEditor;
|
||||||
private selectionListener: monaco.IDisposable;
|
private selectionListener: monaco.IDisposable;
|
||||||
|
|
||||||
public constructor(props: EditorReactProps) {
|
public constructor(props: EditorReactProps) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
|
||||||
showEditor: false,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public componentDidMount(): void {
|
public componentDidMount(): void {
|
||||||
this.createEditor(this.configureEditor.bind(this));
|
this.createEditor(this.configureEditor.bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
public componentDidUpdate(previous: EditorReactProps) {
|
public shouldComponentUpdate(): boolean {
|
||||||
if (this.props.content !== previous.content) {
|
// Prevents component re-rendering
|
||||||
this.editor.setValue(this.props.content);
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public componentWillUnmount(): void {
|
public componentWillUnmount(): void {
|
||||||
@@ -44,19 +35,14 @@ export class EditorReact extends React.Component<EditorReactProps, EditorReactSt
|
|||||||
}
|
}
|
||||||
|
|
||||||
public render(): JSX.Element {
|
public render(): JSX.Element {
|
||||||
return (
|
return <div className="jsonEditor" ref={(elt: HTMLElement) => this.setRef(elt)} />;
|
||||||
<React.Fragment>
|
|
||||||
{!this.state.showEditor && <Spinner size={SpinnerSize.large} className="spinner" />}
|
|
||||||
<div className="jsonEditor" ref={(elt: HTMLElement) => this.setRef(elt)} />
|
|
||||||
</React.Fragment>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected configureEditor(editor: monaco.editor.IStandaloneCodeEditor) {
|
protected configureEditor(editor: monaco.editor.IStandaloneCodeEditor) {
|
||||||
this.editor = editor;
|
this.editor = editor;
|
||||||
const queryEditorModel = this.editor.getModel();
|
const queryEditorModel = this.editor.getModel();
|
||||||
if (!this.props.isReadOnly && this.props.onContentChanged) {
|
if (!this.props.isReadOnly && this.props.onContentChanged) {
|
||||||
queryEditorModel.onDidChangeContent(() => {
|
queryEditorModel.onDidChangeContent((e: monaco.editor.IModelContentChangedEvent) => {
|
||||||
const queryEditorModel = this.editor.getModel();
|
const queryEditorModel = this.editor.getModel();
|
||||||
this.props.onContentChanged(queryEditorModel.getValue());
|
this.props.onContentChanged(queryEditorModel.getValue());
|
||||||
});
|
});
|
||||||
@@ -90,12 +76,6 @@ export class EditorReact extends React.Component<EditorReactProps, EditorReactSt
|
|||||||
this.rootNode.innerHTML = "";
|
this.rootNode.innerHTML = "";
|
||||||
const monaco = await loadMonaco();
|
const monaco = await loadMonaco();
|
||||||
createCallback(monaco.editor.create(this.rootNode, options));
|
createCallback(monaco.editor.create(this.rootNode, options));
|
||||||
|
|
||||||
if (this.rootNode.innerHTML) {
|
|
||||||
this.setState({
|
|
||||||
showEditor: true,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private setRef(element: HTMLElement): void {
|
private setRef(element: HTMLElement): void {
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ export class GitHubReposComponent extends React.Component<GitHubReposComponentPr
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div>{content}</div>
|
<div className={"paneMainContent"}>{content}</div>
|
||||||
{!this.props.showAuthorizeAccess && (
|
{!this.props.showAuthorizeAccess && (
|
||||||
<>
|
<>
|
||||||
<div className={"paneFooter"} style={ContentFooterStyle}>
|
<div className={"paneFooter"} style={ContentFooterStyle}>
|
||||||
|
|||||||
20
src/Explorer/Controls/GitHub/GitHubReposComponentAdapter.tsx
Normal file
20
src/Explorer/Controls/GitHub/GitHubReposComponentAdapter.tsx
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import ko from "knockout";
|
||||||
|
import * as React from "react";
|
||||||
|
import { ReactAdapter } from "../../../Bindings/ReactBindingHandler";
|
||||||
|
import { GitHubReposComponent, GitHubReposComponentProps } from "./GitHubReposComponent";
|
||||||
|
|
||||||
|
export class GitHubReposComponentAdapter implements ReactAdapter {
|
||||||
|
public parameters: ko.Observable<number>;
|
||||||
|
|
||||||
|
constructor(private props: GitHubReposComponentProps) {
|
||||||
|
this.parameters = ko.observable<number>(Date.now());
|
||||||
|
}
|
||||||
|
|
||||||
|
public renderComponent(): JSX.Element {
|
||||||
|
return <GitHubReposComponent {...this.props} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
public triggerRender(): void {
|
||||||
|
window.requestAnimationFrame(() => this.parameters(Date.now()));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -5,9 +5,6 @@
|
|||||||
display: inline-block;
|
display: inline-block;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|
||||||
.input-type-head-text-field {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
textarea {
|
textarea {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
@@ -24,11 +21,4 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.input-typeahead-chocies-container {
|
|
||||||
border: 1px solid lightgrey;
|
|
||||||
padding: 5px 10px 5px 10px;
|
|
||||||
cursor: pointer;
|
|
||||||
.choice-caption{
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -6,13 +6,14 @@
|
|||||||
* typeaheadOverrideOptions: { dynamic:false }
|
* typeaheadOverrideOptions: { dynamic:false }
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
import { getTheme, IconButton, IIconProps, List, Stack, TextField } from "@fluentui/react";
|
import "jquery-typeahead";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
|
import { KeyCodes } from "../../../Common/Constants";
|
||||||
import "./InputTypeahead.less";
|
import "./InputTypeahead.less";
|
||||||
|
|
||||||
export interface Item {
|
export interface Item {
|
||||||
caption: string;
|
caption: string;
|
||||||
value: string;
|
value: any;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -74,125 +75,170 @@ export interface InputTypeaheadComponentProps {
|
|||||||
useTextarea?: boolean;
|
useTextarea?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface InputTypeaheadComponentState {
|
interface OnClickItem {
|
||||||
isSuggestionVisible: boolean;
|
matchedKey: string;
|
||||||
selectedChoice: Item;
|
value: any;
|
||||||
filteredChoices: Item[];
|
caption: string;
|
||||||
|
group: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface Cache {
|
||||||
|
inputValue: string;
|
||||||
|
selection: Item;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface InputTypeaheadComponentState {}
|
||||||
|
|
||||||
export class InputTypeaheadComponent extends React.Component<
|
export class InputTypeaheadComponent extends React.Component<
|
||||||
InputTypeaheadComponentProps,
|
InputTypeaheadComponentProps,
|
||||||
InputTypeaheadComponentState
|
InputTypeaheadComponentState
|
||||||
> {
|
> {
|
||||||
constructor(props: InputTypeaheadComponentProps) {
|
private inputElt: HTMLElement;
|
||||||
|
private containerElt: HTMLElement;
|
||||||
|
|
||||||
|
private cache: Cache;
|
||||||
|
private inputValue: string;
|
||||||
|
private selection: Item;
|
||||||
|
|
||||||
|
public constructor(props: InputTypeaheadComponentProps) {
|
||||||
super(props);
|
super(props);
|
||||||
this.state = {
|
this.cache = {
|
||||||
isSuggestionVisible: false,
|
inputValue: null,
|
||||||
filteredChoices: [],
|
selection: null,
|
||||||
selectedChoice: {
|
|
||||||
caption: "",
|
|
||||||
value: "",
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private onRenderCell = (item: Item): JSX.Element => {
|
/**
|
||||||
return (
|
* Props have changed
|
||||||
<div className="input-typeahead-chocies-container" onClick={() => this.onChoiceClick(item)}>
|
* @param prevProps
|
||||||
<p className="choice-caption">{item.caption}</p>
|
* @param prevState
|
||||||
<span>{item.value}</span>
|
* @param snapshot
|
||||||
</div>
|
*/
|
||||||
);
|
public componentDidUpdate(
|
||||||
};
|
prevProps: InputTypeaheadComponentProps,
|
||||||
|
prevState: InputTypeaheadComponentState,
|
||||||
private onChoiceClick = (item: Item): void => {
|
snapshot: any
|
||||||
this.props.onNewValue(item.caption);
|
): void {
|
||||||
this.setState({ isSuggestionVisible: false, selectedChoice: item });
|
if (prevProps.defaultValue !== this.props.defaultValue) {
|
||||||
};
|
$(this.inputElt).val(this.props.defaultValue);
|
||||||
|
this.initializeTypeahead();
|
||||||
private handleChange = (value: string): void => {
|
|
||||||
if (!value) {
|
|
||||||
this.setState({ isSuggestionVisible: true });
|
|
||||||
}
|
}
|
||||||
this.props.onNewValue(value);
|
}
|
||||||
const filteredChoices = this.filterChoiceByValue(this.props.choices, value);
|
|
||||||
this.setState({ filteredChoices });
|
|
||||||
};
|
|
||||||
|
|
||||||
private onSubmit = (event: React.KeyboardEvent<HTMLElement>): void => {
|
/**
|
||||||
if (event.key === "Enter") {
|
* Executed once react is done building the DOM for this component
|
||||||
|
*/
|
||||||
|
public componentDidMount(): void {
|
||||||
|
this.initializeTypeahead();
|
||||||
|
}
|
||||||
|
|
||||||
|
public render(): JSX.Element {
|
||||||
|
return (
|
||||||
|
<span className="input-typeahead-container">
|
||||||
|
<div
|
||||||
|
className="input-typehead"
|
||||||
|
onKeyDown={(event: React.KeyboardEvent<HTMLDivElement>) => this.onKeyDown(event)}
|
||||||
|
>
|
||||||
|
<div className="typeahead__container" ref={(input) => (this.containerElt = input)}>
|
||||||
|
<div className="typeahead__field">
|
||||||
|
<span className="typeahead__query">
|
||||||
|
{this.props.useTextarea ? (
|
||||||
|
<textarea
|
||||||
|
rows={1}
|
||||||
|
name="q"
|
||||||
|
autoComplete="off"
|
||||||
|
aria-label="Input query"
|
||||||
|
ref={(input) => (this.inputElt = input)}
|
||||||
|
defaultValue={this.props.defaultValue}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<input
|
||||||
|
name="q"
|
||||||
|
type="search"
|
||||||
|
autoComplete="off"
|
||||||
|
aria-label="Input query"
|
||||||
|
ref={(input) => (this.inputElt = input)}
|
||||||
|
defaultValue={this.props.defaultValue}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
{this.props.showSearchButton && (
|
||||||
|
<span className="typeahead__button">
|
||||||
|
<button type="submit">
|
||||||
|
<span className="typeahead__search-icon" />
|
||||||
|
</button>
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private onKeyDown(event: React.KeyboardEvent<HTMLElement>) {
|
||||||
|
if (event.keyCode === KeyCodes.Enter) {
|
||||||
if (this.props.submitFct) {
|
if (this.props.submitFct) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
this.props.submitFct(this.props.defaultValue, this.state.selectedChoice);
|
this.props.submitFct(this.cache.inputValue, this.cache.selection);
|
||||||
this.setState({ isSuggestionVisible: false });
|
$(this.containerElt).children(".typeahead__result").hide();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
private filterChoiceByValue = (choices: Item[], searchKeyword: string): Item[] => {
|
/**
|
||||||
return choices.filter((choice) =>
|
* Must execute once ko is rendered, so that it can find the input element by id
|
||||||
// @ts-ignore
|
*/
|
||||||
Object.keys(choice).some((key) => choice[key].toLowerCase().includes(searchKeyword.toLowerCase()))
|
private initializeTypeahead(): void {
|
||||||
);
|
const props = this.props;
|
||||||
};
|
let cache = this.cache;
|
||||||
|
let options: any = {
|
||||||
public render(): JSX.Element {
|
input: this.inputElt,
|
||||||
const { defaultValue, useTextarea, placeholder, onNewValue } = this.props;
|
order: "asc",
|
||||||
const { isSuggestionVisible, selectedChoice, filteredChoices } = this.state;
|
minLength: 0,
|
||||||
const theme = getTheme();
|
searchOnFocus: true,
|
||||||
|
source: {
|
||||||
const iconButtonStyles = {
|
display: "caption",
|
||||||
root: {
|
data: () => {
|
||||||
color: theme.palette.neutralPrimary,
|
return props.choices;
|
||||||
marginLeft: "10px !important",
|
},
|
||||||
marginTop: "0px",
|
|
||||||
marginRight: "2px",
|
|
||||||
width: "42px",
|
|
||||||
},
|
},
|
||||||
rootHovered: {
|
callback: {
|
||||||
color: theme.palette.neutralDark,
|
onClick: (node: any, a: any, item: OnClickItem, event: any) => {
|
||||||
|
cache.selection = item;
|
||||||
|
|
||||||
|
if (props.onSelected) {
|
||||||
|
props.onSelected(item);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onResult(node: any, query: any, result: any, resultCount: any, resultCountPerGroup: any) {
|
||||||
|
cache.inputValue = query;
|
||||||
|
if (props.onNewValue) {
|
||||||
|
props.onNewValue(query);
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
template: (query: string, item: any) => {
|
||||||
|
// Don't display id if caption *IS* the id
|
||||||
|
return item.caption === item.value
|
||||||
|
? "<span>{{caption}}</span>"
|
||||||
|
: "<span><div>{{caption}}</div><div><small>{{value}}</small></div></span>";
|
||||||
|
},
|
||||||
|
dynamic: true,
|
||||||
};
|
};
|
||||||
const cancelIcon: IIconProps = { iconName: "cancel" };
|
|
||||||
const searchIcon: IIconProps = { iconName: "Search" };
|
|
||||||
|
|
||||||
return (
|
// Override options
|
||||||
<div className="input-typeahead-container">
|
if (props.typeaheadOverrideOptions) {
|
||||||
<Stack horizontal>
|
for (const p in props.typeaheadOverrideOptions) {
|
||||||
<TextField
|
options[p] = props.typeaheadOverrideOptions[p];
|
||||||
multiline={useTextarea}
|
}
|
||||||
rows={1}
|
}
|
||||||
defaultValue={defaultValue}
|
|
||||||
ariaLabel="Input query"
|
if (props.hasOwnProperty("showCancelButton")) {
|
||||||
placeholder={placeholder}
|
options.cancelButton = props.showCancelButton;
|
||||||
className="input-type-head-text-field"
|
}
|
||||||
value={defaultValue}
|
|
||||||
onKeyDown={this.onSubmit}
|
$(this.inputElt).typeahead(options);
|
||||||
onFocus={() => this.setState({ isSuggestionVisible: true })}
|
|
||||||
onChange={(_event, newValue?: string) => this.handleChange(newValue)}
|
|
||||||
/>
|
|
||||||
{this.props.showCancelButton && (
|
|
||||||
<IconButton
|
|
||||||
styles={iconButtonStyles}
|
|
||||||
iconProps={cancelIcon}
|
|
||||||
ariaLabel="cancel Button"
|
|
||||||
onClick={() => onNewValue("")}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{this.props.showSearchButton && (
|
|
||||||
<IconButton
|
|
||||||
styles={iconButtonStyles}
|
|
||||||
iconProps={searchIcon}
|
|
||||||
ariaLabel="Search Button"
|
|
||||||
onClick={() => this.props.submitFct(defaultValue, selectedChoice)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Stack>
|
|
||||||
{filteredChoices.length && isSuggestionVisible ? (
|
|
||||||
<List items={filteredChoices} onRenderCell={this.onRenderCell} />
|
|
||||||
) : undefined}
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,43 +1,61 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
exports[`inputTypeahead renders <input /> 1`] = `
|
exports[`inputTypeahead renders <input /> 1`] = `
|
||||||
<div
|
<span
|
||||||
className="input-typeahead-container"
|
className="input-typeahead-container"
|
||||||
>
|
>
|
||||||
<Stack
|
<div
|
||||||
horizontal={true}
|
className="input-typehead"
|
||||||
|
onKeyDown={[Function]}
|
||||||
>
|
>
|
||||||
<StyledTextFieldBase
|
<div
|
||||||
ariaLabel="Input query"
|
className="typeahead__container"
|
||||||
className="input-type-head-text-field"
|
>
|
||||||
multiline={false}
|
<div
|
||||||
onChange={[Function]}
|
className="typeahead__field"
|
||||||
onFocus={[Function]}
|
>
|
||||||
onKeyDown={[Function]}
|
<span
|
||||||
placeholder="placeholder"
|
className="typeahead__query"
|
||||||
rows={1}
|
>
|
||||||
/>
|
<input
|
||||||
</Stack>
|
aria-label="Input query"
|
||||||
</div>
|
autoComplete="off"
|
||||||
|
name="q"
|
||||||
|
type="search"
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
exports[`inputTypeahead renders <textarea /> 1`] = `
|
exports[`inputTypeahead renders <textarea /> 1`] = `
|
||||||
<div
|
<span
|
||||||
className="input-typeahead-container"
|
className="input-typeahead-container"
|
||||||
>
|
>
|
||||||
<Stack
|
<div
|
||||||
horizontal={true}
|
className="input-typehead"
|
||||||
|
onKeyDown={[Function]}
|
||||||
>
|
>
|
||||||
<StyledTextFieldBase
|
<div
|
||||||
ariaLabel="Input query"
|
className="typeahead__container"
|
||||||
className="input-type-head-text-field"
|
>
|
||||||
multiline={true}
|
<div
|
||||||
onChange={[Function]}
|
className="typeahead__field"
|
||||||
onFocus={[Function]}
|
>
|
||||||
onKeyDown={[Function]}
|
<span
|
||||||
placeholder="placeholder"
|
className="typeahead__query"
|
||||||
rows={1}
|
>
|
||||||
/>
|
<textarea
|
||||||
</Stack>
|
aria-label="Input query"
|
||||||
</div>
|
autoComplete="off"
|
||||||
|
name="q"
|
||||||
|
rows={1}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</span>
|
||||||
`;
|
`;
|
||||||
|
|||||||
@@ -1,90 +1,154 @@
|
|||||||
import { shallow } from "enzyme";
|
|
||||||
import React from "react";
|
|
||||||
import * as DataModels from "../../../Contracts/DataModels";
|
import * as DataModels from "../../../Contracts/DataModels";
|
||||||
import { NotebookTerminalComponent, NotebookTerminalComponentProps } from "./NotebookTerminalComponent";
|
import { NotebookTerminalComponent } from "./NotebookTerminalComponent";
|
||||||
|
|
||||||
const testAccount: DataModels.DatabaseAccount = {
|
const createTestDatabaseAccount = (): DataModels.DatabaseAccount => {
|
||||||
id: "id",
|
return {
|
||||||
kind: "kind",
|
id: "testId",
|
||||||
location: "location",
|
kind: "testKind",
|
||||||
name: "name",
|
location: "testLocation",
|
||||||
properties: {
|
name: "testName",
|
||||||
documentEndpoint: "https://testDocumentEndpoint.azure.com/",
|
properties: {
|
||||||
},
|
cassandraEndpoint: null,
|
||||||
type: "type",
|
documentEndpoint: "https://testDocumentEndpoint.azure.com/",
|
||||||
|
gremlinEndpoint: null,
|
||||||
|
tableEndpoint: null,
|
||||||
|
},
|
||||||
|
type: "testType",
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const testMongo32Account: DataModels.DatabaseAccount = {
|
const createTestMongo32DatabaseAccount = (): DataModels.DatabaseAccount => {
|
||||||
...testAccount,
|
return {
|
||||||
|
id: "testId",
|
||||||
|
kind: "testKind",
|
||||||
|
location: "testLocation",
|
||||||
|
name: "testName",
|
||||||
|
properties: {
|
||||||
|
cassandraEndpoint: null,
|
||||||
|
documentEndpoint: "https://testDocumentEndpoint.azure.com/",
|
||||||
|
gremlinEndpoint: null,
|
||||||
|
tableEndpoint: null,
|
||||||
|
},
|
||||||
|
type: "testType",
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const testMongo36Account: DataModels.DatabaseAccount = {
|
const createTestMongo36DatabaseAccount = (): DataModels.DatabaseAccount => {
|
||||||
...testAccount,
|
return {
|
||||||
properties: {
|
id: "testId",
|
||||||
mongoEndpoint: "https://testMongoEndpoint.azure.com/",
|
kind: "testKind",
|
||||||
},
|
location: "testLocation",
|
||||||
|
name: "testName",
|
||||||
|
properties: {
|
||||||
|
cassandraEndpoint: null,
|
||||||
|
documentEndpoint: "https://testDocumentEndpoint.azure.com/",
|
||||||
|
gremlinEndpoint: null,
|
||||||
|
tableEndpoint: null,
|
||||||
|
mongoEndpoint: "https://testMongoEndpoint.azure.com/",
|
||||||
|
},
|
||||||
|
type: "testType",
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const testCassandraAccount: DataModels.DatabaseAccount = {
|
const createTestCassandraDatabaseAccount = (): DataModels.DatabaseAccount => {
|
||||||
...testAccount,
|
return {
|
||||||
properties: {
|
id: "testId",
|
||||||
cassandraEndpoint: "https://testCassandraEndpoint.azure.com/",
|
kind: "testKind",
|
||||||
},
|
location: "testLocation",
|
||||||
|
name: "testName",
|
||||||
|
properties: {
|
||||||
|
cassandraEndpoint: "https://testCassandraEndpoint.azure.com/",
|
||||||
|
documentEndpoint: null,
|
||||||
|
gremlinEndpoint: null,
|
||||||
|
tableEndpoint: null,
|
||||||
|
},
|
||||||
|
type: "testType",
|
||||||
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const testNotebookServerInfo: DataModels.NotebookWorkspaceConnectionInfo = {
|
const createTerminal = (): NotebookTerminalComponent => {
|
||||||
authToken: "authToken",
|
return new NotebookTerminalComponent({
|
||||||
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com",
|
notebookServerInfo: {
|
||||||
|
authToken: "testAuthToken",
|
||||||
|
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/",
|
||||||
|
},
|
||||||
|
databaseAccount: createTestDatabaseAccount(),
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const testMongoNotebookServerInfo: DataModels.NotebookWorkspaceConnectionInfo = {
|
const createMongo32Terminal = (): NotebookTerminalComponent => {
|
||||||
authToken: "authToken",
|
return new NotebookTerminalComponent({
|
||||||
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/mongo",
|
notebookServerInfo: {
|
||||||
|
authToken: "testAuthToken",
|
||||||
|
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/mongo",
|
||||||
|
},
|
||||||
|
databaseAccount: createTestMongo32DatabaseAccount(),
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const testCassandraNotebookServerInfo: DataModels.NotebookWorkspaceConnectionInfo = {
|
const createMongo36Terminal = (): NotebookTerminalComponent => {
|
||||||
authToken: "authToken",
|
return new NotebookTerminalComponent({
|
||||||
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/cassandra",
|
notebookServerInfo: {
|
||||||
|
authToken: "testAuthToken",
|
||||||
|
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/mongo",
|
||||||
|
},
|
||||||
|
databaseAccount: createTestMongo36DatabaseAccount(),
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const createCassandraTerminal = (): NotebookTerminalComponent => {
|
||||||
|
return new NotebookTerminalComponent({
|
||||||
|
notebookServerInfo: {
|
||||||
|
authToken: "testAuthToken",
|
||||||
|
notebookServerEndpoint: "https://testNotebookServerEndpoint.azure.com/cassandra",
|
||||||
|
},
|
||||||
|
databaseAccount: createTestCassandraDatabaseAccount(),
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
describe("NotebookTerminalComponent", () => {
|
describe("NotebookTerminalComponent", () => {
|
||||||
it("renders terminal", () => {
|
it("getTerminalParams: Test for terminal", () => {
|
||||||
const props: NotebookTerminalComponentProps = {
|
const terminal: NotebookTerminalComponent = createTerminal();
|
||||||
databaseAccount: testAccount,
|
const params: Map<string, string> = terminal.getTerminalParams();
|
||||||
notebookServerInfo: testNotebookServerInfo,
|
|
||||||
};
|
|
||||||
|
|
||||||
const wrapper = shallow(<NotebookTerminalComponent {...props} />);
|
expect(params).toEqual(
|
||||||
expect(wrapper).toMatchSnapshot();
|
new Map<string, string>([["terminal", "true"]])
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("renders mongo 3.2 shell", () => {
|
it("getTerminalParams: Test for Mongo 3.2 terminal", () => {
|
||||||
const props: NotebookTerminalComponentProps = {
|
const terminal: NotebookTerminalComponent = createMongo32Terminal();
|
||||||
databaseAccount: testMongo32Account,
|
const params: Map<string, string> = terminal.getTerminalParams();
|
||||||
notebookServerInfo: testMongoNotebookServerInfo,
|
|
||||||
};
|
|
||||||
|
|
||||||
const wrapper = shallow(<NotebookTerminalComponent {...props} />);
|
expect(params).toEqual(
|
||||||
expect(wrapper).toMatchSnapshot();
|
new Map<string, string>([
|
||||||
|
["terminal", "true"],
|
||||||
|
["terminalEndpoint", new URL(terminal.props.databaseAccount.properties.documentEndpoint).host],
|
||||||
|
])
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("renders mongo 3.6 shell", () => {
|
it("getTerminalParams: Test for Mongo 3.6 terminal", () => {
|
||||||
const props: NotebookTerminalComponentProps = {
|
const terminal: NotebookTerminalComponent = createMongo36Terminal();
|
||||||
databaseAccount: testMongo36Account,
|
const params: Map<string, string> = terminal.getTerminalParams();
|
||||||
notebookServerInfo: testMongoNotebookServerInfo,
|
|
||||||
};
|
|
||||||
|
|
||||||
const wrapper = shallow(<NotebookTerminalComponent {...props} />);
|
expect(params).toEqual(
|
||||||
expect(wrapper).toMatchSnapshot();
|
new Map<string, string>([
|
||||||
|
["terminal", "true"],
|
||||||
|
["terminalEndpoint", new URL(terminal.props.databaseAccount.properties.mongoEndpoint).host],
|
||||||
|
])
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("renders cassandra shell", () => {
|
it("getTerminalParams: Test for Cassandra terminal", () => {
|
||||||
const props: NotebookTerminalComponentProps = {
|
const terminal: NotebookTerminalComponent = createCassandraTerminal();
|
||||||
databaseAccount: testCassandraAccount,
|
const params: Map<string, string> = terminal.getTerminalParams();
|
||||||
notebookServerInfo: testCassandraNotebookServerInfo,
|
|
||||||
};
|
|
||||||
|
|
||||||
const wrapper = shallow(<NotebookTerminalComponent {...props} />);
|
expect(params).toEqual(
|
||||||
expect(wrapper).toMatchSnapshot();
|
new Map<string, string>([
|
||||||
|
["terminal", "true"],
|
||||||
|
["terminalEndpoint", new URL(terminal.props.databaseAccount.properties.cassandraEndpoint).host],
|
||||||
|
])
|
||||||
|
);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,12 +2,12 @@
|
|||||||
* Wrapper around Notebook server terminal
|
* Wrapper around Notebook server terminal
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import postRobot from "post-robot";
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import * as DataModels from "../../../Contracts/DataModels";
|
import * as DataModels from "../../../Contracts/DataModels";
|
||||||
import { TerminalProps } from "../../../Terminal/TerminalProps";
|
|
||||||
import { userContext } from "../../../UserContext";
|
|
||||||
import * as StringUtils from "../../../Utils/StringUtils";
|
import * as StringUtils from "../../../Utils/StringUtils";
|
||||||
|
import { userContext } from "../../../UserContext";
|
||||||
|
import { TerminalQueryParams } from "../../../Common/Constants";
|
||||||
|
import { handleError } from "../../../Common/ErrorHandlingUtils";
|
||||||
|
|
||||||
export interface NotebookTerminalComponentProps {
|
export interface NotebookTerminalComponentProps {
|
||||||
notebookServerInfo: DataModels.NotebookWorkspaceConnectionInfo;
|
notebookServerInfo: DataModels.NotebookWorkspaceConnectionInfo;
|
||||||
@@ -15,69 +15,79 @@ export interface NotebookTerminalComponentProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class NotebookTerminalComponent extends React.Component<NotebookTerminalComponentProps> {
|
export class NotebookTerminalComponent extends React.Component<NotebookTerminalComponentProps> {
|
||||||
private terminalWindow: Window;
|
|
||||||
|
|
||||||
constructor(props: NotebookTerminalComponentProps) {
|
constructor(props: NotebookTerminalComponentProps) {
|
||||||
super(props);
|
super(props);
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount(): void {
|
|
||||||
this.sendPropsToTerminalFrame();
|
|
||||||
}
|
|
||||||
|
|
||||||
public render(): JSX.Element {
|
public render(): JSX.Element {
|
||||||
return (
|
return (
|
||||||
<div className="notebookTerminalContainer">
|
<div className="notebookTerminalContainer">
|
||||||
<iframe
|
<iframe
|
||||||
title="Terminal to Notebook Server"
|
title="Terminal to Notebook Server"
|
||||||
onLoad={(event) => this.handleFrameLoad(event)}
|
src={NotebookTerminalComponent.createNotebookAppSrc(this.props.notebookServerInfo, this.getTerminalParams())}
|
||||||
src="terminal.html"
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
handleFrameLoad(event: React.SyntheticEvent<HTMLIFrameElement, Event>): void {
|
public getTerminalParams(): Map<string, string> {
|
||||||
this.terminalWindow = (event.target as HTMLIFrameElement).contentWindow;
|
let params: Map<string, string> = new Map<string, string>();
|
||||||
this.sendPropsToTerminalFrame();
|
params.set(TerminalQueryParams.Terminal, "true");
|
||||||
}
|
|
||||||
|
|
||||||
sendPropsToTerminalFrame(): void {
|
const terminalEndpoint: string = this.tryGetTerminalEndpoint();
|
||||||
if (!this.terminalWindow) {
|
if (terminalEndpoint) {
|
||||||
return;
|
params.set(TerminalQueryParams.TerminalEndpoint, terminalEndpoint);
|
||||||
}
|
}
|
||||||
|
|
||||||
const props: TerminalProps = {
|
return params;
|
||||||
terminalEndpoint: this.tryGetTerminalEndpoint(),
|
|
||||||
notebookServerEndpoint: this.props.notebookServerInfo?.notebookServerEndpoint,
|
|
||||||
authToken: this.props.notebookServerInfo?.authToken,
|
|
||||||
subscriptionId: userContext.subscriptionId,
|
|
||||||
apiType: userContext.apiType,
|
|
||||||
authType: userContext.authType,
|
|
||||||
databaseAccount: userContext.databaseAccount,
|
|
||||||
};
|
|
||||||
|
|
||||||
postRobot.send(this.terminalWindow, "props", props, {
|
|
||||||
domain: window.location.origin,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public tryGetTerminalEndpoint(): string | undefined {
|
public tryGetTerminalEndpoint(): string | null {
|
||||||
let terminalEndpoint: string | undefined;
|
let terminalEndpoint: string | null;
|
||||||
|
|
||||||
const notebookServerEndpoint = this.props.notebookServerInfo?.notebookServerEndpoint;
|
const notebookServerEndpoint: string = this.props.notebookServerInfo.notebookServerEndpoint;
|
||||||
if (StringUtils.endsWith(notebookServerEndpoint, "mongo")) {
|
if (StringUtils.endsWith(notebookServerEndpoint, "mongo")) {
|
||||||
// mongoEndpoint is only available for Mongo 3.6 and higher, fallback to documentEndpoint otherwise
|
let mongoShellEndpoint: string = this.props.databaseAccount.properties.mongoEndpoint;
|
||||||
terminalEndpoint =
|
if (!mongoShellEndpoint) {
|
||||||
this.props.databaseAccount?.properties.mongoEndpoint || this.props.databaseAccount?.properties.documentEndpoint;
|
// mongoEndpoint is only available for Mongo 3.6 and higher.
|
||||||
|
// Fallback to documentEndpoint otherwise.
|
||||||
|
mongoShellEndpoint = this.props.databaseAccount.properties.documentEndpoint;
|
||||||
|
}
|
||||||
|
terminalEndpoint = mongoShellEndpoint;
|
||||||
} else if (StringUtils.endsWith(notebookServerEndpoint, "cassandra")) {
|
} else if (StringUtils.endsWith(notebookServerEndpoint, "cassandra")) {
|
||||||
terminalEndpoint = this.props.databaseAccount?.properties.cassandraEndpoint;
|
terminalEndpoint = this.props.databaseAccount.properties.cassandraEndpoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (terminalEndpoint) {
|
if (terminalEndpoint) {
|
||||||
return new URL(terminalEndpoint).host;
|
return new URL(terminalEndpoint).host;
|
||||||
}
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
return undefined;
|
public static createNotebookAppSrc(
|
||||||
|
serverInfo: DataModels.NotebookWorkspaceConnectionInfo,
|
||||||
|
params: Map<string, string>
|
||||||
|
): string {
|
||||||
|
if (!serverInfo.notebookServerEndpoint) {
|
||||||
|
handleError(
|
||||||
|
"Notebook server endpoint not defined. Terminal will fail to connect to jupyter server.",
|
||||||
|
"NotebookTerminalComponent/createNotebookAppSrc"
|
||||||
|
);
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
|
||||||
|
params.set(TerminalQueryParams.Server, serverInfo.notebookServerEndpoint);
|
||||||
|
if (serverInfo.authToken && serverInfo.authToken.length > 0) {
|
||||||
|
params.set(TerminalQueryParams.Token, serverInfo.authToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
params.set(TerminalQueryParams.SubscriptionId, userContext.subscriptionId);
|
||||||
|
|
||||||
|
let result: string = "terminal.html?";
|
||||||
|
for (let key of params.keys()) {
|
||||||
|
result += `${key}=${encodeURIComponent(params.get(key))}&`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
|
||||||
|
|
||||||
exports[`NotebookTerminalComponent renders cassandra shell 1`] = `
|
|
||||||
<div
|
|
||||||
className="notebookTerminalContainer"
|
|
||||||
>
|
|
||||||
<iframe
|
|
||||||
onLoad={[Function]}
|
|
||||||
src="terminal.html"
|
|
||||||
title="Terminal to Notebook Server"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`NotebookTerminalComponent renders mongo 3.2 shell 1`] = `
|
|
||||||
<div
|
|
||||||
className="notebookTerminalContainer"
|
|
||||||
>
|
|
||||||
<iframe
|
|
||||||
onLoad={[Function]}
|
|
||||||
src="terminal.html"
|
|
||||||
title="Terminal to Notebook Server"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`NotebookTerminalComponent renders mongo 3.6 shell 1`] = `
|
|
||||||
<div
|
|
||||||
className="notebookTerminalContainer"
|
|
||||||
>
|
|
||||||
<iframe
|
|
||||||
onLoad={[Function]}
|
|
||||||
src="terminal.html"
|
|
||||||
title="Terminal to Notebook Server"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
|
|
||||||
exports[`NotebookTerminalComponent renders terminal 1`] = `
|
|
||||||
<div
|
|
||||||
className="notebookTerminalContainer"
|
|
||||||
>
|
|
||||||
<iframe
|
|
||||||
onLoad={[Function]}
|
|
||||||
src="terminal.html"
|
|
||||||
title="Terminal to Notebook Server"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
@@ -55,7 +55,7 @@ export class NotebookViewerComponent
|
|||||||
databaseAccountName: undefined,
|
databaseAccountName: undefined,
|
||||||
defaultExperience: "NotebookViewer",
|
defaultExperience: "NotebookViewer",
|
||||||
isReadOnly: true,
|
isReadOnly: true,
|
||||||
cellEditorType: "codemirror",
|
cellEditorType: "monaco",
|
||||||
autoSaveInterval: 365 * 24 * 3600 * 1000, // There is no way to turn off auto-save, set to 1 year
|
autoSaveInterval: 365 * 24 * 3600 * 1000, // There is no way to turn off auto-save, set to 1 year
|
||||||
contentProvider: contents.JupyterContentProvider, // NotebookViewer only knows how to talk to Jupyter contents API
|
contentProvider: contents.JupyterContentProvider, // NotebookViewer only knows how to talk to Jupyter contents API
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -29,9 +29,8 @@ import { QueriesClient } from "../../../Common/QueriesClient";
|
|||||||
import * as DataModels from "../../../Contracts/DataModels";
|
import * as DataModels from "../../../Contracts/DataModels";
|
||||||
import { Action } from "../../../Shared/Telemetry/TelemetryConstants";
|
import { Action } from "../../../Shared/Telemetry/TelemetryConstants";
|
||||||
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
||||||
import { useDialog } from "../Dialog";
|
|
||||||
|
|
||||||
const title = "Open Saved Queries";
|
const title: string = "Open Saved Queries";
|
||||||
|
|
||||||
export interface QueriesGridComponentProps {
|
export interface QueriesGridComponentProps {
|
||||||
queriesClient: QueriesClient;
|
queriesClient: QueriesClient;
|
||||||
@@ -197,9 +196,9 @@ export class QueriesGridComponent extends React.Component<QueriesGridComponentPr
|
|||||||
{
|
{
|
||||||
key: "Action",
|
key: "Action",
|
||||||
name: "Action",
|
name: "Action",
|
||||||
fieldName: undefined,
|
fieldName: null,
|
||||||
minWidth: 70,
|
minWidth: 70,
|
||||||
onRender: (query: Query) => {
|
onRender: (query: Query, index: number, column: IColumn) => {
|
||||||
const buttonProps: IButtonProps = {
|
const buttonProps: IButtonProps = {
|
||||||
iconProps: {
|
iconProps: {
|
||||||
iconName: "More",
|
iconName: "More",
|
||||||
@@ -215,50 +214,47 @@ export class QueriesGridComponent extends React.Component<QueriesGridComponentPr
|
|||||||
{
|
{
|
||||||
key: "Open",
|
key: "Open",
|
||||||
text: "Open query",
|
text: "Open query",
|
||||||
onClick: () => {
|
onClick: (event: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>, menuItem: any) => {
|
||||||
this.props.onQuerySelect(query);
|
this.props.onQuerySelect(query);
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
key: "Delete",
|
key: "Delete",
|
||||||
text: "Delete query",
|
text: "Delete query",
|
||||||
onClick: async () => {
|
onClick: async (
|
||||||
useDialog.getState().showOkCancelModalDialog(
|
event: React.MouseEvent<HTMLElement> | React.KeyboardEvent<HTMLElement>,
|
||||||
"Confirm delete",
|
menuItem: any
|
||||||
"Are you sure you want to delete this query?",
|
) => {
|
||||||
"Delete",
|
if (window.confirm("Are you sure you want to delete this query?")) {
|
||||||
async () => {
|
const container = window.dataExplorer;
|
||||||
const startKey: number = TelemetryProcessor.traceStart(Action.DeleteSavedQuery, {
|
const startKey: number = TelemetryProcessor.traceStart(Action.DeleteSavedQuery, {
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
paneTitle: title,
|
paneTitle: title,
|
||||||
});
|
});
|
||||||
try {
|
try {
|
||||||
await this.props.queriesClient.deleteQuery(query);
|
await this.props.queriesClient.deleteQuery(query);
|
||||||
TelemetryProcessor.traceSuccess(
|
TelemetryProcessor.traceSuccess(
|
||||||
Action.DeleteSavedQuery,
|
Action.DeleteSavedQuery,
|
||||||
{
|
{
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
paneTitle: title,
|
paneTitle: title,
|
||||||
},
|
},
|
||||||
startKey
|
startKey
|
||||||
);
|
);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
TelemetryProcessor.traceFailure(
|
TelemetryProcessor.traceFailure(
|
||||||
Action.DeleteSavedQuery,
|
Action.DeleteSavedQuery,
|
||||||
{
|
{
|
||||||
dataExplorerArea: Constants.Areas.ContextualPane,
|
dataExplorerArea: Constants.Areas.ContextualPane,
|
||||||
paneTitle: title,
|
paneTitle: title,
|
||||||
error: getErrorMessage(error),
|
error: getErrorMessage(error),
|
||||||
errorStack: getErrorStack(error),
|
errorStack: getErrorStack(error),
|
||||||
},
|
},
|
||||||
startKey
|
startKey
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
await this.fetchSavedQueries(); // get latest state
|
await this.fetchSavedQueries(); // get latest state
|
||||||
},
|
}
|
||||||
"Cancel",
|
|
||||||
undefined
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -38,6 +38,8 @@ describe("SettingsComponent", () => {
|
|||||||
title: "Scale & Settings",
|
title: "Scale & Settings",
|
||||||
tabPath: "",
|
tabPath: "",
|
||||||
node: undefined,
|
node: undefined,
|
||||||
|
hashLocation: "settings",
|
||||||
|
onUpdateTabsButtons: undefined,
|
||||||
}),
|
}),
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -126,6 +128,7 @@ describe("SettingsComponent", () => {
|
|||||||
isDatabaseExpanded: undefined,
|
isDatabaseExpanded: undefined,
|
||||||
isDatabaseShared: ko.computed(() => true),
|
isDatabaseShared: ko.computed(() => true),
|
||||||
selectedSubnodeKind: undefined,
|
selectedSubnodeKind: undefined,
|
||||||
|
selectDatabase: undefined,
|
||||||
expandDatabase: undefined,
|
expandDatabase: undefined,
|
||||||
collapseDatabase: undefined,
|
collapseDatabase: undefined,
|
||||||
loadCollections: undefined,
|
loadCollections: undefined,
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ import { userContext } from "../../../UserContext";
|
|||||||
import { MongoDBCollectionResource, MongoIndex } from "../../../Utils/arm/generatedClients/cosmos/types";
|
import { MongoDBCollectionResource, MongoIndex } from "../../../Utils/arm/generatedClients/cosmos/types";
|
||||||
import * as AutoPilotUtils from "../../../Utils/AutoPilotUtils";
|
import * as AutoPilotUtils from "../../../Utils/AutoPilotUtils";
|
||||||
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
||||||
import { useCommandBar } from "../../Menus/CommandBar/CommandBarComponentAdapter";
|
import Explorer from "../../Explorer";
|
||||||
import { SettingsTabV2 } from "../../Tabs/SettingsTabV2";
|
import { SettingsTabV2 } from "../../Tabs/SettingsTabV2";
|
||||||
import "./SettingsComponent.less";
|
import "./SettingsComponent.less";
|
||||||
import { mongoIndexingPolicyAADError } from "./SettingsRenderUtils";
|
import { mongoIndexingPolicyAADError } from "./SettingsRenderUtils";
|
||||||
@@ -121,6 +121,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
private collection: ViewModels.Collection;
|
private collection: ViewModels.Collection;
|
||||||
private database: ViewModels.Database;
|
private database: ViewModels.Database;
|
||||||
private offer: DataModels.Offer;
|
private offer: DataModels.Offer;
|
||||||
|
private container: Explorer;
|
||||||
private changeFeedPolicyVisible: boolean;
|
private changeFeedPolicyVisible: boolean;
|
||||||
private isFixedContainer: boolean;
|
private isFixedContainer: boolean;
|
||||||
private shouldShowIndexingPolicyEditor: boolean;
|
private shouldShowIndexingPolicyEditor: boolean;
|
||||||
@@ -132,6 +133,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
this.isCollectionSettingsTab = this.props.settingsTab.tabKind === ViewModels.CollectionTabKind.CollectionSettingsV2;
|
this.isCollectionSettingsTab = this.props.settingsTab.tabKind === ViewModels.CollectionTabKind.CollectionSettingsV2;
|
||||||
if (this.isCollectionSettingsTab) {
|
if (this.isCollectionSettingsTab) {
|
||||||
this.collection = this.props.settingsTab.collection as ViewModels.Collection;
|
this.collection = this.props.settingsTab.collection as ViewModels.Collection;
|
||||||
|
this.container = this.collection?.container;
|
||||||
this.offer = this.collection?.offer();
|
this.offer = this.collection?.offer();
|
||||||
this.isAnalyticalStorageEnabled = !!this.collection?.analyticalStorageTtl();
|
this.isAnalyticalStorageEnabled = !!this.collection?.analyticalStorageTtl();
|
||||||
this.shouldShowIndexingPolicyEditor = userContext.apiType !== "Cassandra" && userContext.apiType !== "Mongo";
|
this.shouldShowIndexingPolicyEditor = userContext.apiType !== "Cassandra" && userContext.apiType !== "Mongo";
|
||||||
@@ -143,6 +145,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
userContext.apiType === "Mongo" && (!this.collection?.partitionKey || this.collection?.partitionKey.systemKey);
|
userContext.apiType === "Mongo" && (!this.collection?.partitionKey || this.collection?.partitionKey.systemKey);
|
||||||
} else {
|
} else {
|
||||||
this.database = this.props.settingsTab.database;
|
this.database = this.props.settingsTab.database;
|
||||||
|
this.container = this.database?.container;
|
||||||
this.offer = this.database?.offer();
|
this.offer = this.database?.offer();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -219,13 +222,13 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
this.setAutoPilotStates();
|
this.setAutoPilotStates();
|
||||||
this.setBaseline();
|
this.setBaseline();
|
||||||
if (this.props.settingsTab.isActive()) {
|
if (this.props.settingsTab.isActive()) {
|
||||||
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
this.props.settingsTab.getContainer().onUpdateTabsButtons(this.getTabsButtons());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidUpdate(): void {
|
componentDidUpdate(): void {
|
||||||
if (this.props.settingsTab.isActive()) {
|
if (this.props.settingsTab.isActive()) {
|
||||||
useCommandBar.getState().setContextButtons(this.getTabsButtons());
|
this.props.settingsTab.getContainer().onUpdateTabsButtons(this.getTabsButtons());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -290,7 +293,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
this.state.wasAutopilotOriginallySet !== this.state.isAutoPilotSelected;
|
this.state.wasAutopilotOriginallySet !== this.state.isAutoPilotSelected;
|
||||||
|
|
||||||
public shouldShowKeyspaceSharedThroughputMessage = (): boolean =>
|
public shouldShowKeyspaceSharedThroughputMessage = (): boolean =>
|
||||||
userContext.apiType === "Cassandra" && hasDatabaseSharedThroughput(this.collection);
|
this.container && userContext.apiType === "Cassandra" && hasDatabaseSharedThroughput(this.collection);
|
||||||
|
|
||||||
public hasConflictResolution = (): boolean =>
|
public hasConflictResolution = (): boolean =>
|
||||||
userContext?.databaseAccount?.properties?.enableMultipleWriteLocations &&
|
userContext?.databaseAccount?.properties?.enableMultipleWriteLocations &&
|
||||||
@@ -880,6 +883,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
const scaleComponentProps: ScaleComponentProps = {
|
const scaleComponentProps: ScaleComponentProps = {
|
||||||
collection: this.collection,
|
collection: this.collection,
|
||||||
database: this.database,
|
database: this.database,
|
||||||
|
container: this.container,
|
||||||
isFixedContainer: this.isFixedContainer,
|
isFixedContainer: this.isFixedContainer,
|
||||||
onThroughputChange: this.onThroughputChange,
|
onThroughputChange: this.onThroughputChange,
|
||||||
throughput: this.state.throughput,
|
throughput: this.state.throughput,
|
||||||
@@ -907,6 +911,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
|
|
||||||
const subSettingsComponentProps: SubSettingsComponentProps = {
|
const subSettingsComponentProps: SubSettingsComponentProps = {
|
||||||
collection: this.collection,
|
collection: this.collection,
|
||||||
|
container: this.container,
|
||||||
isAnalyticalStorageEnabled: this.isAnalyticalStorageEnabled,
|
isAnalyticalStorageEnabled: this.isAnalyticalStorageEnabled,
|
||||||
changeFeedPolicyVisible: this.changeFeedPolicyVisible,
|
changeFeedPolicyVisible: this.changeFeedPolicyVisible,
|
||||||
timeToLive: this.state.timeToLive,
|
timeToLive: this.state.timeToLive,
|
||||||
@@ -959,6 +964,7 @@ export class SettingsComponent extends React.Component<SettingsComponentProps, S
|
|||||||
|
|
||||||
const conflictResolutionPolicyComponentProps: ConflictResolutionComponentProps = {
|
const conflictResolutionPolicyComponentProps: ConflictResolutionComponentProps = {
|
||||||
collection: this.collection,
|
collection: this.collection,
|
||||||
|
container: this.container,
|
||||||
conflictResolutionPolicyMode: this.state.conflictResolutionPolicyMode,
|
conflictResolutionPolicyMode: this.state.conflictResolutionPolicyMode,
|
||||||
conflictResolutionPolicyModeBaseline: this.state.conflictResolutionPolicyModeBaseline,
|
conflictResolutionPolicyModeBaseline: this.state.conflictResolutionPolicyModeBaseline,
|
||||||
onConflictResolutionPolicyModeChange: this.onConflictResolutionPolicyModeChange,
|
onConflictResolutionPolicyModeChange: this.onConflictResolutionPolicyModeChange,
|
||||||
|
|||||||
14
src/Explorer/Controls/Settings/SettingsComponentAdapter.tsx
Normal file
14
src/Explorer/Controls/Settings/SettingsComponentAdapter.tsx
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import ko from "knockout";
|
||||||
|
import * as React from "react";
|
||||||
|
import { ReactAdapter } from "../../../Bindings/ReactBindingHandler";
|
||||||
|
import { SettingsComponent, SettingsComponentProps } from "./SettingsComponent";
|
||||||
|
|
||||||
|
export class SettingsComponentAdapter implements ReactAdapter {
|
||||||
|
public parameters: ko.Computed<boolean>;
|
||||||
|
|
||||||
|
constructor(private props: SettingsComponentProps) {}
|
||||||
|
|
||||||
|
public renderComponent(): JSX.Element {
|
||||||
|
return this.parameters() ? <SettingsComponent {...this.props} /> : <></>;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
import { shallow } from "enzyme";
|
import { shallow } from "enzyme";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { ConflictResolutionComponentProps, ConflictResolutionComponent } from "./ConflictResolutionComponent";
|
||||||
|
import { container, collection } from "../TestUtils";
|
||||||
import * as DataModels from "../../../../Contracts/DataModels";
|
import * as DataModels from "../../../../Contracts/DataModels";
|
||||||
import { collection } from "../TestUtils";
|
|
||||||
import { ConflictResolutionComponent, ConflictResolutionComponentProps } from "./ConflictResolutionComponent";
|
|
||||||
|
|
||||||
describe("ConflictResolutionComponent", () => {
|
describe("ConflictResolutionComponent", () => {
|
||||||
const baseProps: ConflictResolutionComponentProps = {
|
const baseProps: ConflictResolutionComponentProps = {
|
||||||
collection: collection,
|
collection: collection,
|
||||||
|
container: container,
|
||||||
conflictResolutionPolicyMode: DataModels.ConflictResolutionMode.Custom,
|
conflictResolutionPolicyMode: DataModels.ConflictResolutionMode.Custom,
|
||||||
conflictResolutionPolicyModeBaseline: DataModels.ConflictResolutionMode.Custom,
|
conflictResolutionPolicyModeBaseline: DataModels.ConflictResolutionMode.Custom,
|
||||||
onConflictResolutionPolicyModeChange: () => {
|
onConflictResolutionPolicyModeChange: () => {
|
||||||
|
|||||||
@@ -1,19 +1,21 @@
|
|||||||
import { ChoiceGroup, IChoiceGroupOption, ITextFieldProps, Stack, TextField } from "@fluentui/react";
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import * as DataModels from "../../../../Contracts/DataModels";
|
|
||||||
import * as ViewModels from "../../../../Contracts/ViewModels";
|
import * as ViewModels from "../../../../Contracts/ViewModels";
|
||||||
|
import * as DataModels from "../../../../Contracts/DataModels";
|
||||||
|
import Explorer from "../../../Explorer";
|
||||||
import {
|
import {
|
||||||
conflictResolutionCustomToolTip,
|
|
||||||
conflictResolutionLwwTooltip,
|
|
||||||
getChoiceGroupStyles,
|
|
||||||
getTextFieldStyles,
|
getTextFieldStyles,
|
||||||
|
conflictResolutionLwwTooltip,
|
||||||
|
conflictResolutionCustomToolTip,
|
||||||
subComponentStackProps,
|
subComponentStackProps,
|
||||||
|
getChoiceGroupStyles,
|
||||||
} from "../SettingsRenderUtils";
|
} from "../SettingsRenderUtils";
|
||||||
import { isDirty } from "../SettingsUtils";
|
import { TextField, ITextFieldProps, Stack, IChoiceGroupOption, ChoiceGroup } from "@fluentui/react";
|
||||||
import { ToolTipLabelComponent } from "./ToolTipLabelComponent";
|
import { ToolTipLabelComponent } from "./ToolTipLabelComponent";
|
||||||
|
import { isDirty } from "../SettingsUtils";
|
||||||
|
|
||||||
export interface ConflictResolutionComponentProps {
|
export interface ConflictResolutionComponentProps {
|
||||||
collection: ViewModels.Collection;
|
collection: ViewModels.Collection;
|
||||||
|
container: Explorer;
|
||||||
conflictResolutionPolicyMode: DataModels.ConflictResolutionMode;
|
conflictResolutionPolicyMode: DataModels.ConflictResolutionMode;
|
||||||
conflictResolutionPolicyModeBaseline: DataModels.ConflictResolutionMode;
|
conflictResolutionPolicyModeBaseline: DataModels.ConflictResolutionMode;
|
||||||
onConflictResolutionPolicyModeChange: (newMode: DataModels.ConflictResolutionMode) => void;
|
onConflictResolutionPolicyModeChange: (newMode: DataModels.ConflictResolutionMode) => void;
|
||||||
|
|||||||
@@ -7,17 +7,20 @@ import * as SharedConstants from "../../../../Shared/Constants";
|
|||||||
import { updateUserContext } from "../../../../UserContext";
|
import { updateUserContext } from "../../../../UserContext";
|
||||||
import Explorer from "../../../Explorer";
|
import Explorer from "../../../Explorer";
|
||||||
import { throughputUnit } from "../SettingsRenderUtils";
|
import { throughputUnit } from "../SettingsRenderUtils";
|
||||||
import { collection } from "../TestUtils";
|
import { collection, container } from "../TestUtils";
|
||||||
import { ScaleComponent, ScaleComponentProps } from "./ScaleComponent";
|
import { ScaleComponent, ScaleComponentProps } from "./ScaleComponent";
|
||||||
import { ThroughputInputAutoPilotV3Component } from "./ThroughputInputComponents/ThroughputInputAutoPilotV3Component";
|
import { ThroughputInputAutoPilotV3Component } from "./ThroughputInputComponents/ThroughputInputAutoPilotV3Component";
|
||||||
|
|
||||||
describe("ScaleComponent", () => {
|
describe("ScaleComponent", () => {
|
||||||
const nonNationalCloudContainer = new Explorer();
|
const nonNationalCloudContainer = new Explorer();
|
||||||
|
nonNationalCloudContainer.isRunningOnNationalCloud = () => false;
|
||||||
|
|
||||||
const targetThroughput = 6000;
|
const targetThroughput = 6000;
|
||||||
|
|
||||||
const baseProps: ScaleComponentProps = {
|
const baseProps: ScaleComponentProps = {
|
||||||
collection: collection,
|
collection: collection,
|
||||||
database: undefined,
|
database: undefined,
|
||||||
|
container: container,
|
||||||
isFixedContainer: false,
|
isFixedContainer: false,
|
||||||
onThroughputChange: () => {
|
onThroughputChange: () => {
|
||||||
return;
|
return;
|
||||||
@@ -108,7 +111,7 @@ describe("ScaleComponent", () => {
|
|||||||
let scaleComponent = new ScaleComponent(baseProps);
|
let scaleComponent = new ScaleComponent(baseProps);
|
||||||
expect(scaleComponent.getThroughputTitle()).toEqual("Throughput (6,000 - unlimited RU/s)");
|
expect(scaleComponent.getThroughputTitle()).toEqual("Throughput (6,000 - unlimited RU/s)");
|
||||||
|
|
||||||
let newProps = { ...baseProps };
|
let newProps = { ...baseProps, container: nonNationalCloudContainer };
|
||||||
scaleComponent = new ScaleComponent(newProps);
|
scaleComponent = new ScaleComponent(newProps);
|
||||||
expect(scaleComponent.getThroughputTitle()).toEqual("Throughput (6,000 - unlimited RU/s)");
|
expect(scaleComponent.getThroughputTitle()).toEqual("Throughput (6,000 - unlimited RU/s)");
|
||||||
|
|
||||||
@@ -121,7 +124,7 @@ describe("ScaleComponent", () => {
|
|||||||
let scaleComponent = new ScaleComponent(baseProps);
|
let scaleComponent = new ScaleComponent(baseProps);
|
||||||
expect(scaleComponent.canThroughputExceedMaximumValue()).toEqual(true);
|
expect(scaleComponent.canThroughputExceedMaximumValue()).toEqual(true);
|
||||||
|
|
||||||
const newProps = { ...baseProps };
|
const newProps = { ...baseProps, container: nonNationalCloudContainer };
|
||||||
scaleComponent = new ScaleComponent(newProps);
|
scaleComponent = new ScaleComponent(newProps);
|
||||||
expect(scaleComponent.canThroughputExceedMaximumValue()).toEqual(true);
|
expect(scaleComponent.canThroughputExceedMaximumValue()).toEqual(true);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import * as ViewModels from "../../../../Contracts/ViewModels";
|
|||||||
import * as SharedConstants from "../../../../Shared/Constants";
|
import * as SharedConstants from "../../../../Shared/Constants";
|
||||||
import { userContext } from "../../../../UserContext";
|
import { userContext } from "../../../../UserContext";
|
||||||
import * as AutoPilotUtils from "../../../../Utils/AutoPilotUtils";
|
import * as AutoPilotUtils from "../../../../Utils/AutoPilotUtils";
|
||||||
import { isRunningOnNationalCloud } from "../../../../Utils/CloudUtils";
|
import Explorer from "../../../Explorer";
|
||||||
import {
|
import {
|
||||||
getTextFieldStyles,
|
getTextFieldStyles,
|
||||||
getThroughputApplyLongDelayMessage,
|
getThroughputApplyLongDelayMessage,
|
||||||
@@ -23,6 +23,7 @@ import { ThroughputInputAutoPilotV3Component } from "./ThroughputInputComponents
|
|||||||
export interface ScaleComponentProps {
|
export interface ScaleComponentProps {
|
||||||
collection: ViewModels.Collection;
|
collection: ViewModels.Collection;
|
||||||
database: ViewModels.Database;
|
database: ViewModels.Database;
|
||||||
|
container: Explorer;
|
||||||
isFixedContainer: boolean;
|
isFixedContainer: boolean;
|
||||||
onThroughputChange: (newThroughput: number) => void;
|
onThroughputChange: (newThroughput: number) => void;
|
||||||
throughput: number;
|
throughput: number;
|
||||||
@@ -108,7 +109,11 @@ export class ScaleComponent extends React.Component<ScaleComponentProps> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
public canThroughputExceedMaximumValue = (): boolean => {
|
public canThroughputExceedMaximumValue = (): boolean => {
|
||||||
return !this.props.isFixedContainer && configContext.platform === Platform.Portal && !isRunningOnNationalCloud();
|
return (
|
||||||
|
!this.props.isFixedContainer &&
|
||||||
|
configContext.platform === Platform.Portal &&
|
||||||
|
!this.props.container.isRunningOnNationalCloud()
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
public getInitialNotificationElement = (): JSX.Element => {
|
public getInitialNotificationElement = (): JSX.Element => {
|
||||||
|
|||||||
@@ -4,12 +4,14 @@ import { DatabaseAccount } from "../../../../Contracts/DataModels";
|
|||||||
import { updateUserContext } from "../../../../UserContext";
|
import { updateUserContext } from "../../../../UserContext";
|
||||||
import Explorer from "../../../Explorer";
|
import Explorer from "../../../Explorer";
|
||||||
import { ChangeFeedPolicyState, GeospatialConfigType, TtlOff, TtlOn, TtlOnNoDefault, TtlType } from "../SettingsUtils";
|
import { ChangeFeedPolicyState, GeospatialConfigType, TtlOff, TtlOn, TtlOnNoDefault, TtlType } from "../SettingsUtils";
|
||||||
import { collection } from "../TestUtils";
|
import { collection, container } from "../TestUtils";
|
||||||
import { SubSettingsComponent, SubSettingsComponentProps } from "./SubSettingsComponent";
|
import { SubSettingsComponent, SubSettingsComponentProps } from "./SubSettingsComponent";
|
||||||
|
|
||||||
describe("SubSettingsComponent", () => {
|
describe("SubSettingsComponent", () => {
|
||||||
const baseProps: SubSettingsComponentProps = {
|
const baseProps: SubSettingsComponentProps = {
|
||||||
collection: collection,
|
collection: collection,
|
||||||
|
container: container,
|
||||||
|
|
||||||
timeToLive: TtlType.On,
|
timeToLive: TtlType.On,
|
||||||
timeToLiveBaseline: TtlType.On,
|
timeToLiveBaseline: TtlType.On,
|
||||||
onTtlChange: () => {
|
onTtlChange: () => {
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { ChoiceGroup, IChoiceGroupOption, Label, Link, MessageBar, Stack, Text,
|
|||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import * as ViewModels from "../../../../Contracts/ViewModels";
|
import * as ViewModels from "../../../../Contracts/ViewModels";
|
||||||
import { userContext } from "../../../../UserContext";
|
import { userContext } from "../../../../UserContext";
|
||||||
|
import Explorer from "../../../Explorer";
|
||||||
import { Int32 } from "../../../Panes/Tables/Validators/EntityPropertyValidationCommon";
|
import { Int32 } from "../../../Panes/Tables/Validators/EntityPropertyValidationCommon";
|
||||||
import {
|
import {
|
||||||
changeFeedPolicyToolTip,
|
changeFeedPolicyToolTip,
|
||||||
@@ -27,6 +28,8 @@ import { ToolTipLabelComponent } from "./ToolTipLabelComponent";
|
|||||||
|
|
||||||
export interface SubSettingsComponentProps {
|
export interface SubSettingsComponentProps {
|
||||||
collection: ViewModels.Collection;
|
collection: ViewModels.Collection;
|
||||||
|
container: Explorer;
|
||||||
|
|
||||||
timeToLive: TtlType;
|
timeToLive: TtlType;
|
||||||
timeToLiveBaseline: TtlType;
|
timeToLiveBaseline: TtlType;
|
||||||
|
|
||||||
|
|||||||
@@ -36,6 +36,7 @@ describe("SettingsUtils", () => {
|
|||||||
isDatabaseExpanded: ko.observable(false),
|
isDatabaseExpanded: ko.observable(false),
|
||||||
isDatabaseShared: ko.computed(() => true),
|
isDatabaseShared: ko.computed(() => true),
|
||||||
selectedSubnodeKind: ko.observable(undefined),
|
selectedSubnodeKind: ko.observable(undefined),
|
||||||
|
selectDatabase: undefined,
|
||||||
expandDatabase: undefined,
|
expandDatabase: undefined,
|
||||||
collapseDatabase: undefined,
|
collapseDatabase: undefined,
|
||||||
loadCollections: undefined,
|
loadCollections: undefined,
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -58,7 +58,7 @@ export class TabComponent extends React.Component<TabComponentProps> {
|
|||||||
as="span"
|
as="span"
|
||||||
className={className}
|
className={className}
|
||||||
role="presentation"
|
role="presentation"
|
||||||
onActivated={() => this.setActiveTab(index)}
|
onActivated={(e) => this.setActiveTab(index)}
|
||||||
aria-label={`Select tab: ${tab.title}`}
|
aria-label={`Select tab: ${tab.title}`}
|
||||||
>
|
>
|
||||||
{tab.title}
|
{tab.title}
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import { ThroughputInput } from "./ThroughputInput";
|
|||||||
const props = {
|
const props = {
|
||||||
isDatabase: false,
|
isDatabase: false,
|
||||||
showFreeTierExceedThroughputTooltip: true,
|
showFreeTierExceedThroughputTooltip: true,
|
||||||
isSharded: true,
|
isSharded: false,
|
||||||
setThroughputValue: () => jest.fn(),
|
setThroughputValue: () => jest.fn(),
|
||||||
setIsAutoscale: () => jest.fn(),
|
setIsAutoscale: () => jest.fn(),
|
||||||
onCostAcknowledgeChange: () => jest.fn(),
|
onCostAcknowledgeChange: () => jest.fn(),
|
||||||
|
|||||||
@@ -41,16 +41,9 @@ export const ThroughputInput: FunctionComponent<ThroughputInputProps> = ({
|
|||||||
throughputHeaderText = AutoPilotUtils.getAutoPilotHeaderText().toLocaleLowerCase();
|
throughputHeaderText = AutoPilotUtils.getAutoPilotHeaderText().toLocaleLowerCase();
|
||||||
} else {
|
} else {
|
||||||
const minRU: string = SharedConstants.CollectionCreation.DefaultCollectionRUs400.toLocaleString();
|
const minRU: string = SharedConstants.CollectionCreation.DefaultCollectionRUs400.toLocaleString();
|
||||||
|
const maxRU: string = userContext.isTryCosmosDBSubscription
|
||||||
let maxRU: string;
|
? Constants.TryCosmosExperience.maxRU.toLocaleString()
|
||||||
if (userContext.isTryCosmosDBSubscription) {
|
: "unlimited";
|
||||||
maxRU = Constants.TryCosmosExperience.maxRU.toLocaleString();
|
|
||||||
} else if (!isSharded) {
|
|
||||||
maxRU = "10000";
|
|
||||||
} else {
|
|
||||||
maxRU = "unlimited";
|
|
||||||
}
|
|
||||||
|
|
||||||
throughputHeaderText = `throughput (${minRU} - ${maxRU} RU/s)`;
|
throughputHeaderText = `throughput (${minRU} - ${maxRU} RU/s)`;
|
||||||
}
|
}
|
||||||
return `${isDatabase ? "Database" : getCollectionName()} ${throughputHeaderText}`;
|
return `${isDatabase ? "Database" : getCollectionName()} ${throughputHeaderText}`;
|
||||||
|
|||||||
@@ -0,0 +1,308 @@
|
|||||||
|
import * as ko from "knockout";
|
||||||
|
import { KeyCodes } from "../../../Common/Constants";
|
||||||
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
|
import { FreeTierLimits } from "../../../Shared/Constants";
|
||||||
|
import { Action, ActionModifiers } from "../../../Shared/Telemetry/TelemetryConstants";
|
||||||
|
import * as TelemetryProcessor from "../../../Shared/Telemetry/TelemetryProcessor";
|
||||||
|
import * as AutoPilotUtils from "../../../Utils/AutoPilotUtils";
|
||||||
|
import { WaitsForTemplateViewModel } from "../../WaitsForTemplateViewModel";
|
||||||
|
import ThroughputInputComponentAutoscaleV3 from "./ThroughputInputComponentAutoscaleV3.html";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Throughput Input:
|
||||||
|
*
|
||||||
|
* Creates a set of controls to input, sanitize and increase/decrease throughput
|
||||||
|
*
|
||||||
|
* How to use in your markup:
|
||||||
|
* <throughput-input params="{ value: anObservableToHoldTheValue, minimum: anObservableWithMinimum, maximum: anObservableWithMaximum }">
|
||||||
|
* </throughput-input>
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Parameters for this component
|
||||||
|
*/
|
||||||
|
export interface ThroughputInputParams {
|
||||||
|
/**
|
||||||
|
* Callback triggered when the template is bound to the component (for testing purposes)
|
||||||
|
*/
|
||||||
|
onTemplateReady?: () => void;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Observable to bind the Throughput value to
|
||||||
|
*/
|
||||||
|
value: ViewModels.Editable<number>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text to use as id for testing
|
||||||
|
*/
|
||||||
|
testId: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text to use as aria-label
|
||||||
|
*/
|
||||||
|
ariaLabel?: ko.Observable<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Minimum value in the range
|
||||||
|
*/
|
||||||
|
minimum: ko.Observable<number>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Maximum value in the range
|
||||||
|
*/
|
||||||
|
maximum: ko.Observable<number>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Step value for increase/decrease
|
||||||
|
*/
|
||||||
|
step?: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Observable to bind the Throughput enabled status
|
||||||
|
*/
|
||||||
|
isEnabled?: ko.Observable<boolean>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Should show pricing controls
|
||||||
|
*/
|
||||||
|
costsVisible: ko.Observable<boolean>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* RU price
|
||||||
|
*/
|
||||||
|
requestUnitsUsageCost: ko.Computed<string>; // Our code assigns to ko.Computed, but unit test assigns to ko.Observable
|
||||||
|
|
||||||
|
/**
|
||||||
|
* State of the spending acknowledge checkbox
|
||||||
|
*/
|
||||||
|
spendAckChecked?: ko.Observable<boolean>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* id of the spending acknowledge checkbox
|
||||||
|
*/
|
||||||
|
spendAckId?: ko.Observable<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* spending acknowledge text
|
||||||
|
*/
|
||||||
|
spendAckText?: ko.Observable<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Show spending acknowledge controls
|
||||||
|
*/
|
||||||
|
spendAckVisible?: ko.Observable<boolean>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Display * to the left of the label
|
||||||
|
*/
|
||||||
|
showAsMandatory: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If true, it will display a text to prompt users to use unlimited collections to go beyond max for fixed
|
||||||
|
*/
|
||||||
|
isFixed: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Label of the provisioned throughut control
|
||||||
|
*/
|
||||||
|
label: ko.Observable<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Text of the info bubble for provisioned throughut control
|
||||||
|
*/
|
||||||
|
infoBubbleText?: ko.Observable<string>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Computed value that decides if value can exceed maximum allowable value
|
||||||
|
*/
|
||||||
|
canExceedMaximumValue?: ko.Computed<boolean>;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CSS classes to apply on input element
|
||||||
|
*/
|
||||||
|
cssClass?: string;
|
||||||
|
|
||||||
|
isAutoPilotSelected: ko.Observable<boolean>;
|
||||||
|
throughputAutoPilotRadioId: string;
|
||||||
|
throughputProvisionedRadioId: string;
|
||||||
|
throughputModeRadioName: string;
|
||||||
|
maxAutoPilotThroughputSet: ViewModels.Editable<number>;
|
||||||
|
autoPilotUsageCost: ko.Computed<string>;
|
||||||
|
overrideWithAutoPilotSettings: ko.Observable<boolean>;
|
||||||
|
overrideWithProvisionedThroughputSettings: ko.Observable<boolean>;
|
||||||
|
freeTierExceedThroughputTooltip?: ko.Observable<string>;
|
||||||
|
freeTierExceedThroughputWarning?: ko.Observable<string>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ThroughputInputViewModel extends WaitsForTemplateViewModel {
|
||||||
|
public ariaLabel: ko.Observable<string>;
|
||||||
|
public canExceedMaximumValue: ko.Computed<boolean>;
|
||||||
|
public step: ko.Computed<number>;
|
||||||
|
public testId: string;
|
||||||
|
public value: ViewModels.Editable<number>;
|
||||||
|
public minimum: ko.Observable<number>;
|
||||||
|
public maximum: ko.Observable<number>;
|
||||||
|
public isEnabled: ko.Observable<boolean>;
|
||||||
|
public cssClass: string;
|
||||||
|
public decreaseButtonAriaLabel: string;
|
||||||
|
public increaseButtonAriaLabel: string;
|
||||||
|
public costsVisible: ko.Observable<boolean>;
|
||||||
|
public requestUnitsUsageCost: ko.Computed<string>;
|
||||||
|
public spendAckChecked: ko.Observable<boolean>;
|
||||||
|
public spendAckId: ko.Observable<string>;
|
||||||
|
public spendAckText: ko.Observable<string>;
|
||||||
|
public spendAckVisible: ko.Observable<boolean>;
|
||||||
|
public showAsMandatory: boolean;
|
||||||
|
public infoBubbleText: string | ko.Observable<string>;
|
||||||
|
public label: ko.Observable<string>;
|
||||||
|
public isFixed: boolean;
|
||||||
|
public isAutoPilotSelected: ko.Observable<boolean>;
|
||||||
|
public throughputAutoPilotRadioId: string;
|
||||||
|
public throughputProvisionedRadioId: string;
|
||||||
|
public throughputModeRadioName: string;
|
||||||
|
public maxAutoPilotThroughputSet: ko.Observable<number>;
|
||||||
|
public autoPilotUsageCost: ko.Computed<string>;
|
||||||
|
public minAutoPilotThroughput: ko.Observable<number>;
|
||||||
|
public overrideWithAutoPilotSettings: ko.Observable<boolean>;
|
||||||
|
public overrideWithProvisionedThroughputSettings: ko.Observable<boolean>;
|
||||||
|
public isManualThroughputInputFieldRequired: ko.Computed<boolean>;
|
||||||
|
public isAutoscaleThroughputInputFieldRequired: ko.Computed<boolean>;
|
||||||
|
public freeTierExceedThroughputTooltip: ko.Observable<string>;
|
||||||
|
public freeTierExceedThroughputWarning: ko.Observable<string>;
|
||||||
|
public showFreeTierExceedThroughputTooltip: ko.Computed<boolean>;
|
||||||
|
public showFreeTierExceedThroughputWarning: ko.Computed<boolean>;
|
||||||
|
|
||||||
|
public constructor(options: ThroughputInputParams) {
|
||||||
|
super();
|
||||||
|
super.onTemplateReady((isTemplateReady: boolean) => {
|
||||||
|
if (isTemplateReady && options.onTemplateReady) {
|
||||||
|
options.onTemplateReady();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const params: ThroughputInputParams = options;
|
||||||
|
this.testId = params.testId || "ThroughputValue";
|
||||||
|
this.ariaLabel = ko.observable((params.ariaLabel && params.ariaLabel()) || "");
|
||||||
|
this.canExceedMaximumValue = params.canExceedMaximumValue || ko.computed(() => false);
|
||||||
|
this.isEnabled = params.isEnabled || ko.observable(true);
|
||||||
|
this.cssClass = params.cssClass || "textfontclr collid migration";
|
||||||
|
this.minimum = params.minimum;
|
||||||
|
this.maximum = params.maximum;
|
||||||
|
this.value = params.value;
|
||||||
|
this.costsVisible = options.costsVisible;
|
||||||
|
this.requestUnitsUsageCost = options.requestUnitsUsageCost;
|
||||||
|
this.spendAckChecked = options.spendAckChecked || ko.observable<boolean>(false);
|
||||||
|
this.spendAckId = options.spendAckId || ko.observable<string>();
|
||||||
|
this.spendAckText = options.spendAckText || ko.observable<string>();
|
||||||
|
this.spendAckVisible = options.spendAckVisible || ko.observable<boolean>(false);
|
||||||
|
this.showAsMandatory = !!options.showAsMandatory;
|
||||||
|
this.isFixed = !!options.isFixed;
|
||||||
|
this.infoBubbleText = options.infoBubbleText || ko.observable<string>();
|
||||||
|
this.label = options.label || ko.observable<string>();
|
||||||
|
this.isAutoPilotSelected = options.isAutoPilotSelected || ko.observable<boolean>(false);
|
||||||
|
this.isAutoPilotSelected.subscribe((value) => {
|
||||||
|
TelemetryProcessor.trace(Action.ToggleAutoscaleSetting, ActionModifiers.Mark, {
|
||||||
|
changedSelectedValueTo: value ? ActionModifiers.ToggleAutoscaleOn : ActionModifiers.ToggleAutoscaleOff,
|
||||||
|
dataExplorerArea: "Scale Tab V1",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
this.throughputAutoPilotRadioId = options.throughputAutoPilotRadioId;
|
||||||
|
this.throughputProvisionedRadioId = options.throughputProvisionedRadioId;
|
||||||
|
this.throughputModeRadioName = options.throughputModeRadioName;
|
||||||
|
this.overrideWithAutoPilotSettings = options.overrideWithAutoPilotSettings || ko.observable<boolean>(false);
|
||||||
|
this.overrideWithProvisionedThroughputSettings =
|
||||||
|
options.overrideWithProvisionedThroughputSettings || ko.observable<boolean>(false);
|
||||||
|
|
||||||
|
this.maxAutoPilotThroughputSet =
|
||||||
|
options.maxAutoPilotThroughputSet || ko.observable<number>(AutoPilotUtils.minAutoPilotThroughput);
|
||||||
|
this.autoPilotUsageCost = options.autoPilotUsageCost;
|
||||||
|
this.minAutoPilotThroughput = ko.observable<number>(AutoPilotUtils.minAutoPilotThroughput);
|
||||||
|
|
||||||
|
this.step = ko.pureComputed(() => {
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
return AutoPilotUtils.autoPilotIncrementStep;
|
||||||
|
}
|
||||||
|
return params.step || ThroughputInputViewModel._defaultStep;
|
||||||
|
});
|
||||||
|
this.decreaseButtonAriaLabel = "Decrease throughput by " + this.step().toString();
|
||||||
|
this.increaseButtonAriaLabel = "Increase throughput by " + this.step().toString();
|
||||||
|
this.isManualThroughputInputFieldRequired = ko.pureComputed(() => this.isEnabled() && !this.isAutoPilotSelected());
|
||||||
|
this.isAutoscaleThroughputInputFieldRequired = ko.pureComputed(
|
||||||
|
() => this.isEnabled() && this.isAutoPilotSelected()
|
||||||
|
);
|
||||||
|
|
||||||
|
this.freeTierExceedThroughputTooltip = options.freeTierExceedThroughputTooltip || ko.observable<string>();
|
||||||
|
this.freeTierExceedThroughputWarning = options.freeTierExceedThroughputWarning || ko.observable<string>();
|
||||||
|
this.showFreeTierExceedThroughputTooltip = ko.pureComputed<boolean>(
|
||||||
|
() => !!this.freeTierExceedThroughputTooltip() && this.value() > FreeTierLimits.RU
|
||||||
|
);
|
||||||
|
|
||||||
|
this.showFreeTierExceedThroughputWarning = ko.pureComputed<boolean>(
|
||||||
|
() => !!this.freeTierExceedThroughputWarning() && this.value() > FreeTierLimits.RU
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public decreaseThroughput() {
|
||||||
|
let offerThroughput: number = this._getSanitizedValue();
|
||||||
|
|
||||||
|
if (offerThroughput > this.minimum()) {
|
||||||
|
offerThroughput -= this.step();
|
||||||
|
if (offerThroughput < this.minimum()) {
|
||||||
|
offerThroughput = this.minimum();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.value(offerThroughput);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public increaseThroughput() {
|
||||||
|
let offerThroughput: number = this._getSanitizedValue();
|
||||||
|
|
||||||
|
if (offerThroughput < this.maximum() || this.canExceedMaximumValue()) {
|
||||||
|
offerThroughput += this.step();
|
||||||
|
if (offerThroughput > this.maximum() && !this.canExceedMaximumValue()) {
|
||||||
|
offerThroughput = this.maximum();
|
||||||
|
}
|
||||||
|
|
||||||
|
this.value(offerThroughput);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public onIncreaseKeyDown = (source: any, event: KeyboardEvent): boolean => {
|
||||||
|
if (event.keyCode === KeyCodes.Enter || event.keyCode === KeyCodes.Space) {
|
||||||
|
this.increaseThroughput();
|
||||||
|
event.stopPropagation();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
public onDecreaseKeyDown = (source: any, event: KeyboardEvent): boolean => {
|
||||||
|
if (event.keyCode === KeyCodes.Enter || event.keyCode === KeyCodes.Space) {
|
||||||
|
this.decreaseThroughput();
|
||||||
|
event.stopPropagation();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
};
|
||||||
|
|
||||||
|
private _getSanitizedValue(): number {
|
||||||
|
let throughput = this.value();
|
||||||
|
|
||||||
|
if (this.isAutoPilotSelected()) {
|
||||||
|
throughput = this.maxAutoPilotThroughputSet();
|
||||||
|
}
|
||||||
|
return isNaN(throughput) ? 0 : Number(throughput);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static _defaultStep: number = 100;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ThroughputInputComponentAutoPilotV3 = {
|
||||||
|
viewModel: ThroughputInputViewModel,
|
||||||
|
template: ThroughputInputComponentAutoscaleV3,
|
||||||
|
};
|
||||||
@@ -0,0 +1,194 @@
|
|||||||
|
<div>
|
||||||
|
<div>
|
||||||
|
<p class="pkPadding">
|
||||||
|
<!-- ko if: showAsMandatory -->
|
||||||
|
<span class="mandatoryStar">*</span>
|
||||||
|
<!-- /ko -->
|
||||||
|
|
||||||
|
<span data-bind="text: label"></span>
|
||||||
|
|
||||||
|
<!-- ko if: infoBubbleText -->
|
||||||
|
<span class="infoTooltip" role="tooltip" tabindex="0">
|
||||||
|
<img class="infoImg" src="../../../../images/info-bubble.svg" alt="More information" />
|
||||||
|
<span data-bind="text: infoBubbleText" class="tooltiptext throughputRuInfo"></span>
|
||||||
|
</span>
|
||||||
|
<!-- /ko -->
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- ko if: !isFixed -->
|
||||||
|
<div class="throughputModeContainer">
|
||||||
|
<input
|
||||||
|
class="throughputModeRadio"
|
||||||
|
aria-label="Autopilot mode"
|
||||||
|
type="radio"
|
||||||
|
role="radio"
|
||||||
|
tabindex="0"
|
||||||
|
data-bind="
|
||||||
|
checked: isAutoPilotSelected,
|
||||||
|
checkedValue: true,
|
||||||
|
attr: {
|
||||||
|
id: throughputAutoPilotRadioId,
|
||||||
|
name: throughputModeRadioName,
|
||||||
|
'aria-checked': isAutoPilotSelected() ? 'true' : 'false'
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
class="throughputModeSpace"
|
||||||
|
data-bind="
|
||||||
|
attr: {
|
||||||
|
for: throughputAutoPilotRadioId
|
||||||
|
}"
|
||||||
|
>Autoscale
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<input
|
||||||
|
class="throughputModeRadio nonFirstRadio"
|
||||||
|
aria-label="Manual mode"
|
||||||
|
type="radio"
|
||||||
|
role="radio"
|
||||||
|
tabindex="0"
|
||||||
|
data-bind="
|
||||||
|
checked: isAutoPilotSelected,
|
||||||
|
checkedValue: false,
|
||||||
|
attr: {
|
||||||
|
id: throughputProvisionedRadioId,
|
||||||
|
name: throughputModeRadioName,
|
||||||
|
'aria-checked': !isAutoPilotSelected() ? 'true' : 'false'
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
<span
|
||||||
|
class="throughputModeSpace"
|
||||||
|
data-bind="attr: {
|
||||||
|
for: throughputProvisionedRadioId
|
||||||
|
}"
|
||||||
|
>Manual
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<!-- /ko -->
|
||||||
|
|
||||||
|
<div data-bind="visible: isAutoPilotSelected">
|
||||||
|
<p>
|
||||||
|
<span
|
||||||
|
>Provision maximum RU/s required by this resource. Estimate your required RU/s with
|
||||||
|
<a target="_blank" href="https://cosmos.azure.com/capacitycalculator/">capacity calculator</a>.</span
|
||||||
|
>
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<span>Max RU/s</span>
|
||||||
|
</p>
|
||||||
|
<div data-bind="setTemplateReady: true">
|
||||||
|
<input
|
||||||
|
data-bind="textInput: overrideWithProvisionedThroughputSettings() ? '' : maxAutoPilotThroughputSet, attr:{
|
||||||
|
disabled: overrideWithProvisionedThroughputSettings(),
|
||||||
|
step: step,
|
||||||
|
'class':'migration collid select-font-size',
|
||||||
|
min: minAutoPilotThroughput,
|
||||||
|
'aria-label': 'Max request units per second',
|
||||||
|
type: isAutoscaleThroughputInputFieldRequired() ? 'number' : 'hidden',
|
||||||
|
css: {
|
||||||
|
dirty: maxAutoPilotThroughputSet.editableIsDirty
|
||||||
|
}
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<p data-bind="visible: overrideWithProvisionedThroughputSettings && !overrideWithProvisionedThroughputSettings()">
|
||||||
|
<span
|
||||||
|
data-bind="
|
||||||
|
html: autoPilotUsageCost"
|
||||||
|
></span>
|
||||||
|
</p>
|
||||||
|
<p
|
||||||
|
data-bind="visible: costsVisible && overrideWithProvisionedThroughputSettings && !overrideWithProvisionedThroughputSettings()"
|
||||||
|
>
|
||||||
|
<span data-bind="html: requestUnitsUsageCost"></span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- ko if: spendAckVisible -->
|
||||||
|
<p class="pkPadding">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
aria-label="acknowledge spend throughput"
|
||||||
|
data-bind="
|
||||||
|
attr: {
|
||||||
|
title: spendAckText,
|
||||||
|
id: spendAckId
|
||||||
|
},
|
||||||
|
checked: spendAckChecked"
|
||||||
|
/>
|
||||||
|
<span data-bind="text: spendAckText, attr: { for: spendAckId }"></span>
|
||||||
|
</p>
|
||||||
|
<!-- /ko -->
|
||||||
|
|
||||||
|
<!-- ko if: isFixed -->
|
||||||
|
<p>Choose unlimited storage capacity for more than 10,000 RU/s.</p>
|
||||||
|
<!-- /ko -->
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div data-bind="visible: !isAutoPilotSelected()">
|
||||||
|
<p>
|
||||||
|
<span
|
||||||
|
>Estimate your required throughput with
|
||||||
|
<a target="_blank" href="https://cosmos.azure.com/capacitycalculator/">capacity calculator</a></span
|
||||||
|
>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<div class="inputTooltip">
|
||||||
|
<span
|
||||||
|
data-bind="text: freeTierExceedThroughputTooltip, visible: showFreeTierExceedThroughputTooltip"
|
||||||
|
class="inputTooltipText"
|
||||||
|
></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div data-bind="setTemplateReady: true">
|
||||||
|
<input
|
||||||
|
data-bind="
|
||||||
|
textInput: overrideWithAutoPilotSettings() ? maxAutoPilotThroughputSet : value,
|
||||||
|
css: {
|
||||||
|
dirty: value.editableIsDirty
|
||||||
|
},
|
||||||
|
enable: isEnabled,
|
||||||
|
attr:{
|
||||||
|
type: isManualThroughputInputFieldRequired() ? 'number' : 'hidden',
|
||||||
|
'data-test': testId,
|
||||||
|
'class': cssClass,
|
||||||
|
step: step,
|
||||||
|
min: minimum,
|
||||||
|
max: canExceedMaximumValue() ? null : maximum,
|
||||||
|
'aria-label': ariaLabel,
|
||||||
|
disabled: overrideWithAutoPilotSettings(),
|
||||||
|
required: isManualThroughputInputFieldRequired()
|
||||||
|
}"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="freeTierInlineWarning" data-bind="visible: showFreeTierExceedThroughputWarning">
|
||||||
|
<span class="freeTierWarningIcon"><img src="/warning.svg" alt="Warning" /></span>
|
||||||
|
<span class="freeTierWarningMessage" data-bind="text: freeTierExceedThroughputWarning"></span>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<p data-bind="visible: costsVisible">
|
||||||
|
<span data-bind="html: requestUnitsUsageCost"></span>
|
||||||
|
</p>
|
||||||
|
|
||||||
|
<!-- ko if: spendAckVisible -->
|
||||||
|
<p class="pkPadding">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
aria-label="acknowledge spend throughput"
|
||||||
|
data-bind="
|
||||||
|
attr: {
|
||||||
|
title: spendAckText,
|
||||||
|
id: spendAckId
|
||||||
|
},
|
||||||
|
checked: spendAckChecked"
|
||||||
|
/>
|
||||||
|
<span data-bind="text: spendAckText, attr: { for: spendAckId }"></span>
|
||||||
|
</p>
|
||||||
|
<!-- /ko -->
|
||||||
|
|
||||||
|
<!-- ko if: isFixed -->
|
||||||
|
<p>Choose unlimited storage capacity for more than 10,000 RU/s.</p>
|
||||||
|
<!-- /ko -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -3,7 +3,7 @@
|
|||||||
exports[`ThroughputInput Pane should render Default properly 1`] = `
|
exports[`ThroughputInput Pane should render Default properly 1`] = `
|
||||||
<ThroughputInput
|
<ThroughputInput
|
||||||
isDatabase={false}
|
isDatabase={false}
|
||||||
isSharded={true}
|
isSharded={false}
|
||||||
onCostAcknowledgeChange={[Function]}
|
onCostAcknowledgeChange={[Function]}
|
||||||
setIsAutoscale={[Function]}
|
setIsAutoscale={[Function]}
|
||||||
setThroughputValue={[Function]}
|
setThroughputValue={[Function]}
|
||||||
|
|||||||
@@ -58,7 +58,7 @@ export interface TreeComponentProps {
|
|||||||
export class TreeComponent extends React.Component<TreeComponentProps> {
|
export class TreeComponent extends React.Component<TreeComponentProps> {
|
||||||
public render(): JSX.Element {
|
public render(): JSX.Element {
|
||||||
return (
|
return (
|
||||||
<div style={this.props.style} className={`treeComponent ${this.props.className}`} role="tree">
|
<div style={this.props.style} className={`treeComponent ${this.props.className}`}>
|
||||||
<TreeNodeComponent paddingLeft={0} node={this.props.rootNode} generation={0} />
|
<TreeNodeComponent paddingLeft={0} node={this.props.rootNode} generation={0} />
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -172,7 +172,6 @@ export class TreeNodeComponent extends React.Component<TreeNodeComponentProps, T
|
|||||||
className={`${this.props.node.className || ""} main${generation} nodeItem ${showSelected ? "selected" : ""}`}
|
className={`${this.props.node.className || ""} main${generation} nodeItem ${showSelected ? "selected" : ""}`}
|
||||||
onClick={(event: React.MouseEvent<HTMLDivElement>) => this.onNodeClick(event, node)}
|
onClick={(event: React.MouseEvent<HTMLDivElement>) => this.onNodeClick(event, node)}
|
||||||
onKeyPress={(event: React.KeyboardEvent<HTMLDivElement>) => this.onNodeKeyPress(event, node)}
|
onKeyPress={(event: React.KeyboardEvent<HTMLDivElement>) => this.onNodeKeyPress(event, node)}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className={`treeNodeHeader ${this.state.isMenuShowing ? "showingMenu" : ""}`}
|
className={`treeNodeHeader ${this.state.isMenuShowing ? "showingMenu" : ""}`}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
exports[`TreeComponent renders a simple tree 1`] = `
|
exports[`TreeComponent renders a simple tree 1`] = `
|
||||||
<div
|
<div
|
||||||
className="treeComponent tree"
|
className="treeComponent tree"
|
||||||
role="tree"
|
|
||||||
>
|
>
|
||||||
<TreeNodeComponent
|
<TreeNodeComponent
|
||||||
generation={0}
|
generation={0}
|
||||||
@@ -38,7 +37,6 @@ exports[`TreeNodeComponent does not render children by default 1`] = `
|
|||||||
className=" main2 nodeItem "
|
className=" main2 nodeItem "
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onKeyPress={[Function]}
|
onKeyPress={[Function]}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="treeNodeHeader "
|
className="treeNodeHeader "
|
||||||
@@ -139,7 +137,6 @@ exports[`TreeNodeComponent renders a simple node (sorted children, expanded) 1`]
|
|||||||
className="nodeClassname main12 nodeItem "
|
className="nodeClassname main12 nodeItem "
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onKeyPress={[Function]}
|
onKeyPress={[Function]}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="treeNodeHeader "
|
className="treeNodeHeader "
|
||||||
@@ -288,7 +285,6 @@ exports[`TreeNodeComponent renders loading icon 1`] = `
|
|||||||
className=" main2 nodeItem "
|
className=" main2 nodeItem "
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onKeyPress={[Function]}
|
onKeyPress={[Function]}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="treeNodeHeader "
|
className="treeNodeHeader "
|
||||||
@@ -360,7 +356,6 @@ exports[`TreeNodeComponent renders sorted children, expanded, leaves and parents
|
|||||||
className="nodeClassname main12 nodeItem "
|
className="nodeClassname main12 nodeItem "
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onKeyPress={[Function]}
|
onKeyPress={[Function]}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="treeNodeHeader "
|
className="treeNodeHeader "
|
||||||
@@ -528,7 +523,6 @@ exports[`TreeNodeComponent renders unsorted children by default 1`] = `
|
|||||||
className=" main2 nodeItem "
|
className=" main2 nodeItem "
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
onKeyPress={[Function]}
|
onKeyPress={[Function]}
|
||||||
role="treeitem"
|
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="treeNodeHeader "
|
className="treeNodeHeader "
|
||||||
|
|||||||
@@ -2,22 +2,23 @@ jest.mock("../Graph/GraphExplorerComponent/GremlinClient");
|
|||||||
jest.mock("../../Common/dataAccess/createCollection");
|
jest.mock("../../Common/dataAccess/createCollection");
|
||||||
jest.mock("../../Common/dataAccess/createDocument");
|
jest.mock("../../Common/dataAccess/createDocument");
|
||||||
import * as ko from "knockout";
|
import * as ko from "knockout";
|
||||||
|
import Q from "q";
|
||||||
import { createDocument } from "../../Common/dataAccess/createDocument";
|
import { createDocument } from "../../Common/dataAccess/createDocument";
|
||||||
import { DatabaseAccount } from "../../Contracts/DataModels";
|
import { DatabaseAccount } from "../../Contracts/DataModels";
|
||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
import { updateUserContext } from "../../UserContext";
|
import { updateUserContext } from "../../UserContext";
|
||||||
import Explorer from "../Explorer";
|
import Explorer from "../Explorer";
|
||||||
import { useDatabases } from "../useDatabases";
|
|
||||||
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
||||||
|
|
||||||
describe("ContainerSampleGenerator", () => {
|
describe("ContainerSampleGenerator", () => {
|
||||||
let explorerStub: Explorer;
|
const createExplorerStub = (database: ViewModels.Database): Explorer => {
|
||||||
|
const explorerStub = {} as Explorer;
|
||||||
beforeAll(() => {
|
explorerStub.databases = ko.observableArray<ViewModels.Database>([database]);
|
||||||
explorerStub = {
|
explorerStub.canExceedMaximumValue = ko.computed<boolean>(() => false);
|
||||||
refreshAllDatabases: () => {},
|
explorerStub.findDatabaseWithId = () => database;
|
||||||
} as Explorer;
|
explorerStub.refreshAllDatabases = () => Q.resolve();
|
||||||
});
|
return explorerStub;
|
||||||
|
};
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
(createDocument as jest.Mock).mockResolvedValue(undefined);
|
(createDocument as jest.Mock).mockResolvedValue(undefined);
|
||||||
@@ -59,7 +60,8 @@ describe("ContainerSampleGenerator", () => {
|
|||||||
loadCollections: () => {},
|
loadCollections: () => {},
|
||||||
} as ViewModels.Database;
|
} as ViewModels.Database;
|
||||||
database.findCollectionWithId = () => collection;
|
database.findCollectionWithId = () => collection;
|
||||||
useDatabases.getState().addDatabases([database]);
|
|
||||||
|
const explorerStub = createExplorerStub(database);
|
||||||
|
|
||||||
const generator = await ContainerSampleGenerator.createSampleGeneratorAsync(explorerStub);
|
const generator = await ContainerSampleGenerator.createSampleGeneratorAsync(explorerStub);
|
||||||
generator.setData(sampleData);
|
generator.setData(sampleData);
|
||||||
@@ -107,8 +109,8 @@ describe("ContainerSampleGenerator", () => {
|
|||||||
} as ViewModels.Database;
|
} as ViewModels.Database;
|
||||||
database.findCollectionWithId = () => collection;
|
database.findCollectionWithId = () => collection;
|
||||||
collection.databaseId = database.id();
|
collection.databaseId = database.id();
|
||||||
useDatabases.getState().addDatabases([database]);
|
|
||||||
|
|
||||||
|
const explorerStub = createExplorerStub(database);
|
||||||
updateUserContext({
|
updateUserContext({
|
||||||
databaseAccount: {
|
databaseAccount: {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -125,6 +127,7 @@ describe("ContainerSampleGenerator", () => {
|
|||||||
|
|
||||||
it("should not create any sample for Mongo API account", async () => {
|
it("should not create any sample for Mongo API account", async () => {
|
||||||
const experience = "Sample generation not supported for this API Mongo";
|
const experience = "Sample generation not supported for this API Mongo";
|
||||||
|
const explorerStub = createExplorerStub(undefined);
|
||||||
updateUserContext({
|
updateUserContext({
|
||||||
databaseAccount: {
|
databaseAccount: {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -139,6 +142,7 @@ describe("ContainerSampleGenerator", () => {
|
|||||||
|
|
||||||
it("should not create any sample for Table API account", async () => {
|
it("should not create any sample for Table API account", async () => {
|
||||||
const experience = "Sample generation not supported for this API Tables";
|
const experience = "Sample generation not supported for this API Tables";
|
||||||
|
const explorerStub = createExplorerStub(undefined);
|
||||||
updateUserContext({
|
updateUserContext({
|
||||||
databaseAccount: {
|
databaseAccount: {
|
||||||
properties: {
|
properties: {
|
||||||
@@ -160,6 +164,7 @@ describe("ContainerSampleGenerator", () => {
|
|||||||
},
|
},
|
||||||
} as DatabaseAccount,
|
} as DatabaseAccount,
|
||||||
});
|
});
|
||||||
|
const explorerStub = createExplorerStub(undefined);
|
||||||
// Rejects with error that contains experience
|
// Rejects with error that contains experience
|
||||||
await expect(ContainerSampleGenerator.createSampleGeneratorAsync(explorerStub)).rejects.toMatch(experience);
|
await expect(ContainerSampleGenerator.createSampleGeneratorAsync(explorerStub)).rejects.toMatch(experience);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -7,7 +7,6 @@ import * as NotificationConsoleUtils from "../../Utils/NotificationConsoleUtils"
|
|||||||
import GraphTab from ".././Tabs/GraphTab";
|
import GraphTab from ".././Tabs/GraphTab";
|
||||||
import Explorer from "../Explorer";
|
import Explorer from "../Explorer";
|
||||||
import { GremlinClient } from "../Graph/GraphExplorerComponent/GremlinClient";
|
import { GremlinClient } from "../Graph/GraphExplorerComponent/GremlinClient";
|
||||||
import { useDatabases } from "../useDatabases";
|
|
||||||
|
|
||||||
interface SampleDataFile extends DataModels.CreateCollectionParams {
|
interface SampleDataFile extends DataModels.CreateCollectionParams {
|
||||||
data: any[];
|
data: any[];
|
||||||
@@ -60,7 +59,7 @@ export class ContainerSampleGenerator {
|
|||||||
|
|
||||||
await createCollection(createRequest);
|
await createCollection(createRequest);
|
||||||
await this.container.refreshAllDatabases();
|
await this.container.refreshAllDatabases();
|
||||||
const database = useDatabases.getState().findDatabaseWithId(this.sampleDataFile.databaseId);
|
const database = this.container.findDatabaseWithId(this.sampleDataFile.databaseId);
|
||||||
if (!database) {
|
if (!database) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import * as ko from "knockout";
|
|||||||
import * as sinon from "sinon";
|
import * as sinon from "sinon";
|
||||||
import { Collection, Database } from "../../Contracts/ViewModels";
|
import { Collection, Database } from "../../Contracts/ViewModels";
|
||||||
import Explorer from "../Explorer";
|
import Explorer from "../Explorer";
|
||||||
import { useDatabases } from "../useDatabases";
|
|
||||||
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
||||||
import { DataSamplesUtil } from "./DataSamplesUtil";
|
import { DataSamplesUtil } from "./DataSamplesUtil";
|
||||||
|
|
||||||
@@ -17,7 +16,8 @@ describe("DataSampleUtils", () => {
|
|||||||
collections: ko.observableArray<Collection>([collection]),
|
collections: ko.observableArray<Collection>([collection]),
|
||||||
} as Database;
|
} as Database;
|
||||||
const explorer = {} as Explorer;
|
const explorer = {} as Explorer;
|
||||||
useDatabases.getState().addDatabases([database]);
|
explorer.databases = ko.observableArray<Database>([database]);
|
||||||
|
explorer.showOkModalDialog = () => {};
|
||||||
const dataSamplesUtil = new DataSamplesUtil(explorer);
|
const dataSamplesUtil = new DataSamplesUtil(explorer);
|
||||||
|
|
||||||
const fakeGenerator = sinon.createStubInstance<ContainerSampleGenerator>(ContainerSampleGenerator as any);
|
const fakeGenerator = sinon.createStubInstance<ContainerSampleGenerator>(ContainerSampleGenerator as any);
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
import * as ViewModels from "../../Contracts/ViewModels";
|
import * as ViewModels from "../../Contracts/ViewModels";
|
||||||
import { userContext } from "../../UserContext";
|
import { userContext } from "../../UserContext";
|
||||||
import { logConsoleError, logConsoleInfo } from "../../Utils/NotificationConsoleUtils";
|
import { logConsoleError, logConsoleInfo } from "../../Utils/NotificationConsoleUtils";
|
||||||
import { useDialog } from "../Controls/Dialog";
|
|
||||||
import Explorer from "../Explorer";
|
import Explorer from "../Explorer";
|
||||||
import { useDatabases } from "../useDatabases";
|
|
||||||
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
import { ContainerSampleGenerator } from "./ContainerSampleGenerator";
|
||||||
|
|
||||||
export class DataSamplesUtil {
|
export class DataSamplesUtil {
|
||||||
@@ -19,9 +17,9 @@ export class DataSamplesUtil {
|
|||||||
|
|
||||||
const databaseName = generator.getDatabaseId();
|
const databaseName = generator.getDatabaseId();
|
||||||
const containerName = generator.getCollectionId();
|
const containerName = generator.getCollectionId();
|
||||||
if (this.hasContainer(databaseName, containerName, useDatabases.getState().databases)) {
|
if (this.hasContainer(databaseName, containerName, this.container.databases())) {
|
||||||
const msg = `The container ${containerName} in database ${databaseName} already exists. Please delete it and retry.`;
|
const msg = `The container ${containerName} in database ${databaseName} already exists. Please delete it and retry.`;
|
||||||
useDialog.getState().showOkModalDialog(DataSamplesUtil.DialogTitle, msg);
|
this.container.showOkModalDialog(DataSamplesUtil.DialogTitle, msg);
|
||||||
logConsoleError(msg);
|
logConsoleError(msg);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -30,7 +28,7 @@ export class DataSamplesUtil {
|
|||||||
.createSampleContainerAsync()
|
.createSampleContainerAsync()
|
||||||
.catch((error) => logConsoleError(`Error creating sample container: ${error}`));
|
.catch((error) => logConsoleError(`Error creating sample container: ${error}`));
|
||||||
const msg = `The sample ${containerName} in database ${databaseName} has been successfully created.`;
|
const msg = `The sample ${containerName} in database ${databaseName} has been successfully created.`;
|
||||||
useDialog.getState().showOkModalDialog(DataSamplesUtil.DialogTitle, msg);
|
this.container.showOkModalDialog(DataSamplesUtil.DialogTitle, msg);
|
||||||
logConsoleInfo(msg);
|
logConsoleInfo(msg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
43
src/Explorer/Explorer.test.tsx
Normal file
43
src/Explorer/Explorer.test.tsx
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
jest.mock("./../Common/dataAccess/deleteDatabase");
|
||||||
|
jest.mock("./../Shared/Telemetry/TelemetryProcessor");
|
||||||
|
import * as ko from "knockout";
|
||||||
|
import { deleteDatabase } from "./../Common/dataAccess/deleteDatabase";
|
||||||
|
import * as ViewModels from "./../Contracts/ViewModels";
|
||||||
|
import Explorer from "./Explorer";
|
||||||
|
|
||||||
|
describe("Explorer.isLastDatabase() and Explorer.isLastNonEmptyDatabase()", () => {
|
||||||
|
let explorer: Explorer;
|
||||||
|
beforeAll(() => {
|
||||||
|
(deleteDatabase as jest.Mock).mockResolvedValue(undefined);
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
explorer = new Explorer();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if only 1 database", () => {
|
||||||
|
const database = {} as ViewModels.Database;
|
||||||
|
explorer.databases = ko.observableArray<ViewModels.Database>([database]);
|
||||||
|
expect(explorer.isLastDatabase()).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be false if only 2 databases", () => {
|
||||||
|
const database = {} as ViewModels.Database;
|
||||||
|
const database2 = {} as ViewModels.Database;
|
||||||
|
explorer.databases = ko.observableArray<ViewModels.Database>([database, database2]);
|
||||||
|
expect(explorer.isLastDatabase()).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be false if not last empty database", () => {
|
||||||
|
const database = {} as ViewModels.Database;
|
||||||
|
explorer.databases = ko.observableArray<ViewModels.Database>([database]);
|
||||||
|
expect(explorer.isLastNonEmptyDatabase()).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be true if last non empty database", () => {
|
||||||
|
const database = {} as ViewModels.Database;
|
||||||
|
database.collections = ko.observableArray<ViewModels.Collection>([{} as ViewModels.Collection]);
|
||||||
|
explorer.databases = ko.observableArray<ViewModels.Database>([database]);
|
||||||
|
expect(explorer.isLastNonEmptyDatabase()).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -4,12 +4,12 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import AddPropertyIcon from "../../../../images/Add-property.svg";
|
import { NeighborVertexBasicInfo, EditedEdges, GraphNewEdgeData, PossibleVertex } from "./GraphExplorer";
|
||||||
import DeleteIcon from "../../../../images/delete.svg";
|
|
||||||
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
|
||||||
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
|
||||||
import { EditedEdges, GraphNewEdgeData, NeighborVertexBasicInfo, PossibleVertex } from "./GraphExplorer";
|
|
||||||
import * as GraphUtil from "./GraphUtil";
|
import * as GraphUtil from "./GraphUtil";
|
||||||
|
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
||||||
|
import DeleteIcon from "../../../../images/delete.svg";
|
||||||
|
import AddPropertyIcon from "../../../../images/Add-property.svg";
|
||||||
|
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
||||||
|
|
||||||
export interface EditorNeighborsComponentProps {
|
export interface EditorNeighborsComponentProps {
|
||||||
isSource: boolean;
|
isSource: boolean;
|
||||||
@@ -83,11 +83,11 @@ export class EditorNeighborsComponent extends React.Component<EditorNeighborsCom
|
|||||||
}
|
}
|
||||||
|
|
||||||
private removeCurrentNeighborEdge(index: number): void {
|
private removeCurrentNeighborEdge(index: number): void {
|
||||||
const sources = this.props.editedNeighbors.currentNeighbors;
|
let sources = this.props.editedNeighbors.currentNeighbors;
|
||||||
const id = sources[index].edgeId;
|
let id = sources[index].edgeId;
|
||||||
sources.splice(index, 1);
|
sources.splice(index, 1);
|
||||||
|
|
||||||
const droppedIds = this.props.editedNeighbors.droppedIds;
|
let droppedIds = this.props.editedNeighbors.droppedIds;
|
||||||
droppedIds.push(id);
|
droppedIds.push(id);
|
||||||
this.onUpdateEdges();
|
this.onUpdateEdges();
|
||||||
}
|
}
|
||||||
@@ -215,7 +215,7 @@ export class EditorNeighborsComponent extends React.Component<EditorNeighborsCom
|
|||||||
</td>
|
</td>
|
||||||
<td className="actionCol">
|
<td className="actionCol">
|
||||||
<span className="rightPaneTrashIcon rightPaneBtns">
|
<span className="rightPaneTrashIcon rightPaneBtns">
|
||||||
<img src={DeleteIcon} alt="Delete" onClick={() => this.removeAddedEdgeToNeighbor(index)} />
|
<img src={DeleteIcon} alt="Delete" onClick={(e) => this.removeAddedEdgeToNeighbor(index)} />
|
||||||
</span>
|
</span>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
import { shallow } from "enzyme";
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
import { shallow } from "enzyme";
|
||||||
|
import { GraphHighlightedNodeData, EditedProperties } from "./GraphExplorer";
|
||||||
|
|
||||||
import { EditorNodePropertiesComponent, EditorNodePropertiesComponentProps } from "./EditorNodePropertiesComponent";
|
import { EditorNodePropertiesComponent, EditorNodePropertiesComponentProps } from "./EditorNodePropertiesComponent";
|
||||||
|
|
||||||
describe("<EditorNodePropertiesComponent />", () => {
|
describe("<EditorNodePropertiesComponent />", () => {
|
||||||
// Tests that: single value prop is rendered with a textbox and a delete button
|
// Tests that: single value prop is rendered with a textbox and a delete button
|
||||||
// multi-value prop only a delete button (cannot be edited)
|
// multi-value prop only a delete button (cannot be edited)
|
||||||
const onUpdateProperties = jest.fn();
|
|
||||||
it("renders component", () => {
|
it("renders component", () => {
|
||||||
const props: EditorNodePropertiesComponentProps = {
|
const props: EditorNodePropertiesComponentProps = {
|
||||||
editedProperties: {
|
editedProperties: {
|
||||||
@@ -23,6 +24,7 @@ describe("<EditorNodePropertiesComponent />", () => {
|
|||||||
{ value: true, type: "boolean" },
|
{ value: true, type: "boolean" },
|
||||||
{ value: false, type: "boolean" },
|
{ value: false, type: "boolean" },
|
||||||
{ value: undefined, type: "null" },
|
{ value: undefined, type: "null" },
|
||||||
|
{ value: null, type: "null" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@@ -39,13 +41,14 @@ describe("<EditorNodePropertiesComponent />", () => {
|
|||||||
{ value: true, type: "boolean" },
|
{ value: true, type: "boolean" },
|
||||||
{ value: false, type: "boolean" },
|
{ value: false, type: "boolean" },
|
||||||
{ value: undefined, type: "null" },
|
{ value: undefined, type: "null" },
|
||||||
|
{ value: null, type: "null" },
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
addedProperties: [],
|
addedProperties: [],
|
||||||
droppedKeys: [],
|
droppedKeys: [],
|
||||||
},
|
},
|
||||||
onUpdateProperties,
|
onUpdateProperties: (editedProperties: EditedProperties): void => {},
|
||||||
};
|
};
|
||||||
const wrapper = shallow(<EditorNodePropertiesComponent {...props} />);
|
const wrapper = shallow(<EditorNodePropertiesComponent {...props} />);
|
||||||
expect(wrapper).toMatchSnapshot();
|
expect(wrapper).toMatchSnapshot();
|
||||||
@@ -78,7 +81,7 @@ describe("<EditorNodePropertiesComponent />", () => {
|
|||||||
addedProperties: [],
|
addedProperties: [],
|
||||||
droppedKeys: [],
|
droppedKeys: [],
|
||||||
},
|
},
|
||||||
onUpdateProperties,
|
onUpdateProperties: (editedProperties: EditedProperties): void => {},
|
||||||
};
|
};
|
||||||
const wrapper = shallow(<EditorNodePropertiesComponent {...props} />);
|
const wrapper = shallow(<EditorNodePropertiesComponent {...props} />);
|
||||||
expect(wrapper).toMatchSnapshot();
|
expect(wrapper).toMatchSnapshot();
|
||||||
|
|||||||
@@ -4,12 +4,12 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import AddIcon from "../../../../images/Add-property.svg";
|
|
||||||
import DeleteIcon from "../../../../images/delete.svg";
|
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
|
||||||
import { EditedProperties } from "./GraphExplorer";
|
import { EditedProperties } from "./GraphExplorer";
|
||||||
|
import DeleteIcon from "../../../../images/delete.svg";
|
||||||
|
import AddIcon from "../../../../images/Add-property.svg";
|
||||||
import { ReadOnlyNodePropertiesComponent } from "./ReadOnlyNodePropertiesComponent";
|
import { ReadOnlyNodePropertiesComponent } from "./ReadOnlyNodePropertiesComponent";
|
||||||
|
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
||||||
|
|
||||||
export interface EditorNodePropertiesComponentProps {
|
export interface EditorNodePropertiesComponentProps {
|
||||||
editedProperties: EditedProperties;
|
editedProperties: EditedProperties;
|
||||||
@@ -48,7 +48,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
const editedProperties = this.props.editedProperties;
|
const editedProperties = this.props.editedProperties;
|
||||||
// search for it
|
// search for it
|
||||||
for (let i = 0; i < editedProperties.existingProperties.length; i++) {
|
for (let i = 0; i < editedProperties.existingProperties.length; i++) {
|
||||||
const ip = editedProperties.existingProperties[i];
|
let ip = editedProperties.existingProperties[i];
|
||||||
if (ip.key === key) {
|
if (ip.key === key) {
|
||||||
editedProperties.existingProperties.splice(i, 1);
|
editedProperties.existingProperties.splice(i, 1);
|
||||||
editedProperties.droppedKeys.push(key);
|
editedProperties.droppedKeys.push(key);
|
||||||
@@ -60,7 +60,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
|
|
||||||
private removeAddedProperty(index: number): void {
|
private removeAddedProperty(index: number): void {
|
||||||
const editedProperties = this.props.editedProperties;
|
const editedProperties = this.props.editedProperties;
|
||||||
const ap = editedProperties.addedProperties;
|
let ap = editedProperties.addedProperties;
|
||||||
ap.splice(index, 1);
|
ap.splice(index, 1);
|
||||||
|
|
||||||
this.props.onUpdateProperties(editedProperties);
|
this.props.onUpdateProperties(editedProperties);
|
||||||
@@ -68,7 +68,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
|
|
||||||
private addProperty(): void {
|
private addProperty(): void {
|
||||||
const editedProperties = this.props.editedProperties;
|
const editedProperties = this.props.editedProperties;
|
||||||
const ap = editedProperties.addedProperties;
|
let ap = editedProperties.addedProperties;
|
||||||
ap.push({ key: "", values: [{ value: "", type: EditorNodePropertiesComponent.DEFAULT_PROPERTY_TYPE }] });
|
ap.push({ key: "", values: [{ value: "", type: EditorNodePropertiesComponent.DEFAULT_PROPERTY_TYPE }] });
|
||||||
this.props.onUpdateProperties(editedProperties);
|
this.props.onUpdateProperties(editedProperties);
|
||||||
}
|
}
|
||||||
@@ -126,7 +126,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
onChange={(e) => {
|
onChange={(e) => {
|
||||||
singleValue.type = e.target.value as ViewModels.InputPropertyValueTypeString;
|
singleValue.type = e.target.value as ViewModels.InputPropertyValueTypeString;
|
||||||
if (singleValue.type === "null") {
|
if (singleValue.type === "null") {
|
||||||
singleValue.value = undefined;
|
singleValue.value = null;
|
||||||
}
|
}
|
||||||
this.props.onUpdateProperties(this.props.editedProperties);
|
this.props.onUpdateProperties(this.props.editedProperties);
|
||||||
}}
|
}}
|
||||||
@@ -144,7 +144,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
className="rightPaneTrashIcon rightPaneBtns"
|
className="rightPaneTrashIcon rightPaneBtns"
|
||||||
as="span"
|
as="span"
|
||||||
aria-label="Delete property"
|
aria-label="Delete property"
|
||||||
onActivated={() => this.removeExistingProperty(key)}
|
onActivated={(e) => this.removeExistingProperty(key)}
|
||||||
>
|
>
|
||||||
<img src={DeleteIcon} alt="Delete" />
|
<img src={DeleteIcon} alt="Delete" />
|
||||||
</AccessibleElement>
|
</AccessibleElement>
|
||||||
@@ -166,7 +166,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
className="rightPaneTrashIcon rightPaneBtns"
|
className="rightPaneTrashIcon rightPaneBtns"
|
||||||
as="span"
|
as="span"
|
||||||
aria-label="Remove existing property"
|
aria-label="Remove existing property"
|
||||||
onActivated={() => this.removeExistingProperty(nodeProp.key)}
|
onActivated={(e) => this.removeExistingProperty(nodeProp.key)}
|
||||||
>
|
>
|
||||||
<img src={DeleteIcon} alt="Delete" />
|
<img src={DeleteIcon} alt="Delete" />
|
||||||
</AccessibleElement>
|
</AccessibleElement>
|
||||||
@@ -206,7 +206,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
onChange={(e) => {
|
onChange={(e) => {
|
||||||
firstValue.value = e.target.value;
|
firstValue.value = e.target.value;
|
||||||
if (firstValue.type === "null") {
|
if (firstValue.type === "null") {
|
||||||
firstValue.value = undefined;
|
firstValue.value = null;
|
||||||
}
|
}
|
||||||
this.props.onUpdateProperties(this.props.editedProperties);
|
this.props.onUpdateProperties(this.props.editedProperties);
|
||||||
}}
|
}}
|
||||||
@@ -235,7 +235,7 @@ export class EditorNodePropertiesComponent extends React.Component<EditorNodePro
|
|||||||
className="rightPaneTrashIcon rightPaneBtns"
|
className="rightPaneTrashIcon rightPaneBtns"
|
||||||
as="span"
|
as="span"
|
||||||
aria-label="Remove property"
|
aria-label="Remove property"
|
||||||
onActivated={() => this.removeAddedProperty(index)}
|
onActivated={(e) => this.removeAddedProperty(index)}
|
||||||
>
|
>
|
||||||
<img src={DeleteIcon} alt="Delete" />
|
<img src={DeleteIcon} alt="Delete" />
|
||||||
</AccessibleElement>
|
</AccessibleElement>
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
import { GraphData, GremlinEdge, GremlinVertex } from "./GraphData";
|
import { GraphData, GremlinVertex, GremlinEdge } from "./GraphData";
|
||||||
|
|
||||||
describe("Graph Data", () => {
|
describe("Graph Data", () => {
|
||||||
it("should set only one node as root", () => {
|
it("should set only one node as root", () => {
|
||||||
const graphData = new GraphData<GremlinVertex, GremlinEdge>();
|
const graphData = new GraphData<GremlinVertex, GremlinEdge>();
|
||||||
const v1: GremlinVertex = { id: "1", label: undefined };
|
const v1: GremlinVertex = { id: "1", label: null };
|
||||||
const v2: GremlinVertex = { id: "2", label: undefined };
|
const v2: GremlinVertex = { id: "2", label: null };
|
||||||
const v3: GremlinVertex = { id: "3", label: undefined };
|
const v3: GremlinVertex = { id: "3", label: null };
|
||||||
v3._isRoot = true;
|
v3._isRoot = true;
|
||||||
|
|
||||||
graphData.addVertex(v1);
|
graphData.addVertex(v1);
|
||||||
@@ -28,9 +28,9 @@ describe("Graph Data", () => {
|
|||||||
|
|
||||||
it("should properly find root id", () => {
|
it("should properly find root id", () => {
|
||||||
const graphData = new GraphData();
|
const graphData = new GraphData();
|
||||||
const v1: GremlinVertex = { id: "1", label: undefined };
|
const v1: GremlinVertex = { id: "1", label: null };
|
||||||
const v2: GremlinVertex = { id: "2", label: undefined };
|
const v2: GremlinVertex = { id: "2", label: null };
|
||||||
const v3: GremlinVertex = { id: "3", label: undefined };
|
const v3: GremlinVertex = { id: "3", label: null };
|
||||||
|
|
||||||
graphData.addVertex(v1);
|
graphData.addVertex(v1);
|
||||||
graphData.addVertex(v2);
|
graphData.addVertex(v2);
|
||||||
@@ -44,12 +44,12 @@ describe("Graph Data", () => {
|
|||||||
it("should remove edge from graph", () => {
|
it("should remove edge from graph", () => {
|
||||||
const graphData = new GraphData();
|
const graphData = new GraphData();
|
||||||
|
|
||||||
graphData.addVertex({ id: "v1", label: undefined });
|
graphData.addVertex({ id: "v1", label: null });
|
||||||
graphData.addVertex({ id: "v2", label: undefined });
|
graphData.addVertex({ id: "v2", label: null });
|
||||||
graphData.addVertex({ id: "v3", label: undefined });
|
graphData.addVertex({ id: "v3", label: null });
|
||||||
|
|
||||||
graphData.addEdge({ id: "e1", inV: "v1", outV: "v2", label: "" });
|
graphData.addEdge({ id: "e1", inV: "v1", outV: "v2", label: null });
|
||||||
graphData.addEdge({ id: "e2", inV: "v1", outV: "v3", label: "" });
|
graphData.addEdge({ id: "e2", inV: "v1", outV: "v3", label: null });
|
||||||
|
|
||||||
// in edge
|
// in edge
|
||||||
graphData.removeEdge("e1", false);
|
graphData.removeEdge("e1", false);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { SimulationLinkDatum, SimulationNodeDatum } from "d3";
|
|
||||||
import _ from "underscore";
|
import _ from "underscore";
|
||||||
|
import { SimulationNodeDatum, SimulationLinkDatum } from "d3";
|
||||||
|
|
||||||
export interface PaginationInfo {
|
export interface PaginationInfo {
|
||||||
total: number;
|
total: number;
|
||||||
@@ -10,7 +10,7 @@ export interface PaginationInfo {
|
|||||||
|
|
||||||
export interface GremlinVertex {
|
export interface GremlinVertex {
|
||||||
id: string;
|
id: string;
|
||||||
label?: string;
|
label: string;
|
||||||
inE?: { [label: string]: GremlinShortInEdge[] };
|
inE?: { [label: string]: GremlinShortInEdge[] };
|
||||||
outE?: { [label: string]: GremlinShortOutEdge[] };
|
outE?: { [label: string]: GremlinShortOutEdge[] };
|
||||||
properties?: { [propName: string]: GremlinProperty[] };
|
properties?: { [propName: string]: GremlinProperty[] };
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ import { queryDocumentsPage } from "../../../Common/dataAccess/queryDocumentsPag
|
|||||||
import * as DataModels from "../../../Contracts/DataModels";
|
import * as DataModels from "../../../Contracts/DataModels";
|
||||||
import * as StorageUtility from "../../../Shared/StorageUtility";
|
import * as StorageUtility from "../../../Shared/StorageUtility";
|
||||||
import { TabComponent } from "../../Controls/Tabs/TabComponent";
|
import { TabComponent } from "../../Controls/Tabs/TabComponent";
|
||||||
import { ConsoleDataType } from "../../Menus/NotificationConsole/ConsoleData";
|
import { ConsoleDataType } from "../../Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
import GraphTab from "../../Tabs/GraphTab";
|
import GraphTab from "../../Tabs/GraphTab";
|
||||||
import * as D3ForceGraph from "./D3ForceGraph";
|
import * as D3ForceGraph from "./D3ForceGraph";
|
||||||
import { GraphData } from "./GraphData";
|
import { GraphData } from "./GraphData";
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ import { logConsoleError, logConsoleInfo, logConsoleProgress } from "../../../Ut
|
|||||||
import { EditorReact } from "../../Controls/Editor/EditorReact";
|
import { EditorReact } from "../../Controls/Editor/EditorReact";
|
||||||
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
||||||
import * as TabComponent from "../../Controls/Tabs/TabComponent";
|
import * as TabComponent from "../../Controls/Tabs/TabComponent";
|
||||||
import { ConsoleDataType } from "../../Menus/NotificationConsole/ConsoleData";
|
import { ConsoleDataType } from "../../Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
import { IGraphConfig } from "../../Tabs/GraphTab";
|
import { IGraphConfig } from "../../Tabs/GraphTab";
|
||||||
import { ArraysByKeyCache } from "./ArraysByKeyCache";
|
import { ArraysByKeyCache } from "./ArraysByKeyCache";
|
||||||
import * as D3ForceGraph from "./D3ForceGraph";
|
import * as D3ForceGraph from "./D3ForceGraph";
|
||||||
|
|||||||
@@ -0,0 +1,74 @@
|
|||||||
|
import * as React from "react";
|
||||||
|
import { ReactAdapter } from "../../../Bindings/ReactBindingHandler";
|
||||||
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
|
import { IGraphConfig } from "../../Tabs/GraphTab";
|
||||||
|
import { GraphAccessor, GraphExplorer } from "./GraphExplorer";
|
||||||
|
interface Parameter {
|
||||||
|
onIsNewVertexDisabledChange: (isEnabled: boolean) => void;
|
||||||
|
onGraphAccessorCreated: (instance: GraphAccessor) => void;
|
||||||
|
onIsFilterQueryLoading: (isFilterQueryLoading: boolean) => void;
|
||||||
|
onIsValidQuery: (isValidQuery: boolean) => void;
|
||||||
|
onIsPropertyEditing: (isEditing: boolean) => void;
|
||||||
|
onIsGraphDisplayed: (isDisplayed: boolean) => void;
|
||||||
|
onResetDefaultGraphConfigValues: () => void;
|
||||||
|
|
||||||
|
collectionPartitionKeyProperty: string;
|
||||||
|
graphBackendEndpoint: string;
|
||||||
|
databaseId: string;
|
||||||
|
collectionId: string;
|
||||||
|
masterKey: string;
|
||||||
|
|
||||||
|
onLoadStartKey: number;
|
||||||
|
onLoadStartKeyChange: (newKey: number) => void;
|
||||||
|
resourceId: string;
|
||||||
|
|
||||||
|
igraphConfigUiData: ViewModels.IGraphConfigUiData;
|
||||||
|
igraphConfig: IGraphConfig;
|
||||||
|
setIConfigUiData?: (data: string[]) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IGraphExplorerProps {
|
||||||
|
isChanged: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IGraphExplorerStates {
|
||||||
|
isChangedState: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface GraphExplorerAdapter
|
||||||
|
extends ReactAdapter,
|
||||||
|
React.Component<IGraphExplorerProps, IGraphExplorerStates> {}
|
||||||
|
export class GraphExplorerAdapter implements ReactAdapter {
|
||||||
|
public params: Parameter;
|
||||||
|
public parameters = {};
|
||||||
|
public isNewVertexDisabled: boolean;
|
||||||
|
|
||||||
|
public constructor(params: Parameter, props?: IGraphExplorerProps) {
|
||||||
|
this.params = params;
|
||||||
|
}
|
||||||
|
|
||||||
|
public renderComponent(): JSX.Element {
|
||||||
|
return (
|
||||||
|
<GraphExplorer
|
||||||
|
onIsNewVertexDisabledChange={this.params.onIsNewVertexDisabledChange}
|
||||||
|
onGraphAccessorCreated={this.params.onGraphAccessorCreated}
|
||||||
|
onIsFilterQueryLoadingChange={this.params.onIsFilterQueryLoading}
|
||||||
|
onIsValidQueryChange={this.params.onIsValidQuery}
|
||||||
|
onIsPropertyEditing={this.params.onIsPropertyEditing}
|
||||||
|
onIsGraphDisplayed={this.params.onIsGraphDisplayed}
|
||||||
|
onResetDefaultGraphConfigValues={this.params.onResetDefaultGraphConfigValues}
|
||||||
|
collectionPartitionKeyProperty={this.params.collectionPartitionKeyProperty}
|
||||||
|
graphBackendEndpoint={this.params.graphBackendEndpoint}
|
||||||
|
databaseId={this.params.databaseId}
|
||||||
|
collectionId={this.params.collectionId}
|
||||||
|
masterKey={this.params.masterKey}
|
||||||
|
onLoadStartKey={this.params.onLoadStartKey}
|
||||||
|
onLoadStartKeyChange={this.params.onLoadStartKeyChange}
|
||||||
|
resourceId={this.params.resourceId}
|
||||||
|
igraphConfigUiData={this.params.igraphConfigUiData}
|
||||||
|
igraphConfig={this.params.igraphConfig}
|
||||||
|
setIConfigUiData={this.params.setIConfigUiData}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,7 +1,9 @@
|
|||||||
import * as sinon from "sinon";
|
|
||||||
import { GraphData, GremlinEdge, GremlinVertex } from "./GraphData";
|
|
||||||
import { GraphExplorer } from "./GraphExplorer";
|
|
||||||
import * as GraphUtil from "./GraphUtil";
|
import * as GraphUtil from "./GraphUtil";
|
||||||
|
import { GraphData, GremlinVertex, GremlinEdge } from "./GraphData";
|
||||||
|
import * as sinon from "sinon";
|
||||||
|
import { GraphExplorer } from "./GraphExplorer";
|
||||||
|
window.$ = window.jQuery = require("jquery");
|
||||||
|
|
||||||
const OUT_E_MATCHER = "g\\.V\\(.*\\).outE\\(\\).*\\.as\\('e'\\).inV\\(\\)\\.as\\('v'\\)\\.select\\('e', *'v'\\)";
|
const OUT_E_MATCHER = "g\\.V\\(.*\\).outE\\(\\).*\\.as\\('e'\\).inV\\(\\)\\.as\\('v'\\)\\.select\\('e', *'v'\\)";
|
||||||
const IN_E_MATCHER = "g\\.V\\(.*\\).inE\\(\\).*\\.as\\('e'\\).outV\\(\\)\\.as\\('v'\\)\\.select\\('e', *'v'\\)";
|
const IN_E_MATCHER = "g\\.V\\(.*\\).inE\\(\\).*\\.as\\('e'\\).outV\\(\\)\\.as\\('v'\\)\\.select\\('e', *'v'\\)";
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ export interface GremlinSimpleClientParameters {
|
|||||||
password: string;
|
password: string;
|
||||||
successCallback: (result: Result) => void;
|
successCallback: (result: Result) => void;
|
||||||
progressCallback: (result: Result) => void;
|
progressCallback: (result: Result) => void;
|
||||||
failureCallback: (result: Result, error: string) => void;
|
failureCallback: (result: Result | null, error: string) => void;
|
||||||
infoCallback: (msg: string) => void;
|
infoCallback: (msg: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -62,7 +62,6 @@ export class GremlinSimpleClient {
|
|||||||
private static readonly requestChargeHeader = "x-ms-request-charge";
|
private static readonly requestChargeHeader = "x-ms-request-charge";
|
||||||
|
|
||||||
public params: GremlinSimpleClientParameters;
|
public params: GremlinSimpleClientParameters;
|
||||||
private protocols: string | string[];
|
|
||||||
private ws: WebSocket;
|
private ws: WebSocket;
|
||||||
|
|
||||||
public requestsToSend: { [requestId: string]: GremlinRequestMessage };
|
public requestsToSend: { [requestId: string]: GremlinRequestMessage };
|
||||||
@@ -72,6 +71,7 @@ export class GremlinSimpleClient {
|
|||||||
this.params = params;
|
this.params = params;
|
||||||
this.pendingRequests = {};
|
this.pendingRequests = {};
|
||||||
this.requestsToSend = {};
|
this.requestsToSend = {};
|
||||||
|
this.ws = GremlinSimpleClient.createWebSocket(this.params.endpoint);
|
||||||
}
|
}
|
||||||
|
|
||||||
public connect() {
|
public connect() {
|
||||||
@@ -117,7 +117,7 @@ export class GremlinSimpleClient {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public decodeMessage(msg: MessageEvent): GremlinResponseMessage {
|
public decodeMessage(msg: MessageEvent): GremlinResponseMessage | null {
|
||||||
if (msg.data === null) {
|
if (msg.data === null) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -280,7 +280,7 @@ export class GremlinSimpleClient {
|
|||||||
|
|
||||||
public static utf8ToB64(utf8Str: string) {
|
public static utf8ToB64(utf8Str: string) {
|
||||||
return btoa(
|
return btoa(
|
||||||
encodeURIComponent(utf8Str).replace(/%([0-9A-F]{2})/g, function (match, p1) {
|
encodeURIComponent(utf8Str).replace(/%([0-9A-F]{2})/g, function (_match, p1) {
|
||||||
return String.fromCharCode(parseInt(p1, 16));
|
return String.fromCharCode(parseInt(p1, 16));
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
@@ -305,7 +305,7 @@ export class GremlinSimpleClient {
|
|||||||
return binaryMessage;
|
return binaryMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
private onOpen(event: any) {
|
private onOpen(_event: any) {
|
||||||
this.executeRequestsToSend();
|
this.executeRequestsToSend();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,26 +5,21 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import CancelIcon from "../../../../images/cancel.svg";
|
import { GraphHighlightedNodeData, EditedProperties, EditedEdges, PossibleVertex } from "./GraphExplorer";
|
||||||
import CheckIcon from "../../../../images/check-1.svg";
|
|
||||||
import DeleteIcon from "../../../../images/delete.svg";
|
|
||||||
import EditIcon from "../../../../images/edit-1.svg";
|
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
|
||||||
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
|
||||||
import { CollapsiblePanel } from "../../Controls/CollapsiblePanel/CollapsiblePanel";
|
import { CollapsiblePanel } from "../../Controls/CollapsiblePanel/CollapsiblePanel";
|
||||||
import { Item } from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
|
||||||
import { ConsoleDataType } from "../../Menus/NotificationConsole/ConsoleData";
|
|
||||||
import * as EditorNeighbors from "./EditorNeighborsComponent";
|
|
||||||
import { EditorNodePropertiesComponent } from "./EditorNodePropertiesComponent";
|
|
||||||
import {
|
|
||||||
EditedEdges,
|
|
||||||
EditedProperties,
|
|
||||||
GraphExplorer,
|
|
||||||
GraphHighlightedNodeData,
|
|
||||||
PossibleVertex,
|
|
||||||
} from "./GraphExplorer";
|
|
||||||
import { ReadOnlyNeighborsComponent } from "./ReadOnlyNeighborsComponent";
|
|
||||||
import { ReadOnlyNodePropertiesComponent } from "./ReadOnlyNodePropertiesComponent";
|
import { ReadOnlyNodePropertiesComponent } from "./ReadOnlyNodePropertiesComponent";
|
||||||
|
import { EditorNodePropertiesComponent } from "./EditorNodePropertiesComponent";
|
||||||
|
import { ReadOnlyNeighborsComponent } from "./ReadOnlyNeighborsComponent";
|
||||||
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
|
import { Item } from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
||||||
|
import * as EditorNeighbors from "./EditorNeighborsComponent";
|
||||||
|
import EditIcon from "../../../../images/edit.svg";
|
||||||
|
import DeleteIcon from "../../../../images/delete.svg";
|
||||||
|
import CheckIcon from "../../../../images/check.svg";
|
||||||
|
import CancelIcon from "../../../../images/cancel.svg";
|
||||||
|
import { GraphExplorer } from "./GraphExplorer";
|
||||||
|
import { ConsoleDataType } from "../../Menus/NotificationConsole/NotificationConsoleComponent";
|
||||||
|
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
||||||
|
|
||||||
export enum Mode {
|
export enum Mode {
|
||||||
READONLY_PROP,
|
READONLY_PROP,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import CloseIcon from "../../../../images/close-black.svg";
|
|
||||||
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
import * as InputTypeaheadComponent from "../../Controls/InputTypeahead/InputTypeaheadComponent";
|
||||||
|
import CloseIcon from "../../../../images/close-black.svg";
|
||||||
|
|
||||||
export interface QueryContainerComponentProps {
|
export interface QueryContainerComponentProps {
|
||||||
initialQuery: string;
|
initialQuery: string;
|
||||||
@@ -82,7 +82,7 @@ export class QueryContainerComponent extends React.Component<
|
|||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="filterbtnstyle queryButton"
|
className="filterbtnstyle queryButton"
|
||||||
onClick={() => this.props.onExecuteClick(this.state.query)}
|
onClick={(e) => this.props.onExecuteClick(this.state.query)}
|
||||||
disabled={this.props.isLoading || !QueryContainerComponent.isQueryValid(this.state.query)}
|
disabled={this.props.isLoading || !QueryContainerComponent.isQueryValid(this.state.query)}
|
||||||
>
|
>
|
||||||
Execute Gremlin Query
|
Execute Gremlin Query
|
||||||
|
|||||||
@@ -4,9 +4,9 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
|
||||||
import { GraphHighlightedNodeData, NeighborVertexBasicInfo } from "./GraphExplorer";
|
import { GraphHighlightedNodeData, NeighborVertexBasicInfo } from "./GraphExplorer";
|
||||||
import * as GraphUtil from "./GraphUtil";
|
import * as GraphUtil from "./GraphUtil";
|
||||||
|
import { AccessibleElement } from "../../Controls/AccessibleElement/AccessibleElement";
|
||||||
|
|
||||||
export interface ReadOnlyNeighborsComponentProps {
|
export interface ReadOnlyNeighborsComponentProps {
|
||||||
node: GraphHighlightedNodeData;
|
node: GraphHighlightedNodeData;
|
||||||
@@ -48,7 +48,7 @@ export class ReadOnlyNeighborsComponent extends React.Component<ReadOnlyNeighbor
|
|||||||
className="clickableLink"
|
className="clickableLink"
|
||||||
as="a"
|
as="a"
|
||||||
aria-label={_neighbor.name}
|
aria-label={_neighbor.name}
|
||||||
onActivated={() => this.props.selectNode(_neighbor.id)}
|
onActivated={(e) => this.props.selectNode(_neighbor.id)}
|
||||||
title={GraphUtil.getNeighborTitle(_neighbor)}
|
title={GraphUtil.getNeighborTitle(_neighbor)}
|
||||||
>
|
>
|
||||||
{_neighbor.name}
|
{_neighbor.name}
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
|
||||||
import { GraphHighlightedNodeData } from "./GraphExplorer";
|
import { GraphHighlightedNodeData } from "./GraphExplorer";
|
||||||
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
|
|
||||||
export interface ReadOnlyNodePropertiesComponentProps {
|
export interface ReadOnlyNodePropertiesComponentProps {
|
||||||
node: GraphHighlightedNodeData;
|
node: GraphHighlightedNodeData;
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ exports[`<EditorNodePropertiesComponent /> renders component 1`] = `
|
|||||||
</td>
|
</td>
|
||||||
<td
|
<td
|
||||||
className="valueCol"
|
className="valueCol"
|
||||||
title="efgh, 1234, true, false, undefined"
|
title="efgh, 1234, true, false, undefined, null"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="propertyValue"
|
className="propertyValue"
|
||||||
@@ -69,6 +69,12 @@ exports[`<EditorNodePropertiesComponent /> renders component 1`] = `
|
|||||||
>
|
>
|
||||||
undefined
|
undefined
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
className="propertyValue isNull"
|
||||||
|
key="null"
|
||||||
|
>
|
||||||
|
null
|
||||||
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr
|
<tr
|
||||||
@@ -172,6 +178,12 @@ exports[`<EditorNodePropertiesComponent /> renders component 1`] = `
|
|||||||
>
|
>
|
||||||
undefined
|
undefined
|
||||||
</div>
|
</div>
|
||||||
|
<div
|
||||||
|
className="propertyValue isNull"
|
||||||
|
key="null"
|
||||||
|
>
|
||||||
|
null
|
||||||
|
</div>
|
||||||
</td>
|
</td>
|
||||||
<td />
|
<td />
|
||||||
<td
|
<td
|
||||||
|
|||||||
@@ -3,72 +3,103 @@
|
|||||||
* If the component signals a change through the callback passed in the properties, it must render the React component when appropriate
|
* If the component signals a change through the callback passed in the properties, it must render the React component when appropriate
|
||||||
* and update any knockout observables passed from the parent.
|
* and update any knockout observables passed from the parent.
|
||||||
*/
|
*/
|
||||||
import { CommandBar as FluentCommandBar, ICommandBarItemProps } from "@fluentui/react";
|
import { CommandBar, ICommandBarItemProps } from "@fluentui/react";
|
||||||
|
import * as ko from "knockout";
|
||||||
import * as React from "react";
|
import * as React from "react";
|
||||||
import create, { UseStore } from "zustand";
|
import { ReactAdapter } from "../../../Bindings/ReactBindingHandler";
|
||||||
import { StyleConstants } from "../../../Common/Constants";
|
import { StyleConstants } from "../../../Common/Constants";
|
||||||
import * as ViewModels from "../../../Contracts/ViewModels";
|
import * as ViewModels from "../../../Contracts/ViewModels";
|
||||||
import { useTabs } from "../../../hooks/useTabs";
|
import { userContext } from "../../../UserContext";
|
||||||
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
import { CommandButtonComponentProps } from "../../Controls/CommandButton/CommandButtonComponent";
|
||||||
import Explorer from "../../Explorer";
|
import Explorer from "../../Explorer";
|
||||||
import { useSelectedNode } from "../../useSelectedNode";
|
|
||||||
import * as CommandBarComponentButtonFactory from "./CommandBarComponentButtonFactory";
|
import * as CommandBarComponentButtonFactory from "./CommandBarComponentButtonFactory";
|
||||||
import * as CommandBarUtil from "./CommandBarUtil";
|
import * as CommandBarUtil from "./CommandBarUtil";
|
||||||
|
|
||||||
interface Props {
|
export class CommandBarComponentAdapter implements ReactAdapter {
|
||||||
container: Explorer;
|
public parameters: ko.Observable<number>;
|
||||||
|
public container: Explorer;
|
||||||
|
private tabsButtons: CommandButtonComponentProps[];
|
||||||
|
private isNotebookTabActive: ko.Computed<boolean>;
|
||||||
|
|
||||||
|
constructor(container: Explorer) {
|
||||||
|
this.container = container;
|
||||||
|
this.tabsButtons = [];
|
||||||
|
this.isNotebookTabActive = ko.computed(
|
||||||
|
() => container.tabsManager.activeTab()?.tabKind === ViewModels.CollectionTabKind.NotebookV2
|
||||||
|
);
|
||||||
|
|
||||||
|
// These are the parameters watched by the react binding that will trigger a renderComponent() if one of the ko mutates
|
||||||
|
const toWatch = [
|
||||||
|
container.deleteCollectionText,
|
||||||
|
container.deleteDatabaseText,
|
||||||
|
container.addCollectionText,
|
||||||
|
container.isDatabaseNodeOrNoneSelected,
|
||||||
|
container.isDatabaseNodeSelected,
|
||||||
|
container.isNoneSelected,
|
||||||
|
container.isResourceTokenCollectionNodeSelected,
|
||||||
|
container.isHostedDataExplorerEnabled,
|
||||||
|
container.isSynapseLinkUpdating,
|
||||||
|
userContext?.databaseAccount,
|
||||||
|
this.isNotebookTabActive,
|
||||||
|
container.isServerlessEnabled,
|
||||||
|
];
|
||||||
|
|
||||||
|
ko.computed(() => ko.toJSON(toWatch)).subscribe(() => this.triggerRender());
|
||||||
|
this.parameters = ko.observable(Date.now());
|
||||||
|
}
|
||||||
|
|
||||||
|
public onUpdateTabsButtons(buttons: CommandButtonComponentProps[]): void {
|
||||||
|
this.tabsButtons = buttons;
|
||||||
|
this.triggerRender();
|
||||||
|
}
|
||||||
|
|
||||||
|
public renderComponent(): JSX.Element {
|
||||||
|
const backgroundColor = StyleConstants.BaseLight;
|
||||||
|
|
||||||
|
const staticButtons = CommandBarComponentButtonFactory.createStaticCommandBarButtons(this.container);
|
||||||
|
const contextButtons = (this.tabsButtons || []).concat(
|
||||||
|
CommandBarComponentButtonFactory.createContextCommandBarButtons(this.container)
|
||||||
|
);
|
||||||
|
const controlButtons = CommandBarComponentButtonFactory.createControlCommandBarButtons(this.container);
|
||||||
|
|
||||||
|
const uiFabricStaticButtons = CommandBarUtil.convertButton(staticButtons, backgroundColor);
|
||||||
|
if (this.tabsButtons && this.tabsButtons.length > 0) {
|
||||||
|
uiFabricStaticButtons.forEach((btn: ICommandBarItemProps) => (btn.iconOnly = true));
|
||||||
|
}
|
||||||
|
|
||||||
|
const uiFabricTabsButtons: ICommandBarItemProps[] = CommandBarUtil.convertButton(contextButtons, backgroundColor);
|
||||||
|
|
||||||
|
if (uiFabricTabsButtons.length > 0) {
|
||||||
|
uiFabricStaticButtons.push(CommandBarUtil.createDivider("commandBarDivider"));
|
||||||
|
}
|
||||||
|
|
||||||
|
const uiFabricControlButtons = CommandBarUtil.convertButton(controlButtons, backgroundColor);
|
||||||
|
uiFabricControlButtons.forEach((btn: ICommandBarItemProps) => (btn.iconOnly = true));
|
||||||
|
|
||||||
|
if (this.isNotebookTabActive()) {
|
||||||
|
uiFabricControlButtons.unshift(
|
||||||
|
CommandBarUtil.createMemoryTracker("memoryTracker", this.container.memoryUsageInfo)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<React.Fragment>
|
||||||
|
<div className="commandBarContainer">
|
||||||
|
<CommandBar
|
||||||
|
ariaLabel="Use left and right arrow keys to navigate between commands"
|
||||||
|
items={uiFabricStaticButtons.concat(uiFabricTabsButtons)}
|
||||||
|
farItems={uiFabricControlButtons}
|
||||||
|
styles={{
|
||||||
|
root: { backgroundColor: backgroundColor },
|
||||||
|
}}
|
||||||
|
overflowButtonProps={{ ariaLabel: "More commands" }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</React.Fragment>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private triggerRender() {
|
||||||
|
window.requestAnimationFrame(() => this.parameters(Date.now()));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface CommandBarStore {
|
|
||||||
contextButtons: CommandButtonComponentProps[];
|
|
||||||
setContextButtons: (contextButtons: CommandButtonComponentProps[]) => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const useCommandBar: UseStore<CommandBarStore> = create((set) => ({
|
|
||||||
contextButtons: [],
|
|
||||||
setContextButtons: (contextButtons: CommandButtonComponentProps[]) => set((state) => ({ ...state, contextButtons })),
|
|
||||||
}));
|
|
||||||
|
|
||||||
export const CommandBar: React.FC<Props> = ({ container }: Props) => {
|
|
||||||
const selectedNodeState = useSelectedNode();
|
|
||||||
const buttons = useCommandBar((state) => state.contextButtons);
|
|
||||||
const backgroundColor = StyleConstants.BaseLight;
|
|
||||||
|
|
||||||
const staticButtons = CommandBarComponentButtonFactory.createStaticCommandBarButtons(container, selectedNodeState);
|
|
||||||
const contextButtons = (buttons || []).concat(
|
|
||||||
CommandBarComponentButtonFactory.createContextCommandBarButtons(container, selectedNodeState)
|
|
||||||
);
|
|
||||||
const controlButtons = CommandBarComponentButtonFactory.createControlCommandBarButtons(container);
|
|
||||||
|
|
||||||
const uiFabricStaticButtons = CommandBarUtil.convertButton(staticButtons, backgroundColor);
|
|
||||||
if (buttons && buttons.length > 0) {
|
|
||||||
uiFabricStaticButtons.forEach((btn: ICommandBarItemProps) => (btn.iconOnly = true));
|
|
||||||
}
|
|
||||||
|
|
||||||
const uiFabricTabsButtons: ICommandBarItemProps[] = CommandBarUtil.convertButton(contextButtons, backgroundColor);
|
|
||||||
|
|
||||||
if (uiFabricTabsButtons.length > 0) {
|
|
||||||
uiFabricStaticButtons.push(CommandBarUtil.createDivider("commandBarDivider"));
|
|
||||||
}
|
|
||||||
|
|
||||||
const uiFabricControlButtons = CommandBarUtil.convertButton(controlButtons, backgroundColor);
|
|
||||||
uiFabricControlButtons.forEach((btn: ICommandBarItemProps) => (btn.iconOnly = true));
|
|
||||||
|
|
||||||
if (useTabs.getState().activeTab?.tabKind === ViewModels.CollectionTabKind.NotebookV2) {
|
|
||||||
uiFabricControlButtons.unshift(CommandBarUtil.createMemoryTracker("memoryTracker"));
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="commandBarContainer">
|
|
||||||
<FluentCommandBar
|
|
||||||
ariaLabel="Use left and right arrow keys to navigate between commands"
|
|
||||||
items={uiFabricStaticButtons.concat(uiFabricTabsButtons)}
|
|
||||||
farItems={uiFabricControlButtons}
|
|
||||||
styles={{
|
|
||||||
root: { backgroundColor: backgroundColor },
|
|
||||||
}}
|
|
||||||
overflowButtonProps={{ ariaLabel: "More commands" }}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user