mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-02-16 18:46:40 -05:00
User Management - Support new object types: Server Role, Application Role and Database Role (#22889)
* server role dialogs * dialogs for other types * refactor * find object dialog * script button * refactoring * fix issues * fix title * vbump sts * remove language from links
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"downloadUrl": "https://github.com/Microsoft/sqltoolsservice/releases/download/{#version#}/microsoft.sqltools.servicelayer-{#fileName#}",
|
"downloadUrl": "https://github.com/Microsoft/sqltoolsservice/releases/download/{#version#}/microsoft.sqltools.servicelayer-{#fileName#}",
|
||||||
"version": "4.7.0.21",
|
"version": "4.7.0.22",
|
||||||
"downloadFileNames": {
|
"downloadFileNames": {
|
||||||
"Windows_86": "win-x86-net7.0.zip",
|
"Windows_86": "win-x86-net7.0.zip",
|
||||||
"Windows_64": "win-x64-net7.0.zip",
|
"Windows_64": "win-x64-net7.0.zip",
|
||||||
|
|||||||
@@ -494,23 +494,23 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.newObject",
|
"command": "mssql.newObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType == Folder && objectType =~ /^(ServerLevelLogins|Users)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType == Folder && objectType =~ /^(ServerLevelLogins|Users|ServerLevelServerRoles|ApplicationRoles|DatabaseRoles)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "0_query@1"
|
"group": "0_query@1"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.objectProperties",
|
"command": "mssql.objectProperties",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "0_query@1",
|
"group": "0_query@1",
|
||||||
"isDefault": true
|
"isDefault": true
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.deleteObject",
|
"command": "mssql.deleteObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "0_query@2"
|
"group": "0_query@2"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.renameObject",
|
"command": "mssql.renameObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(Database|ServerLevelLogin|User|Table|View)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(Database|ServerLevelLogin|User|Table|View|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "0_query@3"
|
"group": "0_query@3"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -540,22 +540,22 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.newObject",
|
"command": "mssql.newObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType == Folder && objectType =~ /^(ServerLevelLogins|Users)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType == Folder && objectType =~ /^(ServerLevelLogins|Users|ServerLevelServerRoles|ApplicationRoles|DatabaseRoles)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "connection@1"
|
"group": "connection@1"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.objectProperties",
|
"command": "mssql.objectProperties",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "connection@1"
|
"group": "connection@1"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.deleteObject",
|
"command": "mssql.deleteObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(ServerLevelLogin|User|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "connection@2"
|
"group": "connection@2"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "mssql.renameObject",
|
"command": "mssql.renameObject",
|
||||||
"when": "connectionProvider == MSSQL && nodeType =~ /^(Database|ServerLevelLogin|User|Table|View)$/ && config.workbench.enablePreviewFeatures",
|
"when": "connectionProvider == MSSQL && nodeType =~ /^(Database|ServerLevelLogin|User|Table|View|ServerLevelServerRole|ApplicationRole|DatabaseRole)$/ && config.workbench.enablePreviewFeatures",
|
||||||
"group": "connection@3"
|
"group": "connection@3"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1612,6 +1612,17 @@ export namespace DropObjectRequest {
|
|||||||
export const type = new RequestType<DropObjectRequestParams, void, void, void>('objectManagement/drop');
|
export const type = new RequestType<DropObjectRequestParams, void, void, void>('objectManagement/drop');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface SearchObjectRequestParams {
|
||||||
|
contextId: string;
|
||||||
|
searchText: string | undefined;
|
||||||
|
schema: string | undefined;
|
||||||
|
objectTypes: mssql.ObjectManagement.NodeType[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export namespace SearchObjectRequest {
|
||||||
|
export const type = new RequestType<SearchObjectRequestParams, mssql.ObjectManagement.SearchResultItem[], void, void>('objectManagement/search');
|
||||||
|
}
|
||||||
|
|
||||||
// ------------------------------- < Object Management > ------------------------------------
|
// ------------------------------- < Object Management > ------------------------------------
|
||||||
|
|
||||||
// ------------------------------- < Encryption IV/KEY updation Event > ------------------------------------
|
// ------------------------------- < Encryption IV/KEY updation Event > ------------------------------------
|
||||||
|
|||||||
117
extensions/mssql/src/mssql.d.ts
vendored
117
extensions/mssql/src/mssql.d.ts
vendored
@@ -895,9 +895,12 @@ declare module 'mssql' {
|
|||||||
* Object types.
|
* Object types.
|
||||||
*/
|
*/
|
||||||
export const enum NodeType {
|
export const enum NodeType {
|
||||||
|
ApplicationRole = "ApplicationRole",
|
||||||
Column = "Column",
|
Column = "Column",
|
||||||
Database = "Database",
|
Database = "Database",
|
||||||
|
DatabaseRole = "DatabaseRole",
|
||||||
ServerLevelLogin = "ServerLevelLogin",
|
ServerLevelLogin = "ServerLevelLogin",
|
||||||
|
ServerLevelServerRole = "ServerLevelServerRole",
|
||||||
Table = "Table",
|
Table = "Table",
|
||||||
User = "User",
|
User = "User",
|
||||||
View = "View"
|
View = "View"
|
||||||
@@ -1212,6 +1215,112 @@ declare module 'mssql' {
|
|||||||
*/
|
*/
|
||||||
databaseRoles: string[];
|
databaseRoles: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the server role object.
|
||||||
|
*/
|
||||||
|
export interface ServerRoleInfo extends SqlObject {
|
||||||
|
/**
|
||||||
|
* Name of the server principal that owns the server role.
|
||||||
|
*/
|
||||||
|
owner: string;
|
||||||
|
/**
|
||||||
|
* Name of the server principals that are members of the server role.
|
||||||
|
*/
|
||||||
|
members: string[];
|
||||||
|
/**
|
||||||
|
* Server roles that the server role is a member of.
|
||||||
|
*/
|
||||||
|
memberships: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the information required to render the server role view.
|
||||||
|
*/
|
||||||
|
export interface ServerRoleViewInfo extends ObjectViewInfo<ServerRoleInfo> {
|
||||||
|
/**
|
||||||
|
* Whether the server role is a fixed role.
|
||||||
|
*/
|
||||||
|
isFixedRole: boolean;
|
||||||
|
/**
|
||||||
|
* List of all the server roles.
|
||||||
|
*/
|
||||||
|
serverRoles: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the application role object.
|
||||||
|
*/
|
||||||
|
export interface ApplicationRoleInfo extends SqlObject {
|
||||||
|
/**
|
||||||
|
* Default schema of the application role.
|
||||||
|
*/
|
||||||
|
defaultSchema: string;
|
||||||
|
/**
|
||||||
|
* Schemas owned by the application role.
|
||||||
|
*/
|
||||||
|
ownedSchemas: string[];
|
||||||
|
/**
|
||||||
|
* Password of the application role.
|
||||||
|
*/
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the information required to render the application role view.
|
||||||
|
*/
|
||||||
|
export interface ApplicationRoleViewInfo extends ObjectViewInfo<ApplicationRoleInfo> {
|
||||||
|
/**
|
||||||
|
* List of all the schemas in the database.
|
||||||
|
*/
|
||||||
|
schemas: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the database role object.
|
||||||
|
*/
|
||||||
|
export interface DatabaseRoleInfo extends SqlObject {
|
||||||
|
/**
|
||||||
|
* Name of the database principal that owns the database role.
|
||||||
|
*/
|
||||||
|
owner: string;
|
||||||
|
/**
|
||||||
|
* Schemas owned by the database role.
|
||||||
|
*/
|
||||||
|
ownedSchemas: string[];
|
||||||
|
/**
|
||||||
|
* Name of the user or database role that are members of the database role.
|
||||||
|
*/
|
||||||
|
members: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing the information required to render the database role view.
|
||||||
|
*/
|
||||||
|
export interface DatabaseRoleViewInfo extends ObjectViewInfo<DatabaseRoleInfo> {
|
||||||
|
/**
|
||||||
|
* List of all the schemas in the database.
|
||||||
|
*/
|
||||||
|
schemas: string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface representing an item in the search result.
|
||||||
|
*/
|
||||||
|
export interface SearchResultItem {
|
||||||
|
/**
|
||||||
|
* name of the object.
|
||||||
|
*/
|
||||||
|
name: string;
|
||||||
|
/**
|
||||||
|
* type of the object.
|
||||||
|
*/
|
||||||
|
type: NodeType;
|
||||||
|
/**
|
||||||
|
* schema of the object.
|
||||||
|
*/
|
||||||
|
schema: string | undefined;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface IObjectManagementService {
|
export interface IObjectManagementService {
|
||||||
@@ -1258,6 +1367,14 @@ declare module 'mssql' {
|
|||||||
* @param objectUrn SMO Urn of the object to be dropped. More information: https://learn.microsoft.com/sql/relational-databases/server-management-objects-smo/overview-smo
|
* @param objectUrn SMO Urn of the object to be dropped. More information: https://learn.microsoft.com/sql/relational-databases/server-management-objects-smo/overview-smo
|
||||||
*/
|
*/
|
||||||
drop(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string): Thenable<void>;
|
drop(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string): Thenable<void>;
|
||||||
|
/**
|
||||||
|
* Search for objects.
|
||||||
|
* @param contextId The object view's context id.
|
||||||
|
* @param objectTypes The object types to search for.
|
||||||
|
* @param searchText Search text.
|
||||||
|
* @param schema Schema to search in.
|
||||||
|
*/
|
||||||
|
search(contextId: string, objectTypes: ObjectManagement.NodeType[], searchText?: string, schema?: string): Thenable<ObjectManagement.SearchResultItem[]>;
|
||||||
}
|
}
|
||||||
// Object Management - End.
|
// Object Management - End.
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -17,6 +17,9 @@ import * as constants from '../constants';
|
|||||||
import { getNodeTypeDisplayName, refreshParentNode } from './utils';
|
import { getNodeTypeDisplayName, refreshParentNode } from './utils';
|
||||||
import { TelemetryReporter } from '../telemetry';
|
import { TelemetryReporter } from '../telemetry';
|
||||||
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './ui/objectManagementDialogBase';
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './ui/objectManagementDialogBase';
|
||||||
|
import { ServerRoleDialog } from './ui/serverRoleDialog';
|
||||||
|
import { DatabaseRoleDialog } from './ui/databaseRoleDialog';
|
||||||
|
import { ApplicationRoleDialog } from './ui/applicationRoleDialog';
|
||||||
|
|
||||||
export function registerObjectManagementCommands(appContext: AppContext) {
|
export function registerObjectManagementCommands(appContext: AppContext) {
|
||||||
// Notes: Change the second parameter to false to use the actual object management service.
|
// Notes: Change the second parameter to false to use the actual object management service.
|
||||||
@@ -48,13 +51,22 @@ async function handleNewObjectDialogCommand(context: azdata.ObjectExplorerContex
|
|||||||
if (!connectionUri) {
|
if (!connectionUri) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
let newObjectType: ObjectManagement.NodeType;
|
let objectType: ObjectManagement.NodeType;
|
||||||
switch (context.nodeInfo!.objectType) {
|
switch (context.nodeInfo!.objectType) {
|
||||||
|
case FolderType.ApplicationRoles:
|
||||||
|
objectType = ObjectManagement.NodeType.ApplicationRole;
|
||||||
|
break;
|
||||||
|
case FolderType.DatabaseRoles:
|
||||||
|
objectType = ObjectManagement.NodeType.DatabaseRole;
|
||||||
|
break;
|
||||||
case FolderType.ServerLevelLogins:
|
case FolderType.ServerLevelLogins:
|
||||||
newObjectType = ObjectManagement.NodeType.ServerLevelLogin;
|
objectType = ObjectManagement.NodeType.ServerLevelLogin;
|
||||||
|
break;
|
||||||
|
case FolderType.ServerLevelServerRoles:
|
||||||
|
objectType = ObjectManagement.NodeType.ServerLevelServerRole;
|
||||||
break;
|
break;
|
||||||
case FolderType.Users:
|
case FolderType.Users:
|
||||||
newObjectType = ObjectManagement.NodeType.User;
|
objectType = ObjectManagement.NodeType.User;
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unsupported folder type: ${context.nodeInfo!.objectType}`);
|
throw new Error(`Unsupported folder type: ${context.nodeInfo!.objectType}`);
|
||||||
@@ -66,7 +78,7 @@ async function handleNewObjectDialogCommand(context: azdata.ObjectExplorerContex
|
|||||||
connectionUri: connectionUri,
|
connectionUri: connectionUri,
|
||||||
isNewObject: true,
|
isNewObject: true,
|
||||||
database: context.connectionProfile!.databaseName!,
|
database: context.connectionProfile!.databaseName!,
|
||||||
objectType: newObjectType,
|
objectType: objectType,
|
||||||
objectName: '',
|
objectName: '',
|
||||||
parentUrn: parentUrn,
|
parentUrn: parentUrn,
|
||||||
objectExplorerContext: context
|
objectExplorerContext: context
|
||||||
@@ -78,7 +90,8 @@ async function handleNewObjectDialogCommand(context: azdata.ObjectExplorerContex
|
|||||||
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.OpenNewObjectDialog, err).withAdditionalProperties({
|
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.OpenNewObjectDialog, err).withAdditionalProperties({
|
||||||
objectType: context.nodeInfo!.nodeType
|
objectType: context.nodeInfo!.nodeType
|
||||||
}).send();
|
}).send();
|
||||||
await vscode.window.showErrorMessage(localizedConstants.OpenNewObjectDialogError(localizedConstants.LoginTypeDisplayName, getErrorMessage(err)));
|
console.error(err);
|
||||||
|
await vscode.window.showErrorMessage(localizedConstants.OpenNewObjectDialogError(getNodeTypeDisplayName(objectType), getErrorMessage(err)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -87,7 +100,6 @@ async function handleObjectPropertiesDialogCommand(context: azdata.ObjectExplore
|
|||||||
if (!connectionUri) {
|
if (!connectionUri) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const nodeTypeDisplayName = getNodeTypeDisplayName(context.nodeInfo!.nodeType);
|
|
||||||
try {
|
try {
|
||||||
const parentUrn = await getParentUrn(context);
|
const parentUrn = await getParentUrn(context);
|
||||||
const options: ObjectManagementDialogOptions = {
|
const options: ObjectManagementDialogOptions = {
|
||||||
@@ -107,7 +119,8 @@ async function handleObjectPropertiesDialogCommand(context: azdata.ObjectExplore
|
|||||||
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.OpenPropertiesDialog, err).withAdditionalProperties({
|
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.OpenPropertiesDialog, err).withAdditionalProperties({
|
||||||
objectType: context.nodeInfo!.nodeType
|
objectType: context.nodeInfo!.nodeType
|
||||||
}).send();
|
}).send();
|
||||||
await vscode.window.showErrorMessage(localizedConstants.OpenObjectPropertiesDialogError(nodeTypeDisplayName, context.nodeInfo!.label, getErrorMessage(err)));
|
console.error(err);
|
||||||
|
await vscode.window.showErrorMessage(localizedConstants.OpenObjectPropertiesDialogError(getNodeTypeDisplayName(context.nodeInfo!.nodeType), context.nodeInfo!.label, getErrorMessage(err)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -152,6 +165,7 @@ async function handleDeleteObjectCommand(context: azdata.ObjectExplorerContext,
|
|||||||
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.DeleteObject, err).withAdditionalProperties({
|
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.DeleteObject, err).withAdditionalProperties({
|
||||||
objectType: context.nodeInfo!.nodeType
|
objectType: context.nodeInfo!.nodeType
|
||||||
}).send();
|
}).send();
|
||||||
|
console.error(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
operation.updateStatus(azdata.TaskStatus.Succeeded);
|
operation.updateStatus(azdata.TaskStatus.Succeeded);
|
||||||
@@ -204,6 +218,7 @@ async function handleRenameObjectCommand(context: azdata.ObjectExplorerContext,
|
|||||||
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.RenameObject, err).withAdditionalProperties({
|
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, TelemetryActions.RenameObject, err).withAdditionalProperties({
|
||||||
objectType: context.nodeInfo!.nodeType
|
objectType: context.nodeInfo!.nodeType
|
||||||
}).send();
|
}).send();
|
||||||
|
console.error(err);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
operation.updateStatus(azdata.TaskStatus.Succeeded);
|
operation.updateStatus(azdata.TaskStatus.Succeeded);
|
||||||
@@ -214,8 +229,14 @@ async function handleRenameObjectCommand(context: azdata.ObjectExplorerContext,
|
|||||||
|
|
||||||
function getDialog(service: IObjectManagementService, dialogOptions: ObjectManagementDialogOptions): ObjectManagementDialogBase<ObjectManagement.SqlObject, ObjectManagement.ObjectViewInfo<ObjectManagement.SqlObject>> {
|
function getDialog(service: IObjectManagementService, dialogOptions: ObjectManagementDialogOptions): ObjectManagementDialogBase<ObjectManagement.SqlObject, ObjectManagement.ObjectViewInfo<ObjectManagement.SqlObject>> {
|
||||||
switch (dialogOptions.objectType) {
|
switch (dialogOptions.objectType) {
|
||||||
|
case ObjectManagement.NodeType.ApplicationRole:
|
||||||
|
return new ApplicationRoleDialog(service, dialogOptions);
|
||||||
|
case ObjectManagement.NodeType.DatabaseRole:
|
||||||
|
return new DatabaseRoleDialog(service, dialogOptions);
|
||||||
case ObjectManagement.NodeType.ServerLevelLogin:
|
case ObjectManagement.NodeType.ServerLevelLogin:
|
||||||
return new LoginDialog(service, dialogOptions);
|
return new LoginDialog(service, dialogOptions);
|
||||||
|
case ObjectManagement.NodeType.ServerLevelServerRole:
|
||||||
|
return new ServerRoleDialog(service, dialogOptions);
|
||||||
case ObjectManagement.NodeType.User:
|
case ObjectManagement.NodeType.User:
|
||||||
return new UserDialog(service, dialogOptions);
|
return new UserDialog(service, dialogOptions);
|
||||||
default:
|
default:
|
||||||
|
|||||||
@@ -7,16 +7,25 @@
|
|||||||
* The folder types in object explorer.
|
* The folder types in object explorer.
|
||||||
*/
|
*/
|
||||||
export const enum FolderType {
|
export const enum FolderType {
|
||||||
|
ApplicationRoles = 'ApplicationRoles',
|
||||||
|
DatabaseRoles = 'DatabaseRoles',
|
||||||
ServerLevelLogins = 'ServerLevelLogins',
|
ServerLevelLogins = 'ServerLevelLogins',
|
||||||
|
ServerLevelServerRoles = 'ServerLevelServerRoles',
|
||||||
Users = 'Users'
|
Users = 'Users'
|
||||||
}
|
}
|
||||||
|
|
||||||
export const PublicServerRoleName = 'public';
|
export const PublicServerRoleName = 'public';
|
||||||
|
|
||||||
export const CreateUserDocUrl = 'https://learn.microsoft.com/en-us/sql/t-sql/statements/create-user-transact-sql';
|
export const CreateUserDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/create-user-transact-sql';
|
||||||
export const AlterUserDocUrl = 'https://learn.microsoft.com/en-us/sql/t-sql/statements/alter-user-transact-sql';
|
export const AlterUserDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/alter-user-transact-sql';
|
||||||
export const CreateLoginDocUrl = 'https://learn.microsoft.com/en-us/sql/t-sql/statements/create-login-transact-sql';
|
export const CreateLoginDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/create-login-transact-sql';
|
||||||
export const AlterLoginDocUrl = 'https://learn.microsoft.com/en-us/sql/t-sql/statements/alter-login-transact-sql';
|
export const AlterLoginDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/alter-login-transact-sql';
|
||||||
|
export const CreateServerRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/create-server-role-transact-sql';
|
||||||
|
export const AlterServerRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/alter-server-role-transact-sql';
|
||||||
|
export const CreateApplicationRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/create-application-role-transact-sql';
|
||||||
|
export const AlterApplicationRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/alter-application-role-transact-sql';
|
||||||
|
export const CreateDatabaseRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/create-role-transact-sql';
|
||||||
|
export const AlterDatabaseRoleDocUrl = 'https://learn.microsoft.com/sql/t-sql/statements/alter-role-transact-sql';
|
||||||
|
|
||||||
export const enum TelemetryActions {
|
export const enum TelemetryActions {
|
||||||
CreateObject = 'CreateObject',
|
CreateObject = 'CreateObject',
|
||||||
|
|||||||
@@ -15,6 +15,12 @@ export const TableTypeDisplayName: string = localize('objectManagement.TableDisp
|
|||||||
export const ViewTypeDisplayName: string = localize('objectManagement.ViewDisplayName', "view");
|
export const ViewTypeDisplayName: string = localize('objectManagement.ViewDisplayName', "view");
|
||||||
export const ColumnTypeDisplayName: string = localize('objectManagement.ColumnDisplayName', "column");
|
export const ColumnTypeDisplayName: string = localize('objectManagement.ColumnDisplayName', "column");
|
||||||
export const DatabaseTypeDisplayName: string = localize('objectManagement.DatabaseDisplayName', "database");
|
export const DatabaseTypeDisplayName: string = localize('objectManagement.DatabaseDisplayName', "database");
|
||||||
|
export const ServerRoleTypeDisplayName: string = localize('objectManagement.ServerRoleTypeDisplayName', "server role");
|
||||||
|
export const ServerRoleTypeDisplayNameInTitle: string = localize('objectManagement.ServerRoleTypeDisplayNameInTitle', "Server Role");
|
||||||
|
export const ApplicationRoleTypeDisplayName: string = localize('objectManagement.ApplicationRoleTypeDisplayName', "application role");
|
||||||
|
export const ApplicationRoleTypeDisplayNameInTitle: string = localize('objectManagement.ApplicationRoleTypeDisplayNameInTitle', "Application Role");
|
||||||
|
export const DatabaseRoleTypeDisplayName: string = localize('objectManagement.DatabaseRoleTypeDisplayName', "database role");
|
||||||
|
export const DatabaseRoleTypeDisplayNameInTitle: string = localize('objectManagement.DatabaseRoleTypeDisplayNameInTitle', "Database Role");
|
||||||
|
|
||||||
// Shared Strings
|
// Shared Strings
|
||||||
export const HelpText: string = localize('objectManagement.helpText', "Help");
|
export const HelpText: string = localize('objectManagement.helpText', "Help");
|
||||||
@@ -26,6 +32,13 @@ export const RenameObjectDialogTitle: string = localize('objectManagement.rename
|
|||||||
export const ScriptText: string = localize('objectManagement.scriptText', "Script");
|
export const ScriptText: string = localize('objectManagement.scriptText', "Script");
|
||||||
export const NoActionScriptedMessage: string = localize('objectManagement.noActionScriptedMessage', "There is no action to be scripted.");
|
export const NoActionScriptedMessage: string = localize('objectManagement.noActionScriptedMessage', "There is no action to be scripted.");
|
||||||
export const ScriptGeneratedText: string = localize('objectManagement.scriptGenerated', "Script has been generated successfully. You can close the dialog to view it in the newly opened editor.")
|
export const ScriptGeneratedText: string = localize('objectManagement.scriptGenerated', "Script has been generated successfully. You can close the dialog to view it in the newly opened editor.")
|
||||||
|
export const OwnerText: string = localize('objectManagement.ownerText', "Owner");
|
||||||
|
export const BrowseText = localize('objectManagement.browseText', "Browse…");
|
||||||
|
export const BrowseOwnerButtonAriaLabel = localize('objectManagement.browseForOwnerText', "Browse for an owner");
|
||||||
|
export const AddText = localize('objectManagement.addText', "Add…");
|
||||||
|
export const RemoveText = localize('objectManagement.removeText', "Remove");
|
||||||
|
export const AddMemberAriaLabel = localize('objectManagement.addMemberText', "Add a member");
|
||||||
|
export const RemoveMemberAriaLabel = localize('objectManagement.removeMemberText', "Remove selected member");
|
||||||
|
|
||||||
|
|
||||||
export function RefreshObjectExplorerError(error: string): string {
|
export function RefreshObjectExplorerError(error: string): string {
|
||||||
@@ -128,6 +141,9 @@ export const PasswordCannotBeEmptyError = localize('objectManagement.passwordCan
|
|||||||
export const PasswordsNotMatchError = localize('objectManagement.passwordsNotMatchError', "Password must match the confirm password.");
|
export const PasswordsNotMatchError = localize('objectManagement.passwordsNotMatchError', "Password must match the confirm password.");
|
||||||
export const InvalidPasswordError = localize('objectManagement.invalidPasswordError', "Password doesn't meet the complexity requirement. For more information: https://docs.microsoft.com/sql/relational-databases/security/password-policy");
|
export const InvalidPasswordError = localize('objectManagement.invalidPasswordError', "Password doesn't meet the complexity requirement. For more information: https://docs.microsoft.com/sql/relational-databases/security/password-policy");
|
||||||
export const LoginNotSelectedError = localize('objectManagement.loginNotSelectedError', "Login is not selected.");
|
export const LoginNotSelectedError = localize('objectManagement.loginNotSelectedError', "Login is not selected.");
|
||||||
|
export const MembershipSectionHeader = localize('objectManagement.membershipLabel', "Membership");
|
||||||
|
export const MemberSectionHeader = localize('objectManagement.membersLabel', "Members");
|
||||||
|
export const SchemaText = localize('objectManagement.schemaLabel', "Schema");
|
||||||
|
|
||||||
// Login
|
// Login
|
||||||
export const BlankPasswordConfirmationText: string = localize('objectManagement.blankPasswordConfirmation', "Creating a login with a blank password is a security risk. Are you sure you want to continue?");
|
export const BlankPasswordConfirmationText: string = localize('objectManagement.blankPasswordConfirmation', "Creating a login with a blank password is a security risk. Are you sure you want to continue?");
|
||||||
@@ -158,4 +174,22 @@ export const UserWithNoConnectAccess = localize('objectManagement.user.userWithN
|
|||||||
export const DefaultSchemaText = localize('objectManagement.user.defaultSchemaLabel', "Default schema");
|
export const DefaultSchemaText = localize('objectManagement.user.defaultSchemaLabel', "Default schema");
|
||||||
export const LoginText = localize('objectManagement.user.loginLabel', "Login");
|
export const LoginText = localize('objectManagement.user.loginLabel', "Login");
|
||||||
export const OwnedSchemaSectionHeader = localize('objectManagement.user.ownedSchemasLabel', "Owned Schemas");
|
export const OwnedSchemaSectionHeader = localize('objectManagement.user.ownedSchemasLabel', "Owned Schemas");
|
||||||
export const MembershipSectionHeader = localize('objectManagement.user.membershipLabel', "Membership");
|
|
||||||
|
// Database Role
|
||||||
|
export const SelectDatabaseRoleMemberDialogTitle = localize('objectManagement.databaseRole.SelectMemberDialogTitle', "Select Database Role Members");
|
||||||
|
export const SelectDatabaseRoleOwnerDialogTitle = localize('objectManagement.databaseRole.SelectOwnerDialogTitle', "Select Database Role Owner");
|
||||||
|
|
||||||
|
// Server Role
|
||||||
|
export const SelectServerRoleMemberDialogTitle = localize('objectManagement.serverRole.SelectMemberDialogTitle', "Select Server Role Members");
|
||||||
|
export const SelectServerRoleOwnerDialogTitle = localize('objectManagement.serverRole.SelectOwnerDialogTitle', "Select Server Role Owner");
|
||||||
|
|
||||||
|
// Find Object Dialog
|
||||||
|
export const ObjectTypeText = localize('objectManagement.objectTypeLabel', "Object Type");
|
||||||
|
export const FilterText = localize('objectManagement.filterText', "Filter");
|
||||||
|
export const FindText = localize('objectManagement.findText', "Find");
|
||||||
|
export const SelectText = localize('objectManagement.selectText', "Select");
|
||||||
|
export const ObjectsText = localize('objectManagement.objectsLabel', "Objects");
|
||||||
|
export const LoadingObjectsText = localize('objectManagement.loadingObjectsLabel', "Loading objects…");
|
||||||
|
export function LoadingObjectsCompletedText(count: number): string {
|
||||||
|
return localize('objectManagement.loadingObjectsCompletedLabel', "Loading objects completed, {0} objects found", count);
|
||||||
|
}
|
||||||
|
|||||||
@@ -61,39 +61,39 @@ export class ObjectManagementService extends BaseService implements IObjectManag
|
|||||||
const params: contracts.DropObjectRequestParams = { connectionUri, objectUrn, objectType };
|
const params: contracts.DropObjectRequestParams = { connectionUri, objectUrn, objectType };
|
||||||
return this.runWithErrorHandling(contracts.DropObjectRequest.type, params);
|
return this.runWithErrorHandling(contracts.DropObjectRequest.type, params);
|
||||||
}
|
}
|
||||||
|
async search(contextId: string, objectTypes: ObjectManagement.NodeType[], searchText?: string, schema?: string): Promise<ObjectManagement.SearchResultItem[]> {
|
||||||
|
const params: contracts.SearchObjectRequestParams = { contextId, searchText, objectTypes, schema };
|
||||||
|
return this.runWithErrorHandling(contracts.SearchObjectRequest.type, params);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export class TestObjectManagementService implements IObjectManagementService {
|
export class TestObjectManagementService implements IObjectManagementService {
|
||||||
initializeView(contextId: string, objectType: ObjectManagement.NodeType, connectionUri: string, database: string, isNewObject: boolean, parentUrn: string, objectUrn: string): Thenable<ObjectManagement.ObjectViewInfo<ObjectManagement.SqlObject>> {
|
initializeView(contextId: string, objectType: ObjectManagement.NodeType, connectionUri: string, database: string, isNewObject: boolean, parentUrn: string, objectUrn: string): Thenable<ObjectManagement.ObjectViewInfo<ObjectManagement.SqlObject>> {
|
||||||
if (objectType === ObjectManagement.NodeType.ServerLevelLogin) {
|
let obj;
|
||||||
return Promise.resolve(this.getLoginView(isNewObject, objectUrn));
|
if (objectType === ObjectManagement.NodeType.ApplicationRole) {
|
||||||
|
obj = this.getApplicationRoleView(isNewObject, objectUrn);
|
||||||
|
} else if (objectType === ObjectManagement.NodeType.DatabaseRole) {
|
||||||
|
obj = this.getDatabaseRoleView(isNewObject, objectUrn);
|
||||||
|
} else if (objectType === ObjectManagement.NodeType.ServerLevelLogin) {
|
||||||
|
obj = this.getLoginView(isNewObject, objectUrn);
|
||||||
|
} else if (objectType === ObjectManagement.NodeType.ServerLevelServerRole) {
|
||||||
|
obj = this.getServerRoleView(isNewObject, objectUrn);
|
||||||
} else if (objectType === ObjectManagement.NodeType.User) {
|
} else if (objectType === ObjectManagement.NodeType.User) {
|
||||||
return Promise.resolve(this.getUserView(isNewObject, objectUrn));
|
obj = this.getUserView(isNewObject, objectUrn);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
throw Error('Not implemented');
|
throw Error('Not implemented');
|
||||||
}
|
}
|
||||||
|
return this.delayAndResolve(obj);
|
||||||
}
|
}
|
||||||
save(contextId: string, object: ObjectManagement.SqlObject): Thenable<void> {
|
save(contextId: string, object: ObjectManagement.SqlObject): Thenable<void> {
|
||||||
return new Promise((resolve, reject) => {
|
return this.delayAndResolve();
|
||||||
setTimeout(() => {
|
|
||||||
resolve();
|
|
||||||
}, 3000);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
script(contextId: string, object: ObjectManagement.SqlObject): Thenable<string> {
|
script(contextId: string, object: ObjectManagement.SqlObject): Thenable<string> {
|
||||||
return new Promise((resolve, reject) => {
|
return this.delayAndResolve('test script');
|
||||||
setTimeout(() => {
|
|
||||||
resolve('test script');
|
|
||||||
}, 1000);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
disposeView(contextId: string): Thenable<void> {
|
disposeView(contextId: string): Thenable<void> {
|
||||||
return new Promise((resolve, reject) => {
|
return this.delayAndResolve();
|
||||||
setTimeout(() => {
|
|
||||||
resolve();
|
|
||||||
}, 100);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
async rename(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string, newName: string): Promise<void> {
|
async rename(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string, newName: string): Promise<void> {
|
||||||
return this.delayAndResolve();
|
return this.delayAndResolve();
|
||||||
@@ -101,6 +101,23 @@ export class TestObjectManagementService implements IObjectManagementService {
|
|||||||
async drop(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string): Promise<void> {
|
async drop(connectionUri: string, objectType: ObjectManagement.NodeType, objectUrn: string): Promise<void> {
|
||||||
return this.delayAndResolve();
|
return this.delayAndResolve();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async search(contextId: string, objectTypes: ObjectManagement.NodeType[], searchText: string, schema: string): Promise<ObjectManagement.SearchResultItem[]> {
|
||||||
|
const items: ObjectManagement.SearchResultItem[] = [];
|
||||||
|
objectTypes.forEach(type => {
|
||||||
|
items.push(...this.generateSearchResult(type, 15));
|
||||||
|
});
|
||||||
|
return this.delayAndResolve(items);
|
||||||
|
}
|
||||||
|
|
||||||
|
private generateSearchResult(objectType: ObjectManagement.NodeType, count: number): ObjectManagement.SearchResultItem[] {
|
||||||
|
let items: ObjectManagement.SearchResultItem[] = [];
|
||||||
|
for (let i = 0; i < count; i++) {
|
||||||
|
items.push(<ObjectManagement.SearchResultItem>{ name: `${objectType} ${i}`, type: objectType });
|
||||||
|
}
|
||||||
|
return items;
|
||||||
|
}
|
||||||
|
|
||||||
private getLoginView(isNewObject: boolean, name: string): ObjectManagement.LoginViewInfo {
|
private getLoginView(isNewObject: boolean, name: string): ObjectManagement.LoginViewInfo {
|
||||||
const serverRoles = ['sysadmin', 'public', 'bulkadmin', 'dbcreator', 'diskadmin', 'processadmin', 'securityadmin', 'serveradmin'];
|
const serverRoles = ['sysadmin', 'public', 'bulkadmin', 'dbcreator', 'diskadmin', 'processadmin', 'securityadmin', 'serveradmin'];
|
||||||
const languages = ['<default>', 'English'];
|
const languages = ['<default>', 'English'];
|
||||||
@@ -160,6 +177,7 @@ export class TestObjectManagementService implements IObjectManagementService {
|
|||||||
}
|
}
|
||||||
return login;
|
return login;
|
||||||
}
|
}
|
||||||
|
|
||||||
private getUserView(isNewObject: boolean, name: string): ObjectManagement.UserViewInfo {
|
private getUserView(isNewObject: boolean, name: string): ObjectManagement.UserViewInfo {
|
||||||
let viewInfo: ObjectManagement.UserViewInfo;
|
let viewInfo: ObjectManagement.UserViewInfo;
|
||||||
const languages = ['<default>', 'English'];
|
const languages = ['<default>', 'English'];
|
||||||
@@ -213,11 +231,73 @@ export class TestObjectManagementService implements IObjectManagementService {
|
|||||||
}
|
}
|
||||||
return viewInfo;
|
return viewInfo;
|
||||||
}
|
}
|
||||||
private delayAndResolve(): Promise<void> {
|
|
||||||
|
private getServerRoleView(isNewObject: boolean, name: string): ObjectManagement.ServerRoleViewInfo {
|
||||||
|
return isNewObject ? <ObjectManagement.ServerRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: '',
|
||||||
|
members: [],
|
||||||
|
owner: '',
|
||||||
|
memberships: []
|
||||||
|
},
|
||||||
|
isFixedRole: false,
|
||||||
|
serverRoles: ['ServerLevelServerRole 1', 'ServerLevelServerRole 2', 'ServerLevelServerRole 3', 'ServerLevelServerRole 4'],
|
||||||
|
} : <ObjectManagement.ServerRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: 'ServerLevelServerRole 1',
|
||||||
|
members: ['ServerLevelLogin 1', 'ServerLevelServerRole 2'],
|
||||||
|
owner: 'ServerLevelLogin 2',
|
||||||
|
memberships: ['ServerLevelServerRole 3', 'ServerLevelServerRole 4']
|
||||||
|
},
|
||||||
|
isFixedRole: false,
|
||||||
|
serverRoles: ['ServerLevelServerRole 2', 'ServerLevelServerRole 3', 'ServerLevelServerRole 4']
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private getApplicationRoleView(isNewObject: boolean, name: string): ObjectManagement.ApplicationRoleViewInfo {
|
||||||
|
return isNewObject ? <ObjectManagement.ApplicationRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: '',
|
||||||
|
defaultSchema: 'dbo',
|
||||||
|
ownedSchemas: [],
|
||||||
|
},
|
||||||
|
schemas: ['dbo', 'sys', 'admin']
|
||||||
|
} : <ObjectManagement.ApplicationRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: 'app role1',
|
||||||
|
password: '******************',
|
||||||
|
defaultSchema: 'dbo',
|
||||||
|
ownedSchemas: ['dbo'],
|
||||||
|
},
|
||||||
|
schemas: ['dbo', 'sys', 'admin']
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private getDatabaseRoleView(isNewObject: boolean, name: string): ObjectManagement.DatabaseRoleViewInfo {
|
||||||
|
return isNewObject ? <ObjectManagement.DatabaseRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: '',
|
||||||
|
owner: '',
|
||||||
|
members: [],
|
||||||
|
ownedSchemas: []
|
||||||
|
},
|
||||||
|
schemas: ['dbo', 'sys', 'admin']
|
||||||
|
} : <ObjectManagement.DatabaseRoleViewInfo>{
|
||||||
|
objectInfo: {
|
||||||
|
name: 'db role1',
|
||||||
|
owner: '',
|
||||||
|
members: [],
|
||||||
|
ownedSchemas: ['dbo']
|
||||||
|
},
|
||||||
|
schemas: ['dbo', 'sys', 'admin']
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private delayAndResolve(obj?: any): Promise<any> {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
resolve();
|
resolve(obj);
|
||||||
}, 3000);
|
}, 1000);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,94 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
import * as azdata from 'azdata';
|
||||||
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
||||||
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
|
import * as localizedConstants from '../localizedConstants';
|
||||||
|
import { AlterApplicationRoleDocUrl, CreateApplicationRoleDocUrl } from '../constants';
|
||||||
|
import { isValidSQLPassword } from '../utils';
|
||||||
|
import { DefaultMaxTableHeight } from './dialogBase';
|
||||||
|
|
||||||
|
export class ApplicationRoleDialog extends ObjectManagementDialogBase<ObjectManagement.ApplicationRoleInfo, ObjectManagement.ApplicationRoleViewInfo> {
|
||||||
|
// Sections
|
||||||
|
private generalSection: azdata.GroupContainer;
|
||||||
|
private ownedSchemasSection: azdata.GroupContainer;
|
||||||
|
|
||||||
|
// General section content
|
||||||
|
private nameInput: azdata.InputBoxComponent;
|
||||||
|
private defaultSchemaDropdown: azdata.DropDownComponent;
|
||||||
|
private passwordInput: azdata.InputBoxComponent;
|
||||||
|
private confirmPasswordInput: azdata.InputBoxComponent;
|
||||||
|
|
||||||
|
// Owned Schemas section content
|
||||||
|
private ownedSchemaTable: azdata.TableComponent;
|
||||||
|
|
||||||
|
constructor(objectManagementService: IObjectManagementService, options: ObjectManagementDialogOptions) {
|
||||||
|
super(objectManagementService, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override postInitializeData(): void {
|
||||||
|
this.objectInfo.password = this.objectInfo.password ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override get docUrl(): string {
|
||||||
|
return this.options.isNewObject ? CreateApplicationRoleDocUrl : AlterApplicationRoleDocUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async validateInput(): Promise<string[]> {
|
||||||
|
const errors = await super.validateInput();
|
||||||
|
if (!this.objectInfo.password) {
|
||||||
|
errors.push(localizedConstants.PasswordCannotBeEmptyError);
|
||||||
|
}
|
||||||
|
if (this.objectInfo.password && !isValidSQLPassword(this.objectInfo.password, this.objectInfo.name)
|
||||||
|
&& (this.options.isNewObject || this.objectInfo.password !== this.originalObjectInfo.password)) {
|
||||||
|
errors.push(localizedConstants.InvalidPasswordError);
|
||||||
|
}
|
||||||
|
if (this.objectInfo.password !== this.confirmPasswordInput.value) {
|
||||||
|
errors.push(localizedConstants.PasswordsNotMatchError);
|
||||||
|
}
|
||||||
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async initializeUI(): Promise<void> {
|
||||||
|
this.initializeGeneralSection();
|
||||||
|
this.initializeOwnedSchemasSection();
|
||||||
|
this.formContainer.addItems([this.generalSection, this.ownedSchemasSection]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeGeneralSection(): void {
|
||||||
|
this.nameInput = this.createInputBox(localizedConstants.NameText, async (newValue) => {
|
||||||
|
this.objectInfo.name = newValue;
|
||||||
|
}, this.objectInfo.name, this.options.isNewObject);
|
||||||
|
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
||||||
|
|
||||||
|
this.defaultSchemaDropdown = this.createDropdown(localizedConstants.DefaultSchemaText, async (newValue) => {
|
||||||
|
this.objectInfo.defaultSchema = newValue;
|
||||||
|
}, this.viewInfo.schemas, this.objectInfo.defaultSchema!);
|
||||||
|
const defaultSchemaContainer = this.createLabelInputContainer(localizedConstants.DefaultSchemaText, this.defaultSchemaDropdown);
|
||||||
|
|
||||||
|
this.passwordInput = this.createPasswordInputBox(localizedConstants.PasswordText, async (newValue) => {
|
||||||
|
this.objectInfo.password = newValue;
|
||||||
|
}, this.objectInfo.password ?? '');
|
||||||
|
const passwordContainer = this.createLabelInputContainer(localizedConstants.PasswordText, this.passwordInput);
|
||||||
|
|
||||||
|
this.confirmPasswordInput = this.createPasswordInputBox(localizedConstants.ConfirmPasswordText, async () => { }, this.objectInfo.password ?? '');
|
||||||
|
const confirmPasswordContainer = this.createLabelInputContainer(localizedConstants.ConfirmPasswordText, this.confirmPasswordInput);
|
||||||
|
|
||||||
|
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [nameContainer, defaultSchemaContainer, passwordContainer, confirmPasswordContainer], false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeOwnedSchemasSection(): void {
|
||||||
|
this.ownedSchemaTable = this.createTableList<string>(localizedConstants.OwnedSchemaSectionHeader,
|
||||||
|
[localizedConstants.SchemaText],
|
||||||
|
this.viewInfo.schemas,
|
||||||
|
this.objectInfo.ownedSchemas,
|
||||||
|
DefaultMaxTableHeight,
|
||||||
|
(item) => {
|
||||||
|
// It is not allowed to have unassigned schema.
|
||||||
|
return this.objectInfo.ownedSchemas.indexOf(item) === -1;
|
||||||
|
});
|
||||||
|
this.ownedSchemasSection = this.createGroup(localizedConstants.OwnedSchemaSectionHeader, [this.ownedSchemaTable]);
|
||||||
|
}
|
||||||
|
}
|
||||||
130
extensions/mssql/src/objectManagement/ui/databaseRoleDialog.ts
Normal file
130
extensions/mssql/src/objectManagement/ui/databaseRoleDialog.ts
Normal file
@@ -0,0 +1,130 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
import * as azdata from 'azdata';
|
||||||
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
||||||
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
|
import * as localizedConstants from '../localizedConstants';
|
||||||
|
import { AlterDatabaseRoleDocUrl, CreateDatabaseRoleDocUrl } from '../constants';
|
||||||
|
import { FindObjectDialog } from './findObjectDialog';
|
||||||
|
import { DefaultMaxTableHeight } from './dialogBase';
|
||||||
|
|
||||||
|
export class DatabaseRoleDialog extends ObjectManagementDialogBase<ObjectManagement.DatabaseRoleInfo, ObjectManagement.DatabaseRoleViewInfo> {
|
||||||
|
// Sections
|
||||||
|
private generalSection: azdata.GroupContainer;
|
||||||
|
private ownedSchemasSection: azdata.GroupContainer;
|
||||||
|
private memberSection: azdata.GroupContainer;
|
||||||
|
|
||||||
|
// General section content
|
||||||
|
private nameInput: azdata.InputBoxComponent;
|
||||||
|
private ownerInput: azdata.InputBoxComponent;
|
||||||
|
|
||||||
|
// Owned Schemas section content
|
||||||
|
private ownedSchemaTable: azdata.TableComponent;
|
||||||
|
|
||||||
|
// Member section content
|
||||||
|
private memberTable: azdata.TableComponent;
|
||||||
|
|
||||||
|
constructor(objectManagementService: IObjectManagementService, options: ObjectManagementDialogOptions) {
|
||||||
|
super(objectManagementService, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override get docUrl(): string {
|
||||||
|
return this.options.isNewObject ? CreateDatabaseRoleDocUrl : AlterDatabaseRoleDocUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async initializeUI(): Promise<void> {
|
||||||
|
this.initializeGeneralSection();
|
||||||
|
this.initializeOwnedSchemasSection();
|
||||||
|
this.initializeMemberSection();
|
||||||
|
this.formContainer.addItems([this.generalSection, this.ownedSchemasSection, this.memberSection]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeGeneralSection(): void {
|
||||||
|
this.nameInput = this.createInputBox(localizedConstants.NameText, async (newValue) => {
|
||||||
|
this.objectInfo.name = newValue;
|
||||||
|
}, this.objectInfo.name, this.options.isNewObject);
|
||||||
|
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
||||||
|
|
||||||
|
this.ownerInput = this.createInputBox(localizedConstants.OwnerText, async (newValue) => {
|
||||||
|
this.objectInfo.owner = newValue;
|
||||||
|
}, this.objectInfo.owner, true, 'text', 210);
|
||||||
|
const browseOwnerButton = this.createButton(localizedConstants.BrowseText, localizedConstants.BrowseOwnerButtonAriaLabel, async () => {
|
||||||
|
const dialog = new FindObjectDialog(this.objectManagementService, {
|
||||||
|
objectTypes: [ObjectManagement.NodeType.ApplicationRole, ObjectManagement.NodeType.DatabaseRole, ObjectManagement.NodeType.User],
|
||||||
|
multiSelect: false,
|
||||||
|
contextId: this.contextId,
|
||||||
|
title: localizedConstants.SelectDatabaseRoleOwnerDialogTitle
|
||||||
|
});
|
||||||
|
await dialog.open();
|
||||||
|
const result = await dialog.waitForClose();
|
||||||
|
if (result.selectedObjects.length > 0) {
|
||||||
|
this.ownerInput.value = result.selectedObjects[0].name;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const ownerContainer = this.createLabelInputContainer(localizedConstants.OwnerText, this.ownerInput);
|
||||||
|
ownerContainer.addItems([browseOwnerButton], { flex: '0 0 auto' });
|
||||||
|
|
||||||
|
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [nameContainer, ownerContainer], false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeMemberSection(): void {
|
||||||
|
this.memberTable = this.createTable(localizedConstants.MemberSectionHeader, [
|
||||||
|
{
|
||||||
|
type: azdata.ColumnType.text,
|
||||||
|
value: localizedConstants.NameText
|
||||||
|
}
|
||||||
|
], this.objectInfo.members.map(m => [m]));
|
||||||
|
const buttonContainer = this.addButtonsForTable(this.memberTable, localizedConstants.AddMemberAriaLabel, localizedConstants.RemoveMemberAriaLabel,
|
||||||
|
async () => {
|
||||||
|
const dialog = new FindObjectDialog(this.objectManagementService, {
|
||||||
|
objectTypes: [ObjectManagement.NodeType.DatabaseRole, ObjectManagement.NodeType.User],
|
||||||
|
multiSelect: true,
|
||||||
|
contextId: this.contextId,
|
||||||
|
title: localizedConstants.SelectDatabaseRoleMemberDialogTitle
|
||||||
|
});
|
||||||
|
await dialog.open();
|
||||||
|
const result = await dialog.waitForClose();
|
||||||
|
this.addMembers(result.selectedObjects.map(r => r.name));
|
||||||
|
},
|
||||||
|
async () => {
|
||||||
|
if (this.memberTable.selectedRows.length === 1) {
|
||||||
|
this.removeMember(this.memberTable.selectedRows[0]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.memberSection = this.createGroup(localizedConstants.MemberSectionHeader, [this.memberTable, buttonContainer]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private addMembers(names: string[]): void {
|
||||||
|
names.forEach(n => {
|
||||||
|
if (this.objectInfo.members.indexOf(n) === -1) {
|
||||||
|
this.objectInfo.members.push(n);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.updateMembersTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
private removeMember(idx: number): void {
|
||||||
|
this.objectInfo.members.splice(idx, 1);
|
||||||
|
this.updateMembersTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
private updateMembersTable(): void {
|
||||||
|
this.setTableData(this.memberTable, this.objectInfo.members.map(m => [m]));
|
||||||
|
this.onFormFieldChange();
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeOwnedSchemasSection(): void {
|
||||||
|
this.ownedSchemaTable = this.createTableList<string>(localizedConstants.OwnedSchemaSectionHeader,
|
||||||
|
[localizedConstants.SchemaText],
|
||||||
|
this.viewInfo.schemas,
|
||||||
|
this.objectInfo.ownedSchemas,
|
||||||
|
DefaultMaxTableHeight,
|
||||||
|
(item) => {
|
||||||
|
// It is not allowed to have unassigned schema.
|
||||||
|
return this.objectInfo.ownedSchemas.indexOf(item) === -1;
|
||||||
|
});
|
||||||
|
this.ownedSchemasSection = this.createGroup(localizedConstants.OwnedSchemaSectionHeader, [this.ownedSchemaTable]);
|
||||||
|
}
|
||||||
|
}
|
||||||
332
extensions/mssql/src/objectManagement/ui/dialogBase.ts
Normal file
332
extensions/mssql/src/objectManagement/ui/dialogBase.ts
Normal file
@@ -0,0 +1,332 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
import * as azdata from 'azdata';
|
||||||
|
import * as vscode from 'vscode';
|
||||||
|
import { EOL } from 'os';
|
||||||
|
import * as localizedConstants from '../localizedConstants';
|
||||||
|
|
||||||
|
export const DefaultLabelWidth = 150;
|
||||||
|
export const DefaultInputWidth = 300;
|
||||||
|
export const DefaultTableWidth = DefaultInputWidth + DefaultLabelWidth;
|
||||||
|
export const DefaultMaxTableHeight = 400;
|
||||||
|
export const DefaultMinTableRowCount = 1;
|
||||||
|
export const TableRowHeight = 25;
|
||||||
|
export const TableColumnHeaderHeight = 30;
|
||||||
|
|
||||||
|
export function getTableHeight(rowCount: number, minRowCount: number = DefaultMinTableRowCount, maxHeight: number = DefaultMaxTableHeight): number {
|
||||||
|
return Math.min(Math.max(rowCount, minRowCount) * TableRowHeight + TableColumnHeaderHeight, maxHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
export type TableListItemEnabledStateGetter<T> = (item: T) => boolean;
|
||||||
|
export type TableListItemValueGetter<T> = (item: T) => string[];
|
||||||
|
export type TableListItemComparer<T> = (item1: T, item2: T) => boolean;
|
||||||
|
export const DefaultTableListItemEnabledStateGetter: TableListItemEnabledStateGetter<any> = (item: any) => true;
|
||||||
|
export const DefaultTableListItemValueGetter: TableListItemValueGetter<any> = (item: any) => [item?.toString() ?? ''];
|
||||||
|
export const DefaultTableListItemComparer: TableListItemComparer<any> = (item1: any, item2: any) => item1 === item2;
|
||||||
|
|
||||||
|
|
||||||
|
export abstract class DialogBase<DialogResult> {
|
||||||
|
protected readonly disposables: vscode.Disposable[] = [];
|
||||||
|
protected readonly dialogObject: azdata.window.Dialog;
|
||||||
|
|
||||||
|
private _modelView: azdata.ModelView;
|
||||||
|
private _loadingComponent: azdata.LoadingComponent;
|
||||||
|
private _formContainer: azdata.DivContainer;
|
||||||
|
private _closePromise: Promise<DialogResult | undefined>;
|
||||||
|
|
||||||
|
constructor(title: string, name: string, width: azdata.window.DialogWidth = 'narrow', style: azdata.window.DialogStyle = 'flyout') {
|
||||||
|
this.dialogObject = azdata.window.createModelViewDialog(title, name, width, style);
|
||||||
|
this.dialogObject.okButton.label = localizedConstants.OkText;
|
||||||
|
this.dialogObject.registerCloseValidator(async (): Promise<boolean> => {
|
||||||
|
const confirmed = await this.onConfirmation();
|
||||||
|
if (!confirmed) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return await this.runValidation();
|
||||||
|
});
|
||||||
|
this._closePromise = new Promise<DialogResult | undefined>(resolve => {
|
||||||
|
this.disposables.push(this.dialogObject.onClosed(async (reason: azdata.window.CloseReason) => {
|
||||||
|
await this.dispose(reason);
|
||||||
|
const result = reason === 'ok' ? this.dialogResult : undefined;
|
||||||
|
resolve(result);
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public waitForClose(): Promise<DialogResult | undefined> {
|
||||||
|
return this._closePromise;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected get dialogResult(): DialogResult | undefined { return undefined; }
|
||||||
|
|
||||||
|
protected async onConfirmation(): Promise<boolean> { return true; }
|
||||||
|
|
||||||
|
protected abstract initialize(): Promise<void>;
|
||||||
|
|
||||||
|
protected get formContainer(): azdata.DivContainer { return this._formContainer; }
|
||||||
|
|
||||||
|
protected get modelView(): azdata.ModelView { return this._modelView; }
|
||||||
|
|
||||||
|
protected onFormFieldChange(): void { }
|
||||||
|
|
||||||
|
protected validateInput(): Promise<string[]> { return Promise.resolve([]); }
|
||||||
|
|
||||||
|
public async open(): Promise<void> {
|
||||||
|
try {
|
||||||
|
this.onLoadingStatusChanged(true);
|
||||||
|
const initializeDialogPromise = new Promise<void>((async resolve => {
|
||||||
|
await this.dialogObject.registerContent(async view => {
|
||||||
|
this._modelView = view;
|
||||||
|
this._formContainer = this.createFormContainer([]);
|
||||||
|
this._loadingComponent = view.modelBuilder.loadingComponent().withItem(this._formContainer).withProps({
|
||||||
|
loading: true,
|
||||||
|
loadingText: localizedConstants.LoadingDialogText,
|
||||||
|
showText: true,
|
||||||
|
CSSStyles: {
|
||||||
|
width: "100%",
|
||||||
|
height: "100%"
|
||||||
|
}
|
||||||
|
}).component();
|
||||||
|
await view.initializeModel(this._loadingComponent);
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
}));
|
||||||
|
azdata.window.openDialog(this.dialogObject);
|
||||||
|
await initializeDialogPromise;
|
||||||
|
await this.initialize();
|
||||||
|
this.onLoadingStatusChanged(false);
|
||||||
|
} catch (err) {
|
||||||
|
azdata.window.closeDialog(this.dialogObject);
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async dispose(reason: azdata.window.CloseReason): Promise<void> {
|
||||||
|
this.disposables.forEach(disposable => disposable.dispose());
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async runValidation(showErrorMessage: boolean = true): Promise<boolean> {
|
||||||
|
const errors = await this.validateInput();
|
||||||
|
if (errors.length > 0 && (this.dialogObject.message?.text || showErrorMessage)) {
|
||||||
|
this.dialogObject.message = {
|
||||||
|
text: errors.join(EOL),
|
||||||
|
level: azdata.window.MessageLevel.Error
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
this.dialogObject.message = undefined;
|
||||||
|
}
|
||||||
|
return errors.length === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createLabelInputContainer(label: string, input: azdata.InputBoxComponent | azdata.DropDownComponent): azdata.FlexContainer {
|
||||||
|
const labelComponent = this.modelView.modelBuilder.text().withProps({ width: DefaultLabelWidth, value: label, requiredIndicator: input.required }).component();
|
||||||
|
const container = this.modelView.modelBuilder.flexContainer().withLayout({ flexFlow: 'horizontal', flexWrap: 'nowrap', alignItems: 'center' }).withItems([labelComponent], { flex: '0 0 auto' }).component();
|
||||||
|
container.addItem(input, { flex: '1 1 auto' });
|
||||||
|
return container;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createCheckbox(label: string, handler: (checked: boolean) => Promise<void>, checked: boolean = false, enabled: boolean = true): azdata.CheckBoxComponent {
|
||||||
|
const checkbox = this.modelView.modelBuilder.checkBox().withProps({
|
||||||
|
label: label,
|
||||||
|
checked: checked,
|
||||||
|
enabled: enabled
|
||||||
|
}).component();
|
||||||
|
this.disposables.push(checkbox.onChanged(async () => {
|
||||||
|
await handler(checkbox.checked!);
|
||||||
|
this.onFormFieldChange();
|
||||||
|
await this.runValidation(false);
|
||||||
|
}));
|
||||||
|
return checkbox;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createPasswordInputBox(ariaLabel: string, textChangeHandler: (newValue: string) => Promise<void>, value: string = '', enabled: boolean = true, width: number = DefaultInputWidth): azdata.InputBoxComponent {
|
||||||
|
return this.createInputBox(ariaLabel, textChangeHandler, value, enabled, 'password', width);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createInputBox(ariaLabel: string, textChangeHandler: (newValue: string) => Promise<void>, value: string = '', enabled: boolean = true, type: azdata.InputBoxInputType = 'text', width: number = DefaultInputWidth): azdata.InputBoxComponent {
|
||||||
|
const inputbox = this.modelView.modelBuilder.inputBox().withProps({ inputType: type, enabled: enabled, ariaLabel: ariaLabel, value: value, width: width }).component();
|
||||||
|
this.disposables.push(inputbox.onTextChanged(async () => {
|
||||||
|
await textChangeHandler(inputbox.value!);
|
||||||
|
this.onFormFieldChange();
|
||||||
|
await this.runValidation(false);
|
||||||
|
}));
|
||||||
|
return inputbox;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createGroup(header: string, items: azdata.Component[], collapsible: boolean = true, collapsed: boolean = false): azdata.GroupContainer {
|
||||||
|
return this.modelView.modelBuilder.groupContainer().withLayout({
|
||||||
|
header: header,
|
||||||
|
collapsible: collapsible,
|
||||||
|
collapsed: collapsed
|
||||||
|
}).withItems(items).component();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createTableList<T>(ariaLabel: string,
|
||||||
|
columnNames: string[],
|
||||||
|
allItems: T[],
|
||||||
|
selectedItems: T[],
|
||||||
|
maxHeight: number = DefaultMaxTableHeight,
|
||||||
|
enabledStateGetter: TableListItemEnabledStateGetter<T> = DefaultTableListItemEnabledStateGetter,
|
||||||
|
rowValueGetter: TableListItemValueGetter<T> = DefaultTableListItemValueGetter,
|
||||||
|
itemComparer: TableListItemComparer<T> = DefaultTableListItemComparer): azdata.TableComponent {
|
||||||
|
const data = this.getDataForTableList(allItems, selectedItems, enabledStateGetter, rowValueGetter, itemComparer);
|
||||||
|
const table = this.modelView.modelBuilder.table().withProps(
|
||||||
|
{
|
||||||
|
ariaLabel: ariaLabel,
|
||||||
|
data: data,
|
||||||
|
columns: [
|
||||||
|
{
|
||||||
|
value: localizedConstants.SelectedText,
|
||||||
|
type: azdata.ColumnType.checkBox,
|
||||||
|
options: { actionOnCheckbox: azdata.ActionOnCellCheckboxCheck.customAction }
|
||||||
|
}, ...columnNames.map(name => {
|
||||||
|
return { value: name };
|
||||||
|
})
|
||||||
|
],
|
||||||
|
width: DefaultTableWidth,
|
||||||
|
height: getTableHeight(data.length, DefaultMinTableRowCount, maxHeight)
|
||||||
|
}
|
||||||
|
).component();
|
||||||
|
this.disposables.push(table.onCellAction!((arg: azdata.ICheckboxCellActionEventArgs) => {
|
||||||
|
const item = allItems[arg.row];
|
||||||
|
const idx = selectedItems.findIndex(i => itemComparer(i, item));
|
||||||
|
if (arg.checked && idx === -1) {
|
||||||
|
selectedItems.push(item);
|
||||||
|
} else if (!arg.checked && idx !== -1) {
|
||||||
|
selectedItems.splice(idx, 1)
|
||||||
|
}
|
||||||
|
this.onFormFieldChange();
|
||||||
|
}));
|
||||||
|
return table;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected setTableData(table: azdata.TableComponent, data: any[][], maxHeight: number = DefaultMaxTableHeight) {
|
||||||
|
table.data = data;
|
||||||
|
table.height = getTableHeight(data.length, DefaultMinTableRowCount, maxHeight);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected getDataForTableList<T>(
|
||||||
|
allItems: T[],
|
||||||
|
selectedItems: T[],
|
||||||
|
enabledStateGetter: TableListItemEnabledStateGetter<T> = DefaultTableListItemEnabledStateGetter,
|
||||||
|
rowValueGetter: TableListItemValueGetter<T> = DefaultTableListItemValueGetter,
|
||||||
|
itemComparer: TableListItemComparer<T> = DefaultTableListItemComparer): any[][] {
|
||||||
|
return allItems.map(item => {
|
||||||
|
const idx = selectedItems.findIndex(i => itemComparer(i, item));
|
||||||
|
const stateColumnValue = { checked: idx !== -1, enabled: enabledStateGetter(item) };
|
||||||
|
return [stateColumnValue, ...rowValueGetter(item)];
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createTable(ariaLabel: string, columns: azdata.TableColumn[], data: any[][], maxHeight: number = DefaultMaxTableHeight): azdata.TableComponent {
|
||||||
|
const table = this.modelView.modelBuilder.table().withProps(
|
||||||
|
{
|
||||||
|
ariaLabel: ariaLabel,
|
||||||
|
data: data,
|
||||||
|
columns: columns,
|
||||||
|
width: DefaultTableWidth,
|
||||||
|
height: getTableHeight(data.length, DefaultMinTableRowCount, maxHeight)
|
||||||
|
}
|
||||||
|
).component();
|
||||||
|
return table;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected addButtonsForTable(table: azdata.TableComponent, addButtonAriaLabel: string, removeButtonAriaLabel: string, addHandler: () => Promise<void>, removeHandler: () => void): azdata.FlexContainer {
|
||||||
|
let addButton: azdata.ButtonComponent;
|
||||||
|
let removeButton: azdata.ButtonComponent;
|
||||||
|
const updateButtons = () => {
|
||||||
|
removeButton.enabled = table.selectedRows.length > 0;
|
||||||
|
}
|
||||||
|
addButton = this.createButton(localizedConstants.AddText, addButtonAriaLabel, async () => {
|
||||||
|
await addHandler();
|
||||||
|
updateButtons();
|
||||||
|
});
|
||||||
|
removeButton = this.createButton(localizedConstants.RemoveText, removeButtonAriaLabel, async () => {
|
||||||
|
await removeHandler();
|
||||||
|
updateButtons();
|
||||||
|
}, false);
|
||||||
|
this.disposables.push(table.onRowSelected(() => {
|
||||||
|
updateButtons();
|
||||||
|
}));
|
||||||
|
return this.createButtonContainer([addButton, removeButton]);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createDropdown(ariaLabel: string, handler: (newValue: string) => Promise<void>, values: string[], value: string | undefined, enabled: boolean = true, width: number = DefaultInputWidth): azdata.DropDownComponent {
|
||||||
|
// Automatically add an empty item to the beginning of the list if the current value is not specified.
|
||||||
|
// This is needed when no meaningful default value can be provided.
|
||||||
|
// Create a new array so that the original array isn't modified.
|
||||||
|
const dropdownValues = [];
|
||||||
|
dropdownValues.push(...values);
|
||||||
|
if (!value) {
|
||||||
|
dropdownValues.unshift('');
|
||||||
|
}
|
||||||
|
const dropdown = this.modelView.modelBuilder.dropDown().withProps({
|
||||||
|
ariaLabel: ariaLabel,
|
||||||
|
values: dropdownValues,
|
||||||
|
value: value,
|
||||||
|
width: width,
|
||||||
|
enabled: enabled
|
||||||
|
}).component();
|
||||||
|
this.disposables.push(dropdown.onValueChanged(async () => {
|
||||||
|
await handler(<string>dropdown.value!);
|
||||||
|
this.onFormFieldChange();
|
||||||
|
await this.runValidation(false);
|
||||||
|
}));
|
||||||
|
return dropdown;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createButton(label: string, ariaLabel: string, handler: () => Promise<void>, enabled: boolean = true): azdata.ButtonComponent {
|
||||||
|
const button = this.modelView.modelBuilder.button().withProps({
|
||||||
|
label: label,
|
||||||
|
ariaLabel: ariaLabel,
|
||||||
|
enabled: enabled,
|
||||||
|
secondary: true,
|
||||||
|
CSSStyles: { 'min-width': '70px', 'margin-left': '5px' }
|
||||||
|
}).component();
|
||||||
|
this.disposables.push(button.onDidClick(async () => {
|
||||||
|
await handler();
|
||||||
|
}));
|
||||||
|
return button;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected createButtonContainer(items: azdata.ButtonComponent[], justifyContent: azdata.JustifyContentType = 'flex-end'): azdata.FlexContainer {
|
||||||
|
return this.modelView.modelBuilder.flexContainer().withProps({
|
||||||
|
CSSStyles: { 'margin': '5px 0' }
|
||||||
|
}).withLayout({
|
||||||
|
flexFlow: 'horizontal',
|
||||||
|
flexWrap: 'nowrap',
|
||||||
|
justifyContent: justifyContent
|
||||||
|
}).withItems(items, { flex: '0 0 auto' }).component();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected removeItem(container: azdata.DivContainer | azdata.FlexContainer, item: azdata.Component): void {
|
||||||
|
if (container.items.indexOf(item) !== -1) {
|
||||||
|
container.removeItem(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected addItem(container: azdata.DivContainer | azdata.FlexContainer, item: azdata.Component, index?: number): void {
|
||||||
|
if (container.items.indexOf(item) === -1) {
|
||||||
|
if (index === undefined) {
|
||||||
|
container.addItem(item);
|
||||||
|
} else {
|
||||||
|
container.insertItem(item, index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected onLoadingStatusChanged(isLoading: boolean): void {
|
||||||
|
if (this._loadingComponent) {
|
||||||
|
this._loadingComponent.loading = isLoading;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private createFormContainer(items: azdata.Component[]): azdata.DivContainer {
|
||||||
|
return this.modelView.modelBuilder.divContainer().withLayout({ width: 'calc(100% - 20px)', height: 'calc(100% - 20px)' }).withProps({
|
||||||
|
CSSStyles: { 'padding': '10px' }
|
||||||
|
}).withItems(items, { CSSStyles: { 'margin-block-end': '10px' } }).component();
|
||||||
|
}
|
||||||
|
}
|
||||||
136
extensions/mssql/src/objectManagement/ui/findObjectDialog.ts
Normal file
136
extensions/mssql/src/objectManagement/ui/findObjectDialog.ts
Normal file
@@ -0,0 +1,136 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
import * as azdata from 'azdata';
|
||||||
|
import * as mssql from 'mssql';
|
||||||
|
import { DefaultTableListItemEnabledStateGetter, DefaultMaxTableHeight, DialogBase, TableListItemComparer, TableListItemValueGetter } from './dialogBase';
|
||||||
|
import * as localizedConstants from '../localizedConstants';
|
||||||
|
import { getNodeTypeDisplayName } from '../utils';
|
||||||
|
import { getErrorMessage } from '../../utils';
|
||||||
|
|
||||||
|
export interface FindObjectDialogOptions {
|
||||||
|
objectTypes: mssql.ObjectManagement.NodeType[];
|
||||||
|
multiSelect: boolean;
|
||||||
|
contextId: string;
|
||||||
|
title: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface FindObjectDialogResult {
|
||||||
|
selectedObjects: mssql.ObjectManagement.SearchResultItem[];
|
||||||
|
}
|
||||||
|
|
||||||
|
const ObjectComparer: TableListItemComparer<mssql.ObjectManagement.SearchResultItem> =
|
||||||
|
(item1, item2) => {
|
||||||
|
return item1.name === item2.name && item1.type === item2.type;
|
||||||
|
};
|
||||||
|
|
||||||
|
const ObjectRowValueGetter: TableListItemValueGetter<mssql.ObjectManagement.SearchResultItem> =
|
||||||
|
(item) => {
|
||||||
|
return [item.name, getNodeTypeDisplayName(item.type, true)];
|
||||||
|
};
|
||||||
|
|
||||||
|
const ObjectsTableMaxHeight = 700;
|
||||||
|
|
||||||
|
export class FindObjectDialog extends DialogBase<FindObjectDialogResult> {
|
||||||
|
private objectTypesTable: azdata.TableComponent;
|
||||||
|
private findButton: azdata.ButtonComponent;
|
||||||
|
private objectsTable: azdata.TableComponent;
|
||||||
|
private objectsLoadingComponent: azdata.LoadingComponent;
|
||||||
|
private result: FindObjectDialogResult;
|
||||||
|
private selectedObjectTypes: string[] = [];
|
||||||
|
private allObjects: mssql.ObjectManagement.SearchResultItem[] = [];
|
||||||
|
|
||||||
|
constructor(private readonly objectManagementService: mssql.IObjectManagementService, private readonly options: FindObjectDialogOptions) {
|
||||||
|
super(options.title, 'FindObjectDialog');
|
||||||
|
this.dialogObject.okButton.label = localizedConstants.SelectText;
|
||||||
|
this.result = {
|
||||||
|
selectedObjects: []
|
||||||
|
};
|
||||||
|
this.selectedObjectTypes = [...options.objectTypes];
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async initialize(): Promise<void> {
|
||||||
|
this.dialogObject.okButton.enabled = false;
|
||||||
|
this.objectTypesTable = this.createTableList<string>(localizedConstants.ObjectTypeText,
|
||||||
|
[localizedConstants.ObjectTypeText],
|
||||||
|
this.options.objectTypes,
|
||||||
|
this.selectedObjectTypes,
|
||||||
|
DefaultMaxTableHeight,
|
||||||
|
DefaultTableListItemEnabledStateGetter, (item) => {
|
||||||
|
return [getNodeTypeDisplayName(item, true)];
|
||||||
|
});
|
||||||
|
this.findButton = this.createButton(localizedConstants.FindText, localizedConstants.FindText, async () => {
|
||||||
|
await this.onFindObjectButtonClick();
|
||||||
|
});
|
||||||
|
const buttonContainer = this.createButtonContainer([this.findButton]);
|
||||||
|
const objectTypeSection = this.createGroup(localizedConstants.ObjectTypeText, [this.objectTypesTable, buttonContainer]);
|
||||||
|
|
||||||
|
if (this.options.multiSelect) {
|
||||||
|
this.objectsTable = this.createTableList<mssql.ObjectManagement.SearchResultItem>(localizedConstants.ObjectsText,
|
||||||
|
[localizedConstants.NameText, localizedConstants.ObjectTypeText],
|
||||||
|
this.allObjects,
|
||||||
|
this.result.selectedObjects,
|
||||||
|
ObjectsTableMaxHeight,
|
||||||
|
DefaultTableListItemEnabledStateGetter,
|
||||||
|
ObjectRowValueGetter,
|
||||||
|
ObjectComparer);
|
||||||
|
} else {
|
||||||
|
this.objectsTable = this.createTable(localizedConstants.ObjectsText, [{
|
||||||
|
value: localizedConstants.NameText,
|
||||||
|
}, {
|
||||||
|
value: localizedConstants.ObjectTypeText
|
||||||
|
}], []);
|
||||||
|
this.disposables.push(this.objectsTable.onRowSelected(async () => {
|
||||||
|
if (this.objectsTable.selectedRows.length > 0) {
|
||||||
|
this.result.selectedObjects = [this.allObjects[this.objectsTable.selectedRows[0]]];
|
||||||
|
}
|
||||||
|
await this.onFormFieldChange();
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
this.objectsLoadingComponent = this.modelView.modelBuilder.loadingComponent().withItem(this.objectsTable).withProps({
|
||||||
|
loadingText: localizedConstants.LoadingObjectsText,
|
||||||
|
showText: true,
|
||||||
|
loading: false
|
||||||
|
}).component();
|
||||||
|
const objectsSection = this.createGroup(localizedConstants.ObjectsText, [this.objectsLoadingComponent]);
|
||||||
|
|
||||||
|
this.formContainer.addItems([objectTypeSection, objectsSection]);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override get dialogResult(): FindObjectDialogResult | undefined {
|
||||||
|
return this.result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async onFindObjectButtonClick(): Promise<void> {
|
||||||
|
this.dialogObject.okButton.enabled = false;
|
||||||
|
this.objectsLoadingComponent.loading = true;
|
||||||
|
this.findButton.enabled = false;
|
||||||
|
try {
|
||||||
|
const results = await this.objectManagementService.search(this.options.contextId, <mssql.ObjectManagement.NodeType[]>this.selectedObjectTypes);
|
||||||
|
this.allObjects.splice(0, this.allObjects.length, ...results);
|
||||||
|
let data;
|
||||||
|
if (this.options.multiSelect) {
|
||||||
|
data = this.getDataForTableList(this.allObjects, this.result.selectedObjects, DefaultTableListItemEnabledStateGetter, ObjectRowValueGetter, ObjectComparer);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
data = this.allObjects.map(item => ObjectRowValueGetter(item));
|
||||||
|
}
|
||||||
|
this.setTableData(this.objectsTable, data, ObjectsTableMaxHeight);
|
||||||
|
this.objectsLoadingComponent.loadingCompletedText = localizedConstants.LoadingObjectsCompletedText(results.length);
|
||||||
|
} catch (err) {
|
||||||
|
this.dialogObject.message = {
|
||||||
|
text: getErrorMessage(err),
|
||||||
|
level: azdata.window.MessageLevel.Error
|
||||||
|
};
|
||||||
|
}
|
||||||
|
this.findButton.enabled = true;
|
||||||
|
this.objectsLoadingComponent.loading = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async onFormFieldChange(): Promise<void> {
|
||||||
|
this.findButton.enabled = this.selectedObjectTypes.length > 0;
|
||||||
|
this.dialogObject.okButton.enabled = this.result.selectedObjects.length > 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -4,11 +4,12 @@
|
|||||||
*--------------------------------------------------------------------------------------------*/
|
*--------------------------------------------------------------------------------------------*/
|
||||||
import * as azdata from 'azdata';
|
import * as azdata from 'azdata';
|
||||||
import * as vscode from 'vscode';
|
import * as vscode from 'vscode';
|
||||||
import { DefaultInputWidth, ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
||||||
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
import * as localizedConstants from '../localizedConstants';
|
import * as localizedConstants from '../localizedConstants';
|
||||||
import { AlterLoginDocUrl, CreateLoginDocUrl, PublicServerRoleName } from '../constants';
|
import { AlterLoginDocUrl, CreateLoginDocUrl, PublicServerRoleName } from '../constants';
|
||||||
import { getAuthenticationTypeByDisplayName, getAuthenticationTypeDisplayName, isValidSQLPassword } from '../utils';
|
import { getAuthenticationTypeByDisplayName, getAuthenticationTypeDisplayName, isValidSQLPassword } from '../utils';
|
||||||
|
import { DefaultMaxTableHeight } from './dialogBase';
|
||||||
|
|
||||||
export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Login, ObjectManagement.LoginViewInfo> {
|
export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Login, ObjectManagement.LoginViewInfo> {
|
||||||
private generalSection: azdata.GroupContainer;
|
private generalSection: azdata.GroupContainer;
|
||||||
@@ -52,11 +53,8 @@ export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Log
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async validateInput(): Promise<string[]> {
|
protected override async validateInput(): Promise<string[]> {
|
||||||
const errors: string[] = [];
|
const errors = await super.validateInput();
|
||||||
if (!this.objectInfo.name) {
|
|
||||||
errors.push(localizedConstants.NameCannotBeEmptyError);
|
|
||||||
}
|
|
||||||
if (this.objectInfo.authenticationType === ObjectManagement.AuthenticationType.Sql) {
|
if (this.objectInfo.authenticationType === ObjectManagement.AuthenticationType.Sql) {
|
||||||
if (!this.objectInfo.password && !(this.viewInfo.supportAdvancedPasswordOptions && !this.objectInfo.enforcePasswordPolicy)) {
|
if (!this.objectInfo.password && !(this.viewInfo.supportAdvancedPasswordOptions && !this.objectInfo.enforcePasswordPolicy)) {
|
||||||
errors.push(localizedConstants.PasswordCannotBeEmptyError);
|
errors.push(localizedConstants.PasswordCannotBeEmptyError);
|
||||||
@@ -104,17 +102,9 @@ export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Log
|
|||||||
}
|
}
|
||||||
|
|
||||||
private initializeGeneralSection(): void {
|
private initializeGeneralSection(): void {
|
||||||
this.nameInput = this.modelView.modelBuilder.inputBox().withProps({
|
this.nameInput = this.createInputBox(localizedConstants.NameText, async (newValue) => {
|
||||||
ariaLabel: localizedConstants.NameText,
|
this.objectInfo.name = newValue;
|
||||||
enabled: this.options.isNewObject,
|
}, this.objectInfo.name, this.options.isNewObject);
|
||||||
value: this.objectInfo.name,
|
|
||||||
width: DefaultInputWidth
|
|
||||||
}).component();
|
|
||||||
this.disposables.push(this.nameInput.onTextChanged(async () => {
|
|
||||||
this.objectInfo.name = this.nameInput.value!;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
|
|
||||||
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
||||||
const authTypes = [];
|
const authTypes = [];
|
||||||
@@ -127,93 +117,72 @@ export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Log
|
|||||||
if (this.viewInfo.supportAADAuthentication) {
|
if (this.viewInfo.supportAADAuthentication) {
|
||||||
authTypes.push(localizedConstants.AADAuthenticationTypeDisplayText);
|
authTypes.push(localizedConstants.AADAuthenticationTypeDisplayText);
|
||||||
}
|
}
|
||||||
this.authTypeDropdown = this.createDropdown(localizedConstants.AuthTypeText, authTypes, getAuthenticationTypeDisplayName(this.objectInfo.authenticationType), this.options.isNewObject);
|
this.authTypeDropdown = this.createDropdown(localizedConstants.AuthTypeText, async (newValue) => {
|
||||||
this.disposables.push(this.authTypeDropdown.onValueChanged(async () => {
|
this.objectInfo.authenticationType = getAuthenticationTypeByDisplayName(newValue);
|
||||||
this.objectInfo.authenticationType = getAuthenticationTypeByDisplayName(<string>this.authTypeDropdown.value);
|
|
||||||
this.setViewByAuthenticationType();
|
this.setViewByAuthenticationType();
|
||||||
this.onObjectValueChange();
|
}, authTypes, getAuthenticationTypeDisplayName(this.objectInfo.authenticationType), this.options.isNewObject);
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
const authTypeContainer = this.createLabelInputContainer(localizedConstants.AuthTypeText, this.authTypeDropdown);
|
const authTypeContainer = this.createLabelInputContainer(localizedConstants.AuthTypeText, this.authTypeDropdown);
|
||||||
|
|
||||||
this.enabledCheckbox = this.createCheckbox(localizedConstants.EnabledText, this.objectInfo.isEnabled);
|
this.enabledCheckbox = this.createCheckbox(localizedConstants.EnabledText, async (checked) => {
|
||||||
this.disposables.push(this.enabledCheckbox.onChanged(() => {
|
this.objectInfo.isEnabled = checked;
|
||||||
this.objectInfo.isEnabled = this.enabledCheckbox.checked!;
|
}, this.objectInfo.isEnabled);
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [nameContainer, authTypeContainer, this.enabledCheckbox], false);
|
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [nameContainer, authTypeContainer, this.enabledCheckbox], false);
|
||||||
}
|
}
|
||||||
|
|
||||||
private initializeSqlAuthSection(): void {
|
private initializeSqlAuthSection(): void {
|
||||||
const items: azdata.Component[] = [];
|
const items: azdata.Component[] = [];
|
||||||
this.passwordInput = this.createPasswordInputBox(localizedConstants.PasswordText, this.objectInfo.password ?? '');
|
this.passwordInput = this.createPasswordInputBox(localizedConstants.PasswordText, async (newValue) => {
|
||||||
|
this.objectInfo.password = newValue;
|
||||||
|
}, this.objectInfo.password ?? '');
|
||||||
const passwordRow = this.createLabelInputContainer(localizedConstants.PasswordText, this.passwordInput);
|
const passwordRow = this.createLabelInputContainer(localizedConstants.PasswordText, this.passwordInput);
|
||||||
this.confirmPasswordInput = this.createPasswordInputBox(localizedConstants.ConfirmPasswordText, this.objectInfo.password ?? '');
|
this.confirmPasswordInput = this.createPasswordInputBox(localizedConstants.ConfirmPasswordText, async () => { }, this.objectInfo.password ?? '');
|
||||||
this.disposables.push(this.passwordInput.onTextChanged(async () => {
|
|
||||||
this.objectInfo.password = this.passwordInput.value;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
this.disposables.push(this.confirmPasswordInput.onTextChanged(async () => {
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
const confirmPasswordRow = this.createLabelInputContainer(localizedConstants.ConfirmPasswordText, this.confirmPasswordInput);
|
const confirmPasswordRow = this.createLabelInputContainer(localizedConstants.ConfirmPasswordText, this.confirmPasswordInput);
|
||||||
items.push(passwordRow, confirmPasswordRow);
|
items.push(passwordRow, confirmPasswordRow);
|
||||||
|
|
||||||
if (!this.options.isNewObject) {
|
if (!this.options.isNewObject) {
|
||||||
this.specifyOldPasswordCheckbox = this.createCheckbox(localizedConstants.SpecifyOldPasswordText);
|
this.specifyOldPasswordCheckbox = this.createCheckbox(localizedConstants.SpecifyOldPasswordText, async (checked) => {
|
||||||
this.oldPasswordInput = this.createPasswordInputBox(localizedConstants.OldPasswordText, '', false);
|
|
||||||
const oldPasswordRow = this.createLabelInputContainer(localizedConstants.OldPasswordText, this.oldPasswordInput);
|
|
||||||
this.disposables.push(this.specifyOldPasswordCheckbox.onChanged(async () => {
|
|
||||||
this.oldPasswordInput.enabled = this.specifyOldPasswordCheckbox.checked;
|
this.oldPasswordInput.enabled = this.specifyOldPasswordCheckbox.checked;
|
||||||
this.objectInfo.oldPassword = '';
|
this.objectInfo.oldPassword = '';
|
||||||
if (!this.specifyOldPasswordCheckbox.checked) {
|
if (!this.specifyOldPasswordCheckbox.checked) {
|
||||||
this.oldPasswordInput.value = '';
|
this.oldPasswordInput.value = '';
|
||||||
}
|
}
|
||||||
this.onObjectValueChange();
|
});
|
||||||
await this.runValidation(false);
|
this.oldPasswordInput = this.createPasswordInputBox(localizedConstants.OldPasswordText, async (newValue) => {
|
||||||
}));
|
this.objectInfo.oldPassword = newValue;
|
||||||
this.disposables.push(this.oldPasswordInput.onTextChanged(async () => {
|
}, '', false);
|
||||||
this.objectInfo.oldPassword = this.oldPasswordInput.value;
|
const oldPasswordRow = this.createLabelInputContainer(localizedConstants.OldPasswordText, this.oldPasswordInput);
|
||||||
this.onObjectValueChange();
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
items.push(this.specifyOldPasswordCheckbox, oldPasswordRow);
|
items.push(this.specifyOldPasswordCheckbox, oldPasswordRow);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.viewInfo.supportAdvancedPasswordOptions) {
|
if (this.viewInfo.supportAdvancedPasswordOptions) {
|
||||||
this.enforcePasswordPolicyCheckbox = this.createCheckbox(localizedConstants.EnforcePasswordPolicyText, this.objectInfo.enforcePasswordPolicy);
|
this.enforcePasswordPolicyCheckbox = this.createCheckbox(localizedConstants.EnforcePasswordPolicyText, async (checked) => {
|
||||||
this.enforcePasswordExpirationCheckbox = this.createCheckbox(localizedConstants.EnforcePasswordExpirationText, this.objectInfo.enforcePasswordPolicy);
|
const enforcePolicy = checked;
|
||||||
this.mustChangePasswordCheckbox = this.createCheckbox(localizedConstants.MustChangePasswordText, this.objectInfo.mustChangePassword);
|
|
||||||
this.disposables.push(this.enforcePasswordPolicyCheckbox.onChanged(async () => {
|
|
||||||
const enforcePolicy = this.enforcePasswordPolicyCheckbox.checked;
|
|
||||||
this.objectInfo.enforcePasswordPolicy = enforcePolicy;
|
this.objectInfo.enforcePasswordPolicy = enforcePolicy;
|
||||||
this.enforcePasswordExpirationCheckbox.enabled = enforcePolicy;
|
this.enforcePasswordExpirationCheckbox.enabled = enforcePolicy;
|
||||||
this.mustChangePasswordCheckbox.enabled = enforcePolicy;
|
this.mustChangePasswordCheckbox.enabled = enforcePolicy;
|
||||||
this.enforcePasswordExpirationCheckbox.checked = enforcePolicy;
|
this.enforcePasswordExpirationCheckbox.checked = enforcePolicy;
|
||||||
this.mustChangePasswordCheckbox.checked = enforcePolicy;
|
this.mustChangePasswordCheckbox.checked = enforcePolicy;
|
||||||
this.onObjectValueChange();
|
}, this.objectInfo.enforcePasswordPolicy);
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
this.enforcePasswordExpirationCheckbox = this.createCheckbox(localizedConstants.EnforcePasswordExpirationText, async (checked) => {
|
||||||
this.disposables.push(this.enforcePasswordExpirationCheckbox.onChanged(() => {
|
const enforceExpiration = checked;
|
||||||
const enforceExpiration = this.enforcePasswordExpirationCheckbox.checked;
|
|
||||||
this.objectInfo.enforcePasswordExpiration = enforceExpiration;
|
this.objectInfo.enforcePasswordExpiration = enforceExpiration;
|
||||||
this.mustChangePasswordCheckbox.enabled = enforceExpiration;
|
this.mustChangePasswordCheckbox.enabled = enforceExpiration;
|
||||||
this.mustChangePasswordCheckbox.checked = enforceExpiration;
|
this.mustChangePasswordCheckbox.checked = enforceExpiration;
|
||||||
this.onObjectValueChange();
|
}, this.objectInfo.enforcePasswordPolicy);
|
||||||
}));
|
|
||||||
this.disposables.push(this.mustChangePasswordCheckbox.onChanged(() => {
|
this.mustChangePasswordCheckbox = this.createCheckbox(localizedConstants.MustChangePasswordText, async (checked) => {
|
||||||
this.objectInfo.mustChangePassword = this.mustChangePasswordCheckbox.checked;
|
this.objectInfo.mustChangePassword = checked;
|
||||||
this.onObjectValueChange();
|
}, this.objectInfo.mustChangePassword);
|
||||||
}));
|
|
||||||
items.push(this.enforcePasswordPolicyCheckbox, this.enforcePasswordExpirationCheckbox, this.mustChangePasswordCheckbox);
|
items.push(this.enforcePasswordPolicyCheckbox, this.enforcePasswordExpirationCheckbox, this.mustChangePasswordCheckbox);
|
||||||
|
|
||||||
if (!this.options.isNewObject) {
|
if (!this.options.isNewObject) {
|
||||||
this.lockedOutCheckbox = this.createCheckbox(localizedConstants.LoginLockedOutText, this.objectInfo.isLockedOut, this.viewInfo.canEditLockedOutState);
|
this.lockedOutCheckbox = this.createCheckbox(localizedConstants.LoginLockedOutText, async (checked) => {
|
||||||
|
this.objectInfo.isLockedOut = checked;
|
||||||
|
}, this.objectInfo.isLockedOut, this.viewInfo.canEditLockedOutState);
|
||||||
items.push(this.lockedOutCheckbox);
|
items.push(this.lockedOutCheckbox);
|
||||||
this.disposables.push(this.lockedOutCheckbox.onChanged(() => {
|
|
||||||
this.objectInfo.isLockedOut = this.lockedOutCheckbox.checked!;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -223,25 +192,19 @@ export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Log
|
|||||||
private initializeAdvancedSection(): void {
|
private initializeAdvancedSection(): void {
|
||||||
const items: azdata.Component[] = [];
|
const items: azdata.Component[] = [];
|
||||||
if (this.viewInfo.supportAdvancedOptions) {
|
if (this.viewInfo.supportAdvancedOptions) {
|
||||||
this.defaultDatabaseDropdown = this.createDropdown(localizedConstants.DefaultDatabaseText, this.viewInfo.databases, this.objectInfo.defaultDatabase);
|
this.defaultDatabaseDropdown = this.createDropdown(localizedConstants.DefaultDatabaseText, async (newValue) => {
|
||||||
|
this.objectInfo.defaultDatabase = newValue;
|
||||||
|
}, this.viewInfo.databases, this.objectInfo.defaultDatabase);
|
||||||
const defaultDatabaseContainer = this.createLabelInputContainer(localizedConstants.DefaultDatabaseText, this.defaultDatabaseDropdown);
|
const defaultDatabaseContainer = this.createLabelInputContainer(localizedConstants.DefaultDatabaseText, this.defaultDatabaseDropdown);
|
||||||
this.disposables.push(this.defaultDatabaseDropdown.onValueChanged(() => {
|
|
||||||
this.objectInfo.defaultDatabase = <string>this.defaultDatabaseDropdown.value;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
|
|
||||||
this.defaultLanguageDropdown = this.createDropdown(localizedConstants.DefaultLanguageText, this.viewInfo.languages, this.objectInfo.defaultLanguage);
|
this.defaultLanguageDropdown = this.createDropdown(localizedConstants.DefaultLanguageText, async (newValue) => {
|
||||||
|
this.objectInfo.defaultLanguage = newValue;
|
||||||
|
}, this.viewInfo.languages, this.objectInfo.defaultLanguage);
|
||||||
const defaultLanguageContainer = this.createLabelInputContainer(localizedConstants.DefaultLanguageText, this.defaultLanguageDropdown);
|
const defaultLanguageContainer = this.createLabelInputContainer(localizedConstants.DefaultLanguageText, this.defaultLanguageDropdown);
|
||||||
this.disposables.push(this.defaultLanguageDropdown.onValueChanged(() => {
|
|
||||||
this.objectInfo.defaultLanguage = <string>this.defaultLanguageDropdown.value;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
|
|
||||||
this.connectPermissionCheckbox = this.createCheckbox(localizedConstants.PermissionToConnectText, this.objectInfo.connectPermission);
|
this.connectPermissionCheckbox = this.createCheckbox(localizedConstants.PermissionToConnectText, async (checked) => {
|
||||||
this.disposables.push(this.connectPermissionCheckbox.onChanged(() => {
|
this.objectInfo.connectPermission = checked;
|
||||||
this.objectInfo.connectPermission = this.connectPermissionCheckbox.checked!;
|
}, this.objectInfo.connectPermission);
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
items.push(defaultDatabaseContainer, defaultLanguageContainer, this.connectPermissionCheckbox);
|
items.push(defaultDatabaseContainer, defaultLanguageContainer, this.connectPermissionCheckbox);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -249,12 +212,14 @@ export class LoginDialog extends ObjectManagementDialogBase<ObjectManagement.Log
|
|||||||
}
|
}
|
||||||
|
|
||||||
private initializeServerRolesSection(): void {
|
private initializeServerRolesSection(): void {
|
||||||
const serverRolesData = this.viewInfo.serverRoles.map(name => {
|
this.serverRoleTable = this.createTableList(localizedConstants.ServerRoleSectionHeader,
|
||||||
const isRoleSelected = this.objectInfo.serverRoles.indexOf(name) !== -1;
|
[localizedConstants.ServerRoleTypeDisplayNameInTitle],
|
||||||
const isRoleSelectionEnabled = name !== PublicServerRoleName;
|
this.viewInfo.serverRoles,
|
||||||
return [{ enabled: isRoleSelectionEnabled, checked: isRoleSelected }, name];
|
this.objectInfo.serverRoles,
|
||||||
|
DefaultMaxTableHeight,
|
||||||
|
(item) => {
|
||||||
|
return item !== PublicServerRoleName
|
||||||
});
|
});
|
||||||
this.serverRoleTable = this.createTableList(localizedConstants.ServerRoleSectionHeader, this.viewInfo.serverRoles, this.objectInfo.serverRoles, serverRolesData);
|
|
||||||
this.serverRoleSection = this.createGroup(localizedConstants.ServerRoleSectionHeader, [this.serverRoleTable]);
|
this.serverRoleSection = this.createGroup(localizedConstants.ServerRoleSectionHeader, [this.serverRoleTable]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,36 +3,19 @@
|
|||||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
*--------------------------------------------------------------------------------------------*/
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
// TODO:
|
|
||||||
// 1. include server properties and other properties in the telemetry.
|
|
||||||
|
|
||||||
import * as azdata from 'azdata';
|
import * as azdata from 'azdata';
|
||||||
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
import * as vscode from 'vscode';
|
import * as vscode from 'vscode';
|
||||||
import { EOL } from 'os';
|
|
||||||
import { generateUuid } from 'vscode-languageclient/lib/utils/uuid';
|
import { generateUuid } from 'vscode-languageclient/lib/utils/uuid';
|
||||||
import { getErrorMessage } from '../../utils';
|
import * as localizedConstants from '../localizedConstants';
|
||||||
import { TelemetryActions, ObjectManagementViewName } from '../constants';
|
import { deepClone, getNodeTypeDisplayName, refreshNode, refreshParentNode } from '../utils';
|
||||||
import {
|
import { DialogBase } from './dialogBase';
|
||||||
CreateObjectOperationDisplayName, HelpText, LoadingDialogText,
|
import { ObjectManagementViewName, TelemetryActions } from '../constants';
|
||||||
NameText,
|
|
||||||
NewObjectDialogTitle, NoActionScriptedMessage, ObjectPropertiesDialogTitle, OkText, ScriptError, ScriptGeneratedText, ScriptText, SelectedText, UpdateObjectOperationDisplayName
|
|
||||||
} from '../localizedConstants';
|
|
||||||
import { deepClone, getNodeTypeDisplayName, refreshNode } from '../utils';
|
|
||||||
import { TelemetryReporter } from '../../telemetry';
|
import { TelemetryReporter } from '../../telemetry';
|
||||||
|
import { getErrorMessage } from '../../utils';
|
||||||
import { providerId } from '../../constants';
|
import { providerId } from '../../constants';
|
||||||
|
import { equals } from '../../util/objects';
|
||||||
|
|
||||||
export const DefaultLabelWidth = 150;
|
|
||||||
export const DefaultInputWidth = 300;
|
|
||||||
export const DefaultTableWidth = DefaultInputWidth + DefaultLabelWidth;
|
|
||||||
export const DefaultTableMaxHeight = 400;
|
|
||||||
export const DefaultTableMinRowCount = 2;
|
|
||||||
export const TableRowHeight = 25;
|
|
||||||
export const TableColumnHeaderHeight = 30;
|
|
||||||
|
|
||||||
export function getTableHeight(rowCount: number, minRowCount: number = DefaultTableMinRowCount, maxHeight: number = DefaultTableMaxHeight): number {
|
|
||||||
return Math.min(Math.max(rowCount, minRowCount) * TableRowHeight + TableColumnHeaderHeight, maxHeight);
|
|
||||||
}
|
|
||||||
|
|
||||||
function getDialogName(type: ObjectManagement.NodeType, isNewObject: boolean): string {
|
function getDialogName(type: ObjectManagement.NodeType, isNewObject: boolean): string {
|
||||||
return isNewObject ? `New${type}` : `${type}Properties`
|
return isNewObject ? `New${type}` : `${type}Properties`
|
||||||
@@ -50,116 +33,70 @@ export interface ObjectManagementDialogOptions {
|
|||||||
objectName?: string;
|
objectName?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectManagement.SqlObject, ViewInfoType extends ObjectManagement.ObjectViewInfo<ObjectInfoType>> {
|
export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectManagement.SqlObject, ViewInfoType extends ObjectManagement.ObjectViewInfo<ObjectInfoType>> extends DialogBase<void> {
|
||||||
protected readonly disposables: vscode.Disposable[] = [];
|
|
||||||
protected readonly dialogObject: azdata.window.Dialog;
|
|
||||||
private _contextId: string;
|
private _contextId: string;
|
||||||
private _viewInfo: ViewInfoType;
|
private _viewInfo: ViewInfoType;
|
||||||
private _originalObjectInfo: ObjectInfoType;
|
private _originalObjectInfo: ObjectInfoType;
|
||||||
private _modelView: azdata.ModelView;
|
|
||||||
private _loadingComponent: azdata.LoadingComponent;
|
|
||||||
private _formContainer: azdata.DivContainer;
|
|
||||||
private _helpButton: azdata.window.Button;
|
private _helpButton: azdata.window.Button;
|
||||||
private _scriptButton: azdata.window.Button;
|
private _scriptButton: azdata.window.Button;
|
||||||
|
|
||||||
constructor(protected readonly objectManagementService: IObjectManagementService, protected readonly options: ObjectManagementDialogOptions) {
|
constructor(protected readonly objectManagementService: IObjectManagementService, protected readonly options: ObjectManagementDialogOptions) {
|
||||||
this.options.width = this.options.width || 'narrow';
|
super(options.isNewObject ? localizedConstants.NewObjectDialogTitle(getNodeTypeDisplayName(options.objectType, true)) :
|
||||||
const objectTypeDisplayName = getNodeTypeDisplayName(options.objectType, true);
|
localizedConstants.ObjectPropertiesDialogTitle(getNodeTypeDisplayName(options.objectType, true), options.objectName),
|
||||||
const dialogTitle = options.isNewObject ? NewObjectDialogTitle(objectTypeDisplayName) : ObjectPropertiesDialogTitle(objectTypeDisplayName, options.objectName);
|
getDialogName(options.objectType, options.isNewObject),
|
||||||
this.dialogObject = azdata.window.createModelViewDialog(dialogTitle, getDialogName(options.objectType, options.isNewObject), options.width);
|
options.width || 'narrow', 'flyout'
|
||||||
this.dialogObject.okButton.label = OkText;
|
);
|
||||||
this.disposables.push(this.dialogObject.onClosed(async (reason: azdata.window.CloseReason) => { await this.dispose(reason); }));
|
this._helpButton = azdata.window.createButton(localizedConstants.HelpText, 'left');
|
||||||
this._helpButton = azdata.window.createButton(HelpText, 'left');
|
|
||||||
this.disposables.push(this._helpButton.onClick(async () => {
|
this.disposables.push(this._helpButton.onClick(async () => {
|
||||||
await vscode.commands.executeCommand('vscode.open', vscode.Uri.parse(this.docUrl));
|
await vscode.commands.executeCommand('vscode.open', vscode.Uri.parse(this.docUrl));
|
||||||
}));
|
}));
|
||||||
this._scriptButton = azdata.window.createButton(ScriptText, 'left');
|
this._scriptButton = azdata.window.createButton(localizedConstants.ScriptText, 'left');
|
||||||
this.disposables.push(this._scriptButton.onClick(async () => { await this.onScriptButtonClick(); }));
|
this.disposables.push(this._scriptButton.onClick(async () => { await this.onScriptButtonClick(); }));
|
||||||
this.dialogObject.customButtons = [this._helpButton, this._scriptButton];
|
this.dialogObject.customButtons = [this._helpButton, this._scriptButton];
|
||||||
this.updateLoadingStatus(true);
|
|
||||||
this._contextId = generateUuid();
|
this._contextId = generateUuid();
|
||||||
this.dialogObject.registerCloseValidator(async (): Promise<boolean> => {
|
|
||||||
const confirmed = await this.onConfirmation();
|
|
||||||
if (!confirmed) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
return await this.runValidation();
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
protected abstract initializeUI(): Promise<void>;
|
protected abstract initializeUI(): Promise<void>;
|
||||||
protected abstract validateInput(): Promise<string[]>;
|
|
||||||
protected abstract get docUrl(): string;
|
protected abstract get docUrl(): string;
|
||||||
|
|
||||||
protected postInitializeData(): void {
|
protected postInitializeData(): void { }
|
||||||
|
|
||||||
}
|
protected override onFormFieldChange(): void {
|
||||||
protected onObjectValueChange(): void {
|
this._scriptButton.enabled = this.isDirty;
|
||||||
this.dialogObject.okButton.enabled = this.isDirty;
|
this.dialogObject.okButton.enabled = this.isDirty;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async onConfirmation(): Promise<boolean> {
|
protected override async validateInput(): Promise<string[]> {
|
||||||
return true;
|
const errors: string[] = [];
|
||||||
|
if (!this.objectInfo.name) {
|
||||||
|
errors.push(localizedConstants.NameCannotBeEmptyError);
|
||||||
|
}
|
||||||
|
return errors;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected get viewInfo(): ViewInfoType {
|
protected override async initialize(): Promise<void> {
|
||||||
return this._viewInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected get objectInfo(): ObjectInfoType {
|
|
||||||
return this._viewInfo.objectInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected get originalObjectInfo(): ObjectInfoType {
|
|
||||||
return this._originalObjectInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected get formContainer(): azdata.DivContainer {
|
|
||||||
return this._formContainer;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected get modelView(): azdata.ModelView {
|
|
||||||
return this._modelView;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async open(): Promise<void> {
|
|
||||||
try {
|
|
||||||
const initializeViewPromise = new Promise<void>((async resolve => {
|
|
||||||
await this.dialogObject.registerContent(async view => {
|
|
||||||
this._modelView = view;
|
|
||||||
resolve();
|
|
||||||
this._formContainer = this.createFormContainer([]);
|
|
||||||
this._loadingComponent = view.modelBuilder.loadingComponent().withItem(this._formContainer).withProps({
|
|
||||||
loading: true,
|
|
||||||
loadingText: LoadingDialogText,
|
|
||||||
showText: true,
|
|
||||||
CSSStyles: {
|
|
||||||
width: "100%",
|
|
||||||
height: "100%"
|
|
||||||
}
|
|
||||||
}).component();
|
|
||||||
await view.initializeModel(this._loadingComponent);
|
|
||||||
});
|
|
||||||
}));
|
|
||||||
azdata.window.openDialog(this.dialogObject);
|
|
||||||
await this.initializeData();
|
await this.initializeData();
|
||||||
await initializeViewPromise;
|
|
||||||
await this.initializeUI();
|
await this.initializeUI();
|
||||||
this._originalObjectInfo = deepClone(this.objectInfo);
|
|
||||||
const typeDisplayName = getNodeTypeDisplayName(this.options.objectType);
|
const typeDisplayName = getNodeTypeDisplayName(this.options.objectType);
|
||||||
this.dialogObject.registerOperation({
|
this.dialogObject.registerOperation({
|
||||||
displayName: this.options.isNewObject ? CreateObjectOperationDisplayName(typeDisplayName)
|
displayName: this.options.isNewObject ? localizedConstants.CreateObjectOperationDisplayName(typeDisplayName)
|
||||||
: UpdateObjectOperationDisplayName(typeDisplayName, this.options.objectName),
|
: localizedConstants.UpdateObjectOperationDisplayName(typeDisplayName, this.options.objectName),
|
||||||
description: '',
|
description: '',
|
||||||
isCancelable: false,
|
isCancelable: false,
|
||||||
operation: async (operation: azdata.BackgroundOperation): Promise<void> => {
|
operation: async (operation: azdata.BackgroundOperation): Promise<void> => {
|
||||||
const actionName = this.options.isNewObject ? TelemetryActions.CreateObject : TelemetryActions.UpdateObject;
|
const actionName = this.options.isNewObject ? TelemetryActions.CreateObject : TelemetryActions.UpdateObject;
|
||||||
try {
|
try {
|
||||||
if (JSON.stringify(this.objectInfo) !== JSON.stringify(this._originalObjectInfo)) {
|
if (this.isDirty) {
|
||||||
const startTime = Date.now();
|
const startTime = Date.now();
|
||||||
await this.objectManagementService.save(this._contextId, this.objectInfo);
|
await this.objectManagementService.save(this._contextId, this.objectInfo);
|
||||||
if (this.options.isNewObject && this.options.objectExplorerContext) {
|
if (this.options.objectExplorerContext) {
|
||||||
|
if (this.options.isNewObject) {
|
||||||
await refreshNode(this.options.objectExplorerContext);
|
await refreshNode(this.options.objectExplorerContext);
|
||||||
|
} else {
|
||||||
|
// For edit mode, the node context is the object itself, we need to refresh the parent node to reflect the changes.
|
||||||
|
await refreshParentNode(this.options.objectExplorerContext);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
TelemetryReporter.sendTelemetryEvent(actionName, {
|
TelemetryReporter.sendTelemetryEvent(actionName, {
|
||||||
@@ -180,19 +117,26 @@ export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectMa
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
this.updateLoadingStatus(false);
|
|
||||||
} catch (err) {
|
|
||||||
const actionName = this.options.isNewObject ? TelemetryActions.OpenNewObjectDialog : TelemetryActions.OpenPropertiesDialog;
|
|
||||||
TelemetryReporter.createErrorEvent2(ObjectManagementViewName, actionName, err).withAdditionalProperties({
|
|
||||||
objectType: this.options.objectType
|
|
||||||
}).send();
|
|
||||||
void vscode.window.showErrorMessage(getErrorMessage(err));
|
|
||||||
azdata.window.closeDialog(this.dialogObject);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async dispose(reason: azdata.window.CloseReason): Promise<void> {
|
protected get viewInfo(): ViewInfoType {
|
||||||
this.disposables.forEach(disposable => disposable.dispose());
|
return this._viewInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected get objectInfo(): ObjectInfoType {
|
||||||
|
return this._viewInfo.objectInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected get originalObjectInfo(): ObjectInfoType {
|
||||||
|
return this._originalObjectInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected get contextId(): string {
|
||||||
|
return this._contextId;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override async dispose(reason: azdata.window.CloseReason): Promise<void> {
|
||||||
|
await super.dispose(reason);
|
||||||
if (reason !== 'ok') {
|
if (reason !== 'ok') {
|
||||||
await this.disposeView();
|
await this.disposeView();
|
||||||
}
|
}
|
||||||
@@ -206,140 +150,17 @@ export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectMa
|
|||||||
const viewInfo = await this.objectManagementService.initializeView(this._contextId, this.options.objectType, this.options.connectionUri, this.options.database, this.options.isNewObject, this.options.parentUrn, this.options.objectUrn);
|
const viewInfo = await this.objectManagementService.initializeView(this._contextId, this.options.objectType, this.options.connectionUri, this.options.database, this.options.isNewObject, this.options.parentUrn, this.options.objectUrn);
|
||||||
this._viewInfo = viewInfo as ViewInfoType;
|
this._viewInfo = viewInfo as ViewInfoType;
|
||||||
this.postInitializeData();
|
this.postInitializeData();
|
||||||
|
this._originalObjectInfo = deepClone(this.objectInfo);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async runValidation(showErrorMessage: boolean = true): Promise<boolean> {
|
protected override onLoadingStatusChanged(isLoading: boolean): void {
|
||||||
const errors = await this.validateInput();
|
super.onLoadingStatusChanged(isLoading);
|
||||||
if (errors.length > 0 && (this.dialogObject.message?.text || showErrorMessage)) {
|
|
||||||
this.dialogObject.message = {
|
|
||||||
text: errors.join(EOL),
|
|
||||||
level: azdata.window.MessageLevel.Error
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
this.dialogObject.message = undefined;
|
|
||||||
}
|
|
||||||
return errors.length === 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createLabelInputContainer(label: string, input: azdata.InputBoxComponent | azdata.DropDownComponent): azdata.FlexContainer {
|
|
||||||
const labelComponent = this.modelView.modelBuilder.text().withProps({ width: DefaultLabelWidth, value: label, requiredIndicator: input.required }).component();
|
|
||||||
const row = this.modelView.modelBuilder.flexContainer().withLayout({ flexFlow: 'horizontal', flexWrap: 'nowrap', alignItems: 'center' }).withItems([labelComponent, input]).component();
|
|
||||||
return row;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createCheckbox(label: string, checked: boolean = false, enabled: boolean = true): azdata.CheckBoxComponent {
|
|
||||||
return this.modelView.modelBuilder.checkBox().withProps({
|
|
||||||
label: label,
|
|
||||||
checked: checked,
|
|
||||||
enabled: enabled
|
|
||||||
}).component();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createPasswordInputBox(ariaLabel: string, value: string = '', enabled: boolean = true, width: number = DefaultInputWidth): azdata.InputBoxComponent {
|
|
||||||
return this.createInputBox(ariaLabel, value, enabled, 'password', width);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createInputBox(ariaLabel: string, value: string = '', enabled: boolean = true, type: azdata.InputBoxInputType = 'text', width: number = DefaultInputWidth): azdata.InputBoxComponent {
|
|
||||||
return this.modelView.modelBuilder.inputBox().withProps({ inputType: type, enabled: enabled, ariaLabel: ariaLabel, value: value, width: width }).component();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createGroup(header: string, items: azdata.Component[], collapsible: boolean = true, collapsed: boolean = false): azdata.GroupContainer {
|
|
||||||
return this.modelView.modelBuilder.groupContainer().withLayout({
|
|
||||||
header: header,
|
|
||||||
collapsible: collapsible,
|
|
||||||
collapsed: collapsed
|
|
||||||
}).withItems(items).component();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createFormContainer(items: azdata.Component[]): azdata.DivContainer {
|
|
||||||
return this.modelView.modelBuilder.divContainer().withLayout({ width: 'calc(100% - 20px)', height: 'calc(100% - 20px)' }).withProps({
|
|
||||||
CSSStyles: { 'padding': '10px' }
|
|
||||||
}).withItems(items, { CSSStyles: { 'margin-block-end': '10px' } }).component();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createTableList(ariaLabel: string, listValues: string[], selectedValues: string[], data?: any[][]): azdata.TableComponent {
|
|
||||||
let tableData = data;
|
|
||||||
if (tableData === undefined) {
|
|
||||||
tableData = listValues.map(name => {
|
|
||||||
const isSelected = selectedValues.indexOf(name) !== -1;
|
|
||||||
return [isSelected, name];
|
|
||||||
});
|
|
||||||
}
|
|
||||||
const table = this.modelView.modelBuilder.table().withProps(
|
|
||||||
{
|
|
||||||
ariaLabel: ariaLabel,
|
|
||||||
data: tableData,
|
|
||||||
columns: [
|
|
||||||
{
|
|
||||||
value: SelectedText,
|
|
||||||
type: azdata.ColumnType.checkBox,
|
|
||||||
options: { actionOnCheckbox: azdata.ActionOnCellCheckboxCheck.customAction }
|
|
||||||
}, {
|
|
||||||
value: NameText,
|
|
||||||
}
|
|
||||||
],
|
|
||||||
width: DefaultTableWidth,
|
|
||||||
height: getTableHeight(tableData.length)
|
|
||||||
}
|
|
||||||
).component();
|
|
||||||
this.disposables.push(table.onCellAction!((arg: azdata.ICheckboxCellActionEventArgs) => {
|
|
||||||
const name = listValues[arg.row];
|
|
||||||
const idx = selectedValues.indexOf(name);
|
|
||||||
if (arg.checked && idx === -1) {
|
|
||||||
selectedValues.push(name);
|
|
||||||
} else if (!arg.checked && idx !== -1) {
|
|
||||||
selectedValues.splice(idx, 1)
|
|
||||||
}
|
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
return table;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected createDropdown(ariaLabel: string, values: string[], value: string | undefined, enabled: boolean = true, width: number = DefaultInputWidth): azdata.DropDownComponent {
|
|
||||||
// Automatically add an empty item to the beginning of the list if the current value is not specified.
|
|
||||||
// This is needed when no meaningful default value can be provided.
|
|
||||||
// Create a new array so that the original array isn't modified.
|
|
||||||
const dropdownValues = [];
|
|
||||||
dropdownValues.push(...values);
|
|
||||||
if (!value) {
|
|
||||||
dropdownValues.unshift('');
|
|
||||||
}
|
|
||||||
return this.modelView.modelBuilder.dropDown().withProps({
|
|
||||||
ariaLabel: ariaLabel,
|
|
||||||
values: dropdownValues,
|
|
||||||
value: value,
|
|
||||||
width: width,
|
|
||||||
enabled: enabled
|
|
||||||
}).component();
|
|
||||||
}
|
|
||||||
|
|
||||||
protected removeItem(container: azdata.DivContainer | azdata.FlexContainer, item: azdata.Component): void {
|
|
||||||
if (container.items.indexOf(item) !== -1) {
|
|
||||||
container.removeItem(item);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected addItem(container: azdata.DivContainer | azdata.FlexContainer, item: azdata.Component, index?: number): void {
|
|
||||||
if (container.items.indexOf(item) === -1) {
|
|
||||||
if (index === undefined) {
|
|
||||||
container.addItem(item);
|
|
||||||
} else {
|
|
||||||
container.insertItem(item, index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private updateLoadingStatus(isLoading: boolean): void {
|
|
||||||
this._scriptButton.enabled = !isLoading;
|
|
||||||
this._helpButton.enabled = !isLoading;
|
this._helpButton.enabled = !isLoading;
|
||||||
this.dialogObject.okButton.enabled = isLoading ? false : this.isDirty;
|
this.dialogObject.okButton.enabled = this._scriptButton.enabled = isLoading ? false : this.isDirty;
|
||||||
if (this._loadingComponent) {
|
|
||||||
this._loadingComponent.loading = isLoading;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async onScriptButtonClick(): Promise<void> {
|
private async onScriptButtonClick(): Promise<void> {
|
||||||
this.updateLoadingStatus(true);
|
this.onLoadingStatusChanged(true);
|
||||||
try {
|
try {
|
||||||
const isValid = await this.runValidation();
|
const isValid = await this.runValidation();
|
||||||
if (!isValid) {
|
if (!isValid) {
|
||||||
@@ -348,10 +169,10 @@ export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectMa
|
|||||||
let message: string;
|
let message: string;
|
||||||
const script = await this.objectManagementService.script(this._contextId, this.objectInfo);
|
const script = await this.objectManagementService.script(this._contextId, this.objectInfo);
|
||||||
if (script) {
|
if (script) {
|
||||||
message = ScriptGeneratedText;
|
message = localizedConstants.ScriptGeneratedText;
|
||||||
await azdata.queryeditor.openQueryDocument({ content: script }, providerId);
|
await azdata.queryeditor.openQueryDocument({ content: script }, providerId);
|
||||||
} else {
|
} else {
|
||||||
message = NoActionScriptedMessage;
|
message = localizedConstants.NoActionScriptedMessage;
|
||||||
}
|
}
|
||||||
this.dialogObject.message = {
|
this.dialogObject.message = {
|
||||||
text: message,
|
text: message,
|
||||||
@@ -359,15 +180,15 @@ export abstract class ObjectManagementDialogBase<ObjectInfoType extends ObjectMa
|
|||||||
};
|
};
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.dialogObject.message = {
|
this.dialogObject.message = {
|
||||||
text: ScriptError(getErrorMessage(err)),
|
text: localizedConstants.ScriptError(getErrorMessage(err)),
|
||||||
level: azdata.window.MessageLevel.Error
|
level: azdata.window.MessageLevel.Error
|
||||||
};
|
};
|
||||||
} finally {
|
} finally {
|
||||||
this.updateLoadingStatus(false);
|
this.onLoadingStatusChanged(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private get isDirty(): boolean {
|
private get isDirty(): boolean {
|
||||||
return JSON.stringify(this.objectInfo) !== JSON.stringify(this._originalObjectInfo);
|
return !equals(this.objectInfo, this._originalObjectInfo, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
128
extensions/mssql/src/objectManagement/ui/serverRoleDialog.ts
Normal file
128
extensions/mssql/src/objectManagement/ui/serverRoleDialog.ts
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
import * as azdata from 'azdata';
|
||||||
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
||||||
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
|
import * as localizedConstants from '../localizedConstants';
|
||||||
|
import { AlterServerRoleDocUrl, CreateServerRoleDocUrl } from '../constants';
|
||||||
|
import { FindObjectDialog } from './findObjectDialog';
|
||||||
|
|
||||||
|
export class ServerRoleDialog extends ObjectManagementDialogBase<ObjectManagement.ServerRoleInfo, ObjectManagement.ServerRoleViewInfo> {
|
||||||
|
// Sections
|
||||||
|
private generalSection: azdata.GroupContainer;
|
||||||
|
private membershipSection: azdata.GroupContainer;
|
||||||
|
private memberSection: azdata.GroupContainer;
|
||||||
|
|
||||||
|
// General section content
|
||||||
|
private nameInput: azdata.InputBoxComponent;
|
||||||
|
private ownerInput: azdata.InputBoxComponent;
|
||||||
|
|
||||||
|
// Member section content
|
||||||
|
private memberTable: azdata.TableComponent;
|
||||||
|
|
||||||
|
// Membership section content
|
||||||
|
private membershipTable: azdata.TableComponent;
|
||||||
|
|
||||||
|
|
||||||
|
constructor(objectManagementService: IObjectManagementService, options: ObjectManagementDialogOptions) {
|
||||||
|
super(objectManagementService, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override get docUrl(): string {
|
||||||
|
return this.options.isNewObject ? CreateServerRoleDocUrl : AlterServerRoleDocUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected async initializeUI(): Promise<void> {
|
||||||
|
this.initializeGeneralSection();
|
||||||
|
this.initializeMemberSection();
|
||||||
|
const sections: azdata.Component[] = [this.generalSection, this.memberSection];
|
||||||
|
if (!this.viewInfo.isFixedRole) {
|
||||||
|
this.initializeMembershipSection();
|
||||||
|
sections.push(this.membershipSection);
|
||||||
|
}
|
||||||
|
this.formContainer.addItems(sections);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeGeneralSection(): void {
|
||||||
|
this.nameInput = this.createInputBox(localizedConstants.NameText, async (newValue) => {
|
||||||
|
this.objectInfo.name = newValue;
|
||||||
|
}, this.objectInfo.name, this.options.isNewObject);
|
||||||
|
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
||||||
|
|
||||||
|
this.ownerInput = this.createInputBox(localizedConstants.OwnerText, async (newValue) => {
|
||||||
|
this.objectInfo.owner = newValue;
|
||||||
|
}, this.objectInfo.owner, !this.viewInfo.isFixedRole, 'text', 210);
|
||||||
|
const browseOwnerButton = this.createButton(localizedConstants.BrowseText, localizedConstants.BrowseOwnerButtonAriaLabel, async () => {
|
||||||
|
const dialog = new FindObjectDialog(this.objectManagementService, {
|
||||||
|
objectTypes: [ObjectManagement.NodeType.ServerLevelLogin, ObjectManagement.NodeType.ServerLevelServerRole],
|
||||||
|
multiSelect: false,
|
||||||
|
contextId: this.contextId,
|
||||||
|
title: localizedConstants.SelectServerRoleOwnerDialogTitle
|
||||||
|
});
|
||||||
|
await dialog.open();
|
||||||
|
const result = await dialog.waitForClose();
|
||||||
|
if (result.selectedObjects.length > 0) {
|
||||||
|
this.ownerInput.value = result.selectedObjects[0].name;
|
||||||
|
}
|
||||||
|
}, !this.viewInfo.isFixedRole);
|
||||||
|
const ownerContainer = this.createLabelInputContainer(localizedConstants.OwnerText, this.ownerInput);
|
||||||
|
ownerContainer.addItems([browseOwnerButton], { flex: '0 0 auto' });
|
||||||
|
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [
|
||||||
|
nameContainer,
|
||||||
|
ownerContainer
|
||||||
|
], false);
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeMemberSection(): void {
|
||||||
|
this.memberTable = this.createTable(localizedConstants.MemberSectionHeader, [
|
||||||
|
{
|
||||||
|
type: azdata.ColumnType.text,
|
||||||
|
value: localizedConstants.NameText
|
||||||
|
}
|
||||||
|
], this.objectInfo.members.map(m => [m]));
|
||||||
|
const buttonContainer = this.addButtonsForTable(this.memberTable, localizedConstants.AddMemberAriaLabel, localizedConstants.RemoveMemberAriaLabel,
|
||||||
|
async () => {
|
||||||
|
const dialog = new FindObjectDialog(this.objectManagementService, {
|
||||||
|
objectTypes: [ObjectManagement.NodeType.ServerLevelLogin, ObjectManagement.NodeType.ServerLevelServerRole],
|
||||||
|
multiSelect: true,
|
||||||
|
contextId: this.contextId,
|
||||||
|
title: localizedConstants.SelectServerRoleMemberDialogTitle
|
||||||
|
});
|
||||||
|
await dialog.open();
|
||||||
|
const result = await dialog.waitForClose();
|
||||||
|
this.addMembers(result.selectedObjects.map(r => r.name));
|
||||||
|
},
|
||||||
|
async () => {
|
||||||
|
if (this.memberTable.selectedRows.length === 1) {
|
||||||
|
this.removeMember(this.memberTable.selectedRows[0]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.memberSection = this.createGroup(localizedConstants.MemberSectionHeader, [this.memberTable, buttonContainer]);
|
||||||
|
}
|
||||||
|
|
||||||
|
private addMembers(names: string[]): void {
|
||||||
|
names.forEach(n => {
|
||||||
|
if (this.objectInfo.members.indexOf(n) === -1) {
|
||||||
|
this.objectInfo.members.push(n);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.updateMembersTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
private removeMember(idx: number): void {
|
||||||
|
this.objectInfo.members.splice(idx, 1);
|
||||||
|
this.updateMembersTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
private updateMembersTable(): void {
|
||||||
|
this.setTableData(this.memberTable, this.objectInfo.members.map(m => [m]));
|
||||||
|
this.onFormFieldChange();
|
||||||
|
}
|
||||||
|
|
||||||
|
private initializeMembershipSection(): void {
|
||||||
|
this.membershipTable = this.createTableList<string>(localizedConstants.MembershipSectionHeader, [localizedConstants.ServerRoleTypeDisplayNameInTitle], this.viewInfo.serverRoles, this.objectInfo.memberships);
|
||||||
|
this.membershipSection = this.createGroup(localizedConstants.MembershipSectionHeader, [this.membershipTable]);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,11 +3,12 @@
|
|||||||
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
*--------------------------------------------------------------------------------------------*/
|
*--------------------------------------------------------------------------------------------*/
|
||||||
import * as azdata from 'azdata';
|
import * as azdata from 'azdata';
|
||||||
import { DefaultInputWidth, ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
import { ObjectManagementDialogBase, ObjectManagementDialogOptions } from './objectManagementDialogBase';
|
||||||
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
import { IObjectManagementService, ObjectManagement } from 'mssql';
|
||||||
import * as localizedConstants from '../localizedConstants';
|
import * as localizedConstants from '../localizedConstants';
|
||||||
import { AlterUserDocUrl, CreateUserDocUrl } from '../constants';
|
import { AlterUserDocUrl, CreateUserDocUrl } from '../constants';
|
||||||
import { getAuthenticationTypeByDisplayName, getAuthenticationTypeDisplayName, getUserTypeByDisplayName, getUserTypeDisplayName, isValidSQLPassword } from '../utils';
|
import { getAuthenticationTypeByDisplayName, getAuthenticationTypeDisplayName, getUserTypeByDisplayName, getUserTypeDisplayName, isValidSQLPassword } from '../utils';
|
||||||
|
import { DefaultMaxTableHeight } from './dialogBase';
|
||||||
|
|
||||||
export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User, ObjectManagement.UserViewInfo> {
|
export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User, ObjectManagement.UserViewInfo> {
|
||||||
private generalSection: azdata.GroupContainer;
|
private generalSection: azdata.GroupContainer;
|
||||||
@@ -43,11 +44,8 @@ export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User
|
|||||||
this.objectInfo.password = this.objectInfo.password ?? '';
|
this.objectInfo.password = this.objectInfo.password ?? '';
|
||||||
}
|
}
|
||||||
|
|
||||||
protected async validateInput(): Promise<string[]> {
|
protected override async validateInput(): Promise<string[]> {
|
||||||
const errors: string[] = [];
|
const errors = await super.validateInput();
|
||||||
if (!this.objectInfo.name) {
|
|
||||||
errors.push(localizedConstants.NameCannotBeEmptyError);
|
|
||||||
}
|
|
||||||
if (this.objectInfo.type === ObjectManagement.UserType.Contained && this.objectInfo.authenticationType === ObjectManagement.AuthenticationType.Sql) {
|
if (this.objectInfo.type === ObjectManagement.UserType.Contained && this.objectInfo.authenticationType === ObjectManagement.AuthenticationType.Sql) {
|
||||||
if (!this.objectInfo.password) {
|
if (!this.objectInfo.password) {
|
||||||
errors.push(localizedConstants.PasswordCannotBeEmptyError);
|
errors.push(localizedConstants.PasswordCannotBeEmptyError);
|
||||||
@@ -79,41 +77,27 @@ export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User
|
|||||||
}
|
}
|
||||||
|
|
||||||
private initializeGeneralSection(): void {
|
private initializeGeneralSection(): void {
|
||||||
this.nameInput = this.modelView.modelBuilder.inputBox().withProps({
|
this.nameInput = this.createInputBox(localizedConstants.NameText, async (newValue) => {
|
||||||
ariaLabel: localizedConstants.NameText,
|
this.objectInfo.name = newValue;
|
||||||
enabled: this.options.isNewObject,
|
}, this.objectInfo.name, this.options.isNewObject);
|
||||||
value: this.objectInfo.name,
|
|
||||||
width: DefaultInputWidth
|
|
||||||
}).component();
|
|
||||||
this.disposables.push(this.nameInput.onTextChanged(async () => {
|
|
||||||
this.objectInfo.name = this.nameInput.value!;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
const nameContainer = this.createLabelInputContainer(localizedConstants.NameText, this.nameInput);
|
||||||
this.defaultSchemaDropdown = this.createDropdown(localizedConstants.DefaultSchemaText, this.viewInfo.schemas, this.objectInfo.defaultSchema!);
|
|
||||||
|
this.defaultSchemaDropdown = this.createDropdown(localizedConstants.DefaultSchemaText, async (newValue) => {
|
||||||
|
this.objectInfo.defaultSchema = newValue;
|
||||||
|
}, this.viewInfo.schemas, this.objectInfo.defaultSchema!);
|
||||||
this.defaultSchemaContainer = this.createLabelInputContainer(localizedConstants.DefaultSchemaText, this.defaultSchemaDropdown);
|
this.defaultSchemaContainer = this.createLabelInputContainer(localizedConstants.DefaultSchemaText, this.defaultSchemaDropdown);
|
||||||
this.disposables.push(this.defaultSchemaDropdown.onValueChanged(() => {
|
|
||||||
this.objectInfo.defaultSchema = <string>this.defaultSchemaDropdown.value;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
|
|
||||||
// only supporting user with login for initial preview
|
// only supporting user with login for initial preview
|
||||||
const userTypes = [localizedConstants.UserWithLoginText, localizedConstants.UserWithWindowsGroupLoginText, localizedConstants.ContainedUserText, localizedConstants.UserWithNoConnectAccess];
|
const userTypes = [localizedConstants.UserWithLoginText, localizedConstants.UserWithWindowsGroupLoginText, localizedConstants.ContainedUserText, localizedConstants.UserWithNoConnectAccess];
|
||||||
this.typeDropdown = this.createDropdown(localizedConstants.UserTypeText, userTypes, getUserTypeDisplayName(this.objectInfo.type), this.options.isNewObject);
|
this.typeDropdown = this.createDropdown(localizedConstants.UserTypeText, async (newValue) => {
|
||||||
this.disposables.push(this.typeDropdown.onValueChanged(async () => {
|
this.objectInfo.type = getUserTypeByDisplayName(newValue);
|
||||||
this.objectInfo.type = getUserTypeByDisplayName(<string>this.typeDropdown.value);
|
|
||||||
this.onObjectValueChange();
|
|
||||||
this.setViewByUserType();
|
this.setViewByUserType();
|
||||||
await this.runValidation(false);
|
}, userTypes, getUserTypeDisplayName(this.objectInfo.type), this.options.isNewObject);
|
||||||
}));
|
|
||||||
this.typeContainer = this.createLabelInputContainer(localizedConstants.UserTypeText, this.typeDropdown);
|
this.typeContainer = this.createLabelInputContainer(localizedConstants.UserTypeText, this.typeDropdown);
|
||||||
this.loginDropdown = this.createDropdown(localizedConstants.LoginText, this.viewInfo.logins, this.objectInfo.loginName, this.options.isNewObject);
|
|
||||||
this.disposables.push(this.loginDropdown.onValueChanged(async () => {
|
this.loginDropdown = this.createDropdown(localizedConstants.LoginText, async (newValue) => {
|
||||||
this.objectInfo.loginName = <string>this.loginDropdown.value;
|
this.objectInfo.loginName = newValue;
|
||||||
this.onObjectValueChange();
|
}, this.viewInfo.logins, this.objectInfo.loginName, this.options.isNewObject);
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
this.loginContainer = this.createLabelInputContainer(localizedConstants.LoginText, this.loginDropdown);
|
this.loginContainer = this.createLabelInputContainer(localizedConstants.LoginText, this.loginDropdown);
|
||||||
|
|
||||||
const authTypes = [];
|
const authTypes = [];
|
||||||
@@ -126,27 +110,18 @@ export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User
|
|||||||
if (this.viewInfo.supportAADAuthentication) {
|
if (this.viewInfo.supportAADAuthentication) {
|
||||||
authTypes.push(localizedConstants.AADAuthenticationTypeDisplayText);
|
authTypes.push(localizedConstants.AADAuthenticationTypeDisplayText);
|
||||||
}
|
}
|
||||||
this.authTypeDropdown = this.createDropdown(localizedConstants.AuthTypeText, authTypes, getAuthenticationTypeDisplayName(this.objectInfo.authenticationType), this.options.isNewObject);
|
this.authTypeDropdown = this.createDropdown(localizedConstants.AuthTypeText, async (newValue) => {
|
||||||
this.authTypeContainer = this.createLabelInputContainer(localizedConstants.AuthTypeText, this.authTypeDropdown);
|
this.objectInfo.authenticationType = getAuthenticationTypeByDisplayName(newValue);
|
||||||
this.disposables.push(this.authTypeDropdown.onValueChanged(async () => {
|
|
||||||
this.objectInfo.authenticationType = getAuthenticationTypeByDisplayName(<string>this.authTypeDropdown.value);
|
|
||||||
this.onObjectValueChange();
|
|
||||||
this.setViewByAuthenticationType();
|
this.setViewByAuthenticationType();
|
||||||
await this.runValidation(false);
|
}, authTypes, getAuthenticationTypeDisplayName(this.objectInfo.authenticationType), this.options.isNewObject);
|
||||||
}));
|
this.authTypeContainer = this.createLabelInputContainer(localizedConstants.AuthTypeText, this.authTypeDropdown);
|
||||||
|
|
||||||
this.passwordInput = this.createPasswordInputBox(localizedConstants.PasswordText, this.objectInfo.password ?? '');
|
this.passwordInput = this.createPasswordInputBox(localizedConstants.PasswordText, async (newValue) => {
|
||||||
|
this.objectInfo.password = newValue;
|
||||||
|
}, this.objectInfo.password ?? '');
|
||||||
this.passwordContainer = this.createLabelInputContainer(localizedConstants.PasswordText, this.passwordInput);
|
this.passwordContainer = this.createLabelInputContainer(localizedConstants.PasswordText, this.passwordInput);
|
||||||
this.confirmPasswordInput = this.createPasswordInputBox(localizedConstants.ConfirmPasswordText, this.objectInfo.password ?? '');
|
this.confirmPasswordInput = this.createPasswordInputBox(localizedConstants.ConfirmPasswordText, async () => { }, this.objectInfo.password ?? '');
|
||||||
this.confirmPasswordContainer = this.createLabelInputContainer(localizedConstants.ConfirmPasswordText, this.confirmPasswordInput);
|
this.confirmPasswordContainer = this.createLabelInputContainer(localizedConstants.ConfirmPasswordText, this.confirmPasswordInput);
|
||||||
this.disposables.push(this.passwordInput.onTextChanged(async () => {
|
|
||||||
this.objectInfo.password = this.passwordInput.value;
|
|
||||||
this.onObjectValueChange();
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
this.disposables.push(this.confirmPasswordInput.onTextChanged(async () => {
|
|
||||||
await this.runValidation(false);
|
|
||||||
}));
|
|
||||||
|
|
||||||
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [
|
this.generalSection = this.createGroup(localizedConstants.GeneralSectionHeader, [
|
||||||
nameContainer,
|
nameContainer,
|
||||||
@@ -160,25 +135,27 @@ export class UserDialog extends ObjectManagementDialogBase<ObjectManagement.User
|
|||||||
}
|
}
|
||||||
|
|
||||||
private initializeOwnedSchemaSection(): void {
|
private initializeOwnedSchemaSection(): void {
|
||||||
const ownedSchemaData = this.viewInfo.schemas.map(name => {
|
this.ownedSchemaTable = this.createTableList<string>(localizedConstants.OwnedSchemaSectionHeader,
|
||||||
const isSelected = this.objectInfo.ownedSchemas.indexOf(name) !== -1;
|
[localizedConstants.SchemaText],
|
||||||
return [{ enabled: !isSelected, checked: isSelected }, name];
|
this.viewInfo.schemas,
|
||||||
|
this.objectInfo.ownedSchemas,
|
||||||
|
DefaultMaxTableHeight,
|
||||||
|
(item) => {
|
||||||
|
// It is not allowed to have unassigned schema.
|
||||||
|
return this.objectInfo.ownedSchemas.indexOf(item) === -1;
|
||||||
});
|
});
|
||||||
this.ownedSchemaTable = this.createTableList(localizedConstants.OwnedSchemaSectionHeader, this.viewInfo.schemas, this.objectInfo.ownedSchemas, ownedSchemaData);
|
|
||||||
this.ownedSchemaSection = this.createGroup(localizedConstants.OwnedSchemaSectionHeader, [this.ownedSchemaTable]);
|
this.ownedSchemaSection = this.createGroup(localizedConstants.OwnedSchemaSectionHeader, [this.ownedSchemaTable]);
|
||||||
}
|
}
|
||||||
|
|
||||||
private initializeMembershipSection(): void {
|
private initializeMembershipSection(): void {
|
||||||
this.membershipTable = this.createTableList(localizedConstants.MembershipSectionHeader, this.viewInfo.databaseRoles, this.objectInfo.databaseRoles);
|
this.membershipTable = this.createTableList<string>(localizedConstants.MembershipSectionHeader, [localizedConstants.DatabaseRoleTypeDisplayNameInTitle], this.viewInfo.databaseRoles, this.objectInfo.databaseRoles);
|
||||||
this.membershipSection = this.createGroup(localizedConstants.MembershipSectionHeader, [this.membershipTable]);
|
this.membershipSection = this.createGroup(localizedConstants.MembershipSectionHeader, [this.membershipTable]);
|
||||||
}
|
}
|
||||||
|
|
||||||
private initializeAdvancedSection(): void {
|
private initializeAdvancedSection(): void {
|
||||||
this.defaultLanguageDropdown = this.createDropdown(localizedConstants.DefaultLanguageText, this.viewInfo.languages, this.objectInfo.defaultLanguage);
|
this.defaultLanguageDropdown = this.createDropdown(localizedConstants.DefaultLanguageText, async (newValue) => {
|
||||||
this.disposables.push(this.defaultLanguageDropdown.onValueChanged(() => {
|
this.objectInfo.defaultLanguage = newValue;
|
||||||
this.objectInfo.defaultLanguage = <string>this.defaultLanguageDropdown.value;
|
}, this.viewInfo.languages, this.objectInfo.defaultLanguage);
|
||||||
this.onObjectValueChange();
|
|
||||||
}));
|
|
||||||
const container = this.createLabelInputContainer(localizedConstants.DefaultLanguageText, this.defaultLanguageDropdown);
|
const container = this.createLabelInputContainer(localizedConstants.DefaultLanguageText, this.defaultLanguageDropdown);
|
||||||
this.advancedSection = this.createGroup(localizedConstants.AdvancedSectionHeader, [container]);
|
this.advancedSection = this.createGroup(localizedConstants.AdvancedSectionHeader, [container]);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import * as azdata from 'azdata';
|
|||||||
import * as vscode from 'vscode';
|
import * as vscode from 'vscode';
|
||||||
import { getErrorMessage } from '../utils';
|
import { getErrorMessage } from '../utils';
|
||||||
import { ObjectManagement } from 'mssql';
|
import { ObjectManagement } from 'mssql';
|
||||||
import { AADAuthenticationTypeDisplayText, ColumnTypeDisplayName, ContainedUserText, DatabaseTypeDisplayName, LoginTypeDisplayName, LoginTypeDisplayNameInTitle, RefreshObjectExplorerError, SQLAuthenticationTypeDisplayText, TableTypeDisplayName, UserTypeDisplayName, UserTypeDisplayNameInTitle, UserWithLoginText, UserWithNoConnectAccess, UserWithWindowsGroupLoginText, ViewTypeDisplayName, WindowsAuthenticationTypeDisplayText } from './localizedConstants';
|
import * as localizedConstants from './localizedConstants';
|
||||||
|
|
||||||
export function deepClone<T>(obj: T): T {
|
export function deepClone<T>(obj: T): T {
|
||||||
if (!obj || typeof obj !== 'object') {
|
if (!obj || typeof obj !== 'object') {
|
||||||
@@ -36,7 +36,7 @@ export async function refreshParentNode(context: azdata.ObjectExplorerContext):
|
|||||||
await parentNode?.refresh();
|
await parentNode?.refresh();
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
await vscode.window.showErrorMessage(RefreshObjectExplorerError(getErrorMessage(err)));
|
await vscode.window.showErrorMessage(localizedConstants.RefreshObjectExplorerError(getErrorMessage(err)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -48,27 +48,33 @@ export async function refreshNode(context: azdata.ObjectExplorerContext): Promis
|
|||||||
await node?.refresh();
|
await node?.refresh();
|
||||||
}
|
}
|
||||||
catch (err) {
|
catch (err) {
|
||||||
await vscode.window.showErrorMessage(RefreshObjectExplorerError(getErrorMessage(err)));
|
await vscode.window.showErrorMessage(localizedConstants.RefreshObjectExplorerError(getErrorMessage(err)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getNodeTypeDisplayName(type: string, inTitle: boolean = false): string {
|
export function getNodeTypeDisplayName(type: string, inTitle: boolean = false): string {
|
||||||
switch (type) {
|
switch (type) {
|
||||||
|
case ObjectManagement.NodeType.ApplicationRole:
|
||||||
|
return inTitle ? localizedConstants.ApplicationRoleTypeDisplayNameInTitle : localizedConstants.ApplicationRoleTypeDisplayName;
|
||||||
|
case ObjectManagement.NodeType.DatabaseRole:
|
||||||
|
return inTitle ? localizedConstants.DatabaseRoleTypeDisplayNameInTitle : localizedConstants.DatabaseRoleTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.ServerLevelLogin:
|
case ObjectManagement.NodeType.ServerLevelLogin:
|
||||||
return inTitle ? LoginTypeDisplayNameInTitle : LoginTypeDisplayName;
|
return inTitle ? localizedConstants.LoginTypeDisplayNameInTitle : localizedConstants.LoginTypeDisplayName;
|
||||||
|
case ObjectManagement.NodeType.ServerLevelServerRole:
|
||||||
|
return inTitle ? localizedConstants.ServerRoleTypeDisplayNameInTitle : localizedConstants.ServerRoleTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.User:
|
case ObjectManagement.NodeType.User:
|
||||||
return inTitle ? UserTypeDisplayNameInTitle : UserTypeDisplayName;
|
return inTitle ? localizedConstants.UserTypeDisplayNameInTitle : localizedConstants.UserTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.Table:
|
case ObjectManagement.NodeType.Table:
|
||||||
return TableTypeDisplayName;
|
return localizedConstants.TableTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.View:
|
case ObjectManagement.NodeType.View:
|
||||||
return ViewTypeDisplayName;
|
return localizedConstants.ViewTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.Column:
|
case ObjectManagement.NodeType.Column:
|
||||||
return ColumnTypeDisplayName;
|
return localizedConstants.ColumnTypeDisplayName;
|
||||||
case ObjectManagement.NodeType.Database:
|
case ObjectManagement.NodeType.Database:
|
||||||
return DatabaseTypeDisplayName;
|
return localizedConstants.DatabaseTypeDisplayName;
|
||||||
default:
|
default:
|
||||||
throw new Error(`Unkown node type: ${type}`);
|
throw new Error(`Unknown node type: ${type}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -77,19 +83,19 @@ export function getAuthenticationTypeDisplayName(authType: ObjectManagement.Auth
|
|||||||
|
|
||||||
switch (authType) {
|
switch (authType) {
|
||||||
case ObjectManagement.AuthenticationType.Windows:
|
case ObjectManagement.AuthenticationType.Windows:
|
||||||
return WindowsAuthenticationTypeDisplayText;
|
return localizedConstants.WindowsAuthenticationTypeDisplayText;
|
||||||
case ObjectManagement.AuthenticationType.AzureActiveDirectory:
|
case ObjectManagement.AuthenticationType.AzureActiveDirectory:
|
||||||
return AADAuthenticationTypeDisplayText;
|
return localizedConstants.AADAuthenticationTypeDisplayText;
|
||||||
default:
|
default:
|
||||||
return SQLAuthenticationTypeDisplayText;
|
return localizedConstants.SQLAuthenticationTypeDisplayText;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getAuthenticationTypeByDisplayName(displayValue: string): ObjectManagement.AuthenticationType {
|
export function getAuthenticationTypeByDisplayName(displayValue: string): ObjectManagement.AuthenticationType {
|
||||||
switch (displayValue) {
|
switch (displayValue) {
|
||||||
case WindowsAuthenticationTypeDisplayText:
|
case localizedConstants.WindowsAuthenticationTypeDisplayText:
|
||||||
return ObjectManagement.AuthenticationType.Windows;
|
return ObjectManagement.AuthenticationType.Windows;
|
||||||
case AADAuthenticationTypeDisplayText:
|
case localizedConstants.AADAuthenticationTypeDisplayText:
|
||||||
return ObjectManagement.AuthenticationType.AzureActiveDirectory;
|
return ObjectManagement.AuthenticationType.AzureActiveDirectory;
|
||||||
default:
|
default:
|
||||||
return ObjectManagement.AuthenticationType.Sql;
|
return ObjectManagement.AuthenticationType.Sql;
|
||||||
@@ -99,23 +105,23 @@ export function getAuthenticationTypeByDisplayName(displayValue: string): Object
|
|||||||
export function getUserTypeDisplayName(userType: ObjectManagement.UserType): string {
|
export function getUserTypeDisplayName(userType: ObjectManagement.UserType): string {
|
||||||
switch (userType) {
|
switch (userType) {
|
||||||
case ObjectManagement.UserType.WithLogin:
|
case ObjectManagement.UserType.WithLogin:
|
||||||
return UserWithLoginText;
|
return localizedConstants.UserWithLoginText;
|
||||||
case ObjectManagement.UserType.WithWindowsGroupLogin:
|
case ObjectManagement.UserType.WithWindowsGroupLogin:
|
||||||
return UserWithWindowsGroupLoginText;
|
return localizedConstants.UserWithWindowsGroupLoginText;
|
||||||
case ObjectManagement.UserType.Contained:
|
case ObjectManagement.UserType.Contained:
|
||||||
return ContainedUserText;
|
return localizedConstants.ContainedUserText;
|
||||||
default:
|
default:
|
||||||
return UserWithNoConnectAccess;
|
return localizedConstants.UserWithNoConnectAccess;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUserTypeByDisplayName(userTypeDisplayName: string): ObjectManagement.UserType {
|
export function getUserTypeByDisplayName(userTypeDisplayName: string): ObjectManagement.UserType {
|
||||||
switch (userTypeDisplayName) {
|
switch (userTypeDisplayName) {
|
||||||
case UserWithLoginText:
|
case localizedConstants.UserWithLoginText:
|
||||||
return ObjectManagement.UserType.WithLogin;
|
return ObjectManagement.UserType.WithLogin;
|
||||||
case UserWithWindowsGroupLoginText:
|
case localizedConstants.UserWithWindowsGroupLoginText:
|
||||||
return ObjectManagement.UserType.WithWindowsGroupLogin;
|
return ObjectManagement.UserType.WithWindowsGroupLogin;
|
||||||
case ContainedUserText:
|
case localizedConstants.ContainedUserText:
|
||||||
return ObjectManagement.UserType.Contained;
|
return ObjectManagement.UserType.Contained;
|
||||||
default:
|
default:
|
||||||
return ObjectManagement.UserType.NoConnectAccess;
|
return ObjectManagement.UserType.NoConnectAccess;
|
||||||
@@ -124,7 +130,7 @@ export function getUserTypeByDisplayName(userTypeDisplayName: string): ObjectMan
|
|||||||
|
|
||||||
// https://docs.microsoft.com/sql/relational-databases/security/password-policy
|
// https://docs.microsoft.com/sql/relational-databases/security/password-policy
|
||||||
export function isValidSQLPassword(password: string, userName: string = 'sa'): boolean {
|
export function isValidSQLPassword(password: string, userName: string = 'sa'): boolean {
|
||||||
const containsUserName = password && userName !== undefined && password.toUpperCase().includes(userName.toUpperCase());
|
const containsUserName = password && userName && password.toUpperCase().includes(userName.toUpperCase());
|
||||||
const hasUpperCase = /[A-Z]/.test(password) ? 1 : 0;
|
const hasUpperCase = /[A-Z]/.test(password) ? 1 : 0;
|
||||||
const hasLowerCase = /[a-z]/.test(password) ? 1 : 0;
|
const hasLowerCase = /[a-z]/.test(password) ? 1 : 0;
|
||||||
const hasNumbers = /\d/.test(password) ? 1 : 0;
|
const hasNumbers = /\d/.test(password) ? 1 : 0;
|
||||||
|
|||||||
70
extensions/mssql/src/util/objects.ts
Normal file
70
extensions/mssql/src/util/objects.ts
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
/*---------------------------------------------------------------------------------------------
|
||||||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||||||
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
||||||
|
*--------------------------------------------------------------------------------------------*/
|
||||||
|
|
||||||
|
export function equals(one: any, other: any, strictArrayCompare: boolean = true): boolean {
|
||||||
|
if (one === other) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (one === null || one === undefined || other === null || other === undefined) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (typeof one !== typeof other) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (typeof one !== 'object') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if ((Array.isArray(one)) !== (Array.isArray(other))) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let i: number;
|
||||||
|
let key: string;
|
||||||
|
|
||||||
|
if (Array.isArray(one)) {
|
||||||
|
if (one.length !== other.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i = 0; i < one.length; i++) {
|
||||||
|
if (strictArrayCompare) {
|
||||||
|
if (!equals(one[i], other[i], strictArrayCompare)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
let match = false;
|
||||||
|
for (let j = 0; j < other.length; j++) {
|
||||||
|
if (equals(one[i], other[j], strictArrayCompare)) {
|
||||||
|
match = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!match) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
const oneKeys: string[] = [];
|
||||||
|
|
||||||
|
for (key in one) {
|
||||||
|
oneKeys.push(key);
|
||||||
|
}
|
||||||
|
oneKeys.sort();
|
||||||
|
const otherKeys: string[] = [];
|
||||||
|
for (key in other) {
|
||||||
|
otherKeys.push(key);
|
||||||
|
}
|
||||||
|
otherKeys.sort();
|
||||||
|
if (!equals(oneKeys, otherKeys, strictArrayCompare)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i = 0; i < oneKeys.length; i++) {
|
||||||
|
if (!equals(one[oneKeys[i]], other[oneKeys[i]], strictArrayCompare)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user